Fix a potential race condition
git-svn-id: svn://svn.code.sf.net/p/nuttx/code/trunk@2234 42af7a65-404d-4744-a932-0658087f49c3
This commit is contained in:
parent
7f84a8bea1
commit
bf7555cc4a
@ -101,6 +101,13 @@ EXTERN pid_t g_worker;
|
||||
* Queue work to be performed at a later time. All queued work will be
|
||||
* performed on the worker thread of of execution (not the caller's).
|
||||
*
|
||||
* The work structure is allocated by caller, but completely managed by
|
||||
* the work queue logic. The caller should never modify the contents of
|
||||
* the work queue structure; the caller should not call work_queue()
|
||||
* again until either (1) the previous work has been performed and removed
|
||||
* from the queue, or (2) work_cancel() has been called to cancel the work
|
||||
* and remove it from the work queue.
|
||||
*
|
||||
* Input parameters:
|
||||
* work - The work structure to queue
|
||||
* worker - The worker callback to be invoked. The callback will invoked
|
||||
@ -121,7 +128,9 @@ EXTERN int work_queue(struct work_s *work, worker_t worker, FAR void *arg, uint3
|
||||
* Name: work_cancel
|
||||
*
|
||||
* Description:
|
||||
* Cancel previously queued work.
|
||||
* Cancel previously queued work. This removes work from the work queue.
|
||||
* After work has been canceled, it may be re-queue by calling work_queue()
|
||||
* again.
|
||||
*
|
||||
* Input parameters:
|
||||
* work - The previously queue work structure to cancel
|
||||
@ -137,7 +146,9 @@ EXTERN int work_cancel(struct work_s *work);
|
||||
* Name: work_signal
|
||||
*
|
||||
* Description:
|
||||
* Signal the worker thread to process the work queue now.
|
||||
* Signal the worker thread to process the work queue now. This function
|
||||
* is used internally by the work logic but could also be used by the
|
||||
* user to force an immediate re-assessment of pending work.
|
||||
*
|
||||
* Input parameters:
|
||||
* None
|
||||
|
@ -79,7 +79,9 @@
|
||||
* Name: work_cancel
|
||||
*
|
||||
* Description:
|
||||
* Cancel previously queued work.
|
||||
* Cancel previously queued work. This removes work from the work queue.
|
||||
* After work has been canceled, it may be re-queue by calling work_queue()
|
||||
* again.
|
||||
*
|
||||
* Input parameters:
|
||||
* work - The previously queue work structure to cancel
|
||||
|
@ -83,6 +83,13 @@
|
||||
* Queue work to be performed at a later time. All queued work will be
|
||||
* performed on the worker thread of of execution (not the caller's).
|
||||
*
|
||||
* The work structure is allocated by caller, but completely managed by
|
||||
* the work queue logic. The caller should never modify the contents of
|
||||
* the work queue structure; the caller should not call work_queue()
|
||||
* again until either (1) the previous work has been performed and removed
|
||||
* from the queue, or (2) work_cancel() has been called to cancel the work
|
||||
* and remove it from the work queue.
|
||||
*
|
||||
* Input parameters:
|
||||
* work - The work structure to queue
|
||||
* worker - The worker callback to be invoked. The callback will invoked
|
||||
|
@ -101,6 +101,8 @@ struct dq_queue_s g_work;
|
||||
int work_thread(int argc, char *argv[])
|
||||
{
|
||||
volatile FAR struct work_s *work;
|
||||
worker_t worker;
|
||||
FAR void *arg;
|
||||
irqstate_t flags;
|
||||
|
||||
/* Loop forever */
|
||||
@ -130,37 +132,44 @@ int work_thread(int argc, char *argv[])
|
||||
work = (FAR struct work_s *)g_work.head;
|
||||
while (work)
|
||||
{
|
||||
/* Is this work ready? It is ready if there is no delay or if
|
||||
* the delay has elapsed.
|
||||
*/
|
||||
/* Is this work ready? It is ready if there is no delay or if
|
||||
* the delay has elapsed.
|
||||
*/
|
||||
|
||||
if (work->delay == 0 || g_system_timer - work->qtime > work->delay)
|
||||
{
|
||||
/* Remove the ready-to-execute work from the list */
|
||||
if (work->delay == 0 || g_system_timer - work->qtime > work->delay)
|
||||
{
|
||||
/* Remove the ready-to-execute work from the list */
|
||||
|
||||
(void)dq_rem((struct dq_entry_s *)work, &g_work);
|
||||
(void)dq_rem((struct dq_entry_s *)work, &g_work);
|
||||
|
||||
/* Do the work. Re-enable interrupts while the work is being
|
||||
* performed... we don't have any idea how long that will take!
|
||||
*/
|
||||
/* Extract the work description from the entry (in case the work
|
||||
* instance by the re-used after it has been de-queued).
|
||||
*/
|
||||
|
||||
irqrestore(flags);
|
||||
work->worker(work->arg);
|
||||
worker = work->worker;
|
||||
arg = work->arg;
|
||||
|
||||
/* Now, unfortunately, since we re-enabled interrupts we don't
|
||||
* the state of the work list and we will have to start back at
|
||||
* the head of the list.
|
||||
*/
|
||||
/* Do the work. Re-enable interrupts while the work is being
|
||||
* performed... we don't have any idea how long that will take!
|
||||
*/
|
||||
|
||||
flags = irqsave();
|
||||
work = (FAR struct work_s *)g_work.head;
|
||||
}
|
||||
else
|
||||
{
|
||||
/* This one is not ready, try the next in the list. */
|
||||
irqrestore(flags);
|
||||
worker(arg);
|
||||
|
||||
work = (FAR struct work_s *)work->dq.flink;
|
||||
}
|
||||
/* Now, unfortunately, since we re-enabled interrupts we don't
|
||||
* know the state of the work list and we will have to start
|
||||
* back at the head of the list.
|
||||
*/
|
||||
|
||||
flags = irqsave();
|
||||
work = (FAR struct work_s *)g_work.head;
|
||||
}
|
||||
else
|
||||
{
|
||||
/* This one is not ready, try the next in the list. */
|
||||
|
||||
work = (FAR struct work_s *)work->dq.flink;
|
||||
}
|
||||
}
|
||||
irqrestore(flags);
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user