1 // TortoiseGit - a Windows shell extension for easy version control
3 // Copyright (C) 2016 - TortoiseGit
4 // Copyright (C) 2009-2010 - TortoiseSVN
6 // This program is free software; you can redistribute it and/or
7 // modify it under the terms of the GNU General Public License
8 // as published by the Free Software Foundation; either version 2
9 // of the License, or (at your option) any later version.
11 // This program is distributed in the hope that it will be useful,
12 // but WITHOUT ANY WARRANTY; without even the implied warranty of
13 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 // GNU General Public License for more details.
16 // You should have received a copy of the GNU General Public License
17 // along with this program; if not, write to the Free Software Foundation,
18 // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
21 #include "scope_exit_noexcept.h"
24 * A helper class for invoking CreateProcess(). The lpProcessInformation
25 * can point to an uninitialized struct - it's memset to all zeroes inside.
28 class CCreateProcessHelper
31 static bool CreateProcess(LPCTSTR lpApplicationName
,
33 LPCTSTR lpCurrentDirectory
,
34 LPPROCESS_INFORMATION lpProcessInformation
);
35 static bool CreateProcess(LPCTSTR lpApplicationName
,
37 LPPROCESS_INFORMATION lpProcessInformation
);
39 static bool CreateProcessDetached(LPCTSTR lpApplicationName
,
41 LPCTSTR lpCurrentDirectory
);
42 static bool CreateProcessDetached(LPCTSTR lpApplicationName
,
43 LPTSTR lpCommandLine
);
44 static bool CreateProcessDetached(LPCTSTR lpApplicationName
, LPCTSTR lpCommandLine
);
47 inline bool CCreateProcessHelper::CreateProcess(LPCTSTR applicationName
,
48 LPTSTR commandLine
, LPCTSTR currentDirectory
,
49 LPPROCESS_INFORMATION processInfo
)
51 STARTUPINFO startupInfo
= { 0 };
52 startupInfo
.cb
= sizeof(STARTUPINFO
);
54 memset(processInfo
, 0, sizeof(PROCESS_INFORMATION
));
55 const BOOL result
= ::CreateProcess(applicationName
, commandLine
, nullptr, nullptr, FALSE
, CREATE_UNICODE_ENVIRONMENT
, nullptr, currentDirectory
, &startupInfo
, processInfo
);
59 inline bool CCreateProcessHelper::CreateProcess(LPCTSTR applicationName
,
60 LPTSTR commandLine
, LPPROCESS_INFORMATION processInformation
)
62 return CreateProcess( applicationName
, commandLine
, nullptr, processInformation
);
65 inline bool CCreateProcessHelper::CreateProcessDetached(LPCTSTR lpApplicationName
,
66 LPTSTR lpCommandLine
, LPCTSTR lpCurrentDirectory
)
68 PROCESS_INFORMATION process
;
69 if (!CreateProcess(lpApplicationName
, lpCommandLine
, lpCurrentDirectory
, &process
))
72 CloseHandle(process
.hThread
);
73 CloseHandle(process
.hProcess
);
77 inline bool CCreateProcessHelper::CreateProcessDetached(LPCTSTR lpApplicationName
,
80 return CreateProcessDetached(lpApplicationName
, lpCommandLine
, nullptr);
83 inline bool CCreateProcessHelper::CreateProcessDetached(LPCTSTR lpApplicationName
, LPCTSTR lpCommandLine
)
85 // CreateProcess may modify buffer specified by lpCommandLine:
86 // https://msdn.microsoft.com/en-us/library/windows/desktop/ms682425
88 // The Unicode version of this function, CreateProcessW, can modify
89 // the contents of this string. Therefore, this parameter cannot be a
90 // pointer to read - only memory(such as a const variable or a literal
91 // string).If this parameter is a constant string, the function may
92 // cause an access violation.
94 LPWSTR commandLineBuf
= nullptr;
96 commandLineBuf
= _wcsdup(lpCommandLine
);
97 SCOPE_EXIT
{ free((void*)commandLineBuf
); };
99 return CreateProcessDetached(lpApplicationName
, commandLineBuf
, nullptr);