2012-03-02 15:00:11 +01:00
|
|
|
/*******************************************************************************
|
2011-08-27 20:57:39 +02:00
|
|
|
|
2012-03-02 15:00:11 +01:00
|
|
|
COPYRIGHT(C) JONAS 'SORTIE' TERMANSEN 2011, 2012.
|
2011-08-27 20:57:39 +02:00
|
|
|
|
|
|
|
This file is part of LibMaxsi.
|
|
|
|
|
|
|
|
LibMaxsi is free software: you can redistribute it and/or modify it under
|
|
|
|
the terms of the GNU Lesser General Public License as published by the Free
|
|
|
|
Software Foundation, either version 3 of the License, or (at your option)
|
|
|
|
any later version.
|
|
|
|
|
|
|
|
LibMaxsi is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
|
|
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
2012-03-02 15:00:11 +01:00
|
|
|
FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
|
|
|
|
details.
|
2011-08-27 20:57:39 +02:00
|
|
|
|
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
|
|
|
along with LibMaxsi. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
process.cpp
|
|
|
|
Exposes system calls for process creation and management.
|
|
|
|
|
2012-03-02 15:00:11 +01:00
|
|
|
*******************************************************************************/
|
2011-08-27 20:57:39 +02:00
|
|
|
|
2012-04-04 01:38:45 +02:00
|
|
|
#define _WANT_ENVIRON
|
2012-02-12 02:03:34 +01:00
|
|
|
#include <libmaxsi/platform.h>
|
|
|
|
#include <libmaxsi/syscall.h>
|
|
|
|
#include <libmaxsi/process.h>
|
Implemented large parts of the stdio(3), including fprintf.
Made FILE an interface to various backends. This allows application writers
to override the standard FILE API functions with their own backends. This
is highly unportable - it'd be nice if a real standard existed for this.
glibc already does something like this internally, but AFAIK you can't hook
into it.
Added fdopen(3), fopen(3), fregister(3), funregister(3), fread(3),
fwrite(3), fseek(3), clearerr(3), ferror(3), feof(3), rewind(3), ftell(3),
fflush(3), fclose(3), fileno(3), fnewline(3), fcloseall(3), memset(3),
stdio(3), vfprintf(3), fprintf(3), and vprintf(3).
Added a file-descriptor backend to the FILE API.
fd's {0, 1, 2} are now initialized as stdin, stdout, and stderr when the
standard library initializes.
fcloseall(3) is now called on exit(3).
decl/intn_t_.h now @include(size_t.h) instead of declaring it itself.
Added <stdint.h>.
The following programs now flush stdout: cat(1), clear(1), editor(1),
init(1), mxsh(1).
printf(3) is now hooked up against vprintf(3), while Maxsi::PrintF
remains using the system call, for now.
2011-12-24 04:08:10 +01:00
|
|
|
#include <stdio.h>
|
2012-04-29 14:37:11 +02:00
|
|
|
#include <stdarg.h>
|
Added readdir(3), closedir(3), rewinddir(3), dirfd(3), fdopendir(3),
opendir(3).
Also added non-standard dregister(3), dunregister(3), dclearerr(3),
derror(3), deof(3), dnewdir(3), dcloseall(3).
2012-01-14 20:54:02 +01:00
|
|
|
#include <dirent.h>
|
2012-04-04 01:38:45 +02:00
|
|
|
#include <unistd.h>
|
2012-04-29 14:37:11 +02:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2011-08-27 20:57:39 +02:00
|
|
|
|
|
|
|
namespace Maxsi
|
|
|
|
{
|
|
|
|
namespace Process
|
|
|
|
{
|
2011-12-09 12:41:06 +01:00
|
|
|
DEFN_SYSCALL1_VOID(SysExit, SYSCALL_EXIT, int);
|
2012-03-02 15:00:11 +01:00
|
|
|
DEFN_SYSCALL3(int, SysExecVE, SYSCALL_EXEC, const char*, char* const*, char* const*);
|
2012-04-05 23:00:47 +02:00
|
|
|
DEFN_SYSCALL2(pid_t, SysSForkR, SYSCALL_SFORKR, int, sforkregs_t*);
|
2011-12-09 12:41:06 +01:00
|
|
|
DEFN_SYSCALL0(pid_t, SysGetPID, SYSCALL_GETPID);
|
|
|
|
DEFN_SYSCALL0(pid_t, SysGetParentPID, SYSCALL_GETPPID);
|
|
|
|
DEFN_SYSCALL3(pid_t, SysWait, SYSCALL_WAIT, pid_t, int*, int);
|
2011-08-27 20:57:39 +02:00
|
|
|
|
2011-09-13 16:49:08 +02:00
|
|
|
void Abort()
|
|
|
|
{
|
|
|
|
// TODO: Send SIGABRT instead!
|
|
|
|
Exit(128 + 6);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" void abort() { return Abort(); }
|
|
|
|
|
2011-11-05 18:52:11 +01:00
|
|
|
extern "C" void _exit(int status)
|
2011-09-13 16:49:08 +02:00
|
|
|
{
|
2011-11-05 18:52:11 +01:00
|
|
|
SysExit(status);
|
|
|
|
}
|
|
|
|
|
2012-03-02 15:00:11 +01:00
|
|
|
extern "C" int execve(const char* pathname, char* const* argv,
|
|
|
|
char* const* envp)
|
|
|
|
{
|
|
|
|
return SysExecVE(pathname, argv, envp);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" int execv(const char* pathname, char* const* argv)
|
|
|
|
{
|
2012-04-04 01:38:45 +02:00
|
|
|
return execve(pathname, argv, environ);
|
2012-03-02 15:00:11 +01:00
|
|
|
}
|
|
|
|
|
2012-04-29 14:37:11 +02:00
|
|
|
// Note that the only PATH variable in Sortix is the one used here.
|
|
|
|
extern "C" int execvpe(const char* filename, char* const* argv,
|
|
|
|
char* const* envp)
|
|
|
|
{
|
|
|
|
if ( strchr(filename, '/') )
|
|
|
|
return execve(filename, argv, envp);
|
|
|
|
size_t filenamelen = strlen(filename);
|
|
|
|
const char* PATH = "/bin";
|
|
|
|
size_t pathlen = strlen(PATH);
|
|
|
|
char* pathname = (char*) malloc(filenamelen + 1 + pathlen + 1);
|
|
|
|
if ( !pathname ) { return -1; }
|
|
|
|
stpcpy(stpcpy(stpcpy(pathname, PATH), "/"), filename);
|
|
|
|
int result = execve(pathname, argv, envp);
|
|
|
|
free(pathname);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" int execvp(const char* filename, char* const* argv)
|
|
|
|
{
|
|
|
|
return execvpe(filename, argv, environ);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" int vexecl(const char* pathname, va_list args)
|
|
|
|
{
|
|
|
|
va_list iter;
|
|
|
|
va_copy(iter, args);
|
|
|
|
size_t numargs = 0;
|
|
|
|
while ( va_arg(iter, const char*) ) { numargs++; }
|
|
|
|
va_end(iter);
|
|
|
|
char** argv = (char**) malloc(sizeof(char*) * (numargs+1));
|
|
|
|
if ( !argv ) { return -1; }
|
|
|
|
for ( size_t i = 0; i <= numargs; i++ )
|
|
|
|
{
|
|
|
|
argv[i] = (char*) va_arg(args, const char*);
|
|
|
|
}
|
|
|
|
int result = execv(pathname, argv);
|
|
|
|
free(argv);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" int vexeclp(const char* filename, va_list args)
|
|
|
|
{
|
|
|
|
va_list iter;
|
|
|
|
va_copy(iter, args);
|
|
|
|
size_t numargs = 0;
|
|
|
|
while ( va_arg(iter, const char*) ) { numargs++; }
|
|
|
|
va_end(iter);
|
|
|
|
char** argv = (char**) malloc(sizeof(char*) * (numargs+1));
|
|
|
|
if ( !argv ) { return -1; }
|
|
|
|
for ( size_t i = 0; i <= numargs; i++ )
|
|
|
|
{
|
|
|
|
argv[i] = (char*) va_arg(args, const char*);
|
|
|
|
}
|
|
|
|
int result = execvp(filename, argv);
|
|
|
|
free(argv);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" int vexecle(const char* pathname, va_list args)
|
|
|
|
{
|
|
|
|
va_list iter;
|
|
|
|
va_copy(iter, args);
|
|
|
|
size_t numargs = 0;
|
|
|
|
while ( va_arg(iter, const char*) ) { numargs++; }
|
|
|
|
va_end(iter);
|
|
|
|
numargs--; // envp
|
|
|
|
char** argv = (char**) malloc(sizeof(char*) * (numargs+1));
|
|
|
|
if ( !argv ) { return -1; }
|
|
|
|
for ( size_t i = 0; i < numargs; i++ )
|
|
|
|
{
|
|
|
|
argv[i] = (char*) va_arg(args, const char*);
|
|
|
|
}
|
|
|
|
argv[numargs] = NULL;
|
|
|
|
char* const* envp = va_arg(args, char* const*);
|
|
|
|
int result = execve(pathname, argv, envp);
|
|
|
|
free(argv);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" int execl(const char* pathname, ...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start(args, pathname);
|
|
|
|
int result = vexecl(pathname, args);
|
|
|
|
va_end(args);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" int execlp(const char* filename, ...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start(args, filename);
|
|
|
|
int result = vexeclp(filename, args);
|
|
|
|
va_end(args);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" int execle(const char* pathname, ...)
|
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start(args, pathname);
|
|
|
|
int result = vexecle(pathname, args);
|
|
|
|
va_end(args);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2012-05-29 00:05:27 +02:00
|
|
|
extern "C" void call_exit_handlers(int status);
|
|
|
|
|
2011-11-05 18:52:11 +01:00
|
|
|
DUAL_FUNCTION(void, exit, Exit, (int status))
|
|
|
|
{
|
2012-05-29 00:05:27 +02:00
|
|
|
call_exit_handlers(status);
|
Added readdir(3), closedir(3), rewinddir(3), dirfd(3), fdopendir(3),
opendir(3).
Also added non-standard dregister(3), dunregister(3), dclearerr(3),
derror(3), deof(3), dnewdir(3), dcloseall(3).
2012-01-14 20:54:02 +01:00
|
|
|
dcloseall();
|
Implemented large parts of the stdio(3), including fprintf.
Made FILE an interface to various backends. This allows application writers
to override the standard FILE API functions with their own backends. This
is highly unportable - it'd be nice if a real standard existed for this.
glibc already does something like this internally, but AFAIK you can't hook
into it.
Added fdopen(3), fopen(3), fregister(3), funregister(3), fread(3),
fwrite(3), fseek(3), clearerr(3), ferror(3), feof(3), rewind(3), ftell(3),
fflush(3), fclose(3), fileno(3), fnewline(3), fcloseall(3), memset(3),
stdio(3), vfprintf(3), fprintf(3), and vprintf(3).
Added a file-descriptor backend to the FILE API.
fd's {0, 1, 2} are now initialized as stdin, stdout, and stderr when the
standard library initializes.
fcloseall(3) is now called on exit(3).
decl/intn_t_.h now @include(size_t.h) instead of declaring it itself.
Added <stdint.h>.
The following programs now flush stdout: cat(1), clear(1), editor(1),
init(1), mxsh(1).
printf(3) is now hooked up against vprintf(3), while Maxsi::PrintF
remains using the system call, for now.
2011-12-24 04:08:10 +01:00
|
|
|
fcloseall();
|
2011-11-06 23:51:02 +01:00
|
|
|
_exit(status);
|
2011-09-13 16:49:08 +02:00
|
|
|
}
|
|
|
|
|
2012-04-05 23:00:47 +02:00
|
|
|
extern "C" pid_t sforkr(int flags, sforkregs_t* regs)
|
2011-09-21 20:52:29 +02:00
|
|
|
{
|
2012-04-05 23:00:47 +02:00
|
|
|
return SysSForkR(flags, regs);
|
2011-09-21 20:52:29 +02:00
|
|
|
}
|
|
|
|
|
2012-04-05 23:00:47 +02:00
|
|
|
extern "C" pid_t __call_sforkr_with_regs(int flags);
|
|
|
|
|
2012-04-04 00:29:25 +02:00
|
|
|
extern "C" pid_t sfork(int flags)
|
2012-04-02 16:21:49 +02:00
|
|
|
{
|
2012-04-05 23:00:47 +02:00
|
|
|
return __call_sforkr_with_regs(flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
DUAL_FUNCTION(pid_t, fork, Fork, ())
|
|
|
|
{
|
|
|
|
return sfork(SFFORK);
|
2012-04-02 16:21:49 +02:00
|
|
|
}
|
|
|
|
|
2011-09-21 20:52:29 +02:00
|
|
|
DUAL_FUNCTION(pid_t, getpid, GetPID, ())
|
|
|
|
{
|
|
|
|
return SysGetPID();
|
|
|
|
}
|
|
|
|
|
|
|
|
DUAL_FUNCTION(pid_t, getppid, GetParentPID, ())
|
|
|
|
{
|
|
|
|
return SysGetParentPID();
|
|
|
|
}
|
2011-11-06 22:00:29 +01:00
|
|
|
|
|
|
|
extern "C" pid_t waitpid(pid_t pid, int* status, int options)
|
|
|
|
{
|
|
|
|
return SysWait(pid, status, options);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" pid_t wait(int* status)
|
|
|
|
{
|
|
|
|
return waitpid(-1, status, 0);
|
|
|
|
}
|
2011-08-27 20:57:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|