MPTRAC
tropo.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 Functions...
29 ------------------------------------------------------------ */
30
32void usage(
33 void);
34
35/* ------------------------------------------------------------
36 Main...
37 ------------------------------------------------------------ */
38
39int main(
40 int argc,
41 char *argv[]) {
42
43 ctl_t ctl;
44
45 clim_t *clim;
46
47 met_t *met;
48
49 dd_t *dd;
50
51 static double ps[EX * EY], pt[EX * EY], qt[EX * EY], o3t[EX * EY],
52 zs[EX * EY], zt[EX * EY], tt[EX * EY], lon, lons[EX], lat, lats[EY];
53
54 static int init, nx, ny, nt, ncid, varid, dims[3];
55
56 static size_t count[10], start[10];
57
58 /* Allocate... */
59 ALLOC(clim, clim_t, 1);
60 ALLOC(met, met_t, 1);
61 ALLOC(dd, dd_t, 1);
62
63 /* Print usage information... */
64 USAGE;
65
66 /* Check arguments... */
67 if (argc < 4)
68 ERRMSG("Missing or invalid command-line arguments.\n\n"
69 "Usage: tropo <ctl> <tropo.nc> <met0> [<met1> ...]\n\n"
70 "Use -h for full help.");
71
72 /* Read control parameters... */
73 mptrac_read_ctl(argv[1], argc, argv, &ctl);
74 double lon0 = scan_ctl(argv[1], argc, argv, "TROPO_LON0", -1, "-180", NULL);
75 double lon1 = scan_ctl(argv[1], argc, argv, "TROPO_LON1", -1, "180", NULL);
76 double dlon = scan_ctl(argv[1], argc, argv, "TROPO_DLON", -1, "-999", NULL);
77 double lat0 = scan_ctl(argv[1], argc, argv, "TROPO_LAT0", -1, "-90", NULL);
78 double lat1 = scan_ctl(argv[1], argc, argv, "TROPO_LAT1", -1, "90", NULL);
79 double dlat = scan_ctl(argv[1], argc, argv, "TROPO_DLAT", -1, "-999", NULL);
80 int h2o = (int) scan_ctl(argv[1], argc, argv, "TROPO_H2O", -1, "1", NULL);
81 int o3 = (int) scan_ctl(argv[1], argc, argv, "TROPO_O3", -1, "1", NULL);
82
83 /* Read climatological data... */
84 mptrac_read_clim(&ctl, clim);
85
86 /* Loop over files... */
87 for (int i = 3; i < argc; i++) {
88
89 /* Set control parameters... */
90 ctl.met_tropo = 0;
91
92 /* Read meteorological data... */
93 if (!mptrac_read_met(argv[i], &ctl, clim, met, dd))
94 continue;
95
96 /* Set horizontal grid... */
97 if (!init) {
98 init = 1;
99
100 /* Get grid... */
101 if (dlon <= 0)
102 dlon = fabs(met->lon[1] - met->lon[0]);
103 if (dlat <= 0)
104 dlat = fabs(met->lat[1] - met->lat[0]);
105 if (lon0 < -360 && lon1 > 360) {
106 lon0 = gsl_stats_min(met->lon, 1, (size_t) met->nx);
107 lon1 = gsl_stats_max(met->lon, 1, (size_t) met->nx);
108 }
109 nx = ny = 0;
110 for (lon = lon0; lon <= lon1 + 0.001; lon += dlon) {
111 lons[nx] = round(lon * 1e3) / 1e3;
112 if ((++nx) >= EX)
113 ERRMSG("Too many longitudes!");
114 }
115 if (lat0 < -90 && lat1 > 90) {
116 lat0 = gsl_stats_min(met->lat, 1, (size_t) met->ny);
117 lat1 = gsl_stats_max(met->lat, 1, (size_t) met->ny);
118 }
119 for (lat = lat0; lat <= lat1 + 0.001; lat += dlat) {
120 lats[ny] = round(lat * 1e3) / 1e3;
121 if ((++ny) >= EY)
122 ERRMSG("Too many latitudes!");
123 }
124
125 /* Create netCDF file... */
126 LOG(1, "Write tropopause data file: %s", argv[2]);
127 NC(nc_create(argv[2], NC_NETCDF4, &ncid));
128
129 /* Create dimensions... */
130 NC(nc_def_dim(ncid, "time", (size_t) NC_UNLIMITED, &dims[0]));
131 NC(nc_def_dim(ncid, "lat", (size_t) ny, &dims[1]));
132 NC(nc_def_dim(ncid, "lon", (size_t) nx, &dims[2]));
133
134 /* Create variables... */
135 NC_DEF_VAR("time", NC_DOUBLE, 1, &dims[0], "time",
136 "seconds since 2000-01-01 00:00:00 UTC", 0, 0);
137 NC_DEF_VAR("lat", NC_DOUBLE, 1, &dims[1], "latitude", "degrees_north",
138 0, 0);
139 NC_DEF_VAR("lon", NC_DOUBLE, 1, &dims[2], "longitude", "degrees_east",
140 0, 0);
141
142 NC_DEF_VAR("clp_z", NC_FLOAT, 3, &dims[0], "cold point height", "km", 0,
143 0);
144 NC_DEF_VAR("clp_p", NC_FLOAT, 3, &dims[0], "cold point pressure", "hPa",
145 0, 0);
146 NC_DEF_VAR("clp_t", NC_FLOAT, 3, &dims[0], "cold point temperature",
147 "K", 0, 0);
148 if (h2o)
149 NC_DEF_VAR("clp_q", NC_FLOAT, 3, &dims[0], "cold point water vapor",
150 "ppv", 0, 0);
151 if (o3)
152 NC_DEF_VAR("clp_o3", NC_FLOAT, 3, &dims[0], "cold point ozone",
153 "ppv", 0, 0);
154
155 NC_DEF_VAR("dyn_z", NC_FLOAT, 3, &dims[0],
156 "dynamical tropopause height", "km", 0, 0);
157 NC_DEF_VAR("dyn_p", NC_FLOAT, 3, &dims[0],
158 "dynamical tropopause pressure", "hPa", 0, 0);
159 NC_DEF_VAR("dyn_t", NC_FLOAT, 3, &dims[0],
160 "dynamical tropopause temperature", "K", 0, 0);
161 if (h2o)
162 NC_DEF_VAR("dyn_q", NC_FLOAT, 3, &dims[0],
163 "dynamical tropopause water vapor", "ppv", 0, 0);
164 if (o3)
165 NC_DEF_VAR("dyn_o3", NC_FLOAT, 3, &dims[0],
166 "dynamical tropopause ozone", "ppv", 0, 0);
167
168 NC_DEF_VAR("wmo_1st_z", NC_FLOAT, 3, &dims[0],
169 "WMO 1st tropopause height", "km", 0, 0);
170 NC_DEF_VAR("wmo_1st_p", NC_FLOAT, 3, &dims[0],
171 "WMO 1st tropopause pressure", "hPa", 0, 0);
172 NC_DEF_VAR("wmo_1st_t", NC_FLOAT, 3, &dims[0],
173 "WMO 1st tropopause temperature", "K", 0, 0);
174 if (h2o)
175 NC_DEF_VAR("wmo_1st_q", NC_FLOAT, 3, &dims[0],
176 "WMO 1st tropopause water vapor", "ppv", 0, 0);
177 if (o3)
178 NC_DEF_VAR("wmo_1st_o3", NC_FLOAT, 3, &dims[0],
179 "WMO 1st tropopause ozone", "ppv", 0, 0);
180
181 NC_DEF_VAR("wmo_2nd_z", NC_FLOAT, 3, &dims[0],
182 "WMO 2nd tropopause height", "km", 0, 0);
183 NC_DEF_VAR("wmo_2nd_p", NC_FLOAT, 3, &dims[0],
184 "WMO 2nd tropopause pressure", "hPa", 0, 0);
185 NC_DEF_VAR("wmo_2nd_t", NC_FLOAT, 3, &dims[0],
186 "WMO 2nd tropopause temperature", "K", 0, 0);
187 if (h2o)
188 NC_DEF_VAR("wmo_2nd_q", NC_FLOAT, 3, &dims[0],
189 "WMO 2nd tropopause water vapor", "ppv", 0, 0);
190 if (o3)
191 NC_DEF_VAR("wmo_2nd_o3", NC_FLOAT, 3, &dims[0],
192 "WMO 2nd tropopause ozone", "ppv", 0, 0);
193
194 NC_DEF_VAR("ps", NC_FLOAT, 3, &dims[0], "surface pressure", "hPa", 0,
195 0);
196 NC_DEF_VAR("zs", NC_FLOAT, 3, &dims[0], "surface height", "km", 0, 0);
197
198 /* End definition... */
199 NC(nc_enddef(ncid));
200
201 /* Write longitude and latitude... */
202 NC_PUT_DOUBLE("lat", lats, 0);
203 NC_PUT_DOUBLE("lon", lons, 0);
204 }
205
206 /* Write time... */
207 start[0] = (size_t) nt;
208 count[0] = 1;
209 start[1] = 0;
210 count[1] = (size_t) ny;
211 start[2] = 0;
212 count[2] = (size_t) nx;
213 NC_PUT_DOUBLE("time", &met->time, 1);
214
215 /* Get cold point... */
216 get_tropo(2, &ctl, clim, met, lons, nx, lats, ny, pt, zt, tt, qt, o3t, ps,
217 zs);
218 NC_PUT_DOUBLE("clp_z", zt, 1);
219 NC_PUT_DOUBLE("clp_p", pt, 1);
220 NC_PUT_DOUBLE("clp_t", tt, 1);
221 if (h2o)
222 NC_PUT_DOUBLE("clp_q", qt, 1);
223 if (o3)
224 NC_PUT_DOUBLE("clp_o3", o3t, 1);
225
226 /* Get dynamical tropopause... */
227 get_tropo(5, &ctl, clim, met, lons, nx, lats, ny, pt, zt, tt, qt, o3t, ps,
228 zs);
229 NC_PUT_DOUBLE("dyn_z", zt, 1);
230 NC_PUT_DOUBLE("dyn_p", pt, 1);
231 NC_PUT_DOUBLE("dyn_t", tt, 1);
232 if (h2o)
233 NC_PUT_DOUBLE("dyn_q", qt, 1);
234 if (o3)
235 NC_PUT_DOUBLE("dyn_o3", o3t, 1);
236
237 /* Get WMO 1st tropopause... */
238 get_tropo(3, &ctl, clim, met, lons, nx, lats, ny, pt, zt, tt, qt, o3t, ps,
239 zs);
240 NC_PUT_DOUBLE("wmo_1st_z", zt, 1);
241 NC_PUT_DOUBLE("wmo_1st_p", pt, 1);
242 NC_PUT_DOUBLE("wmo_1st_t", tt, 1);
243 if (h2o)
244 NC_PUT_DOUBLE("wmo_1st_q", qt, 1);
245 if (o3)
246 NC_PUT_DOUBLE("wmo_1st_o3", o3t, 1);
247
248 /* Get WMO 2nd tropopause... */
249 get_tropo(4, &ctl, clim, met, lons, nx, lats, ny, pt, zt, tt, qt, o3t, ps,
250 zs);
251 NC_PUT_DOUBLE("wmo_2nd_z", zt, 1);
252 NC_PUT_DOUBLE("wmo_2nd_p", pt, 1);
253 NC_PUT_DOUBLE("wmo_2nd_t", tt, 1);
254 if (h2o)
255 NC_PUT_DOUBLE("wmo_2nd_q", qt, 1);
256 if (o3)
257 NC_PUT_DOUBLE("wmo_2nd_o3", o3t, 1);
258
259 /* Write surface data... */
260 NC_PUT_DOUBLE("ps", ps, 1);
261 NC_PUT_DOUBLE("zs", zs, 1);
262
263 /* Increment time step counter... */
264 nt++;
265 }
266
267 /* Close file... */
268 NC(nc_close(ncid));
269
270 /* Free... */
271 free(clim);
272 free(met);
273 free(dd);
274
275 return EXIT_SUCCESS;
276}
277
278/*****************************************************************************/
279
281void usage(
282 void) {
283
284 printf("\nMPTRAC tropo tool.\n\n");
285 printf("Create tropopause data sets from meteorological data.\n");
286 printf("\n");
287 printf("Usage:\n");
288 printf(" tropo <ctl> <tropo.nc> <met0> [<met1> ...]\n");
289 printf("\n");
290 printf("Arguments:\n");
291 printf(" <ctl> Control file.\n");
292 printf(" <tropo.nc> Output tropopause netCDF file.\n");
293 printf(" <met*> Meteorological input files.\n");
294 printf("\nFurther information:\n");
295 printf(" Manual: https://slcs-jsc.github.io/mptrac/\n");
296}
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
void get_tropo(const int met_tropo, ctl_t *ctl, const clim_t *clim, met_t *met, const double *lons, const int nx, const double *lats, const int ny, double *pt, double *zt, double *tt, double *qt, double *o3t, double *ps, double *zs)
Calculate tropopause data.
Definition: mptrac.c:1783
MPTRAC library declarations.
#define NC(cmd)
Execute a NetCDF command and check for errors.
Definition: mptrac.h:1204
#define ERRMSG(...)
Print an error message with contextual information and terminate the program.
Definition: mptrac.h:2102
#define EY
Maximum number of latitudes for meteo data.
Definition: mptrac.h:340
#define USAGE
Print usage information on -h or --help.
Definition: mptrac.h:1909
#define EX
Maximum number of longitudes for meteo data.
Definition: mptrac.h:335
#define ALLOC(ptr, type, n)
Allocate memory for a pointer with error handling.
Definition: mptrac.h:453
#define LOG(level,...)
Print a log message with a specified logging level.
Definition: mptrac.h:2032
#define NC_DEF_VAR(varname, type, ndims, dims, long_name, units, level, quant)
Define a NetCDF variable with attributes.
Definition: mptrac.h:1233
#define NC_PUT_DOUBLE(varname, ptr, hyperslab)
Write double precision data to a NetCDF variable.
Definition: mptrac.h:1317
Climatological data.
Definition: mptrac.h:3436
Control parameters.
Definition: mptrac.h:2190
int met_tropo
Tropopause definition (0=none, 1=clim, 2=cold point, 3=WMO_1st, 4=WMO_2nd, 5=dynamical).
Definition: mptrac.h:2687
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
double lon[EX]
Longitudes [deg].
Definition: mptrac.h:3513
double time
Time [s].
Definition: mptrac.h:3498
double lat[EY]
Latitudes [deg].
Definition: mptrac.h:3516
int main(int argc, char *argv[])
Definition: tropo.c:39
void usage(void)
Print command-line help.
Definition: tropo.c:281