move VMWare SVGA driver to generic location
[AROS.git] / rom / exec / childwait.c
blobd5676029a1f14d181ce573d551a601e2208f5140
1 /*
2 Copyright © 1995-2007, The AROS Development Team. All rights reserved.
3 $Id$
5 ChildWait() - Wait for a task to finish it processing.
6 */
7 #include "exec_intern.h"
8 #include <proto/exec.h>
10 /*****************************************************************************
12 NAME */
14 AROS_LH1(IPTR, ChildWait,
16 /* SYNOPSIS */
17 AROS_LHA(APTR, tid, D0),
19 /* LOCATION */
20 struct ExecBase *, SysBase, 126, Exec)
22 /* FUNCTION
23 Wait for either a specific child task, or any child task to finish.
24 If you specify tid = 0, then this call will return when any child
25 task exits, otherwise it will not return until the requested task
26 finishes.
28 Note that the tid is NOT the task control block (ie struct Task *),
29 rather it is the value of the ETask et_UniqueID field. Passing in a
30 Task pointer will cause your Task to deadlock.
32 You must call ChildFree() on the returned task before calling
33 ChildWait() again. Ie.
35 struct ETask *et;
37 et = ChildWait(0);
38 ChildFree(et->et_UniqueID);
40 INPUTS
41 tid - The UniqueID of a task.
43 RESULT
44 Returns either the ETask structure of the child, or one of the
45 CHILD_* values on error.
47 This allows you to work out which of the children has exited.
49 NOTES
50 Calling ChildWait() on a task that isn't your child will result in
51 a deadlock.
53 EXAMPLE
55 BUGS
56 Be careful with the return result of this function.
58 SEE ALSO
60 INTERNALS
62 *****************************************************************************/
64 AROS_LIBFUNC_INIT
66 struct Task *this = FindTask(NULL);
67 struct ETask *et;
68 struct ETask *child;
71 ChildWait() will wait for either a specific or any child task to
72 finish. We we get it we return the task unique id to the caller.
74 Firstly, are we a new-style Task?
76 if ((this->tc_Flags & TF_ETASK) == 0)
77 return CHILD_NOTNEW;
79 et = this->tc_UnionETask.tc_ETask;
82 Scanning this list is unsafe, I need to Forbid(). Note that the
83 Wait() below will break the Forbid() condition. This is how I need
84 it to be.
86 Forbid();
88 /* We do not return until the condition is met */
89 for (;;)
91 /* Check if it has returned already. This will also take the first. */
92 ForeachNode(&et->et_TaskMsgPort.mp_MsgList, child)
94 if ((ULONG)tid == 0 || child->et_UniqueID == (ULONG)tid)
95 break;
98 /* No matching children, we have to wait */
99 SetSignal(0, SIGF_CHILD);
100 Wait(SIGF_CHILD);
103 Permit();
104 return (IPTR)child;
106 AROS_LIBFUNC_EXIT
107 } /* ChildWait */