프로그래밍/Let's Share it
프로세스 생성 종결자 (ShellExecute / CreateProcess)
초록생선
2012. 3. 23. 14:50
Vista 이상에서 되도록 ShellExecute를 권장하고 있습니다.
그래서 ShellExecute를 그냥 사용하게 되는데,
간혹 Shell 쪽의 COM이 깨져 ShellExecute가 실패하는 경우가 있습니다.
그때는 CreateProcess를 해줘야 합니다.
ShellExecute / CreateProcess 모두 많은 Argument가 있어 사용하기 복잡합니다.
그래서,
- 실행 경로 (lpszExePath)
- 종료때 까지 기다릴지 여부 (bBlock)
- Exit Code (pnExitCode, Optional)
- 실행 파라미터 (lpszParam, Optional)
- 실행 프로세스의 Current Directory (lpszDirectory, Optional)
와 같이 실제 자주 사용되는 Argument로 종합한 함수를 공유합니다.
물론, Optional은 NULL이나 0을 전달해도 무방합니다.
내부에서 ShellExecute가 실패하면 CreateProcess로 연결되니, 걱정않으셔도 됩니다. :)
// TRACELOG는 TRACE로그를 찍는 것일 수 있음 // 각자 정의한 것으로 Replace 요망 // 삭제 가능 // Block이 TRUE이고 ERROR_SUCCESS 리턴일때 pnExitCode 의미 있음 DWORD ExecuteProcess(IN LPCTSTR lpszExePath, IN BOOL bBlock, OPTIONAL OUT PINT pnExitcode, OPTIONAL IN LPCTSTR lpszParam, OPTIONAL IN LPCTSTR lpszDirectory) { DWORD dwRtnValue = ERROR_SUCCESS; DWORD dwResult = 0; SHELLEXECUTEINFO stInfo = {0,}; HANDLE hCreateProcess = NULL; STARTUPINFO si = {0,}; PROCESS_INFORMATION pi = {0,}; LPTSTR lpszCmd = {0,}; CString strCmdLine; if (NULL != pnExitcode) { (*pnExitcode) = -1; } if (NULL == lpszExePath) { dwRtnValue = ERROR_INVALID_PARAMETER; goto FINAL; } TRACELOG(TEXT("[INFO] ExecuteCalled")); TRACELOG(TEXT("[INFO] lpszExePath : %s"), lpszExePath); if (NULL != lpszParam) { TRACELOG(TEXT("[INFO] lpszParam : %s"), lpszParam); } if (NULL != lpszDirectory) { TRACELOG(TEXT("[INFO] lpszDirectory : %s"), lpszDirectory); } TRACELOG(TEXT("[INFO] Block : %d"), bBlock); stInfo.cbSize = sizeof(stInfo); stInfo.lpFile = lpszExePath; stInfo.lpDirectory = lpszDirectory; stInfo.lpParameters = lpszParam; stInfo.fMask = SEE_MASK_NOCLOSEPROCESS | SEE_MASK_FLAG_NO_UI | SEE_MASK_CLASSNAME; stInfo.nShow = SW_SHOW; stInfo.lpClass = TEXT("exefile"); if (FALSE == ::ShellExecuteEx(&stInfo)) { // ShellExcute가 실패했다. TRACELOG(TEXT("[ERROR] Fail to ShellExecuteEx, %d"), ::GetLastError()); // CreateProcess로 재도전 strCmdLine = lpszExePath; strCmdLine.TrimRight(TEXT("\"")); strCmdLine.TrimLeft (TEXT("\"")); strCmdLine.Insert(0, TEXT("\"")); strCmdLine += TEXT("\""); si.cb = sizeof(si); if (NULL != lpszParam) { strCmdLine += TEXT(" "); strCmdLine += lpszParam; } lpszCmd = new TCHAR[_tcslen(strCmdLine)+1]; if (NULL == lpszCmd) { dwRtnValue = ERROR_NOT_ENOUGH_MEMORY; goto FINAL; } ZeroMemory(lpszCmd, sizeof(TCHAR)*(_tcslen(strCmdLine)+1)); ::StringCchCopy(lpszCmd, _tcslen(strCmdLine)+1, strCmdLine); strCmdLine.Empty(); if (FALSE == ::CreateProcess(NULL, lpszCmd, NULL, NULL, FALSE, 0, NULL, lpszDirectory, &si, &pi)) { dwRtnValue = ::GetLastError(); TRACELOG(TEXT("[INFO] Fail to CreateProcess, %d"), dwRtnValue); goto FINAL; } if (TRUE == bBlock) { TRACELOG(TEXT("[INFO] Try to Wait process(CP) %s"), lpszExePath); dwRtnValue = ::WaitForSingleObject(pi.hProcess, INFINITE); TRACELOG(TEXT("[INFO] Finished to Wait process(CP) %s"), lpszExePath); if (NULL != pnExitcode) { if (FALSE == ::GetExitCodeProcess(pi.hProcess, (DWORD*)pnExitcode)) { dwRtnValue = ::GetLastError(); TRACELOG(TEXT("[ERROR] Fail to GetExitCode(CP), GetLastError=%d"), dwRtnValue); goto FINAL; } TRACELOG(TEXT("[INFO] ExitCode(CP)=%d"), (*pnExitcode)); } } if (NULL != pi.hProcess) { ::CloseHandle(pi.hProcess); pi.hProcess = NULL; } if (NULL != pi.hThread) { ::CloseHandle(pi.hThread); pi.hThread = NULL; } } else { if (NULL != stInfo.hProcess) { if (TRUE == bBlock) { TRACELOG(TEXT("[INFO] Try to Wait process %s"), lpszExePath); dwResult = ::WaitForSingleObject(stInfo.hProcess, INFINITE); TRACELOG(TEXT("[INFO] Finished to Wait process %s"), lpszExePath); if (NULL != pnExitcode) { if (FALSE == ::GetExitCodeProcess(stInfo.hProcess, (DWORD*)pnExitcode)) { dwRtnValue = ::GetLastError(); TRACELOG(TEXT("[ERROR] Fail to GetExitCode, GetLastError=%d"), dwRtnValue); goto FINAL; } TRACELOG(TEXT("[INFO] ExitCode=%d"), (*pnExitcode)); } } ::CloseHandle(stInfo.hProcess); stInfo.hProcess = NULL; } } dwRtnValue = ERROR_SUCCESS; FINAL: if (NULL == lpszCmd) { delete [] lpszCmd; lpszCmd = NULL; } if (NULL != stInfo.hProcess) { ::CloseHandle(stInfo.hProcess); stInfo.hProcess = NULL; } if (NULL != pi.hProcess) { ::CloseHandle(pi.hProcess); pi.hProcess = NULL; } if (NULL != pi.hThread) { ::CloseHandle(pi.hThread); pi.hThread = NULL; } return dwRtnValue; }