mirror of
https://github.com/nasa/trick.git
synced 2025-01-10 23:12:44 +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
204 lines
7.4 KiB
C++
204 lines
7.4 KiB
C++
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <iostream>
|
|
#ifdef __linux
|
|
#include <cxxabi.h>
|
|
#endif
|
|
#include <netdb.h>
|
|
|
|
|
|
#include "trick/VariableServer.hh"
|
|
#include "trick/variable_server_sync_types.h"
|
|
#include "trick/input_processor_proto.h"
|
|
#include "trick/tc_proto.h"
|
|
#include "trick/message_proto.h"
|
|
#include "trick/message_type.h"
|
|
#include "trick/realtimesync_proto.h"
|
|
#include "trick/ExecutiveException.hh"
|
|
#include "trick/exec_proto.h"
|
|
|
|
void exit_var_thread(void *in_vst) ;
|
|
|
|
void * Trick::VariableServerThread::thread_body() {
|
|
|
|
int ii , jj;
|
|
int msg_len;
|
|
int ret;
|
|
int nbytes = -1;
|
|
char *last_newline ;
|
|
unsigned int size ;
|
|
socklen_t sock_size ;
|
|
|
|
// We need to make the thread to VariableServerThread map before we accept the connection.
|
|
// Otherwise we have a race where this thread is unknown to the variable server and the
|
|
// client gets confirmation that the connection is ready for communication.
|
|
vs->add_vst( pthread_self() , this ) ;
|
|
|
|
if ( listen_dev->socket_type == SOCK_STREAM ) {
|
|
tc_accept(listen_dev, &connection);
|
|
tc_blockio(&connection, TC_COMM_ALL_OR_NOTHING);
|
|
}
|
|
connection_accepted = true ;
|
|
|
|
pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, NULL) ;
|
|
pthread_cleanup_push(exit_var_thread, (void *) this);
|
|
|
|
/* Save off the host and port information of the source port. We want to ignore messages from
|
|
this port if we are a multicast socket */
|
|
struct sockaddr_in self_s_in ;
|
|
int s_in_size = sizeof(self_s_in) ;
|
|
if ( conn_type == MCAST ) {
|
|
getsockname( connection.socket , (struct sockaddr *)&self_s_in, (socklen_t *)&s_in_size) ;
|
|
if ( self_s_in.sin_addr.s_addr == 0 ) {
|
|
char hname[80];
|
|
struct hostent *ip_host;
|
|
gethostname(hname, (size_t) 80);
|
|
ip_host = gethostbyname(hname);
|
|
memcpy(&(self_s_in.sin_addr.s_addr), ip_host->h_addr, (size_t) ip_host->h_length);
|
|
}
|
|
}
|
|
|
|
// if log is set on for variable server (e.g., in input file), turn log on for each client
|
|
if (vs->get_log()) {
|
|
log = true ;
|
|
}
|
|
|
|
try {
|
|
while (1) {
|
|
|
|
// Pause here if we are in a restart condition
|
|
pthread_mutex_lock(&restart_pause) ;
|
|
|
|
/* Check the length of the message on the socket */
|
|
nbytes = recvfrom( connection.socket, incoming_msg, MAX_CMD_LEN, MSG_PEEK, NULL, NULL ) ;
|
|
if (nbytes == 0 ) {
|
|
break ;
|
|
}
|
|
|
|
if (nbytes != -1) { // -1 means socket is nonblocking and no data to read
|
|
/* find the last newline that is present on the socket */
|
|
incoming_msg[nbytes] = '\0' ;
|
|
last_newline = rindex( incoming_msg , '\n') ;
|
|
|
|
/* if there is a newline then there is a complete command on the socket */
|
|
if ( last_newline != NULL ) {
|
|
/* only remove up to (and including) the last newline on the socket */
|
|
size = last_newline - incoming_msg + 1;
|
|
if ( conn_type == UDP ) {
|
|
// Save the remote host information, that is where we are going to send replies.
|
|
sock_size = sizeof(connection.remoteServAddr) ;
|
|
nbytes = recvfrom( connection.socket, incoming_msg, size, 0 ,
|
|
(struct sockaddr *)&connection.remoteServAddr, &sock_size ) ;
|
|
} else if ( conn_type == MCAST ) {
|
|
// Save the remove host information for test against ourself.
|
|
struct sockaddr_in s_in ;
|
|
sock_size = sizeof(s_in) ;
|
|
nbytes = recvfrom( connection.socket, incoming_msg, size, 0 ,
|
|
(struct sockaddr *)&s_in, &sock_size ) ;
|
|
// If this message is from us, then ignore it.
|
|
if ( s_in.sin_addr.s_addr == self_s_in.sin_addr.s_addr and s_in.sin_port == self_s_in.sin_port) {
|
|
nbytes = 0 ;
|
|
}
|
|
} else {
|
|
// We know where we are sending information, no need to save it.
|
|
nbytes = recvfrom( connection.socket, incoming_msg, size, 0 , NULL, NULL ) ;
|
|
}
|
|
} else {
|
|
nbytes = 0 ;
|
|
}
|
|
}
|
|
|
|
if ( nbytes > 0 ) {
|
|
|
|
msg_len = nbytes ;
|
|
if (debug >= 3) {
|
|
message_publish(MSG_DEBUG, "%p tag=<%s> var_server received bytes = msg_len = %d\n", &connection, connection.client_tag, nbytes);
|
|
}
|
|
|
|
incoming_msg[msg_len] = '\0' ;
|
|
|
|
if (vs->get_info_msg() || (debug >= 1)) {
|
|
message_publish(MSG_DEBUG, "%p tag=<%s> var_server received: %s", &connection, connection.client_tag, incoming_msg) ;
|
|
}
|
|
if (log) {
|
|
message_publish(MSG_PLAYBACK, "tag=<%s> time=%f %s", connection.client_tag, exec_get_sim_time(), incoming_msg) ;
|
|
}
|
|
|
|
for( ii = 0 , jj = 0 ; ii <= msg_len ; ii++ ) {
|
|
if ( incoming_msg[ii] != '\r' ) {
|
|
stripped_msg[jj++] = incoming_msg[ii] ;
|
|
}
|
|
}
|
|
|
|
ip_parse(stripped_msg); /* returns 0 if no parsing error */
|
|
|
|
}
|
|
|
|
/* break out of loop if exit command found */
|
|
if (exit_cmd == true) {
|
|
break;
|
|
}
|
|
|
|
if ( copy_mode == VS_COPY_ASYNC ) {
|
|
copy_sim_data() ;
|
|
}
|
|
|
|
if ( (write_mode == VS_WRITE_ASYNC) or
|
|
((copy_mode == VS_COPY_ASYNC) and (write_mode == VS_WRITE_WHEN_COPIED)) or
|
|
(! is_real_time()) ) {
|
|
if ( !pause_cmd ) {
|
|
ret = write_data() ;
|
|
if ( ret < 0 ) {
|
|
break ;
|
|
}
|
|
}
|
|
}
|
|
pthread_mutex_unlock(&restart_pause) ;
|
|
|
|
usleep((unsigned int) (update_rate * 1000000));
|
|
}
|
|
} catch (Trick::ExecutiveException & ex ) {
|
|
message_publish(MSG_ERROR, "\nVARIABLE SERVER COMMANDED exec_terminate\n ROUTINE: %s\n DIAGNOSTIC: %s\n" ,
|
|
ex.file.c_str(), ex.message.c_str()) ;
|
|
exit(ex.ret_code) ;
|
|
} catch (const std::exception &ex) {
|
|
message_publish(MSG_ERROR, "\nVARIABLE SERVER caught std::exception\n DIAGNOSTIC: %s\n" ,
|
|
ex.what()) ;
|
|
exit(-1) ;
|
|
#ifdef __linux
|
|
#ifdef __GNUC__
|
|
#if __GNUC__ >= 4 && __GNUC_MINOR__ >= 2
|
|
// for post gcc 4.1.2 or whatever glibc version is used in RHEL6 and above.
|
|
} catch (abi::__forced_unwind&) {
|
|
//pthread_exit and pthread_cancel will cause an abi::__forced_unwind to be thrown. Rethrow it.
|
|
throw;
|
|
#endif
|
|
#endif
|
|
#endif
|
|
} catch (...) {
|
|
#ifdef __linux
|
|
#ifdef __GNUC__
|
|
#if (__GNUC__ == 4 && __GNUC_MINOR__ == 1) || __GNUC__ == 12
|
|
// for gcc 4.1.2 or whatever glib version in RHEL 5 that does not work with the abi::__forced_unwind
|
|
// Also seems to have a problem with gcc 12
|
|
throw;
|
|
#else
|
|
message_publish(MSG_ERROR, "\nVARIABLE SERVER caught unknown exception\n" ) ;
|
|
exit(-1) ;
|
|
#endif
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
if (debug >= 3) {
|
|
message_publish(MSG_DEBUG, "%p tag=<%s> var_server receive loop exiting\n", &connection, connection.client_tag);
|
|
}
|
|
|
|
pthread_cleanup_pop(1);
|
|
pthread_exit(NULL) ;
|
|
return NULL ;
|
|
|
|
}
|
|
|