/** @page LEVEL2 Variable Server When running a %Trick simulation, unless specifically turned off, a server called the "variable server" is always up and listening in a separate thread of execution. The variable server is privy to simulation parameters and their values since it resides in an asynchronous simulation thread. Threads share the same address space as their siblings and parent. Clients connect to the variable server in order to set/get values of %Trick processed variables. You may already be familiar with the %Trick applications that use the variable server: the simulation control panel, %Trick View (TV) , Event/Malfunction %Trick View (MTV) , and the stripchart. The variable server is a convenient way for external applications to interact with the simulation. Any application that needs to set or get simulation parameters may do so through the variable server. The external application need not be on the same machine since the connection to the variable server is via a %Trick communication TCP/IP socket. @section LEVEL3 Variable Server Broadcast Channel To connect to the variable server for any simulation, a client needs to know the hostname and port. As of 10.5, the port number is determined by the OS. For external applications the best way to find a varible server port is to listen to the variable server broadcast channel. Every simulation variable server will broadcast the host and port number to the broadcast channel. The channel is address 224.3.14.15 port 9265. All simulations on your network sends it's information to this address and port so there may be multiple messages with variable server information available here. Here is some C code that reads all messages on the variable server channel. @code #include #include #include #include int main() { int mcast_socket ; char buf1[1024] ; ssize_t num_bytes ; int value = 1; struct sockaddr_in sockin ; struct ip_mreq mreq; if ((mcast_socket = socket(AF_INET, SOCK_DGRAM, 0)) < 0) { perror("init socket"); } if (setsockopt(mcast_socket, SOL_SOCKET, SO_REUSEADDR, (char *) &value, (socklen_t) sizeof(value)) < 0) { perror("setsockopt: reuseaddr"); } #ifdef SO_REUSEPORT if (setsockopt(mcast_socket, SOL_SOCKET, SO_REUSEPORT, (char *) &value, sizeof(value)) < 0) { perror("setsockopt: reuseport"); } #endif // Use setsockopt() to request that the kernel join a multicast group mreq.imr_multiaddr.s_addr = inet_addr("224.3.14.15"); mreq.imr_interface.s_addr = htonl(INADDR_ANY); if (setsockopt(mcast_socket, IPPROTO_IP, IP_ADD_MEMBERSHIP, (char *) &mreq, (socklen_t) sizeof(mreq)) < 0) { perror("setsockopt: ip_add_membership"); } // Set up destination address sockin.sin_family = AF_INET; sockin.sin_addr.s_addr = htonl(INADDR_ANY); sockin.sin_port = htons(9265); if ( bind(mcast_socket, (struct sockaddr *) &sockin, (socklen_t) sizeof(sockin)) < 0 ) { perror("bind"); } do { num_bytes = recvfrom(mcast_socket, buf1, 1024, 0 , NULL, NULL) ; if ( num_bytes > 0 ) { buf1[num_bytes] = '\0' ; printf("%s\n" , buf1) ; } } while ( num_bytes > 0 ) ; return 0 ; } @endcode The information sent by each variable server is a tab delimited list of strings -# Hostname -# Port -# User -# Process ID (PID) -# Simulation directory -# S_main command line name -# Input file -# %Trick version of simulation -# User defined tag -# Port (duplicate field for backwards compatibility) @section LEVEL3 User accessible routines These commands are for enabling/disabling the variable server, and for getting its status. The variable server is enabled by default. @li int var_server_set_enabled(int on_off) @li int var_server_get_enabled() @note Disabling the variable server will disable all %Trick runtime GUIs: simulation control panel, TV, MTV, and stripchart. These commands are for toggling information messages from the variable server (i.e., commands received from ALL clients). The messages go to the terminal, the simulation control panel, and the "send_hs" file in the RUN directory. The variable server information message capability is off by default. @li int @link Trick::VariableServer::set_var_server_info_msg_off() set_var_server_info_msg_off() @endlink @li int @link Trick::VariableServer::set_var_server_info_msg_on() set_var_server_info_msg_on() @endlink These commands are also for toggling information messages from the variable server (i.e., commands received from ALL clients). The messages only go to a dedicated "varserver_log" file in the RUN directory. The variable server log capability is off by default. @li int @link Trick::VariableServer::set_var_server_log_off() set_var_server_log_off() @endlink @li int @link Trick::VariableServer::set_var_server_log_on() set_var_server_log_on() @endlink @section LEVEL4 Getting and Setting the Variable Server Port Information To set the variable server port to a fixed number in the input file use var_server_set_port() @code trick.var_server_set_port( unsigned int port ) @endcode To get the variable server host and port information in the input file use var_server_get_hostname() and var_server_get_port(). @code trick.var_server_get_hostname() trick.var_server_get_port() @endcode @section LEVEL3 Commands The variable server accepts commands in the form of strings. The variable server parses these commands using the Python input processor. So in theory, any Python valid syntax is acceptable to the variable server. This section lists the commands that are specific for the variable server. Commands are sent over a %Trick communication TCP/IP socket to the variable server. Multiple commands (newline separated) can be sent in the string over the socket. The variable server will send back information to the requesting client. If the command contains a syntax error, Python will print an error message to the screen, but nothing will be returned to the client. @section LEVEL4 Adding a Variable @code trick.var_add( string var_name ) @endcode or @code trick.var_add( string var_name , string units ) @endcode Adding a variable will tell the variable server to send the variable's value back to the client at a specified frequency. An optional units parameter may be attached to the variable as the desired return units. Multiple variables may be added to the list to be sent back to the client. The format of the returned values are described below, Ascii Format or binary format. @section LEVEL4 Time Homogeneous or Synchronous Data @section LEVEL5 Copying Data Out of Simulation. @code trick.var_set_copy_mode(int mode) @endcode There are 3 options to when the variable server will copy data out from the simulation. Each option has unique capabilites. @par Asynchronous Copy (mode = trick.VS_COPY_ASYNC or 0) This is the default. Values are copied out of the sim asynchronously. Copies are done approximately at the var_cycle() rate during run and freeze mode. A separate thread is used to copy the data. The data is not guaranteed to be time homogenous. This mode does not affect the main thread real-time performance. @par End of Main Thread Execution Copy (mode = trick.VS_COPY_SCHEDULED or 1) This mode copies data at the end of execution frame. Copies are done exactly at the var_cycle() rate after the main thread has finished all of it's jobs scheduled to run at that time step both in run and freeze mode. All variables solely calculated in the main thread are guaranteed to be time homogenous. Variables calculated in child threads are not guaranteed to be time homogenous. Copying data may very slightly affect the main thread real-time performance. @par Top of Frame Copy (mode = trick.VS_COPY_TOP_OF_FRAME or 2) This mode copies data at the top of frame. Copies are done at a multiple and offset of the Executive software frame. During freeze mode copies are made at a multiple and offset of the freeze frame. With careful planning, all variables from all threads can be guaranteed to be time homogenous. Copying data may very slightly affect the main thread real-time performance. To set the frame multiplier and frame offset between copies use the following commands. The frame refers to the software frame in the Executive. In freeze mode a different multiplier and offset are used. @code trick.var_set_frame_multiplier(int mult) trick.var_set_frame_offset(int offset) trick.var_set_freeze_frame_multiplier(int mult) trick.var_set_freeze_frame_offset(int offset) @endcode @section LEVEL5 Writing Data Out of Simulation. @code trick.var_set_write_mode(int mode) @endcode There are 2 options when the variable server writes the data. @par Asynchronous Write ( mode = trick.VS_WRITE_ASYNC or 0 ) This is the default. Values are written onto the socket asynchronously. Writes are done approximately at the var_cycle() rate during run and freeze mode. A separate thread is used to copy write data. This mode does not affect the main thread real-time performance. @par Write When Copied ( mode = trick.VS_WRITE_WHEN_COPIED or 1 ) Values are written onto the socket as soon as they are copied from the simulation. The write rate depends on the copy. Writes are done in the main thread of execution. This can greatly affect real-tim performance if a large amount of data is requested. @section LEVEL5 Old Style var_sync() Command @code trick.var_sync(bool mode) @endcode var_sync() was previously used to control the copies and writes from the simulation. The number of options has outgrown what a single var_sync command can configure. It may still be used to configure a subset of the copy/write combinations. @code trick.var_sync(0) # asynchronous copy and asynchronous write. trick.var_sync(1) # end of main thread copy and asynchronous write. trick.var_sync(2) # end of main thread copy and write when copied. @endcode @section LEVEL4 Sending the Return Values Immediately @code trick.var_send() @endcode The var_send command forces the variable server to return the list of values to the client immediately. @section LEVEL4 Changing the Units @code trick.var_units( string var_name , string units ) @endcode The returned values can be converted to other units of measurments. The var_units command tells the variable server what units to use. If the units are changed, then the units are included in the returned string to the client. @section LEVEL4 Removing a Variable @code trick.var_remove( string var_name ) @endcode Removing a variable removes the variable from the list returned to the client. @section LEVEL4 Clearing the List of Variables @code trick.var_clear() @endcode To clear the whole list of variables sent to the client. @section LEVEL4 Exiting the Variable Server @code trick.var_exit() @endcode Disconnects the current client from the variable server. @section LEVEL4 Checking for existence of a variable @code trick.var_exists( string var_name ) @endcode To test if a variable name exists. A special return message is sent to the client when this command is processed. The return message is in this format: @code1<1 byte binary 0 or 1>@endcode @section LEVEL4 Changing the Return Value Cycle Rate @code trick.var_cycle( double cycle_rate ) @endcode Changes the rate of the return messages to the client. This rate is estimated and may not perfectly match the requested rate. @section LEVEL4 Pause the Variable Server @code trick.var_pause() @endcode Pauses the return values sent to the client. Even when paused, the variable server will accept new commands. @section LEVEL4 Unpause the Variable Server @code trick.var_unpause() @endcode Resumes sending the return values to the client. @section LEVEL4 Setting Ascii Return Format @code trick.var_ascii() @endcode Sets the return message format to ASCII. See below for the format of the message. @section LEVEL4 Setting Binary Return Format @code trick.var_binary() @endcode Sets the return message format to Binary. See below for the format of the message. @code trick.var_binary_nonames() @endcode This variation of the binary format reduces the amount of data that is sent to the client. See below for the exact format. @section LEVEL4 Sending stdout and stderr to client @code trick.var_set_send_stdio(bool on_off) @endcode If var_set_send_stdio is called with a true value, then all python stdout and stderr output will be redirected to the client instead of printing to the simulation stdout/stderr location. Note: output from C/C++ code called from python will direct it's output to the simulation stdout/stderr location. See the return message format for Stdio. This is useful to get output from the simulation such as the return values of a function. @code # Example in a variable server client to get the Trick version used to compile a sim # The C prototype is "const char *exec_get_current_version(void) ;" trick.var_set_send_stdio(True) sys.stdout.write(trick.exec_get_current_version()) # The returned text will look like this. See the return message format below 4 1 10 10.7.dev-1 # If a "print" is used instead of sys.stdout.write, a second message is sent containing # a single newline. print "trick.exec_get_current_version()" 4 1 10 10.7.dev-14 1 1 <- a single newline is the second message @endcode @section LEVEL4 Debugging Variable Server Messages @code trick.var_debug(int level) @endcode The level may range from 0-3. The larger the number the more debugging information is printed to the screen (for the current client only). @section LEVEL4 Logging Messages to file. These commands are for toggling information messages from the variable server (for this client only). The messages only go to a dedicated "varserver_log" file in the RUN directory. The variable server log capability is off by default. (See the global variable server commands @link Trick::VariableServer::set_var_server_log_on() set_var_server_log_on() @endlink and @link Trick::VariableServer::set_var_server_log_off() set_var_server_log_off() @endlink for toggling the logging capability for ALL clients.) @code trick.var_server_log_on() trick.var_server_log_off() @endcode @section LEVEL4 Setting Variable Server Client Tag @code trick.var_set_client_tag(string name) @endcode This sets an identifying name tag to be associated with the current client that will be printed with each information message displayed. Information messages are displayed as a result of @link Trick::VariableServer::set_var_server_info_msg_on() set_var_server_info_msg_on() @endlink, @c var_server_log_on() or @c var_debug(). For instance, %Trick sets a name tag for each of its variable server clients (simulation control panel is "SimControl", TV is "TRICK_TV", etc.). @section LEVEL4 Byteswapping @code trick.var_byteswap(bool on_off) @endcode @section LEVEL3 Returned Values By default the values retrieved are sent asynchronously to the client. That is, the values retrieved by the variable server are pulled directly from memory asynchronously and do not guarantee synchronization from the same simulation execution frames unless the var_sync command is used. Values will be returned to the client in the same order that they were issued in the var_add command(s). Typically the client receives the data from the variable server in a buffer via the tc_read command (see TrickComm for more information). @subsection LEVEL3 Ascii Format The default format, or if var_ascii is commanded specifically, causes the variable server to return a buffer containing a tab delimited character string in the following format: @code 0\t[\t. . .\t]\n @endcode where N is the number of variables registered via the var_add command(s). The "\t" represents a tab character, and the "\n" is the newline character that always ends the string. Note that if a value being returned is itself a character string data type, any tab (or other unprintable character) that occurs within the character string value will appear as an escaped character, i.e. preceded by a backslash. The 1st value returned in the list will always be a message indicator. The possible values of the message indicator are: - 0 returned variable value(s) from var_add or var_send - 1 returned value from var_exists - 2 returned value from send_sie_resource (special command used by Trick View) - 3 returned value from send_event_data (special command used by Events/Malfunctions Trick View) . If the variable units are also specified along with the variable name in a var_add or var_units command, then that variable will also have its units specification returned following its associated value separated by a single blank. For example, if the 2nd of N variables was specified with {} in either a var_add or var_units command, the returned string would be in the following format: @code 0\t\t {}. . .\t\n @endcode Note that the maximum message size that the variable server sends to the client is 8192 bytes. If the amount of data requested is larger than that, the ASCII message will be split into multiple messages. The client is responsible for concatenating the multiple messages back together. (Hint: look for the "\n" delimter) If a syntax error occurs when processing the variable server client command, Python will print an error message to the screen, but nothing will be returned to the client. If a var_add command was issued for a non-existent variable, there will be a one time %Trick error message printed to the screen, but the resulting data sent to the client is still ok. The value returned for the non-existent variable is the string "BAD_REF". @subsection LEVEL3 Binary Format By specifying the var_binary or var_binary_nonames command, the variable server will return values in a binary message formatted as follows: @code . . . @endcode Where the first 12 bytes are the message header: - @e message_indicator is the same possible values as in var_ascii shown above : a 4 byte integer - @e message_size is the total size of the message in bytes (NOT including message_indicator) : a 4 byte integer - @e N is the number of variables registered via the var_add command(s) : a 4 byte integer . and the remaining bytes of the message contain the variable data: - @e variable_namelength is the string length of the variable name : a 4 byte integer (NOT present for var_binary_nonames) - @e variable_name is the ASCII variable name string : @e variable_namelength bytes of string (NOT present for var_binary_nonames) - @e variable_type is %Trick data type of the variable : a 4 byte integer (see Trick::MemoryManager::TRICK_TYPE) - @e variable_size is number of bytes the variable occupies in memory : a 4 byte integer - @e variable_value is the variable's current value : @e variable_size bytes of @e variable_type When the client has requested a very large amount of data, it is possible that it may require more than one message to be returned. The maximum message size is 8192 bytes, so if the data returned by the variable server requires more space than that (once formatted into the above message format), then the variable server sends more than one message. This is indicated by the @e N field. For example, if the client has requested 15 variables, and @e N = 15, then everything is contained in that one message. However if @e N < 15, then the client should continue reading messages until all @e N received add up to 15. If a syntax error occurs when processing the variable server client command, Python will print an error message to the screen, but nothing will be returned to the client. If a var_add command was issued for a non-existent variable, there will be a one time %Trick error message printed to the screen, but the resulting data sent to the client is still ok. The message returned for the non-existent variable will have a type of 24 and it's value will be the string "BAD_REF". @subsection LEVEL3 Stdio Format These messages are sent to the client if stdout and stderr are redirected. See "Sending stdout and stderr to client" for more details. @code 4 @endcode - @e message_id Stdio messages are message_id = 4. - @e stream is the stream the message was written to. 1 = stdout, 2 = stderr - @e size is the number of bytes in the section. The newline between the and is not counted in the size. - @e text is the message Only output from python is redirected, i.e. "print" or calls to "sys.stdout.write()". C/C++ code called from python will still direct their stdout/stderr to the simulation output location. The "print" statement will send 2 messages, the text in the print, and an additional newline. Calls to sys.stdout.write() only generate 1 message. @note Error messages printed by python to stderr may be sent in multiple messages. */