From 815c1cad70c68ec2b99e68eb1b65de963085c50d Mon Sep 17 00:00:00 2001 From: Bill Hoffman Date: Wed, 8 Sep 2004 10:41:54 -0400 Subject: BUG: don't close the pipes too early --- Source/cmWin32ProcessExecution.cxx | 136 ++++++++++++++++++++----------------- Source/cmWin32ProcessExecution.h | 18 +++-- 2 files changed, 83 insertions(+), 71 deletions(-) diff --git a/Source/cmWin32ProcessExecution.cxx b/Source/cmWin32ProcessExecution.cxx index e4ee523..6c9b050 100644 --- a/Source/cmWin32ProcessExecution.cxx +++ b/Source/cmWin32ProcessExecution.cxx @@ -483,19 +483,26 @@ bool cmWin32ProcessExecution::PrivateOpen(const char *cmdstring, int mode, int n) { - HANDLE hChildStdinRd, hChildStdinWr, hChildStdoutRd, hChildStdoutWr, - hChildStderrRd, hChildStderrWr, hChildStdinWrDup, hChildStdoutRdDup, - hChildStderrRdDup, hProcess; /* hChildStdoutWrDup; */ + HANDLE hProcess; SECURITY_ATTRIBUTES saAttr; BOOL fSuccess; int fd1, fd2, fd3; - + this->hChildStdinRd = 0; + this->hChildStdinWr = 0; + this->hChildStdoutRd = 0; + this->hChildStdoutWr = 0; + this->hChildStderrRd = 0; + this->hChildStderrWr = 0; + this->hChildStdinWrDup = 0; + this->hChildStdoutRdDup = 0; + this->hChildStderrRdDup = 0; + saAttr.nLength = sizeof(SECURITY_ATTRIBUTES); saAttr.bInheritHandle = TRUE; saAttr.lpSecurityDescriptor = NULL; - - if (!CreatePipe(&hChildStdinRd, &hChildStdinWr, &saAttr, 0)) + + if (!CreatePipe(&this->hChildStdinRd, &this->hChildStdinWr, &saAttr, 0)) { m_Output += "CreatePipeError\n"; return false; @@ -505,8 +512,8 @@ bool cmWin32ProcessExecution::PrivateOpen(const char *cmdstring, * the inheritance properties to FALSE. Otherwise, the child inherits * the these handles; resulting in non-closeable handles to the pipes * being created. */ - fSuccess = DuplicateHandle(GetCurrentProcess(), hChildStdinWr, - GetCurrentProcess(), &hChildStdinWrDup, 0, + fSuccess = DuplicateHandle(GetCurrentProcess(), this->hChildStdinWr, + GetCurrentProcess(), &this->hChildStdinWrDup, 0, FALSE, DUPLICATE_SAME_ACCESS); if (!fSuccess) @@ -520,14 +527,14 @@ bool cmWin32ProcessExecution::PrivateOpen(const char *cmdstring, that we're using. */ CloseHandle(hChildStdinWr); - if (!CreatePipe(&hChildStdoutRd, &hChildStdoutWr, &saAttr, 0)) + if (!CreatePipe(&this->hChildStdoutRd, &this->hChildStdoutWr, &saAttr, 0)) { m_Output += "CreatePipeError\n"; return false; } - fSuccess = DuplicateHandle(GetCurrentProcess(), hChildStdoutRd, - GetCurrentProcess(), &hChildStdoutRdDup, 0, + fSuccess = DuplicateHandle(GetCurrentProcess(), this->hChildStdoutRd, + GetCurrentProcess(), &this->hChildStdoutRdDup, 0, FALSE, DUPLICATE_SAME_ACCESS); if (!fSuccess) { @@ -541,16 +548,16 @@ bool cmWin32ProcessExecution::PrivateOpen(const char *cmdstring, if (n != POPEN_4) { - if (!CreatePipe(&hChildStderrRd, &hChildStderrWr, &saAttr, 0)) + if (!CreatePipe(&this->hChildStderrRd, &this->hChildStderrWr, &saAttr, 0)) { m_Output += "CreatePipeError\n"; return false; } fSuccess = DuplicateHandle(GetCurrentProcess(), - hChildStderrRd, - GetCurrentProcess(), - &hChildStderrRdDup, 0, - FALSE, DUPLICATE_SAME_ACCESS); + this->hChildStderrRd, + GetCurrentProcess(), + &this->hChildStderrRdDup, 0, + FALSE, DUPLICATE_SAME_ACCESS); if (!fSuccess) { m_Output += "DuplicateHandleError\n"; @@ -568,14 +575,14 @@ bool cmWin32ProcessExecution::PrivateOpen(const char *cmdstring, { case _O_WRONLY | _O_TEXT: /* Case for writing to child Stdin in text mode. */ - fd1 = _open_osfhandle(TO_INTPTR(hChildStdinWrDup), mode); + fd1 = _open_osfhandle(TO_INTPTR(this->hChildStdinWrDup), mode); /* We don't care about these pipes anymore, so close them. */ break; case _O_RDONLY | _O_TEXT: /* Case for reading from child Stdout in text mode. */ - fd1 = _open_osfhandle(TO_INTPTR(hChildStdoutRdDup), mode); + fd1 = _open_osfhandle(TO_INTPTR(this->hChildStdoutRdDup), mode); /* We don't care about these pipes anymore, so close them. */ break; @@ -583,14 +590,14 @@ bool cmWin32ProcessExecution::PrivateOpen(const char *cmdstring, case _O_RDONLY | _O_BINARY: /* Case for readinig from child Stdout in binary mode. */ - fd1 = _open_osfhandle(TO_INTPTR(hChildStdoutRdDup), mode); + fd1 = _open_osfhandle(TO_INTPTR(this->hChildStdoutRdDup), mode); /* We don't care about these pipes anymore, so close them. */ break; case _O_WRONLY | _O_BINARY: /* Case for writing to child Stdin in binary mode. */ - fd1 = _open_osfhandle(TO_INTPTR(hChildStdinWrDup), mode); + fd1 = _open_osfhandle(TO_INTPTR(this->hChildStdinWrDup), mode); /* We don't care about these pipes anymore, so close them. */ break; @@ -601,17 +608,17 @@ bool cmWin32ProcessExecution::PrivateOpen(const char *cmdstring, case POPEN_4: if ( 1 ) { - fd1 = _open_osfhandle(TO_INTPTR(hChildStdinWrDup), mode); - fd2 = _open_osfhandle(TO_INTPTR(hChildStdoutRdDup), mode); + fd1 = _open_osfhandle(TO_INTPTR(this->hChildStdinWrDup), mode); + fd2 = _open_osfhandle(TO_INTPTR(this->hChildStdoutRdDup), mode); break; } case POPEN_3: if ( 1) { - fd1 = _open_osfhandle(TO_INTPTR(hChildStdinWrDup), mode); - fd2 = _open_osfhandle(TO_INTPTR(hChildStdoutRdDup), mode); - fd3 = _open_osfhandle(TO_INTPTR(hChildStderrRdDup), mode); + fd1 = _open_osfhandle(TO_INTPTR(this->hChildStdinWrDup), mode); + fd2 = _open_osfhandle(TO_INTPTR(this->hChildStdoutRdDup), mode); + fd3 = _open_osfhandle(TO_INTPTR(this->hChildStderrRdDup), mode); break; } } @@ -621,9 +628,9 @@ bool cmWin32ProcessExecution::PrivateOpen(const char *cmdstring, if (!RealPopenCreateProcess(cmdstring, path, m_ConsoleSpawn.c_str(), - hChildStdinRd, - hChildStdoutWr, - hChildStdoutWr, + this->hChildStdinRd, + this->hChildStdoutWr, + this->hChildStdoutWr, &hProcess, m_HideWindows, m_Output)) return 0; @@ -633,9 +640,9 @@ bool cmWin32ProcessExecution::PrivateOpen(const char *cmdstring, if (!RealPopenCreateProcess(cmdstring, path, m_ConsoleSpawn.c_str(), - hChildStdinRd, - hChildStdoutWr, - hChildStderrWr, + this->hChildStdinRd, + this->hChildStdoutWr, + this->hChildStderrWr, &hProcess, m_HideWindows, m_Output)) return 0; @@ -658,39 +665,6 @@ bool cmWin32ProcessExecution::PrivateOpen(const char *cmdstring, * make sure that no handles to the write end of the output pipe * are maintained in this process or else the pipe will not close * when the child process exits and the ReadFile will hang. */ - - if (!CloseHandle(hChildStdinRd)) - { - m_Output += "CloseHandleError\n"; - return false; - } - if(!CloseHandle(hChildStdoutRdDup)) - { - m_Output += "CloseHandleError\n"; - return false; - } - if(!CloseHandle(hChildStderrRdDup)) - { - m_Output += "CloseHandleError\n"; - return false; - } - if(!CloseHandle(hChildStdinWrDup)) - { - m_Output += "CloseHandleError\n"; - return false; - } - if (!CloseHandle(hChildStdoutWr)) - { - m_Output += "CloseHandleError\n"; - return false; - } - - if ((n != 4) && (!CloseHandle(hChildStderrWr))) - { - m_Output += "CloseHandleError\n"; - return false; - } - m_ProcessHandle = hProcess; if ( fd1 >= 0 ) { @@ -832,6 +806,40 @@ bool cmWin32ProcessExecution::PrivateClose(int /* timeout */) CloseHandle(hProcess); m_ExitValue = result; m_Output += output; + + if (this->hChildStdinRd && !CloseHandle(this->hChildStdinRd)) + { + m_Output += "CloseHandleError\n"; + return false; + } + if(this->hChildStdoutRdDup && !CloseHandle(this->hChildStdoutRdDup)) + { + m_Output += "CloseHandleError\n"; + return false; + } + if(this->hChildStderrRdDup && !CloseHandle(this->hChildStderrRdDup)) + { + m_Output += "CloseHandleError\n"; + return false; + } + if(this->hChildStdinWrDup && !CloseHandle(this->hChildStdinWrDup)) + { + m_Output += "CloseHandleError\n"; + return false; + } + if (this->hChildStdoutWr && !CloseHandle(this->hChildStdoutWr)) + { + m_Output += "CloseHandleError\n"; + return false; + } + + if (this->hChildStderrWr && !CloseHandle(this->hChildStderrWr)) + { + m_Output += "CloseHandleError\n"; + return false; + } + + if ( result < 0 ) { return false; diff --git a/Source/cmWin32ProcessExecution.h b/Source/cmWin32ProcessExecution.h index ea38768..822999c 100644 --- a/Source/cmWin32ProcessExecution.h +++ b/Source/cmWin32ProcessExecution.h @@ -143,13 +143,17 @@ private: bool PrivateClose(int timeout); HANDLE m_ProcessHandle; - - // Comment this out. Maybe we will need it in the future. - // file IO access to the process might be cool. - // FILE* m_StdIn; - // FILE* m_StdOut; - // FILE* m_StdErr; - + HANDLE hChildStdinRd; + HANDLE hChildStdinWr; + HANDLE hChildStdoutRd; + HANDLE hChildStdoutWr; + HANDLE hChildStderrRd; + HANDLE hChildStderrWr; + HANDLE hChildStdinWrDup; + HANDLE hChildStdoutRdDup; + HANDLE hChildStderrRdDup; + + int m_pStdIn; int m_pStdOut; int m_pStdErr; -- cgit v0.12