Copyright (C) Kevin Larke 2009-2020
This file is part of libcm.
libcm is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.
libcm is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
See the GNU General Public License distributed with the libcm
package or look here:
// // TODO: // 0) Figure out an error handling scheme that does not rely on // a global errno. This is not useful in multi-thread environments. // It might be ok to go with an 'all errors are fatal' model // (except in the var-args functions). // Consider the use of a context object for use with functions // that can have runtime errors or need to allocate memory. // // 1) Implement the canConvert and willTruncate functions. // // 2) Make a set of cmDataAllocXXXPtr() functions which take // a flag indicating whether or not to dynamically allocate // the array space. This will allow dynamic allocattion to // occur at runtime. Make var args functions for list and // record objects which also take this flag. // Whereever a function may be implemented using // static/dynamic allocation this flag should be present. // (e.g. string allocation for pair labels) // This choice is common enough that it may be worth // suffixing function names with a capital letter to // be clear what the functions memory policy is. // // 3) Come up with a var-args format which allows a // hierchy of records to be defined in one line. // // 4) Implement the serialization functions. // // 5) Implement an ascii string/parse format for writing/reading. // // 6) Implement fast lookup of record fields. // // 7) Allow for user defined types. For example a 'matrix' // data type. This might be as simple as adding an extra 'user_tid' // field to cmData_t. // // 8) Implement type specific cmDataGetRecordValueXXX() functions. // // 9) Implement cmDataIsEqual(), cmDataIsLtE(), ... // // // enum { kOkDtRC = cmOkRC, kAssertErrDtRC, kConstErrDtRC, kCvtErrDtRC, kInvalidContDtRC, kInvalidTypeDtRC, kMissingFieldDtRC, kLexFailDtRC, kParseStackFailDtRC, kSyntaxErrDtRC, kEolDtRC }; typedef unsigned cmDtRC_t; typedef enum { kInvalidTypeDtId, // 0 kNullDtId, // 1 the data object exists but it has no data kBoolDtId, // 2 kUCharDtId, // 3 kCharDtId, // 4 kUShortDtId, // 5 kShortDtId, // 6 kUIntDtId, // 7 kIntDtId, // 8 kULongDtId, // 9 kLongDtId, // 10 kFloatDtId, // 11 kDoubleDtId, // 12 kStrDtId, // 13 zero terminated string kBlobDtId, // 14 application defined raw memory object kStructDtId, // 15 node is a pair,list, or recd } cmDataTypeId_t; typedef enum { kInvalidCntDtId = kInvalidTypeDtId, kScalarDtId = 0x00100000, kArrayDtId = 0x00200000, kPairDtId = 0x00400000, kListDtId = 0x00800000, kRecordDtId = 0x01000000, kContainerDtMask = 0x01f00000, } cmDataContainerId_t; enum { kNoFlagsDtFl = 0x00, kOptArgDtFl = 0x02000000, // Indicate that the memory used by the data object // was dynamically allocated and should be released // by cmDataFree(). kFreeObjDtFl = 0x04000000, // Indicate that the memory used by strings, blobs // and arrays should be freed by cmDataFree(). kFreeValueDtFl = 0x08000000, // Indicate that the value of the object cannot be changed. // (but the object could be reassigned as a new type). kConstValueDtFl = 0x10000000, // Indicate that the type of the object cannot be changed. // (but the value may be changed). kConstObjDtFl = 0x20000000, // Indicate that the array or string should not be // internally reallocated but rather the source pointer // should be taken as the new value of the object. kNoCopyDtFl = 0x40000000, kFlagsDtMask = 0x7e000000 }; // The kInvalidDtXXX constants are used to indicate an error when returned // from the cmDtXXX() functions below. #define kInvalidDtFloat FLT_MAX #define kInvalidDtDouble DBL_MAX enum { kInvalidDtChar = 0xff, kInvalidDtShort = 0xffff, kInvalidDtInt = 0xffffffff, kInvalidDtLong = kInvalidDtInt, kInvalidDtBool = kInvalidDtInt }; typedef struct cmData_str { cmDataTypeId_t tid; // data format id cmDataContainerId_t cid; // container id unsigned flags; struct cmData_str* parent; // this childs parent struct cmData_str* sibling; // this childs left sibling unsigned cnt; // byte cnt for strings/blobs and ele count for arrays union { bool b; char c; unsigned char uc; short s; unsigned short us; int i; unsigned int ui; long l; unsigned long ul; float f; double d; cmChar_t* z; void* vp; struct cmData_str* child; // first child (list,record,pair) } u; } cmData_t; extern cmData_t cmDataNull; const cmChar_t* cmDataTypeToLabel( cmDataTypeId_t tid ); cmDataTypeId_t cmDataLabelToType( const cmChar_t* typeLabelStr ); // Returns 1 for kStrDtId. // Returns cmInvalidCnt if tid is not recognized. unsigned dmDataByteWidth( cmDataTypeId_t tid ); const cmChar_t* cmDataContainerIdToLabel( cmDataContainerId_t tid ); cmDataContainerId_t cmDataLabelToContainerId( const cmChar_t* contLabelStr ); bool cmDataIsConstObj( const cmData_t* d ); void cmDataEnableConstObj( cmData_t* d, bool enaFl ); bool cmDataIsConstValue( const cmData_t* d ); void cmDataEnableConstValue( cmData_t* d, bool enaFl ); bool cmDataIsFreeValue( const cmData_t* d ); void cmDataEnableFreeValue( cmData_t* d, bool enaFl ); // Returns true if this is a scalar or array node. bool cmDataIsLeaf( const cmData_t* d); // Return true if this is NOT a scalar or array node. bool cmDataIsStruct( const cmData_t* d );
// Scalar related functions // // Dynamically allocate a scalar object and set it's value. // The 'flags' argument may include kConstValueDtFl and kConstObjDtFl. // The string and blob constructors may also use the // kNoCopyDtFl and the kFreeValueDtFl. // Generic: // 'byteCnt' is ignored for all types other than strings and blobs. cmDtRC_t cmDataNewScalar( cmData_t* parent, cmDataTypeId_t tid, unsigned flags, void* vp, unsigned byteCnt, cmData_t** ref ); // Type specific cmDtRC_t cmDataNewNull( cmData_t* parent, unsigned flags, cmData_t** ref ); cmDtRC_t cmDataNewBool( cmData_t* parent, unsigned flags, bool v, cmData_t** ref ); cmDtRC_t cmDataNewChar( cmData_t* parent, unsigned flags, char v, cmData_t** ref ); cmDtRC_t cmDataNewUChar( cmData_t* parent, unsigned flags, unsigned char v, cmData_t** ref ); cmDtRC_t cmDataNewShort( cmData_t* parent, unsigned flags, short v, cmData_t** ref ); cmDtRC_t cmDataNewUShort( cmData_t* parent, unsigned flags, unsigned short v, cmData_t** ref ); cmDtRC_t cmDataNewInt( cmData_t* parent, unsigned flags, int v, cmData_t** ref ); cmDtRC_t cmDataNewUInt( cmData_t* parent, unsigned flags, unsigned int v, cmData_t** ref ); cmDtRC_t cmDataNewLong( cmData_t* parent, unsigned flags, long v, cmData_t** ref ); cmDtRC_t cmDataNewULong( cmData_t* parent, unsigned flags, unsigned long v, cmData_t** ref ); cmDtRC_t cmDataNewFloat( cmData_t* parent, unsigned flags, float v, cmData_t** ref ); cmDtRC_t cmDataNewDouble( cmData_t* parent, unsigned flags, double v, cmData_t** ref ); cmDtRC_t cmDataNewStr( cmData_t* parent, unsigned flags, cmChar_t* str, cmData_t** ref ); cmDtRC_t cmDataNewConstStr( cmData_t* parent, unsigned flags, const cmChar_t* str, cmData_t** ref ); cmDtRC_t cmDataNewStrN( cmData_t* parent, unsigned flags, cmChar_t* str, unsigned charCnt, cmData_t** ref ); cmDtRC_t cmDataNewConstStrN(cmData_t* parent, unsigned flags, const cmChar_t* str, unsigned charCnt, cmData_t** ref ); cmDtRC_t cmDataNewBlob( cmData_t* parent, unsigned flags, void* vp, unsigned byteCnt, cmData_t** ref ); cmDtRC_t cmDataNewConstBlob(cmData_t* parent, unsigned flags, const void* vp, unsigned byteCnt, cmData_t** ref ); // Set the value and type of an existing scalar object. // These functions begin by releasing any resources held by *p // prior to resetting the type and value of the object. // The 'flags' argument to cmDataSetStr() and cmDataSetConstStr() // may use the kNoCopyDtFl and the kFreeValueDtFl cmDtRC_t cmDataSetScalarValue( cmData_t* d, cmDataTypeId_t tid, void* vp, unsigned byteCnt, unsigned flags ); cmDtRC_t cmDataSetNull( cmData_t* p ); cmDtRC_t cmDataSetBool( cmData_t* p, bool v ); cmDtRC_t cmDataSetChar( cmData_t* p, char v ); cmDtRC_t cmDataSetUChar( cmData_t* p, unsigned char v ); cmDtRC_t cmDataSetShort( cmData_t* p, short v ); cmDtRC_t cmDataSetUShort( cmData_t* p, unsigned short v ); cmDtRC_t cmDataSetInt( cmData_t* p, int v ); cmDtRC_t cmDataSetUInt( cmData_t* p, unsigned int v ); cmDtRC_t cmDataSetLong( cmData_t* p, long v ); cmDtRC_t cmDataSetULong( cmData_t* p, unsigned long v ); cmDtRC_t cmDataSetFloat( cmData_t* p, float v ); cmDtRC_t cmDataSetDouble( cmData_t* p, double v ); cmDtRC_t cmDataSetStr( cmData_t* p, unsigned flags, cmChar_t* s ); cmDtRC_t cmDataSetConstStr( cmData_t* p, unsigned flags, const cmChar_t* s ); cmDtRC_t cmDataSetStrN( cmData_t* p, unsigned flags, cmChar_t* s, unsigned charCnt ); cmDtRC_t cmDataSetConstStrN( cmData_t* p, unsigned flags, const cmChar_t* s, unsigned charCnt ); cmDtRC_t cmDataSetBlob( cmData_t* p, unsigned flags, void* v, unsigned byteCnt ); cmDtRC_t cmDataSetConstBlob( cmData_t* p, unsigned flags, const void* v, unsigned byteCnt ); // Get the value of an object. No conversion is applied the // type must match exactly or an error is generated. cmDtRC_t cmDataBool( const cmData_t* p, bool* v ); cmDtRC_t cmDataChar( const cmData_t* p, char* v ); cmDtRC_t cmDataUChar( const cmData_t* p, unsigned char* v ); cmDtRC_t cmDataShort( const cmData_t* p, short* v ); cmDtRC_t cmDataUShort( const cmData_t* p, unsigned short* v ); cmDtRC_t cmDataInt( const cmData_t* p, int* v ); cmDtRC_t cmDataUInt( const cmData_t* p, unsigned int* v ); cmDtRC_t cmDataLong( const cmData_t* p, long* v ); cmDtRC_t cmDataULong( const cmData_t* p, unsigned long* v ); cmDtRC_t cmDataFloat( const cmData_t* p, float* v ); cmDtRC_t cmDataDouble( const cmData_t* p, double* v ); cmDtRC_t cmDataStr( const cmData_t* p, cmChar_t** v ); cmDtRC_t cmDataConstStr( const cmData_t* p, const cmChar_t** v ); cmDtRC_t cmDataBlob( const cmData_t* p, void** v, unsigned* byteCntRef ); cmDtRC_t cmDataConstBlob( const cmData_t* p, const void** v, unsigned* byteCntRef ); // Functions in this group which return pointers will return NULL // on error. The other function indicate an error by returning // kInvalidDtXXX depending on their type. // Note that there is no guarantee, except as determined by the // application, that one of the kInvalidDtXXX is not in fact a legal return value. // These function are simple wrappers around calls to cmDataXXX() and // therefore do NOT do any type conversion. bool cmDtBool( const cmData_t* p ); char cmDtChar( const cmData_t* p ); unsigned char cmDtUChar( const cmData_t* p ); short cmDtShort( const cmData_t* p ); unsigned short cmDtUShort( const cmData_t* p ); int cmDtInt( const cmData_t* p ); unsigned cmDtUInt( const cmData_t* p ); long cmDtLong( const cmData_t* p ); unsigned long cmDtULong( const cmData_t* p ); float cmDtFloat( const cmData_t* p ); double cmDtDouble( const cmData_t* p ); char* cmDtStr( const cmData_t* p ); const char* cmDtConstStr( const cmData_t* p ); void* cmDtBlob( const cmData_t* p, unsigned* byteCntRef ); const void* cmDtConstBlob( const cmData_t* p, unsigned* byteCntRef ); // Get the value of an object with conversion. cmDtRC_t cmDataGetBool( const cmData_t* p, bool* v ); cmDtRC_t cmDataGetChar( const cmData_t* p, char* v ); cmDtRC_t cmDataGetUChar( const cmData_t* p, unsigned char* v ); cmDtRC_t cmDataGetShort( const cmData_t* p, short* v ); cmDtRC_t cmDataGetUShort( const cmData_t* p, unsigned short* v ); cmDtRC_t cmDataGetInt( const cmData_t* p, int* v ); cmDtRC_t cmDataGetUInt( const cmData_t* p, unsigned int* v ); cmDtRC_t cmDataGetLong( const cmData_t* p, long* v ); cmDtRC_t cmDataGetULong( const cmData_t* p, unsigned long* v ); cmDtRC_t cmDataGetFloat( const cmData_t* p, float* v ); cmDtRC_t cmDataGetDouble( const cmData_t* p, double* v ); // Functions in this group which return pointers will return NULL // on error. The other function indicate an error by returning // kInvalidDtXXX depending on their type. // Note that there is no guarantee, except as determined by the // application that one of the kInvalidDtXXX is not in fact a legal return value. // These function are simple wrappers around calls to cmDataGetXXX() and // therefore do type conversion. bool cmDtGetBool( const cmData_t* p ); char cmDtGetChar( const cmData_t* p ); unsigned char cmDtGetUChar( const cmData_t* p ); short cmDtGetShort( const cmData_t* p ); unsigned short cmDtGetUShort( const cmData_t* p ); int cmDtGetInt( const cmData_t* p ); unsigned cmDtGetUInt( const cmData_t* p ); long cmDtGetLong( const cmData_t* p ); unsigned long cmDtGetULong( const cmData_t* p ); float cmDtGetFloat( const cmData_t* p ); double cmDtGetDouble( const cmData_t* p );
// Array related functions // // Notes: // 1) string arrays are arrays of string pointers. // 2) blob arrays (array of void pointers) are not supported because // there is no direct way to determine the length of each blob // and therefore they cannot be internally duplicated - a special scheme // could be devised (length goes in first 4 bytes) to make this // work but we will defer that until the need arises. // // Dynamically allocate a new array data object. // // eleCnt referes to the number of elements in the array pointed // to by 'vp'. The number of bytes pointed to by 'vp' is then // cmDataByteWidth(tid)*eleCnt. // // If no flags are set then the array pointed to by 'vp' is reallocated // and kDataFreeDtFl is set. // // If kFreeValueDtFl is set then the object will take responsibility for // releasing the memory pointed to by 'vp' when the object is destroyed // or the array is reassigned. // // If kNoCopyDtFl is set then 'vp' becomes the internal array // value (vp[cnt]) is NOT reallocated). In this case the client is // responsibile for eventually releasing the associated memory - when // the data object is no longer valid. cmDtRC_t cmDataNewArray( cmData_t* parent, cmDataTypeId_t tid, void* vp, unsigned eleCnt, unsigned flags, cmData_t** ref ); cmDtRC_t cmDataNewBoolArray( cmData_t* parent, bool* v, unsigned eleCnt, unsigned flags, cmData_t** ref ); cmDtRC_t cmDataNewCharArray( cmData_t* parent, char* v, unsigned eleCnt, unsigned flags, cmData_t** ref ); cmDtRC_t cmDataNewUCharArray( cmData_t* parent, unsigned char* v, unsigned eleCnt, unsigned flags, cmData_t** ref ); cmDtRC_t cmDataNewShortArray( cmData_t* parent, short* v, unsigned eleCnt, unsigned flags, cmData_t** ref ); cmDtRC_t cmDataNewUShortArray( cmData_t* parent, unsigned short* v, unsigned eleCnt, unsigned flags, cmData_t** ref ); cmDtRC_t cmDataNewIntArray( cmData_t* parent, int* v, unsigned eleCnt, unsigned flags, cmData_t** ref ); cmDtRC_t cmDataNewUIntArray( cmData_t* parent, unsigned int* v, unsigned eleCnt, unsigned flags, cmData_t** ref ); cmDtRC_t cmDataNewLongArray( cmData_t* parent, long* v, unsigned eleCnt, unsigned flags, cmData_t** ref ); cmDtRC_t cmDataNewULongArray( cmData_t* parent, unsigned long* v, unsigned eleCnt, unsigned flags, cmData_t** ref ); cmDtRC_t cmDataNewFloatArray( cmData_t* parent, float* v, unsigned eleCnt, unsigned flags, cmData_t** ref ); cmDtRC_t cmDataNewDoubleArray( cmData_t* parent, double* v, unsigned eleCnt, unsigned flags, cmData_t** ref ); cmDtRC_t cmDataNewStrArray( cmData_t* parent, cmChar_t** v, unsigned eleCnt, unsigned flags, cmData_t** ref ); cmDtRC_t cmDataNewConstStrArray( cmData_t* parent, const cmChar_t** v,unsigned eleCnt, unsigned flags, cmData_t** ref ); // Set the value and type of an existing scalar object. // // These functions begin by releasing any resources held by *p // prior to resetting the type and value of the object. // The 'flags' argument may include kConstValueDtFl, kConstObjDtFl, // kNoCopyDtFl and the kFreeValueDtFl. // Generic set array functions. 'vp' is assumed to point to an array // of the type defined by 'tid'. cmDtRC_t cmDataSetArrayValue( cmData_t* dt, cmDataTypeId_t tid, void* vp, unsigned eleCnt, unsigned flags ); // Type sepctific set array functions. cmDtRC_t cmDataSetBoolArray( cmData_t* d, bool* v, unsigned eleCnt, unsigned flags ); cmDtRC_t cmDataSetCharArray( cmData_t* d, char* v, unsigned eleCnt, unsigned flags ); cmDtRC_t cmDataSetUCharArray( cmData_t* d, unsigned char* v, unsigned eleCnt, unsigned flags ); cmDtRC_t cmDataSetShortArray( cmData_t* d, short* v, unsigned eleCnt, unsigned flags ); cmDtRC_t cmDataSetUShortArray( cmData_t* d, unsigned short* v, unsigned eleCnt, unsigned flags ); cmDtRC_t cmDataSetIntArray( cmData_t* d, int* v, unsigned eleCnt, unsigned flags ); cmDtRC_t cmDataSetUIntArray( cmData_t* d, unsigned int* v, unsigned eleCnt, unsigned flags ); cmDtRC_t cmDataSetLongArray( cmData_t* d, long* v, unsigned eleCnt, unsigned flags ); cmDtRC_t cmDataSetULongArray( cmData_t* d, unsigned long* v, unsigned eleCnt, unsigned flags ); cmDtRC_t cmDataSetFloatArray( cmData_t* d, float* v, unsigned eleCnt, unsigned flags ); cmDtRC_t cmDataSetDoubleArray( cmData_t* d, double* v, unsigned eleCnt, unsigned flags ); cmDtRC_t cmDataSetStrArray( cmData_t* d, cmChar_t** v, unsigned eleCnt, unsigned flags ); cmDtRC_t cmDataSetConstStrArray(cmData_t* d,const cmChar_t** v,unsigned eleCnt, unsigned flags ); // Return the count of elements in a n array. unsigned cmDataArrayEleCount( const cmData_t* d ); // Get a pointer to the base of an array. // The type must match exactly or an error is generated. // Use cmDataEleCount() to determine the number of elements in the array. cmDtRC_t cmDataBoolArray( const cmData_t* d, bool** v ); cmDtRC_t cmDataCharArray( const cmData_t* d, char** v ); cmDtRC_t cmDataUCharArray( const cmData_t* d, unsigned char** v ); cmDtRC_t cmDataShortArray( const cmData_t* d, short** v ); cmDtRC_t cmDataUShortArray( const cmData_t* d, unsigned short** v ); cmDtRC_t cmDataIntArray( const cmData_t* d, int** v ); cmDtRC_t cmDataUIntArray( const cmData_t* d, unsigned int** v ); cmDtRC_t cmDataLongArray( const cmData_t* d, long** v ); cmDtRC_t cmDataULongArray( const cmData_t* d, unsigned long** v ); cmDtRC_t cmDataFloatArray( const cmData_t* d, float** v ); cmDtRC_t cmDataDoubleArray( const cmData_t* d, double** v ); // This group of functions is a wrapper around calls to the same named // cmDataXXXArray() functions above. On error they return NULL. bool* cmDtBoolArray( const cmData_t* d ); char* cmDtCharArray( const cmData_t* d ); unsigned char* cmDtUCharArray( const cmData_t* d ); short* cmDtShortArray( const cmData_t* d ); unsigned short* cmDtUShortArray( const cmData_t* d ); int* cmDtIntArray( const cmData_t* d ); unsigned* cmDtUIntArray( const cmData_t* d ); long* cmDtLongArray( const cmData_t* d ); unsigned long* cmDtULongArray( const cmData_t* d ); float* cmDtFloatArray( const cmData_t* d ); double* cmDtDoubleArray( const cmData_t* d );
// Structure related functions // // Release an object and any resources held by it. // Note the this function does not unlink the object // from it's parent. Use cmDataUnlinkAndFree() // to remove a object from it's parent list prior // to releasing it. void cmDataFree( cmData_t* p ); // Unlink 'p' from its parents and siblings. // Asserts if parent is not a structure. // Returns 'p'. cmData_t* cmDataUnlink( cmData_t* p ); // Wrapper function to cmDataUnlink() and cmDataFree(). void cmDataUnlinkAndFree( cmData_t* p ); // Replace the 'dst' node with the 'src' node and // return 'src'. This operation does not duplicate // 'src' it simply links in 'src' at the location of // 'dst' and then unlinks and free's 'dst'. cmData_t* cmDataReplace( cmData_t* dst, cmData_t* src ); // Return the count of child nodes. // Scalars and arrays have no children. // Pairs have 2 children. // Lists have one child per element. // Records have one child per pair. unsigned cmDataChildCount( const cmData_t* p ); // Returns the ith child of 'p'. // Returns NULL if p has no children or index is invalid. cmData_t* cmDataChild( cmData_t* p, unsigned index ); // Prepend 'p' to 'parents' child list. // The source node 'p' is not duplicated it is simply linked in. // 'p' is automatically unlinked prior to being prepended. // Returns 'p'. cmData_t* cmDataPrependChild(cmData_t* parent, cmData_t* p ); // Append 'p' to the end of 'parent' child list. // The source node 'p' is not duplicated it is simply linked in. // 'p' is automatically unlinked prior to being appended. // Returns 'p'. cmData_t* cmDataAppendChild( cmData_t* parent, cmData_t* p ); // Insert 'p' at index. Index must be in the range: // 0 to cmDataChildCount(parent). // The source node 'p' is not duplicated it is simply linked in. // 'p' is automatically unlinked prior to being inserted. // Returns 'p'. cmData_t* cmDataInsertChild( cmData_t* parent, unsigned index, cmData_t* p );
// Pair related functions // // Get the key/value of a pair cmData_t* cmDataPairKey( cmData_t* p ); unsigned cmDataPairKeyId( cmData_t* p ); const cmChar_t* cmDataPairKeyLabel( cmData_t* p ); cmData_t* cmDataPairValue( cmData_t* p ); // Set the value of an existing pair node. // 'value' is not duplicated it is simply linked in place of the // previous pair value node. The previous pair value node is // unlinked and freed. // Returns 'p'. cmData_t* cmDataPairSetValue( cmData_t* p, cmData_t* value ); // Set the key of an existing pair node. // The previous key is unlinked and freed. cmData_t* cmDataPairSetKey( cmData_t* p, cmData_t* key ); cmData_t* cmDataPairSetKeyId( cmData_t* p, unsigned id ); // The data space for the 'label' string is dynamically allocated. cmData_t* cmDataPairSetKeyLabel( cmData_t* p, const cmChar_t* label ); // Create a pair value by assigning a key and value to 'p'. // 'p' is unlinked and freed prior to the key value assignment. // 'key' and 'value' are simply linked in they are not duplicated or reallocated. cmData_t* cmDataPairMake( cmData_t* parent, cmData_t* p, cmData_t* key, cmData_t* value ); // Dynamically allocate a pair node. Both the key and value nodes are reallocated. cmRC_t cmDataPairAlloc( cmData_t* parent, const cmData_t* key, const cmData_t* value, cmData_t** ref ); // Dynamically allocate the id but link (w/o realloc) the value. cmRC_t cmDataPairAllocId( cmData_t* parent, unsigned keyId, cmData_t* value, cmData_t** ref ); // Dynamically allocate the label but link (w/o realloc) the value. cmRC_t cmDataPairAllocLabelN(cmData_t* parent, const cmChar_t* label, unsigned charCnt, cmData_t* value, cmData_t** ref); cmRC_t cmDataPairAllocLabel( cmData_t* parent, const cmChar_t* label, cmData_t* value, cmData_t** ref );
// List related functions // // Return the count of ele's in the list. unsigned cmDataListCount( const cmData_t* p ); // Return the ith element in the list. cmData_t* cmDataListEle( cmData_t* p, unsigned index ); // cmData_t* cmDataListMake( cmData_t* parent, cmData_t* p ); cmData_t* cmDataListAlloc( cmData_t* parent); // Dynamically allocate a new list object and fill it with values // using a variable argument list. // // 1) Var-args fmt: // &lttypeId&gt {&ltvalue&gt} {&ltcnt&gt} // // 2) &lttypeId&gt is formed by OR-ing one of the container type flags with a // data type id. (e.g. kArrayDtId | kIntDtId). Note that if no // container is given then the type is assumed to be scalar. // // scalar types: &ltvalue&gt is literal, &ltcnt&gt should not be included // null scalar object should not include &ltvalue&gt or &ltcnt&gt // array types: &ltvalue&gt is pointer,&ltcnt&gt is element count // struct types: &ltvalue&gt is cmData_t pointer, &ltcnt&gt should not be included. // The struct object is appended to the child list of parent. // // 3) Indicate the end of argument list by setting &lttypeId&gt to kInvalidDtId. // 4) Scalar and array objects will be dynamically allocated. // 5) Data attribute flags (kXXXDtFl) make also be OR-ed into the &lttypeId&gt. // 6) Data attribute flags will only be used with scalar and array object and // will be ignored for all other object types. cmRC_t cmDataListAllocV( cmData_t* parent, cmErr_t* err, cmRC_t errRC, cmData_t** ref, va_list vl ); cmRC_t cmDataListAllocA( cmData_t* parent, cmErr_t* err, cmRC_t errRC, cmData_t** ref, ... ); // Returns a ptr to 'ele'. cmData_t* cmDataListAppendEle( cmData_t* p, cmData_t* ele ); cmDtRC_t cmDataListAppendV( cmData_t* p, cmErr_t* err, cmRC_t errRC, va_list vl ); cmDtRC_t cmDataListAppend( cmData_t* p, cmErr_t* err, cmRC_t errRC, ... ); // Return 'p'. cmData_t* cmDataListInsertEle( cmData_t* p, unsigned index, cmData_t* ele ); cmData_t* cmDataListInsertEleN(cmData_t* p, unsigned index, cmData_t* ele[], unsigned n );
// Record related functions // // Return count of pairs. unsigned cmDataRecdCount( const cmData_t* p ); // Return the ith pair. cmData_t* cmDataRecdEle( cmData_t* p, unsigned index ); // Return the ith value. cmData_t* cmDataRecdValueFromIndex( cmData_t* p, unsigned index ); cmData_t* cmDataRecdValueFromId( cmData_t* p, unsigned id ); cmData_t* cmDataRecdValueFromLabel( cmData_t* p, const cmChar_t* label ); // Return the ith key or NULL if it does not exist. cmData_t* cmDataRecdKey( cmData_t* p, unsigned index ); // Returns cmInvalidId if the ith key does not exist or is not an integer. unsigned cmDataRecdKeyId( cmData_t* p, unsigned index ); // Return NULL if the ith key does not exist or is not a string. const cmChar_t* cmDataRecdKeyLabel( cmData_t* p, unsigned index ); cmData_t* cmDataRecdMake( cmData_t* parent, cmData_t* p ); cmData_t* cmDataRecdAlloc( cmData_t* parent ); // Append a pair node by linking the pair node 'pair' to the record node 'p'. // 'pair' is simply linked to 'p' via cmDataAppendChild() no // reallocation or duplicattion takes place. cmData_t* cmDataRecdAppendPair( cmData_t* p, cmData_t* pair ); // Dynamically allocate a new record object and fill it with values // using a variable argument list. // // 1) Var-args format: // &ltlabel|id&gt &ltvalue&gt // // The &ltlabel|id&gt arg. gives the record pair label and the &ltvalue&gt // argument list gives the pair value. // // 2) The &ltvalue&gt argument has the same syntax as the cmDataListAllocLabelV() // call. // 3) The cmDataRecdAllocLabelV() arg. list should be terminated with NULL. // The cmDataRecdAllocIdV() arg. list should be terminated with 'kInvalidDtId'. cmDtRC_t cmDataRecdAllocLabelV( cmData_t* parent, cmErr_t* err, cmRC_t errRC, cmData_t** ref, va_list vl ); cmDtRC_t cmDataRecdAllocLabelA( cmData_t* parent, cmErr_t* err, cmRC_t errRC, cmData_t** ref, ... ); cmDtRC_t cmDataRecdAllocIdV( cmData_t* parent, cmErr_t* err, cmRC_t errRC, cmData_t** ref, va_list vl ); cmDtRC_t cmDataRecdAllocIdA( cmData_t* parent, cmErr_t* err, cmRC_t errRC, cmData_t** ref, ... ); // Extract the data in a record to C variables. // Var-args format: // (label | id) &ltcid&gt &lttypeId&gt &ltptr&gt {cnt_ptr} // The var-args list must be NULL terminated. // The &lt'id' | 'label'&gt identify a pair. // The &ltcid&gt indicates the type of the target container. // The &lttypeId&gt indicates the C type of 'pointer'. // If &ltcid&gt is kArrayDtId then the &ltcnt_ptr} must be include to receive the // count of elements in the array. // The actual field type must be convertable into this pointer type or the // function will fail. // 'err' is an application supplied error object to be used if a required // field is missing. 'errRC' is the client result code to be passed with // the error report. See cmErrMsg(). Both 'err' and 'errRC' are optional. // Set kOptArgDtFl on 'typeId' to indicate that a field is optional. // &ltlabel|id&gt (&lttypeId&gt | kOptArgDtFl) &ltpointer&gt cmDtRC_t cmDataRecdParseLabelV(cmData_t* p, cmErr_t* err, unsigned errRC, va_list vl ); cmDtRC_t cmDataRecdParseLabel( cmData_t* p, cmErr_t* err, unsigned errRC, ... ); cmDtRC_t cmDataRecdParseIdV( cmData_t* p, cmErr_t* err, unsigned errRC, va_list vl ); cmDtRC_t cmDataRecdParseId( cmData_t* p, cmErr_t* err, unsigned errRC, ... );
// Serialization related functions // unsigned cmDataSerializeByteCount( const cmData_t* p ); cmDtRC_t cmDataSerialize( const cmData_t* p, void* buf, unsigned bufByteCnt ); cmDtRC_t cmDataDeserialize( const void* buf, unsigned bufByteCnt, cmData_t** pp );
// Text to Data related functions // typedef cmHandle_t cmDataParserH_t; extern cmDataParserH_t cmDataParserNullHandle; cmDtRC_t cmDataParserCreate( cmCtx_t* ctx, cmDataParserH_t* hp ); cmDtRC_t cmDataParserDestroy( cmDataParserH_t* hp ); bool cmDataParserIsValid( cmDataParserH_t h ); // Parse a text representation into a 'record' type. // Note that the text is wrapped with implied curly braces // (e.g. &quot{ text }&quot). The contents of the text should therefore // fit the record syntax (e.g. the first token should be a // 'pair' label. cmDtRC_t cmDataParserExec( cmDataParserH_t h, const cmChar_t* text, cmData_t** pp );
void cmDataPrint( const cmData_t* p, cmRpt_t* rpt ); void cmDataTest( cmCtx_t* ctx );