2022-08-23 13:47:56 -05:00
# include <string.h>
# include <iostream>
# include <sstream>
# include <stdlib.h>
# include <udunits2.h>
# include "trick/VariableServerSession.hh"
# include "trick/variable_server_message_types.h"
# include "trick/memorymanager_c_intf.h"
# include "trick/exec_proto.h"
# include "trick/command_line_protos.h"
# include "trick/message_proto.h"
# include "trick/message_type.h"
# include "trick/TrickConstant.hh"
# include "trick/sie_c_intf.h"
# include "trick/UdUnits.hh"
# include "trick/map_trick_units_to_udunits.hh"
int Trick : : VariableServerSession : : var_add ( std : : string in_name ) {
VariableReference * new_var ;
if ( in_name = = " time " ) {
2023-03-20 17:53:01 -05:00
new_var = new VariableReference ( in_name , & _time ) ;
2022-08-23 13:47:56 -05:00
} else {
new_var = new VariableReference ( in_name ) ;
}
2023-03-20 17:53:01 -05:00
_session_variables . push_back ( new_var ) ;
2022-08-23 13:47:56 -05:00
return ( 0 ) ;
}
int Trick : : VariableServerSession : : var_add ( std : : string var_name , std : : string units_name ) {
var_add ( var_name ) ;
var_units ( var_name , units_name ) ;
return ( 0 ) ;
}
// Helper function for var_send_once
std : : vector < std : : string > split ( const std : : string & str , const char delim ) {
std : : stringstream ss ( str ) ;
std : : string s ;
std : : vector < std : : string > ret ;
while ( std : : getline ( ss , s , delim ) ) {
ret . push_back ( s ) ;
}
return ret ;
}
int Trick : : VariableServerSession : : var_send_once ( std : : string in_name , int num_vars ) {
std : : vector < std : : string > var_names = split ( in_name , ' , ' ) ;
if ( var_names . size ( ) ! = num_vars ) {
message_publish ( MSG_ERROR , " Number of variables sent to var_send_once (%d) does not match num_vars (%d). \n " , var_names . size ( ) , num_vars ) ;
return - 1 ;
}
std : : vector < VariableReference * > given_vars ;
for ( auto & varName : var_names ) {
VariableReference * new_var ;
if ( varName = = " time " ) {
2023-03-20 17:53:01 -05:00
new_var = new VariableReference ( varName , & _time ) ;
2022-08-23 13:47:56 -05:00
} else {
new_var = new VariableReference ( varName ) ;
}
given_vars . push_back ( new_var ) ;
}
copy_sim_data ( given_vars , false ) ;
write_data ( given_vars , VS_SEND_ONCE ) ;
return ( 0 ) ;
}
int Trick : : VariableServerSession : : var_remove ( std : : string in_name ) {
2023-03-20 17:53:01 -05:00
for ( unsigned int ii = 0 ; ii < _session_variables . size ( ) ; ii + + ) {
std : : string var_name = _session_variables [ ii ] - > getName ( ) ;
2022-08-23 13:47:56 -05:00
if ( ! var_name . compare ( in_name ) ) {
2023-03-20 17:53:01 -05:00
delete _session_variables [ ii ] ;
_session_variables . erase ( _session_variables . begin ( ) + ii ) ;
2022-08-23 13:47:56 -05:00
break ;
}
}
return ( 0 ) ;
}
int Trick : : VariableServerSession : : var_units ( std : : string var_name , std : : string units_name ) {
VariableReference * variable = find_session_variable ( var_name ) ;
if ( variable = = NULL ) {
// TODO: ERROR LOGGER HERE
return - 1 ;
}
return variable - > setRequestedUnits ( units_name ) ;
}
int Trick : : VariableServerSession : : var_exists ( std : : string in_name ) {
char buf1 [ 5 ] ;
bool error = false ;
unsigned int msg_type ;
REF2 * var_ref = ref_attributes ( in_name . c_str ( ) ) ;
if ( var_ref = = ( REF2 * ) NULL ) {
error = true ;
}
2023-03-20 17:53:01 -05:00
if ( _binary_data ) {
2022-08-23 13:47:56 -05:00
/* send binary 1 or 0 */
msg_type = VS_VAR_EXISTS ;
memcpy ( buf1 , & msg_type , sizeof ( msg_type ) ) ;
buf1 [ 4 ] = ( error = = false ) ;
2023-03-20 17:53:01 -05:00
if ( _debug > = 2 ) {
2023-04-18 16:27:32 -05:00
message_publish ( MSG_DEBUG , " %p tag=<%s> var_server sending 1 binary byte \n " , _connection , _connection - > getClientTag ( ) . c_str ( ) ) ;
2022-08-23 13:47:56 -05:00
}
2023-03-20 17:53:01 -05:00
_connection - > write ( buf1 , 5 ) ;
2022-08-23 13:47:56 -05:00
} else {
/* send ascii "1" or "0" */
sprintf ( buf1 , " %d \t %d \n " , VS_VAR_EXISTS , ( error = = false ) ) ;
2023-03-20 17:53:01 -05:00
if ( _debug > = 2 ) {
2023-04-18 16:27:32 -05:00
message_publish ( MSG_DEBUG , " %p tag=<%s> var_server sending: \n %s \n " , _connection , _connection - > getClientTag ( ) . c_str ( ) , buf1 ) ;
2022-08-23 13:47:56 -05:00
}
std : : string write_string ( buf1 ) ;
if ( write_string . length ( ) ! = strlen ( buf1 ) ) {
std : : cout < < " PROBLEM WITH STRING LENGTH: VAR_EXISTS ASCII " < < std : : endl ;
}
2023-03-20 17:53:01 -05:00
_connection - > write ( write_string ) ;
2022-08-23 13:47:56 -05:00
}
return ( 0 ) ;
}
int Trick : : VariableServerSession : : var_clear ( ) {
2023-03-20 17:53:01 -05:00
while ( ! _session_variables . empty ( ) ) {
delete _session_variables . back ( ) ;
_session_variables . pop_back ( ) ;
2022-08-23 13:47:56 -05:00
}
return ( 0 ) ;
}
int Trick : : VariableServerSession : : var_send ( ) {
copy_sim_data ( ) ;
write_data ( ) ;
return ( 0 ) ;
}
int Trick : : VariableServerSession : : var_cycle ( double in_rate ) {
2023-03-20 17:53:01 -05:00
_update_rate = in_rate ;
_cycle_tics = ( long long ) ( _update_rate * exec_get_time_tic_value ( ) ) ;
2022-08-23 13:47:56 -05:00
return ( 0 ) ;
}
int Trick : : VariableServerSession : : var_exit ( ) {
2023-03-20 17:53:01 -05:00
_exit_cmd = true ;
2022-08-23 13:47:56 -05:00
return ( 0 ) ;
}
int Trick : : VariableServerSession : : var_validate_address ( bool on_off ) {
2023-03-20 17:53:01 -05:00
_validate_address = on_off ;
2022-08-23 13:47:56 -05:00
return ( 0 ) ;
}
int Trick : : VariableServerSession : : var_debug ( int level ) {
2023-03-20 17:53:01 -05:00
_debug = level ;
2022-08-23 13:47:56 -05:00
return ( 0 ) ;
}
int Trick : : VariableServerSession : : var_ascii ( ) {
2023-03-20 17:53:01 -05:00
_binary_data = 0 ;
2022-08-23 13:47:56 -05:00
return ( 0 ) ;
}
int Trick : : VariableServerSession : : var_binary ( ) {
2023-03-20 17:53:01 -05:00
_binary_data = 1 ;
2022-08-23 13:47:56 -05:00
return ( 0 ) ;
}
int Trick : : VariableServerSession : : var_binary_nonames ( ) {
2023-03-20 17:53:01 -05:00
_binary_data = 1 ;
_binary_data_nonames = 1 ;
2022-08-23 13:47:56 -05:00
return ( 0 ) ;
}
int Trick : : VariableServerSession : : var_set_copy_mode ( int mode ) {
if ( mode > = VS_COPY_ASYNC and mode < = VS_COPY_TOP_OF_FRAME ) {
2023-03-20 17:53:01 -05:00
_copy_mode = ( VS_COPY_MODE ) mode ;
if ( _copy_mode = = VS_COPY_SCHEDULED ) {
2022-08-23 13:47:56 -05:00
long long sim_time_tics ;
sim_time_tics = exec_get_time_tics ( ) ;
// round the next call time to a multiple of the cycle
2023-03-20 17:53:01 -05:00
sim_time_tics - = sim_time_tics % _cycle_tics ;
_next_tics = sim_time_tics + _cycle_tics ;
2022-08-23 13:47:56 -05:00
sim_time_tics = exec_get_freeze_time_tics ( ) ;
// round the next call time to a multiple of the cycle
2023-03-20 17:53:01 -05:00
sim_time_tics - = sim_time_tics % _cycle_tics ;
_freeze_next_tics = sim_time_tics + _cycle_tics ;
2022-08-23 13:47:56 -05:00
} else {
2023-03-20 17:53:01 -05:00
_next_tics = TRICK_MAX_LONG_LONG ;
2022-08-23 13:47:56 -05:00
}
return 0 ;
}
return - 1 ;
}
int Trick : : VariableServerSession : : var_set_write_mode ( int mode ) {
if ( mode > = VS_WRITE_ASYNC and mode < = VS_WRITE_WHEN_COPIED ) {
2023-03-20 17:53:01 -05:00
_write_mode = ( VS_WRITE_MODE ) mode ;
2022-08-23 13:47:56 -05:00
return 0 ;
}
return - 1 ;
}
int Trick : : VariableServerSession : : var_sync ( int mode ) {
switch ( mode ) {
case 1 :
var_set_copy_mode ( VS_COPY_SCHEDULED ) ;
var_set_write_mode ( VS_WRITE_ASYNC ) ;
break ;
case 2 :
var_set_copy_mode ( VS_COPY_SCHEDULED ) ;
var_set_write_mode ( VS_WRITE_WHEN_COPIED ) ;
break ;
case 0 :
default :
var_set_copy_mode ( VS_COPY_ASYNC ) ;
var_set_write_mode ( VS_WRITE_ASYNC ) ;
break ;
}
return 0 ;
}
int Trick : : VariableServerSession : : var_set_frame_multiple ( unsigned int mult ) {
2023-03-20 17:53:01 -05:00
_frame_multiple = mult ;
2022-08-23 13:47:56 -05:00
return 0 ;
}
int Trick : : VariableServerSession : : var_set_frame_offset ( unsigned int offset ) {
2023-03-20 17:53:01 -05:00
_frame_offset = offset ;
2022-08-23 13:47:56 -05:00
return 0 ;
}
int Trick : : VariableServerSession : : var_set_freeze_frame_multiple ( unsigned int mult ) {
2023-03-20 17:53:01 -05:00
_freeze_frame_multiple = mult ;
2022-08-23 13:47:56 -05:00
return 0 ;
}
int Trick : : VariableServerSession : : var_set_freeze_frame_offset ( unsigned int offset ) {
2023-03-20 17:53:01 -05:00
_freeze_frame_offset = offset ;
2022-08-23 13:47:56 -05:00
return 0 ;
}
int Trick : : VariableServerSession : : var_byteswap ( bool on_off ) {
2023-03-20 17:53:01 -05:00
_byteswap = on_off ;
2022-08-23 13:47:56 -05:00
return ( 0 ) ;
}
bool Trick : : VariableServerSession : : get_send_stdio ( ) {
2023-03-20 17:53:01 -05:00
return _send_stdio ;
2022-08-23 13:47:56 -05:00
}
int Trick : : VariableServerSession : : set_send_stdio ( bool on_off ) {
2023-03-20 17:53:01 -05:00
_send_stdio = on_off ;
2022-08-23 13:47:56 -05:00
return ( 0 ) ;
}
int Trick : : VariableServerSession : : send_list_size ( ) {
unsigned int msg_type = VS_LIST_SIZE ;
2023-03-20 17:53:01 -05:00
int var_count = _session_variables . size ( ) ;
2022-08-23 13:47:56 -05:00
// send number of variables
2023-03-20 17:53:01 -05:00
if ( _binary_data ) {
2022-08-23 13:47:56 -05:00
// send in the binary message header format:
// <message_indicator><message_size><number_of_variables>
char buf1 [ 12 ] ;
unsigned int msg_type = VS_LIST_SIZE ;
memcpy ( buf1 , & msg_type , sizeof ( msg_type ) ) ;
memset ( & ( buf1 [ 4 ] ) , 0 , sizeof ( int ) ) ; // message size = 0
memcpy ( & ( buf1 [ 8 ] ) , & var_count , sizeof ( var_count ) ) ;
2023-03-20 17:53:01 -05:00
if ( _debug > = 2 ) {
message_publish ( MSG_DEBUG , " %p tag=<%s> var_server sending %d event variables \n " , _connection , _connection - > getClientTag ( ) . c_str ( ) , var_count ) ;
2022-08-23 13:47:56 -05:00
}
2023-03-20 17:53:01 -05:00
_connection - > write ( buf1 , sizeof ( buf1 ) ) ;
2022-08-23 13:47:56 -05:00
} else {
std : : stringstream write_string ;
write_string < < VS_LIST_SIZE < < " \t " < < var_count < < " \n " ;
// ascii
2023-03-20 17:53:01 -05:00
if ( _debug > = 2 ) {
message_publish ( MSG_DEBUG , " %p tag=<%s> var_server sending number of event variables: \n %s \n " , _connection , _connection - > getClientTag ( ) . c_str ( ) , write_string . str ( ) . c_str ( ) ) ;
2022-08-23 13:47:56 -05:00
}
2023-03-20 17:53:01 -05:00
_connection - > write ( write_string . str ( ) ) ;
2022-08-23 13:47:56 -05:00
}
return 0 ;
}
2023-04-18 16:27:32 -05:00
int Trick : : VariableServerSession : : transmit_file ( std : : string filename ) {
2022-08-23 13:47:56 -05:00
const unsigned int packet_size = 4095 ;
FILE * fp ;
unsigned int file_size ;
unsigned int current_size = 0 ;
unsigned int bytes_read ;
char buffer [ packet_size + 1 ] ;
int ret ;
2023-03-20 17:53:01 -05:00
if ( _debug > = 2 ) {
2023-04-18 16:27:32 -05:00
message_publish ( MSG_DEBUG , " %p tag=<%s> var_server opening %s. \n " , _connection , _connection - > getClientTag ( ) . c_str ( ) , filename . c_str ( ) ) ;
2022-08-23 13:47:56 -05:00
}
2023-04-18 16:27:32 -05:00
if ( ( fp = fopen ( filename . c_str ( ) , " r " ) ) = = NULL ) {
message_publish ( MSG_ERROR , " Variable Server Error: Cannot open %s. \n " , filename . c_str ( ) ) ;
2022-08-23 13:47:56 -05:00
sprintf ( buffer , " %d \t -1 \n " , VS_SIE_RESOURCE ) ;
std : : string message ( buffer ) ;
2023-03-20 17:53:01 -05:00
_connection - > write ( message ) ;
2022-08-23 13:47:56 -05:00
return ( - 1 ) ;
}
fseek ( fp , 0L , SEEK_END ) ;
file_size = ftell ( fp ) ;
sprintf ( buffer , " %d \t %u \n \0 " , VS_SIE_RESOURCE , file_size ) ;
std : : string message ( buffer ) ;
2023-03-20 17:53:01 -05:00
_connection - > write ( message ) ;
2022-08-23 13:47:56 -05:00
rewind ( fp ) ;
// Switch to blocking writes since this could be a large transfer.
2023-03-20 17:53:01 -05:00
if ( _connection - > setBlockMode ( true ) ) {
2023-02-22 11:35:30 -06:00
message_publish ( MSG_DEBUG , " Variable Server Error: Failed to set socket to blocking mode. \n " ) ;
2022-08-23 13:47:56 -05:00
}
while ( current_size < file_size ) {
bytes_read = fread ( buffer , 1 , packet_size , fp ) ;
message = std : : string ( buffer ) ;
message . resize ( bytes_read ) ;
2023-03-20 17:53:01 -05:00
ret = _connection - > write ( message ) ;
2022-08-23 13:47:56 -05:00
if ( ret ! = ( int ) bytes_read ) {
2023-04-18 16:27:32 -05:00
message_publish ( MSG_ERROR , " Variable Server Error: Failed to send file. Bytes read: %d Bytes sent: %d \n " , bytes_read , ret ) ;
2022-08-23 13:47:56 -05:00
return ( - 1 ) ;
}
current_size + = bytes_read ;
}
// Switch back to non-blocking writes.
2023-03-20 17:53:01 -05:00
if ( _connection - > setBlockMode ( false ) ) {
2023-02-22 11:35:30 -06:00
message_publish ( MSG_DEBUG , " Variable Server Error: Failed to set socket to non-blocking mode. \n " ) ;
2022-08-23 13:47:56 -05:00
return ( - 1 ) ;
}
return ( 0 ) ;
}
int Trick : : VariableServerSession : : send_file ( std : : string file_name ) {
return transmit_file ( file_name ) ;
}
int Trick : : VariableServerSession : : send_sie_resource ( ) {
sie_append_runtime_objs ( ) ;
return transmit_file ( std : : string ( command_line_args_get_default_dir ( ) ) + " /S_sie.resource " ) ;
}
int Trick : : VariableServerSession : : send_sie_class ( ) {
sie_class_attr_map_print_xml ( ) ;
return transmit_file ( std : : string ( command_line_args_get_default_dir ( ) ) + " / " + " S_sie_class.xml " ) ;
}
int Trick : : VariableServerSession : : send_sie_enum ( ) {
sie_enum_attr_map_print_xml ( ) ;
return transmit_file ( std : : string ( command_line_args_get_default_dir ( ) ) + " / " + " S_sie_enum.xml " ) ;
}
int Trick : : VariableServerSession : : send_sie_top_level_objects ( ) {
sie_top_level_objects_print_xml ( ) ;
return transmit_file ( std : : string ( command_line_args_get_default_dir ( ) ) + " / " + " S_sie_top_level_objects.xml " ) ;
}