mirror of
https://github.com/libsdl-org/SDL.git
synced 2025-09-29 22:48:30 +00:00
205 lines
5.8 KiB
C
205 lines
5.8 KiB
C
/*
|
|
Simple DirectMedia Layer
|
|
Copyright (C) 1997-2025 Sam Lantinga <slouken@libsdl.org>
|
|
|
|
This software is provided 'as-is', without any express or implied
|
|
warranty. In no event will the authors be held liable for any damages
|
|
arising from the use of this software.
|
|
|
|
Permission is granted to anyone to use this software for any purpose,
|
|
including commercial applications, and to alter it and redistribute it
|
|
freely, subject to the following restrictions:
|
|
|
|
1. The origin of this software must not be misrepresented; you must not
|
|
claim that you wrote the original software. If you use this software
|
|
in a product, an acknowledgment in the product documentation would be
|
|
appreciated but is not required.
|
|
2. Altered source versions must be plainly marked as such, and must not be
|
|
misrepresented as being the original software.
|
|
3. This notice may not be removed or altered from any source distribution.
|
|
*/
|
|
#include "SDL_internal.h"
|
|
|
|
#include "SDL_sysprocess.h"
|
|
|
|
|
|
SDL_Process *SDL_CreateProcess(const char * const *args, bool pipe_stdio)
|
|
{
|
|
CHECK_PARAM(!args || !args[0] || !args[0][0]) {
|
|
SDL_InvalidParamError("args");
|
|
return NULL;
|
|
}
|
|
|
|
SDL_Process *process;
|
|
SDL_PropertiesID props = SDL_CreateProperties();
|
|
SDL_SetPointerProperty(props, SDL_PROP_PROCESS_CREATE_ARGS_POINTER, (void *)args);
|
|
if (pipe_stdio) {
|
|
SDL_SetNumberProperty(props, SDL_PROP_PROCESS_CREATE_STDIN_NUMBER, SDL_PROCESS_STDIO_APP);
|
|
SDL_SetNumberProperty(props, SDL_PROP_PROCESS_CREATE_STDOUT_NUMBER, SDL_PROCESS_STDIO_APP);
|
|
}
|
|
process = SDL_CreateProcessWithProperties(props);
|
|
SDL_DestroyProperties(props);
|
|
return process;
|
|
}
|
|
|
|
SDL_Process *SDL_CreateProcessWithProperties(SDL_PropertiesID props)
|
|
{
|
|
const char * const *args = SDL_GetPointerProperty(props, SDL_PROP_PROCESS_CREATE_ARGS_POINTER, NULL);
|
|
#if defined(SDL_PLATFORM_WINDOWS)
|
|
const char *cmdline = SDL_GetStringProperty(props, SDL_PROP_PROCESS_CREATE_CMDLINE_STRING, NULL);
|
|
CHECK_PARAM((!args || !args[0] || !args[0][0]) && (!cmdline || !cmdline[0])) {
|
|
SDL_SetError("Either SDL_PROP_PROCESS_CREATE_ARGS_POINTER or SDL_PROP_PROCESS_CREATE_CMDLINE_STRING must be valid");
|
|
return NULL;
|
|
}
|
|
#else
|
|
CHECK_PARAM(!args || !args[0] || !args[0][0]) {
|
|
SDL_InvalidParamError("SDL_PROP_PROCESS_CREATE_ARGS_POINTER");
|
|
return NULL;
|
|
}
|
|
#endif
|
|
|
|
SDL_Process *process = (SDL_Process *)SDL_calloc(1, sizeof(*process));
|
|
if (!process) {
|
|
return NULL;
|
|
}
|
|
process->background = SDL_GetBooleanProperty(props, SDL_PROP_PROCESS_CREATE_BACKGROUND_BOOLEAN, false);
|
|
|
|
process->props = SDL_CreateProperties();
|
|
if (!process->props) {
|
|
SDL_DestroyProcess(process);
|
|
return NULL;
|
|
}
|
|
SDL_SetBooleanProperty(process->props, SDL_PROP_PROCESS_BACKGROUND_BOOLEAN, process->background);
|
|
|
|
if (!SDL_SYS_CreateProcessWithProperties(process, props)) {
|
|
SDL_DestroyProcess(process);
|
|
return NULL;
|
|
}
|
|
process->alive = true;
|
|
return process;
|
|
}
|
|
|
|
SDL_PropertiesID SDL_GetProcessProperties(SDL_Process *process)
|
|
{
|
|
CHECK_PARAM(!process) {
|
|
return SDL_InvalidParamError("process");
|
|
}
|
|
return process->props;
|
|
}
|
|
|
|
void *SDL_ReadProcess(SDL_Process *process, size_t *datasize, int *exitcode)
|
|
{
|
|
void *result;
|
|
|
|
if (datasize) {
|
|
*datasize = 0;
|
|
}
|
|
if (exitcode) {
|
|
*exitcode = -1;
|
|
}
|
|
|
|
CHECK_PARAM(!process) {
|
|
SDL_InvalidParamError("process");
|
|
return NULL;
|
|
}
|
|
|
|
SDL_IOStream *io = (SDL_IOStream *)SDL_GetPointerProperty(process->props, SDL_PROP_PROCESS_STDOUT_POINTER, NULL);
|
|
if (!io) {
|
|
SDL_SetError("Process not created with I/O enabled");
|
|
return NULL;
|
|
}
|
|
|
|
result = SDL_LoadFile_IO(io, datasize, false);
|
|
|
|
SDL_WaitProcess(process, true, exitcode);
|
|
|
|
return result;
|
|
}
|
|
|
|
SDL_IOStream *SDL_GetProcessInput(SDL_Process *process)
|
|
{
|
|
CHECK_PARAM(!process) {
|
|
SDL_InvalidParamError("process");
|
|
return NULL;
|
|
}
|
|
|
|
SDL_IOStream *io = (SDL_IOStream *)SDL_GetPointerProperty(process->props, SDL_PROP_PROCESS_STDIN_POINTER, NULL);
|
|
if (!io) {
|
|
SDL_SetError("Process not created with standard input available");
|
|
return NULL;
|
|
}
|
|
|
|
return io;
|
|
}
|
|
|
|
SDL_IOStream *SDL_GetProcessOutput(SDL_Process *process)
|
|
{
|
|
CHECK_PARAM(!process) {
|
|
SDL_InvalidParamError("process");
|
|
return NULL;
|
|
}
|
|
|
|
SDL_IOStream *io = (SDL_IOStream *)SDL_GetPointerProperty(process->props, SDL_PROP_PROCESS_STDOUT_POINTER, NULL);
|
|
if (!io) {
|
|
SDL_SetError("Process not created with standard output available");
|
|
return NULL;
|
|
}
|
|
|
|
return io;
|
|
}
|
|
|
|
bool SDL_KillProcess(SDL_Process *process, bool force)
|
|
{
|
|
CHECK_PARAM(!process) {
|
|
return SDL_InvalidParamError("process");
|
|
}
|
|
|
|
if (!process->alive) {
|
|
return SDL_SetError("Process isn't running");
|
|
}
|
|
|
|
return SDL_SYS_KillProcess(process, force);
|
|
}
|
|
|
|
bool SDL_WaitProcess(SDL_Process *process, bool block, int *exitcode)
|
|
{
|
|
CHECK_PARAM(!process) {
|
|
return SDL_InvalidParamError("process");
|
|
}
|
|
|
|
if (!process->alive) {
|
|
if (exitcode) {
|
|
*exitcode = process->exitcode;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
if (SDL_SYS_WaitProcess(process, block, &process->exitcode)) {
|
|
process->alive = false;
|
|
if (exitcode) {
|
|
if (process->background) {
|
|
process->exitcode = 0;
|
|
}
|
|
*exitcode = process->exitcode;
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void SDL_DestroyProcess(SDL_Process *process)
|
|
{
|
|
if (!process) {
|
|
return;
|
|
}
|
|
|
|
// Check to see if the process has exited, will reap zombies on POSIX platforms
|
|
if (process->alive) {
|
|
SDL_WaitProcess(process, false, NULL);
|
|
}
|
|
|
|
SDL_SYS_DestroyProcess(process);
|
|
SDL_DestroyProperties(process->props);
|
|
SDL_free(process);
|
|
}
|