mirror of
https://github.com/Unidata/netcdf-c.git
synced 2025-01-12 15:45:21 +08:00
d2316f866c
Primary Fixes: * Add a whole variable optimization -- used in the rare case that nc_get/put_vara covers the whole of a variable and the variable has a single chunk. * Fix chunking error when stride causes whole chunks to be skipped. * Fix some memory leaks * Add test cases * Add one performance test to nczarr_test/. This uses the timer utils from unit_test: timer_utils.[ch]. * Move ncdumpchunks utility from ncdump to nczarr_test Misc. Other Changes: * Make check for aws libraries conditional on --enable-nczarr-s3 * Remove all but one bm tests from nczarr_test until they are working. * Remove another dependency on HDF5 from supposedly non-HDF5 specific code; specifically hdf5_log_hdf5. * Make the BAIL2 macro be hdf5 specific and replace elsewhere with an HDF5 independent equivalent. * Move hdf5cache.c to libsrc4/nc4cache.c because it is used by nczarr. * Modify unit_tests so that some of them are run even if using Windows. * Misc. small bug fixes and refactors and memory leaks. * Rename some conflicting tests for cmake. * Attempted to make nc_perf work with cmake and failed.
648 lines
17 KiB
C
648 lines
17 KiB
C
/**
|
|
* @file
|
|
* @internal Add provenance info for netcdf-4 files.
|
|
*
|
|
* Copyright 2018, UCAR/Unidata See netcdf/COPYRIGHT file for copying
|
|
* and redistribution conditions.
|
|
* @author Dennis Heimbigner
|
|
*/
|
|
|
|
#include "zincludes.h"
|
|
#include "nc_provenance.h"
|
|
|
|
/* Provide a hack to suppress the writing of _NCProperties attribute.
|
|
This is for creating a file without _NCProperties for testing purposes.
|
|
*/
|
|
#undef SUPPRESSNCPROPS
|
|
|
|
/* Various Constants */
|
|
#define NCPROPS_MAX_NAME 1024 /* max key name size */
|
|
#define NCPROPS_MAX_VALUE 1024 /* max value size */
|
|
#define NCZ_MAX_NAME 1024 /**< ZARR max name. */
|
|
|
|
#define ESCAPECHARS "\\=|,"
|
|
|
|
#define NCPNCZLIB "nczarr"
|
|
|
|
/** @internal Check NetCDF return code. */
|
|
#define NCHECK(expr) {if((expr)!=NC_NOERR) {goto done;}}
|
|
|
|
/** @internal Check ZARR return code. */
|
|
#define HCHECK(expr) {if((expr)<0) {ncstat = NC_EHDFERR; goto done;}}
|
|
|
|
static int NCZ_read_ncproperties(NC_FILE_INFO_T* h5, const char* value, char** propstring);
|
|
static int NCZ_write_ncproperties(NC_FILE_INFO_T* h5);
|
|
|
|
static int globalpropinitialized = 0;
|
|
static NC4_Provenance globalprovenance;
|
|
|
|
/**
|
|
* @internal Initialize default provenance info
|
|
* This will only be used for newly created files
|
|
* or for opened files that do not contain an _NCProperties
|
|
* attribute.
|
|
*
|
|
* @return ::NC_NOERR No error.
|
|
* @author Dennis Heimbigner
|
|
*/
|
|
int
|
|
NCZ_provenance_init(void)
|
|
{
|
|
int stat = NC_NOERR;
|
|
char* name = NULL;
|
|
char* value = NULL;
|
|
unsigned long major, minor, release;
|
|
NCbytes* buffer = NULL; /* for constructing the global _NCProperties */
|
|
char printbuf[1024];
|
|
|
|
if(globalpropinitialized)
|
|
return stat;
|
|
|
|
/* Build _NCProperties info */
|
|
|
|
/* Initialize globalpropinfo */
|
|
memset((void*)&globalprovenance,0,sizeof(NC4_Provenance));
|
|
globalprovenance.version = NCPROPS_VERSION;
|
|
|
|
buffer = ncbytesnew();
|
|
|
|
/* Insert version as first entry */
|
|
ncbytescat(buffer,NCPVERSION);
|
|
ncbytescat(buffer,"=");
|
|
|
|
snprintf(printbuf,sizeof(printbuf),"%d",globalprovenance.version);
|
|
ncbytescat(buffer,printbuf);
|
|
|
|
/* Insert the netcdf version */
|
|
ncbytesappend(buffer,NCPROPSSEP2);
|
|
ncbytescat(buffer,NCPNCLIB2);
|
|
ncbytescat(buffer,"=");
|
|
ncbytescat(buffer,PACKAGE_VERSION);
|
|
|
|
/* This should be redundant since netcdf version => zarr format */
|
|
/* Insert the ZARR as underlying storage format library */
|
|
ncbytesappend(buffer,NCPROPSSEP2);
|
|
ncbytescat(buffer,NCPNCZLIB);
|
|
ncbytescat(buffer,"=");
|
|
if((stat = NCZ_get_libversion(&major,&minor,&release))) return stat;
|
|
snprintf(printbuf,sizeof(printbuf),"%lu.%lu.%lu",major,minor,release);
|
|
ncbytescat(buffer,printbuf);
|
|
|
|
#ifdef NCPROPERTIES_EXTRA
|
|
if(NCPROPERTIES_EXTRA != NULL && strlen(NCPROPERTIES_EXTRA) > 0)
|
|
{
|
|
/* Add any extra fields */
|
|
const char* p = NCPROPERTIES_EXTRA;
|
|
if(p != NULL && strlen(p) > 0) {
|
|
if(p[0] == NCPROPSSEP2) p++; /* If leading separator */
|
|
ncbytesappend(buffer,NCPROPSSEP2);
|
|
ncbytescat(buffer,p);
|
|
}
|
|
}
|
|
#endif
|
|
ncbytesnull(buffer);
|
|
globalprovenance.ncproperties = ncbytesextract(buffer);
|
|
|
|
ncbytesfree(buffer);
|
|
if(name != NULL) free(name);
|
|
if(value != NULL) free(value);
|
|
if(stat == NC_NOERR)
|
|
globalpropinitialized = 1; /* avoid repeating it */
|
|
return stat;
|
|
}
|
|
|
|
/**
|
|
* @internal finalize default provenance info
|
|
*
|
|
* @return ::NC_NOERR No error.
|
|
* @author Dennis Heimbigner
|
|
*/
|
|
int
|
|
NCZ_provenance_finalize(void)
|
|
{
|
|
return NCZ_clear_provenance(&globalprovenance);
|
|
}
|
|
|
|
/**
|
|
* @internal
|
|
*
|
|
* Construct the provenance information for a newly created file.
|
|
* Note that creation of the _NCProperties attribute is deferred
|
|
* to the sync_netcdf4_file function.
|
|
*
|
|
* @param file Pointer to file object.
|
|
*
|
|
* @return ::NC_NOERR No error.
|
|
* [Note: other errors are reported via LOG()]
|
|
* @author Dennis Heimbigner
|
|
*/
|
|
int
|
|
NCZ_new_provenance(NC_FILE_INFO_T* file)
|
|
{
|
|
int stat = NC_NOERR;
|
|
NC4_Provenance* provenance = NULL;
|
|
int superblock;
|
|
|
|
LOG((5, "%s: ncid 0x%x", __func__, file->root_grp->hdr.id));
|
|
|
|
assert(file->provenance.ncproperties == NULL); /* not yet defined */
|
|
|
|
provenance = &file->provenance;
|
|
memset(provenance,0,sizeof(NC4_Provenance)); /* make sure */
|
|
|
|
/* Set the version */
|
|
provenance->version = globalprovenance.version;
|
|
|
|
/* Set the superblock number */
|
|
if((stat = NCZ_get_superblock(file,&superblock))) goto done;
|
|
provenance->superblockversion = superblock;
|
|
|
|
if(globalprovenance.ncproperties != NULL) {
|
|
if((provenance->ncproperties = strdup(globalprovenance.ncproperties)) == NULL)
|
|
{stat = NC_ENOMEM; goto done;}
|
|
}
|
|
|
|
done:
|
|
if(stat) {
|
|
LOG((0,"Could not create _NCProperties attribute"));
|
|
}
|
|
return NC_NOERR;
|
|
}
|
|
|
|
/**
|
|
* @internal
|
|
*
|
|
* Construct the provenance information for an existing file.
|
|
*
|
|
* @param file Pointer to file object.
|
|
*
|
|
* @return ::NC_NOERR No error.
|
|
* [Note: other errors are reported via LOG()]
|
|
* @author Dennis Heimbigner
|
|
*/
|
|
int
|
|
NCZ_read_provenance(NC_FILE_INFO_T* file, const char* name, const char* value)
|
|
{
|
|
int stat = NC_NOERR;
|
|
NC4_Provenance* provenance = NULL;
|
|
char* propstring = NULL;
|
|
|
|
LOG((5, "%s: ncid 0x%x", __func__, file->root_grp->hdr.id));
|
|
|
|
assert(file->provenance.version == 0); /* not yet defined */
|
|
|
|
provenance = &file->provenance;
|
|
memset(provenance,0,sizeof(NC4_Provenance)); /* make sure */
|
|
|
|
/* Set the superblock number */
|
|
int superblock = -1;
|
|
if((stat = NCZ_get_superblock(file,&superblock))) goto done;
|
|
provenance->superblockversion = superblock;
|
|
|
|
/* Process the _NCProperties value */
|
|
if(strcmp(name,NCPROPS)==0) {
|
|
if((stat = NCZ_read_ncproperties(file,value,&propstring))) goto done;
|
|
provenance->ncproperties = propstring;
|
|
propstring = NULL;
|
|
}
|
|
|
|
done:
|
|
nullfree(propstring);
|
|
if(stat) {
|
|
LOG((0,"Could not create _NCProperties attribute"));
|
|
}
|
|
return NC_NOERR;
|
|
}
|
|
|
|
/**
|
|
* @internal
|
|
*
|
|
* Add the provenance information to a newly created file.
|
|
*
|
|
* @param file Pointer to file object.
|
|
*
|
|
* @return ::NC_NOERR No error.
|
|
* [Note: other errors are reported via LOG()]
|
|
* @author Dennis Heimbigner
|
|
*/
|
|
int
|
|
NCZ_write_provenance(NC_FILE_INFO_T* file)
|
|
{
|
|
int stat = NC_NOERR;
|
|
if((stat = NCZ_write_ncproperties(file)))
|
|
goto done;
|
|
done:
|
|
return stat;
|
|
}
|
|
|
|
/* ZARR Specific attribute read/write of _NCProperties */
|
|
static int
|
|
NCZ_read_ncproperties(NC_FILE_INFO_T* h5, const char* value, char** propstring)
|
|
{
|
|
int stat = NC_NOERR;
|
|
char* text = NULL;
|
|
size_t len;
|
|
|
|
LOG((5, "%s", __func__));
|
|
|
|
/* NCPROPS Attribute exists, make sure it is legitimate */
|
|
if(value == NULL || strlen(value) == 0)
|
|
{stat = NC_EINVAL; goto done;}
|
|
len = strlen(value);
|
|
text = (char*)malloc(1+len);
|
|
if(text == NULL) {stat = NC_ENOMEM; goto done;}
|
|
memcpy(text,value,len);
|
|
/* Make sure its null terminated */
|
|
text[len] = '\0';
|
|
if(propstring) {*propstring = text; text = NULL;}
|
|
|
|
done:
|
|
if(text != NULL) free(text);
|
|
/* For certain errors, actually fail, else log that attribute was invalid and ignore */
|
|
if(stat != NC_NOERR) {
|
|
if(stat != NC_ENOMEM && stat != NC_EHDFERR) {
|
|
LOG((0,"Invalid _NCProperties attribute: ignored"));
|
|
stat = NC_NOERR;
|
|
}
|
|
}
|
|
return stat;
|
|
}
|
|
|
|
static int
|
|
NCZ_write_ncproperties(NC_FILE_INFO_T* h5)
|
|
{
|
|
#ifdef SUPPRESSNCPROPERTY
|
|
return NC_NOERR;
|
|
#else /*!SUPPRESSNCPROPERTY*/
|
|
int i,stat = NC_NOERR;
|
|
NC4_Provenance* prov = &h5->provenance;
|
|
NC_ATT_INFO_T* ncprops = NULL;
|
|
NCindex* attlist = NULL;
|
|
|
|
LOG((5, "%s", __func__));
|
|
|
|
/* If the file is read-only, return an error. */
|
|
if (h5->no_write)
|
|
{stat = NC_EPERM; goto done;}
|
|
|
|
/* See if it already exists */
|
|
/* Load the root group attributes */
|
|
if((stat = ncz_getattlist(h5->root_grp,NC_GLOBAL,NULL,&attlist)))
|
|
goto done;
|
|
|
|
/* Look for _NCProperties */
|
|
for(i=0; i<ncindexsize(attlist); i++) {
|
|
NC_ATT_INFO_T* att = (NC_ATT_INFO_T*)ncindexith(attlist,i);
|
|
if(strcmp(NCPROPS,att->hdr.name)==0) {
|
|
ncprops = att;
|
|
break;
|
|
}
|
|
}
|
|
if(ncprops != NULL) goto done; /* Already exists, no overwrite */
|
|
|
|
/* Build the property if we have legit value */
|
|
if(prov->ncproperties != NULL) {
|
|
if((stat=nc4_att_list_add(attlist,NCPROPS,&ncprops)))
|
|
goto done;
|
|
ncprops->nc_typeid = NC_CHAR;
|
|
ncprops->len = strlen(prov->ncproperties);
|
|
if((ncprops->data = strdup(prov->ncproperties)) == NULL)
|
|
{stat = NC_ENOMEM; goto done;}
|
|
ncprops->dirty = 1;
|
|
if((ncprops->format_att_info = calloc(1,sizeof(NCZ_ATT_INFO_T)))==NULL)
|
|
{stat = NC_ENOMEM; goto done;}
|
|
((NCZ_ATT_INFO_T*)ncprops->format_att_info)->common.file = h5;
|
|
}
|
|
|
|
done:
|
|
/* For certain errors, actually fail, else log that attribute was invalid and ignore */
|
|
switch (stat) {
|
|
case NC_ENOMEM:
|
|
case NC_EHDFERR:
|
|
case NC_EPERM:
|
|
case NC_EFILEMETA:
|
|
case NC_NOERR:
|
|
break;
|
|
default:
|
|
LOG((0,"Invalid _NCProperties attribute"));
|
|
stat = NC_NOERR;
|
|
break;
|
|
}
|
|
return stat;
|
|
#endif /*!SUPPRESSNCPROPERTY*/
|
|
}
|
|
|
|
/**************************************************/
|
|
/* Utilities */
|
|
|
|
/* Debugging */
|
|
|
|
void
|
|
nczprintprovenance(NC4_Provenance* info)
|
|
{
|
|
fprintf(stderr,"[%p] version=%d superblockversion=%d ncproperties=|%s|\n",
|
|
info,
|
|
info->version,
|
|
info->superblockversion,
|
|
(info->ncproperties==NULL?"":info->ncproperties));
|
|
}
|
|
|
|
/**
|
|
* @internal
|
|
*
|
|
* Clear the NCPROVENANCE object; do not free it
|
|
* @param prov Pointer to provenance object
|
|
*
|
|
* @return ::NC_NOERR No error.
|
|
* @author Dennis Heimbigner
|
|
*/
|
|
int
|
|
NCZ_clear_provenance(NC4_Provenance* prov)
|
|
{
|
|
LOG((5, "%s", __func__));
|
|
|
|
if(prov == NULL) return NC_NOERR;
|
|
nullfree(prov->ncproperties);
|
|
memset(prov,0,sizeof(NC4_Provenance));
|
|
return NC_NOERR;
|
|
}
|
|
|
|
#if 0
|
|
/* Unused functions */
|
|
|
|
/**
|
|
* @internal Parse file properties.
|
|
*
|
|
* @param text0 Text properties.
|
|
* @param pairs list of parsed (key,value) pairs
|
|
*
|
|
* @return ::NC_NOERR No error.
|
|
* @author Dennis Heimbigner
|
|
*/
|
|
static int
|
|
properties_parse(const char* text0, NClist* pairs)
|
|
{
|
|
int ret = NC_NOERR;
|
|
char* p;
|
|
char* q;
|
|
char* text = NULL;
|
|
|
|
if(text0 == NULL || strlen(text0) == 0)
|
|
goto done;
|
|
|
|
text = strdup(text0);
|
|
if(text == NULL) return NC_ENOMEM;
|
|
|
|
/* For back compatibility with version 1, translate '|' -> ',' */
|
|
for(p=text;*p;p++) {
|
|
if(*p == NCPROPSSEP1)
|
|
*p = NCPROPSSEP2;
|
|
}
|
|
|
|
/* Walk and fill in ncinfo */
|
|
p = text;
|
|
while(*p) {
|
|
char* name = p;
|
|
char* value = NULL;
|
|
char* next = NULL;
|
|
|
|
/* Delimit whole (key,value) pair */
|
|
q = locate(p,NCPROPSSEP2);
|
|
if(*q != '\0') /* Never go beyond the final nul term */
|
|
*q++ = '\0';
|
|
next = q;
|
|
/* split key and value */
|
|
q = locate(p,'=');
|
|
name = p;
|
|
*q++ = '\0';
|
|
value = q;
|
|
/* Set up p for next iteration */
|
|
p = next;
|
|
nclistpush(pairs,strdup(name));
|
|
nclistpush(pairs,strdup(value));
|
|
}
|
|
done:
|
|
if(text) free(text);
|
|
return ret;
|
|
}
|
|
|
|
/* Locate a specific character and return its pointer
|
|
or EOS if not found
|
|
take \ escapes into account */
|
|
static char*
|
|
locate(char* p, char tag)
|
|
{
|
|
char* next;
|
|
int c;
|
|
assert(p != NULL);
|
|
for(next = p;(c = *next);next++) {
|
|
if(c == tag)
|
|
return next;
|
|
else if(c == '\\' && next[1] != '\0')
|
|
next++; /* skip escaped char */
|
|
}
|
|
return next; /* not found */
|
|
}
|
|
|
|
/* Utility to transfer a string to a buffer with escaping */
|
|
static void
|
|
escapify(NCbytes* buffer, const char* s)
|
|
{
|
|
const char* p;
|
|
for(p=s;*p;p++) {
|
|
if(strchr(ESCAPECHARS,*p) != NULL)
|
|
ncbytesappend(buffer,'\\');
|
|
ncbytesappend(buffer,*p);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @internal
|
|
*
|
|
* Clear and Free the NC4_Provenance object
|
|
* @param prov Pointer to provenance object
|
|
*
|
|
* @return ::NC_NOERR No error.
|
|
* @author Dennis Heimbigner
|
|
*/
|
|
static int
|
|
NCZ_free_provenance(NC4_Provenance* prov)
|
|
{
|
|
LOG((5, "%s", __func__));
|
|
|
|
if(prov == NULL) return NC_NOERR;
|
|
NCZ_clear_provenance(prov);
|
|
free(prov);
|
|
return NC_NOERR;
|
|
}
|
|
|
|
/**
|
|
* @internal Build _NCProperties attribute value.
|
|
*
|
|
* Convert a NCPROPINFO instance to a single string.
|
|
* Will always convert to current format
|
|
*
|
|
* @param version
|
|
* @param list Properties list
|
|
* @param spropp Pointer that gets properties string.
|
|
* @return ::NC_NOERR No error.
|
|
* @return ::NC_EINVAL failed.
|
|
* @author Dennis Heimbigner
|
|
*/
|
|
static int
|
|
build_propstring(int version, NClist* list, char** spropp)
|
|
{
|
|
int stat = NC_NOERR;
|
|
int i;
|
|
NCbytes* buffer = NULL;
|
|
char sversion[64];
|
|
|
|
LOG((5, "%s version=%d", __func__, version));
|
|
|
|
if(spropp != NULL) *spropp = NULL;
|
|
|
|
if(version == 0 || version > NCPROPS_VERSION) /* unknown case */
|
|
goto done;
|
|
if(list == NULL)
|
|
{stat = NC_EINVAL; goto done;}
|
|
|
|
if((buffer = ncbytesnew()) == NULL)
|
|
{stat = NC_ENOMEM; goto done;}
|
|
|
|
/* start with version */
|
|
ncbytescat(buffer,NCPVERSION);
|
|
ncbytesappend(buffer,'=');
|
|
/* Use current version */
|
|
snprintf(sversion,sizeof(sversion),"%d",version);
|
|
ncbytescat(buffer,sversion);
|
|
|
|
for(i=0;i<nclistlength(list);i+=2) {
|
|
char* value, *name;
|
|
name = nclistget(list,i);
|
|
if(name == NULL) continue;
|
|
value = nclistget(list,i+1);
|
|
ncbytesappend(buffer,NCPROPSSEP2); /* terminate last entry */
|
|
escapify(buffer,name);
|
|
ncbytesappend(buffer,'=');
|
|
escapify(buffer,value);
|
|
}
|
|
/* Force null termination */
|
|
ncbytesnull(buffer);
|
|
if(spropp) *spropp = ncbytesextract(buffer);
|
|
|
|
done:
|
|
if(buffer != NULL) ncbytesfree(buffer);
|
|
return stat;
|
|
}
|
|
|
|
static int
|
|
properties_getversion(const char* propstring, int* versionp)
|
|
{
|
|
int stat = NC_NOERR;
|
|
int version = 0;
|
|
/* propstring should begin with "version=dddd" */
|
|
if(propstring == NULL || strlen(propstring) < strlen("version=") + strlen("1"))
|
|
{stat = NC_EINVAL; goto done;} /* illegal version */
|
|
if(memcmp(propstring,"version=",strlen("version=")) != 0)
|
|
{stat = NC_EINVAL; goto done;} /* illegal version */
|
|
propstring += strlen("version=");
|
|
/* get version */
|
|
version = atoi(propstring);
|
|
if(version < 0)
|
|
{stat = NC_EINVAL; goto done;} /* illegal version */
|
|
if(versionp) *versionp = version;
|
|
done:
|
|
return stat;
|
|
}
|
|
|
|
/**
|
|
* @internal
|
|
*
|
|
* Construct the parsed provenance information
|
|
*
|
|
* @param prov Pointer to provenance object
|
|
*
|
|
* @return ::NC_NOERR No error.
|
|
* @return ::NC_ENOMEM
|
|
* @return ::NC_EINVAL
|
|
* @author Dennis Heimbigner
|
|
*/
|
|
static int
|
|
parse_provenance(NC4_Provenance* prov)
|
|
{
|
|
int stat = NC_NOERR;
|
|
char *name = NULL;
|
|
char *value = NULL;
|
|
int version = 0;
|
|
NClist* list = NULL;
|
|
|
|
LOG((5, "%s: prov 0x%x", __func__, prov));
|
|
|
|
if(prov->ncproperty == NULL || strlen(prov->ncproperty) < strlen("version="))
|
|
{stat = NC_EINVAL; goto done;}
|
|
if((list = nclistnew()) == NULL)
|
|
{stat = NC_ENOMEM; goto done;}
|
|
|
|
/* Do we understand the version? */
|
|
if(prov->version > 0 && prov->version <= NCPROPS_VERSION) {/* recognized version */
|
|
if((stat=properties_parse(prov->ncproperty,list)))
|
|
goto done;
|
|
/* Remove version pair from properties list*/
|
|
if(nclistlength(list) < 2)
|
|
{stat = NC_EINVAL; goto done;} /* bad _NCProperties attribute */
|
|
/* Throw away the purported version=... */
|
|
nclistremove(list,0); /* version key */
|
|
nclistremove(list,0); /* version value */
|
|
|
|
/* Now, rebuild to the latest version */
|
|
switch (version) {
|
|
default: break; /* do nothing */
|
|
case 1: {
|
|
int i;
|
|
for(i=0;i<nclistlength(list);i+=2) {
|
|
char* newname = NULL;
|
|
name = nclistget(list,i);
|
|
if(name == NULL) continue; /* ignore */
|
|
if(strcmp(name,NCPNCLIB1) == 0)
|
|
newname = NCPNCLIB2; /* change name */
|
|
else if(strcmp(name,NCPNCZLIB1) == 0)
|
|
newname = NCPNCZLIB2;
|
|
else continue; /* ignore */
|
|
/* Do any rename */
|
|
nclistset(list,i,strdup(newname));
|
|
if(name) {free(name); name = NULL;}
|
|
}
|
|
} break;
|
|
} /*switch*/
|
|
}
|
|
prov->properties = list;
|
|
list = NULL;
|
|
|
|
done:
|
|
nclistfreeall(list);
|
|
if(name != NULL) free(name);
|
|
if(value != NULL) free(value);
|
|
return stat;
|
|
}
|
|
|
|
/* Utility to copy contents of the dfalt into an NCPROPINFO object */
|
|
static int
|
|
propinfo_default(NCZ_Properties* dst, const NCZ_Properties* dfalt)
|
|
{
|
|
int i;
|
|
if(dst->properties == NULL) {
|
|
dst->properties = nclistnew();
|
|
if(dst->properties == NULL) return NC_ENOMEM;
|
|
}
|
|
dst->version = dfalt->version;
|
|
for(i=0;i<nclistlength(dfalt->properties);i++) {
|
|
char* s = nclistget(dfalt->properties,i);
|
|
s = strdup(s);
|
|
if(s == NULL) return NC_ENOMEM;
|
|
nclistpush(dst->properties,s);
|
|
}
|
|
return NC_NOERR;
|
|
}
|
|
|
|
#endif /*0*/
|