cwFlowValue.h/cpp : Implemented recd_fmt_t to alloc spec. of 'record' in class desc.

Implemented recd_get()/set() for multiple fields at once.
Moved value_from_value() and value_from_cfg() into cwFlowValue from cwFlowType
This commit is contained in:
kevin 2024-11-25 09:55:05 -05:00
parent c450378d78
commit e3423e775e
2 changed files with 621 additions and 234 deletions

View File

@ -70,6 +70,7 @@ namespace cw
if( f->group_fl ) if( f->group_fl )
_recd_type_destroy_field_list(f->u.group_fieldL); _recd_type_destroy_field_list(f->u.group_fieldL);
mem::release(f->doc);
mem::release(f->label); mem::release(f->label);
mem::release(f); mem::release(f);
@ -77,113 +78,175 @@ namespace cw
} }
} }
rc_t _recd_type_add_field( recd_field_t*& fieldL_ref, const char* field_label, recd_field_t*& new_field_ref ) unsigned _recd_field_list_set_index( recd_field_t* fld, unsigned index )
{ {
rc_t rc = kOkRC; for(recd_field_t* f=fld; f!=nullptr; f=f->link)
recd_field_t* f = nullptr; if( f->group_fl )
recd_field_t* new_field = nullptr; index = _recd_field_list_set_index(f->u.group_fieldL,index);
new_field_ref = nullptr;
if( textLength(field_label) == 0 )
{
rc = cwLogError(kInvalidArgRC,"A blank msg field was encountered.");
goto errLabel;
}
for(f=fieldL_ref; f!=nullptr; f=f->link)
{
if( textIsEqual(field_label,f->label))
{
rc = cwLogError(kInvalidArgRC,"A duplicate msg field '%s' has been encountered.", field_label);
goto errLabel;
}
// add the new field to the end of the list
if( f->link == nullptr )
break;
}
new_field = mem::allocZ<recd_field_t>();
new_field->label = mem::duplStr(field_label);
if( f == nullptr )
fieldL_ref = new_field;
else else
f->link = new_field; f->u.index = index++;
return index;
new_field_ref = new_field;
errLabel:
return rc;
} }
rc_t _recd_type_add_value_field( recd_field_t*& fieldL_ref, const char* field_label, unsigned index ) const char* _recd_field_index_to_label(const recd_field_t* fld, unsigned field_idx)
{ {
rc_t rc; const char* label = nullptr;
recd_field_t* new_field = nullptr; for(const recd_field_t* f=fld; f!=nullptr; f=f->link)
if( f->group_fl )
if((rc = _recd_type_add_field( fieldL_ref, field_label, new_field )) != kOkRC ) label = _recd_field_index_to_label(f->u.group_fieldL,field_idx);
else
{ {
rc = cwLogError(rc,"Record add value field '%s' failed.",cwStringNullGuard(field_label)); if(f->u.index == field_idx )
goto errLabel; label = f->label;
} }
new_field->group_fl = false; return label;
new_field->u.index = index;
errLabel:
return rc;
}
rc_t _recd_type_add_group_field( recd_field_t*& fieldL_ref, const char* field_label, recd_field_t*& new_field_ref )
{
rc_t rc;
if((rc = _recd_type_add_field( fieldL_ref, field_label, new_field_ref )) != kOkRC )
{
rc = cwLogError(rc,"Record add group field '%s' failed.",cwStringNullGuard(field_label));
goto errLabel;
}
new_field_ref->group_fl = true;
errLabel:
return rc;
} }
rc_t _recd_type_add_value_fields( recd_type_t* recd_type, recd_field_t*& fieldL_ref, const char* field_labels ) rc_t _recd_field_list_from_cfg( recd_field_t*& field_list_ref, const object_t* field_dict_cfg )
{ {
rc_t rc = kOkRC; rc_t rc = kOkRC;
const char* s0 = field_labels;
while(s0 != nullptr) if( !field_dict_cfg->is_dict() )
{ {
s0 = nextNonWhiteChar(s0); rc = cwLogError(kSyntaxErrorRC,"The field cfg. is not a dictionary.");
goto errLabel;
const char* s1 = nextWhiteChar(s0); }
else
unsigned sn = s1==nullptr ? textLength(s0) : s1-s0;
char s[sn+1];
strncpy(s,s0,sn);
s[sn] = 0;
if((rc = _recd_type_add_value_field(fieldL_ref, s, recd_type->fieldN)) != kOkRC )
{ {
unsigned row_cnt = field_dict_cfg->child_count();
for(unsigned i=0; i<row_cnt; ++i)
{
const object_t* pair = field_dict_cfg->child_ele(i);
recd_field_t* field = nullptr;
const char* type_label = nullptr;
const char* doc_string = nullptr;
const object_t* val_cfg = nullptr;
// parse the required fields
if((rc = pair->pair_value()->getv("type",type_label,
"doc",doc_string)) != kOkRC )
{
rc = cwLogError(rc,"Error parsing the field '%s'.",cwStringNullGuard(pair->pair_label()));
goto errLabel; goto errLabel;
} }
recd_type->fieldN += 1; // allocate the field record
field = mem::allocZ<recd_field_t>();
s0 = s1; // add the new field to the end of the field list
if( field_list_ref == nullptr )
field_list_ref = field;
else
{
recd_field_t* f = field_list_ref;
while( f->link != nullptr )
f = f->link;
assert(f!=nullptr);
f->link = field;
}
field->label = mem::duplStr(pair->pair_label());
field->doc = mem::duplStr(doc_string);
if( textIsEqual(type_label,"group") )
{
const object_t* field_dict;
field->group_fl = true;
// get the group 'fields' dictionary
if((rc = pair->pair_value()->getv("fields",field_dict)) != kOkRC )
{
rc = cwLogError(rc,"The field group '%s' does not have a field list.",pair->pair_label());
goto errLabel;
}
// recursively read the group field list
if((rc = _recd_field_list_from_cfg(field->u.group_fieldL,field_dict)) != kOkRC )
{
rc = cwLogError(rc,"The creation of field group '%s' failed.",pair->pair_label());
goto errLabel;
}
}
else
{
// validate the value type flag
if((field->value.tflag = value_type_label_to_flag( type_label )) == kInvalidTFl )
{
rc = cwLogError(kSyntaxErrorRC,"The value type label '%s' is not valid on the field specifier '%s'.",cwStringNullGuard(type_label),cwStringNullGuard(pair->pair_label()));
goto errLabel;
}
// get the optional default value
if((val_cfg = pair->pair_value()->find("value")) != nullptr )
{
value_t v;
v.tflag = kInvalidTFl;
// parse the value into 'v'
if((rc = value_from_cfg(val_cfg,v)) != kOkRC )
{
rc = cwLogError(rc,"The default value parse failed for the field '%s'.",cwStringNullGuard(pair->pair_label()));
goto errLabel;
}
// convert the value from 'v' into field->value
if((rc = value_from_value(v,field->value)) != kOkRC )
{
rc = cwLogError(rc,"The default value assignment failed for the field '%s'.",cwStringNullGuard(pair->pair_label()));
goto errLabel;
}
}
}
}
}
errLabel:
return rc;
}
void _recd_set_value_type( recd_field_t* fieldL, recd_t* r )
{
recd_field_t* f = fieldL;
for(; f!=nullptr; f=f->link)
if( f->group_fl )
_recd_set_value_type( f->u.group_fieldL, r );
else
r->valA[ f->u.index ].tflag = f->value.tflag;
}
rc_t _recd_set_default_value( recd_field_t* fieldL, recd_t* r )
{
rc_t rc = kOkRC;
recd_field_t* f = fieldL;
for(; f!=nullptr; f=f->link)
{
if( f->group_fl )
_recd_set_default_value( f->u.group_fieldL, r );
else
{
if(f->value.tflag != kInvalidTFl)
{
if((rc = value_from_value( f->value, r->valA[f->u.index] )) != kOkRC )
{
rc = cwLogError(rc,"Set default value failed on the field '%s'.",cwStringNullGuard(f->label));
goto errLabel;
}
}
}
} }
errLabel: errLabel:
return rc; return rc;
} }
@ -780,6 +843,136 @@ void cw::flow::value_duplicate( value_t& dst, const value_t& src )
} }
cw::rc_t cw::flow::value_from_cfg( const object_t* cfg, value_t& value_ref )
{
rc_t rc = kOkRC;
switch( cfg->type->id )
{
case kCharTId:
case kUInt8TId:
case kUInt16TId:
case kUInt32TId:
value_ref.tflag = kUIntTFl;
if((rc = cfg->value(value_ref.u.u)) != kOkRC )
rc = cwLogError(rc,"Conversion to uint failed.");
break;
case kInt8TId:
case kInt16TId:
case kInt32TId:
value_ref.tflag = kIntTFl;
if((rc = cfg->value(value_ref.u.i)) != kOkRC )
rc = cwLogError(rc,"Conversion to int failed.");
break;
case kInt64TId:
case kUInt64TId:
rc = cwLogError(kInvalidArgRC,"The flow system does not currently implement 64bit integers.");
goto errLabel;
break;
case kFloatTId:
value_ref.tflag = kFloatTFl;
if((rc = cfg->value(value_ref.u.f)) != kOkRC )
rc = cwLogError(rc,"Conversion to float failed.");
break;
case kDoubleTId:
value_ref.tflag = kDoubleTFl;
if((rc = cfg->value(value_ref.u.d)) != kOkRC )
rc = cwLogError(rc,"Conversion to double failed.");
break;
case kBoolTId:
value_ref.tflag = kBoolTFl;
if((rc = cfg->value(value_ref.u.b)) != kOkRC )
rc = cwLogError(rc,"Conversion to bool failed.");
break;
case kStringTId:
case kCStringTId:
value_ref.tflag = kStringTFl;
if((rc = cfg->value(value_ref.u.s)) != kOkRC )
rc = cwLogError(rc,"Conversion to string failed.");
break;
default:
value_ref.tflag = kCfgTFl;
value_ref.u.cfg = cfg;
}
errLabel:
return rc;
}
cw::rc_t cw::flow::value_from_value( const value_t& src, value_t& dst )
{
rc_t rc = kOkRC;
if( dst.tflag == kInvalidTFl || dst.tflag & src.tflag)
{
dst = src;
return kOkRC;
}
// we only get here if conversion is necessary
switch( src.tflag )
{
case kInvalidTFl:
rc = cwLogError(kInvalidStateRC,"The src operand does not have a valid type.");
break;
case kBoolTFl:
rc = value_set(&dst,src.u.b);
break;
case kUIntTFl:
rc = value_set(&dst,src.u.u);
break;
case kIntTFl:
rc = value_set(&dst,src.u.i);
break;
case kFloatTFl:
rc = value_set(&dst,src.u.f);
break;
case kDoubleTFl:
rc = value_set(&dst,src.u.d);
break;
case kBoolMtxTFl:
case kUIntMtxTFl:
case kIntMtxTFl:
case kFloatMtxTFl:
case kDoubleMtxTFl:
rc = cwLogError(kNotImplementedRC,"Matrix conversion is not implemented for value to value conversion.");
break;
case kABufTFl:
case kFBufTFl:
case kMBufTFl:
case kRBufTFl:
case kStringTFl:
case kTimeTFl:
case kCfgTFl:
case kMidiTFl:
rc = cwLogError(kOpFailRC,"Value conversion failed during value to value assignement.");
break;
default:
rc = cwLogError(kInvalidArgRC,"An unknown source operand data type 0x%x was encountered.",src.tflag);
}
return rc;
}
void cw::flow::value_print( const value_t* v, bool info_fl ) void cw::flow::value_print( const value_t* v, bool info_fl )
{ {
if( v == nullptr ) if( v == nullptr )
@ -1394,6 +1587,21 @@ cw::rc_t cw::flow::value_get( const value_t* val, midi::ch_msg_t*& valRef )
return rc; return rc;
} }
cw::rc_t cw::flow::value_get( const value_t* val, const midi::ch_msg_t*& valRef )
{
rc_t rc = kOkRC;
if( cwIsFlag(val->tflag & kTypeMask, kMidiTFl) )
valRef = val->u.midi;
else
{
valRef = nullptr;
rc = cwLogError(kTypeMismatchRC,"The type %s (0x%x) could not be converted to a MIDI record.",_typeFlagToLabel(val->tflag),val->tflag);
}
return rc;
}
cw::rc_t cw::flow::value_set(value_t* val, midi::ch_msg_t* v ) cw::rc_t cw::flow::value_set(value_t* val, midi::ch_msg_t* v )
{ {
rc_t rc = kOkRC; rc_t rc = kOkRC;
@ -1406,7 +1614,7 @@ cw::rc_t cw::flow::value_set(value_t* val, midi::ch_msg_t* v )
case kInvalidTFl: case kInvalidTFl:
val->u.midi=v; val->u.midi=v;
val->tflag = kCfgTFl; val->tflag = kMidiTFl;
break; break;
default: default:
@ -1417,18 +1625,75 @@ cw::rc_t cw::flow::value_set(value_t* val, midi::ch_msg_t* v )
} }
cw::rc_t cw::flow::recd_type_create( recd_type_t*& recd_type_ref, const recd_type_t* base, const char* fields_str ) //------------------------------------------------------------------------------------------------------------------------
//
// Record
//
cw::rc_t cw::flow::recd_format_create( recd_fmt_t*& recd_fmt_ref, const object_t* cfg )
{ {
rc_t rc = kOkRC; rc_t rc = kOkRC;
recd_fmt_t* recd_fmt = nullptr;
recd_fmt_ref = nullptr;
recd_fmt = mem::allocZ<recd_fmt_t>();
if((rc = recd_type_create(recd_fmt->recd_type,nullptr,cfg)) != kOkRC )
goto errLabel;
recd_fmt->alloc_cnt = 32; // TODO: this should not be hard coded
if((rc =cfg->getv_opt("alloc_cnt",recd_fmt->alloc_cnt)) != kOkRC )
{
rc = cwLogError(rc,"Error parsing record format 'alloc_cnt'.");
goto errLabel;
}
recd_fmt_ref = recd_fmt;
errLabel:
if(rc != kOkRC )
rc = cwLogError(rc,"Record format creation failed.");
return rc;
}
void cw::flow::recd_format_destroy( recd_fmt_t*& recd_fmt_ref )
{
if( recd_fmt_ref != nullptr )
{
recd_type_destroy(recd_fmt_ref->recd_type);
mem::release(recd_fmt_ref);
}
}
cw::rc_t cw::flow::recd_type_create( recd_type_t*& recd_type_ref, const recd_type_t* base, const object_t* cfg )
{
rc_t rc = kOkRC;
const object_t* fields_dict = nullptr;;
recd_type_t* recd_type = mem::allocZ<recd_type_t>(); recd_type_t* recd_type = mem::allocZ<recd_type_t>();
recd_type_ref = nullptr; recd_type_ref = nullptr;
if((rc = recd_type_add_value_fields( recd_type, fields_str )) != kOkRC ) // get the fields list
if((rc = cfg->getv("fields",fields_dict)) != kOkRC )
{
rc = cwLogError(rc,"The 'fields' dictionary was not found in the record 'fmt' specifier.");
goto errLabel;
}
// load the fields list
if((rc = _recd_field_list_from_cfg(recd_type->fieldL,fields_dict)) != kOkRC )
{ {
goto errLabel; goto errLabel;
} }
// assign the index to the value fields and update recd_type.fieldN
recd_type->fieldN = _recd_field_list_set_index(recd_type->fieldL, 0 );
recd_type->base = base; recd_type->base = base;
recd_type_ref = recd_type; recd_type_ref = recd_type;
@ -1451,34 +1716,6 @@ void cw::flow::recd_type_destroy( recd_type_t*& recd_type_ref )
mem::release(recd_type_ref); mem::release(recd_type_ref);
} }
cw::rc_t cw::flow::recd_type_add_value_fields( recd_type_t* recd_type, const char* field_labels )
{
return _recd_type_add_value_fields( recd_type, recd_type->fieldL, field_labels );
}
cw::rc_t cw::flow::recd_type_add_group( recd_type_t* recd_type, const char* group_label, const char* field_labels )
{
rc_t rc;
recd_field_t* f = nullptr;
if((rc = _recd_type_add_group_field( recd_type->fieldL, group_label, f )) != kOkRC )
{
goto errLabel;
}
if((rc = _recd_type_add_value_fields( recd_type, f->u.group_fieldL, field_labels )) != kOkRC )
{
goto errLabel;
}
errLabel:
if( rc !=kOkRC )
rc = cwLogError(rc,"Record group add failed.");
return rc;
}
unsigned cw::flow::recd_type_max_field_count( const recd_type_t* recd_type ) unsigned cw::flow::recd_type_max_field_count( const recd_type_t* recd_type )
{ {
@ -1502,31 +1739,63 @@ errLabel:
return index; return index;
} }
const char* cw::flow::recd_type_field_index_to_label( const recd_type_t* recd_type, unsigned field_idx )
{
const char* label = nullptr;
if( field_idx >= recd_type->fieldN )
label = recd_type_field_index_to_label(recd_type->base, field_idx - recd_type->fieldN );
else
label = _recd_field_index_to_label(recd_type->fieldL,field_idx);
return label;
}
void cw::flow::recd_type_print( const recd_type_t* recd_type ) void cw::flow::recd_type_print( const recd_type_t* recd_type )
{ {
_recd_type_print(recd_type,recd_type); _recd_type_print(recd_type,recd_type);
} }
cw::rc_t cw::flow::recd_print( const recd_type_t* rt, const recd_t* r ) cw::rc_t cw::flow::recd_init( const recd_type_t* recd_type, const recd_t* base, recd_t* r )
{ return _recd_print( rt, r ); } {
r->base = base;
return _recd_set_default_value( recd_type->fieldL, r );
}
cw::rc_t cw::flow::recd_print( const recd_type_t* recd_type, const recd_t* r )
{ return _recd_print( recd_type, r ); }
cw::rc_t cw::flow::recd_array_create( recd_array_t*& recd_array_ref, recd_type_t* recd_type, unsigned allocRecdN ) cw::rc_t cw::flow::recd_array_create( recd_array_t*& recd_array_ref, recd_type_t* recd_type, const recd_type_t* base, unsigned allocRecdN )
{ {
rc_t rc = kOkRC; rc_t rc = kOkRC;
recd_array_t* recd_array = mem::allocZ<recd_array_t>(); recd_array_t* recd_array = mem::allocZ<recd_array_t>();
recd_array_ref = nullptr; recd_array_ref = nullptr;
recd_array->type = recd_type; recd_array->type = mem::allocZ<recd_type_t>();
recd_array->type->fieldL = recd_type->fieldL;
recd_array->type->fieldN = recd_type->fieldN;
recd_array->type->base = base;
recd_array->valA = mem::allocZ<value_t>(recd_array->type->fieldN * allocRecdN); recd_array->valA = mem::allocZ<value_t>(recd_array->type->fieldN * allocRecdN);
recd_array->recdA = mem::allocZ<recd_t>(allocRecdN); recd_array->recdA = mem::allocZ<recd_t>(allocRecdN);
recd_array->allocRecdN = allocRecdN; recd_array->allocRecdN = allocRecdN;
// for each record
for(unsigned i=0; i<allocRecdN; ++i) for(unsigned i=0; i<allocRecdN; ++i)
{
// set the value array for this record
recd_array->recdA[i].valA = recd_array->valA + (i*recd_array->type->fieldN); recd_array->recdA[i].valA = recd_array->valA + (i*recd_array->type->fieldN);
// set the value type of all records in the array
_recd_set_value_type( recd_array->type->fieldL, recd_array->recdA + i );
}
recd_array_ref = recd_array; recd_array_ref = recd_array;
//if( rc != kOkRC ) //if( rc != kOkRC )
@ -1539,6 +1808,7 @@ cw::rc_t cw::flow::recd_array_destroy( recd_array_t*& recd_array_ref )
{ {
if( recd_array_ref != nullptr ) if( recd_array_ref != nullptr )
{ {
mem::release(recd_array_ref->type);
mem::release(recd_array_ref->valA); mem::release(recd_array_ref->valA);
mem::release(recd_array_ref->recdA); mem::release(recd_array_ref->recdA);
mem::release(recd_array_ref); mem::release(recd_array_ref);
@ -1547,67 +1817,86 @@ cw::rc_t cw::flow::recd_array_destroy( recd_array_t*& recd_array_ref )
return kOkRC; return kOkRC;
} }
cw::rc_t cw::flow::value_test( const test::test_args_t& args ) cw::rc_t cw::flow::value_test( const test::test_args_t& args )
{ {
rc_t rc = kOkRC; rc_t rc = kOkRC;
recd_type_t* rt0 = nullptr; object_t* cfg0 = nullptr;
recd_type_t* rt1 = nullptr; object_t* cfg1 = nullptr;
recd_fmt_t* fmt0 = nullptr;
recd_fmt_t* fmt1 = nullptr;
recd_array_t* ra0 = nullptr; recd_array_t* ra0 = nullptr;
recd_array_t* ra1 = nullptr; recd_array_t* ra1 = nullptr;
if((rc = recd_type_create( rt0, nullptr, "a b c" )) != kOkRC ) const char* s0 = "{ alloc_cnt:3, fields: {"
"a: { type:bool, doc:\"A floater.\" },"
"b: { type:uint, value:1, doc:\"My uint.\" },"
"c: { type:uint, value:2, doc:\"My other uint.\" }"
"g0: { type:group, doc:\"A group.\""
"fields:{ a:{type:int, value:0, doc:\"My int.\" }"
" b:{type:bool, value:true, doc:\"My flag.\" }"
" c:{type:double, value:1, doc:\"Another field.\" }"
"}}"
"}}";
const char* s1 = "{ alloc_cnt:3, fields: {"
"d: { type:double, doc:\"d doc.\" },"
"e: { type:uint, value:-1, doc:\"e doc.\" },"
"f: { type:uint, value:-1, doc:\"f doc.\" }"
"g1: { type:group, doc:\"A group.\""
"fields:{ a:{type:int, value:0, doc:\"My int.\" }"
" b:{type:bool, value:true, doc:\"My flag.\" }"
" c:{type:uint, value:1, doc:\"Another field.\" }"
"}}"
"}}";
if((rc = objectFromString(s0,cfg0)) != kOkRC )
{ {
rc = cwLogError(rc,"rt0 create failed."); rc = cwLogError(rc,"cfg0 parse failed.");
goto errLabel; goto errLabel;
} }
if((rc = recd_type_add_group( rt0, "g0", "a b c" )) != kOkRC ) if((rc = objectFromString(s1,cfg1)) != kOkRC )
{ {
rc = cwLogError(rc,"rt.g0 group create failed."); rc = cwLogError(rc,"cfg1 parse failed.");
goto errLabel; goto errLabel;
} }
if((rc = recd_type_create( rt1, rt0, "d e f" )) != kOkRC ) if((rc = recd_format_create( fmt0, cfg0 )) != kOkRC )
{ {
rc = cwLogError(rc,"rt0 create failed."); rc = cwLogError(rc,"fmt0 create failed.");
goto errLabel; goto errLabel;
} }
if((rc = recd_type_add_group( rt1, "g1", "a b c" )) != kOkRC ) if((rc = recd_format_create( fmt1, cfg1 )) != kOkRC )
{ {
rc = cwLogError(rc,"rt.g0 group create failed."); rc = cwLogError(rc,"fmt1 create failed.");
goto errLabel; goto errLabel;
} }
recd_type_print( rt0 ); if((rc = recd_array_create( ra0, fmt0->recd_type, nullptr, fmt0->alloc_cnt )) != kOkRC )
recd_type_print( rt1 );
if((rc = recd_array_create(ra0,rt0,10)) != kOkRC )
{ {
rc = cwLogError(rc,"ra0 alloc failed."); rc = cwLogError(rc,"recd array 0 create failed.");
goto errLabel; goto errLabel;
} }
if((rc = recd_array_create(ra1,rt1,10)) != kOkRC ) if((rc = recd_array_create( ra1, fmt1->recd_type, fmt0->recd_type, fmt1->alloc_cnt )) != kOkRC )
{ {
rc = cwLogError(rc,"ra1 alloc failed."); rc = cwLogError(rc,"recd array 0 create failed.");
goto errLabel; goto errLabel;
} }
for(unsigned i=0; i<ra0->allocRecdN; ++i) for(unsigned i=0; i<ra0->allocRecdN; ++i)
{ {
recd_t* r = ra0->recdA + i; recd_t* r = ra0->recdA + i;
rc_t rc0 = recd_set( ra0->type, nullptr, r, recd_type_field_index(ra0->type,"a"), 0*i);
rc_t rc1 = recd_set( ra0->type, nullptr, r, recd_type_field_index(ra0->type,"b"), 1*i);
rc_t rc2 = recd_set( ra0->type, nullptr, r, recd_type_field_index(ra0->type,"c"), 2*i);
rc_t rc3 = recd_set( ra0->type, nullptr, r, recd_type_field_index(ra0->type,"g0.a"), 4.0*i);
rc_t rc4 = recd_set( ra0->type, nullptr, r, recd_type_field_index(ra0->type,"g0.b"), 5.0*i);
rc_t rc5 = recd_set( ra0->type, nullptr, r, recd_type_field_index(ra0->type,"g0.c"), 6.0*i);
if((rc = rcSelect(rc0,rc1,rc2,rc3,rc4,rc5)) != kOkRC ) if((rc = recd_set( ra0->type, nullptr, r,
recd_type_field_index(ra0->type,"a"), 0.0f*i,
recd_type_field_index(ra0->type,"g0.a"), 4.0*i,
recd_type_field_index(ra0->type,"g0.b"), 5*i,
recd_type_field_index(ra0->type,"g0.c"), 6*i)) != kOkRC )
{ {
rc = cwLogError(rc,"recd_set() failed on ra0."); cwLogError(rc,"recd_set() failed on ra0.");
goto errLabel; goto errLabel;
} }
} }
@ -1615,20 +1904,19 @@ cw::rc_t cw::flow::value_test( const test::test_args_t& args )
for(unsigned i=0; i<ra0->allocRecdN; ++i) for(unsigned i=0; i<ra0->allocRecdN; ++i)
recd_print(ra0->type,ra0->recdA+i); recd_print(ra0->type,ra0->recdA+i);
for(unsigned i=0; i<ra1->allocRecdN; ++i) for(unsigned i=0; i<ra1->allocRecdN; ++i)
{ {
recd_t* r = ra1->recdA + i; recd_t* r = ra1->recdA + i;
recd_t* r_base = ra0->recdA + i; recd_t* r_base = ra0->recdA + i;
rc_t rc0 = recd_set( ra1->type, r_base, r, recd_type_field_index(ra1->type,"d"), 0*i*2);
rc_t rc1 = recd_set( ra1->type, r_base, r, recd_type_field_index(ra1->type,"e"), 1*i*2);
rc_t rc2 = recd_set( ra1->type, r_base, r, recd_type_field_index(ra1->type,"f"), 2*i*2);
rc_t rc3 = recd_set( ra1->type, r_base, r, recd_type_field_index(ra1->type,"g1.a"), 4.0*i*2);
rc_t rc4 = recd_set( ra1->type, r_base, r, recd_type_field_index(ra1->type,"g1.b"), 5.0*i*2);
rc_t rc5 = recd_set( ra1->type, r_base, r, recd_type_field_index(ra1->type,"g1.c"), 6.0*i*2);
if((rc = rcSelect(rc0,rc1,rc2,rc3,rc4,rc5)) != kOkRC ) if((rc = recd_set( ra1->type, r_base, r,
recd_type_field_index(ra1->type,"d"), 0.0f*i,
recd_type_field_index(ra1->type,"g1.a"), 4.0*i*2,
recd_type_field_index(ra1->type,"g1.b"), 5*i*2,
recd_type_field_index(ra1->type,"g1.c"), 6*i*2)) != kOkRC )
{ {
rc = cwLogError(rc,"recd_set() failed on ra1."); cwLogError(rc,"recd_set() failed on ra1.");
goto errLabel; goto errLabel;
} }
} }
@ -1636,11 +1924,18 @@ cw::rc_t cw::flow::value_test( const test::test_args_t& args )
for(unsigned i=0; i<ra1->allocRecdN; ++i) for(unsigned i=0; i<ra1->allocRecdN; ++i)
recd_print(ra1->type,ra1->recdA+i); recd_print(ra1->type,ra1->recdA+i);
recd_array_destroy( ra0 ); recd_array_destroy( ra0 );
recd_array_destroy( ra1 ); recd_array_destroy( ra1 );
recd_type_destroy(rt0);
recd_type_destroy(rt1); recd_format_destroy( fmt0 );
recd_format_destroy( fmt1 );
cfg0->free();
cfg1->free();
errLabel: errLabel:
return rc; return rc;
} }

View File

@ -83,7 +83,7 @@ namespace cw
kTypeMask = 0x0003ffff, kTypeMask = 0x0003ffff,
kRuntimeTFl = 0x80000000, kRuntimeTFl = 0x80000000, // The type of the value associated with this variable will be set by the proc instances during instantiation
kNumericTFl = kBoolTFl | kUIntTFl | kIntTFl | kFloatTFl | kDoubleTFl, kNumericTFl = kBoolTFl | kUIntTFl | kIntTFl | kFloatTFl | kDoubleTFl,
kMtxTFl = kBoolMtxTFl | kUIntMtxTFl | kIntMtxTFl | kFloatMtxTFl | kDoubleMtxTFl, kMtxTFl = kBoolMtxTFl | kUIntMtxTFl | kIntMtxTFl | kFloatMtxTFl | kDoubleMtxTFl,
@ -100,37 +100,12 @@ namespace cw
} u; } u;
} mtx_t; } mtx_t;
typedef struct recd_field_str struct recd_type_str;
{ struct recd_str;
bool group_fl; // set if this field record is a group
char* label; // field or group label
union
{
unsigned index; // index into recd_t.valA of the value associated with this field
struct recd_field_str* group_fieldL;
} u;
struct recd_field_str* link;
} recd_field_t;
typedef struct recd_type_str
{
recd_field_t* fieldL; // linked list of field spec's
unsigned fieldN; // length of fieldL list (fieldN + base->fieldN) is total field count
const struct recd_type_str* base;
} recd_type_t;
struct value_str;
typedef struct recd_str
{
struct value_str* valA; // varA[ fieldN ] array of field values
const struct recd_str* base; // base record fields
} recd_t;
typedef struct rbuf_str typedef struct rbuf_str
{ {
const recd_type_t* type; // all msgs are formed from this type const struct recd_type_str* type; // all msgs are formed from this type
const recd_t* recdA; // recdA[ recdN ] const struct recd_str* recdA; // recdA[ recdN ]
unsigned recdN; // unsigned recdN; //
} rbuf_t; } rbuf_t;
@ -200,10 +175,10 @@ namespace cw
void mbuf_destroy( mbuf_t*& buf ); void mbuf_destroy( mbuf_t*& buf );
mbuf_t* mbuf_duplicate( const mbuf_t* src ); mbuf_t* mbuf_duplicate( const mbuf_t* src );
rbuf_t* rbuf_create( const recd_type_t* type=nullptr, const recd_t* recdA=nullptr, unsigned recdN=0 ); rbuf_t* rbuf_create( const struct recd_type_str* type=nullptr, const struct recd_str* recdA=nullptr, unsigned recdN=0 );
void rbuf_destroy( rbuf_t*& buf ); void rbuf_destroy( rbuf_t*& buf );
rbuf_t* rbuf_duplicate( const rbuf_t* src ); rbuf_t* rbuf_duplicate( const rbuf_t* src );
void rbuf_setup( rbuf_t* rbuf, recd_type_t* type, recd_t* recdA, unsigned recdN ); void rbuf_setup( rbuf_t* rbuf, struct recd_type_str* type, struct recd_str* recdA, unsigned recdN );
inline bool value_is_abuf( const value_t* v ) { return v->tflag & kABufTFl; } inline bool value_is_abuf( const value_t* v ) { return v->tflag & kABufTFl; }
@ -215,6 +190,14 @@ namespace cw
void value_release( value_t* v ); void value_release( value_t* v );
void value_duplicate( value_t& dst, const value_t& src ); void value_duplicate( value_t& dst, const value_t& src );
// For all numeric types this function set's the type of 'value_ref' to the type of cfg.
// For all other types sets the type of 'value_ref' to kCfgTFl and stores 'cfg' to value_ref.u.cfg;
rc_t value_from_cfg( const object_t* cfg, value_t& value_ref );
// Assigns src to dst. If dst has a value type then src is converted to this type.
// If the conversion is not possible then the function fail.s
rc_t value_from_value( const value_t& src, value_t& dst );
void value_print( const value_t* value, bool info_fl=false); void value_print( const value_t* value, bool info_fl=false);
rc_t value_get( const value_t* val, bool& valRef ); rc_t value_get( const value_t* val, bool& valRef );
@ -255,6 +238,7 @@ namespace cw
rc_t value_set( value_t* val, const object_t* v ); rc_t value_set( value_t* val, const object_t* v );
rc_t value_get( const value_t* val, midi::ch_msg_t*& valRef ); rc_t value_get( const value_t* val, midi::ch_msg_t*& valRef );
rc_t value_get( const value_t* val, const midi::ch_msg_t*& valRef );
rc_t value_set( value_t* val, midi::ch_msg_t* v ); rc_t value_set( value_t* val, midi::ch_msg_t* v );
@ -264,30 +248,76 @@ namespace cw
// Record // Record
// //
typedef struct recd_field_str
{
bool group_fl; // set if this field record is a group
char* label; // field or group label
value_t value; // default value for this field
char* doc; // documentation field for this field
union
{
unsigned index; // index into recd_t.valA of the value associated with this field
struct recd_field_str* group_fieldL;
} u;
struct recd_field_str* link;
} recd_field_t;
typedef struct recd_type_str
{
recd_field_t* fieldL; // linked list of field spec's
unsigned fieldN; // length of fieldL list (fieldN + base->fieldN) is total field count
const struct recd_type_str* base; // base recd type that this field inherits from
} recd_type_t;
typedef struct recd_fmt_str
{
unsigned alloc_cnt; // count of records to pre-allocate
recd_type_t* recd_type; // record type for this variable
} recd_fmt_t;
typedef struct recd_array_str typedef struct recd_array_str
{ {
recd_type_t* type; recd_type_t* type; // recd_type_t of this record array
value_t* valA; // valA[ allocRecdN * type->fieldN ] value_t* valA; // valA[ allocRecdN * type->fieldN ]
recd_t* recdA; struct recd_str* recdA; // recdA[ allocRecdN ]
unsigned allocRecdN; unsigned allocRecdN; //
} recd_array_t; } recd_array_t;
rc_t recd_type_create( recd_type_t*& recd_type_ref, const recd_type_t* base_type, const char* fields_string=nullptr ); typedef struct recd_str
void recd_type_destroy( recd_type_t*& recd_type ); {
struct value_str* valA; // varA[ recd_type_t.fieldN ] array of field values
const struct recd_str* base; // Pointer to the records inherited fields.
} recd_t;
rc_t recd_type_add_value_fields( recd_type_t* recd_type, const char* field_labels );
rc_t recd_type_add_group( recd_type_t* recd_type, const char* group_label, const char* field_labels ); // Create/destroy a recd_format_t object.
rc_t recd_format_create( recd_fmt_t*& recd_fmt_ref, const object_t* cfg );
void recd_format_destroy( recd_fmt_t*& recd_fmt_ref );
// Create a recd_type_t instance from a cfg. description.
rc_t recd_type_create( recd_type_t*& recd_type_ref, const recd_type_t* base_type, const object_t* cfg );
void recd_type_destroy( recd_type_t*& recd_type );
// Count of fields combined local and base record types. // Count of fields combined local and base record types.
rc_t recd_type_max_field_count( const recd_type_t* recd_type ); rc_t recd_type_max_field_count( const recd_type_t* recd_type );
// use '.' notation to separate groups from fields. // Get the field index associated with a named field.
// Use '.' notation to separate groups from fields.
// Note if this is a 'local' field then the high bit in the returned index will be set. // Note if this is a 'local' field then the high bit in the returned index will be set.
unsigned recd_type_field_index( const recd_type_t* recd_type, const char* field_label); unsigned recd_type_field_index( const recd_type_t* recd_type, const char* field_label);
// Given a field index return the field label.
const char* recd_type_field_index_to_label( const recd_type_t* recd_type, unsigned field_idx );
// Set the record base pointer and the value of all fields with default values.
rc_t recd_init( const recd_type_t* recd_type, const recd_t* base, recd_t* r );
// Print the recd_type info. to the console.
void recd_type_print( const recd_type_t* recd_type ); void recd_type_print( const recd_type_t* recd_type );
// Read the value from a single record field
template< typename T > template< typename T >
rc_t recd_get( const recd_type_t* type, const recd_t* recd, unsigned field_idx, T& val_ref ) rc_t recd_get( const recd_type_t* type, const recd_t* recd, unsigned field_idx, T& val_ref )
{ {
@ -297,6 +327,27 @@ namespace cw
return recd_get( type->base, recd->base, field_idx - type->fieldN, val_ref ); return recd_get( type->base, recd->base, field_idx - type->fieldN, val_ref );
} }
inline rc_t _recd_get(const recd_type_t* recd_type, recd_t* r ) { return kOkRC; }
template< typename T1, typename... ARGS >
rc_t _recd_get( const recd_type_t* recd_type, const recd_t* recd, unsigned field_idx, T1& val, ARGS&&... args )
{
rc_t rc = kOkRC;
if((rc = recd_get(recd_type,recd,field_idx,val)) != kOkRC )
return rc;
return _recd_get(recd_type,recd,std::forward<ARGS>(args)...);
}
// Read the value of multiple record fields.
template< typename T1, typename... ARGS >
rc_t recd_get( const recd_type_t* recd_type, const recd_t* recd, unsigned field_idx, T1& val, ARGS&&... args )
{
return _recd_get(recd_type,recd,field_idx,val,args...);
}
// Set the base record pointer for a record with an inherited base
inline rc_t recd_set_base( const recd_type_t* type, recd_t* recd, const recd_t* base ) inline rc_t recd_set_base( const recd_type_t* type, recd_t* recd, const recd_t* base )
{ {
// if we are setting base then the type must have a base type // if we are setting base then the type must have a base type
@ -318,9 +369,50 @@ namespace cw
return value_set( recd->valA + field_idx, val ); return value_set( recd->valA + field_idx, val );
} }
inline rc_t _recd_set( const recd_type_t* recd_type, recd_t* recd ) { return kOkRC; }
template< typename T1, typename... ARGS >
rc_t _recd_set( const recd_type_t* recd_type, recd_t* recd, unsigned field_idx, const T1& val, ARGS&&... args )
{
rc_t rc = kOkRC;
if( field_idx >= recd_type->fieldN )
return cwLogError(kInvalidArgRC,"Fields in the inherited record may not be set.");
if((rc = value_set( recd->valA + field_idx, val)) != kOkRC )
{
rc = cwLogError(rc,"Field set failed on '%s'.", cwStringNullGuard(recd_type_field_index_to_label( recd_type, field_idx )));
goto errLabel;
}
return _recd_set(recd_type,recd,std::forward<ARGS>(args)...);
errLabel:
return rc;
}
// Set multiple fields of a record.
template< typename T1, typename... ARGS >
rc_t recd_set( const recd_type_t* recd_type, const recd_t* base, recd_t* recd, unsigned field_idx, const T1& val, ARGS&&... args )
{
rc_t rc = kOkRC;
if((rc = recd_init( recd_type, base, recd )) != kOkRC )
goto errLabel;
if((rc = _recd_set(recd_type,recd,field_idx,val,args...)) != kOkRC )
goto errLabel;
errLabel:
return rc;
}
// Print a record to the console.
rc_t recd_print( const recd_type_t* recd_type, const recd_t* r ); rc_t recd_print( const recd_type_t* recd_type, const recd_t* r );
rc_t recd_array_create( recd_array_t*& recd_array_ref, recd_type_t* recd_type, unsigned allocRecdN ); // Create/destroy a buffer of records.
rc_t recd_array_create( recd_array_t*& recd_array_ref, recd_type_t* recd_type, const recd_type_t* base, unsigned allocRecdN );
rc_t recd_array_destroy( recd_array_t*& recd_array_ref ); rc_t recd_array_destroy( recd_array_t*& recd_array_ref );
rc_t value_test( const test::test_args_t& args ); rc_t value_test( const test::test_args_t& args );