mirror of
https://github.com/nasa/trick.git
synced 2025-01-28 07:04:01 +00:00
2335f6bb69
Also change the Cannon sim monte_post jobs to use mc_read/write instead of tc_read/write Refs #396
197 lines
6.9 KiB
C++
197 lines
6.9 KiB
C++
#include "trick/MonteCarlo.hh"
|
|
#include "trick/message_proto.h"
|
|
#include "trick/message_type.h"
|
|
#include "trick/tc_proto.h"
|
|
|
|
/**
|
|
* @par Detailed Design:
|
|
* This function performs a non-blocking accept on the listening socket.
|
|
*/
|
|
void Trick::MonteCarlo::receive_results() {
|
|
|
|
/** <li> While there are pending connections: */
|
|
while (tc_accept(&listen_device, &connection_device) == TC_SUCCESS) {
|
|
|
|
/** <ul><li> Read the slave id. */
|
|
int id;
|
|
tc_read(&connection_device, (char*)&id, (int)sizeof(id));
|
|
id = ntohl(id);
|
|
MonteSlave* slave = get_slave(id);
|
|
if (!slave) {
|
|
message_publish(
|
|
MSG_ERROR,
|
|
"Monte [Master] Slave returned an invalid id (%d)\n",
|
|
id) ;
|
|
tc_disconnect(&connection_device);
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* <li> If the slave is in the INITIALIZING state, it is sending us the
|
|
* machine name and port over which it is listening for new runs.
|
|
*/
|
|
if (slave->state == MonteSlave::INITIALIZING) {
|
|
handle_initialization(*slave);
|
|
}
|
|
/** <li> Otherwise, it's sending us run data. */
|
|
else {
|
|
handle_run_data(*slave);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Trick::MonteCarlo::handle_initialization(Trick::MonteSlave& slave) {
|
|
if (verbosity >= ALL) {
|
|
message_publish(
|
|
MSG_INFO,
|
|
"Monte [Master] Receiving initialization information from %s:%d.\n",
|
|
slave.machine_name.c_str(), slave.id);
|
|
}
|
|
|
|
int size;
|
|
if (tc_read(&connection_device, (char*)&size, (int)sizeof(size)) != (int)sizeof(size)) {
|
|
set_disconnected_state(slave);
|
|
return;
|
|
}
|
|
size = ntohl(size);
|
|
|
|
char name[size + 1];
|
|
name[size] = '\0';
|
|
if (tc_read(&connection_device, name, size) != size) {
|
|
set_disconnected_state(slave);
|
|
return;
|
|
}
|
|
slave.machine_name = std::string(name);
|
|
|
|
size = (int)sizeof(slave.port);
|
|
if (tc_read(&connection_device, (char*)&slave.port, size) != size) {
|
|
set_disconnected_state(slave) ;
|
|
return;
|
|
}
|
|
slave.port = ntohl(slave.port);
|
|
|
|
slave.state = MonteSlave::READY;
|
|
tc_disconnect(&connection_device);
|
|
}
|
|
|
|
void Trick::MonteCarlo::handle_run_data(Trick::MonteSlave& slave) {
|
|
if (verbosity >= INFORMATIONAL) {
|
|
message_publish(MSG_INFO, "Monte [Master] Receiving results for run %d from %s:%d.\n",
|
|
slave.current_run->id, slave.machine_name.c_str(), slave.id) ;
|
|
}
|
|
|
|
/**
|
|
* <ul><li> This run may have already been resolved by another slave if
|
|
* this slave was marked as having timed out. If that is the case,
|
|
* discard these results.
|
|
*/
|
|
if (slave.current_run->exit_status != MonteRun::INCOMPLETE) {
|
|
// TODO: If a slave times out or core dumps in it's monte_slave_post
|
|
// jobs, the master will receive a COMPLETE status from the slave's
|
|
// child process and then an error status from the parent, rendering
|
|
// this message incorrect.
|
|
if (verbosity >= ALL) {
|
|
message_publish(
|
|
MSG_INFO,
|
|
"Monte [Master] Run %d has already been resolved. Discarding results.\n",
|
|
slave.current_run->id) ;
|
|
}
|
|
tc_disconnect(&connection_device);
|
|
return;
|
|
}
|
|
|
|
/** <li> Otherwise, check the exit status: */
|
|
int exit_status;
|
|
int size = sizeof(exit_status);
|
|
if (tc_read(&connection_device, (char*)&exit_status, size) != size) {
|
|
set_disconnected_state(slave) ;
|
|
return;
|
|
}
|
|
exit_status = ntohl(exit_status);
|
|
|
|
switch (exit_status) {
|
|
|
|
case MonteRun::COMPLETE:
|
|
resolve_run(slave, MonteRun::COMPLETE);
|
|
run_queue(&master_post_queue, "in master_post queue") ;
|
|
break;
|
|
|
|
case MonteRun::BAD_INPUT:
|
|
if (verbosity >= ERROR) {
|
|
message_publish(
|
|
MSG_ERROR,
|
|
"Monte [Master] %s:%d reported bad input for run %d. Skipping.\n",
|
|
slave.machine_name.c_str(), slave.id, slave.current_run->id) ;
|
|
}
|
|
resolve_run(slave, MonteRun::BAD_INPUT);
|
|
break;
|
|
|
|
case MonteRun::CORED:
|
|
if (verbosity >= ERROR) {
|
|
message_publish(
|
|
MSG_ERROR,
|
|
"Monte [Master] %s:%d reported core dump for run %d. Skipping.\n",
|
|
slave.machine_name.c_str(), slave.id, slave.current_run->id) ;
|
|
}
|
|
resolve_run(slave, MonteRun::CORED);
|
|
break;
|
|
|
|
case MonteRun::NO_PERM:
|
|
if (verbosity >= ERROR) {
|
|
message_publish(
|
|
MSG_ERROR,
|
|
"Monte [Master] %s:%d reported a failure to create output directories for run %d.\n",
|
|
slave.machine_name.c_str(), slave.id, slave.current_run->id);
|
|
}
|
|
handle_retry(slave, MonteRun::NO_PERM);
|
|
break;
|
|
|
|
/**
|
|
* <li> Timeouts and are redispatched. However, we must first check to
|
|
* see if this run has already been processed in #check_timeouts, which
|
|
* can occur when the master determines that a slave has timed out, and
|
|
* then that slave itself reports a timeout. </ul>
|
|
*/
|
|
case MonteRun::TIMEDOUT:
|
|
if (verbosity >= ERROR) {
|
|
message_publish(
|
|
MSG_ERROR,
|
|
"Monte [Master] %s:%d reported a timeout for run %d.\n",
|
|
slave.machine_name.c_str(), slave.id, slave.current_run->id);
|
|
}
|
|
if (slave.state != MonteSlave::UNRESPONSIVE_RUNNING &&
|
|
slave.state != MonteSlave::UNRESPONSIVE_STOPPING) {
|
|
handle_retry(slave, MonteRun::TIMEDOUT);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
if (verbosity >= ERROR) {
|
|
message_publish(
|
|
MSG_ERROR,
|
|
"Monte [Master] %s:%d reported unrecognized exit status (%d) for run %d. Skipping.\n",
|
|
slave.machine_name.c_str(), slave.id, exit_status, slave.current_run->id);
|
|
}
|
|
resolve_run(slave, MonteRun::UNKNOWN);
|
|
break;
|
|
}
|
|
|
|
tc_disconnect(&connection_device);
|
|
|
|
/** <li> Update the slave's state. */
|
|
if (slave.state == MonteSlave::RUNNING || slave.state == MonteSlave::UNRESPONSIVE_RUNNING) {
|
|
slave.state = MonteSlave::READY;
|
|
} else if (slave.state == MonteSlave::STOPPING || slave.state == MonteSlave::UNRESPONSIVE_STOPPING) {
|
|
slave.state = MonteSlave::STOPPED;
|
|
}
|
|
}
|
|
|
|
void Trick::MonteCarlo::set_disconnected_state(Trick::MonteSlave& slave) {
|
|
slave.state = Trick::MonteSlave::DISCONNECTED;
|
|
if (verbosity >= ERROR) {
|
|
message_publish(MSG_ERROR, "Monte [Master] Lost connection to %s:%d.\n",
|
|
slave.machine_name.c_str(), slave.id) ;
|
|
}
|
|
tc_disconnect(&connection_device);
|
|
}
|