29 {
30
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
52
53
54 if (argc < 3)
55 ERRMSG(
"Give parameters: <ctl> <metbase>");
56
57
59 double t0 =
scan_ctl(argv[1], argc, argv,
"WIND_T0", -1,
"0", NULL);
60 int nx = (int)
scan_ctl(argv[1], argc, argv,
"WIND_NX", -1,
"360", NULL);
61 int ny = (int)
scan_ctl(argv[1], argc, argv,
"WIND_NY", -1,
"181", NULL);
62 int nz = (int)
scan_ctl(argv[1], argc, argv,
"WIND_NZ", -1,
"61", NULL);
63 double z0 =
scan_ctl(argv[1], argc, argv,
"WIND_Z0", -1,
"0", NULL);
64 double z1 =
scan_ctl(argv[1], argc, argv,
"WIND_Z1", -1,
"60", NULL);
65 double u0 =
66 scan_ctl(argv[1], argc, argv,
"WIND_U0", -1,
"38.587660177302", NULL);
67 double u1 =
68 scan_ctl(argv[1], argc, argv,
"WIND_U1", -1,
"38.587660177302", NULL);
69 double w0 =
scan_ctl(argv[1], argc, argv,
"WIND_W0", -1,
"0", NULL);
70 double alpha =
scan_ctl(argv[1], argc, argv,
"WIND_ALPHA", -1,
"0.0", NULL);
71
72
73 if (nx < 1 || nx >
EX)
74 ERRMSG(
"Set 1 <= NX <= MAX!");
75 if (ny < 1 || ny >
EY)
76 ERRMSG(
"Set 1 <= NY <= MAX!");
77 if (nz < 1 || nz >
EP)
78 ERRMSG(
"Set 1 <= NZ <= MAX!");
79
80
81 jsec2time(t0, &year, &mon, &day, &hour, &min, &sec, &r);
82 t0 = year * 10000. + mon * 100. + day + hour / 24.;
83
84
85 sprintf(filename, "%s_%d_%02d_%02d_%02d.nc", argv[2], year, mon, day, hour);
86
87
88 NC(nc_create(filename, NC_CLOBBER, &ncid));
89
90
91 NC(nc_def_dim(ncid,
"time", 1, &dims[0]));
92 NC(nc_def_dim(ncid,
"lev", (
size_t) nz, &dims[1]));
93 NC(nc_def_dim(ncid,
"lat", (
size_t) ny, &dims[2]));
94 NC(nc_def_dim(ncid,
"lon", (
size_t) nx, &dims[3]));
95
96
97 NC_DEF_VAR(
"time", NC_DOUBLE, 1, &dims[0],
"time",
"day as %Y%m%d.%f");
98 NC_DEF_VAR(
"lev", NC_DOUBLE, 1, &dims[1],
"air_pressure",
"Pa");
99 NC_DEF_VAR(
"lat", NC_DOUBLE, 1, &dims[2],
"latitude",
"degrees_north");
100 NC_DEF_VAR(
"lon", NC_DOUBLE, 1, &dims[3],
"longitude",
"degrees_east");
101 NC_DEF_VAR(
"T", NC_FLOAT, 4, &dims[0],
"Temperature",
"K");
102 NC_DEF_VAR(
"U", NC_FLOAT, 4, &dims[0],
"zonal wind",
"m s**-1");
103 NC_DEF_VAR(
"V", NC_FLOAT, 4, &dims[0],
"meridional wind",
"m s**-1");
104 NC_DEF_VAR(
"W", NC_FLOAT, 4, &dims[0],
"vertical velocity",
"Pa s**-1");
105
106
108
109
110 for (int ix = 0; ix < nx; ix++)
111 dataLon[ix] = 360.0 / nx * (double) ix;
112 for (int iy = 0; iy < ny; iy++)
113 dataLat[iy] = 180.0 / (ny - 1) * (double) iy - 90;
114 for (int iz = 0; iz < nz; iz++)
115 dataZ[iz] = 100. *
P(
LIN(0.0, z0, nz - 1.0, z1, iz));
116
117
122
123
124 for (int ix = 0; ix < nx; ix++)
125 for (int iy = 0; iy < ny; iy++)
126 for (int iz = 0; iz < nz; iz++) {
127 int idx = (iz * ny + iy) * nx + ix;
128 dataU[idx] = (float) (
LIN(0.0, u0, nz - 1.0, u1, iz)
129 * (cos(dataLat[iy] * M_PI / 180.0)
130 * cos(alpha * M_PI / 180.0)
131 + sin(dataLat[iy] * M_PI / 180.0)
132 * cos(dataLon[ix] * M_PI / 180.0)
133 * sin(alpha * M_PI / 180.0)));
134 dataV[idx] = (float) (-
LIN(0.0, u0, nz - 1.0, u1, iz)
135 * sin(dataLon[ix] * M_PI / 180.0)
136 * sin(alpha * M_PI / 180.0));
137 dataW[idx] = (float)
DZ2DP(1e-3 * w0, dataZ[iz]);
138 }
139
140
145
146
148
149
150 free(dataT);
151 free(dataU);
152 free(dataV);
153 free(dataW);
154
155 return EXIT_SUCCESS;
156}
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.
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.
double scan_ctl(const char *filename, int argc, char *argv[], const char *varname, int arridx, const char *defvalue, char *value)
Scans a control file or command-line arguments for a specified variable.
#define LEN
Maximum length of ASCII data lines.
#define NC(cmd)
Execute a NetCDF command and check for errors.
#define ERRMSG(...)
Print an error message with contextual information and terminate the program.
#define EY
Maximum number of latitudes for meteo data.
#define P(z)
Compute pressure at given altitude.
#define EX
Maximum number of longitudes for meteo data.
#define DZ2DP(dz, p)
Convert a change in altitude to a change in pressure.
#define ALLOC(ptr, type, n)
Allocate memory for a pointer with error handling.
#define NC_PUT_FLOAT(varname, ptr, hyperslab)
Write a float array to a NetCDF file.
#define EP
Maximum number of pressure levels for meteo data.
#define NC_PUT_DOUBLE(varname, ptr, hyperslab)
Write double precision data to a NetCDF variable.
#define LIN(x0, y0, x1, y1, x)
Linear interpolation.
#define NC_DEF_VAR(varname, type, ndims, dims, long_name, units)
Define a NetCDF variable with attributes.