ReactOS 0.4.16-dev-937-g7afcd2a
spawnv.cpp File Reference
#include <corecrt_internal.h>
#include <corecrt_internal_lowio.h>
#include <corecrt_internal_traits.h>
#include <errno.h>
#include <io.h>
#include <mbstring.h>
#include <process.h>
#include <stdlib.h>
#include <string.h>
Include dependency graph for spawnv.cpp:

Go to the source code of this file.

Functions

static bool accumulate_inheritable_handles (BYTE **const data, size_t *const size, bool const include_std_handles) throw ()
 
static bool __cdecl should_create_unicode_environment (char) throw ()
 
static bool __cdecl should_create_unicode_environment (wchar_t) throw ()
 
template<typename Character >
static intptr_t __cdecl execute_command (int const mode, Character const *const file_name, Character const *const *const arguments, Character const *const *const environment) throw ()
 
template<typename Character >
static intptr_t __cdecl common_spawnv (int const mode, Character const *const file_name, Character const *const *const arguments, Character const *const *const environment) throw ()
 
intptr_t __cdecl _execv (char const *const file_name, char const *const *const arguments)
 
intptr_t __cdecl _execve (char const *const file_name, char const *const *const arguments, char const *const *const environment)
 
intptr_t __cdecl _spawnv (int const mode, char const *const file_name, char const *const *const arguments)
 
intptr_t __cdecl _spawnve (int const mode, char const *const file_name, char const *const *const arguments, char const *const *const environment)
 
intptr_t __cdecl _wexecv (wchar_t const *const file_name, wchar_t const *const *const arguments)
 
intptr_t __cdecl _wexecve (wchar_t const *const file_name, wchar_t const *const *const arguments, wchar_t const *const *const environment)
 
intptr_t __cdecl _wspawnv (int const mode, wchar_t const *const file_name, wchar_t const *const *const arguments)
 
intptr_t __cdecl _wspawnve (int const mode, wchar_t const *const file_name, wchar_t const *const *const arguments, wchar_t const *const *const environment)
 

Function Documentation

◆ _execv()

intptr_t __cdecl _execv ( char const *const  file_name,
char const *const *const  arguments 
)

Definition at line 379 of file spawnv.cpp.

383{
384 return common_spawnv(_P_OVERLAY, file_name, arguments, static_cast<char const* const*>(nullptr));
385}
static LPCWSTR file_name
Definition: protocol.c:147
#define _P_OVERLAY
Definition: port.h:372
static intptr_t __cdecl common_spawnv(int const mode, Character const *const file_name, Character const *const *const arguments, Character const *const *const environment)
Definition: spawnv.cpp:273
#define const
Definition: zconf.h:233

◆ _execve()

intptr_t __cdecl _execve ( char const *const  file_name,
char const *const *const  arguments,
char const *const *const  environment 
)

Definition at line 387 of file spawnv.cpp.

392{
393 return common_spawnv(_P_OVERLAY, file_name, arguments, environment);
394}

◆ _spawnv()

intptr_t __cdecl _spawnv ( int const  mode,
char const *const  file_name,
char const *const *const  arguments 
)

Definition at line 396 of file spawnv.cpp.

401{
402 return common_spawnv(mode, file_name, arguments, static_cast<char const* const*>(nullptr));
403}
GLenum mode
Definition: glext.h:6217

◆ _spawnve()

intptr_t __cdecl _spawnve ( int const  mode,
char const *const  file_name,
char const *const *const  arguments,
char const *const *const  environment 
)

Definition at line 405 of file spawnv.cpp.

411{
412 return common_spawnv(mode, file_name, arguments, environment);
413}

◆ _wexecv()

intptr_t __cdecl _wexecv ( wchar_t const *const  file_name,
wchar_t const *const *const  arguments 
)

Definition at line 417 of file spawnv.cpp.

421{
422 return common_spawnv(_P_OVERLAY, file_name, arguments, static_cast<wchar_t const* const*>(nullptr));
423}

◆ _wexecve()

intptr_t __cdecl _wexecve ( wchar_t const *const  file_name,
wchar_t const *const *const  arguments,
wchar_t const *const *const  environment 
)

Definition at line 425 of file spawnv.cpp.

