/** \file \internal Basic NC_INMEMORY API tests both for netcdf-3 and netcdf-4 Copyright 2018, UCAR/Unidata. See COPYRIGHT file for copying and redistribution conditions. */ #undef DDBG #include "config.h" #include #include #ifdef HAVE_UNISTD_H #include #endif #include "netcdf.h" #include "netcdf_mem.h" #include "ncbytes.h" #include "nc_tests.h" #include "err_macros.h" #ifdef USE_HDF5 #include extern int H5Eprint1(FILE * stream); #endif #define FLAGS4 (NC_INMEMORY|NC_NETCDF4|NC_CLOBBER) #define FLAGS3 (NC_INMEMORY|NCCLOBBER) #define NC_NETCDF3 0 #define MODIFIED 1 #define EXTRA 1 #define LARGE_SPACE (1<<18) #define FILE3 "tst_inmemory3.nc" #define FILE4 "tst_inmemory4.nc" #define CREATE3 "tst_inmemory3_create.nc" #define CREATE4 "tst_inmemory4_create.nc" #define XFAIL "tst_xfail.nc" #define MISC "tst_misc.nc" #define CREATEFILE3 "tst_memcreate3.nc" #define CREATEFILE4 "tst_memcreate4.nc" /* Make no dimension larger than this */ #define MAXDIMLEN (1024*12) #define MAXDIMS 3 /* for defining arrays; includes all dimensions */ #define MAXVARS 4 /* for defining arrays; includes all variables */ #define NDIMS0 2 /* # dimensions in define_metadata: 1 unlimited + 1 fixed */ #define DIM0_NAME "fun" /* unlimited */ #define DIM0_LEN 2 #define DIM1_NAME "money" #define DIM1_LEN 400 #define ATT0_NAME "home" #define ATT0_TEXT "earthship" #define NVARS0 3 /* # variables in define_metadata */ #define VAR0_NAME "nightlife" #define VAR1_NAME "taxi_distance" #define VAR2_NAME "time" /* Variable added by modify_file */ #define VAR3_NAME "miles" /* Added variable info for modify_file_extra */ #define DIMX_NAME "bigmoney" #define DIMX_LEN 8000 #define VARX_NAME "expenses" #define FLOATVAL ((float)42.22) /* CDL for created file: netcdf tst_inmemory3 { dimensions: fun = UNLIMITED ; // (2 currently) money = 8 ; variables: int nightlife(fun, money) ; float time ; short taxi_distance(money) ; // global attributes: :home = "earthship" ; data: nightlife = 0, 100, 200, 300, 400, 500, 600, 700, 800, 900, 1000, 1100, 1200, 1300, 1400, 1500 ; time = 42.22 ; taxi_distance = 0, 1, 2, 3, 4, 5, 6, 7 ; } */ #ifdef DDBG #undef ERR static void fail(int line) { fflush(stdout); fprintf(stderr,"\nline=%d\n",line); fflush(stderr); exit(1); } #define ERR fail(__LINE__) #endif static int fail(int stat, const char* file, int line, int xfail) { fflush(stdout); fprintf(stderr,"***%sFail: line: %d; status=%d %s\n", (xfail?"X":""),line,stat,nc_strerror(stat)); err++; fflush(stderr); return stat; } static int xxfail(int stat, const char* file, int line, int xfail) { fflush(stdout); if(stat == NC_NOERR) return fail(NC_EINVAL,file,line,xfail); fprintf(stderr,"\t***XFail Pass: status=%d %s\n", stat,nc_strerror(stat)); stat = NC_NOERR; /* because xfail */ fflush(stderr); return stat; } static int check(int stat, const char* file, int line, int xfail) { if(!xfail && stat != NC_NOERR) return fail(stat,file,line,xfail); if(xfail) return xxfail(stat,file,line,xfail); return stat; } #define CHECK(expr) {stat = check((expr),__FILE__,__LINE__,0); if(stat) return stat;} #define XCHECK(expr) {stat = check((expr),__FILE__,__LINE__,1); if(stat) return stat;} #define REPORT(xfail,expr) {if((xfail)) {XCHECK((expr));} else {CHECK((expr));}} /**************************************************/ static void removefile(const char* path) { unlink(path); } static int readfile(const char* path, NC_memio* memio) { int status = NC_NOERR; FILE* f = NULL; size_t filesize = 0; size_t count = 0; char* memory = NULL; char* p = NULL; /* Open the file for reading */ #ifdef _MSC_VER f = fopen(path,"rb"); #else f = fopen(path,"r"); #endif if(f == NULL) {status = errno; goto done;} /* get current filesize */ if(fseek(f,0,SEEK_END) < 0) {status = errno; goto done;} filesize = (size_t)ftell(f); /* allocate memory */ memory = malloc((size_t)filesize); if(memory == NULL) {status = NC_ENOMEM; goto done;} /* move pointer back to beginning of file */ rewind(f); count = filesize; p = memory; while(count > 0) { size_t actual; actual = fread(p,1,count,f); if(actual == 0 || ferror(f)) {status = NC_EIO; goto done;} count -= actual; p += actual; } if(memio) { memio->size = (size_t)filesize; memio->memory = memory; } done: if(status != NC_NOERR && memory != NULL) free(memory); if(f != NULL) fclose(f); return status; } static int writefile(const char* path, NC_memio* memio) { int status = NC_NOERR; FILE* f = NULL; size_t count = 0; char* p = NULL; /* Open the file for writing */ #ifdef _MSC_VER f = fopen(path,"wb"); #else f = fopen(path,"w"); #endif if(f == NULL) {status = errno; goto done;} count = memio->size; p = memio->memory; while(count > 0) { size_t actual; actual = fwrite(p,1,count,f); if(actual == 0 || ferror(f)) {status = NC_EIO; goto done;} count -= actual; p += actual; } done: if(f != NULL) fclose(f); return status; } /* Duplicate an NC_memio instance; needed to avoid attempting to use memory that might have been realloc'd Allow the new memory to be larger than the src memory */ int duplicatememory(NC_memio* src, NC_memio* target, size_t alloc, void** original) { if(src == NULL || target == NULL || src->size == 0 || src->memory == NULL) return NC_EINVAL; *target = *src; if(alloc == 0) alloc = src->size; target->memory = malloc(alloc); if(target->memory == NULL) return NC_ENOMEM; if(original) *original = target->memory; memcpy(target->memory,src->memory,src->size); target->size = alloc; return NC_NOERR; } /* Given an ncid of a created file, fill in the meta-data and data as described by the above CDL. Do not close the file. */ static int define_metadata(int ncid) { int stat = NC_NOERR; int dimid[MAXDIMS], varid0, varid1, varid2; size_t start[1] = {0}; size_t count[1] = {DIM1_LEN}; int dimprod = (DIM0_LEN*DIM1_LEN); int i; float float_data; int nightdata[DIM0_LEN*DIM1_LEN] ; short taxi_distance[DIM1_LEN] ; /* Create data to write */ float_data = FLOATVAL; for (i = 0; i < DIM1_LEN; i++) taxi_distance[i] = i; for (i = 0; i < dimprod; i++) nightdata[i] = (100*i); CHECK(nc_put_att_text(ncid, NC_GLOBAL, ATT0_NAME, sizeof(ATT0_TEXT), ATT0_TEXT)); CHECK(nc_def_dim(ncid, DIM0_NAME, NC_UNLIMITED, &dimid[0])); CHECK(nc_def_dim(ncid, DIM1_NAME, DIM1_LEN, &dimid[1])); CHECK(nc_def_var(ncid, VAR0_NAME, NC_INT, 2, dimid, &varid0)); CHECK(nc_def_var(ncid, VAR1_NAME, NC_SHORT, 1, &dimid[1], &varid1)); CHECK(nc_def_var(ncid, VAR2_NAME, NC_FLOAT, 0, NULL, &varid2)); CHECK(nc_enddef(ncid)); CHECK(nc_put_vara_short(ncid, varid1, start, count, taxi_distance)); CHECK(nc_put_var_float(ncid, varid2, &float_data)); { size_t start[2] = {0,0}; size_t count[2] = {2,DIM1_LEN}; CHECK(nc_put_vara_int(ncid, varid0, start, count, nightdata)); } return stat; } /* Create our reference file as a real on-disk file and read it back into memory */ static int create_reference_file(const char* filename, int mode, NC_memio* filedata) { int stat = NC_NOERR; int ncid; CHECK(nc_create(filename, mode|NC_CLOBBER, &ncid)); /* overwrite */ CHECK(define_metadata(ncid)); CHECK(nc_close(ncid)); /* Read back the contents of the file into memory */ if(filedata != NULL) { memset(filedata,0,sizeof(NC_memio)); CHECK(readfile(filename,filedata)); } return stat; } static int modify_file(int ncid) { int stat = NC_NOERR; size_t i; int varid3; int dimid[1]; size_t len; int data[MAXDIMLEN]; /* Get id of the unlimited dimension */ if((stat=nc_inq_dimid(ncid, DIM0_NAME, dimid))) goto done; /* get current dim length */ if((stat=nc_inq_dimlen(ncid, dimid[0], &len))) goto done; /* open file for new meta-data */ if((stat=nc_redef(ncid))) goto done; /* Define a new variable */ if((stat=nc_def_var(ncid, VAR3_NAME, NC_INT, 1, dimid, &varid3))) goto done; /* close metadata */ if((stat=nc_enddef(ncid))) goto done; /* Write data to new variable */ for(i=0;imemory != NULL) free(memio->memory); memio->memory = NULL; } if(original) *original = NULL; } static int test_open(const char* path, NC_memio* filedata, int mode) { int stat = NC_NOERR; NC_memio duplicate; void* original = NULL; NC_memio finaldata; int ncid; int xmode = mode; /* modified mode */ finaldata.memory = NULL; finaldata.size = 0; finaldata.flags = 0; fprintf(stderr,"\n\t***Test open 1: nc_open_mem(): read-only\n"); CHECK(duplicatememory(filedata,&duplicate,0,&original)); CHECK(nc_open_mem(path, xmode, duplicate.size, duplicate.memory, &ncid)); CHECK(verify_file(ncid,!MODIFIED,!EXTRA)); CHECK(nc_close(ncid)); memiofree(&duplicate,&original); fprintf(stderr,"\n\t***Test open 2: nc_open_memio(): read-only\n"); CHECK(duplicatememory(filedata,&duplicate,0,&original)); duplicate.flags = NC_MEMIO_LOCKED; CHECK(nc_open_memio(path, xmode, &duplicate, &ncid)) CHECK(verify_file(ncid,!MODIFIED,!EXTRA)); CHECK(nc_close_memio(ncid,&finaldata)); /* Published returned finaldata */ fprintf(stderr,"\tfinaldata: size=%lld memory=%p original=%p\n",(unsigned long long)finaldata.size,finaldata.memory,original); /* Verify that finaldata is same */ if(finaldata.size != duplicate.size) CHECK(NC_EINVAL); if(finaldata.memory != duplicate.memory) CHECK(NC_EINVAL); memiofree(&finaldata,&original); fprintf(stderr,"\n\t***Test open 3: nc_open_memio(): read-write, copy, no size increase\n"); fprintf(stderr,"\t*** Not testable\n"); #if 0 { fprintf(stderr,"\n\t***Test open 3: nc_open_memio(): read-write, copy, no size increase\n"); xmode |= NC_WRITE; /* allow file to be modified */ CHECK(duplicatememory(filedata,&duplicate,0,&original)); CHECK(nc_open_memio(path, xmode, &duplicate, &ncid)) /* modify file */ CHECK(modify_file(ncid)); CHECK(verify_file(ncid,MODIFIED,!EXTRA)); CHECK(nc_close_memio(ncid,&finaldata)); /* Published returned finaldata */ fprintf(stderr,"\tfinaldata: size=%lld memory=%p original=%p\n",(unsigned long long)finaldata.size,finaldata.memory,original); /* Verify that finaldata is same */ if(finaldata.size < filedata->size) CHECK(NC_EINVAL); /* As a safeguard, the memory in duplicate should have been set to NULL*/ memiofree(&finaldata,&original); } #endif fprintf(stderr,"\n\t***Test open 4: nc_open_memio(): read-write, copy, size increase\n"); xmode |= NC_WRITE; /* allow file to be modified */ CHECK(duplicatememory(filedata,&duplicate,0,&original)); CHECK(nc_open_memio(path, xmode, &duplicate, &ncid)) /* modify file */ CHECK(modify_file(ncid)); CHECK(modify_file_extra(ncid)); CHECK(verify_file(ncid,MODIFIED,EXTRA)); CHECK(nc_close_memio(ncid,&finaldata)); /* Published returned finaldata */ fprintf(stderr,"\tfinaldata: size=%lld memory=%p original=%p\n",(unsigned long long)finaldata.size,finaldata.memory,original); /* Verify that finaldata is same */ if(finaldata.size < filedata->size) CHECK(NC_EINVAL); /* As a safeguard, the memory in duplicate should have been set to NULL*/ memiofree(&finaldata,&original); fprintf(stderr,"\n\t***Test open 5: nc_open_memio(): read-write, locked, extra space\n"); /* Store the filedata in a memory chunk that leaves room for modification */ CHECK(duplicatememory(filedata,&duplicate,LARGE_SPACE,&original)); /* Lock the duplicate memory */ duplicate.flags |= NC_MEMIO_LOCKED; xmode |= NC_WRITE; /* allow file to be modified */ CHECK(nc_open_memio(path, xmode, &duplicate, &ncid)) /* modify file */ CHECK(modify_file(ncid)); CHECK(verify_file(ncid,MODIFIED,!EXTRA)); CHECK(nc_close_memio(ncid,&finaldata)); /* Published returned finaldata */ fprintf(stderr,"\tfinaldata: size=%lld memory=%p original=%p\n",(unsigned long long)finaldata.size,finaldata.memory,original); /* Check returned finaldata: should have same memory but actual used final size should not exceed the original */ if(finaldata.size > duplicate.size) CHECK(NC_EINVAL); if(finaldata.memory != duplicate.memory) CHECK(NC_EINVAL); memiofree(&finaldata,&original); return stat; } static int test_create(const char* path, int mode) { int stat = NC_NOERR; NC_memio finaldata; int ncid; int xmode = mode; finaldata.memory = NULL; finaldata.size = 0; finaldata.flags = 0; fprintf(stderr,"\n\t***Test create 1: nc_create_memio(): no initialsize\n"); CHECK(nc_create_mem(path, xmode, 0, &ncid)) /* create file metadata */ CHECK(define_metadata(ncid)); CHECK(verify_file(ncid,!MODIFIED,!EXTRA)); CHECK(nc_close_memio(ncid,&finaldata)); /* Published returned finaldata */ fprintf(stderr,"\tfinaldata: size=%lld memory=%p\n",(unsigned long long)finaldata.size,finaldata.memory); free(finaldata.memory); fprintf(stderr,"\n\t***Test create 2: nc_create_memio(): initialsize; save file\n"); CHECK(nc_create_mem(path, xmode, LARGE_SPACE, &ncid)) /* create file metadata */ CHECK(define_metadata(ncid)); CHECK(verify_file(ncid,!MODIFIED,!EXTRA)); CHECK(nc_close_memio(ncid,&finaldata)); /* Published returned finaldata */ fprintf(stderr,"\tfinaldata: size=%lld memory=%p\n",(unsigned long long)finaldata.size,finaldata.memory); /* Write out the final data as a .nc file */ CHECK(writefile(path,&finaldata)); if(finaldata.memory != NULL) free(finaldata.memory); finaldata.memory = NULL; return stat; } static int test_misc(const char* path, int mode, NC_memio* filedata) { int stat = NC_NOERR; int ncid; int xmode = mode; NC_memio duplicate; void* original = NULL; fprintf(stderr,"\n\t***Test misc 1: use nc_close on created inmemory file\n"); CHECK(nc_create_mem(MISC, xmode, 0, &ncid)) CHECK(nc_close(ncid)); fprintf(stderr,"\t***Pass\n"); removefile(MISC); fprintf(stderr,"\n\t***Test misc 2: use nc_close on opened inmemory file\n"); CHECK(duplicatememory(filedata,&duplicate,0,&original)); CHECK(nc_open_memio(path, xmode, &duplicate, &ncid)) CHECK(verify_file(ncid,!MODIFIED,!EXTRA)); CHECK(nc_close(ncid)); fprintf(stderr,"\t***Pass\n"); /* Do not free: nc_close will have done it memiofree(&duplicate,&original); */ removefile(MISC); return stat; } /* Test various edge conditions to ensure they fail correctly */ static int test_xfail(const char* path, int mode, NC_memio* filedata) { int stat = NC_NOERR; NC_memio duplicate = {0,NULL,0}; int ncid; int xmode = mode; /* modified mode */ void* original = NULL; fprintf(stderr,"\n\t***Test xfail 1: nc_open_mem(): write to read-only\n"); CHECK(duplicatememory(filedata,&duplicate,0,&original)); CHECK(nc_open_mem(XFAIL, xmode, duplicate.size, duplicate.memory, &ncid)); XCHECK(nc_redef(ncid)); CHECK(nc_abort(ncid)); memiofree(&duplicate,&original); fprintf(stderr,"\n\t***Test xfail 2: nc_open_memio(): modify without overallocating\n"); if((mode & NC_NETCDF4)) { fprintf(stderr,"\t*** Suppressed because of HDF5 library bug\n"); } else { /* With HDF5 1.8.20, and possibly other versions, this tests causes a seg fault in the HDF5 Library. So until it is fixed, just leave well enough alone */ NC_memio finaldata; memset(&finaldata,0,sizeof(finaldata)); CHECK(duplicatememory(filedata,&duplicate,0,&original)); duplicate.flags = NC_MEMIO_LOCKED; xmode |= NC_WRITE; CHECK(nc_open_memio(XFAIL, xmode, &duplicate, &ncid)) XCHECK(modify_file(ncid)); CHECK(nc_abort(ncid)); memiofree(&finaldata,&original); memiofree(&duplicate,&original); } return stat; } int main(int argc, char **argv) { int stat = NC_NOERR; NC_memio filedata3; void* original = NULL; #ifdef USE_HDF5 NC_memio filedata4; #endif fprintf(stderr,"\n*** Testing the inmemory API: netcdf-3.\n"); CHECK(create_reference_file(FILE3,NC_NETCDF3,&filedata3)); /* netcdf-3 */ CHECK(test_open(FILE3,&filedata3,NC_NETCDF3)); CHECK(test_create(CREATE3,NC_NETCDF3)); CHECK(test_misc(FILE3, NC_NETCDF3, &filedata3)); CHECK(test_xfail(FILE3, NC_NETCDF3, &filedata3)); memiofree(&filedata3,&original); #ifdef USE_HDF5 fprintf(stderr,"\n*** Testing the inmemory API: netcdf-4.\n"); CHECK(create_reference_file(FILE4,NC_NETCDF4,&filedata4)); CHECK(test_open(FILE4,&filedata4,NC_NETCDF4)); CHECK(test_create(CREATE4,NC_NETCDF4)); CHECK(test_misc(FILE4,NC_NETCDF4, &filedata4)); CHECK(test_xfail(FILE4, NC_NETCDF4, &filedata4)); memiofree(&filedata4,&original); #endif SUMMARIZE_ERR; FINAL_RESULTS; return 0; }