2015-02-26 09:02:31 -06:00

507 lines
20 KiB
Plaintext

/**
@page LEVEL2 Executive Scheduler
This scheduler or derivative of this class is required for Trick simulations to run.
The scheduler is in charge of simulation execution. The scheduler maintains simulation
elapsed time. The scheduler has 4 modes of operation, Initialization, Run, Freeze,
and Shutdown. The scheduler maintains the simulation mode. Within each mode of operation
the executive/scheduler calls different sets of user and system jobs.
The scheduler is implemented by the Trick::Executive class.
@section LEVEL3 Executive Flow
The next set of flowcharts details how the Trick main executable runs.
@image html ../../../doxygen/images/executive_flowchart/initialization.png
@image rtf ../../../doxygen/images/executive_flowchart/initialization.png
<center><b>Figure Exec_1 Executive Initialization</b></center>
During the Initialization phase three job classes are called, default data, input_processor, and
initialization. All jobs of these classes are executed in the main thread. Not shown in the
picture is any job returning a non-zero value to "trick_ret" will cause the simulation immediately
to go to shutdown. Execution continues to the Scheduled loop.
@image html ../../../doxygen/images/executive_flowchart/scheduled_loop.png
@image rtf ../../../doxygen/images/executive_flowchart/scheduled_loop.png
<center><b>Figure Exec_2 Scheduled Loop</b></center>
Run mode is also called the Scheduled loop. The Scheduled loop calls the top_of_frame, the scheduled
jobs, and the end_of_frame jobs. Both the top_of_frame and end_of_frame
jobs are called at the software_frame cycle rate. The scheduled jobs include these job classes:
- integ_loop. The integ_loop includes derivative, integration, dynamic_event, and post_integration.
- automatic
- environment
- sensor
- sensor_receiver
- scheduled, effector
- effector_emitter
- effector_receiver
- automatic_last
- logging
- advance_sim_time
Receiving a freeze command diverts execution to the Freeze loop. Execution is resumed in the
scheduled loop when returing to run mode.
The loop is run until an exit condition is met. The exit conditions include receiving
an exit command, reaching the termination time, or a job returning a non-zero value to
"trick_ret". Upon meeting an exit condition, execution continues to the Shutdown phase.
@image html ../../../doxygen/images/executive_flowchart/freeze_loop.png
@image rtf ../../../doxygen/images/executive_flowchart/freeze_loop.png
<center><b>Figure Exec_3 Freeze Loop</b></center>
Execution is diverted to the Freeze loop when the Freeze command is set in the Scheduled
loop. When entering freeze the freeze_init jobs are called. The freeze loop cyclically
calls the freeze scheduled jobs, freeze_automatic, and freeze jobs. Elapsed freeze time
is kept and cycle times for the freeze_scheduled jobs are followed. freeze_automatic
jobs are assumed to set their own next call time. Freeze jobs are called at the end of
the freeze_software_frame. The loop is run until a run command is received or an exit
condtion is met. Upon receiving a run command, execution is returned to the Scheduled loop.
The exit conditions include receiving an exit command or a job returning a non-zero
value to "trick_ret". Upon meeting an exit condition, execution continues to the Shutdown phase.
@image html ../../../doxygen/images/executive_flowchart/shutdown.png
@image rtf ../../../doxygen/images/executive_flowchart/shutdown.png
<center><b>Figure Exec_4 Shutdown</b></center>
The shutdown phase calls the shutdown phase jobs and exits. Easy peasy.
@section LEVEL3 Executive Time
Accessing the simulation time is one of the more common user interactions with the
executive. The executive keeps track of only simulation time. The executive does not keep track of
realtime. That is the job of Trick::RealtimeSync. The executive also does not keep track of
other times including mission elapsed time, universal time, or other model generated time.
@section LEVEL4 Reading the Time
The executive provides 2 calls to get the simulation time.
@code
double exec_get_sim_time() ;
long long exec_get_time_tics() ;
@endcode
exec_get_sim_time() is a returns a double representing simulation elapsed seconds.
@note exec_get_sim_time() will not return an exact value when time values are large due to roundoff error.
exec_get_time_tics() returns the number of elapsed time tics. The time tic is explained below.
@section LEVEL4 Setting the Time
Trick provides 2 calls to set the simulation time.
@code
int exec_set_time( double in_time ) ;
int exec_set_time_tics( long long in_time_tics ) ;
@endcode
exec_set_time() sets the current simulation time to the incoming value. This should normally be done
at initialization. Doing this while running is not defined.
exec_set_time_tics() sets the simulation time based on time tics. The time tic is explained below.
@section LEVEL4 Time tic
@code
int exec_get_time_tic_value( void ) ;
int exec_set_time_tic_value( int num_of_tics_per_second ) ;
@endcode
A time tic is a fraction of a second equal to 1s/<number_of_tics_per_second>. The time tic default is 1us.
Scheduled job cycles are converted to numbers of tics. For instance a 1ms cycle job will be run every
1000 tics. All job cycles should be a multiple of the time tic value. If they are not the number of
time tics is rounded down to the nearest number of tics. The time tic value can be changed with the
exec_set_time_tic_value() call.
Example 1: A job is set to run at 128Hz or at 0.0078125 seconds. This is converted to 7812.5 tics using
the default tic value. The job will end up running incorrectly every 7812 tics. To correct this we
call exec_set_time_tic_value(10000000) to set the time tic value to 100ns. With a 100ns time tic value
the job will be called every 78125 tics, which is (10000000/78125) = 128Hz.
Example 2: Job 1 is set to run at 300Hz, 0.003333 seconds. This is converted to 3333 tics and will
not run at exactly 300Hz using the default 1us time tic. Job 2 is set to run at 100Hz. To run
both jobs at their correct rates, we call exec_set_time_tic_value(3000000). Job 1 will run every
10000 tics. Job 2 will run every 30000 tics.
@section LEVEL3 Software Frame
The software frame sets the cycle rates of the top_of_frame and end_of_frame class jobs.
Jobs run at the software frame are used to synchronize to real-time, synchronize multiple simulations
together, and inject variables values. The scheduled loop and the freeze loop have separate software
frame values. They are read and written with the following commands.
@code
int exec_set_software_frame(double) ;
double exec_get_software_frame() ;
long long exec_get_software_frame_tics() ;
int exec_set_freeze_frame(double) ;
double exec_get_freeze_frame() ;
long long exec_get_freeze_frame_tics() ;
@endcode
exec_set_software_frame(double) and exec_set_freeze_frame(double) both take frame values in seconds.
exec_get_software_frame() and exec_get_freeze_frame() return the frame values in seconds.
exec_get_software_frame_tics() and exec_get_freeze_frame_tics() return the frame values in number of tics.
@section LEVEL3 User Mode Control
User's may toggle the Executive mode between Freeze and Run as well as command simulation Shutdown.
@section LEVEL4 Getting the current mode
@code
# Python code
SIM_MODE trick.exec_get_mode()
@endcode
exec_get_mode returns the current mode of the simulation. See the SIM_MODE definition for the
enumeration values.
@section LEVEL4 Commanding to Freeze
@code
// C/C++ code
#include "sim_services/Executive/include/exec_proto.h"
exec_freeze()
@endcode
@code
# Python code
trick.freeze()
trick.freeze(<double freeze_time>)
@endcode
Users may command freeze in model code by calling exec_freeze(). Inside the input file user's
may call freeze with an optional time argument. Calling trick.freeze() without an argument will
freeze the simulation when all jobs at the current time step are complete. Using a freeze_time
argument will freeze the simulation in the future. The freeze_time argument must be greater
than the current time. The freeze_time argument is elapsed simulation seconds.
@code
# Python code
trick.exec_set_freeze_command(int on_off)
@endcode
Calling exec_set_freeze_command() with a non-zero argument will start the simulation in freeze.
This is here to replicate a call from previous versions of Trick. A call to trick.freeze() in
the input file will accomplish the same result. A call to trick.freeze(0.0) will not work. This
is because freeze times must be greater than the current sim time.
@code
# Python code
trick.exec_set_freeze_on_frame_boundary(int on_off)
@endcode
Calling exec_set_freeze_on_frame_boundary() with a non-zero argument will force all freezes to
synchronize with the software frame. Freeze commands received in the middle of the frame will
be saved and executed at the next avaialbe frame boundary.
@code
# Python code
trick.exec_set_enable_freeze( int on_off )
int trick.exec_get_enable_freeze() ;
@endcode
Calling exec_set_enable_freeze() with a non-zero argument will enable the CTRL-C keyboard interrupt
signal to freeze/unfreeze the simulation. The default is off. When enable_freeze is off, a CTRL-C
keystroke will terminate the simulation.
@section LEVEL4 Commanding to Run
@code
exec_run()
@endcode
exec_run() is called when the Run button on an attached sim_control panel is pressed. It is rare that a
user job calls exec_run.
@section LEVEL4 Commanding to Shutdown
@code
# Python code
trick.exec_set_terminate_time(double time_value)
double trick.exec_get_terminate_time() ;
@endcode
The most common way to shutdown a simulation is setting a terminate time. When simulation elapsed
time reaches the terminate time, the shutdown jobs are executed and the program terminates.
@code
exec_terminate(const char *file_name, const char *error)
exec_terminate_with_return(int ret_code, const char *file_name, int line, const char *error)
@endcode
Users may terminate simulation execution at any time by calling one of the exec_terminate varieties.
Both versions set the the Executive mode to shutdown. The exec_terminate routine The
exec_terminate_with_return allows users to set the simulation exit code as well as include a line
number in the error message.
@section LEVEL3 Job Control
The executive provides routines to control job execution.
@section LEVEL4 Turning Jobs On/Off
@code
# Python code
trick.exec_set_job_onoff(char * job_name , int instance_num , int on) ;
@endcode
The exec_set_job_onoff() routine allows users to turn individual jobs on and off. The job name is
the concatenation of the sim_object name and the job name. The instance_num argument is used
when jobs have the same name. Jobs with unique names are always instance_num 1. Jobs with
identical names start at instance_num 1 and increment for each job as they appear in the
sim object.
If there is a job tag specified for one of more jobs in the S_define file, you can turn all jobs
with that tag on/off by specifying the tag as the job_name argument.
@code
# Python code
trick.exec_set_sim_object_onoff(char * job_name , int on) ;
@endcode
The exec_set_sim_object_onoff routine allows users to turn individual whole sim_objects on and off.
@section LEVEL4 Job Cycle Time
@code
# Python code
trick.exec_set_job_cycle(char * job_name, int instance_num, double in_cycle)
double trick.exec_get_job_cycle(char * job_name)
@endcode
Each job cycle time is availabe to read and to set. Some job classes ignore cycle times, i.e. initialization
and shutdown. The user may change the cycle time set in the S_define file with the exec_set_job_cycle() call.
The instance_num argument is used when jobs have the same name (see exec_set_job_onoff above).
If there is a job tag specified for one of more jobs in the S_define file, you can set the cycle for all jobs
with that tag by specifying the tag as the job_name argument.
@section LEVEL3 Thread Control
Jobs may be assigned to specific threads. See the Simulation Definition File -> Child Thread Specification section
for information about assigning jobs to threads.
Trick provides 3 types of threads. Each thread type synchronizes to the main threads differently. Setting the
thread process type determines how the executive synchronizes child threads to run.
@code
# Python code
trick.exec_set_thread_process_type( unsigned int thread_id , int process_type )
@endcode
Sets the synchronization type of a child thread. There are three synchronization types, PROCESS_TYPE_SCHEDULED,
PROCESS_TYPE_ASYNC_CHILD, and PROCESS_TYPE_AMF_CHILD. See the Trick::ProcessType.
@section LEVEL4 PROCESS_TYPE_SCHEDULED Threads
Jobs in scheduled threads run in step with the main thread. The main thread will wait for jobs in scheduled
threads to finish before advancing to the simulation time step. Scheduled thread simulation time always matches
the main thread.
@section LEVEL4 PROCESS_TYPE_ASYNC_CHILD Threads
Asynchronous threads have 2 modes of operation depending on if a synchronization time is specified. A cycle
syncrhonization time is set through the exec_set_thread_async_cycle_time call.
@code
# Python code
trick.exec_set_thread_async_cycle_time( unsigned int thread_id , double cycle_time ) ;
@endcode
If the synchronization cycle time is set to zero, then the asynchronous threads do not synchronize to the main thread.
Asynchronous jobs are often infinite loops. If all jobs in an asynchronous thread finish, the thread is
immediately restarted. Execution frequencies for jobs in asynchronous threads are ignored. All jobs are run
each time an asynchronous thread is executed. Asynchronous thread simulation time is set to the thread start
time and is not updated while the thread is executing.
If the synchronization cycle time is non zero, then the asynchronous thread attempts to synchronize to the main thread.
At the end of the synchronization cycle if the asynchronous thread has completed, then it will be triggered to run its
next frame of jobs. If the asynchronous thread has not completed, then it will not be triggered to run the next frame
of jobs. This condition is not considered an overrun and is not logged as one. The non completed thread will be
checked at the next syncrhonization cycle time for completion. Between synchronization times, async threads maintain
their own simulation time. The simulation time on the thread may not match the main thread.
@section LEVEL4 PROCESS_TYPE_AMF_CHILD Threads
AMF stands for Asynchronous Must Finish. Threads of this type synchronize to the main thread at regular
intervals set by exec_set_thread_amf_cycle_time(). Between synchronizations, AMF threads maintain their
own simulation time. Jobs in AMF threads are run as fast as possible. The AMF thread simulation time may
not match the main thread
When using an AMF thread the executive needs to know how often to synchronize with the child thread.
@code
# Python code
trick.exec_set_thread_amf_cycle_time( unsigned int thread_id , double cycle_time ) ;
@endcode
exec_set_thread_amf_cycle_time sets the synchronization cycle rate with the main thread.
@section LEVEL4 Main Thread CPU release.
@code
# Python code
trick.exec_set_rt_nap(int on_off)
trick.exec_get_rt_nap()
@endcode
While the main thread is waiting for child threads to finish execution, it furiously spins waiting for
them to finish. Calling exec_set_rt_nap() with a non-zero argument will tell the main thread to
momentarily give up the CPU if it needs to wait for child threads to finish.
@section LEVEL4 Asynchronous Threads at Shutdown
@code
# Python code
trick.exec_set_thread_async_wait( unsigned int thread_id , int yes_no ) ;
@endcode
By default the executive does not wait for asynchronous or AMF threads during shutdown. Calling
exec_set_thread_async_wait() will tell the executive to wait for a thread before calling shutdown
routines and exiting.
@section LEVEL4 Thread Priorities
@code
# Python code
trick.exec_set_thread_priority(unsigned int thread_id , unsigned int req_priority)
@endcode
exec_set_thread_priority() will set the thread's priority. The main thread is thread_id=0, 1-n are the
child threads. for req_priority, 1 is the highest priority. This number is converted internally
to the highest priority of the system automatically.
@note Setting a simulation to run as priority 1 may lock out keyboard and mouse processing.
@code
# Python code
trick.exec_set_thread_cpu_affinity(unsigned int thread_id , int cpu_num) ;
@endcode
exec_set_thread_cpu_affinity assigns a thread to a specific CPU. The main thread is thread_id=0. 1-n are
the child threads. Setting a thread to run on a specific CPU does not exclude other processes to
continue to run on the same CPU.
@section LEVEL4 Thread Priorities
@code
# Python code
trick.exec_set_lock_memory(int yes_no) ;
@endcode
Lock all of the process memory for best real-time performance. This prevents process memory to be
swapped out to virtual memory on disk. This option may move to the Trick::RealtimeSync class in
the future.
@section LEVEL4 Thread Job Dependencies
@code
# Python code
trick.exec_add_depends_on_job(char * target_job_string , unsigned int t_instance ,
char * depend_job_string , unsigned int d_instance )
@endcode
Jobs in different threads may need other jobs in other threads to run first before executing.
Trick provides a depends_on feature. Jobs that depend on other jobs will not execute until all
dependencies have finished. The instance value in the above call to take care of the case
where the same job name is called multiple times in a sim_object. Instance values start at 1.
@section LEVEL4 Getting Thread ID
@code
unsigned int exec_get_process_id() ;
unsigned int exec_get_num_threads() ;
@endcode
exec_get_process_id() will return the current thread the caller is on. 0 = main thread, 1-n
are the child threads. If this call is issued through a variable server client, or a thread
that was not spawned by the executive, a -1 error code is returned.
exec_get_num_threads() returns the number of child threads spawned by the main thread.
@section LEVEL3 Debugging Help
The Executive provides several parameters that can help a model developer debug a simulation.
@section LEVEL4 Trapping signals
@code
# Python code
trick.exec_set_trap_sigbus(int on_off)
trick.exec_set_trap_sigfpe(int on_off)
trick.exec_set_trap_sigsegv(int on_off)
trick.exec_get_trap_sigbus()
trick.exec_get_trap_sigfpe()
trick.exec_get_trap_sigsegv()
@endcode
The set_trap routines listed above set a signal handler for the SIGBUS, SIGFPE, and SIGSEGV
signals respectively. The get_trap routines return the on/off status of the trap. Trapping
the signals allows the Trick to gracefully shutdown the simulation and to possibly write important
information about the signal before exitting execution. Turning off the traps will revert
signal handling to the default system signal handler. By default the traps for SIGBUS and
SIGSEGV are true. SIGFPE is not trapped by default.
@section LEVEL4 Printing a Stack (Call) Trace on Signal
@code
# Python code
trick.exec_set_stack_trace(int on_off)
@endcode
This is a Linux only option. By default, printing a stack trace when a signal is trapped
is enabled. When a signal is trapped a debugger is automatically connected to the
running simulation and a printout of the calling stack is printed before the simulation
exits.
@section LEVEL4 Attaching a debugger on Signal
@code
# Python code
trick.exec_set_attach_debugger(int on_off)
@endcode
This is a Linux only option. By default, this option is off. If enabled, when a signal
is trapped a debugger is automatically connected to the running simulation and an interactive
debugging session is presented to the user.
The debugger executable may be set with the following. The default is "/usr/bin/gdb".
@code
# Python code
trick.exec_set_debugger_command(char * command)
char * trick.exec_get_debugger_command()
@endcode
@section LEVEL4 Getting Build Information
@code
# Python code
char * trick.exec_get_current_version()
@endcode
The executive stores the Trick version that was used to build the executable. Use
exec_get_current_version() to return the version string.
@section LEVEL3 Executive Scheduler Related Pages
@subpage scheduler_requirements "Scheduler Requirements"\n
@subpage scheduler_design "Scheduler Design"
*/
#include "requirements.dox"
#include "design.dox"