mirror of
https://github.com/Unidata/netcdf-c.git
synced 2025-01-18 15:55:12 +08:00
334 lines
9.7 KiB
C
334 lines
9.7 KiB
C
/*********************************************************************
|
|
* Copyright 1993, UCAR/Unidata
|
|
* See netcdf/COPYRIGHT file for copying and redistribution conditions.
|
|
* $Header: /upc/share/CVS/netcdf-3/ncgen/getfill.c,v 1.8 2010/04/14 22:04:56 dmh Exp $
|
|
*********************************************************************/
|
|
|
|
#include "includes.h"
|
|
#include "dump.h"
|
|
|
|
/* mnemonic*/
|
|
#define TOPLEVEL 1
|
|
|
|
/*Forward*/
|
|
static Datalist* buildfill(Symbol* tsym);
|
|
static void fill(Symbol* tsym, Datalist*);
|
|
static void fillarray(Symbol* tsym, Dimset* dimset, int index, Datalist*);
|
|
static void filllist(Symbol* tvsym, Datalist* dl);
|
|
static int checkfill(Symbol* tvsym, Datasrc* src, Symbol*);
|
|
static int checkarray(Symbol* tsym, Dimset* dimset, int index, Datasrc*, Symbol*, int);
|
|
|
|
|
|
/* Construct a Datalist representing a complete fill value*/
|
|
/* for a specified type*/
|
|
/* Cache if needed later*/
|
|
|
|
Datalist*
|
|
getfiller(Symbol* tvsym, Datalist* fillsrc)
|
|
{
|
|
Datalist* filler = fillsrc;
|
|
if(filler == NULL)
|
|
filler = tvsym->var.special._Fillvalue; /* get cached value (if any)*/
|
|
if(filler == NULL) { /* need to compute it*/
|
|
filler = buildfill(tvsym);
|
|
tvsym->var.special._Fillvalue = filler; /* cache value*/
|
|
}
|
|
if(debug >= 1)
|
|
dumpdatalist(filler,"getfiller");
|
|
return filler;
|
|
}
|
|
|
|
static Datalist*
|
|
buildfill(Symbol* tvasym)
|
|
{
|
|
Datalist* filler = builddatalist(0);
|
|
ASSERT(tvasym->objectclass == NC_VAR
|
|
|| tvasym->objectclass == NC_TYPE
|
|
|| tvasym->objectclass == NC_ATT);
|
|
if(tvasym->objectclass == NC_VAR) {
|
|
if(tvasym->typ.dimset.ndims > 0) {
|
|
fillarray(tvasym->typ.basetype,&tvasym->typ.dimset,0,filler);
|
|
} else
|
|
fill(tvasym->typ.basetype,filler);
|
|
} else if(tvasym->objectclass == NC_ATT) {
|
|
fill(tvasym->typ.basetype,filler);
|
|
} else /*NC_TYPE*/
|
|
fill(tvasym,filler);
|
|
return filler;
|
|
}
|
|
|
|
static void
|
|
fill(Symbol* tvsym, Datalist* filler)
|
|
{
|
|
int i;
|
|
Constant con;
|
|
Datalist* sublist;
|
|
/* NC_TYPE case*/
|
|
switch (tvsym->subclass) {
|
|
case NC_ENUM: case NC_OPAQUE: case NC_PRIM:
|
|
con.nctype = tvsym->typ.typecode;
|
|
nc_getfill(&con);
|
|
break;
|
|
case NC_COMPOUND:
|
|
sublist = builddatalist(listlength(tvsym->subnodes));
|
|
for(i=0;i<listlength(tvsym->subnodes);i++) {
|
|
Symbol* field = (Symbol*)listget(tvsym->subnodes,i);
|
|
if(field->typ.dimset.ndims > 0) {
|
|
fillarray(field->typ.basetype,&field->typ.dimset,0,filler);
|
|
} else
|
|
filllist(field->typ.basetype,sublist);
|
|
}
|
|
con = builddatasublist(sublist);
|
|
break;
|
|
case NC_VLEN:
|
|
sublist = builddatalist(0);
|
|
filllist(tvsym->typ.basetype,sublist); /* generate a single instance*/
|
|
con = builddatasublist(sublist);
|
|
break;
|
|
default: PANIC1("fill: unexpected subclass %d",tvsym->subclass);
|
|
}
|
|
dlappend(filler,&con);
|
|
}
|
|
|
|
static void
|
|
filllist(Symbol* tvsym, Datalist* dl)
|
|
{
|
|
int i;
|
|
Datalist* sublist;
|
|
Constant con;
|
|
|
|
/* NC_TYPE case*/
|
|
switch (tvsym->subclass) {
|
|
case NC_ENUM: case NC_OPAQUE: case NC_PRIM:
|
|
con.nctype = tvsym->typ.typecode;
|
|
nc_getfill(&con);
|
|
dlappend(dl,&con);
|
|
break;
|
|
case NC_COMPOUND:
|
|
sublist = builddatalist(listlength(tvsym->subnodes));
|
|
for(i=0;i<listlength(tvsym->subnodes);i++) {
|
|
Symbol* field = (Symbol*)listget(tvsym->subnodes,i);
|
|
filllist(field->typ.basetype,sublist);
|
|
}
|
|
con = builddatasublist(sublist);
|
|
dlappend(dl,&con);
|
|
break;
|
|
case NC_VLEN:
|
|
sublist = builddatalist(0);
|
|
filllist(tvsym->typ.basetype,sublist); /* generate a single instance*/
|
|
con = builddatasublist(sublist);
|
|
dlappend(dl,&con);
|
|
break;
|
|
default: PANIC1("fill: unexpected subclass %d",tvsym->subclass);
|
|
}
|
|
}
|
|
|
|
static void
|
|
fillarray(Symbol* basetype, Dimset* dimset, int index, Datalist* arraylist)
|
|
{
|
|
int i;
|
|
Symbol* dim = dimset->dimsyms[index];
|
|
unsigned int size = dim->dim.declsize;
|
|
int isunlimited = (size == 0);
|
|
int lastdim = (index == (dimset->ndims - 1));
|
|
int firstdim = (index == 0);
|
|
Datalist* sublist;
|
|
|
|
sublist = (firstdim?builddatalist(0):arraylist);
|
|
if(isunlimited) {
|
|
/* do a single entry to satisfy*/
|
|
if(lastdim) {
|
|
filllist(basetype,sublist);
|
|
} else {
|
|
fillarray(basetype->typ.basetype,dimset,index+1,sublist);
|
|
}
|
|
} else { /* bounded*/
|
|
if(lastdim) {
|
|
for(i=0;i<size;i++) filllist(basetype,sublist);
|
|
} else {
|
|
for(i=0;i<size;i++) {
|
|
fillarray(basetype->typ.basetype,dimset,index+1,sublist);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Verify that a user provided fill value does in fact*/
|
|
/* conform to the structure of a given type or variable*/
|
|
/* Structure conforms to fill procedures above.*/
|
|
|
|
int
|
|
checkfillvalue(Symbol* tvsym, Datalist* filler)
|
|
{
|
|
Datasrc* src = datalist2src(filler);
|
|
int result;
|
|
ASSERT(tvsym->objectclass == NC_VAR || tvsym->objectclass == NC_TYPE);
|
|
if(tvsym->objectclass == NC_VAR) {
|
|
if(tvsym->typ.dimset.ndims > 0) {
|
|
result = checkarray(tvsym->typ.basetype,&tvsym->typ.dimset,0,src,tvsym,TOPLEVEL);
|
|
} else
|
|
result = checkfill(tvsym->typ.basetype,src,tvsym);
|
|
} else /* NC_TYPE*/
|
|
result = checkfill(tvsym,src,tvsym);
|
|
freedatasrc(src);
|
|
return result;
|
|
}
|
|
|
|
static int
|
|
checkfill(Symbol* tvsym, Datasrc* src, Symbol* original)
|
|
{
|
|
int i,iscmpd,result;
|
|
Constant* con;
|
|
|
|
result = 1;
|
|
switch (tvsym->subclass) {
|
|
case NC_ENUM: case NC_OPAQUE: case NC_PRIM:
|
|
con = srcnext(src);
|
|
if(src == NULL) {
|
|
semerror(srcline(src),"%s: malformed _FillValue",original->name);
|
|
result = 0;
|
|
} else if(con->nctype != tvsym->typ.typecode) result = 0; /* wrong type*/
|
|
break;
|
|
|
|
case NC_COMPOUND:
|
|
if(!issublist(src)) {/* fail on no compound*/
|
|
semerror(srcline(src),"Compound constants must be enclosed in {..}");
|
|
}
|
|
srcpush(src);
|
|
for(i=0;i<listlength(tvsym->subnodes);i++) {
|
|
Symbol* field = (Symbol*)listget(tvsym->subnodes,i);
|
|
result = checkfill(field,src,original);
|
|
if(!result) break;
|
|
}
|
|
srcpop(src);
|
|
break;
|
|
|
|
case NC_VLEN:
|
|
if(!issublist(src)) {
|
|
semerror(srcline(src),"%s: vlen instances in _FillValue must be enclosed in {...}",original->name);
|
|
result = 0;
|
|
} else {
|
|
srcpush(src);
|
|
while(srcmore(src)) {
|
|
result = checkfill(tvsym->typ.basetype,src,original);
|
|
if(!result) break;
|
|
}
|
|
srcpop(src);
|
|
}
|
|
break;
|
|
|
|
case NC_FIELD:
|
|
/* Braces are optional */
|
|
if((iscmpd=issublist(src))) srcpush(src);
|
|
if(tvsym->typ.dimset.ndims > 0) {
|
|
result = checkarray(tvsym->typ.basetype,&tvsym->typ.dimset,0,src,original,!TOPLEVEL);
|
|
} else
|
|
result = checkfill(tvsym->typ.basetype,src,original);
|
|
if(iscmpd) srcpop(src);
|
|
break;
|
|
|
|
default: PANIC1("checkfillvalue: unexpected subclass %d",tvsym->subclass);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
static int
|
|
checkarray(Symbol* basetype, Dimset* dimset, int index, Datasrc* src, Symbol* original, int toplevel)
|
|
{
|
|
int i,result;
|
|
Symbol* dim = dimset->dimsyms[index];
|
|
unsigned int size = dim->dim.declsize;
|
|
int lastdim = (index == (dimset->ndims - 1));
|
|
int isunlimited = (size == 0);
|
|
|
|
result = 1;
|
|
if(isunlimited) {
|
|
if(!toplevel) {
|
|
if(!issublist(src)) {
|
|
semerror(srcline(src),"UNLIMITED dimension constants (other than top level) must be enclosed in {...}");
|
|
result = 0;
|
|
goto done;
|
|
} else
|
|
srcpush(src);
|
|
}
|
|
if(lastdim) {
|
|
while(srcmore(src) && result) {
|
|
result = checkfill(basetype,src,original);
|
|
}
|
|
} else { /*!lastdim*/
|
|
while(srcmore(src) && result) {
|
|
result = checkarray(basetype,dimset,index+1,src,original,toplevel);
|
|
}
|
|
}
|
|
} else { /* bounded*/
|
|
if(lastdim) {
|
|
for(i=0;i<size && result;i++) {
|
|
result = checkfill(basetype,src,original);
|
|
}
|
|
} else { /* !lastdim*/
|
|
for(i=0;i<size && result;i++) {
|
|
result = checkarray(basetype,dimset,index+1,src,original,toplevel);
|
|
}
|
|
}
|
|
}
|
|
done:
|
|
return result;
|
|
}
|
|
|
|
/*
|
|
* Given primitive netCDF type, return a default fill_value appropriate for
|
|
* that type.
|
|
*/
|
|
void
|
|
nc_getfill(Constant* value)
|
|
{
|
|
switch(value->nctype) {
|
|
case NC_CHAR: value->value.charv = NC_FILL_CHAR; break;
|
|
case NC_BYTE: value->value.int8v = NC_FILL_BYTE; break;
|
|
case NC_SHORT: value->value.int16v = NC_FILL_SHORT; break;
|
|
case NC_INT: value->value.int32v = NC_FILL_INT; break;
|
|
case NC_FLOAT: value->value.floatv = NC_FILL_FLOAT; break;
|
|
case NC_DOUBLE: value->value.doublev = NC_FILL_DOUBLE; break;
|
|
case NC_UBYTE: value->value.uint8v = NC_FILL_UBYTE; break;
|
|
case NC_USHORT: value->value.uint16v = NC_FILL_USHORT; break;
|
|
case NC_UINT: value->value.uint32v = NC_FILL_UINT; break;
|
|
case NC_INT64: value->value.int64v = NC_FILL_INT64; break;
|
|
case NC_UINT64: value->value.uint64v = NC_FILL_UINT64; break;
|
|
case NC_STRING:
|
|
value->value.stringv.stringv = nulldup(NC_FILL_STRING);
|
|
value->value.stringv.len = strlen(NC_FILL_STRING);
|
|
/* Exception: if string is null, then make it's length be 1 */
|
|
if(value->value.stringv.len == 0)
|
|
value->value.stringv.len = 1;
|
|
break;
|
|
case NC_OPAQUE:
|
|
value->value.opaquev.len = 2;
|
|
value->value.opaquev.stringv = nulldup("00");
|
|
break;
|
|
default:
|
|
derror("nc_getfill: unrecognized type: %d",value->nctype);
|
|
}
|
|
}
|
|
|
|
char*
|
|
nc_dfaltfillname(nc_type nctype)
|
|
{
|
|
switch (nctype) {
|
|
case NC_BYTE: return "NC_FILL_BYTE";
|
|
case NC_CHAR: return "NC_FILL_CHAR";
|
|
case NC_SHORT: return "NC_FILL_SHORT";
|
|
case NC_INT: return "NC_FILL_INT";
|
|
case NC_FLOAT: return "NC_FILL_FLOAT";
|
|
case NC_DOUBLE: return "NC_FILL_DOUBLE";
|
|
case NC_UBYTE: return "NC_FILL_UBYTE";
|
|
case NC_USHORT: return "NC_FILL_USHORT";
|
|
case NC_UINT: return "NC_FILL_UINT";
|
|
case NC_INT64: return "NC_FILL_INT64";
|
|
case NC_UINT64: return "NC_FILL_UINT64";
|
|
case NC_STRING: return "NC_FILL_STRING";
|
|
default: PANIC("unexpected default fill name");
|
|
}
|
|
return NULL;
|
|
}
|
|
|