Added: #1440 Initial handling of build abort

--HG--
branch : build_pipeline_v3
hg/feature/build_pipeline_v3
kaetemi 13 years ago
parent 3c236fc336
commit aff9193e48

@ -58,6 +58,7 @@ namespace PIPELINE {
// permanent flags // permanent flags
#define PIPELINE_INFO_CODE_ERROR_UNMACRO "#CODE_ERROR_UNMACRO" #define PIPELINE_INFO_CODE_ERROR_UNMACRO "#CODE_ERROR_UNMACRO"
#define PIPELINE_INFO_SLAVE_REJECTED "#M_SLAVE_REJECT" #define PIPELINE_INFO_SLAVE_REJECTED "#M_SLAVE_REJECT"
#define PIPELINE_INFO_SLAVE_ABORTED "#M_SLAVE_ABORTED"
#define PIPELINE_INFO_SLAVE_CRASHED "#M_SLAVE_CRASH" #define PIPELINE_INFO_SLAVE_CRASHED "#M_SLAVE_CRASH"
#define PIPELINE_INFO_SLAVE_NOT_READY "#M_SLAVE_NOT_R" #define PIPELINE_INFO_SLAVE_NOT_READY "#M_SLAVE_NOT_R"
#define PIPELINE_INFO_SLAVE_CB_GONE "#M_SLAVE_CB_GONE" #define PIPELINE_INFO_SLAVE_CB_GONE "#M_SLAVE_CB_GONE"
@ -271,6 +272,7 @@ public:
// if it goes down while busy on a task it crashed (or was poorly stopped by user...) // if it goes down while busy on a task it crashed (or was poorly stopped by user...)
CInfoFlags::getInstance()->addFlag(PIPELINE_INFO_SLAVE_CRASHED); CInfoFlags::getInstance()->addFlag(PIPELINE_INFO_SLAVE_CRASHED);
// ... TODO ... // ... TODO ...
slaveAbortedBuildTask(moduleProxy); // see if this works
} }
m_Slaves.erase(slaveIt); m_Slaves.erase(slaveIt);
@ -392,14 +394,26 @@ public:
/////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////
/// When the slave task finishes, with or without error
virtual void slaveFinishedBuildTask(NLNET::IModuleProxy *sender, uint8 errorLevel) virtual void slaveFinishedBuildTask(NLNET::IModuleProxy *sender, uint8 errorLevel)
{ {
// TODO // TODO
} }
/// When the user aborts slave-side, when slave-side exits, etc
virtual void slaveAbortedBuildTask(NLNET::IModuleProxy *sender) virtual void slaveAbortedBuildTask(NLNET::IModuleProxy *sender)
{ {
// TODO // TODO
//m_SlavesMutex.lock();
CSlave *slave = m_Slaves[sender];
if (slave == NULL) { nlerror("Received 'slaveAbortedBuildTask' from unknown slave at '%s'", sender->getModuleName().c_str()); m_Slaves.erase(sender); /*m_SlavesMutex.unlock();*/ return; }
//m_SlavesMutex.unlock();
m_BuildTaskQueue.abortedTask(slave->ActiveTaskId);
slave->ActiveTaskId = 0;
// --slave->SaneBehaviour; // legal behaviour
// slave->TimeOutStamp = NLMISC::CTime::getSecondsSince1970() + 30; // timeout for 30 seconds on this slave
CInfoFlags::getInstance()->addFlag(PIPELINE_INFO_SLAVE_ABORTED);
// TODO
} }
// in fact slaves are not allowed to refuse tasks, but they may do this if the user is toying around with the slave service // in fact slaves are not allowed to refuse tasks, but they may do this if the user is toying around with the slave service
@ -475,6 +489,7 @@ public:
nlwarning("Slave disconnected before callback could be delivered"); nlwarning("Slave disconnected before callback could be delivered");
CInfoFlags::getInstance()->removeFlag(PIPELINE_INFO_MASTER_UPDATE_DATABASE_FOR_SLAVE); CInfoFlags::getInstance()->removeFlag(PIPELINE_INFO_MASTER_UPDATE_DATABASE_FOR_SLAVE);
CInfoFlags::getInstance()->addFlag(PIPELINE_INFO_SLAVE_CB_GONE); CInfoFlags::getInstance()->addFlag(PIPELINE_INFO_SLAVE_CB_GONE);
delete this;
return; return;
} }