430{
431 return common_spawnv(_P_OVERLAY, file_name, arguments, environment);
432}

◆ _wspawnv()

intptr_t __cdecl _wspawnv ( int const  mode,
wchar_t const *const  file_name,
wchar_t const *const *const  arguments 
)

Definition at line 434 of file spawnv.cpp.

439{
440 return common_spawnv(mode, file_name, arguments, static_cast<wchar_t const* const*>(nullptr));
441}

◆ _wspawnve()

intptr_t __cdecl _wspawnve ( int const  mode,
wchar_t const *const  file_name,
wchar_t const *const *const  arguments,
wchar_t const *const *const  environment 
)

Definition at line 443 of file spawnv.cpp.

449{
450 return common_spawnv(mode, file_name, arguments, environment);
451}

◆ accumulate_inheritable_handles()

static bool accumulate_inheritable_handles ( BYTE **const  data,
size_t *const  size,
bool const  include_std_handles 
)
throw (
)
static

Definition at line 82 of file spawnv.cpp.

87{
88 return __acrt_lock_and_call(__acrt_lowio_index_lock, [&]() -> bool
89 {
90 *data = nullptr;
91 *size = 0;
92
93 // Count the number of handles to be inherited:
94 size_t handle_count = 0;
95 for (handle_count = _nhandle; handle_count != 0 && _osfile(handle_count - 1) != 0; --handle_count)
96 {
97 }
98
99 size_t const max_handle_count = (USHRT_MAX - sizeof(int)) / (sizeof(char) + sizeof(intptr_t));
100 _VALIDATE_RETURN_NOEXC(handle_count < max_handle_count, ENOMEM, false);
101
102 size_t const handle_data_header_size = sizeof(int);
103 size_t const handle_data_element_size = sizeof(char) + sizeof(intptr_t);
104
105 unsigned short const handle_data_size = static_cast<unsigned short>(
106 handle_data_header_size +
107 handle_count * handle_data_element_size);
108
109 __crt_unique_heap_ptr<BYTE> handle_data(_calloc_crt_t(BYTE, handle_data_size));
110 _VALIDATE_RETURN_NOEXC(handle_data.get() != nullptr, ENOMEM, false);
111
112 // Set the handle count in the data:
113 *reinterpret_cast<int*>(handle_data.get()) = static_cast<int>(handle_count);
114
115 auto const first_flags = reinterpret_cast<char*>(handle_data.get() + sizeof(int));
116 auto const first_handle = reinterpret_cast<intptr_t UNALIGNED*>(first_flags + (handle_count * sizeof(char)));
117
118 // Copy the handle data:
119 auto flags_it = first_flags;
120 auto handle_it = first_handle;
121 for (size_t i = 0; i != handle_count; ++i, ++flags_it, ++handle_it)
122 {
123 __crt_lowio_handle_data* const pio = _pioinfo(i);
124 if ((pio->osfile & FNOINHERIT) == 0)
125 {
126 *flags_it = pio->osfile;
127 *handle_it = pio->osfhnd;
128 }
129 else
130 {
131 *flags_it = 0;
132 *handle_it = reinterpret_cast<intptr_t>(INVALID_HANDLE_VALUE);
133 }
134 }
135
136 // Exclude the first three handles (stdin, stdout, stderr) if asked:
137 if (!include_std_handles)
138 {
139 flags_it = first_flags;
140 handle_it = first_handle;
141 for (size_t i = 0; i != __min(handle_count, 3); ++i, ++flags_it, ++handle_it)
142 {
143 *flags_it = 0;
144 *handle_it = reinterpret_cast<intptr_t>(INVALID_HANDLE_VALUE);
145 }
146 }
147
148 *data = handle_data.detach();
149 *size = handle_data_size;
150 return true;
151 });
152}
#define ENOMEM
Definition: acclib.h:84
@ __acrt_lowio_index_lock
int _nhandle
Definition: ioinit.cpp:34
#define INVALID_HANDLE_VALUE
Definition: compat.h:731
unsigned char
Definition: typeof.h:29
unsigned int(__cdecl typeof(jpeg_read_scanlines))(struct jpeg_decompress_struct *
Definition: typeof.h:31
GLint GLenum GLsizei GLsizei GLsizei GLint GLsizei const GLvoid * data
Definition: gl.h:1950
GLsizeiptr size
Definition: glext.h:5919
GLsizei GLenum const GLvoid GLsizei GLenum GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLint GLint GLint GLshort GLshort GLshort GLubyte GLubyte GLubyte GLuint GLuint GLuint GLushort GLushort GLushort GLbyte GLbyte GLbyte GLbyte GLdouble GLdouble GLdouble GLdouble GLfloat GLfloat GLfloat GLfloat GLint GLint GLint GLint GLshort GLshort GLshort GLshort GLubyte GLubyte GLubyte GLubyte GLuint GLuint GLuint GLuint GLushort GLushort GLushort GLushort GLboolean const GLdouble const GLfloat const GLint const GLshort const GLbyte const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLdouble const GLfloat const GLfloat const GLint const GLint const GLshort const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort const GLdouble const GLfloat const GLint const GLshort GLenum GLenum GLenum GLfloat GLenum GLint GLenum GLenum GLenum GLfloat GLenum GLenum GLint GLenum GLfloat GLenum GLint GLint GLushort GLenum GLenum GLfloat GLenum GLenum GLint GLfloat const GLubyte GLenum GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLint GLint GLsizei GLsizei GLint GLenum GLenum const GLvoid GLenum GLenum const GLfloat GLenum GLenum const GLint GLenum GLenum const GLdouble GLenum GLenum const GLfloat GLenum GLenum const GLint GLsizei GLuint GLfloat GLuint GLbitfield GLfloat GLint GLuint GLboolean GLenum GLfloat GLenum GLbitfield GLenum GLfloat GLfloat GLint GLint const GLfloat GLenum GLfloat GLfloat GLint GLint GLfloat GLfloat GLint GLint const GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat GLint GLfloat GLfloat const GLdouble const GLfloat const GLdouble const GLfloat GLint i
Definition: glfuncs.h:248
#define __min(a, b)
Definition: stdlib.h:102
#define USHRT_MAX
Definition: limits.h:38
#define _VALIDATE_RETURN_NOEXC(expr, errorcode, retexpr)
#define UNALIGNED
Definition: pecoff.h:227
#define _pioinfo(i)
Definition: internal.h:71
#define _osfile(i)
Definition: internal.h:72
static BOOL handle_data(struct parsed_symbol *sym)
Definition: undname.c:1061
int intptr_t
Definition: vcruntime.h:134
unsigned char BYTE
Definition: xxhash.c:193

Referenced by execute_command().

◆ common_spawnv()

template<typename Character >
static intptr_t __cdecl common_spawnv ( int const  mode,
Character const *const  file_name,
Character const *const *const  arguments,
Character const *const *const  environment 
)
throw (
)
static

Definition at line 273 of file spawnv.cpp.

279{
280 typedef __crt_char_traits<Character> traits;
281
282 _VALIDATE_RETURN(file_name != nullptr, EINVAL, -1);
283 _VALIDATE_RETURN(file_name[0] != '\0', EINVAL, -1);
284 _VALIDATE_RETURN(arguments != nullptr, EINVAL, -1);
285 _VALIDATE_RETURN(arguments[0] != nullptr, EINVAL, -1);
286 _VALIDATE_RETURN(arguments[0][0] != '\0', EINVAL, -1);
287
288 Character const* const final_backslash = traits::tcsrchr(file_name, '\\');
289 Character const* const final_slash = traits::tcsrchr(file_name, '/');
290
291 Character const* mutated_file_name = file_name;
292 Character const* end_of_directory = final_backslash;
293 if (!final_slash)
294 {
295 if (!final_backslash)
296 {
297 Character const* const final_colon = traits::tcsrchr(file_name, ':');
298 if (final_colon)
299 {
300 end_of_directory = final_colon;
301 }
302 else
303 {
304 // The path is a file name only. We force it to be a relative
305 // path name.
306 size_t const file_name_size = traits::tcslen(file_name) + 3;
307 __crt_unique_heap_ptr<Character> buffer(_calloc_crt_t(Character, file_name_size));
308 if (!buffer)
309 return -1;
310
311 static Character const dot_slash[] = { '.', '\\', '\0' };
312 _ERRCHECK(traits::tcscpy_s(buffer.get(), file_name_size, dot_slash));
313 _ERRCHECK(traits::tcscat_s(buffer.get(), file_name_size, file_name));
314
315 mutated_file_name = buffer.detach();
316 end_of_directory = mutated_file_name + 2; // Adjust for ".\"
317 }
318 }
319 }
320 else if (!final_backslash || final_slash > final_backslash)
321 {
322 end_of_directory = final_slash;
323 }
324
325 // If we allocated a file name above, make sure we clean it up:
326 __crt_unique_heap_ptr<Character const> const mutated_file_name_cleanup(file_name == mutated_file_name
327 ? nullptr
328 : mutated_file_name);
329
330 if (traits::tcsrchr(end_of_directory, '.'))
331 {
332 // If an extension was provided, just invoke the path:
333 if (traits::taccess_s(mutated_file_name, 0) == 0)
334 {
335 return execute_command(mode, mutated_file_name, arguments, environment);
336 }
337 }
338 else
339 {
340 // If no extension was provided, try known executable extensions:
341 size_t const buffer_size = traits::tcslen(mutated_file_name) + 5;
342 __crt_unique_heap_ptr<Character> const buffer(_calloc_crt_t(Character, buffer_size));
343 if (!buffer)
344 return -1;
345
346 _ERRCHECK(traits::tcscpy_s(buffer.get(), buffer_size, mutated_file_name));
347 Character* extension_buffer = buffer.get() + buffer_size - 5;
348
349 typedef Character const extension_type[5];
350 static extension_type const extensions[4] =
351 {
352 { '.', 'c', 'o', 'm', '\0' },
353 { '.', 'e', 'x', 'e', '\0' },
354 { '.', 'b', 'a', 't', '\0' },
355 { '.', 'c', 'm', 'd', '\0' }
356 };
357
358 errno_t const saved_errno = errno;
359
360 extension_type const* const first_extension = extensions;
361 extension_type const* const last_extension = first_extension + _countof(extensions);
362 for (auto it = first_extension; it != last_extension; ++it)
363 {
364 _ERRCHECK(traits::tcscpy_s(extension_buffer, 5, *it));
365
366 if (traits::taccess_s(buffer.get(), 0) == 0)
367 {
368 errno = saved_errno;
369 return execute_command(mode, buffer.get(), arguments, environment);
370 }
371 }
372 }
373
374 return -1;
375}
#define EINVAL
Definition: acclib.h:90
#define _ERRCHECK(e)
#define _VALIDATE_RETURN(expr, errorcode, retexpr)
GLuint buffer
Definition: glext.h:5915
#define errno
Definition: errno.h:18
#define _countof(array)
Definition: sndvol32.h:70
static intptr_t __cdecl execute_command(int const mode, Character const *const file_name, Character const *const *const arguments, Character const *const *const environment)
Definition: spawnv.cpp:168
wchar_t const *const size_t const buffer_size
Definition: stat.cpp:95
int errno_t
Definition: corecrt.h:615

Referenced by _execv(), _execve(), _spawnv(), _spawnve(), _wexecv(), _wexecve(), _wspawnv(), and _wspawnve().

◆ execute_command()

template<typename Character >
static intptr_t __cdecl execute_command ( int const  mode,
Character const *const  file_name,
Character const *const *const  arguments,
Character const *const *const  environment 
)
throw (
)
static

Definition at line 168 of file spawnv.cpp.

174{
175 typedef __crt_char_traits<Character> traits;
176
177 _VALIDATE_RETURN(file_name != nullptr, EINVAL, -1);
178 _VALIDATE_RETURN(arguments != nullptr, EINVAL, -1);
179
181
182 __crt_unique_heap_ptr<Character> command_line;
183 __crt_unique_heap_ptr<Character> environment_block;
184 if (traits::pack_command_line_and_environment(
185 arguments,
186 environment,
187 command_line.get_address_of(),
188 environment_block.get_address_of()
189 ) == -1)
190 {
191 return -1;
192 }
193
194 __crt_unique_heap_ptr<BYTE> handle_data;
195 size_t handle_data_size;
196 if (!accumulate_inheritable_handles(handle_data.get_address_of(), &handle_data_size, mode != _P_DETACH))
197 return -1;
198
199 DWORD creation_flags = 0;
200 if (mode == _P_DETACH)
201 creation_flags |= DETACHED_PROCESS;
202
203 if (should_create_unicode_environment(Character()))
204 creation_flags |= CREATE_UNICODE_ENVIRONMENT;
205
206 _doserrno = 0;
207
208 STARTUPINFOW startup_info = { };
209 startup_info.cb = sizeof(startup_info);
210 startup_info.cbReserved2 = static_cast<WORD>(handle_data_size);
211 startup_info.lpReserved2 = handle_data.get();
212
213 PROCESS_INFORMATION process_info;
214 BOOL const create_process_status = traits::create_process(
215 const_cast<Character*>(file_name),
216 command_line.get(),
217 nullptr,
218 nullptr,
219 TRUE,
220 creation_flags,
221 environment_block.get(),
222 nullptr,
223 &startup_info,
224 &process_info);
225
226 __crt_unique_handle process_handle(process_info.hProcess);
227 __crt_unique_handle thread_handle(process_info.hThread);
228
229 if (!create_process_status)
230 {
232 return -1;
233 }
234
235 if (mode == _P_OVERLAY)
236 {
237 // Destroy ourselves:
238 _exit(0);
239 }
240 else if (mode == _P_WAIT)
241 {
242 WaitForSingleObject(process_info.hProcess, static_cast<DWORD>(-1));
243
244 // Return the termination code and exit code. Note that we return
245 // the full exit code.
247 if (0 != GetExitCodeProcess(process_info.hProcess, &exit_code))
248 {
249 return static_cast<int>(exit_code);
250 }
251 else
252 {
254 return -1;
255 }
256 }
257 else if (mode == _P_DETACH)
258 {
259 /* like totally detached asynchronous spawn, dude,
260 close process handle, return 0 for success */
261 return 0;
262 }
263 else
264 {
265 // Asynchronous spawn: return process handle:
266 return reinterpret_cast<intptr_t>(process_handle.detach());
267 }
268}
void _exit(int exitcode)
Definition: _exit.c:25
void __cdecl __acrt_errno_map_os_error(unsigned long)
Definition: errno.cpp:91
#define TRUE
Definition: types.h:120
BOOL WINAPI GetExitCodeProcess(IN HANDLE hProcess, IN LPDWORD lpExitCode)
Definition: proc.c:1168
unsigned int BOOL
Definition: ntddk_ex.h:94
unsigned long DWORD
Definition: ntddk_ex.h:95
unsigned short WORD
Definition: ntddk_ex.h:93
#define _doserrno
Definition: stdlib.h:131
#define _VALIDATE_CLEAR_OSSERR_RETURN(expr, errorcode, retexpr)
static UINT exit_code
Definition: process.c:78
#define _P_DETACH
Definition: port.h:374
#define _P_WAIT
Definition: port.h:370
static bool accumulate_inheritable_handles(BYTE **const data, size_t *const size, bool const include_std_handles)
Definition: spawnv.cpp:82
static bool __cdecl should_create_unicode_environment(char)
Definition: spawnv.cpp:156
PBYTE lpReserved2
Definition: winbase.h:891
DWORD cb
Definition: winbase.h:877
WORD cbReserved2
Definition: winbase.h:890
DWORD WINAPI WaitForSingleObject(IN HANDLE hHandle, IN DWORD dwMilliseconds)
Definition: synch.c:82
DWORD WINAPI GetLastError(void)
Definition: except.c:1042
#define CREATE_UNICODE_ENVIRONMENT
Definition: winbase.h:189
#define DETACHED_PROCESS
Definition: winbase.h:182

Referenced by common_spawnv().

◆ should_create_unicode_environment() [1/2]

static bool __cdecl should_create_unicode_environment ( char  )
throw (
)
static

Definition at line 156 of file spawnv.cpp.

156{ return false; }

Referenced by execute_command().

◆ should_create_unicode_environment() [2/2]

static bool __cdecl should_create_unicode_environment ( wchar_t  )
throw (
)
static

Definition at line 157 of file spawnv.cpp.

157{ return true; }