mirror of
https://github.com/Unidata/netcdf-c.git
synced 2024-11-21 03:13:42 +08:00
479 lines
13 KiB
C
479 lines
13 KiB
C
/*********************************************************************
|
|
* Copyright 2010, UCAR/Unidata
|
|
* See netcdf/COPYRIGHT file for copying and redistribution conditions.
|
|
* $Id$
|
|
* $Header$
|
|
*********************************************************************/
|
|
|
|
#include "includes.h"
|
|
#include "nccrgetvarx.h"
|
|
#include "nccrcvt.h"
|
|
|
|
|
|
#define DATAPREFIX "?req=data&var="
|
|
|
|
|
|
/* Forward */
|
|
static int makegetvar(NCCDMR*, CRnode*, void*, nc_type, NCCRgetvarx**);
|
|
|
|
static int crfetchdata(NCCDMR* cdmr, char* p, bytes_t* bufp, long* filetimep);
|
|
|
|
static int nccr_getcontent(NCCDMR* cdmr, NCCRgetvarx* varxinfo, bytes_t, size_t, void* data);
|
|
|
|
static void freegetvarx(NCCRgetvarx* varx);
|
|
|
|
static CRnode* locatevar(NCCDMR* cdmr, Data* data);
|
|
|
|
static char* urlconstraintstring(CCEprojection*);
|
|
|
|
|
|
/**************************************************/
|
|
int
|
|
NCCR_getvarx(int ncid, int varid,
|
|
const size_t* startp,
|
|
const size_t* countp,
|
|
const ptrdiff_t* stridep,
|
|
void* data,
|
|
nc_type externaltype0)
|
|
{
|
|
int ncstat = NC_NOERR;
|
|
unsigned int i;
|
|
NCCRgetvarx* varxinfo = NULL;
|
|
CCEprojection* varxprojection = NULL;
|
|
NC* drno;
|
|
NCCDMR* cdmr;
|
|
size_t localcount[NC_MAX_VAR_DIMS];
|
|
NClist* vars = NULL;
|
|
DataType datatype;
|
|
nc_type internaltype;
|
|
nc_type externaltype = externaltype0;
|
|
CRshape shape;
|
|
bytes_t buf = bytes_t_null;
|
|
char* projectionstring = NULL;
|
|
size_t offset;
|
|
CRnode* streamvar; /* stream node mapping to var*/
|
|
|
|
LOG((2, "nccr_get_varx: ncid 0x%x varid %d", ncid, varid));
|
|
|
|
ncstat = NC_check_id(ncid, (NC**)&drno);
|
|
if(ncstat != NC_NOERR) goto done;
|
|
cdmr = (NCCDMR*)drno->dispatchdata;
|
|
|
|
/* Find the CRnode instance for this ncid */
|
|
for(i=0;i<nclistlength(cdmr->variables);i++) {
|
|
streamvar = (CRnode*)nclistget(cdmr->variables,i);
|
|
if(streamvar->ncid == varid) break;
|
|
}
|
|
if(streamvar == NULL) {ncstat = NC_ENOTVAR; goto done;}
|
|
|
|
ASSERT((streamvar->sort == _Variable || streamvar->sort == _Structure));
|
|
|
|
/* Get the dimension info and typing */
|
|
if(streamvar->sort == _Variable) {
|
|
Variable* var = (Variable*)streamvar;
|
|
crextractshape((CRnode*)var,&shape);
|
|
datatype = var->dataType;
|
|
} else if(streamvar->sort == _Structure) {
|
|
Structure* var = (Structure*)streamvar;
|
|
crextractshape((CRnode*)var,&shape);
|
|
datatype = var->dataType;
|
|
}
|
|
|
|
/* Fill in missing arguments */
|
|
if(startp == NULL)
|
|
startp = nc_sizevector0;
|
|
|
|
if(countp == NULL) {
|
|
/* Accumulate the dimension sizes */
|
|
for(i=0;i<shape.rank;i++) {
|
|
Dimension* dim = shape.dims[i];
|
|
localcount[i] = dimsize(dim);
|
|
}
|
|
countp = localcount;
|
|
}
|
|
|
|
if(stridep == NULL)
|
|
stridep = nc_ptrdiffvector1;
|
|
|
|
/* Validate the dimension sizes */
|
|
for(i=0;i<shape.rank;i++) {
|
|
Dimension* dim = shape.dims[i];
|
|
if(startp[i] > dimsize(dim)
|
|
|| startp[i]+countp[i] > dimsize(dim)) {
|
|
ncstat = NC_EINVALCOORDS;
|
|
goto done;
|
|
}
|
|
}
|
|
|
|
/* Default to using the var type */
|
|
internaltype = cvtstreamtonc(datatype);
|
|
if(externaltype == NC_NAT) externaltype = internaltype;
|
|
|
|
/* Validate any implied type conversion*/
|
|
if(internaltype != externaltype && externaltype == NC_CHAR) {
|
|
/* The only disallowed conversion is to/from char and non-byte
|
|
numeric types*/
|
|
switch (internaltype) {
|
|
case NC_STRING:
|
|
case NC_CHAR: case NC_BYTE: case NC_UBYTE:
|
|
break;
|
|
default:
|
|
THROWCHK(NC_ECHAR);
|
|
goto done;
|
|
}
|
|
}
|
|
|
|
ncstat = makegetvar(cdmr,streamvar,data,externaltype,&varxinfo);
|
|
if(ncstat) {THROWCHK(NC_ENOMEM); goto done;}
|
|
|
|
/* Load with constraints */
|
|
vars = nclistnew();
|
|
nclistpush(vars,(ncelem)varxinfo->target);
|
|
|
|
/* Find the relevant projection from the predefined projections */
|
|
for(i=0;i<nclistlength(cdmr->urlconstraint->projections);i++) {
|
|
CCEprojection* p = (CCEprojection*)nclistget(cdmr->urlconstraint->projections,i);
|
|
if(p->decl == streamvar) {varxinfo->projection = (CCEprojection*)cceclone((CCEnode*)p); break;}
|
|
}
|
|
ASSERT(varxinfo->projection != NULL); /* By construction */
|
|
|
|
/* merge the getvarx start/stride/stop into the existing projection (as cloned) */
|
|
ncstat = ccerestrictprojection(varxinfo->projection,shape.rank,startp,countp,stridep);
|
|
if(ncstat != NC_NOERR) {THROWCHK(ncstat); goto done;}
|
|
|
|
#ifdef DEBUG
|
|
fprintf(stderr,"varx merge: %s\n",
|
|
ccetostring((CCEnode*)varxinfo->projection));
|
|
#endif
|
|
|
|
/* Convert the projections into a string for use
|
|
with crfetchdata*/
|
|
|
|
projectionstring = urlconstraintstring(varxinfo->projection);
|
|
#ifdef DEBUG
|
|
fprintf(stderr,"projectionstring: %s\n",projectionstring);
|
|
#endif
|
|
|
|
/* Fetch */
|
|
|
|
buf = bytes_t_null;
|
|
ncstat = crfetchdata(cdmr,projectionstring,&buf,NULL);
|
|
if(ncstat != NC_NOERR) {THROWCHK(ncstat); goto done;}
|
|
|
|
if(projectionstring) free(projectionstring); /* no longer needed */
|
|
|
|
/* Parse the data header */
|
|
ncstat = nccr_decodedatamessage(&buf,&cdmr->datahdr,&offset);
|
|
if(ncstat != NC_NOERR) {THROWCHK(ncstat); goto done;}
|
|
|
|
/* Verify the variable in the data header */
|
|
ASSERT((locatevar(cdmr,cdmr->datahdr) == streamvar));
|
|
|
|
/* move the data into user's memory; watch out buf is structure copy */
|
|
ncstat = nccr_getcontent(cdmr,varxinfo,buf,offset,data);
|
|
if(ncstat != NC_NOERR) {THROWCHK(ncstat); goto done;}
|
|
|
|
done:
|
|
if(buf.bytes != NULL) free(buf.bytes);
|
|
ccefree((CCEnode*)varxprojection);
|
|
freegetvarx(varxinfo);
|
|
return THROW(ncstat);
|
|
}
|
|
|
|
static char*
|
|
urlconstraintstring(CCEprojection* projection)
|
|
{
|
|
char* projectionstring = ccetostring((CCEnode*)projection);
|
|
return projectionstring;
|
|
}
|
|
|
|
static void
|
|
freegetvarx(NCCRgetvarx* varx)
|
|
{
|
|
if(varx == NULL) return;
|
|
ccefree((CCEnode*)varx->projection);
|
|
free(varx);
|
|
}
|
|
|
|
static int
|
|
makegetvar(NCCDMR* cdmr, CRnode* var, void* data, nc_type externaltype, NCCRgetvarx** varxp)
|
|
{
|
|
NCCRgetvarx* varx;
|
|
int ncstat = NC_NOERR;
|
|
|
|
varx = (NCCRgetvarx*)calloc(1,sizeof(NCCRgetvarx));
|
|
MEMCHECK(varx,NC_ENOMEM);
|
|
if(varxp) *varxp = varx;
|
|
|
|
varx->target = var;
|
|
varx->externaltype = externaltype;
|
|
varx->internaltype = cvtstreamtonc(nccr_gettype(var));
|
|
varx->projection = NULL;
|
|
return ncstat;
|
|
}
|
|
|
|
#ifdef IGNORE
|
|
/* In order to construct the projection,
|
|
we need to make sure to match the relevant dimensions
|
|
against the relevant nodes in which the ultimate target
|
|
is contained.
|
|
*/
|
|
static int
|
|
buildvarxprojection(NCCRgetvarx* varxinfo,
|
|
const size_t* startp,
|
|
const size_t* countp,
|
|
const ptrdiff_t* stridep,
|
|
CCEprojection** projectionp)
|
|
{
|
|
int i;
|
|
int ncstat = NC_NOERR;
|
|
CRnode* var = varxinfo->target;
|
|
CCEprojection* projection = NULL;
|
|
NClist* segments = NULL;
|
|
CCEsegment* segment;
|
|
CRshape shape;
|
|
|
|
segment = (CCEsegment*)ccecreate(CES_SEGMENT);
|
|
segment->decl = var;
|
|
ASSERT((segment->decl != NULL));
|
|
segment->name = nulldup(nccr_getname(segment->decl));
|
|
segment->slicesdefined = 0; /* temporary */
|
|
segment->slicesdeclized = 0; /* temporary */
|
|
segments = nclistnew();
|
|
nclistpush(segments,(ncelem)segment);
|
|
|
|
projection = (CCEprojection*)ccecreate(CES_PROJECT);
|
|
projection->decl = var;
|
|
projection->segments = segments;
|
|
|
|
crextractshape((CRnode*)var,&shape);
|
|
segment->rank = shape.rank;
|
|
for(i=0;i<segment->rank;i++) {
|
|
CCEslice* slice = &segment->slices[i];
|
|
Dimension* dim = shape.dims[i];
|
|
slice->first = startp[i];
|
|
slice->stride = stridep[i];
|
|
slice->count = countp[i];
|
|
slice->length = slice->count * slice->stride;
|
|
slice->stop = (slice->first + slice->length);
|
|
ASSERT(dimsize(dim) > 0);
|
|
slice->declsize = dimsize(dim);
|
|
}
|
|
segment->slicesdefined = 1;
|
|
segment->slicesdeclized = 1;
|
|
|
|
if(projectionp) *projectionp = projection;
|
|
if(ncstat) ccefree((CCEnode*)projection);
|
|
return ncstat;
|
|
}
|
|
#endif
|
|
|
|
static int
|
|
crfetchdata(NCCDMR* cdmr, char* projection, bytes_t* bufp, long* filetimep)
|
|
{
|
|
int ncstat = NC_NOERR;
|
|
char* curlurl;
|
|
long filetime;
|
|
bytes_t buf;
|
|
char* fullprojection;
|
|
size_t len;
|
|
|
|
/* Construct the complete projection suffix */
|
|
len = strlen(projection);
|
|
len += strlen(DATAPREFIX);
|
|
len++; /* null terminator */
|
|
fullprojection = (char*) malloc(len);
|
|
if(fullprojection == NULL) {ncstat=NC_ENOMEM; goto done;}
|
|
strcpy(fullprojection,DATAPREFIX);
|
|
strcat(fullprojection,projection);
|
|
|
|
/* fetch data */
|
|
buf = bytes_t_null;
|
|
curlurl = nc_uribuild(cdmr->uri,NULL,fullprojection,0);
|
|
free(fullprojection);
|
|
if(curlurl == NULL) {ncstat=NC_ENOMEM; goto done;}
|
|
ncstat = nccr_fetchurl(cdmr,cdmr->curl.curl,curlurl,&buf,&filetime);
|
|
free(curlurl);
|
|
if(ncstat != NC_NOERR) {THROWCHK(ncstat); goto done;}
|
|
|
|
if(filetimep) *filetimep = filetime;
|
|
if(bufp) *bufp = buf;
|
|
|
|
done:
|
|
return ncstat;
|
|
}
|
|
|
|
/*
|
|
Find the node in the Header* tree matching the var in Data tree.
|
|
Note that this needs to eventually change because storing the
|
|
variables's full path name introduces the possibility of ambiguity
|
|
if the name uses non-standard characters. The solution is
|
|
to replace the varName in struct Data with a vector of names
|
|
(similar to CRpath).
|
|
*/
|
|
|
|
static CRnode*
|
|
locatevar(NCCDMR* cdmr, Data* data)
|
|
{
|
|
char segment[1024];
|
|
char* p;
|
|
CRnode* match = NULL;
|
|
int i;
|
|
CRpath* path = NULL;
|
|
|
|
/* Split the name at the dots and store the pieces */
|
|
p = data->varName;
|
|
for(;;) {
|
|
ptrdiff_t len;
|
|
char* dot = strchr(p,'.');
|
|
if(dot == NULL) break;
|
|
len = (dot-p);
|
|
if(len >= sizeof(segment)) len = sizeof(segment-1);
|
|
strncpy(segment,dot,len);
|
|
segment[len] = '\0';
|
|
path = crpathappend(path,segment);
|
|
p = dot+1;
|
|
}
|
|
/* do the last segment */
|
|
if(*p) {
|
|
path = crpathappend(path,p);
|
|
}
|
|
|
|
/* Now locate the matching variable */
|
|
for(i=0;i<nclistlength(cdmr->variables);i++) {
|
|
CRnode* test = (CRnode*)nclistget(cdmr->variables,i);
|
|
if(crpathmatch(test->pathname,path) == 1) {
|
|
if(match != NULL) {
|
|
nclog(NCLOGERR,"Ambiguous name in Data object");
|
|
} else match = test;
|
|
}
|
|
}
|
|
if(match == NULL) {
|
|
nclog(NCLOGERR,"Cannot locate variable in Data object");
|
|
}
|
|
return match;
|
|
}
|
|
|
|
|
|
/*
|
|
Assumptions:
|
|
1. there is only 1 projection variable in response
|
|
|
|
Not-Assumed:
|
|
*/
|
|
|
|
|
|
static int
|
|
nccr_getcontent(NCCDMR* cdmr, NCCRgetvarx* varxinfo, bytes_t data, size_t offset, void* memory)
|
|
{
|
|
int ncstat = NC_NOERR;
|
|
int i;
|
|
size_t count;
|
|
int localbig = ((cdmr->controls & BIGENDIAN)?1:0);
|
|
int databig = ((cdmr->datahdr->bigend.defined
|
|
&& cdmr->datahdr->bigend.value)?1:0);
|
|
int swap = (localbig == databig?0:1);
|
|
int internaltypesize = nctypelen(varxinfo->internaltype);
|
|
|
|
/* modify data */
|
|
data.nbytes -= offset;
|
|
data.bytes += offset;
|
|
|
|
/* First, check to see if any type conversion will be necessary */
|
|
if(varxinfo->internaltype == varxinfo->externaltype) {
|
|
/* We can short circuit the transfer and not do any type conversion */
|
|
/* Separate out the integer-like types from the string types */
|
|
|
|
switch(varxinfo->internaltype) {
|
|
|
|
default: { /* numeric types */
|
|
/* Compute the amount to move */
|
|
size_t vlen = 0;
|
|
ncstat = nccr_decodedatacount(&data,&vlen,&count);
|
|
if(ncstat) {goto done;}
|
|
count = count/internaltypesize;
|
|
/* swap and convert */
|
|
ncstat = nccrconvert(varxinfo->internaltype,varxinfo->externaltype,
|
|
data.bytes+vlen,memory,
|
|
count,swap);
|
|
if(ncstat) {goto done;}
|
|
} break;
|
|
|
|
case ENUM1:
|
|
case ENUM2:
|
|
case ENUM4:
|
|
break;
|
|
|
|
/* Variable length types */
|
|
case OPAQUE:
|
|
case STRING: {
|
|
/* Get count of the number of objects */
|
|
size_t nobjects;
|
|
size_t len;
|
|
char** pp = (char**)memory;
|
|
size_t localoffset = offset;
|
|
ncstat = nccr_decodedatacount(&data,&localoffset,&nobjects);
|
|
if(ncstat) {goto done;}
|
|
for(i=0;i<nobjects;i++) {
|
|
char* p;
|
|
/* Get the length (|length|) */
|
|
ncstat = nccr_decodedatacount(&data,&localoffset,&len);
|
|
if(ncstat) {goto done;}
|
|
p = (char*)malloc(len+1);
|
|
memcpy(p,data.bytes+localoffset,len);
|
|
p[len] = '\0';
|
|
*pp++ = p;
|
|
}
|
|
} break;
|
|
|
|
/* We should never see these */
|
|
case SEQUENCE:
|
|
case STRUCTURE:
|
|
PANIC("unexpected internal type");
|
|
}
|
|
} else {/* Guess we have to do the conversions */
|
|
size_t vlen;
|
|
ncstat = nccr_decodedatacount(&data,&vlen,&count);
|
|
if(ncstat) {goto done;}
|
|
|
|
switch(varxinfo->internaltype) {
|
|
|
|
default: {
|
|
ncstat = nccrconvert(varxinfo->internaltype,varxinfo->externaltype,
|
|
data.bytes+vlen,memory,
|
|
count,swap);
|
|
if(ncstat) {goto done;}
|
|
} break;
|
|
|
|
/* Variable length types */
|
|
case OPAQUE:
|
|
case STRING: {
|
|
/* Get count of the number of objects */
|
|
size_t nobjects;
|
|
size_t len, vlen;
|
|
char** pp = (char**)memory;
|
|
ncstat = nccr_decodedatacount(&data,&vlen,&nobjects);
|
|
if(ncstat) {goto done;}
|
|
for(i=0;i<nobjects;i++) {
|
|
char* p;
|
|
/* Get the length (|length|) */
|
|
ncstat = nccr_decodedatacount(&data,&vlen,&len);
|
|
if(ncstat) {goto done;}
|
|
p = (char*)malloc(len+1);
|
|
memcpy(p,data.bytes+vlen,len);
|
|
p[len] = '\0';
|
|
*pp++ = p;
|
|
}
|
|
} break;
|
|
|
|
/* We should never see these */
|
|
case SEQUENCE:
|
|
case STRUCTURE:
|
|
PANIC("unexpected internal type");
|
|
}
|
|
}
|
|
|
|
done:
|
|
return ncstat;
|
|
}
|