48 {
49
51
53
54 FILE *out;
55
56 double *ahtd, *aqtd, *avtd, ahtdm, aqtdm[
NQ], avtdm, *lat1_old, *lat2_old,
57 *lh1, *lh2, *lon1_old, *lon2_old, *lv1, *lv2, *rhtd, *rqtd, *rvtd, rhtdm,
58 rel_min[
NQ], rqtdm[
NQ], rvtdm, t0 =
59 0, x0[3], x1[3], x2[3], z1, *z1_old, z2, *z2_old, *work;
60
61 int init = 0, np;
62
63
65
66
67 if (argc < 6)
68 ERRMSG(
"Missing or invalid command-line arguments.\n\n"
69 "Usage: atm_dist <ctl> <dist.tab> <param> <atm1a> <atm1b> [<atm2a> <atm2b> ...]\n\n"
70 "Use -h for full help.");
71
72
75 ALLOC(lon1_old,
double,
77 ALLOC(lat1_old,
double,
85 ALLOC(lon2_old,
double,
87 ALLOC(lat2_old,
double,
109
110
112
114 ERRMSG(
"atm_dist currently supports only lat/lon grids");
115 const int ens =
116 (int)
scan_ctl(argv[1], argc, argv,
"DIST_ENS", -1,
"-999", NULL);
117 const double p0 =
118 P(
scan_ctl(argv[1], argc, argv,
"DIST_Z0", -1,
"-1000", NULL));
119 const double p1 =
120 P(
scan_ctl(argv[1], argc, argv,
"DIST_Z1", -1,
"1000", NULL));
121 const double lat0 =
122 scan_ctl(argv[1], argc, argv,
"DIST_LAT0", -1,
"-1000", NULL);
123 const double lat1 =
124 scan_ctl(argv[1], argc, argv,
"DIST_LAT1", -1,
"1000", NULL);
125 const double lon0 =
126 scan_ctl(argv[1], argc, argv,
"DIST_LON0", -1,
"-1000", NULL);
127 const double lon1 =
128 scan_ctl(argv[1], argc, argv,
"DIST_LON1", -1,
"1000", NULL);
129 const double zscore =
130 scan_ctl(argv[1], argc, argv,
"DIST_ZSCORE", -1,
"-999", NULL);
131
132
133 LOG(1,
"Write transport deviations: %s", argv[2]);
134
135
136 if (!(out = fopen(argv[2], "w")))
137 ERRMSG(
"Cannot create file!");
138
139
140 fprintf(out,
141 "# $1 = time [s]\n"
142 "# $2 = time difference [s]\n"
143 "# $3 = absolute horizontal distance (%s) [km]\n"
144 "# $4 = relative horizontal distance (%s) [%%]\n"
145 "# $5 = absolute vertical distance (%s) [km]\n"
146 "# $6 = relative vertical distance (%s) [%%]\n",
147 argv[3], argv[3], argv[3], argv[3]);
148 for (
int iq = 0; iq < ctl.
nq; iq++) {
149 rel_min[iq] =
150 scan_ctl(argv[1], argc, argv,
"DIST_REL_MIN", iq,
"0", NULL);
151 if (rel_min[iq] > 0)
152 fprintf(out,
153 "# Relative %s differences are masked where |q1| + |q2| <= %g %s.\n",
155 fprintf(out,
156 "# $%d = %s absolute difference (%s) [%s]\n"
157 "# $%d = %s relative difference (%s) [%%]\n",
159 8 + 2 * iq, ctl.
qnt_name[iq], argv[3]);
160 }
161 fprintf(out,
"# $%d = number of particles\n\n", 7 + 2 * ctl.
nq);
162
163
164 for (int f = 4; f < argc; f += 2) {
165
166
169 continue;
170
171
172 if (atm1->
np != atm2->
np)
173 ERRMSG(
"Different numbers of particles!");
174
175
176 int time_offset = ctl.
atm_type < 2 ? 23 : 22;
178
179
180 if (!init) {
181 init = 1;
182 t0 = t;
183 }
184
185
186 np = 0;
187 for (
int ip = 0; ip < atm1->
np; ip++) {
188 ahtd[ip] = avtd[ip] = rhtd[ip] = rvtd[ip] = 0;
189 for (
int iq = 0; iq < ctl.
nq; iq++)
190 aqtd[iq *
NP + ip] = rqtd[iq *
NP + ip] = 0;
191 }
192
193
194 for (
int ip = 0; ip < atm1->
np; ip++) {
195
196
199 ERRMSG(
"Air parcel index does not match!");
200
201
204 || atm2->
q[ctl.
qnt_ens][ip] != ens))
205 continue;
206
207
208 if (!isfinite(atm1->
time[ip]) || !isfinite(atm2->
time[ip]))
209 continue;
210
211
212 if (atm1->
p[ip] > p0 || atm1->
p[ip] < p1
213 || atm1->
lon[ip] < lon0 || atm1->
lon[ip] > lon1
214 || atm1->
lat[ip] < lat0 || atm1->
lat[ip] > lat1)
215 continue;
216 if (atm2->
p[ip] > p0 || atm2->
p[ip] < p1
217 || atm2->
lon[ip] < lon0 || atm2->
lon[ip] > lon1
218 || atm2->
lat[ip] < lat0 || atm2->
lat[ip] > lat1)
219 continue;
220
221
226
227
228 ahtd[np] =
DIST(x1, x2);
229 avtd[np] = z1 - z2;
230 for (
int iq = 0; iq < ctl.
nq; iq++)
231 aqtd[iq *
NP + np] = atm1->
q[iq][ip] - atm2->
q[iq][ip];
232
233
234 if (f > 4) {
235
236
237 geo2cart(0, lon1_old[ip], lat1_old[ip], x0);
238 lh1[ip] +=
DIST(x0, x1);
239 lv1[ip] += fabs(z1_old[ip] - z1);
240
241 geo2cart(0, lon2_old[ip], lat2_old[ip], x0);
242 lh2[ip] +=
DIST(x0, x2);
243 lv2[ip] += fabs(z2_old[ip] - z2);
244
245
246 if (lh1[ip] + lh2[ip] > 0)
247 rhtd[np] = 200. *
DIST(x1, x2) / (lh1[ip] + lh2[ip]);
248 if (lv1[ip] + lv2[ip] > 0)
249 rvtd[np] = 200. * (z1 - z2) / (lv1[ip] + lv2[ip]);
250 }
251
252
253 for (
int iq = 0; iq < ctl.
nq; iq++) {
254 const double q1 = atm1->
q[iq][ip];
255 const double q2 = atm2->
q[iq][ip];
256 const double denom = fabs(q1) + fabs(q2);
257 if (denom <= rel_min[iq])
258 rqtd[iq *
NP + np] = GSL_NAN;
259 else
260 rqtd[iq *
NP + np] = 200. * (q1 - q2) / denom;
261 }
262
263
264 lon1_old[ip] = atm1->
lon[ip];
265 lat1_old[ip] = atm1->
lat[ip];
266 z1_old[ip] = z1;
267
268 lon2_old[ip] = atm2->
lon[ip];
269 lat2_old[ip] = atm2->
lat[ip];
270 z2_old[ip] = z2;
271
272
273 np++;
274 }
275
276
277 if (zscore > 0 && np > 1) {
278
279
280 const size_t n = (size_t) np;
281 const double muh = gsl_stats_mean(ahtd, 1, n);
282 const double muv = gsl_stats_mean(avtd, 1, n);
283 const double sigh = gsl_stats_sd(ahtd, 1, n);
284 const double sigv = gsl_stats_sd(avtd, 1, n);
285
286
287 np = 0;
288 for (size_t i = 0; i < n; i++)
289 if (fabs((ahtd[i] - muh) / sigh) < zscore
290 && fabs((avtd[i] - muv) / sigv) < zscore) {
291 ahtd[np] = ahtd[i];
292 rhtd[np] = rhtd[i];
293 avtd[np] = avtd[i];
294 rvtd[np] = rvtd[i];
295 for (
int iq = 0; iq < ctl.
nq; iq++) {
296 aqtd[iq *
NP + np] = aqtd[iq *
NP + (int) i];
297 rqtd[iq *
NP + np] = rqtd[iq *
NP + (int) i];
298 }
299 np++;
300 }
301 }
302
303
304 if (strcasecmp(argv[3], "mean") == 0) {
305 ahtdm = gsl_stats_mean(ahtd, 1, (size_t) np);
306 rhtdm = gsl_stats_mean(rhtd, 1, (size_t) np);
307 avtdm = gsl_stats_mean(avtd, 1, (size_t) np);
308 rvtdm = gsl_stats_mean(rvtd, 1, (size_t) np);
309 for (
int iq = 0; iq < ctl.
nq; iq++) {
310 aqtdm[iq] = gsl_stats_mean(&aqtd[iq *
NP], 1, (
size_t) np);
312 }
313 } else if (strcasecmp(argv[3], "stddev") == 0) {
314 ahtdm = gsl_stats_sd(ahtd, 1, (size_t) np);
315 rhtdm = gsl_stats_sd(rhtd, 1, (size_t) np);
316 avtdm = gsl_stats_sd(avtd, 1, (size_t) np);
317 rvtdm = gsl_stats_sd(rvtd, 1, (size_t) np);
318 for (
int iq = 0; iq < ctl.
nq; iq++) {
319 aqtdm[iq] = gsl_stats_sd(&aqtd[iq *
NP], 1, (
size_t) np);
321 }
322 } else if (strcasecmp(argv[3], "min") == 0) {
323 ahtdm = gsl_stats_min(ahtd, 1, (size_t) np);
324 rhtdm = gsl_stats_min(rhtd, 1, (size_t) np);
325 avtdm = gsl_stats_min(avtd, 1, (size_t) np);
326 rvtdm = gsl_stats_min(rvtd, 1, (size_t) np);
327 for (
int iq = 0; iq < ctl.
nq; iq++) {
328 aqtdm[iq] = gsl_stats_min(&aqtd[iq *
NP], 1, (
size_t) np);
330 }
331 } else if (strcasecmp(argv[3], "max") == 0) {
332 ahtdm = gsl_stats_max(ahtd, 1, (size_t) np);
333 rhtdm = gsl_stats_max(rhtd, 1, (size_t) np);
334 avtdm = gsl_stats_max(avtd, 1, (size_t) np);
335 rvtdm = gsl_stats_max(rvtd, 1, (size_t) np);
336 for (
int iq = 0; iq < ctl.
nq; iq++) {
337 aqtdm[iq] = gsl_stats_max(&aqtd[iq *
NP], 1, (
size_t) np);
339 }
340 } else if (strcasecmp(argv[3], "skew") == 0) {
341 ahtdm = gsl_stats_skew(ahtd, 1, (size_t) np);
342 rhtdm = gsl_stats_skew(rhtd, 1, (size_t) np);
343 avtdm = gsl_stats_skew(avtd, 1, (size_t) np);
344 rvtdm = gsl_stats_skew(rvtd, 1, (size_t) np);
345 for (
int iq = 0; iq < ctl.
nq; iq++) {
346 aqtdm[iq] = gsl_stats_skew(&aqtd[iq *
NP], 1, (
size_t) np);
348 }
349 } else if (strcasecmp(argv[3], "kurt") == 0) {
350 ahtdm = gsl_stats_kurtosis(ahtd, 1, (size_t) np);
351 rhtdm = gsl_stats_kurtosis(rhtd, 1, (size_t) np);
352 avtdm = gsl_stats_kurtosis(avtd, 1, (size_t) np);
353 rvtdm = gsl_stats_kurtosis(rvtd, 1, (size_t) np);
354 for (
int iq = 0; iq < ctl.
nq; iq++) {
355 aqtdm[iq] = gsl_stats_kurtosis(&aqtd[iq *
NP], 1, (
size_t) np);
357 }
358 } else if (strcasecmp(argv[3], "absdev") == 0) {
359 ahtdm = gsl_stats_absdev_m(ahtd, 1, (size_t) np, 0.0);
360 rhtdm = gsl_stats_absdev_m(rhtd, 1, (size_t) np, 0.0);
361 avtdm = gsl_stats_absdev_m(avtd, 1, (size_t) np, 0.0);
362 rvtdm = gsl_stats_absdev_m(rvtd, 1, (size_t) np, 0.0);
363 for (
int iq = 0; iq < ctl.
nq; iq++) {
364 aqtdm[iq] = gsl_stats_absdev_m(&aqtd[iq *
NP], 1, (
size_t) np, 0.0);
366 }
367 } else if (strcasecmp(argv[3], "median") == 0) {
368 ahtdm = gsl_stats_median(ahtd, 1, (size_t) np);
369 rhtdm = gsl_stats_median(rhtd, 1, (size_t) np);
370 avtdm = gsl_stats_median(avtd, 1, (size_t) np);
371 rvtdm = gsl_stats_median(rvtd, 1, (size_t) np);
372 for (
int iq = 0; iq < ctl.
nq; iq++) {
373 aqtdm[iq] = gsl_stats_median(&aqtd[iq *
NP], 1, (
size_t) np);
375 }
376 } else if (strcasecmp(argv[3], "mad") == 0) {
377 ahtdm = gsl_stats_mad0(ahtd, 1, (size_t) np, work);
378 rhtdm = gsl_stats_mad0(rhtd, 1, (size_t) np, work);
379 avtdm = gsl_stats_mad0(avtd, 1, (size_t) np, work);
380 rvtdm = gsl_stats_mad0(rvtd, 1, (size_t) np, work);
381 for (
int iq = 0; iq < ctl.
nq; iq++) {
382 aqtdm[iq] = gsl_stats_mad0(&aqtd[iq *
NP], 1, (
size_t) np, work);
384 }
385 } else
386 ERRMSG(
"Unknown parameter!");
387
388
389 fprintf(out, "%.2f %.2f %g %g %g %g", t, t - t0,
390 ahtdm, rhtdm, avtdm, rvtdm);
391 for (
int iq = 0; iq < ctl.
nq; iq++) {
392 fprintf(out, " ");
394 fprintf(out, " ");
396 }
397 fprintf(out, " %d\n", np);
398 }
399
400
401 fclose(out);
402
403
404 free(atm1);
405 free(atm2);
406 free(lon1_old);
407 free(lat1_old);
408 free(z1_old);
409 free(lh1);
410 free(lv1);
411 free(lon2_old);
412 free(lat2_old);
413 free(z2_old);
414 free(lh2);
415 free(lv2);
416 free(ahtd);
417 free(avtd);
418 free(aqtd);
419 free(rhtd);
420 free(rvtd);
421 free(rqtd);
422 free(work);
423
424 return EXIT_SUCCESS;
425}
double finite_stat(const double *data, int np, const char *param, double *work)
Calculate a statistic on finite values only.
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.
int mptrac_read_atm(const char *filename, const ctl_t *ctl, atm_t *atm)
Reads air parcel data from a specified file into the given atmospheric structure.
double time_from_filename(const char *filename, const int offset, const int with_seconds)
Extracts and converts a timestamp from a filename to Julian seconds.
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.
void geo2cart(const double z, const double lon, const double lat, double *x)
Converts geographic coordinates (longitude, latitude, altitude) to Cartesian coordinates.
#define USAGE
Print usage information on -h or --help.
#define Z(p)
Convert pressure to altitude.
#define P(z)
Compute pressure at given altitude.
#define NQ
Maximum number of quantities per data point.
#define ALLOC(ptr, type, n)
Allocate memory for a pointer with error handling.
#define NP
Maximum number of atmospheric data points.
#define LOG(level,...)
Print a log message with a specified logging level.
#define DIST(a, b)
Calculate the distance between two points in Cartesian coordinates.
double lat[NP]
Latitude [deg].
double lon[NP]
Longitude [deg].
int np
Number of air parcels.
double q[NQ][NP]
Quantity data (for various, user-defined attributes).
double p[NP]
Pressure [hPa].
char qnt_format[NQ][LEN]
Quantity output format.
int atm_type
Type of atmospheric data files (0=ASCII, 1=binary, 2=netCDF, 3=CLaMS_traj, 4=CLaMS_pos).
char qnt_unit[NQ][LEN]
Quantity units.
char qnt_name[NQ][LEN]
Quantity names.
int qnt_ens
Quantity array index for ensemble IDs.
int met_coord_type
Type of coordinates for meteo data (-1=detect, 0=lat/lon [deg], 1=UTM [m]).
int qnt_idx
Quantity array index for air parcel IDs.
int nq
Number of quantities.