@ -56,6 +56,8 @@ namespace PIPELINE {
#define PIPELINE_INFO_STATUS_UPDATE_MASTER "S_ST_UPD_MASTER" #define PIPELINE_INFO_STATUS_UPDATE_MASTER "S_ST_UPD_MASTER"
#define PIPELINE_INFO_STATUS_UPDATE_SLAVE "S_ST_UPD_SLAVE" #define PIPELINE_INFO_STATUS_UPDATE_SLAVE "S_ST_UPD_SLAVE"
#define PIPELINE_INFO_ABORTING "S_ABORTING"
enum TRequestState enum TRequestState
{ {
REQUEST_NONE, REQUEST_NONE,
@ -97,8 +99,10 @@ public:
CPipelineProcessImpl *m_ActiveProcess; // TODO: Maybe it would be easier to go directly to CPipelineProject from the plugin and provide an interface therefore. CPipelineProcessImpl *m_ActiveProcess; // TODO: Maybe it would be easier to go directly to CPipelineProject from the plugin and provide an interface therefore.
CProcessPluginInfo m_ActivePlugin; CProcessPluginInfo m_ActivePlugin;
bool m_AbortRequested;
public: public:
CModulePipelineSlave() : m_Master(NULL), m_TestCommand(false), m_ReloadSheetsState(REQUEST_NONE), m_BuildReadyState(false), m_SlaveTaskState(IDLE_WAIT_MASTER), m_TaskManager(NULL), m_StatusUpdateMasterDone("StatusUpdateMasterDone"), m_StatusUpdateSlaveDone("StatusUpdateSlaveDone"), m_ActiveProject(NULL), m_ActiveProcess(NULL) CModulePipelineSlave() : m_Master(NULL), m_TestCommand(false), m_ReloadSheetsState(REQUEST_NONE), m_BuildReadyState(false), m_SlaveTaskState(IDLE_WAIT_MASTER), m_TaskManager(NULL), m_StatusUpdateMasterDone("StatusUpdateMasterDone"), m_StatusUpdateSlaveDone("StatusUpdateSlaveDone"), m_ActiveProject(NULL), m_ActiveProcess(NULL), m_AbortRequested(false)
{ {
NLMISC::CSynchronized<bool>::CAccessor(&m_StatusUpdateMasterDone).value() = false; NLMISC::CSynchronized<bool>::CAccessor(&m_StatusUpdateMasterDone).value() = false;
NLMISC::CSynchronized<bool>::CAccessor(&m_StatusUpdateSlaveDone).value() = false; NLMISC::CSynchronized<bool>::CAccessor(&m_StatusUpdateSlaveDone).value() = false;
@ -202,6 +206,13 @@ public:
{ {
if (NLMISC::CSynchronized<bool>::CAccessor(&m_StatusUpdateMasterDone).value() if (NLMISC::CSynchronized<bool>::CAccessor(&m_StatusUpdateMasterDone).value()
&& NLMISC::CSynchronized<bool>::CAccessor(&m_StatusUpdateSlaveDone).value()) && NLMISC::CSynchronized<bool>::CAccessor(&m_StatusUpdateSlaveDone).value())
{
if (m_AbortRequested)
{
nlinfo("Aborted slave task after status update");
finalizeAbort();
}
else
{ {
nlinfo("Slave task: Status update done"); nlinfo("Slave task: Status update done");
m_SlaveTaskState = SOMEWHERE_INBETWEEN; m_SlaveTaskState = SOMEWHERE_INBETWEEN;
@ -209,6 +220,7 @@ public:
// ... TODO ... // ... TODO ...
} }
} }
}
break; break;
} }
@ -288,6 +300,9 @@ public:
nlassert(m_ActiveProject == NULL); nlassert(m_ActiveProject == NULL);
nlassert(m_ActiveProcess == NULL); nlassert(m_ActiveProcess == NULL);
// TODO: ERROR HANDLING !!! (see sanity check above)
// master->slaveRefusedBuildTask(this);
// Set the task state somewhere inbetween // Set the task state somewhere inbetween
m_SlaveTaskState = SOMEWHERE_INBETWEEN; m_SlaveTaskState = SOMEWHERE_INBETWEEN;
@ -296,24 +311,43 @@ public:
m_ActiveProcess = new CPipelineProcessImpl(m_ActiveProject); m_ActiveProcess = new CPipelineProcessImpl(m_ActiveProject);
g_PipelineWorkspace->getProcessPlugin(m_ActivePlugin, pluginId); g_PipelineWorkspace->getProcessPlugin(m_ActivePlugin, pluginId);
// TODO: ERROR HANDLING !!! // TODO: ERROR HANDLING !!! (see sanity check above)
// master->slaveRefusedBuildTask(this);
//CModulePipelineMasterProxy master(sender);
//master.slaveRefusedBuildTask(this);
// Begin with status update of the dependent directories and files on the master, and previous output files on the slave // Begin with status update of the dependent directories and files on the master, and previous output files on the slave
beginTaskStatusUpdate(); beginTaskStatusUpdate();
} }
void finalizeAbort()
{
m_ActiveProject = NULL;
m_ActiveProcess = NULL;
m_SlaveTaskState = IDLE_WAIT_MASTER;
m_Master->slaveAbortedBuildTask(this);
m_AbortRequested = false;
CInfoFlags::getInstance()->removeFlag(PIPELINE_INFO_ABORTING);
}
/// Master or user request to abort.
virtual void abortBuildTask(NLNET::IModuleProxy *sender) virtual void abortBuildTask(NLNET::IModuleProxy *sender)
{
if (m_ActiveProject)
{ {
// Sender NULL is request from slave (user exit, command or master disconnect), otherwise request from master. // Sender NULL is request from slave (user exit, command or master disconnect), otherwise request from master.
nlassert(m_Master->getModuleProxy() == sender || sender == NULL); // sanity check nlassert(m_Master->getModuleProxy() == sender || sender == NULL); // sanity check
// TODO // ?TODO? Actually wait for the task manager etc to end before sending the aborted confirmation.
CInfoFlags::getInstance()->addFlag(PIPELINE_INFO_ABORTING);
m_AbortRequested = true;
// ?TODO?
//m_ActiveProject = NULL;
//m_ActiveProcess = NULL;
//m_Master->slaveAbortedBuildTask(this);
// ?TODO?
// KABOOM // KABOOM
// m_ActiveProject = NULL; }
} }
virtual void masterUpdatedDatabaseStatus(NLNET::IModuleProxy *sender) virtual void masterUpdatedDatabaseStatus(NLNET::IModuleProxy *sender)
@ -361,7 +395,7 @@ public:
virtual void leaveBuildReadyState(NLNET::IModuleProxy *sender) virtual void leaveBuildReadyState(NLNET::IModuleProxy *sender)
{ {
nlassert(m_Master->getModuleProxy() == sender); // sanity check nlassert(m_Master->getModuleProxy() == sender || sender == NULL); // sanity check
if (m_BuildReadyState) if (m_BuildReadyState)
{ {

Loading…
Cancel
Save