MPTRAC
wind.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-2025 Forschungszentrum Juelich GmbH
18*/
19
25#include "mptrac.h"
26
27int main(
28 int argc,
29 char *argv[]) {
30
31 ctl_t ctl;
32
33 static char filename[LEN];
34
35 static double r, dataLon[EX], dataLat[EY], dataZ[EP];
36
37 static float *dataT, *dataU, *dataV, *dataW;
38
39 static int ncid, varid, dims[4], year, mon, day, hour, min, sec;
40
41 static size_t start[4], count[4];
42
43 /* Allocate... */
44 ALLOC(dataT, float,
45 EP * EY * EX);
46 ALLOC(dataU, float,
47 EP * EY * EX);
48 ALLOC(dataV, float,
49 EP * EY * EX);
50 ALLOC(dataW, float,
51 EP * EY * EX);
52
53 /* Check arguments... */
54 if (argc < 3)
55 ERRMSG("Give parameters: <ctl> <metbase>");
56
57 /* Read control parameters... */
58 read_ctl(argv[1], argc, argv, &ctl);
59 double t0 = scan_ctl(argv[1], argc, argv, "WIND_T0", -1, "0", NULL);
60 const int nx =
61 (int) scan_ctl(argv[1], argc, argv, "WIND_NX", -1, "360", NULL);
62 const int ny =
63 (int) scan_ctl(argv[1], argc, argv, "WIND_NY", -1, "181", NULL);
64 const int nz =
65 (int) scan_ctl(argv[1], argc, argv, "WIND_NZ", -1, "61", NULL);
66 const double z0 = scan_ctl(argv[1], argc, argv, "WIND_Z0", -1, "0", NULL);
67 const double z1 = scan_ctl(argv[1], argc, argv, "WIND_Z1", -1, "60", NULL);
68 const double u0 =
69 scan_ctl(argv[1], argc, argv, "WIND_U0", -1, "38.587660177302", NULL);
70 const double u1 =
71 scan_ctl(argv[1], argc, argv, "WIND_U1", -1, "38.587660177302", NULL);
72 const double w0 = scan_ctl(argv[1], argc, argv, "WIND_W0", -1, "0", NULL);
73 const double alpha =
74 scan_ctl(argv[1], argc, argv, "WIND_ALPHA", -1, "0.0", NULL);
75
76 /* Check dimensions... */
77 if (nx < 1 || nx > EX)
78 ERRMSG("Set 1 <= NX <= MAX!");
79 if (ny < 1 || ny > EY)
80 ERRMSG("Set 1 <= NY <= MAX!");
81 if (nz < 1 || nz > EP)
82 ERRMSG("Set 1 <= NZ <= MAX!");
83
84 /* Get time... */
85 jsec2time(t0, &year, &mon, &day, &hour, &min, &sec, &r);
86 t0 = year * 10000. + mon * 100. + day + hour / 24.;
87
88 /* Set filename... */
89 sprintf(filename, "%s_%d_%02d_%02d_%02d.nc", argv[2], year, mon, day, hour);
90
91 /* Create netCDF file... */
92 NC(nc_create(filename, NC_NETCDF4, &ncid));
93
94 /* Create dimensions... */
95 NC(nc_def_dim(ncid, "time", 1, &dims[0]));
96 NC(nc_def_dim(ncid, "lev", (size_t) nz, &dims[1]));
97 NC(nc_def_dim(ncid, "lat", (size_t) ny, &dims[2]));
98 NC(nc_def_dim(ncid, "lon", (size_t) nx, &dims[3]));
99
100 /* Create variables... */
101 NC_DEF_VAR("time", NC_DOUBLE, 1, &dims[0], "time", "day as %Y%m%d.%f", 0,
102 0);
103 NC_DEF_VAR("lev", NC_DOUBLE, 1, &dims[1], "air_pressure", "Pa", 0, 0);
104 NC_DEF_VAR("lat", NC_DOUBLE, 1, &dims[2], "latitude", "degrees_north", 0,
105 0);
106 NC_DEF_VAR("lon", NC_DOUBLE, 1, &dims[3], "longitude", "degrees_east", 0,
107 0);
108 NC_DEF_VAR("T", NC_FLOAT, 4, &dims[0], "Temperature", "K", 0, 0);
109 NC_DEF_VAR("U", NC_FLOAT, 4, &dims[0], "zonal wind", "m s**-1", 0, 0);
110 NC_DEF_VAR("V", NC_FLOAT, 4, &dims[0], "meridional wind", "m s**-1", 0, 0);
111 NC_DEF_VAR("W", NC_FLOAT, 4, &dims[0], "vertical velocity", "Pa s**-1", 0,
112 0);
113
114 /* End definition... */
115 NC(nc_enddef(ncid));
116
117 /* Set coordinates... */
118 for (int ix = 0; ix < nx; ix++)
119 dataLon[ix] = 360.0 / nx * (double) ix;
120 for (int iy = 0; iy < ny; iy++)
121 dataLat[iy] = 180.0 / (ny - 1) * (double) iy - 90;
122 for (int iz = 0; iz < nz; iz++)
123 dataZ[iz] = 100. * P(LIN(0.0, z0, nz - 1.0, z1, iz));
124
125 /* Write coordinates... */
126 NC_PUT_DOUBLE("time", &t0, 0);
127 NC_PUT_DOUBLE("lev", dataZ, 0);
128 NC_PUT_DOUBLE("lat", dataLat, 0);
129 NC_PUT_DOUBLE("lon", dataLon, 0);
130
131 /* Create wind fields (Williamson et al., 1992)... */
132 for (int ix = 0; ix < nx; ix++)
133 for (int iy = 0; iy < ny; iy++)
134 for (int iz = 0; iz < nz; iz++) {
135 int idx = (iz * ny + iy) * nx + ix;
136 dataU[idx] = (float) (LIN(0.0, u0, nz - 1.0, u1, iz)
137 * (cos(DEG2RAD(dataLat[iy]))
138 * cos(DEG2RAD(alpha))
139 + sin(DEG2RAD(dataLat[iy]))
140 * cos(DEG2RAD(dataLon[ix]))
141 * sin(DEG2RAD(alpha))));
142 dataV[idx] = (float) (-LIN(0.0, u0, nz - 1.0, u1, iz)
143 * sin(DEG2RAD(dataLon[ix]))
144 * sin(DEG2RAD(alpha)));
145 dataW[idx] = (float) DZ2DP(1e-3 * w0, dataZ[iz]);
146 }
147
148 /* Write data... */
149 NC_PUT_FLOAT("T", dataT, 0);
150 NC_PUT_FLOAT("U", dataU, 0);
151 NC_PUT_FLOAT("V", dataV, 0);
152 NC_PUT_FLOAT("W", dataW, 0);
153
154 /* Close file... */
155 NC(nc_close(ncid));
156
157 /* Free... */
158 free(dataT);
159 free(dataU);
160 free(dataV);
161 free(dataW);
162
163 return EXIT_SUCCESS;
164}
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:8503
void 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:5156
void jsec2time(const double jsec, int *year, int *mon, int *day, int *hour, int *min, int *sec, double *remain)
Converts Julian seconds to calendar date and time components.
Definition: mptrac.c:1831
MPTRAC library declarations.
#define LEN
Maximum length of ASCII data lines.
Definition: mptrac.h:241
#define NC(cmd)
Execute a NetCDF command and check for errors.
Definition: mptrac.h:1016
#define ERRMSG(...)
Print an error message with contextual information and terminate the program.
Definition: mptrac.h:1916
#define EY
Maximum number of latitudes for meteo data.
Definition: mptrac.h:271
#define P(z)
Compute pressure at given altitude.
Definition: mptrac.h:1304
#define EX
Maximum number of longitudes for meteo data.
Definition: mptrac.h:266
#define DZ2DP(dz, p)
Convert a change in altitude to a change in pressure.
Definition: mptrac.h:562
#define ALLOC(ptr, type, n)
Allocate memory for a pointer with error handling.
Definition: mptrac.h:349
#define DEG2RAD(deg)
Converts degrees to radians.
Definition: mptrac.h:478
#define NC_PUT_FLOAT(varname, ptr, hyperslab)
Write a float array to a NetCDF file.
Definition: mptrac.h:1150
#define NC_DEF_VAR(varname, type, ndims, dims, long_name, units, level, quant)
Define a NetCDF variable with attributes.
Definition: mptrac.h:1045
#define EP
Maximum number of pressure levels for meteo data.
Definition: mptrac.h:261
#define NC_PUT_DOUBLE(varname, ptr, hyperslab)
Write double precision data to a NetCDF variable.
Definition: mptrac.h:1126
#define LIN(x0, y0, x1, y1, x)
Linear interpolation.
Definition: mptrac.h:859
Control parameters.
Definition: mptrac.h:2170
int main(int argc, char *argv[])
Definition: wind.c:27