#include "ncdap4.h" #include "dapodom.h" #include "dapalign.h" #ifdef DEBUG #include "dapdump.h" #endif #include "ncd3dispatch.h" static NCerror getcontent4(NCDAP4*, Getvara*, CDFnode* rootnode, void* data); static NCerror getcontent4r(NCDAP4*, Getvara*, CDFnode* tnode, OCdata, NCbytes*); static NCerror getcontent4prim(NCDAP4* drno, Getvara*, CDFnode* tnode, DCEsegment*, OCdata currentcontent, NCbytes* memory); static int findfield(CDFnode* node, CDFnode* subnode); static int contiguousdims(Dapodometer* odom); static NCerror ncdap4convert(nc_type srctype, nc_type dsttype, char* memory, char* value, size_t); int NCD4_get_vara(int ncid, int varid, const size_t* startp, const size_t* countp, void* data, nc_type externaltype0) { unsigned int i; NCerror ncstat = NC_NOERR; OCerror ocstat = OC_NOERR; NC_GRP_INFO_T *grp; NC_HDF5_FILE_INFO_T *h5; NC_VAR_INFO_T *var; NCDAP4* drno; NCDAPCOMMON* nccomm; CDFnode* cdfvar; /* cdf node mapping to var*/ NClist* varnodes; Getvara* varainfo = NULL; char* constraint = NULL; CDFnode* xtarget = NULL; CDFnode* target = NULL; DCEprojection* varaprojection = NULL; NCcachenode* cachenode = NULL; nc_type externaltype = externaltype0; size_t localcount[NC_MAX_VAR_DIMS]; NClist* ncdims; size_t ncrank; LOG((2, "nc_get_vara: ncid 0x%x varid %d", ncid, varid)); if((ncstat = nc4_find_nc_grp_h5(ncid, (NC_FILE_INFO_T**)&drno, &grp, &h5))) {THROWCHK(ncstat); goto fail;} nccomm = &drno->dap; /* Find the netcdf-4 var structure */ for(var=grp->var;var!=NULL;var=var->next) { if (var->varid == varid) break; } if(var == NULL) {ncstat = NC_ENOTVAR; goto fail;} /* Find cdfnode corresponding to the var.*/ varnodes = nccomm->cdf.varnodes; cdfvar = NULL; for(i=0;incid == varid) { cdfvar = node; break; } } ASSERT((cdfvar != NULL)); ASSERT((strcmp(cdfvar->ncfullname,var->name)==0)); /* Get the dimension info */ ncdims = cdfvar->array.dimensions; ncrank = nclistlength(ncdims); /* Fill in missing arguments */ if(startp == NULL) startp = dapzerostart3; if(countp == NULL) { /* Accumulate the dimension sizes */ for(i=0;idim.declsize; } countp = localcount; } /* Validate the dimension sizes */ for(i=0;i dim->dim.declsize || startp[i]+countp[i] > dim->dim.declsize) { ncstat = NC_EINVALCOORDS; goto fail; } } #ifdef DEBUG { NClist* dims = cdfvar->array.dimensions; fprintf(stderr,"getvarx: %s/%d",cdfvar->ncfullname,(int)nclistlength(dims)); if(nclistlength(dims) > 0) {int i; for(i=0;i "); for(i=0;iexternaltype; /* Validate any implied type conversion*/ if(cdfvar->nctype == NC_Primitive && cdfvar->etype != externaltype && externaltype == NC_CHAR) { /* The only disallowed conversion is to/from char and non-byte numeric types*/ switch (cdfvar->etype) { case NC_STRING: case NC_URL: case NC_CHAR: case NC_BYTE: case NC_UBYTE: break; default: THROWCHK(NC_ECHAR); goto fail; } } ncstat = makegetvar34(nccomm,cdfvar,data,externaltype,&varainfo); if(ncstat) {THROWCHK(NC_ENOMEM); goto fail;} #ifdef IGNORE freegetvara(nccomm->vara); nccomm->vara = varainfo; #endif ncstat = buildvaraprojection4(varainfo, startp,countp,dapsinglestride3, &varaprojection); if(ncstat != NC_NOERR) {THROWCHK(ncstat); goto fail;} if(FLAGSET(drno->dap.controls,NCF_UNCONSTRAINABLE)) { #ifdef DEBUG fprintf(stderr,"Unconstrained: reusing prefetch\n"); #endif cachenode = nccomm->cdf.cache->prefetch; ASSERT((cachenode != NULL)); } else if(iscached(&drno->dap,varaprojection->var->cdfleaf,&cachenode)) { #ifdef DEBUG fprintf(stderr,"Reusing cached fetch constraint: %s\n", dumpconstraint(cachenode->constraint)); #endif } else { /* Load with constraints */ NClist* vars = nclistnew(); DCEconstraint* constraint; nclistpush(vars,(ncelem)varainfo->target); constraint = (DCEconstraint*)dcecreate(CES_CONSTRAINT); constraint->projections = dceclonelist(nccomm->oc.dapconstraint->projections); if(!FLAGSET(drno->dap.controls,NCF_CACHE)) { /* If we are not caching, then merge the getvara projections */ NClist* tmp = nclistnew(); DCEprojection* clone = (DCEprojection*)dceclone((DCEnode*)varaprojection); nclistpush(tmp,(ncelem)clone); ncstat = mergeprojections3(constraint->projections,tmp); nclistfree(tmp); dcefree((DCEnode*)clone); if(ncstat != NC_NOERR) {THROWCHK(ncstat); goto fail;} #ifdef DEBUG fprintf(stderr,"vara merge: %s\n", dumpprojections(constraint->projections)); #endif } restrictprojection34(vars,constraint->projections); constraint->selections = dceclonelist(nccomm->oc.dapconstraint->selections); /* buildcachenode3 will also fetch the corresponding datadds */ ncstat = buildcachenode34(nccomm,constraint,vars,&cachenode,0); if(ncstat != NC_NOERR) {THROWCHK(ncstat); goto fail;} #ifdef DEBUG fprintf(stderr,"cache.datadds=%s\n",dumptree(cachenode->datadds)); #endif } /* attach DATADDS to DDS */ unattach34(nccomm->cdf.ddsroot); ncstat = attachsubset34(cachenode->datadds,nccomm->cdf.ddsroot); if(ncstat) goto fail; /* Fix up varainfo to use the cache */ varainfo->cache = cachenode; /* However use this current vara projection */ varainfo->varaprojection = varaprojection; varaprojection = NULL; /* Now, get the relevant instance*/ target = varainfo->target; /* xtarget is in the datadds space */ xtarget = target->attachment; if(xtarget == NULL) {THROWCHK(ncstat=NC_ENODATA); goto fail;} /* Switch to datadds tree space*/ varainfo->target = xtarget; ncstat = getcontent4(drno,varainfo,cachenode->datadds,data); if(ncstat != OC_NOERR) {THROWCHK(ncstat); goto fail;} goto ok; fail: if(ocstat != OC_NOERR) ncstat = ocerrtoncerr(ocstat); ok: nullfree(constraint); freegetvara(varainfo); dcefree((DCEnode*)varaprojection); return THROW(ncstat); } static NCerror getcontent4(NCDAP4* drno, Getvara* xgetvar, CDFnode* xroot, void* data) { NCerror ncstat = NC_NOERR; OCerror ocstat = OC_NOERR; NCbytes* memory; size_t alloc; int fieldindex; CDFnode* tnode = xgetvar->target; OCconnection conn = drno->dap.oc.conn; OCdata rootcontent = OCNULL; OCdata fieldcontent = OCNULL; OCdata dimcontent = OCNULL; OCdata seqcontent = OCNULL; OCdata gridcontent = OCNULL; Dapodometer* odom = NULL; OCobject ocroot = OCNULL; int caching = FLAGSET(drno->dap.controls,NCF_CACHE); int unconstrainable = FLAGSET(drno->dap.controls,NCF_UNCONSTRAINABLE); nc_type externaltype = xgetvar->dsttype; /* is var from a toplevel grid? */ int virtual = tnode->virtual; /* is var a toplevel Sequence? */ int seqvar = daptopseq(tnode); if(seqvar) { alloc = sizeof(nc_vlen_t); } if(tnode->nctype == NC_Primitive) { /* compute the # elements */ size_t nelems = dimproduct3(tnode->array.dimensions); /* Pull the field size from the externaltype. */ alloc = (nctypesizeof(externaltype)*nelems); } else { ASSERT((tnode->attachment != NULL)); /* Pull the field size from the node type size. */ alloc = (size_t)(tnode->attachment->typesize.field.size); } #ifdef READCHECK fprintf(stderr,"getcontent4: |%s| = %lu\n",tnode->name,alloc); #endif memory = ncbytesnew(); ncbytessetcontents(memory,data,alloc); /* Get the root content*/ ocroot = xroot->tree->ocroot; rootcontent = oc_data_new(conn); ocstat = oc_data_root(conn,ocroot,rootcontent); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} /* Move to the field for this var; handle grid vars separately */ if(virtual) { int gridindex; /* Get index of the grid node wrt to the root node */ gridindex = findfield(xroot,tnode->container); if(gridindex < 0) {ncstat = NC_ENOTVAR; goto fail;} gridcontent = oc_data_new(conn); ocstat = oc_data_ith(conn,rootcontent,gridindex,gridcontent); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} /* Now move to the relevant grid field */ fieldindex = findfield(tnode->container,tnode); if(fieldindex < 0) {ncstat = NC_ENOTVAR; goto fail;} fieldcontent = oc_data_new(conn); ocstat = oc_data_ith(conn,gridcontent,fieldindex,fieldcontent); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} } else if(seqvar) { /* Move to the seq data */ int seqindex = findfield(xroot,tnode); if(seqindex < 0) {ncstat = NC_ENOTVAR; goto fail;} seqcontent = oc_data_new(conn); ocstat = oc_data_ith(conn,rootcontent,seqindex,seqcontent); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} } else { fieldindex = findfield(xroot,tnode); if(fieldindex < 0) {ncstat = NC_ENOTVAR; goto fail;} if(tnode->virtual) { int structindex; /* Get index of the struct node wrt to the root node */ structindex = findfield(xroot,tnode->container); if(structindex < 0) {ncstat = NC_ENOTVAR; goto fail;} /* Get index of the structure field wrt to the virtual struct */ fieldindex = findfield(tnode->container,tnode); if(fieldindex < 0) {ncstat = NC_ENOTVAR; goto fail;} /* Actual field is structindex + fieldindex */ fieldindex += structindex; fieldcontent = oc_data_new(conn); ocstat = oc_data_ith(conn,fieldcontent,fieldindex,fieldcontent); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} } else { fieldcontent = oc_data_new(conn); ocstat = oc_data_ith(conn,rootcontent,fieldindex,fieldcontent); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} } } if(seqvar) { ASSERT((seqcontent != OCNULL)); ncstat=getcontent4r(drno,xgetvar,tnode,seqcontent,memory); } else if(tnode->nctype == NC_Primitive) { /* Stride the dimensions and get the instances */ DCEsegment* segment = NULL; ASSERT((nclistlength(xgetvar->varaprojection->var->segments)==1)); segment = (DCEsegment*)nclistget(xgetvar->varaprojection->var->segments,0); ASSERT((fieldcontent != OCNULL)); ncstat = getcontent4prim(drno,xgetvar,tnode,segment,fieldcontent,memory); } else if(nclistlength(tnode->array.dimensions) > 0) { /* Stride the dimensions and get the instances */ DCEsegment* segment = NULL; ASSERT((nclistlength(xgetvar->varaprojection->var->segments)==1)); segment = (DCEsegment*)nclistget(xgetvar->varaprojection->var->segments,0); if(caching || unconstrainable) { odom = newdapodometer(segment->slices,0,segment->rank); } else { /*Since vara was projected out, build a simple odometer*/ odom = newsimpledapodometer(segment,segment->rank); } while(dapodometermore(odom)) { /* Compute which instance to move to*/ unsigned int dimoffset = dapodometercount(odom); dimcontent = oc_data_new(conn); ocstat = oc_data_ith(conn,fieldcontent,dimoffset,dimcontent); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} ncstat = getcontent4r(drno,xgetvar,tnode,dimcontent,memory); dapodometerincr(odom); } freedapodometer(odom); } else { ncstat=getcontent4r(drno,xgetvar,tnode,fieldcontent,memory); } /*ok:*/ fail: oc_data_free(conn,gridcontent); oc_data_free(conn,seqcontent); oc_data_free(conn,dimcontent); oc_data_free(conn,fieldcontent); oc_data_free(conn,rootcontent); ncbytesfree(memory);/* leaves contents untouched */ if(ocstat != OC_NOERR) ncstat = ocerrtoncerr(ocstat); return THROW(ncstat); } /* Recursive walker part of getcontent */ static NCerror getcontent4r(NCDAP4* drno, Getvara* xgetvar, CDFnode* tnode, /* type definition */ OCdata currentcontent, NCbytes* memory) { unsigned int i; OCerror ocstat = OC_NOERR; NCerror ncstat = NC_NOERR; size_t rank; OCmode mode; OCconnection conn = drno->dap.oc.conn; OCdata reccontent = OCNULL; OCdata fieldcontent = OCNULL; OCdata dimcontent = OCNULL; Dapodometer* odom = NULL; NCbytes* vlenmemory = NULL; nc_vlen_t vlenref = {0,NULL}; int caching = FLAGSET(drno->dap.controls,NCF_CACHE); int unconstrainable = FLAGSET(drno->dap.controls,NCF_UNCONSTRAINABLE); rank = nclistlength(tnode->array.dimensions); oc_data_mode(conn,currentcontent,&mode); #ifdef READCHECK fprintf(stderr,"getcontent4r: rank=%lu mode=%d nctype=%s\n", (unsigned long)rank,mode,nctypetostring(tnode->nctype)); #endif if(tnode->nctype == NC_Primitive) { DCEsegment seg; /* temporary */ seg.name = tnode->name; seg.cdfnode = tnode; makewholesegment3(&seg,tnode); ncstat = getcontent4prim(drno,xgetvar,tnode,&seg,currentcontent,memory); goto done; } if(tnode->virtual) { fprintf(stderr,"VIRTUAL ENCOUNTER\n"); abort(); } switch (mode) { case OCARRAYMODE: { unsigned int rank = nclistlength(tnode->array.dimensions); DCEsegment seg; /* temporary */ ASSERT((tnode->nctype == NC_Structure)); seg.name = tnode->name; seg.cdfnode = tnode; makewholesegment3(&seg,tnode); /* The goal here is to walk the indices (if any) and extract each instance */ ASSERT((rank > 0)); if(caching || unconstrainable) { odom = newdapodometer(seg.slices,0,rank); } else { /*Since vara was projected out, build a simple odometer*/ odom = newsimpledapodometer(&seg,rank); } dimcontent = oc_data_new(conn); while(dapodometermore(odom)) { /* Compute which instance to move to*/ unsigned int dimoffset = dapodometercount(odom); ocstat = oc_data_ith(conn,currentcontent,dimoffset,dimcontent); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} ncstat = getcontent4r(drno,xgetvar,tnode,dimcontent,memory); } } break; case OCFIELDMODE: /* Walk each field in turn */ /* Align the buffer to struct boundary*/ alignbuffer3(memory,tnode->typesize.instance.alignment); fieldcontent = oc_data_new(conn); for(i=0;isubnodes);i++) { CDFnode* subnode = (CDFnode*)nclistget(tnode->subnodes,i); ocstat = oc_data_ith(conn,currentcontent,i,fieldcontent); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} ncstat = getcontent4r(drno,xgetvar,subnode,fieldcontent,memory); } break; case OCRECORDMODE: /* Collect the set of records as a separate memory structure */ vlenmemory = ncbytesnew(); ncbytessetalloc(vlenmemory,4096); #ifdef READCHECK fprintf(stderr,"getcontent4r: record: vlenmemory=%lx\n", (unsigned long)ncbytescontents(vlenmemory)); #endif reccontent = oc_data_new(conn); for(i=0;;i++) { ocstat = oc_data_ith(conn,currentcontent,i,reccontent); if(ocstat == OC_EINVALCOORDS) {ocstat=OC_NOERR; break;} /* no more records */ if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} ncstat = getcontent4r(drno,xgetvar,tnode,reccontent,vlenmemory); } vlenref.len = i; vlenref.p = ncbytesextract(vlenmemory); #ifdef READCHECK fprintf(stderr,"getcontent4r: record: vlen.p=%lx\n", (unsigned long)vlenref.p); #endif /* Now put an nc_vlen_t into our buffer */ alignbuffer3(memory,tnode->typesize.field.alignment); ncbytesappendn(memory,(char*)&vlenref,sizeof(vlenref)); #ifdef READCHECK fprintf(stderr,"getcontent4r: record: memory=%lx |memory|=%lu\n", (unsigned long)ncbytescontents(memory),(unsigned long)ncbyteslength(memory)); #endif break; default: ncstat = NC_EINVAL; {THROWCHK(ncstat); goto fail;} } done: fail: ncbytesfree(vlenmemory); oc_data_free(conn,dimcontent); oc_data_free(conn,fieldcontent); oc_data_free(conn,reccontent); if(ocstat != OC_NOERR) ncstat = ocerrtoncerr(ocstat); return THROW(ncstat); } static NCerror getcontent4prim(NCDAP4* drno, Getvara* xgetvar, CDFnode* tnode, DCEsegment* segment, OCdata currentcontent, NCbytes* memory) { OCerror ocstat = OC_NOERR; NCerror ncstat = NC_NOERR; unsigned int rank; OCconnection conn = drno->dap.oc.conn; Dapodometer* odom = NULL; unsigned int memoffset; DCEslice* slices = segment->slices; int caching = FLAGSET(drno->dap.controls,NCF_CACHE); int unconstrainable = FLAGSET(drno->dap.controls,NCF_UNCONSTRAINABLE); size_t internaltypesize, externaltypesize; nc_type internaltype = segment->cdfnode->etype; nc_type externaltype = xgetvar->dsttype; int bit8type; /* is this an 8 bit type? */ if(externaltype == NC_NAT) externaltype = internaltype; internaltypesize = nctypesizeof(internaltype); externaltypesize = nctypesizeof(externaltype); switch (internaltype) { case NC_CHAR: case NC_BYTE: case NC_UBYTE: bit8type = 1; break; default: bit8type = 0; break; } rank = nclistlength(tnode->array.dimensions); ASSERT((rank == segment->rank)); #ifdef READCHECK fprintf(stderr,"getcontent4prim: tnode=%s/%d segment=%s internaltype=%s externaltype=%s memory=%lx internaltypesize=%lu externaltypesize=%lu\n", tnode->name,rank,dumpsegment(segment), nctypetostring(internaltype),nctypetostring(externaltype), (unsigned long)ncbytescontents(memory), (unsigned long)internaltypesize, (unsigned long)externaltypesize); #endif alignbuffer3(memory,ncctypealignment(externaltype)); /* Align to the proper boundary */ memoffset = ncbyteslength(memory); #ifdef READCHECK fprintf(stderr,"getcontent4prim: alignment=%d memory.after=%lx\n", ncctypealignment(externaltype),(unsigned long)ncbytescontents(memory)); #endif if(rank == 0) { /* singleton data item*/ char value[32]; /* big enough to hold any single primitive value*/ void* memdata; ASSERT((internaltypesize <= sizeof(value))); if(!ncbytessetlength(memory,memoffset+externaltypesize)) return NC_ENOMEM; #ifdef READCHECK fprintf(stderr,"getcontent4prim: scalar: memory=%lx |memory|=%lu\n", (unsigned long)ncbytescontents(memory),(unsigned long)ncbyteslength(memory)); #endif memdata = ncbytescontents(memory)+memoffset; #ifdef READCHECK fprintf(stderr,"getcontent4prim: scalar: memdata=%lx\n",(unsigned long)memdata); #endif ocstat = oc_data_get(conn,currentcontent,value,sizeof(value),0,1); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} ncstat = ncdap4convert(internaltype,externaltype,memdata,value,1); if(ncstat != NC_NOERR) {THROWCHK(ncstat); goto fail;} } else { /* rank > 0 */ size_t requested, arraysize; char* memdata; int contiguous; if(caching || unconstrainable) { odom = newdapodometer(slices,0,rank); } else { odom = newsimpledapodometer(segment,rank); } /* Compute the actual # of values requested by the caller */ requested = dapodometerpoints(odom); if(bit8type && externaltype == NC_STRING) { /* special case */ arraysize = nctypesizeof(NC_STRING); /* allocate a single string */ } else arraysize = externaltypesize*requested; if(!ncbytessetlength(memory,memoffset+arraysize)) return NC_ENOMEM; memdata = ncbytescontents(memory)+memoffset; #ifdef TEST { size_t dimcount; oc_data_count(conn,currentcontent,&dimcount); printf("read: %s: memoffset=%lu externaltypesize=%lu dimcount=%lu requested=%lu arraysize=%lu\n", tnode->name,(unsigned long)memoffset,(unsigned long)externaltypesize, (unsigned long)dimcount,(unsigned long)requested,arraysize); fflush(stdout); } #endif /* Optimize off the use of the odometer by checking the slicing to locate the largest possible suffix of dimensions that represent a contiguous chunk; However do not do this if external type conversion is needed. */ contiguous = contiguousdims(odom); if(contiguous < odom->rank && internaltype == externaltype) { /* Compute the chunk size */ size_t chunkcount; ASSERT((odom->rank != 1 || contiguous == 0)); chunkcount = dapodometerspace(odom,contiguous); if(contiguous == 0) { /* The whole vara slice is usable: read the whole chunk at one shot. */ /* copy the data locally before conversion */ ocstat = oc_data_get(conn,currentcontent,memdata, arraysize,0,chunkcount); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} } else { /* odom->rank > 1 && contiguous > 0 */ size_t memchunk = (externaltypesize * chunkcount); /* Use the odometer to find each contiguous chunk. */ odom->rank = contiguous; /* move in chunksize chunks */ while(dapodometermore(odom)) { size_t nchunks = dapodometercount(odom); size_t chunkoffset = nchunks * chunkcount; ocstat = oc_data_get(conn,currentcontent,memdata, arraysize,chunkoffset,chunkcount); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} memdata += memchunk; arraysize -= memchunk; dapodometerincr(odom); } } } else if(bit8type && externaltype == NC_STRING) { /* special case conversion */ char* s = (char*)malloc(requested+1); /* bytes will go into a single string */ char** sp = (char**)memdata; ocstat = oc_data_get(conn,currentcontent,s,requested,0,requested); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} *sp = s; memdata += externaltypesize; } else { /*Oh well,use the odometer to walk to the appropriate fields*/ char value[32]; /* to hold any value*/ while(dapodometermore(odom)) { size_t dimoffset = dapodometercount(odom); ocstat = oc_data_get(conn,currentcontent,value,sizeof(value),dimoffset,1); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} ocstat = ncdap4convert(internaltype,externaltype,memdata,value,1); if(ocstat != OC_NOERR) {THROWCHK(ocstat); goto fail;} memdata += externaltypesize; dapodometerincr(odom); } } } /*ok:*/ fail: freedapodometer(odom); if(ocstat != OC_NOERR) ncstat = ocerrtoncerr(ocstat); return THROW(ncstat); } static int findfield(CDFnode* node, CDFnode* subnode) { size_t i; for(i=0;isubnodes);i++) { CDFnode* test = (CDFnode*)nclistget(node->subnodes,i); if(test == subnode) return i; } return -1; } /* Find the largest prefix of n dimensions such that: n .. odom.rank are complete: they have start of 0, stride of 1 and count == dimensions size. */ static int contiguousdims(Dapodometer* odom) { unsigned int i; int mark = 0; if(odom->rank <= 1) return 0; for(i=0;irank;i++) { ASSERT((odom->slices[i].declsize != 0)); if(odom->slices[i].stride != 1 || odom->slices[i].first > 0 || odom->slices[i].length != odom->slices[i].declsize) mark=i; } return mark+1; } static NCerror ncdap4convert(nc_type srctype, nc_type dsttype, char* memory, char* value, size_t count) { NCerror ncstat = NC_NOERR; size_t i; #define CASE(nc1,nc2) (nc1*256+nc2) switch (CASE(srctype,dsttype)) { /* the NC_STRING/URL case is only used by ncdap4 */ case CASE(NC_STRING,NC_STRING): case CASE(NC_STRING,NC_URL): case CASE(NC_URL,NC_STRING): case CASE(NC_URL,NC_URL): for(i=0;itarget; NClist* path = NULL; #ifdef DEBUG fprintf(stderr,"vara: %s\n",getvaraprint(getvar)); #endif if(!getvar->projected) return NC_NOERR; path = nclistnew(); collectnodepath3(target,path,WITHOUTDATASET); if(nclistlength(path) == 0) return NC_NOERR; ncbytesclear(buf); for(dimdex=0,i=0;i0) ncbytescat(buf,"."); /* we are past the first projection*/ ncbytescat(buf,node->name); #ifdef IGNORE if(node->nctype == NC_Sequence) { ASSERT((node->usesequence == 1)); dimdex++; } else if(nclistlength(node->array.dimensions) > 0) { makevarprojection3(getvar->slices,dimdex,nclistlength(node->array.dimensions),buf); dimdex += nclistlength(node->array.dimensions); } #endif } #ifdef DEBUG fprintf(stderr,"projection=|%s|",ncbytescontents(buf)); fflush(stderr); #endif nclistfree(path); return NC_NOERR; } static void makevarprojection3(NCslice* slices, int dimdex, int ndims, NCbytes* buf) { int i; char tmp[1024]; NCslice* slice = slices+dimdex; for(i=0;ifirst, (unsigned long)slice->stride, (unsigned long)(slice->stop - 1)); ncbytescat(buf,tmp); } } #endif