MPTRAC
met_lapse.c
Go to the documentation of this file.
1/*
2 This file is part of MPTRAC.
3
4 MPTRAC is free software: you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation, either version 3 of the License, or
7 (at your option) any later version.
8
9 MPTRAC is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
13
14 You should have received a copy of the GNU General Public License
15 along with MPTRAC. If not, see <http://www.gnu.org/licenses/>.
16
17 Copyright (C) 2013-2026 Forschungszentrum Juelich GmbH
18*/
19
25#include "mptrac.h"
26
27/* ------------------------------------------------------------
28 Dimensions...
29 ------------------------------------------------------------ */
30
32#define LAPSEMIN -20.0
33
35#define DLAPSE 0.1
36
38#define IDXMAX 400
39
40/* ------------------------------------------------------------
41 Functions...
42 ------------------------------------------------------------ */
43
45void usage(
46 void);
47
48/* ------------------------------------------------------------
49 Main...
50 ------------------------------------------------------------ */
51
52int main(
53 int argc,
54 char *argv[]) {
55
56 ctl_t ctl;
57
58 clim_t *clim;
59
60 met_t *met;
61
62 dd_t *dd;
63
64 FILE *out;
65
66 static double p2[1000], t[1000], t2[1000], z[1000], z2[1000], lat_mean,
67 z_mean;
68
69 static int hist_max[1000], hist_min[1000], hist_mean[1000], hist_sig[1000],
70 nhist_max, nhist_min, nhist_mean, nhist_sig, np;
71
72 /* Allocate... */
73 ALLOC(clim, clim_t, 1);
74 ALLOC(met, met_t, 1);
75 ALLOC(dd, dd_t, 1);
76
77 /* Print usage information... */
78 USAGE;
79
80 /* Check arguments... */
81 if (argc < 4)
82 ERRMSG("Missing or invalid command-line arguments.\n\n"
83 "Usage: met_lapse <ctl> <lapse.tab> <met0> [<met1> ...]\n\n"
84 "Use -h for full help.");
85
86 /* Read control parameters... */
87 mptrac_read_ctl(argv[1], argc, argv, &ctl);
88 const int dz =
89 (int) scan_ctl(argv[1], argc, argv, "LAPSE_DZ", -1, "20", NULL);
90 const double lat0 =
91 (int) scan_ctl(argv[1], argc, argv, "LAPSE_LAT0", -1, "-90", NULL);
92 const double lat1 =
93 (int) scan_ctl(argv[1], argc, argv, "LAPSE_LAT1", -1, "90", NULL);
94 const double z0 =
95 (int) scan_ctl(argv[1], argc, argv, "LAPSE_Z0", -1, "0", NULL);
96 const double z1 =
97 (int) scan_ctl(argv[1], argc, argv, "LAPSE_Z1", -1, "100", NULL);
98 const int intpol =
99 (int) scan_ctl(argv[1], argc, argv, "LAPSE_INTPOL", -1, "1", NULL);
100
101 /* Read climatological data... */
102 mptrac_read_clim(&ctl, clim);
103
104 /* Loop over files... */
105 for (int i = 3; i < argc; i++) {
106
107 /* Read meteorological data... */
108 if (!mptrac_read_met(argv[i], &ctl, clim, met, dd))
109 continue;
110
111 /* Get altitude and pressure profiles... */
112 for (int iz = 0; iz < met->np; iz++)
113 z[iz] = Z(met->p[iz]);
114 for (int iz = 0; iz <= 250; iz++) {
115 z2[iz] = 0.0 + 0.1 * iz;
116 p2[iz] = P(z2[iz]);
117 }
118
119 /* Loop over grid points... */
120 for (int ix = 0; ix < met->nx; ix++)
121 for (int iy = 0; iy < met->ny; iy++) {
122
123 /* Check latitude range... */
124 if (met->lat[iy] < lat0 || met->lat[iy] > lat1)
125 continue;
126
127 /* Interpolate temperature profile... */
128 for (int iz = 0; iz < met->np; iz++)
129 t[iz] = met->t[ix][iy][iz];
130 if (intpol == 1)
131 spline(z, t, met->np, z2, t2, 251, ctl.met_tropo_spline);
132 else
133 for (int iz = 0; iz <= 250; iz++) {
134 int idx = locate_irr(z, met->np, z2[iz]);
135 t2[iz] = LIN(z[idx], t[idx], z[idx + 1], t[idx + 1], z2[iz]);
136 }
137
138 /* Loop over vertical levels... */
139 for (int iz = 0; iz <= 250; iz++) {
140
141 /* Check height range... */
142 if (z2[iz] < z0 || z2[iz] > z1)
143 continue;
144
145 /* Check surface pressure... */
146 if (p2[iz] > met->ps[ix][iy])
147 continue;
148
149 /* Get mean latitude and height... */
150 lat_mean += met->lat[iy];
151 z_mean += z2[iz];
152 np++;
153
154 /* Get lapse rates within a vertical layer... */
155 int nlapse = 0;
156 double lapse_max = -1e99, lapse_min = 1e99, lapse_mean =
157 0, lapse_sig = 0;
158 for (int iz2 = iz + 1; iz2 <= iz + dz; iz2++) {
159 lapse_max =
160 MAX(LAPSE(p2[iz], t2[iz], p2[iz2], t2[iz2]), lapse_max);
161 lapse_min =
162 MIN(LAPSE(p2[iz], t2[iz], p2[iz2], t2[iz2]), lapse_min);
163 lapse_mean += LAPSE(p2[iz], t2[iz], p2[iz2], t2[iz2]);
164 lapse_sig += SQR(LAPSE(p2[iz], t2[iz], p2[iz2], t2[iz2]));
165 nlapse++;
166 }
167 lapse_mean /= nlapse;
168 lapse_sig = sqrt(MAX(lapse_sig / nlapse - SQR(lapse_mean), 0));
169
170 /* Get histograms... */
171 int idx = (int) ((lapse_max - LAPSEMIN) / DLAPSE);
172 if (idx >= 0 && idx < IDXMAX) {
173 hist_max[idx]++;
174 nhist_max++;
175 }
176
177 idx = (int) ((lapse_min - LAPSEMIN) / DLAPSE);
178 if (idx >= 0 && idx < IDXMAX) {
179 hist_min[idx]++;
180 nhist_min++;
181 }
182
183 idx = (int) ((lapse_mean - LAPSEMIN) / DLAPSE);
184 if (idx >= 0 && idx < IDXMAX) {
185 hist_mean[idx]++;
186 nhist_mean++;
187 }
188
189 idx = (int) ((lapse_sig - LAPSEMIN) / DLAPSE);
190 if (idx >= 0 && idx < IDXMAX) {
191 hist_sig[idx]++;
192 nhist_sig++;
193 }
194 }
195 }
196 }
197
198 /* Create output file... */
199 LOG(1, "Write lapse rate data: %s", argv[2]);
200 if (!(out = fopen(argv[2], "w")))
201 ERRMSG("Cannot create file!");
202
203 /* Write header... */
204 fprintf(out,
205 "# $1 = mean altitude [km]\n"
206 "# $2 = mean latitude [deg]\n"
207 "# $3 = lapse rate [K/km]\n"
208 "# $4 = counts of maxima per bin\n"
209 "# $5 = total number of maxima\n"
210 "# $6 = normalized frequency of maxima\n"
211 "# $7 = counts of minima per bin\n"
212 "# $8 = total number of minima\n"
213 "# $9 = normalized frequency of minima\n"
214 "# $10 = counts of means per bin\n"
215 "# $11 = total number of means\n"
216 "# $12 = normalized frequency of means\n"
217 "# $13 = counts of sigmas per bin\n"
218 "# $14 = total number of sigmas\n"
219 "# $15 = normalized frequency of sigmas\n\n");
220
221 /* Write data... */
222 double nmax_max = 0, nmax_min = 0, nmax_mean = 0, nmax_sig = 0;
223 for (int idx = 0; idx < IDXMAX; idx++) {
224 nmax_max = MAX(hist_max[idx], nmax_max);
225 nmax_min = MAX(hist_min[idx], nmax_min);
226 nmax_mean = MAX(hist_mean[idx], nmax_mean);
227 nmax_sig = MAX(hist_sig[idx], nmax_sig);
228 }
229 for (int idx = 0; idx < IDXMAX; idx++)
230 fprintf(out,
231 "%g %g %g %d %d %g %d %d %g %d %d %g %d %d %g\n",
232 z_mean / np, lat_mean / np, (idx + .5) * DLAPSE + LAPSEMIN,
233 hist_max[idx], nhist_max,
234 (double) hist_max[idx] / (double) nmax_max, hist_min[idx],
235 nhist_min, (double) hist_min[idx] / (double) nmax_min,
236 hist_mean[idx], nhist_mean,
237 (double) hist_mean[idx] / (double) nmax_mean, hist_sig[idx],
238 nhist_sig, (double) hist_sig[idx] / (double) nmax_sig);
239
240 /* Close file... */
241 fclose(out);
242
243 /* Free... */
244 free(clim);
245 free(met);
246 free(dd);
247
248 return EXIT_SUCCESS;
249}
250
251/*****************************************************************************/
252
254void usage(
255 void) {
256
257 printf("\nMPTRAC met_lapse tool.\n\n");
258 printf("Calculate lapse-rate statistics from meteorological data.\n");
259 printf("\n");
260 printf("Usage:\n");
261 printf(" met_lapse <ctl> <lapse.tab> <met0> [<met1> ...]\n");
262 printf("\n");
263 printf("Arguments:\n");
264 printf(" <ctl> Control file.\n");
265 printf(" <lapse.tab> Output table for lapse-rate statistics.\n");
266 printf(" <met*> Meteorological input files.\n");
267 printf("\nFurther information:\n");
268 printf(" Manual: https://slcs-jsc.github.io/mptrac/\n");
269}
int main(int argc, char *argv[])
Definition: met_lapse.c:52
#define LAPSEMIN
Lapse rate minimum [K/km].
Definition: met_lapse.c:32
#define IDXMAX
Maximum number of histogram bins.
Definition: met_lapse.c:38
#define DLAPSE
Lapse rate bin size [K/km].
Definition: met_lapse.c:35
void usage(void)
Print command-line help.
Definition: met_lapse.c:254
int locate_irr(const double *xx, const int n, const double x)
Locate the index of the interval containing a given value in a sorted array.
Definition: mptrac.c:2524
void spline(const double *x, const double *y, const int n, const double *x2, double *y2, const int n2, const int method)
Performs spline interpolation or linear interpolation.
Definition: mptrac.c:10850
double scan_ctl(const char *filename, int argc, char *argv[], const char *varname, const int arridx, const char *defvalue, char *value)
Scans a control file or command-line arguments for a specified variable.
Definition: mptrac.c:10745
void mptrac_read_clim(const ctl_t *ctl, clim_t *clim)
Reads various climatological data and populates the given climatology structure.
Definition: mptrac.c:5188
int mptrac_read_met(const char *filename, const ctl_t *ctl, const clim_t *clim, met_t *met, dd_t *dd)
Reads meteorological data from a file, supporting multiple formats and MPI broadcasting.
Definition: mptrac.c:6151
void mptrac_read_ctl(const char *filename, int argc, char *argv[], ctl_t *ctl)
Reads control parameters from a configuration file and populates the given structure.
Definition: mptrac.c:5248
MPTRAC library declarations.
#define LAPSE(p1, t1, p2, t2)
Calculate lapse rate.
Definition: mptrac.h:1028
#define MIN(a, b)
Macro to determine the minimum of two values.
Definition: mptrac.h:1175
#define ERRMSG(...)
Print an error message with contextual information and terminate the program.
Definition: mptrac.h:2102
#define USAGE
Print usage information on -h or --help.
Definition: mptrac.h:1909
#define Z(p)
Convert pressure to altitude.
Definition: mptrac.h:1939
#define P(z)
Compute pressure at given altitude.
Definition: mptrac.h:1480
#define ALLOC(ptr, type, n)
Allocate memory for a pointer with error handling.
Definition: mptrac.h:453
#define SQR(x)
Compute the square of a value.
Definition: mptrac.h:1733
#define LOG(level,...)
Print a log message with a specified logging level.
Definition: mptrac.h:2032
#define LIN(x0, y0, x1, y1, x)
Linear interpolation.
Definition: mptrac.h:1047
#define MAX(a, b)
Macro to determine the maximum of two values.
Definition: mptrac.h:1074
Climatological data.
Definition: mptrac.h:3436
Control parameters.
Definition: mptrac.h:2190
int met_tropo_spline
Tropopause interpolation method (0=linear, 1=spline).
Definition: mptrac.h:2696
Domain decomposition data structure.
Definition: mptrac.h:3669
Meteo data structure.
Definition: mptrac.h:3495
int nx
Number of longitudes.
Definition: mptrac.h:3501
int ny
Number of latitudes.
Definition: mptrac.h:3504
float ps[EX][EY]
Surface pressure [hPa].
Definition: mptrac.h:3534
int np
Number of pressure levels.
Definition: mptrac.h:3507
float t[EX][EY][EP]
Temperature [K].
Definition: mptrac.h:3609
double lat[EY]
Latitudes [deg].
Definition: mptrac.h:3516
double p[EP]
Pressure levels [hPa].
Definition: mptrac.h:3519