Timselpctl.c 7.11 KB
Newer Older
Uwe Schulzweida's avatar
Uwe Schulzweida committed
1
2
3
4
/*
  This file is part of CDO. CDO is a collection of Operators to
  manipulate and analyse Climate model Data.

Uwe Schulzweida's avatar
Uwe Schulzweida committed
5
  Copyright (C) 2006 Brockmann Consult
Uwe Schulzweida's avatar
Uwe Schulzweida committed
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
  See COPYING file for copying and redistribution conditions.

  This program is free software; you can redistribute it and/or modify
  it under the terms of the GNU General Public License as published by
  the Free Software Foundation; version 2 of the License.

  This program is distributed in the hope that it will be useful,
  but WITHOUT ANY WARRANTY; without even the implied warranty of
  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  GNU General Public License for more details.
*/

/*
   This module contains the following operators:

21
      Timselpctl    timselpctl         Time range percentiles
Uwe Schulzweida's avatar
Uwe Schulzweida committed
22
23
24
*/


Ralf Mueller's avatar
Ralf Mueller committed
25
#include <cdi.h>
Uwe Schulzweida's avatar
Uwe Schulzweida committed
26
27
28
#include "cdo.h"
#include "cdo_int.h"
#include "pstream.h"
29
#include "percentiles_hist.h"
Uwe Schulzweida's avatar
Uwe Schulzweida committed
30
31


