diff options
Diffstat (limited to 'Source/kwsys')
-rw-r--r-- | Source/kwsys/ProcessWin32.c | 186 |
1 files changed, 106 insertions, 80 deletions
diff --git a/Source/kwsys/ProcessWin32.c b/Source/kwsys/ProcessWin32.c index 5f77282..538f4b4 100644 --- a/Source/kwsys/ProcessWin32.c +++ b/Source/kwsys/ProcessWin32.c @@ -78,10 +78,12 @@ typedef struct kwsysProcessCreateInformation_s /*--------------------------------------------------------------------------*/ typedef struct kwsysProcessPipeData_s kwsysProcessPipeData; -static DWORD WINAPI kwsysProcessPipeThread(LPVOID ptd); -static void kwsysProcessPipeThreadReadPipe(kwsysProcess* cp, kwsysProcessPipeData* td); +static DWORD WINAPI kwsysProcessPipeThreadRead(LPVOID ptd); +static void kwsysProcessPipeThreadReadPipe(kwsysProcess* cp, + kwsysProcessPipeData* td); static DWORD WINAPI kwsysProcessPipeThreadWake(LPVOID ptd); -static void kwsysProcessPipeThreadWakePipe(kwsysProcess* cp, kwsysProcessPipeData* td); +static void kwsysProcessPipeThreadWakePipe(kwsysProcess* cp, + kwsysProcessPipeData* td); static int kwsysProcessInitialize(kwsysProcess* cp); static int kwsysProcessCreate(kwsysProcess* cp, int index, kwsysProcessCreateInformation* si, @@ -108,30 +110,34 @@ static void kwsysProcessDisablePipeThreads(kwsysProcess* cp); extern kwsysEXPORT int kwsysEncodedWriteArrayProcessFwd9x(const char* fname); /*--------------------------------------------------------------------------*/ -/* A structure containing data for each pipe's thread. */ -struct kwsysProcessPipeData_s +/* A structure containing synchronization data for each thread. */ +typedef struct kwsysProcessPipeSync_s kwsysProcessPipeSync; +struct kwsysProcessPipeSync_s { - /* ------------- Data managed per instance of kwsysProcess ------------- */ - - /* Handle for the thread for this pipe. */ + /* Handle to the thread. */ HANDLE Thread; - HANDLE ThreadWake; - /* Semaphore indicating a process and pipe are available. */ + /* Semaphore indicating to the thread that a process has started. */ HANDLE Ready; - /* Semaphore indicating when this thread's buffer is empty. */ - HANDLE Empty; + /* Semaphore indicating to the thread that it should begin work. */ + HANDLE Go; - /* Semaphore indicating a pipe thread has reset for another process. */ + /* Semaphore indicating thread has reset for another process. */ HANDLE Reset; +}; - /* Semaphore indicating the wake thread should unblock. */ - HANDLE Wake; +/*--------------------------------------------------------------------------*/ +/* A structure containing data for each pipe's threads. */ +struct kwsysProcessPipeData_s +{ + /* ------------- Data managed per instance of kwsysProcess ------------- */ + + /* Synchronization data for reading thread. */ + kwsysProcessPipeSync Reader; - /* Semaphore indicating the wake thread has reset for another process. */ - HANDLE ReadyWake; - HANDLE ResetWake; + /* Synchronization data for waking thread. */ + kwsysProcessPipeSync Waker; /* Index of this pipe. */ int Index; @@ -428,53 +434,54 @@ kwsysProcess* kwsysProcess_New() /* Give the thread a pointer back to the kwsysProcess instance. */ cp->Pipe[i].Process = cp; - /* The pipe is not yet ready to read. Initialize semaphore to 0. */ - if(!(cp->Pipe[i].Ready = CreateSemaphore(0, 0, 1, 0))) + /* No process is yet running. Initialize semaphore to 0. */ + if(!(cp->Pipe[i].Reader.Ready = CreateSemaphore(0, 0, 1, 0))) { kwsysProcess_Delete(cp); return 0; } /* The pipe is not yet reset. Initialize semaphore to 0. */ - if(!(cp->Pipe[i].Reset = CreateSemaphore(0, 0, 1, 0))) + if(!(cp->Pipe[i].Reader.Reset = CreateSemaphore(0, 0, 1, 0))) { kwsysProcess_Delete(cp); return 0; } /* The thread's buffer is initially empty. Initialize semaphore to 1. */ - if(!(cp->Pipe[i].Empty = CreateSemaphore(0, 1, 1, 0))) + if(!(cp->Pipe[i].Reader.Go = CreateSemaphore(0, 1, 1, 0))) { kwsysProcess_Delete(cp); return 0; } - /* Create the thread. It will block immediately. The thread will - not make deeply nested calls, so we need only a small - stack. */ - if(!(cp->Pipe[i].Thread = CreateThread(0, 1024, kwsysProcessPipeThread, - &cp->Pipe[i], 0, &dummy))) + /* Create the reading thread. It will block immediately. The + thread will not make deeply nested calls, so we need only a + small stack. */ + if(!(cp->Pipe[i].Reader.Thread = CreateThread(0, 1024, + kwsysProcessPipeThreadRead, + &cp->Pipe[i], 0, &dummy))) { kwsysProcess_Delete(cp); return 0; } - /* The wake thread should block. Initialize semaphore to 0. */ - if(!(cp->Pipe[i].ReadyWake = CreateSemaphore(0, 0, 1, 0))) + /* No process is yet running. Initialize semaphore to 0. */ + if(!(cp->Pipe[i].Waker.Ready = CreateSemaphore(0, 0, 1, 0))) { kwsysProcess_Delete(cp); return 0; } - /* The wake thread need not reset yet. Initialize semaphore to 0. */ - if(!(cp->Pipe[i].ResetWake = CreateSemaphore(0, 0, 1, 0))) + /* The pipe is not yet reset. Initialize semaphore to 0. */ + if(!(cp->Pipe[i].Waker.Reset = CreateSemaphore(0, 0, 1, 0))) { kwsysProcess_Delete(cp); return 0; } - /* The pipe will not need to wake yet. Initialize semaphore to 0. */ - if(!(cp->Pipe[i].Wake = CreateSemaphore(0, 0, 1, 0))) + /* The waker should not wake immediately. Initialize semaphore to 0. */ + if(!(cp->Pipe[i].Waker.Go = CreateSemaphore(0, 0, 1, 0))) { kwsysProcess_Delete(cp); return 0; @@ -483,9 +490,9 @@ kwsysProcess* kwsysProcess_New() /* Create the waking thread. It will block immediately. The thread will not make deeply nested calls, so we need only a small stack. */ - if(!(cp->Pipe[i].ThreadWake = CreateThread(0, 1024, - kwsysProcessPipeThreadWake, - &cp->Pipe[i], 0, &dummy))) + if(!(cp->Pipe[i].Waker.Thread = CreateThread(0, 1024, + kwsysProcessPipeThreadWake, + &cp->Pipe[i], 0, &dummy))) { kwsysProcess_Delete(cp); return 0; @@ -525,39 +532,41 @@ void kwsysProcess_Delete(kwsysProcess* cp) /* Terminate each of the threads. */ for(i=0; i < KWSYSPE_PIPE_COUNT; ++i) { - if(cp->Pipe[i].Thread) + /* Terminate this reading thread. */ + if(cp->Pipe[i].Reader.Thread) { /* Signal the thread we are ready for it. It will terminate immediately since Deleting is set. */ - ReleaseSemaphore(cp->Pipe[i].Ready, 1, 0); + ReleaseSemaphore(cp->Pipe[i].Reader.Ready, 1, 0); /* Wait for the thread to exit. */ - WaitForSingleObject(cp->Pipe[i].Thread, INFINITE); + WaitForSingleObject(cp->Pipe[i].Reader.Thread, INFINITE); /* Close the handle to the thread. */ - kwsysProcessCleanupHandle(&cp->Pipe[i].Thread); + kwsysProcessCleanupHandle(&cp->Pipe[i].Reader.Thread); } - if(cp->Pipe[i].ThreadWake) + /* Terminate this waking thread. */ + if(cp->Pipe[i].Waker.Thread) { /* Signal the thread we are ready for it. It will terminate immediately since Deleting is set. */ - ReleaseSemaphore(cp->Pipe[i].ReadyWake, 1, 0); + ReleaseSemaphore(cp->Pipe[i].Waker.Ready, 1, 0); /* Wait for the thread to exit. */ - WaitForSingleObject(cp->Pipe[i].ThreadWake, INFINITE); + WaitForSingleObject(cp->Pipe[i].Waker.Thread, INFINITE); /* Close the handle to the thread. */ - kwsysProcessCleanupHandle(&cp->Pipe[i].ThreadWake); + kwsysProcessCleanupHandle(&cp->Pipe[i].Waker.Thread); } /* Cleanup the pipe's semaphores. */ - kwsysProcessCleanupHandle(&cp->Pipe[i].Reset); - kwsysProcessCleanupHandle(&cp->Pipe[i].Ready); - kwsysProcessCleanupHandle(&cp->Pipe[i].Empty); - kwsysProcessCleanupHandle(&cp->Pipe[i].ReadyWake); - kwsysProcessCleanupHandle(&cp->Pipe[i].ResetWake); - kwsysProcessCleanupHandle(&cp->Pipe[i].Wake); + kwsysProcessCleanupHandle(&cp->Pipe[i].Reader.Ready); + kwsysProcessCleanupHandle(&cp->Pipe[i].Reader.Go); + kwsysProcessCleanupHandle(&cp->Pipe[i].Reader.Reset); + kwsysProcessCleanupHandle(&cp->Pipe[i].Waker.Ready); + kwsysProcessCleanupHandle(&cp->Pipe[i].Waker.Go); + kwsysProcessCleanupHandle(&cp->Pipe[i].Waker.Reset); } /* Close the shared semaphores. */ @@ -1194,8 +1203,8 @@ void kwsysProcess_Execute(kwsysProcess* cp) /* Tell the pipe threads that a process has started. */ for(i=0; i < KWSYSPE_PIPE_COUNT; ++i) { - ReleaseSemaphore(cp->Pipe[i].Ready, 1, 0); - ReleaseSemaphore(cp->Pipe[i].ReadyWake, 1, 0); + ReleaseSemaphore(cp->Pipe[i].Reader.Ready, 1, 0); + ReleaseSemaphore(cp->Pipe[i].Waker.Ready, 1, 0); } /* We don't care about the children's main threads. */ @@ -1231,8 +1240,8 @@ void kwsysProcess_Disown(kwsysProcess* cp) /* Wait for all pipe threads to reset. */ for(i=0; i < KWSYSPE_PIPE_COUNT; ++i) { - WaitForSingleObject(cp->Pipe[i].Reset, INFINITE); - WaitForSingleObject(cp->Pipe[i].ResetWake, INFINITE); + WaitForSingleObject(cp->Pipe[i].Reader.Reset, INFINITE); + WaitForSingleObject(cp->Pipe[i].Waker.Reset, INFINITE); } /* We will not wait for exit, so cleanup now. */ @@ -1278,7 +1287,7 @@ int kwsysProcess_WaitForData(kwsysProcess* cp, char** data, int* length, done with the data. */ if(cp->CurrentIndex < KWSYSPE_PIPE_COUNT) { - ReleaseSemaphore(cp->Pipe[cp->CurrentIndex].Empty, 1, 0); + ReleaseSemaphore(cp->Pipe[cp->CurrentIndex].Reader.Go, 1, 0); cp->CurrentIndex = KWSYSPE_PIPE_COUNT; } @@ -1317,8 +1326,10 @@ int kwsysProcess_WaitForData(kwsysProcess* cp, char** data, int* length, /* Data are available or a pipe closed. */ if(cp->Pipe[cp->CurrentIndex].Closed) { - /* The pipe closed. */ - ReleaseSemaphore(cp->Pipe[cp->CurrentIndex].Wake, 1, 0); + /* The pipe closed at the write end. Close the read end and + inform the wakeup thread it is done with this process. */ + kwsysProcessCleanupHandle(&cp->Pipe[cp->CurrentIndex].Read); + ReleaseSemaphore(cp->Pipe[cp->CurrentIndex].Waker.Go, 1, 0); --cp->PipesLeft; } else if(data && length) @@ -1413,15 +1424,15 @@ int kwsysProcess_WaitForExit(kwsysProcess* cp, double* userTimeout) without releaseing the pipe's thread. Release it now. */ if(cp->CurrentIndex < KWSYSPE_PIPE_COUNT) { - ReleaseSemaphore(cp->Pipe[cp->CurrentIndex].Empty, 1, 0); + ReleaseSemaphore(cp->Pipe[cp->CurrentIndex].Reader.Go, 1, 0); cp->CurrentIndex = KWSYSPE_PIPE_COUNT; } /* Wait for all pipe threads to reset. */ for(i=0; i < KWSYSPE_PIPE_COUNT; ++i) { - WaitForSingleObject(cp->Pipe[i].Reset, INFINITE); - WaitForSingleObject(cp->Pipe[i].ResetWake, INFINITE); + WaitForSingleObject(cp->Pipe[i].Reader.Reset, INFINITE); + WaitForSingleObject(cp->Pipe[i].Waker.Reset, INFINITE); } /* ---- It is now safe again to call kwsysProcessCleanup. ----- */ @@ -1503,23 +1514,19 @@ void kwsysProcess_Kill(kwsysProcess* cp) Function executed for each pipe's thread. Argument is a pointer to the kwsysProcessPipeData instance for this thread. */ -DWORD WINAPI kwsysProcessPipeThread(LPVOID ptd) +DWORD WINAPI kwsysProcessPipeThreadRead(LPVOID ptd) { kwsysProcessPipeData* td = (kwsysProcessPipeData*)ptd; kwsysProcess* cp = td->Process; /* Wait for a process to be ready. */ - while((WaitForSingleObject(td->Ready, INFINITE), !cp->Deleting)) + while((WaitForSingleObject(td->Reader.Ready, INFINITE), !cp->Deleting)) { /* Read output from the process for this thread's pipe. */ kwsysProcessPipeThreadReadPipe(cp, td); - /* We were signalled to exit with our buffer empty. Reset the - mutex for a new process. */ - ReleaseSemaphore(td->Empty, 1, 0); - /* Signal the main thread we have reset for a new process. */ - ReleaseSemaphore(td->Reset, 1, 0); + ReleaseSemaphore(td->Reader.Reset, 1, 0); } return 0; } @@ -1533,7 +1540,7 @@ DWORD WINAPI kwsysProcessPipeThread(LPVOID ptd) void kwsysProcessPipeThreadReadPipe(kwsysProcess* cp, kwsysProcessPipeData* td) { /* Wait for space in the thread's buffer. */ - while((WaitForSingleObject(td->Empty, INFINITE), !td->Closed)) + while((WaitForSingleObject(td->Reader.Go, INFINITE), !td->Closed)) { /* Read data from the pipe. This may block until data are available. */ if(!ReadFile(td->Read, td->DataBuffer, KWSYSPE_PIPE_BUFFER_SIZE, @@ -1555,6 +1562,10 @@ void kwsysProcessPipeThreadReadPipe(kwsysProcess* cp, kwsysProcessPipeData* td) cp->SharedIndex = td->Index; ReleaseSemaphore(cp->Full, 1, 0); } + + /* We were signalled to exit with our buffer empty. Reset the + mutex for a new process. */ + ReleaseSemaphore(td->Reader.Go, 1, 0); } /*--------------------------------------------------------------------------*/ @@ -1569,29 +1580,34 @@ DWORD WINAPI kwsysProcessPipeThreadWake(LPVOID ptd) kwsysProcess* cp = td->Process; /* Wait for a process to be ready. */ - while((WaitForSingleObject(td->ReadyWake, INFINITE), !cp->Deleting)) + while((WaitForSingleObject(td->Waker.Ready, INFINITE), !cp->Deleting)) { /* Wait for a possible wakeup. */ kwsysProcessPipeThreadWakePipe(cp, td); /* Signal the main thread we have reset for a new process. */ - ReleaseSemaphore(td->ResetWake, 1, 0); + ReleaseSemaphore(td->Waker.Reset, 1, 0); } return 0; } +/*--------------------------------------------------------------------------*/ + +/* + Function called in each pipe's thread to handle reading thread + wakeup for one execution of a subprocess. +*/ void kwsysProcessPipeThreadWakePipe(kwsysProcess* cp, kwsysProcessPipeData* td) { /* Wait for a possible wake command. */ - if((WaitForSingleObject(td->Wake, INFINITE), !td->Closed)) + WaitForSingleObject(td->Waker.Go, INFINITE); + + /* If the pipe is not closed, we need to wake up the reading thread. */ + if(!td->Closed) { - /* The pipe is not closed. We need to wake up the reading thread. */ DWORD dummy; WriteFile(td->Write, "", 1, &dummy, 0); } - - /* We have processed the wake command. */ - ReleaseSemaphore(td->Wake, 1, 0); } /*--------------------------------------------------------------------------*/ @@ -2666,11 +2682,11 @@ static void kwsysProcessDisablePipeThreads(kwsysProcess* cp) /* If data were just reported data, release the pipe's thread. */ if(cp->CurrentIndex < KWSYSPE_PIPE_COUNT) { - ReleaseSemaphore(cp->Pipe[cp->CurrentIndex].Empty, 1, 0); + ReleaseSemaphore(cp->Pipe[cp->CurrentIndex].Reader.Go, 1, 0); cp->CurrentIndex = KWSYSPE_PIPE_COUNT; } - /* Signal all the pipe wakeup threads. */ + /* Wakeup all reading threads that are not on closed pipes. */ for(i=0; i < KWSYSPE_PIPE_COUNT; ++i) { /* The wakeup threads will write one byte to the pipe write ends. @@ -2684,18 +2700,28 @@ static void kwsysProcessDisablePipeThreads(kwsysProcess* cp) thread to call WriteFile. If it blocks, that is okay because it will unblock when we close the read end and break the pipe below. */ - ReleaseSemaphore(cp->Pipe[i].Wake, 1, 0); + if(cp->Pipe[i].Read) + { + ReleaseSemaphore(cp->Pipe[i].Waker.Go, 1, 0); + } } /* Tell pipe threads to reset until we run another process. */ while(cp->PipesLeft > 0) { + /* The waking threads will cause all reading threads to report. + Wait for the next one and save its index. */ WaitForSingleObject(cp->Full, INFINITE); cp->CurrentIndex = cp->SharedIndex; ReleaseSemaphore(cp->SharedIndexMutex, 1, 0); - kwsysProcessCleanupHandle(&cp->Pipe[cp->CurrentIndex].Read); + + /* We are done reading this pipe. Close its read handle. */ cp->Pipe[cp->CurrentIndex].Closed = 1; - ReleaseSemaphore(cp->Pipe[cp->CurrentIndex].Empty, 1, 0); + kwsysProcessCleanupHandle(&cp->Pipe[cp->CurrentIndex].Read); --cp->PipesLeft; + + /* Tell the reading thread we are done with the data. It will + reset immediately because the pipe is closed. */ + ReleaseSemaphore(cp->Pipe[cp->CurrentIndex].Reader.Go, 1, 0); } } |