mirror of
https://github.com/nasa/trick.git
synced 2025-01-15 01:09:59 +00:00
9c8799610f
* Adds the send_once command and message type, which allows a user to request a variable to be sent immediately and only once (intended to replace the var_add, var_send, var_clear idiom that is commonly used for this purpose) * Minor refactoring of variable server internals to reduce repeated code * Adds SIM_test_varserv to integration test to test basic variable server functionality * Changes graphics client for SIM_billiards to use var_send_once as an example of intended use * Add documentation for var_send_once in docs and tutorial * Set exit_code_enabled in trick unit tests to be true by default * Patch for failing bookworm build
325 lines
12 KiB
C++
325 lines
12 KiB
C++
/*
|
|
PURPOSE: (Allows clients to get and set Trick parameters)
|
|
PROGRAMMERS: (((Alex Lin) (NASA) (8/06) (--)))
|
|
*/
|
|
|
|
#include <iostream>
|
|
#include <pthread.h>
|
|
#include "trick/VariableServer.hh"
|
|
#include "trick/parameter_types.h"
|
|
#include "trick/bitfield_proto.h"
|
|
#include "trick/trick_byteswap.h"
|
|
#include "trick/tc_proto.h"
|
|
#include "trick/message_proto.h"
|
|
#include "trick/message_type.h"
|
|
|
|
|
|
extern "C" {
|
|
void *trick_bswap_buffer(void *out, void *in, ATTRIBUTES * attr, int tofrom) ;
|
|
}
|
|
|
|
#define MAX_MSG_LEN 8192
|
|
|
|
|
|
int Trick::VariableServerThread::write_binary_data( int Start, char *buf1, const std::vector<VariableReference *>& given_vars, VS_MESSAGE_TYPE message_type) {
|
|
int vars = 0;
|
|
int i;
|
|
int ret ;
|
|
int HeaderSize, MessageSize;
|
|
int NumVariablesProcessed;
|
|
unsigned int msg_type , offset, len ;
|
|
unsigned int size ;
|
|
unsigned int swap_int ;
|
|
char * address = 0 ;
|
|
char* param_name;
|
|
|
|
/* start the offset 4 bytes into the message, we'll subtract the sizeof offset at the end */
|
|
offset = sizeof(msg_type) + sizeof(offset) ;
|
|
|
|
if (byteswap) {
|
|
/* Swap message type bytes */
|
|
msg_type = trick_byteswap_int((int)message_type) ;
|
|
} else {
|
|
msg_type = message_type;
|
|
}
|
|
memcpy(buf1, &msg_type , sizeof(msg_type)) ;
|
|
HeaderSize = sizeof(msg_type);
|
|
|
|
offset += sizeof(unsigned int) ;
|
|
HeaderSize += sizeof(unsigned int);
|
|
|
|
for (i = Start; i < (int)given_vars.size() ; i++) {
|
|
|
|
// data to send was copied to buffer in copy_sim_data
|
|
address = (char *)given_vars[i]->buffer_out;
|
|
size = given_vars[i]->size ;
|
|
|
|
param_name = given_vars[i]->ref->reference;
|
|
len = strlen(param_name) ;
|
|
// when var_binary_nonames, do not put the variable names into the message to be sent
|
|
if (binary_data_nonames) {
|
|
MessageSize = sizeof(int) + sizeof(size) + size ;
|
|
} else {
|
|
MessageSize = sizeof(len) + len + sizeof(int) + sizeof(size) + size ;
|
|
}
|
|
|
|
/* make sure this message will fit in a packet by itself */
|
|
if ( (HeaderSize + MessageSize) > MAX_MSG_LEN ) {
|
|
message_publish(MSG_WARNING, "%p Variable Server buffer[%d] too small (need %d) for symbol %s, SKIPPING IT.\n",
|
|
&connection, MAX_MSG_LEN,
|
|
(int)(HeaderSize + MessageSize),
|
|
given_vars[i]->ref->reference );
|
|
continue;
|
|
}
|
|
|
|
if ( (offset + MessageSize) < MAX_MSG_LEN ) {
|
|
if (byteswap) {
|
|
if (!binary_data_nonames) {
|
|
swap_int = trick_byteswap_int((int)len) ;
|
|
memcpy(&buf1[offset] , &swap_int , sizeof(len)) ;
|
|
offset += sizeof(len) ;
|
|
|
|
memcpy(&buf1[offset] , param_name , (size_t)len) ;
|
|
offset += len ;
|
|
}
|
|
|
|
swap_int = trick_byteswap_int(given_vars[i]->ref->attr->type) ;
|
|
memcpy(&buf1[offset] , &swap_int , sizeof(int)) ;
|
|
offset += sizeof(int) ;
|
|
|
|
swap_int = trick_byteswap_int((int)size) ;
|
|
memcpy(&buf1[offset] , &swap_int , sizeof(size)) ;
|
|
offset += sizeof(size) ;
|
|
|
|
/* TODO: There is a bug here, this call will want to swap the entire buffer, we may not have the whole buffer */
|
|
trick_bswap_buffer(&buf1[offset], address, given_vars[i]->ref->attr, 1);
|
|
offset += size ;
|
|
}
|
|
else {
|
|
int temp_i ;
|
|
unsigned int temp_ui ;
|
|
|
|
if (!binary_data_nonames) {
|
|
memcpy(&buf1[offset] , &len , sizeof(len)) ;
|
|
offset += sizeof(len) ;
|
|
|
|
memcpy(&buf1[offset] , param_name , (size_t)len) ;
|
|
offset += len ;
|
|
}
|
|
|
|
memcpy(&buf1[offset] , &given_vars[i]->ref->attr->type , sizeof(int)) ;
|
|
offset += sizeof(int) ;
|
|
|
|
memcpy(&buf1[offset] , &size , sizeof(size)) ;
|
|
offset += sizeof(size) ;
|
|
|
|
switch ( given_vars[i]->ref->attr->type ) {
|
|
case TRICK_BITFIELD:
|
|
temp_i = GET_BITFIELD(address , given_vars[i]->ref->attr->size ,
|
|
given_vars[i]->ref->attr->index[0].start, given_vars[i]->ref->attr->index[0].size) ;
|
|
memcpy(&buf1[offset] , &temp_i , (size_t)size) ;
|
|
break ;
|
|
case TRICK_UNSIGNED_BITFIELD:
|
|
temp_ui = GET_UNSIGNED_BITFIELD(address , given_vars[i]->ref->attr->size ,
|
|
given_vars[i]->ref->attr->index[0].start, given_vars[i]->ref->attr->index[0].size) ;
|
|
memcpy(&buf1[offset] , &temp_ui , (size_t)size) ;
|
|
break ;
|
|
case TRICK_NUMBER_OF_TYPES:
|
|
// TRICK_NUMBER_OF_TYPES is an error case
|
|
temp_i = 0 ;
|
|
memcpy(&buf1[offset] , &temp_i , (size_t)size) ;
|
|
break ;
|
|
default:
|
|
memcpy(&buf1[offset] , address , (size_t)size) ;
|
|
break ;
|
|
}
|
|
offset += size ;
|
|
}
|
|
}
|
|
else {
|
|
/* indicate that we're over the maximum size */
|
|
if (debug >= 2) {
|
|
message_publish(MSG_DEBUG, "%p tag=<%s> var_server buffer[%d] too small (need %d), sending multiple binary packets.\n",
|
|
&connection, connection.client_tag, MAX_MSG_LEN,
|
|
(int)(offset + MessageSize) );
|
|
}
|
|
break ;
|
|
}
|
|
}
|
|
|
|
/* adjust the header with the correct information reflecting what has been accomplished */
|
|
NumVariablesProcessed = i - Start;
|
|
|
|
offset -= sizeof(offset) ;
|
|
if (byteswap) {
|
|
swap_int = trick_byteswap_int((int)offset) ;
|
|
memcpy(buf1 + sizeof(msg_type) , &swap_int , sizeof(offset)) ;
|
|
|
|
swap_int = trick_byteswap_int( NumVariablesProcessed ) ;
|
|
memcpy( buf1 + sizeof(msg_type) + sizeof(offset), &swap_int , sizeof(swap_int)) ;
|
|
}
|
|
else {
|
|
memcpy(buf1 + sizeof(msg_type) , &offset , sizeof(offset)) ;
|
|
memcpy( buf1 + sizeof(msg_type) + sizeof(offset), &NumVariablesProcessed , sizeof( NumVariablesProcessed )) ;
|
|
}
|
|
|
|
if (debug >= 2) {
|
|
message_publish(MSG_DEBUG, "%p tag=<%s> var_server sending %u binary bytes containing %d variables.\n", &connection,
|
|
connection.client_tag, (unsigned int)(offset + sizeof(offset)), NumVariablesProcessed);
|
|
}
|
|
|
|
len = offset + sizeof(msg_type) ;
|
|
ret = tc_write(&connection, (char *) buf1, len);
|
|
if ( ret != (int)len ) {
|
|
return(-1) ;
|
|
}
|
|
|
|
/* return the index to the next symbol to send or V->num_vars if all done */
|
|
return i;
|
|
}
|
|
|
|
int Trick::VariableServerThread::write_ascii_data(char * dest_buf, const std::vector<VariableReference *>& given_vars, VS_MESSAGE_TYPE message_type ) {
|
|
|
|
sprintf(dest_buf, "%d\t", message_type) ;
|
|
|
|
for (int i = 0; i < given_vars.size(); i++) {
|
|
char curr_buf[MAX_MSG_LEN];
|
|
int ret = vs_format_ascii( given_vars[i] , curr_buf);
|
|
|
|
if (ret < 0) {
|
|
message_publish(MSG_WARNING, "%p Variable Server string buffer[%d] too small for symbol %s, TRUNCATED IT.\n",
|
|
&connection, MAX_MSG_LEN, given_vars[i]->ref->reference );
|
|
}
|
|
|
|
/* make sure this message will fit in a packet by itself */
|
|
if( strlen( curr_buf ) + 2 > MAX_MSG_LEN ) {
|
|
message_publish(MSG_WARNING, "%p Variable Server buffer[%d] too small for symbol %s, TRUNCATED IT.\n",
|
|
&connection, MAX_MSG_LEN, given_vars[i]->ref->reference );
|
|
curr_buf[MAX_MSG_LEN - 1] = '\0';
|
|
}
|
|
|
|
int len = strlen(dest_buf) ;
|
|
|
|
/* make sure there is space for the next tab or next newline and null */
|
|
if( len + strlen( curr_buf ) + 2 > MAX_MSG_LEN ) {
|
|
// If there isn't, send incomplete message
|
|
if (debug >= 2) {
|
|
message_publish(MSG_DEBUG, "%p tag=<%s> var_server sending %d ascii bytes:\n%s\n",
|
|
&connection, connection.client_tag, (int)strlen(dest_buf), dest_buf) ;
|
|
}
|
|
|
|
ret = tc_write(&connection, (char *) dest_buf, len);
|
|
if ( ret != len ) {
|
|
return(-1) ;
|
|
}
|
|
dest_buf[0] = '\0';
|
|
}
|
|
|
|
strcat(dest_buf, curr_buf);
|
|
strcat(dest_buf, "\t");
|
|
}
|
|
|
|
int len = strlen(dest_buf) ;
|
|
|
|
if ( len > 0 ) {
|
|
dest_buf[ strlen(dest_buf) - 1 ] = '\n';
|
|
|
|
if (debug >= 2) {
|
|
message_publish(MSG_DEBUG, "%p tag=<%s> var_server sending %d ascii bytes:\n%s\n",
|
|
&connection, connection.client_tag, (int)strlen(dest_buf), dest_buf) ;
|
|
}
|
|
int ret = tc_write(&connection, (char *) dest_buf, (int)strlen(dest_buf));
|
|
if ( ret != (int)strlen(dest_buf) ) {
|
|
return(-1) ;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int Trick::VariableServerThread::write_data() {
|
|
|
|
int ret;
|
|
unsigned int i ;
|
|
char buf1[ MAX_MSG_LEN ];
|
|
int len ;
|
|
|
|
// do not send anything when there are no variables!
|
|
if ( vars.size() == 0 or packets_copied == 0 ) {
|
|
return 0;
|
|
}
|
|
|
|
/* Acquire sole access to vars[ii]->buffer_in. */
|
|
if ( var_data_staged and pthread_mutex_trylock(©_mutex) == 0 ) {
|
|
unsigned int ii;
|
|
void * temp_p;
|
|
// Swap buffer_in and buffer_out for each vars[ii].
|
|
for ( ii = 0 ; ii < vars.size() ; ii++ ) {
|
|
temp_p = vars[ii]->buffer_in;
|
|
vars[ii]->buffer_in = vars[ii]->buffer_out;
|
|
vars[ii]->buffer_out = temp_p;
|
|
}
|
|
var_data_staged = false;
|
|
|
|
/* Relinquish sole access to vars[ii]->buffer_in. */
|
|
pthread_mutex_unlock(©_mutex) ;
|
|
|
|
if (binary_data) {
|
|
int index = 0;
|
|
|
|
do {
|
|
ret = write_binary_data( index, buf1, vars, VS_VAR_LIST );
|
|
if ( ret >= 0 ) {
|
|
index = ret ;
|
|
} else {
|
|
return(-1) ;
|
|
}
|
|
} while( index < (int)vars.size() );
|
|
|
|
return 0;
|
|
|
|
} else { /* ascii mode */
|
|
return write_ascii_data(buf1, vars, VS_VAR_LIST );
|
|
}
|
|
}
|
|
}
|
|
|
|
int Trick::VariableServerThread::write_data(std::vector<VariableReference *> given_vars) {
|
|
// do not send anything when there are no variables!
|
|
if ( given_vars.size() == 0) {
|
|
return(0);
|
|
}
|
|
|
|
/* Acquire sole access to vars[ii]->buffer_in. */
|
|
if ( pthread_mutex_trylock(©_mutex) == 0 ) {
|
|
// Swap buffer_in and buffer_out for each vars[ii].
|
|
for (int i = 0 ; i < given_vars.size() ; i++ ) {
|
|
void *temp_p = given_vars[i]->buffer_in;
|
|
given_vars[i]->buffer_in = given_vars[i]->buffer_out;
|
|
given_vars[i]->buffer_out = temp_p;
|
|
}
|
|
/* Relinquish sole access to vars[ii]->buffer_in. */
|
|
pthread_mutex_unlock(©_mutex) ;
|
|
|
|
char buf1[ MAX_MSG_LEN ];
|
|
|
|
if (binary_data) {
|
|
int index = 0;
|
|
|
|
do {
|
|
int ret = write_binary_data( index, buf1, given_vars, VS_SEND_ONCE );
|
|
if ( ret >= 0 ) {
|
|
index = ret ;
|
|
} else {
|
|
return(-1) ;
|
|
}
|
|
} while( index < (int)given_vars.size() );
|
|
|
|
return 0;
|
|
|
|
} else { /* ascii mode */
|
|
return write_ascii_data(buf1, given_vars, VS_SEND_ONCE);
|
|
}
|
|
}
|
|
}
|