32
void *Timselpctl(void *argument)
Uwe Schulzweida's avatar
Uwe Schulzweida committed
33
{
34
  int timestat_date = TIMESTAT_MEAN;
Uwe Schulzweida's avatar
Uwe Schulzweida committed
35
36
  int vdate2 = 0, vtime2 = 0;
  int vdate3 = 0, vtime3 = 0;
37
  int nrecs = 0;
Uwe Schulzweida's avatar
Uwe Schulzweida committed
38
39
  int gridID, varID, levelID, recID;
  int tsID;
Uwe Schulzweida's avatar
Uwe Schulzweida committed
40
  int nsets = 0;
Uwe Schulzweida's avatar
Uwe Schulzweida committed
41
42
  int i;
  int nmiss;
43
  int nlevels;
Uwe Schulzweida's avatar
Uwe Schulzweida committed
44
45
  field_t **vars1 = NULL;
  field_t field;
Uwe Schulzweida's avatar
Uwe Schulzweida committed
46
47
48
49
  HISTOGRAM_SET *hset = NULL;

  cdoInitialize(argument);

50
  cdoOperatorAdd("timselpctl", func_pctl,  0, NULL);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
51
52
53

  operatorInputArg("percentile number, nsets <,noffset <,nskip>>");

54
  int nargc = operatorArgc();
55
  if ( nargc < 2 ) cdoAbort("Too few arguments! Need %d found %d.", 2, nargc);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
56

57
  double pn  = parameter2double(operatorArgv()[0]);
58
  percentile_check_number(pn);
59
  int ndates = parameter2int(operatorArgv()[1]);
60
  int noffset = 0, nskip = 0;
61
62
  if ( nargc > 2 ) noffset = parameter2int(operatorArgv()[2]);
  if ( nargc > 3 ) nskip   = parameter2int(operatorArgv()[3]);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
63
64
65

  if ( cdoVerbose ) cdoPrint("nsets = %d, noffset = %d, nskip = %d", ndates, noffset, nskip);

66
67
68
  int streamID1 = streamOpenRead(cdoStreamName(0));
  int streamID2 = streamOpenRead(cdoStreamName(1));
  int streamID3 = streamOpenRead(cdoStreamName(2));
Uwe Schulzweida's avatar
Uwe Schulzweida committed
69

70
71
72
73
  int vlistID1 = streamInqVlist(streamID1);
  int vlistID2 = streamInqVlist(streamID2);
  int vlistID3 = streamInqVlist(streamID3);
  int vlistID4 = vlistDuplicate(vlistID1);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
74

Uwe Schulzweida's avatar
Uwe Schulzweida committed
75
76
  vlistCompare(vlistID1, vlistID2, CMP_ALL);
  vlistCompare(vlistID1, vlistID3, CMP_ALL);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
77

78
79
80
  int taxisID1 = vlistInqTaxis(vlistID1);
  int taxisID2 = vlistInqTaxis(vlistID2);
  int taxisID3 = vlistInqTaxis(vlistID3);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
81
82
  /* TODO - check that time axes 2 and 3 are equal */

83
  int taxisID4 = taxisDuplicate(taxisID1);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
84
85
  vlistDefTaxis(vlistID4, taxisID4);

86
  int streamID4 = streamOpenWrite(cdoStreamName(3), cdoFiletype());
Uwe Schulzweida's avatar
Uwe Schulzweida committed
87
88
89

  streamDefVlist(streamID4, vlistID4);

90
91
  int nvars    = vlistNvars(vlistID1);
  int nrecords = vlistNrecs(vlistID1);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
92

93
94
  int *recVarID   = (int*) Malloc(nrecords*sizeof(int));
  int *recLevelID = (int*) Malloc(nrecords*sizeof(int));
Uwe Schulzweida's avatar
Uwe Schulzweida committed
95

96
97
98
99
  dtlist_type *dtlist = dtlist_new();
  dtlist_set_stat(dtlist, timestat_date);
  dtlist_set_calendar(dtlist, taxisInqCalendar(taxisID1));

100
  int gridsize = vlistGridsizeMax(vlistID1);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
101

102
  field_init(&field);
103
  field.ptr = (double*) Malloc(gridsize * sizeof(double));
Uwe Schulzweida's avatar
Uwe Schulzweida committed
104

105
  vars1 = field_malloc(vlistID1, FIELD_PTR);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
106
107
108
109
110
111
112
113
114
115
116
117
118
119
  hset = hsetCreate(nvars);

  for ( varID = 0; varID < nvars; varID++ )
    {
      gridID   = vlistInqVarGrid(vlistID1, varID);
      nlevels   = zaxisInqSize(vlistInqVarZaxis(vlistID1, varID));

      hsetCreateVarLevels(hset, varID, nlevels, gridID);
    }

  for ( tsID = 0; tsID < noffset; tsID++ )
    {
      nrecs = streamInqTimestep(streamID1, tsID);
      if ( nrecs == 0 ) break;
120
121
122
123
124
125
126
127
128
129
130

      for ( recID = 0; recID < nrecs; recID++ )
	{
	  streamInqRecord(streamID1, &varID, &levelID);

	  if ( tsID == 0 )
	    {
	      recVarID[recID]   = varID;
	      recLevelID[recID] = levelID;
	    }
	}
Uwe Schulzweida's avatar
Uwe Schulzweida committed
131
132
    }

Oliver Heidmann's avatar
Oliver Heidmann committed
133
  int otsID = 0;
Uwe Schulzweida's avatar
Uwe Schulzweida committed
134
135
  if ( tsID < noffset )
    {
136
      cdoWarning("noffset is larger than number of timesteps!");
Uwe Schulzweida's avatar
Uwe Schulzweida committed
137
138
139
140
141
142
143
      goto LABEL_END;
    }

  while ( TRUE )
    {
      nrecs = streamInqTimestep(streamID2, otsID);
      if ( nrecs != streamInqTimestep(streamID3, otsID) )
144
        cdoAbort("Number of records at time step %d of %s and %s differ!", otsID+1, cdoStreamName(1)->args, cdoStreamName(2)->args);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
145
146
147
148
149
150

      vdate2 = taxisInqVdate(taxisID2);
      vtime2 = taxisInqVtime(taxisID2);
      vdate3 = taxisInqVdate(taxisID3);
      vtime3 = taxisInqVtime(taxisID3);
      if ( vdate2 != vdate3 || vtime2 != vtime3 )
151
        cdoAbort("Verification dates at time step %d of %s and %s differ!", otsID+1, cdoStreamName(1)->args, cdoStreamName(2)->args);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
152
153
154
155
156
157
158
      
      for ( recID = 0; recID < nrecs; recID++ )
        {
          streamInqRecord(streamID2, &varID, &levelID);
          streamReadRecord(streamID2, vars1[varID][levelID].ptr, &nmiss);
          vars1[varID][levelID].nmiss = nmiss;
        }
159

Uwe Schulzweida's avatar
Uwe Schulzweida committed
160
161
162
163
164
165
166
167
168
169
170
      for ( recID = 0; recID < nrecs; recID++ )
        {
          streamInqRecord(streamID3, &varID, &levelID);
          streamReadRecord(streamID3, field.ptr, &nmiss);
          field.nmiss   = nmiss;
          field.grid    = vars1[varID][levelID].grid;
          field.missval = vars1[varID][levelID].missval;
          
          hsetDefVarLevelBounds(hset, varID, levelID, &vars1[varID][levelID], &field);
        }

171
172
173
174
175
176
177
      nsets = 0;
      if ( nrecs )
	for ( nsets = 0; nsets < ndates; nsets++ )
	  {
	    nrecs = streamInqTimestep(streamID1, tsID);
	    if ( nrecs == 0 ) break;

178
	    dtlist_taxisInqTimestep(dtlist, taxisID1, nsets);
179
180
181
182
183
184
185
186
187
188
189
190
191

	    for ( recID = 0; recID < nrecs; recID++ )
	      {
		streamInqRecord(streamID1, &varID, &levelID);

		if ( tsID == 0 )
		  {
		    recVarID[recID]   = varID;
		    recLevelID[recID] = levelID;
		  }

		streamReadRecord(streamID1, vars1[varID][levelID].ptr, &nmiss);
		vars1[varID][levelID].nmiss = nmiss;
Uwe Schulzweida's avatar
Uwe Schulzweida committed
192
                  
193
194
		hsetAddVarLevelValues(hset, varID, levelID, &vars1[varID][levelID]);
	      }
Uwe Schulzweida's avatar
Uwe Schulzweida committed
195

196
197
	    tsID++;
	  }
Uwe Schulzweida's avatar
Uwe Schulzweida committed
198
199
200
201
202

      if ( nrecs == 0 && nsets == 0 ) break;

      for ( varID = 0; varID < nvars; varID++ )
        {
Uwe Schulzweida's avatar
Uwe Schulzweida committed
203
          if ( vlistInqVarTsteptype(vlistID1, varID) == TSTEP_CONSTANT ) continue;
Uwe Schulzweida's avatar
Uwe Schulzweida committed
204
205
206
207
208
209
          nlevels = zaxisInqSize(vlistInqVarZaxis(vlistID1, varID));
          
          for ( levelID = 0; levelID < nlevels; levelID++ )
            hsetGetVarLevelPercentiles(&vars1[varID][levelID], hset, varID, levelID, pn);
        }

210
      dtlist_stat_taxisDefTimestep(dtlist, taxisID4, nsets);
211
      streamDefTimestep(streamID4, otsID);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
212
213
214
215
216
217

      for ( recID = 0; recID < nrecords; recID++ )
	{
	  varID   = recVarID[recID];
	  levelID = recLevelID[recID];

Uwe Schulzweida's avatar
Uwe Schulzweida committed
218
	  if ( otsID && vlistInqVarTsteptype(vlistID1, varID) == TSTEP_CONSTANT ) continue;
219
220
221

	  streamDefRecord(streamID4, varID, levelID);
	  streamWriteRecord(streamID4, vars1[varID][levelID].ptr,  vars1[varID][levelID].nmiss);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
222
223
224
	}

      if ( nrecs == 0 ) break;
225
      otsID++;
Uwe Schulzweida's avatar
Uwe Schulzweida committed
226
227
228
229
230
231
232
233
234
235
236
237
238

      for ( i = 0; i < nskip; i++ )
	{
	  nrecs = streamInqTimestep(streamID1, tsID);
	  if ( nrecs == 0 ) break;
	  tsID++;
	}

      if ( nrecs == 0 ) break;
    }

 LABEL_END:

239
  field_free(vars1, vlistID1);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
240
241
  hsetDestroy(hset);

242
243
  dtlist_delete(dtlist);

244
  if ( field.ptr ) Free(field.ptr);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
245

246
247
  if ( recVarID   ) Free(recVarID);
  if ( recLevelID ) Free(recLevelID);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
248
249
250
251
252
253
254
255

  streamClose(streamID4);
  streamClose(streamID3);
  streamClose(streamID2);
  streamClose(streamID1);

  cdoFinish();

Uwe Schulzweida's avatar
Uwe Schulzweida committed
256
  return 0;
Uwe Schulzweida's avatar
Uwe Schulzweida committed
257
}