Cat.c 3.77 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.

5
  Copyright (C) 2003-2010 Uwe Schulzweida, Uwe.Schulzweida@zmaw.de
Uwe Schulzweida's avatar
Uwe Schulzweida committed
6
7
8
9
10
11
12
13
14
15
16
17
  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.
*/

18
19
20
21
22
23
/*
   This module contains the following operators:

      Copy       cat             Concatenate datasets
*/

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

30
void    vlistDefVarTime(int vlistID, int varID, int timeID);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
31
32
33
34
35
36

void *Cat(void *argument)
{
  int streamID1, streamID2 = CDI_UNDEFID;
  int nrecs;
  int tsID1, tsID2 = 0, recID, varID, levelID;
37
  int vlistID1, vlistID2 = CDI_UNDEFID;
Uwe Schulzweida's avatar
Uwe Schulzweida committed
38
39
40
41
42
  int streamCnt, nfiles, indf;
  int taxisID1, taxisID2 = CDI_UNDEFID;
  int lcopy = FALSE;
  int gridsize;
  int nmiss;
43
  int ntsteps;
Uwe Schulzweida's avatar
Uwe Schulzweida committed
44
45
46
47
48
49
50
51
52
53
54
  double *array = NULL;

  cdoInitialize(argument);

  if ( UNCHANGED_RECORD ) lcopy = TRUE;

  streamCnt = cdoStreamCnt();
  nfiles = streamCnt - 1;

  for ( indf = 0; indf < nfiles; indf++ )
    {
Uwe Schulzweida's avatar
Uwe Schulzweida committed
55
56
      if ( cdoVerbose ) cdoPrint("Process file: %s", cdoStreamName(indf));

Uwe Schulzweida's avatar
Uwe Schulzweida committed
57
58
59
60
61
62
63
64
      streamID1 = streamOpenRead(cdoStreamName(indf));
      if ( streamID1 < 0 ) cdiError(streamID1, "Open failed on %s", cdoStreamName(indf));

      vlistID1 = streamInqVlist(streamID1);
      taxisID1 = vlistInqTaxis(vlistID1);

      if ( indf == 0 )
	{
65
	  if ( fileExist(cdoStreamName(nfiles)) )
Uwe Schulzweida's avatar
Uwe Schulzweida committed
66
67
68
69
70
71
72
73
	    {
	      streamID2 = streamOpenAppend(cdoStreamName(nfiles));
	      if ( streamID2 < 0 )
		cdiError(streamID2, "Open failed on %s", cdoStreamName(nfiles));

	      vlistID2 = streamInqVlist(streamID2);
	      taxisID2 = vlistInqTaxis(vlistID2);

Uwe Schulzweida's avatar
Uwe Schulzweida committed
74
	      vlistCompare(vlistID1, vlistID2, CMP_SFT);
75

Uwe Schulzweida's avatar
Uwe Schulzweida committed
76
	      tsID2 = vlistNtsteps(vlistID2);
77
	      if ( tsID2 == 0 ) tsID2 = 1; /* bug fix for time constant data only */
Uwe Schulzweida's avatar
Uwe Schulzweida committed
78
79
80
81
	    }
	  else
	    {
	      if ( cdoVerbose )
82
		cdoPrint("Output file doesn't exist, creating: %s", cdoStreamName(nfiles));
Uwe Schulzweida's avatar
Uwe Schulzweida committed
83
84
85
86
87
88
89
90
91

	      streamID2 = streamOpenWrite(cdoStreamName(nfiles), cdoFiletype());
	      if ( streamID2 < 0 )
		cdiError(streamID2, "Open failed on %s", cdoStreamName(nfiles));

	      vlistID2 = vlistDuplicate(vlistID1);
	      taxisID2 = taxisDuplicate(taxisID1);
	      vlistDefTaxis(vlistID2, taxisID2);
	  
92
93
94
95
96
97
98
99
100
	      ntsteps = vlistNtsteps(vlistID1);
	      if ( ntsteps == 0 && nfiles > 1 )
		{
		  int nvars = vlistNvars(vlistID1);
		  
		  for ( varID = 0; varID < nvars; ++varID )
		    vlistDefVarTime(vlistID2, varID, TIME_VARIABLE);
		}

Uwe Schulzweida's avatar
Uwe Schulzweida committed
101
102
103
104
105
106
107
108
109
	      streamDefVlist(streamID2, vlistID2);
	    }

	  if ( ! lcopy )
	    {
	      gridsize = vlistGridsizeMax(vlistID1);
	      array = (double *) malloc(gridsize*sizeof(double));
	    }
	}
110
111
112
113
      else
	{
	  vlistCompare(vlistID1, vlistID2, CMP_SFT);
	}
Uwe Schulzweida's avatar
Uwe Schulzweida committed
114
115
116
117
118
119
120
121
122
123
124
125

      tsID1 = 0;
      while ( (nrecs = streamInqTimestep(streamID1, tsID1)) )
	{
	  taxisCopyTimestep(taxisID2, taxisID1);

	  streamDefTimestep(streamID2, tsID2);
	       
	  for ( recID = 0; recID < nrecs; recID++ )
	    {
	      streamInqRecord(streamID1, &varID, &levelID);
	      streamDefRecord(streamID2,  varID,  levelID);
126

Uwe Schulzweida's avatar
Uwe Schulzweida committed
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
	      if ( lcopy )
		{
		  streamCopyRecord(streamID2, streamID1); 
		}
	      else
		{
		  streamReadRecord(streamID1, array, &nmiss);
		  streamWriteRecord(streamID2, array, nmiss);
		}
	    }
	  tsID1++;
	  tsID2++;
	}
      streamClose(streamID1);
    }

  streamClose(streamID2);
 
145
  if ( array ) free(array);
Uwe Schulzweida's avatar
Uwe Schulzweida committed
146
147
148
149
150

  cdoFinish();

  return (0);
}