MXS-2002 Make Worker excecution mode explicit
This is the first step in some cleanup of the Worker interface. The execution mode must now be explicitly specified, but that is just a temporary step. Further down the road, _posting_ will *always* mean via the message loop while _executing_ will optionally and by default mean direct execution if the calling thread is that of the worker.
This commit is contained in:
@ -730,7 +730,12 @@ public:
|
|||||||
* MyResult& result = task.result();
|
* MyResult& result = task.result();
|
||||||
* @endcode
|
* @endcode
|
||||||
*/
|
*/
|
||||||
bool post(Task* pTask, Semaphore* pSem = NULL, enum execute_mode_t mode = EXECUTE_AUTO);
|
bool post(Task* pTask, Semaphore* pSem, enum execute_mode_t mode);
|
||||||
|
|
||||||
|
bool post(Task* pTask, enum execute_mode_t mode)
|
||||||
|
{
|
||||||
|
return post(pTask, NULL, mode);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Posts a task to a worker for execution.
|
* Posts a task to a worker for execution.
|
||||||
@ -742,10 +747,10 @@ public:
|
|||||||
*
|
*
|
||||||
* @attention Once the task has been executed, it will be deleted.
|
* @attention Once the task has been executed, it will be deleted.
|
||||||
*/
|
*/
|
||||||
bool post(std::auto_ptr<DisposableTask> sTask, enum execute_mode_t mode = EXECUTE_AUTO);
|
bool post(std::auto_ptr<DisposableTask> sTask, enum execute_mode_t mode);
|
||||||
|
|
||||||
template<class T>
|
template<class T>
|
||||||
bool post(std::auto_ptr<T> sTask, enum execute_mode_t mode = EXECUTE_AUTO)
|
bool post(std::auto_ptr<T> sTask, enum execute_mode_t mode)
|
||||||
{
|
{
|
||||||
return post(std::auto_ptr<DisposableTask>(sTask.release()), mode);
|
return post(std::auto_ptr<DisposableTask>(sTask.release()), mode);
|
||||||
}
|
}
|
||||||
@ -768,10 +773,11 @@ public:
|
|||||||
* semaphore.
|
* semaphore.
|
||||||
*
|
*
|
||||||
* @param func Function to execute
|
* @param func Function to execute
|
||||||
|
* @param mode Execution mode
|
||||||
*
|
*
|
||||||
* @return True if function was executed on the worker
|
* @return True if function was executed on the worker
|
||||||
*/
|
*/
|
||||||
bool execute(GenericFunction func);
|
bool execute(GenericFunction func, enum execute_mode_t mode);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Post a message to a worker.
|
* Post a message to a worker.
|
||||||
@ -933,7 +939,7 @@ protected:
|
|||||||
pTask->dec_ref();
|
pTask->dec_ref();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool post_disposable(DisposableTask* pTask, enum execute_mode_t mode = EXECUTE_AUTO);
|
bool post_disposable(DisposableTask* pTask, enum execute_mode_t mode);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Called by Worker::run() before starting the epoll loop.
|
* Called by Worker::run() before starting the epoll loop.
|
||||||
|
@ -177,6 +177,12 @@ public:
|
|||||||
* should either not have any sharable data or then it should
|
* should either not have any sharable data or then it should
|
||||||
* have data specific to each worker that can be accessed
|
* have data specific to each worker that can be accessed
|
||||||
* without locks.
|
* without locks.
|
||||||
|
*
|
||||||
|
* @attention The task will be posted to each routing worker using the
|
||||||
|
* EXECUTE_AUTO execution mode. That is, if the calling thread
|
||||||
|
* is that of a routing worker, then the task will be executed
|
||||||
|
* directly without going through the message loop of the worker,
|
||||||
|
* otherwise the task is delivered via the message loop.
|
||||||
*/
|
*/
|
||||||
static size_t broadcast(Task* pTask, Semaphore* pSem = NULL);
|
static size_t broadcast(Task* pTask, Semaphore* pSem = NULL);
|
||||||
|
|
||||||
@ -194,6 +200,12 @@ public:
|
|||||||
*
|
*
|
||||||
* @attention Once the task has been executed by all workers, it will
|
* @attention Once the task has been executed by all workers, it will
|
||||||
* be deleted.
|
* be deleted.
|
||||||
|
*
|
||||||
|
* @attention The task will be posted to each routing worker using the
|
||||||
|
* EXECUTE_AUTO execution mode. That is, if the calling thread
|
||||||
|
* is that of a routing worker, then the task will be executed
|
||||||
|
* directly without going through the message loop of the worker,
|
||||||
|
* otherwise the task is delivered via the message loop.
|
||||||
*/
|
*/
|
||||||
static size_t broadcast(std::auto_ptr<DisposableTask> sTask);
|
static size_t broadcast(std::auto_ptr<DisposableTask> sTask);
|
||||||
|
|
||||||
@ -215,6 +227,12 @@ public:
|
|||||||
* @warning This function is extremely inefficient and will be slow compared
|
* @warning This function is extremely inefficient and will be slow compared
|
||||||
* to the other functions. Only use this function when printing thread-specific
|
* to the other functions. Only use this function when printing thread-specific
|
||||||
* data to stdout.
|
* data to stdout.
|
||||||
|
*
|
||||||
|
* @attention The task will be posted to each routing worker using the
|
||||||
|
* EXECUTE_AUTO execution mode. That is, if the calling thread
|
||||||
|
* is that of a routing worker, then the task will be executed
|
||||||
|
* directly without going through the message loop of the worker,
|
||||||
|
* otherwise the task is delivered via the message loop.
|
||||||
*/
|
*/
|
||||||
static size_t execute_serially(Task& task);
|
static size_t execute_serially(Task& task);
|
||||||
|
|
||||||
@ -226,6 +244,12 @@ public:
|
|||||||
* @param task The task to be executed.
|
* @param task The task to be executed.
|
||||||
*
|
*
|
||||||
* @return How many workers the task was posted to.
|
* @return How many workers the task was posted to.
|
||||||
|
*
|
||||||
|
* @attention The task will be posted to each routing worker using the
|
||||||
|
* EXECUTE_AUTO execution mode. That is, if the calling thread
|
||||||
|
* is that of a routing worker, then the task will be executed
|
||||||
|
* directly without going through the message loop of the worker,
|
||||||
|
* otherwise the task is delivered via the message loop.
|
||||||
*/
|
*/
|
||||||
static size_t execute_concurrently(Task& task);
|
static size_t execute_concurrently(Task& task);
|
||||||
|
|
||||||
|
@ -1280,7 +1280,7 @@ HttpResponse resource_handle_request(const HttpRequest& request)
|
|||||||
mxs::Semaphore sem;
|
mxs::Semaphore sem;
|
||||||
ResourceTask task(request);
|
ResourceTask task(request);
|
||||||
|
|
||||||
worker->post(&task, &sem);
|
worker->post(&task, &sem, mxs::Worker::EXECUTE_AUTO);
|
||||||
sem.wait();
|
sem.wait();
|
||||||
|
|
||||||
return task.result();
|
return task.result();
|
||||||
|
@ -604,7 +604,7 @@ size_t RoutingWorker::broadcast(Task* pTask, Semaphore* pSem)
|
|||||||
Worker* pWorker = this_unit.ppWorkers[i];
|
Worker* pWorker = this_unit.ppWorkers[i];
|
||||||
ss_dassert(pWorker);
|
ss_dassert(pWorker);
|
||||||
|
|
||||||
if (pWorker->post(pTask, pSem))
|
if (pWorker->post(pTask, pSem, EXECUTE_AUTO))
|
||||||
{
|
{
|
||||||
++n;
|
++n;
|
||||||
}
|
}
|
||||||
@ -627,7 +627,7 @@ size_t RoutingWorker::broadcast(std::auto_ptr<DisposableTask> sTask)
|
|||||||
RoutingWorker* pWorker = this_unit.ppWorkers[i];
|
RoutingWorker* pWorker = this_unit.ppWorkers[i];
|
||||||
ss_dassert(pWorker);
|
ss_dassert(pWorker);
|
||||||
|
|
||||||
if (pWorker->post_disposable(pTask))
|
if (pWorker->post_disposable(pTask, EXECUTE_AUTO))
|
||||||
{
|
{
|
||||||
++n;
|
++n;
|
||||||
}
|
}
|
||||||
@ -650,7 +650,7 @@ size_t RoutingWorker::execute_serially(Task& task)
|
|||||||
RoutingWorker* pWorker = this_unit.ppWorkers[i];
|
RoutingWorker* pWorker = this_unit.ppWorkers[i];
|
||||||
ss_dassert(pWorker);
|
ss_dassert(pWorker);
|
||||||
|
|
||||||
if (pWorker->post(&task, &sem))
|
if (pWorker->post(&task, &sem, EXECUTE_AUTO))
|
||||||
{
|
{
|
||||||
sem.wait();
|
sem.wait();
|
||||||
++n;
|
++n;
|
||||||
@ -894,7 +894,7 @@ bool RoutingWorker::get_qc_stats(int id, QC_CACHE_STATS* pStats)
|
|||||||
{
|
{
|
||||||
Semaphore sem;
|
Semaphore sem;
|
||||||
Task task(pStats);
|
Task task(pStats);
|
||||||
pWorker->post(&task, &sem);
|
pWorker->post(&task, &sem, EXECUTE_AUTO);
|
||||||
sem.wait();
|
sem.wait();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1000,6 +1000,14 @@ std::unique_ptr<json_t> RoutingWorker::get_qc_stats_as_json(const char* zHost)
|
|||||||
return std::unique_ptr<json_t>(mxs_json_resource(zHost, MXS_JSON_API_QC_STATS, sAll_stats.release()));
|
return std::unique_ptr<json_t>(mxs_json_resource(zHost, MXS_JSON_API_QC_STATS, sAll_stats.release()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// static
|
||||||
|
RoutingWorker* RoutingWorker::pick_worker()
|
||||||
|
{
|
||||||
|
static int id_generator = 0;
|
||||||
|
int id = this_unit.id_min_worker + (atomic_add(&id_generator, 1) % this_unit.nWorkers);
|
||||||
|
return get(id);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t mxs_rworker_broadcast_message(uint32_t msg_id, intptr_t arg1, intptr_t arg2)
|
size_t mxs_rworker_broadcast_message(uint32_t msg_id, intptr_t arg1, intptr_t arg2)
|
||||||
@ -1197,7 +1205,7 @@ json_t* mxs_rworker_to_json(const char* zHost, int id)
|
|||||||
WorkerInfoTask task(zHost, id + 1);
|
WorkerInfoTask task(zHost, id + 1);
|
||||||
mxs::Semaphore sem;
|
mxs::Semaphore sem;
|
||||||
|
|
||||||
target->post(&task, &sem);
|
target->post(&task, &sem, mxs::Worker::EXECUTE_AUTO);
|
||||||
sem.wait();
|
sem.wait();
|
||||||
|
|
||||||
return task.resource(id);
|
return task.resource(id);
|
||||||
@ -1209,11 +1217,3 @@ json_t* mxs_rworker_list_to_json(const char* host)
|
|||||||
RoutingWorker::execute_concurrently(task);
|
RoutingWorker::execute_concurrently(task);
|
||||||
return task.resource();
|
return task.resource();
|
||||||
}
|
}
|
||||||
|
|
||||||
// static
|
|
||||||
RoutingWorker* RoutingWorker::pick_worker()
|
|
||||||
{
|
|
||||||
static int id_generator = 0;
|
|
||||||
int id = this_unit.id_min_worker + (atomic_add(&id_generator, 1) % this_unit.nWorkers);
|
|
||||||
return get(id);
|
|
||||||
}
|
|
||||||
|
@ -359,7 +359,8 @@ static void session_free(MXS_SESSION *session)
|
|||||||
{
|
{
|
||||||
// Destroy the service in the main routing worker thread
|
// Destroy the service in the main routing worker thread
|
||||||
mxs::RoutingWorker* main_worker = mxs::RoutingWorker::get(mxs::RoutingWorker::MAIN);
|
mxs::RoutingWorker* main_worker = mxs::RoutingWorker::get(mxs::RoutingWorker::MAIN);
|
||||||
main_worker->post(std::auto_ptr<ServiceDestroyTask>(new ServiceDestroyTask(service)));
|
main_worker->post(std::auto_ptr<ServiceDestroyTask>(new ServiceDestroyTask(service)),
|
||||||
|
mxs::Worker::EXECUTE_AUTO);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -516,10 +516,10 @@ bool Worker::post(GenericFunction func, Semaphore* pSem, execute_mode_t mode)
|
|||||||
return rval;
|
return rval;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Worker::execute(GenericFunction func)
|
bool Worker::execute(GenericFunction func, execute_mode_t mode)
|
||||||
{
|
{
|
||||||
Semaphore sem;
|
Semaphore sem;
|
||||||
return post(func, &sem, EXECUTE_AUTO) && sem.wait();
|
return post(func, &sem, mode) && sem.wait();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Worker::post_message(uint32_t msg_id, intptr_t arg1, intptr_t arg2)
|
bool Worker::post_message(uint32_t msg_id, intptr_t arg1, intptr_t arg2)
|
||||||
|
@ -263,7 +263,7 @@ void MariaDBMonitor::diagnostics(DCB *dcb) const
|
|||||||
diag_str = diagnostics_to_string();
|
diag_str = diagnostics_to_string();
|
||||||
};
|
};
|
||||||
|
|
||||||
if (!mutable_ptr->execute(func))
|
if (!mutable_ptr->execute(func, Worker::EXECUTE_AUTO))
|
||||||
{
|
{
|
||||||
diag_str = DIAG_ERROR;
|
diag_str = DIAG_ERROR;
|
||||||
}
|
}
|
||||||
@ -310,7 +310,7 @@ json_t* MariaDBMonitor::diagnostics_json() const
|
|||||||
rval = diagnostics_to_json();
|
rval = diagnostics_to_json();
|
||||||
};
|
};
|
||||||
|
|
||||||
if (!mutable_ptr->execute(func))
|
if (!mutable_ptr->execute(func, Worker::EXECUTE_AUTO))
|
||||||
{
|
{
|
||||||
rval = mxs_json_error_append(rval, "%s", DIAG_ERROR);
|
rval = mxs_json_error_append(rval, "%s", DIAG_ERROR);
|
||||||
}
|
}
|
||||||
|
@ -335,7 +335,7 @@ static bool conversion_task_ctl(Avro *inst, bool start)
|
|||||||
|
|
||||||
if (task.get())
|
if (task.get())
|
||||||
{
|
{
|
||||||
worker->post(task);
|
worker->post(task, Worker::EXECUTE_AUTO);
|
||||||
rval = true;
|
rval = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user