2015-02-26 15:02:31 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <iostream>
|
2016-05-16 22:14:50 +00:00
|
|
|
#include <udunits2.h>
|
2022-08-23 18:47:56 +00:00
|
|
|
#include <math.h> // for fpclassify
|
|
|
|
#include <iomanip> // for setprecision
|
|
|
|
#include <string.h>
|
|
|
|
#include <sstream>
|
|
|
|
|
|
|
|
#include "trick/VariableReference.hh"
|
2015-06-01 15:28:29 +00:00
|
|
|
#include "trick/memorymanager_c_intf.h"
|
|
|
|
#include "trick/wcs_ext.h"
|
2022-08-23 18:47:56 +00:00
|
|
|
#include "trick/map_trick_units_to_udunits.hh"
|
2015-06-01 15:28:29 +00:00
|
|
|
#include "trick/message_proto.h"
|
|
|
|
#include "trick/message_type.h"
|
2022-08-23 18:47:56 +00:00
|
|
|
#include "trick/UdUnits.hh"
|
|
|
|
#include "trick/bitfield_proto.h"
|
|
|
|
#include "trick/trick_byteswap.h"
|
|
|
|
|
|
|
|
|
|
|
|
// Static variables to be addresses that are known to be the error ref address
|
2023-03-20 22:53:01 +00:00
|
|
|
int Trick::VariableReference::_bad_ref_int = 0 ;
|
|
|
|
int Trick::VariableReference::_do_not_resolve_bad_ref_int = 0 ;
|
2022-08-23 18:47:56 +00:00
|
|
|
|
|
|
|
REF2* Trick::VariableReference::make_error_ref(std::string in_name) {
|
|
|
|
REF2* new_ref;
|
|
|
|
new_ref = (REF2*)calloc(1, sizeof(REF2));
|
|
|
|
new_ref->reference = strdup(in_name.c_str()) ;
|
|
|
|
new_ref->units = NULL ;
|
2023-03-20 22:53:01 +00:00
|
|
|
new_ref->address = (char *)&_bad_ref_int ;
|
2022-08-23 18:47:56 +00:00
|
|
|
new_ref->attr = (ATTRIBUTES*)calloc(1, sizeof(ATTRIBUTES)) ;
|
|
|
|
new_ref->attr->type = TRICK_NUMBER_OF_TYPES ;
|
|
|
|
new_ref->attr->units = (char *)"--" ;
|
|
|
|
new_ref->attr->size = sizeof(int) ;
|
|
|
|
return new_ref;
|
|
|
|
}
|
|
|
|
|
|
|
|
REF2* Trick::VariableReference::make_do_not_resolve_ref(std::string in_name) {
|
|
|
|
REF2* new_ref;
|
|
|
|
new_ref = (REF2*)calloc(1, sizeof(REF2));
|
|
|
|
new_ref->reference = strdup(in_name.c_str()) ;
|
|
|
|
new_ref->units = NULL ;
|
2023-03-20 22:53:01 +00:00
|
|
|
new_ref->address = (char *)&_do_not_resolve_bad_ref_int ;
|
2022-08-23 18:47:56 +00:00
|
|
|
new_ref->attr = (ATTRIBUTES*)calloc(1, sizeof(ATTRIBUTES)) ;
|
|
|
|
new_ref->attr->type = TRICK_NUMBER_OF_TYPES ;
|
|
|
|
new_ref->attr->units = (char *)"--" ;
|
|
|
|
new_ref->attr->size = sizeof(int) ;
|
|
|
|
return new_ref;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Helper function to deal with time variable
|
|
|
|
REF2* make_time_ref(double * time) {
|
|
|
|
REF2* new_ref;
|
|
|
|
new_ref = (REF2*)calloc(1, sizeof(REF2));
|
|
|
|
new_ref->reference = strdup("time") ;
|
|
|
|
new_ref->units = strdup("s") ;
|
|
|
|
new_ref->address = (char *)time ;
|
|
|
|
new_ref->attr = (ATTRIBUTES*)calloc(1, sizeof(ATTRIBUTES)) ;
|
|
|
|
new_ref->attr->type = TRICK_DOUBLE ;
|
|
|
|
new_ref->attr->units = strdup("s") ;
|
|
|
|
new_ref->attr->size = sizeof(double) ;
|
|
|
|
return new_ref;
|
|
|
|
}
|
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
Trick::VariableReference::VariableReference(std::string var_name, double* time) : _staged(false), _write_ready(false) {
|
2022-08-23 18:47:56 +00:00
|
|
|
if (var_name != "time") {
|
|
|
|
ASSERT(0);
|
|
|
|
}
|
2015-02-26 15:02:31 +00:00
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
_var_info = make_time_ref(time);
|
2015-02-26 15:02:31 +00:00
|
|
|
|
2022-08-23 18:47:56 +00:00
|
|
|
// Set up member variables
|
2023-03-20 22:53:01 +00:00
|
|
|
_address = _var_info->address;
|
|
|
|
_size = _var_info->attr->size ;
|
|
|
|
_deref = false;
|
2015-02-26 15:02:31 +00:00
|
|
|
|
2022-08-23 18:47:56 +00:00
|
|
|
// Deal with weirdness around string vs wstring
|
2023-03-20 22:53:01 +00:00
|
|
|
_trick_type = _var_info->attr->type ;
|
2015-02-26 15:02:31 +00:00
|
|
|
|
2022-08-23 18:47:56 +00:00
|
|
|
// Allocate stage and write buffers
|
2023-03-20 22:53:01 +00:00
|
|
|
_stage_buffer = calloc(_size, 1) ;
|
|
|
|
_write_buffer = calloc(_size, 1) ;
|
2016-05-16 22:14:50 +00:00
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
_conversion_factor = cv_get_trivial();
|
|
|
|
_base_units = _var_info->attr->units;
|
|
|
|
_requested_units = "s";
|
|
|
|
_name = _var_info->reference;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
Trick::VariableReference::VariableReference(std::string var_name) : _staged(false), _write_ready(false) {
|
2022-08-23 18:47:56 +00:00
|
|
|
|
|
|
|
if (var_name == "time") {
|
|
|
|
ASSERT(0);
|
|
|
|
} else {
|
|
|
|
// get variable attributes from memory manager
|
2023-03-20 22:53:01 +00:00
|
|
|
_var_info = ref_attributes(var_name.c_str());
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Handle error cases
|
2023-03-20 22:53:01 +00:00
|
|
|
if ( _var_info == NULL ) {
|
2022-08-23 18:47:56 +00:00
|
|
|
// TODO: ERROR LOGGER sendErrorMessage("Variable Server could not find variable %s.\n", var_name);
|
|
|
|
// PRINTF IS NOT AN ERROR LOGGER @me
|
2023-04-18 21:27:32 +00:00
|
|
|
message_publish(MSG_ERROR, "Variable Server could not find variable %s.\n", var_name.c_str());
|
2023-03-20 22:53:01 +00:00
|
|
|
_var_info = make_error_ref(var_name);
|
|
|
|
} else if ( _var_info->attr ) {
|
|
|
|
if ( _var_info->attr->type == TRICK_STRUCTURED ) {
|
2022-08-23 18:47:56 +00:00
|
|
|
// sendErrorMessage("Variable Server: var_add cant add \"%s\" because its a composite variable.\n", var_name);
|
2023-04-18 21:27:32 +00:00
|
|
|
message_publish(MSG_ERROR, "Variable Server: var_add cant add \"%s\" because its a composite variable.\n", var_name.c_str());
|
2015-02-26 15:02:31 +00:00
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
free(_var_info);
|
|
|
|
_var_info = make_do_not_resolve_ref(var_name);
|
2022-08-23 18:47:56 +00:00
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
} else if ( _var_info->attr->type == TRICK_STL ) {
|
2022-08-23 18:47:56 +00:00
|
|
|
// sendErrorMessage("Variable Server: var_add cant add \"%s\" because its an STL variable.\n", var_name);
|
2023-04-18 21:27:32 +00:00
|
|
|
message_publish(MSG_ERROR,"Variable Server: var_add cant add \"%s\" because its an STL variable.\n", var_name.c_str());
|
2022-08-23 18:47:56 +00:00
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
free(_var_info);
|
|
|
|
_var_info = make_do_not_resolve_ref(var_name);
|
2015-02-26 15:02:31 +00:00
|
|
|
}
|
|
|
|
} else {
|
2022-08-23 18:47:56 +00:00
|
|
|
// sendErrorMessage("Variable Server: BAD MOJO - Missing ATTRIBUTES.");
|
2023-04-18 21:27:32 +00:00
|
|
|
message_publish(MSG_ERROR, "Variable Server: BAD MOJO - Missing ATTRIBUTES.");
|
2022-08-23 18:47:56 +00:00
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
free(_var_info);
|
|
|
|
_var_info = make_error_ref(var_name);
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Set up member variables
|
2023-03-20 22:53:01 +00:00
|
|
|
_var_info->units = NULL;
|
|
|
|
_address = _var_info->address;
|
|
|
|
_size = _var_info->attr->size ;
|
|
|
|
_deref = false;
|
2022-08-23 18:47:56 +00:00
|
|
|
|
|
|
|
// Deal with weirdness around string vs wstring
|
2023-03-20 22:53:01 +00:00
|
|
|
_trick_type = _var_info->attr->type ;
|
2022-08-23 18:47:56 +00:00
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
if ( _var_info->num_index == _var_info->attr->num_index ) {
|
2022-08-23 18:47:56 +00:00
|
|
|
// single value - nothing else necessary
|
2023-03-20 22:53:01 +00:00
|
|
|
} else if ( _var_info->attr->index[_var_info->attr->num_index - 1].size != 0 ) {
|
2022-08-23 18:47:56 +00:00
|
|
|
// Constrained array
|
2023-03-20 22:53:01 +00:00
|
|
|
for ( int i = _var_info->attr->num_index-1; i > _var_info->num_index-1 ; i-- ) {
|
|
|
|
_size *= _var_info->attr->index[i].size ;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Unconstrained array
|
2023-03-20 22:53:01 +00:00
|
|
|
if ((_var_info->attr->num_index - _var_info->num_index) > 1 ) {
|
2023-04-18 21:27:32 +00:00
|
|
|
message_publish(MSG_ERROR, "Variable Server Error: var_add(%s) requests more than one dimension of dynamic array.\n", _var_info->reference);
|
|
|
|
message_publish(MSG_ERROR, "Data is not contiguous so returned values are unpredictable.\n") ;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
2023-03-20 22:53:01 +00:00
|
|
|
if ( _var_info->attr->type == TRICK_CHARACTER ) {
|
|
|
|
_trick_type = TRICK_STRING ;
|
|
|
|
_deref = true;
|
|
|
|
} else if ( _var_info->attr->type == TRICK_WCHAR ) {
|
|
|
|
_trick_type = TRICK_WSTRING ;
|
|
|
|
_deref = true;
|
2015-02-26 15:02:31 +00:00
|
|
|
} else {
|
2023-03-20 22:53:01 +00:00
|
|
|
_deref = true ;
|
|
|
|
_size *= get_size((char*)_address) ;
|
2015-02-26 15:02:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// handle strings: set a max buffer size, the copy size may vary so will be set in copy_sim_data
|
2023-03-20 22:53:01 +00:00
|
|
|
if (( _trick_type == TRICK_STRING ) || ( _trick_type == TRICK_WSTRING )) {
|
|
|
|
_size = MAX_ARRAY_LENGTH ;
|
2015-02-26 15:02:31 +00:00
|
|
|
}
|
|
|
|
|
2022-08-23 18:47:56 +00:00
|
|
|
// Allocate stage and write buffers
|
2023-03-20 22:53:01 +00:00
|
|
|
_stage_buffer = calloc(_size, 1) ;
|
|
|
|
_write_buffer = calloc(_size, 1) ;
|
2015-02-26 15:02:31 +00:00
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
_conversion_factor = cv_get_trivial();
|
|
|
|
_base_units = _var_info->attr->units;
|
|
|
|
_requested_units = "";
|
|
|
|
_name = _var_info->reference;
|
2015-02-26 15:02:31 +00:00
|
|
|
|
2022-08-23 18:47:56 +00:00
|
|
|
// Done!
|
2015-02-26 15:02:31 +00:00
|
|
|
}
|
|
|
|
|
2022-08-23 18:47:56 +00:00
|
|
|
Trick::VariableReference::~VariableReference() {
|
2023-03-20 22:53:01 +00:00
|
|
|
if (_var_info != NULL) {
|
|
|
|
free( _var_info );
|
|
|
|
_var_info = NULL;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
2023-03-20 22:53:01 +00:00
|
|
|
if (_stage_buffer != NULL) {
|
|
|
|
free (_stage_buffer);
|
|
|
|
_stage_buffer = NULL;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
2023-03-20 22:53:01 +00:00
|
|
|
if (_write_buffer != NULL) {
|
|
|
|
free (_write_buffer);
|
|
|
|
_write_buffer = NULL;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
2023-03-20 22:53:01 +00:00
|
|
|
if (_conversion_factor != NULL) {
|
|
|
|
cv_free(_conversion_factor);
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
}
|
2019-02-18 23:11:41 +00:00
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
std::string Trick::VariableReference::getName() const {
|
|
|
|
return _name;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int Trick::VariableReference::getSizeBinary() const {
|
2023-03-20 22:53:01 +00:00
|
|
|
return _size;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TRICK_TYPE Trick::VariableReference::getType() const {
|
2023-03-20 22:53:01 +00:00
|
|
|
return _trick_type;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
std::string Trick::VariableReference::getBaseUnits() const {
|
|
|
|
return _base_units;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int Trick::VariableReference::setRequestedUnits(std::string units_name) {
|
|
|
|
// Some error logging lambdas - these should probably go somewhere else
|
|
|
|
// But I do kinda like them
|
|
|
|
auto publish = [](MESSAGE_TYPE type, const std::string& message) {
|
|
|
|
std::ostringstream oss;
|
|
|
|
oss << "Variable Server: " << message << std::endl;
|
|
|
|
message_publish(type, oss.str().c_str());
|
|
|
|
};
|
|
|
|
|
|
|
|
auto publishError = [&](const std::string& units) {
|
|
|
|
std::ostringstream oss;
|
|
|
|
oss << "units error for [" << getName() << "] [" << units << "]";
|
|
|
|
publish(MSG_ERROR, oss.str());
|
|
|
|
};
|
|
|
|
|
|
|
|
// If the units_name parameter is "xx", set it to the current units.
|
|
|
|
if (!units_name.compare("xx")) {
|
|
|
|
units_name = getBaseUnits();
|
2019-02-18 23:11:41 +00:00
|
|
|
}
|
2022-08-23 18:47:56 +00:00
|
|
|
|
|
|
|
// if unitless ('--') then do not convert to udunits
|
|
|
|
if(units_name.compare("--")) {
|
|
|
|
// Check to see if this is an old style Trick unit that needs to be converted to new udunits
|
|
|
|
std::string new_units = map_trick_units_to_udunits(units_name) ;
|
|
|
|
// Warn if a conversion has taken place
|
|
|
|
if ( units_name.compare(new_units) ) {
|
|
|
|
// TODO: MAKE BETTER SYSTEM FOR ERROR LOGGING
|
|
|
|
std::ostringstream oss;
|
|
|
|
oss << "[" << getName() << "] old-style units converted from ["
|
|
|
|
<< units_name << "] to [" << new_units << "]";
|
|
|
|
publish(MSG_WARNING, oss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Interpret base unit
|
2023-03-20 22:53:01 +00:00
|
|
|
ut_unit * from = ut_parse(Trick::UdUnits::get_u_system(), getBaseUnits().c_str(), UT_ASCII) ;
|
2022-08-23 18:47:56 +00:00
|
|
|
if ( !from ) {
|
|
|
|
std::cout << "Error in interpreting base units" << std::endl;
|
|
|
|
publishError(getBaseUnits());
|
|
|
|
ut_free(from) ;
|
|
|
|
return -1 ;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Interpret requested unit
|
|
|
|
ut_unit * to = ut_parse(Trick::UdUnits::get_u_system(), new_units.c_str(), UT_ASCII) ;
|
|
|
|
if ( !to ) {
|
|
|
|
std::cout << "Error in interpreting requested units" << std::endl;
|
|
|
|
publishError(new_units);
|
|
|
|
ut_free(from) ;
|
|
|
|
ut_free(to) ;
|
|
|
|
return -1 ;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a converter from the base to the requested
|
|
|
|
auto new_conversion_factor = ut_get_converter(from, to) ;
|
|
|
|
ut_free(from) ;
|
|
|
|
ut_free(to) ;
|
|
|
|
if ( !new_conversion_factor ) {
|
|
|
|
std::ostringstream oss;
|
|
|
|
oss << "[" << getName() << "] cannot convert units from [" << getBaseUnits()
|
|
|
|
<< "] to [" << new_units << "]";
|
|
|
|
publish(MSG_ERROR, oss.str());
|
|
|
|
return -1 ;
|
|
|
|
} else {
|
2023-03-20 22:53:01 +00:00
|
|
|
_conversion_factor = new_conversion_factor;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Set the requested units. This will cause the unit string to be printed in write_value_ascii
|
2023-03-20 22:53:01 +00:00
|
|
|
_requested_units = new_units;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
return 0;
|
2019-02-18 23:11:41 +00:00
|
|
|
}
|
|
|
|
|
2022-08-23 18:47:56 +00:00
|
|
|
int Trick::VariableReference::stageValue(bool validate_address) {
|
2023-03-20 22:53:01 +00:00
|
|
|
_write_ready = false;
|
2022-08-23 18:47:56 +00:00
|
|
|
|
|
|
|
// Copy <size> bytes from <address> to staging_point.
|
|
|
|
|
|
|
|
// Try to recreate connection if it has been broken
|
2023-03-20 22:53:01 +00:00
|
|
|
if (_var_info->address == &_bad_ref_int) {
|
|
|
|
REF2 *new_ref = ref_attributes(_var_info->reference);
|
2022-08-23 18:47:56 +00:00
|
|
|
if (new_ref != NULL) {
|
2023-03-20 22:53:01 +00:00
|
|
|
_var_info = new_ref;
|
|
|
|
_address = _var_info->address;
|
|
|
|
// _requested_units = "";
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// if there's a pointer somewhere in the address path, follow it in case pointer changed
|
2023-03-20 22:53:01 +00:00
|
|
|
if ( _var_info->pointer_present == 1 ) {
|
|
|
|
_address = follow_address_path(_var_info) ;
|
|
|
|
if (_address == NULL) {
|
2022-08-23 18:47:56 +00:00
|
|
|
tagAsInvalid();
|
|
|
|
} else if ( validate_address ) {
|
|
|
|
validate();
|
|
|
|
} else {
|
2023-03-20 22:53:01 +00:00
|
|
|
_var_info->address = _address ;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// if this variable is a string we need to get the raw character string out of it.
|
2023-03-20 22:53:01 +00:00
|
|
|
if (( _trick_type == TRICK_STRING ) && !_deref) {
|
|
|
|
std::string * str_ptr = (std::string *)_var_info->address ;
|
2022-08-23 18:47:56 +00:00
|
|
|
// Get a pointer to the internal character array
|
2023-03-20 22:53:01 +00:00
|
|
|
_address = (void *)(str_ptr->c_str()) ;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// if this variable itself is a pointer, dereference it
|
2023-03-20 22:53:01 +00:00
|
|
|
if ( _deref ) {
|
|
|
|
_address = *(void**)_var_info->address ;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// handle c++ string and char*
|
2023-03-20 22:53:01 +00:00
|
|
|
if ( _trick_type == TRICK_STRING ) {
|
|
|
|
if (_address == NULL) {
|
|
|
|
_size = 0 ;
|
2022-08-23 18:47:56 +00:00
|
|
|
} else {
|
2023-03-20 22:53:01 +00:00
|
|
|
_size = strlen((char*)_address) + 1 ;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// handle c++ wstring and wchar_t*
|
2023-03-20 22:53:01 +00:00
|
|
|
if ( _trick_type == TRICK_WSTRING ) {
|
|
|
|
if (_address == NULL) {
|
|
|
|
_size = 0 ;
|
2022-08-23 18:47:56 +00:00
|
|
|
} else {
|
2023-03-20 22:53:01 +00:00
|
|
|
_size = wcslen((wchar_t *)_address) * sizeof(wchar_t);
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
}
|
2023-03-20 22:53:01 +00:00
|
|
|
if(_address != NULL) {
|
|
|
|
memcpy( _stage_buffer , _address , _size ) ;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
_staged = true;
|
2022-08-23 18:47:56 +00:00
|
|
|
return 0;
|
2015-02-26 15:02:31 +00:00
|
|
|
}
|
2022-08-23 18:47:56 +00:00
|
|
|
|
|
|
|
bool Trick::VariableReference::validate() {
|
|
|
|
// The address is not NULL.
|
|
|
|
// Should be called by VariableServer Session if validateAddress is on.
|
|
|
|
// check the memory manager if the address falls into
|
|
|
|
// any of the memory blocks it knows of. Don't do this if we have a std::string or
|
|
|
|
// wstring type, or we already are pointing to a bad ref.
|
2023-03-20 22:53:01 +00:00
|
|
|
if ( (_trick_type != TRICK_STRING) and
|
|
|
|
(_trick_type != TRICK_WSTRING) and
|
|
|
|
(_var_info->address != &_bad_ref_int) and
|
|
|
|
(get_alloc_info_of(_address) == NULL) ) {
|
2022-08-23 18:47:56 +00:00
|
|
|
|
|
|
|
// This variable is broken, make it into an error ref
|
|
|
|
tagAsInvalid();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Everything is fine
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void write_escaped_string( std::ostream& os, const char* s) {
|
|
|
|
for (int ii=0 ; ii<strlen(s) ; ii++) {
|
|
|
|
if (isprint(s[ii])) {
|
|
|
|
os << s[ii];
|
|
|
|
} else {
|
|
|
|
switch ((s)[ii]) {
|
|
|
|
case '\n': os << "\\n"; break;
|
|
|
|
case '\t': os << "\\t"; break;
|
|
|
|
case '\b': os << "\\b"; break;
|
|
|
|
case '\a': os << "\\a"; break;
|
|
|
|
case '\f': os << "\\f"; break;
|
|
|
|
case '\r': os << "\\n"; break;
|
|
|
|
case '\v': os << "\\v"; break;
|
|
|
|
case '\"': os << "\\\""; break;
|
|
|
|
default : {
|
|
|
|
// Replicating behavior from original vs_format_ascii
|
|
|
|
char temp_s[6];
|
|
|
|
sprintf(temp_s, "\\x%02x", s[ii]);
|
|
|
|
os << temp_s ;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int Trick::VariableReference::getSizeAscii() const {
|
|
|
|
std::stringstream ss;
|
|
|
|
writeValueAscii(ss);
|
|
|
|
return ss.str().length();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Trick::VariableReference::writeValueAscii( std::ostream& out ) const {
|
|
|
|
// This is copied and modified from vs_format_ascii
|
|
|
|
|
|
|
|
if (!isWriteReady()) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int bytes_written = 0;
|
2023-03-20 22:53:01 +00:00
|
|
|
void * buf_ptr = _write_buffer ;
|
|
|
|
while (bytes_written < _size) {
|
|
|
|
bytes_written += _var_info->attr->size ;
|
2022-08-23 18:47:56 +00:00
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
switch (_trick_type) {
|
2022-08-23 18:47:56 +00:00
|
|
|
|
|
|
|
case TRICK_CHARACTER:
|
2023-03-20 22:53:01 +00:00
|
|
|
if (_var_info->attr->num_index == _var_info->num_index) {
|
2022-08-23 18:47:56 +00:00
|
|
|
// Single char
|
2023-03-20 22:53:01 +00:00
|
|
|
out << (int)cv_convert_double(_conversion_factor, *(char *)buf_ptr);
|
2022-08-23 18:47:56 +00:00
|
|
|
} else {
|
|
|
|
// All but last dim specified, leaves a char array
|
|
|
|
write_escaped_string(out, (const char *) buf_ptr);
|
2023-03-20 22:53:01 +00:00
|
|
|
bytes_written = _size ;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TRICK_UNSIGNED_CHARACTER:
|
2023-03-20 22:53:01 +00:00
|
|
|
if (_var_info->attr->num_index == _var_info->num_index) {
|
2022-08-23 18:47:56 +00:00
|
|
|
// Single char
|
2023-03-20 22:53:01 +00:00
|
|
|
out << (unsigned int)cv_convert_double(_conversion_factor,*(unsigned char *)buf_ptr);
|
2022-08-23 18:47:56 +00:00
|
|
|
} else {
|
|
|
|
// All but last dim specified, leaves a char array
|
|
|
|
write_escaped_string(out, (const char *) buf_ptr);
|
2023-03-20 22:53:01 +00:00
|
|
|
bytes_written = _size ;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TRICK_WCHAR:{
|
2023-03-20 22:53:01 +00:00
|
|
|
if (_var_info->attr->num_index == _var_info->num_index) {
|
2022-08-23 18:47:56 +00:00
|
|
|
out << *(wchar_t *) buf_ptr;
|
|
|
|
} else {
|
|
|
|
// convert wide char string char string
|
|
|
|
size_t len = wcs_to_ncs_len((wchar_t *)buf_ptr) + 1 ;
|
|
|
|
|
|
|
|
char temp_buf[len];
|
|
|
|
wcs_to_ncs((wchar_t *) buf_ptr, temp_buf, len);
|
|
|
|
out << temp_buf;
|
2023-03-20 22:53:01 +00:00
|
|
|
bytes_written = _size ;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TRICK_STRING:
|
|
|
|
if ((char *) buf_ptr != NULL) {
|
|
|
|
write_escaped_string(out, (const char *) buf_ptr);
|
2023-03-20 22:53:01 +00:00
|
|
|
bytes_written = _size ;
|
2022-08-23 18:47:56 +00:00
|
|
|
} else {
|
|
|
|
out << '\0';
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TRICK_WSTRING:
|
|
|
|
if ((wchar_t *) buf_ptr != NULL) {
|
|
|
|
// convert wide char string char string
|
|
|
|
size_t len = wcs_to_ncs_len( (wchar_t *)buf_ptr) + 1 ;
|
|
|
|
|
|
|
|
char temp_buf[len];
|
|
|
|
wcs_to_ncs( (wchar_t *) buf_ptr, temp_buf, len);
|
|
|
|
out << temp_buf;
|
2023-03-20 22:53:01 +00:00
|
|
|
bytes_written = _size ;
|
2022-08-23 18:47:56 +00:00
|
|
|
} else {
|
|
|
|
out << '\0';
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TRICK_SHORT:
|
2023-03-20 22:53:01 +00:00
|
|
|
out << (short)cv_convert_double(_conversion_factor,*(short *)buf_ptr);
|
2022-08-23 18:47:56 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TRICK_UNSIGNED_SHORT:
|
2023-03-20 22:53:01 +00:00
|
|
|
out << (unsigned short)cv_convert_double(_conversion_factor,*(unsigned short *)buf_ptr);
|
2022-08-23 18:47:56 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TRICK_INTEGER:
|
|
|
|
case TRICK_ENUMERATED:
|
2023-03-20 22:53:01 +00:00
|
|
|
out << (int)cv_convert_double(_conversion_factor,*(int *)buf_ptr);
|
2022-08-23 18:47:56 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TRICK_BOOLEAN:
|
2023-03-20 22:53:01 +00:00
|
|
|
out << (int)cv_convert_double(_conversion_factor,*(bool *)buf_ptr);
|
2022-08-23 18:47:56 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TRICK_BITFIELD:
|
2023-03-20 22:53:01 +00:00
|
|
|
out << (GET_BITFIELD(buf_ptr, _var_info->attr->size, _var_info->attr->index[0].start, _var_info->attr->index[0].size));
|
2022-08-23 18:47:56 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TRICK_UNSIGNED_BITFIELD:
|
2023-03-20 22:53:01 +00:00
|
|
|
out << (GET_UNSIGNED_BITFIELD(buf_ptr, _var_info->attr->size, _var_info->attr->index[0].start, _var_info->attr->index[0].size));
|
2022-08-23 18:47:56 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TRICK_UNSIGNED_INTEGER:
|
2023-03-20 22:53:01 +00:00
|
|
|
out << (unsigned int)cv_convert_double(_conversion_factor,*(unsigned int *)buf_ptr);
|
2022-08-23 18:47:56 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TRICK_LONG: {
|
|
|
|
long l = *(long *)buf_ptr;
|
2023-03-20 22:53:01 +00:00
|
|
|
if (_conversion_factor != cv_get_trivial()) {
|
|
|
|
l = (long)cv_convert_double(_conversion_factor, l);
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
out << l;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case TRICK_UNSIGNED_LONG: {
|
|
|
|
unsigned long ul = *(unsigned long *)buf_ptr;
|
2023-03-20 22:53:01 +00:00
|
|
|
if (_conversion_factor != cv_get_trivial()) {
|
|
|
|
ul = (unsigned long)cv_convert_double(_conversion_factor, ul);
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
out << ul;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case TRICK_FLOAT:
|
2023-03-20 22:53:01 +00:00
|
|
|
out << std::setprecision(8) << cv_convert_float(_conversion_factor,*(float *)buf_ptr);
|
2022-08-23 18:47:56 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TRICK_DOUBLE:
|
2023-03-20 22:53:01 +00:00
|
|
|
out << std::setprecision(16) << cv_convert_double(_conversion_factor,*(double *)buf_ptr);
|
2022-08-23 18:47:56 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TRICK_LONG_LONG: {
|
|
|
|
long long ll = *(long long *)buf_ptr;
|
2023-03-20 22:53:01 +00:00
|
|
|
if (_conversion_factor != cv_get_trivial()) {
|
|
|
|
ll = (long long)cv_convert_double(_conversion_factor, ll);
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
out << ll;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case TRICK_UNSIGNED_LONG_LONG: {
|
|
|
|
unsigned long long ull = *(unsigned long long *)buf_ptr;
|
2023-03-20 22:53:01 +00:00
|
|
|
if (_conversion_factor != cv_get_trivial()) {
|
|
|
|
ull = (unsigned long long)cv_convert_double(_conversion_factor, ull);
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
out << ull;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case TRICK_NUMBER_OF_TYPES:
|
|
|
|
out << "BAD_REF";
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:{
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} // end switch
|
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
if (bytes_written < _size) {
|
2022-08-23 18:47:56 +00:00
|
|
|
// if returning an array, continue array as comma separated values
|
|
|
|
out << ",";
|
2023-03-20 22:53:01 +00:00
|
|
|
buf_ptr = (void*) ((long)buf_ptr + _var_info->attr->size) ;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
} //end while
|
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
if (_requested_units != "") {
|
|
|
|
if ( _var_info->attr->mods & TRICK_MODS_UNITSDASHDASH ) {
|
2022-08-23 18:47:56 +00:00
|
|
|
out << " {--}";
|
|
|
|
} else {
|
2023-03-20 22:53:01 +00:00
|
|
|
out << " {" << _requested_units << "}";
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Trick::VariableReference::tagAsInvalid () {
|
|
|
|
std::string save_name(getName()) ;
|
2023-03-20 22:53:01 +00:00
|
|
|
free(_var_info) ;
|
|
|
|
_var_info = make_error_ref(save_name) ;
|
|
|
|
_address = _var_info->address ;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Trick::VariableReference::prepareForWrite() {
|
2023-03-20 22:53:01 +00:00
|
|
|
if (!_staged) {
|
2022-08-23 18:47:56 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
void * temp_p = _stage_buffer;
|
|
|
|
_stage_buffer = _write_buffer;
|
|
|
|
_write_buffer = temp_p;
|
2022-08-23 18:47:56 +00:00
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
_staged = false;
|
|
|
|
_write_ready = true;
|
2022-08-23 18:47:56 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Trick::VariableReference::isStaged() const {
|
2023-03-20 22:53:01 +00:00
|
|
|
return _staged;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Trick::VariableReference::isWriteReady() const {
|
2023-03-20 22:53:01 +00:00
|
|
|
return _write_ready;
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int Trick::VariableReference::writeTypeBinary( std::ostream& out, bool byteswap ) const {
|
2023-03-20 22:53:01 +00:00
|
|
|
int local_type = _trick_type;
|
2022-08-23 18:47:56 +00:00
|
|
|
if (byteswap) {
|
|
|
|
local_type = trick_byteswap_int(local_type);
|
|
|
|
}
|
|
|
|
out.write(const_cast<const char *>(reinterpret_cast<char *>(&local_type)), sizeof(int));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Trick::VariableReference::writeSizeBinary( std::ostream& out, bool byteswap ) const {
|
2023-03-20 22:53:01 +00:00
|
|
|
int local_size = _size;
|
2022-08-23 18:47:56 +00:00
|
|
|
if (byteswap) {
|
|
|
|
local_size = trick_byteswap_int(local_size);
|
|
|
|
}
|
|
|
|
out.write(const_cast<const char *>(reinterpret_cast<char *>(&local_size)), sizeof(int));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Trick::VariableReference::writeNameBinary( std::ostream& out, bool byteswap ) const {
|
2023-03-20 22:53:01 +00:00
|
|
|
std::string name = getName();
|
2023-04-18 21:27:32 +00:00
|
|
|
out.write(name.c_str(), name.size());
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2022-08-23 18:47:56 +00:00
|
|
|
|
2023-04-18 21:27:32 +00:00
|
|
|
int Trick::VariableReference::writeNameLengthBinary( std::ostream& out, bool byteswap ) const {
|
|
|
|
int name_size = getName().size();
|
2022-08-23 18:47:56 +00:00
|
|
|
if (byteswap) {
|
|
|
|
name_size = trick_byteswap_int(name_size);
|
|
|
|
}
|
|
|
|
|
|
|
|
out.write(const_cast<const char *>(reinterpret_cast<char *>(&name_size)), sizeof(int));
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2023-04-18 21:27:32 +00:00
|
|
|
|
2022-08-23 18:47:56 +00:00
|
|
|
void Trick::VariableReference::byteswap_var (char * out, char * in) const {
|
|
|
|
byteswap_var(out, in, *this);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Trick::VariableReference::byteswap_var (char * out, char * in, const VariableReference& ref) {
|
2023-03-20 22:53:01 +00:00
|
|
|
ATTRIBUTES * attr = ref._var_info->attr;
|
2022-08-23 18:47:56 +00:00
|
|
|
int array_size = 1;
|
|
|
|
|
|
|
|
// Determine how many elements are in this array if it is an array
|
2023-03-20 22:53:01 +00:00
|
|
|
for (int j = 0; j < ref._var_info->attr->num_index; j++) {
|
2022-08-23 18:47:56 +00:00
|
|
|
array_size *= attr->index[j].size;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (attr->size) {
|
|
|
|
case 1:
|
|
|
|
// If these are just characters, no need to byteswap
|
|
|
|
for (int j = 0; j < array_size; j++) {
|
|
|
|
out[j] = in[j];
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 2: {
|
|
|
|
short * short_in = reinterpret_cast<short *> (in);
|
|
|
|
short * short_out = reinterpret_cast<short *> (out);
|
|
|
|
|
|
|
|
for (int j = 0; j < array_size; j++) {
|
|
|
|
short_out[j] = trick_byteswap_short(short_in[j]);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case 4: {
|
|
|
|
int * int_in = reinterpret_cast<int *> (in);
|
|
|
|
int * int_out = reinterpret_cast<int *> (out);
|
|
|
|
|
|
|
|
for (int j = 0; j < array_size; j++) {
|
|
|
|
int_out[j] = trick_byteswap_int(int_in[j]);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 8: {
|
|
|
|
// We don't actually care if this is double or long, just that it's the right size
|
|
|
|
double * double_in = reinterpret_cast<double *> (in);
|
|
|
|
double * double_out = reinterpret_cast<double *> (out);
|
|
|
|
|
|
|
|
for (int j = 0; j < array_size; j++) {
|
|
|
|
double_out[j] = trick_byteswap_double(double_in[j]);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int Trick::VariableReference::writeValueBinary( std::ostream& out, bool byteswap ) const {
|
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
if ( _trick_type == TRICK_BITFIELD ) {
|
|
|
|
int temp_i = GET_BITFIELD(_write_buffer , _var_info->attr->size ,
|
|
|
|
_var_info->attr->index[0].start, _var_info->attr->index[0].size) ;
|
|
|
|
out.write((char *)(&temp_i), _size);
|
|
|
|
return _size;
|
|
|
|
}
|
2022-08-23 18:47:56 +00:00
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
if ( _trick_type == TRICK_UNSIGNED_BITFIELD ) {
|
|
|
|
int temp_unsigned = GET_UNSIGNED_BITFIELD(_write_buffer , _var_info->attr->size ,
|
|
|
|
_var_info->attr->index[0].start, _var_info->attr->index[0].size) ;
|
|
|
|
out.write((char *)(&temp_unsigned), _size);
|
|
|
|
return _size;
|
|
|
|
}
|
2022-08-23 18:47:56 +00:00
|
|
|
|
2023-03-20 22:53:01 +00:00
|
|
|
if (_trick_type == TRICK_NUMBER_OF_TYPES) {
|
|
|
|
// TRICK_NUMBER_OF_TYPES is an error case
|
|
|
|
int temp_zero = 0 ;
|
|
|
|
out.write((char *)(&temp_zero), _size);
|
|
|
|
return _size;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (byteswap) {
|
|
|
|
char * byteswap_buf = (char *) calloc (_size, 1);
|
|
|
|
byteswap_var(byteswap_buf, (char *) _write_buffer);
|
|
|
|
out.write(byteswap_buf, _size);
|
|
|
|
free (byteswap_buf);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
out.write((char *) _write_buffer, _size);
|
|
|
|
}
|
|
|
|
|
|
|
|
return _size;
|
|
|
|
|
2022-08-23 18:47:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::ostream& Trick::operator<< (std::ostream& s, const Trick::VariableReference& ref) {
|
|
|
|
s << " \"" << ref.getName() << "\"";
|
|
|
|
return s;
|
|
|
|
}
|