mirror of
https://github.com/Unidata/netcdf-c.git
synced 2025-01-06 15:34:44 +08:00
751300ec59
This is a follow up to PR https://github.com/Unidata/netcdf-c/pull/1173 Sorry that it is so big, but leak suppression can be complex. This PR fixes all remaining memory leaks -- as determined by -fsanitize=address, and with the exceptions noted below. Unfortunately. there remains a significant leak that I cannot solve. It involves vlens, and it is unclear if the leak is occurring in the netcdf-c library or the HDF5 library. I have added a check_PROGRAM to the ncdump directory to show the problem. The program is called tst_vlen_demo.c To exercise it, build the netcdf library with -fsanitize=address enabled. Then go into ncdump and do a "make clean check". This should build tst_vlen_demo without actually executing it. Then do the command "./tst_vlen_demo" to see the output of the memory checker. Note the the lost malloc is deep in the HDF5 library (in H5Tvlen.c). I am temporarily working around this error in the following way. 1. I modified several test scripts to not execute known vlen tests that fail as described above. 2. Added an environment variable called NC_VLEN_NOTEST. If set, then those specific tests are suppressed. This should mean that the --disable-utilities option to ./configure should not need to be set to get a memory leak clean build. This should allow for detection of any new leaks. Note: I used an environment variable rather than a ./configure option to control the vlen tests. This is because it is temporary (I hope) and because it is a bit tricky for shell scripts to access ./configure options. Finally, as before, this only been tested with netcdf-4 and hdf5 support.
545 lines
14 KiB
C
545 lines
14 KiB
C
/*********************************************************************
|
|
* Copyright 1993, UCAR/Unidata
|
|
* See netcdf/COPYRIGHT file for copying and redistribution conditions.
|
|
* $Header: /upc/share/CVS/netcdf-3/ncgen/gencml.c,v 1.5 2010/04/04 19:39:46 dmh Exp $
|
|
*********************************************************************/
|
|
|
|
#include "includes.h"
|
|
#include <ctype.h> /* for isprint() */
|
|
|
|
#ifdef ENABLE_CML
|
|
|
|
#undef TRACE
|
|
|
|
extern List* vlenconstants; /* List<Constant*>;*/
|
|
|
|
/* Forward */
|
|
/*
|
|
static const char* ncstype(nc_type);
|
|
static const char* nctype(nc_type);
|
|
static const char* ncctype(nc_type);
|
|
static const char* groupncid(Symbol*);
|
|
static const char* typencid(Symbol*);
|
|
static const char* varncid(Symbol*);
|
|
static const char* dimncid(Symbol*);
|
|
*/
|
|
|
|
static void gencml_group(Symbol* group);
|
|
|
|
static char* ncxtype(nc_type type);
|
|
|
|
#ifdef USE_NETCDF4
|
|
static void definextype(Symbol*);
|
|
static char* xprefixed(List* prefix, char* suffix, char* separator);
|
|
static void definespecialattributes(Symbol* vsym);
|
|
#endif
|
|
|
|
static void defineattribute(Symbol* asym);
|
|
static void definevardata(Symbol*);
|
|
|
|
#ifdef USE_NETCDF4
|
|
static void xprint(Bytebuffer* buf);
|
|
#endif
|
|
static void xpartial(char* line);
|
|
static void xline(char* line);
|
|
static void xflush(void);
|
|
|
|
|
|
/*
|
|
Global Bytebuffer into which to store the C code output;
|
|
periodically dumped to file by xflush().
|
|
*/
|
|
Bytebuffer* xcode;
|
|
|
|
/*
|
|
* Generate C code for creating netCDF from in-memory structure.
|
|
*/
|
|
void
|
|
gencml_netcdf(void)
|
|
{
|
|
xcode = bbNew();
|
|
bbSetalloc(xcode,C_MAX_STMT);
|
|
const char *filename = rootgroup->file.filename;
|
|
|
|
/* Dump XML header */
|
|
xline("<?xml version=\"1.0\" encoding=\"UTF-8\"?>");
|
|
xline("<netcdf xmlns=\"http://www.unidata.ucar.edu/namespaces/netcdf/ncml-2.2\">");
|
|
|
|
xline("");
|
|
xline("<explicit/>");
|
|
xline("");
|
|
|
|
xflush();
|
|
|
|
/* Recursively dump all groups via the rootgroup */
|
|
gencml_group(rootgroup);
|
|
}
|
|
|
|
void
|
|
gencml_close(void)
|
|
{
|
|
xline("</netcdf>");
|
|
xflush();
|
|
}
|
|
|
|
static void
|
|
gencml_group(Symbol* group)
|
|
{
|
|
int idim, ivar, iatt;
|
|
int ndims, nvars, natts, ngatts, ngrps, ntyps;
|
|
|
|
#ifdef USE_NETCDF4
|
|
int igrp,ityp;
|
|
#endif
|
|
|
|
ndims = listlength(dimdefs);
|
|
nvars = listlength(vardefs);
|
|
natts = listlength(attdefs);
|
|
ngatts = listlength(gattdefs);
|
|
ngrps = listlength(grpdefs);
|
|
ntyps = listlength(typdefs);
|
|
|
|
#ifdef USE_NETCDF4
|
|
if(group != rootgroup) {
|
|
xline("");
|
|
xpartial("<group name=");
|
|
xpartial("\"");
|
|
xpartial(xname(group));
|
|
xline("\" >");
|
|
}
|
|
#endif
|
|
|
|
#ifdef USE_NETCDF4
|
|
/* Construct code to define types*/
|
|
if(ntyps > 0) {
|
|
xline("<!-- define types -->");
|
|
xline("<types>");
|
|
for(ityp = 0; ityp < ntyps; ityp++) {
|
|
Symbol* tsym = (Symbol*)listget(typdefs,ityp);
|
|
xline("");
|
|
if(tsym->subclass == NC_PRIM
|
|
|| tsym->subclass == NC_ARRAY) continue;/*no need to do these*/
|
|
definextype(tsym);
|
|
}
|
|
xline("");
|
|
xline("</types>");
|
|
}
|
|
xflush();
|
|
#endif
|
|
|
|
/* define dimensions from info in dims array */
|
|
if (ndims > 0) {
|
|
xline("");
|
|
xline("<!-- define dimensions -->");
|
|
for(idim = 0; idim < ndims; idim++) {
|
|
Symbol* dsym = (Symbol*)listget(dimdefs,idim);
|
|
unsigned long declsize = dsym->dim.size;
|
|
char* utag = "";
|
|
if(declsize == 0)
|
|
utag = " isUnlimited=\"true\"";
|
|
nprintf(stmt,sizeof(stmt),
|
|
"<dimension name=\"%s\" length=\"%lu\"%s />",
|
|
xname(dsym),declsize,utag);
|
|
xline(stmt);
|
|
}
|
|
}
|
|
xflush();
|
|
|
|
/* Define the global attributes*/
|
|
if(ngatts > 0) {
|
|
xline("");
|
|
xline("<!-- Define global attributes -->");
|
|
for(iatt = 0; iatt < ngatts; iatt++) {
|
|
Symbol* gasym = (Symbol*)listget(gattdefs,iatt);
|
|
if(gasym->att.var == NULL) /* => global */
|
|
defineattribute(gasym);
|
|
}
|
|
xline("");
|
|
}
|
|
xflush();
|
|
|
|
/* define variables from info in vars array */
|
|
if (nvars > 0) {
|
|
xline("");
|
|
xline("<!-- define variables -->");
|
|
for(ivar = 0; ivar < nvars; ivar++) {
|
|
Symbol* vsym = (Symbol*)listget(vardefs,ivar);
|
|
Dimset* dimset = &vsym->typ.dimset;
|
|
if(ivar > 0) xline("");
|
|
nprintf(stmt,sizeof(stmt),"<variable name=\"%s\"",xname(vsym));
|
|
xpartial(stmt);
|
|
if(dimset->ndims > 0) {
|
|
xpartial(" shape=\"");
|
|
for(idim = 0; idim < dimset->ndims; idim++) {
|
|
Symbol* dsym = dimset->dimsyms[idim];
|
|
if(idim > 0) xpartial(" ");
|
|
xpartial(xname(dsym));
|
|
}
|
|
xpartial("\"");
|
|
}
|
|
xpartial(" type=\"");
|
|
xpartial(xname(vsym->typ.basetype));
|
|
xline(" >");
|
|
#ifdef USE_NETCDF4
|
|
definespecialattributes(vsym);
|
|
#endif /*USE_NETCDF4*/
|
|
if(natts > 0) {
|
|
for(iatt = 0; iatt < natts; iatt++) {
|
|
Symbol* asym = (Symbol*)listget(attdefs,iatt);
|
|
if(asym->att.var == vsym) defineattribute(asym);
|
|
}
|
|
}
|
|
if(vsym->data != NULL) definevardata(vsym);
|
|
xline("</variable>");
|
|
}
|
|
}
|
|
xflush();
|
|
|
|
#ifdef USE_NETCDF4
|
|
/* Recursively define subgroups */
|
|
for(igrp=0;igrp<ngrps;igrp++) {
|
|
Symbol* gsym = (Symbol*)listget(grpdefs,igrp);
|
|
if(gsym->container != group) continue;
|
|
gencml_group(gsym);
|
|
}
|
|
#endif
|
|
|
|
#ifdef USE_NETCDF4
|
|
if(group != rootgroup) {
|
|
xline("</group>");
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#ifdef USE_NETCDF4
|
|
static void
|
|
definespecialattributes(Symbol* vsym)
|
|
{
|
|
Specialdata* special = &vsym->var.special;
|
|
if(usingclassic) return;
|
|
if(special->flags & _STORAGE_FLAG) {
|
|
int storage = special->_Storage;
|
|
size_t* chunks = special->_ChunkSizes;
|
|
nprintf(stmt,sizeof(stmt),"<attribute name=\"_Chunksizes\" storage=\"%s\"",
|
|
(storage == NC_CONTIGUOUS?"contiguous":"chunked"));
|
|
xpartial(stmt);
|
|
if(special->nchunks != 0 && chunks != NULL) {
|
|
int i;
|
|
xpartial(" value=\"");
|
|
for(i=0;i<special->nchunks;i++) {
|
|
if(i > 0) xpartial(" ");
|
|
nprintf(stmt,sizeof(stmt),"%ld",special->_ChunkSizes[i]);
|
|
xpartial(stmt);
|
|
}
|
|
xpartial("\"");
|
|
}
|
|
xline(" />");
|
|
}
|
|
if(special->flags & _FLETCHER32_FLAG) {
|
|
nprintf(stmt,sizeof(stmt),
|
|
"<attribute name=\"_Fletcher32\" value=\"%s\" />",
|
|
(special->_Fletcher32?"true":"false"));
|
|
xline(stmt);
|
|
}
|
|
if(special->flags & (_DEFLATE_FLAG | _SHUFFLE_FLAG)) {
|
|
nprintf(stmt,sizeof(stmt),
|
|
"<attribute name=\"_DeflateLevel\" shuffle=\"%s\" deflate=\"%s\" value=\"%d\" />",
|
|
(special->_Shuffle==1?"shuffle":"noshuffle"),
|
|
(special->_DeflateLevel >= 0?"true":"false"),
|
|
(special->_DeflateLevel >= 0?special->_DeflateLevel:0));
|
|
xline(stmt);
|
|
}
|
|
if(special->flags & _ENDIAN_FLAG) {
|
|
nprintf(stmt,sizeof(stmt),
|
|
"<attribute name=\"_Endianness\" value=\"%s\" />",
|
|
(special->_Endianness == NC_ENDIAN_LITTLE?"little":"big"));
|
|
xline(stmt);
|
|
}
|
|
if(special->flags & _NOFILL_FLAG) {
|
|
nprintf(stmt,sizeof(stmt),
|
|
"<attribute name=\"_Fill\" value=\"%s\" />",
|
|
(special->_Fill?"fill":"nofill"));
|
|
xline(stmt);
|
|
}
|
|
}
|
|
|
|
static void
|
|
definextype(Symbol* tsym)
|
|
{
|
|
int i;
|
|
|
|
ASSERT(tsym->objectclass == NC_TYPE);
|
|
switch (tsym->subclass) {
|
|
case NC_PRIM: break; /* these are already taken care of*/
|
|
case NC_OPAQUE:
|
|
nprintf(stmt,sizeof(stmt),"<opaque name=\"%s\" size=\"%lu\" />",
|
|
xname(tsym),(unsigned long)tsym->typ.size);
|
|
xline(stmt);
|
|
break;
|
|
case NC_ENUM:
|
|
nprintf(stmt,sizeof(stmt),"<enum name=\"%s\" type=\"%s\" >",
|
|
xname(tsym),xtypename(tsym->typ.basetype));
|
|
xline(stmt);
|
|
stmt[0] = 0;
|
|
for(i=0;i<listlength(tsym->subnodes);i++) {
|
|
Symbol* econst = (Symbol*)listget(tsym->subnodes,i);
|
|
ASSERT(econst->subclass == NC_ECONST);
|
|
nprintf(stmt,sizeof(stmt),"<enumconst name=\"%s\" value=\"%s\" />",
|
|
xname(econst),
|
|
xconst(econst->typ.econst));
|
|
xline(stmt);
|
|
}
|
|
xline("</enum>");
|
|
break;
|
|
case NC_VLEN:
|
|
nprintf(stmt,sizeof(stmt),"<vlen name=\"%s\" type=\"%s\" />",
|
|
xname(tsym),xtypename(tsym));
|
|
xline(stmt);
|
|
break;
|
|
case NC_COMPOUND:
|
|
nprintf(stmt,sizeof(stmt),"<compound name=\"%s\" >",
|
|
xname(tsym));
|
|
xline(stmt);
|
|
for(i=0;i<listlength(tsym->subnodes);i++) {
|
|
Symbol* efield = (Symbol*)listget(tsym->subnodes,i);
|
|
ASSERT(efield->subclass == NC_FIELD);
|
|
nprintf(stmt,sizeof(stmt),"<field name=\"%s\"",xname(efield));
|
|
xpartial(stmt);
|
|
/* compute any dimension specification*/
|
|
if(efield->typ.dimset.ndims > 0) {
|
|
Bytebuffer* dimbuf = bbNew();
|
|
int j;
|
|
bbCat(dimbuf," shape=\"");
|
|
for(j=0;j<efield->typ.dimset.ndims;j++) {
|
|
char tmp[32];
|
|
Symbol* dim;
|
|
if(j > 0) bbCat(dimbuf," ");
|
|
dim = efield->typ.dimset.dimsyms[j];
|
|
ASSERT(dim->dim.isconstant);
|
|
sprintf(tmp,"%lu",(unsigned long)dim->dim.size);
|
|
bbCat(dimbuf,tmp);
|
|
}
|
|
bbCat(dimbuf,"\"");
|
|
xprint(dimbuf);
|
|
bbFree(dimbuf);
|
|
}
|
|
nprintf(stmt,sizeof(stmt)," type=\"%s\" />",
|
|
xtypename(efield->typ.basetype));
|
|
xline(stmt);
|
|
}
|
|
xline("</compound>");
|
|
xflush();
|
|
break;
|
|
|
|
case NC_ARRAY:
|
|
/* ignore: this will be handled by def_var*/
|
|
break;
|
|
|
|
default: panic("definextype: unexpected type subclass: %d",tsym->subclass);
|
|
}
|
|
}
|
|
|
|
#endif /*USE_NETCDF4*/
|
|
|
|
static void
|
|
defineattribute(Symbol* asym)
|
|
{
|
|
unsigned long len;
|
|
Datalist* list;
|
|
Symbol* basetype = asym->typ.basetype;
|
|
Bytebuffer* code = NULL; /* capture other decls*/
|
|
|
|
list = asym->data;
|
|
if(list == NULL) PANIC("empty attribute list");
|
|
len = asym->att.count;
|
|
if(len == 0) PANIC("empty attribute list");
|
|
|
|
code = bbNew();
|
|
|
|
gencml_attrdata(asym,code);
|
|
|
|
/* Handle NC_CHAR specially */
|
|
if(basetype->typ.typecode == NC_CHAR) {
|
|
/* revise the length count */
|
|
len = bbLength(code);
|
|
if(len == 0) {bbAppend(code,'\0'); len++;}
|
|
xquotestring(code);
|
|
}
|
|
|
|
if(isprimplus(basetype->typ.typecode)) {
|
|
nprintf(stmt,sizeof(stmt),"<attribute name=\"%s\" type=\"%s\" value=\"%s\" />",
|
|
xname(asym),xtypename(asym->typ.basetype),
|
|
bbContents(code));
|
|
xline(stmt);
|
|
} else {
|
|
nprintf(stmt,sizeof(stmt),"<attribute name=\"%s\" type=\"%s\" >",
|
|
xname(asym),xtypename(asym->typ.basetype));
|
|
xline(stmt);
|
|
xpartial("<values>");
|
|
xpartial(bbContents(code));
|
|
xline("</values>");
|
|
xline("</attribute>");
|
|
}
|
|
xflush();
|
|
}
|
|
|
|
/* Note: no closure is used for CML */
|
|
static void
|
|
definevardata(Symbol* vsym)
|
|
{
|
|
Dimset* dimset = &vsym->typ.dimset;
|
|
int isscalar = (dimset->ndims == 0);
|
|
Bytebuffer* code;
|
|
nc_type typecode = vsym->typ.basetype->typ.typecode;
|
|
|
|
if(vsym->data == NULL) return;
|
|
|
|
code = bbNew();
|
|
|
|
/* Handle special cases first*/
|
|
if(isscalar) {
|
|
gencml_scalardata(vsym,code);
|
|
} else { /* Non-scalar*/
|
|
gencml_arraydata(vsym,code);
|
|
}
|
|
if(typecode == NC_STRING) {
|
|
xpartial("<values>");
|
|
xpartial(bbContents(code));
|
|
xline("</values>");
|
|
} else {
|
|
xline("<values>");
|
|
xpartial(bbContents(code));
|
|
if(isprimplus(vsym->typ.basetype->typ.typecode)) xline("");
|
|
xline("</values>");
|
|
}
|
|
bbFree(code);
|
|
}
|
|
|
|
/* return CML name for netCDF type, given type code */
|
|
static char *
|
|
ncxtype(nc_type type) /* netCDF type code */
|
|
{
|
|
switch (type) {
|
|
case NC_CHAR: return "char";
|
|
case NC_BYTE: return "byte";
|
|
case NC_SHORT: return "short";
|
|
case NC_INT: return "int";
|
|
case NC_FLOAT: return "float";
|
|
case NC_DOUBLE: return "double";
|
|
case NC_UBYTE: return "ubyte";
|
|
case NC_USHORT: return "ushort";
|
|
case NC_UINT: return "uint";
|
|
case NC_INT64: return "int64";
|
|
case NC_UINT64: return "uint64";
|
|
case NC_STRING: return "string";
|
|
default: PANIC("nctype: bad type code");
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/* Compute the CML name for a given type*/
|
|
char*
|
|
xtypename(Symbol* tsym)
|
|
{
|
|
char* name;
|
|
ASSERT(tsym->objectclass == NC_TYPE);
|
|
if(tsym->subclass == NC_PRIM)
|
|
name = ncxtype(tsym->typ.typecode);
|
|
else
|
|
name = xname(tsym);
|
|
return name;
|
|
}
|
|
|
|
/* Compute the CML name for a given symbol*/
|
|
/* Cache in symbol->lname*/
|
|
char*
|
|
xname(Symbol* sym)
|
|
{
|
|
if(sym->lname == NULL) {
|
|
char* name = pooldup(sym->name);
|
|
#ifdef USE_NETCDF4
|
|
if(sym->subclass == NC_FIELD || sym->subclass == NC_ECONST) {
|
|
sym->lname = nulldup(codify(name));
|
|
} else
|
|
sym->lname = nulldup(codify(xprefixed(sym->prefix,name,"/")));
|
|
#else
|
|
sym->lname = nulldup(codify(name)); /* convert to usable form*/
|
|
#endif
|
|
}
|
|
return sym->lname;
|
|
}
|
|
|
|
#ifdef USE_NETCDF4
|
|
/* Result is pool alloc'd*/
|
|
static char*
|
|
xprefixed(List* prefix, char* suffix, char* separator)
|
|
{
|
|
int slen;
|
|
int plen;
|
|
int i;
|
|
char* result;
|
|
|
|
ASSERT(suffix != NULL);
|
|
plen = prefixlen(prefix);
|
|
if(prefix == NULL || plen == 0) return codify(suffix);
|
|
/* plen > 0*/
|
|
slen = 0;
|
|
for(i=0;i<plen;i++) {
|
|
Symbol* sym = (Symbol*)listget(prefix,i);
|
|
slen += (strlen(sym->name)+strlen(separator));
|
|
}
|
|
slen += strlen(suffix);
|
|
slen++; /* for null terminator*/
|
|
result = poolalloc(slen);
|
|
result[0] = '\0';
|
|
/* Leave off the root*/
|
|
i = (rootgroup == (Symbol*)listget(prefix,0))?1:0;
|
|
for(;i<plen;i++) {
|
|
Symbol* sym = (Symbol*)listget(prefix,i);
|
|
strcat(result,sym->name); /* append "<prefix[i]/>"*/
|
|
strcat(result,separator);
|
|
}
|
|
strcat(result,suffix); /* append "<suffix>"*/
|
|
return result;
|
|
}
|
|
#endif
|
|
|
|
/**************************************************/
|
|
/*
|
|
* Output a CML statement
|
|
*/
|
|
|
|
#ifdef USE_NETCDF4
|
|
static void
|
|
xprint(Bytebuffer* buf)
|
|
{
|
|
bbAppendn(xcode,bbContents(buf),bbLength(buf));
|
|
}
|
|
#endif
|
|
|
|
static void
|
|
xpartial(char* line)
|
|
{
|
|
bbCat(xcode,line);
|
|
}
|
|
|
|
static void
|
|
xline(char* line)
|
|
{
|
|
xpartial(line);
|
|
bbCat(xcode,"\n");
|
|
}
|
|
|
|
static void
|
|
xflush(void)
|
|
{
|
|
if(bbLength(xcode) > 0) {
|
|
bbAppend(xcode,'\0');
|
|
fputs(bbContents(xcode),stdout);
|
|
fflush(stdout);
|
|
bbClear(xcode);
|
|
}
|
|
}
|
|
|
|
#endif /*ENABLE_CML*/
|