mirror of
https://github.com/gryf/wmaker.git
synced 2025-12-20 04:48:06 +01:00
Add the BSD version of GetCommandForPid()
- tested on Net, Free, Open and DragonFly - fix up the linux version (terminate argv with a null ptr) - add error handling to file ops in the linux version - add a stub version for platforms not supported
This commit is contained in:
committed by
Carlos R. Mafra
parent
294ea56d84
commit
ab9c85a11d
171
src/osdep/bsd.c
Normal file
171
src/osdep/bsd.c
Normal file
@@ -0,0 +1,171 @@
|
||||
|
||||
#if defined( FREEBSD ) || defined( DRAGONFLYBSD )
|
||||
# include <sys/types.h>
|
||||
#else /* OPENBSD || NETBSD */
|
||||
# include <sys/param.h>
|
||||
#endif
|
||||
#include <sys/sysctl.h>
|
||||
|
||||
#include <assert.h>
|
||||
|
||||
#if defined( OPENBSD )
|
||||
# include <kvm.h>
|
||||
# include <limits.h> /* _POSIX2_LINE_MAX */
|
||||
#endif
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#if defined( OPENBSD )
|
||||
# include <string.h>
|
||||
#endif
|
||||
|
||||
#include <unistd.h>
|
||||
|
||||
#include <WINGs/WUtil.h>
|
||||
|
||||
#include "../wconfig.h"
|
||||
|
||||
/*
|
||||
* copy argc and argv for an existing process identified by `pid'
|
||||
* into suitable storage given in ***argv and *argc.
|
||||
*
|
||||
* subsequent calls use the same static area for argv and argc.
|
||||
*
|
||||
* returns 0 for failure, in which case argc := 0 and argv := NULL
|
||||
* returns 1 for success
|
||||
*/
|
||||
/*
|
||||
* NetBSD, FreeBSD and DragonFlyBSD supply the necessary information via
|
||||
* sysctl(3). The result is a plain simple flat octet stream and its length.
|
||||
* The octet stream represents argv, with members separated by a null character.
|
||||
* The argv array returned by GetCommandForPid() consists of pointers into this
|
||||
* stream (which is stored in a static array, `args'). Net and Free/DFly only
|
||||
* differ slightly in the MIB vector given to sysctl(3). Free and DFly are
|
||||
* identical.
|
||||
*
|
||||
* OpenBSD supplies the necessary informationvia kvm(3) in the form of a real
|
||||
* argv array. This array is flattened to be in the same way as Net/Free/DFly
|
||||
* returns the arguments in the first place. This is done in order for the
|
||||
* storage (`args') to easily be made static, which means some memory bytes
|
||||
* are sacrificed to save hoops of memory management.
|
||||
*/
|
||||
Bool GetCommandForPid(int pid, char ***argv, int *argc)
|
||||
{
|
||||
/*
|
||||
* it just returns failure if the sysctl calls fail; since there's
|
||||
* apparently no harm done to the caller because of this, it seems
|
||||
* more user-friendly than to bomb out.
|
||||
*/
|
||||
int j, mib[4];
|
||||
unsigned int i;
|
||||
size_t count;
|
||||
static char *args = NULL;
|
||||
static int argmax = 0;
|
||||
#if defined( OPENBSD )
|
||||
char kvmerr[_POSIX2_LINE_MAX]; /* for kvm*() error reporting */
|
||||
int procs; /* kvm_getprocs() */
|
||||
kvm_t *kd;
|
||||
struct kinfo_proc *kp;
|
||||
char **nargv; /* kvm_getarg() */
|
||||
#endif
|
||||
|
||||
*argv = NULL;
|
||||
*argc = 0;
|
||||
|
||||
/* the system-wide limit */
|
||||
if (argmax == 0) { /* it hopefully doesn't change at runtime *g* */
|
||||
mib[0] = CTL_KERN;
|
||||
mib[1] = KERN_ARGMAX;
|
||||
mib[2] = 0;
|
||||
mib[4] = 0;
|
||||
|
||||
count = sizeof(argmax);
|
||||
if (sysctl(mib, 2, &argmax, &count, NULL, 0) == -1)
|
||||
return False;
|
||||
}
|
||||
|
||||
/* if argmax is still 0, something went very seriously wrong */
|
||||
assert( argmax > 0);
|
||||
|
||||
/* space for args; no need to free before returning even on errors */
|
||||
if (args == NULL)
|
||||
args = (char *)wmalloc(argmax);
|
||||
|
||||
#if defined( OPENBSD )
|
||||
/* kvm descriptor */
|
||||
if ((kd = kvm_openfiles(NULL, NULL, NULL, KVM_NO_FILES, kvmerr)) == NULL)
|
||||
return False;
|
||||
|
||||
procs = 0;
|
||||
/* the process we are interested in */
|
||||
if ((kp = kvm_getprocs(kd, KERN_PROC_PID, pid, &procs)) == NULL || procs == 0)
|
||||
/* if kvm_getprocs() bombs out or does not find the process */
|
||||
return False;
|
||||
|
||||
/* get it's argv */
|
||||
if ((nargv = kvm_getargv(kd, kp, 0)) == NULL)
|
||||
return False;
|
||||
|
||||
/* flatten nargv into args */
|
||||
count = 0;
|
||||
memset(args, 0, argmax);
|
||||
/*
|
||||
* must have this much free space in `args' in order for the current
|
||||
* iteration not to overflow it: we are at `count', and will append
|
||||
* the next ((*argc)+1) arg and a null (+1)
|
||||
* technically, overflow (or truncation, which isn't handled) can not
|
||||
* happen (should not, at least).
|
||||
*/
|
||||
#define ARGSPACE ( count + strlen(nargv[ (*argc) + 1 ] ) + 1 )
|
||||
while (nargv[*argc] && ARGSPACE < argmax ) {
|
||||
memcpy(args + count, nargv[*argc], strlen(nargv[*argc]));
|
||||
count += strlen(nargv[*argc]) + 1;
|
||||
(*argc)++;
|
||||
}
|
||||
#undef ARGSPACE
|
||||
/* by now *argc is correct as a byproduct */
|
||||
|
||||
kvm_close(kd);
|
||||
#else /* FREEBSD || NETBSD || DRAGONFLYBSD */
|
||||
|
||||
mib[0] = CTL_KERN;
|
||||
#if defined( NETBSD )
|
||||
mib[1] = KERN_PROC_ARGS;
|
||||
mib[2] = pid;
|
||||
mib[3] = KERN_PROC_ARGV;
|
||||
#elif defined( FREEBSD ) || defined( DRAGONFLYBSD )
|
||||
mib[1] = KERN_PROC;
|
||||
mib[2] = KERN_PROC_ARGS;
|
||||
mib[3] = pid;
|
||||
#endif
|
||||
|
||||
count = argmax;
|
||||
/* canary */
|
||||
*args = 0;
|
||||
if (sysctl(mib, 4, args, &count, NULL, 0) == -1 || *args == 0)
|
||||
return False;
|
||||
|
||||
/* args is a flattened series of null-terminated strings */
|
||||
for (i = 0; i < count; i++)
|
||||
if (args[i] == '\0')
|
||||
(*argc)++;
|
||||
#endif
|
||||
|
||||
*argv = (char **)wmalloc(sizeof(char *) * (*argc + 1 /* term. null ptr */));
|
||||
(*argv)[0] = args;
|
||||
|
||||
/* go through args, set argv[$next] to the beginning of each string */
|
||||
for (i = 0, j = 1; i < count; i++) {
|
||||
if (args[i] != '\0')
|
||||
continue;
|
||||
if (i < count - 1)
|
||||
(*argv)[j++] = &args[i + 1];
|
||||
if (j == *argc)
|
||||
break;
|
||||
}
|
||||
|
||||
/* the list of arguments must be terminated by a null pointer */
|
||||
(*argv)[j] = NULL;
|
||||
return True;
|
||||
}
|
||||
@@ -17,7 +17,9 @@
|
||||
* copy argc and argv for an existing process identified by `pid'
|
||||
* into suitable storage given in ***argv and *argc.
|
||||
*
|
||||
* returns 0 for failure, in which case argc := 0 and arv := NULL
|
||||
* subsequent calls use the same static area for argv and argc.
|
||||
*
|
||||
* returns 0 for failure, in which case argc := 0 and argv := NULL
|
||||
* returns 1 for success
|
||||
*/
|
||||
Bool GetCommandForPid(int pid, char ***argv, int *argc)
|
||||
@@ -31,16 +33,27 @@ Bool GetCommandForPid(int pid, char ***argv, int *argc)
|
||||
|
||||
/* cmdline is a flattened series of null-terminated strings */
|
||||
snprintf(buf, sizeof(buf), "/proc/%d/cmdline", pid);
|
||||
if ((fd = open(buf, O_RDONLY)) == -1)
|
||||
return False;
|
||||
|
||||
/* XXX: read/close errors */
|
||||
if ((count = read(fd, buf, sizeof(buf))) == -1) {
|
||||
close(fd);
|
||||
while (1) {
|
||||
if ((fd = open(buf, O_RDONLY)) != -1)
|
||||
break;
|
||||
if (errno == EINTR)
|
||||
continue;
|
||||
return False;
|
||||
}
|
||||
close(fd);
|
||||
|
||||
while (1) {
|
||||
if ((count = read(fd, buf, sizeof(buf))) != -1)
|
||||
break;
|
||||
if (errno == EINTR)
|
||||
continue;
|
||||
return False;
|
||||
}
|
||||
|
||||
do {
|
||||
close(fd);
|
||||
} while (errno == EINTR);
|
||||
|
||||
/* count args */
|
||||
for (i = 0; i < count; i++)
|
||||
if (buf[i] == '\0')
|
||||
(*argc)++;
|
||||
@@ -48,7 +61,7 @@ Bool GetCommandForPid(int pid, char ***argv, int *argc)
|
||||
if (*argc == 0)
|
||||
return False;
|
||||
|
||||
*argv = (char **)wmalloc(sizeof(char *) * *argc);
|
||||
*argv = (char **)wmalloc(sizeof(char *) * (*argc + 1 /* term. null ptr */));
|
||||
(*argv)[0] = buf;
|
||||
|
||||
/* go through buf, set argv[$next] to the beginning of each string */
|
||||
@@ -61,5 +74,7 @@ Bool GetCommandForPid(int pid, char ***argv, int *argc)
|
||||
break;
|
||||
}
|
||||
|
||||
/* the list of arguments must be terminated by a null pointer */
|
||||
(*argv)[j] = NULL;
|
||||
return True;
|
||||
}
|
||||
|
||||
11
src/osdep/stub.c
Normal file
11
src/osdep/stub.c
Normal file
@@ -0,0 +1,11 @@
|
||||
|
||||
#include <WINGs/WUtil.h>
|
||||
#include "../wconfig.h"
|
||||
|
||||
Bool GetCommandForPid(int pid, char ***argv, int *argc)
|
||||
{
|
||||
*argv = NULL;
|
||||
*argc = 0;
|
||||
|
||||
return False;
|
||||
}
|
||||
49
src/osdep/test.c
Normal file
49
src/osdep/test.c
Normal file
@@ -0,0 +1,49 @@
|
||||
|
||||
#include <stdio.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include <WINGs/WUtil.h>
|
||||
|
||||
/*
|
||||
* gcc -D{$os} -I ../../WINGs ${includes...} -Wall -Wextra -g -ggdb3 -c -o ${plat}.o ${plat}.c
|
||||
* gcc -I ../../WINGs ${includes...} -g -ggdb3 -c -o test.o test.c
|
||||
* gcc -g -ggdb3 -o test ${plat}.o ../../WINGs/.libs/libWUtil.a test.o
|
||||
*
|
||||
* $ ./test 1 2 'foo bar' "`date`" `date`
|
||||
* arg[0] = [./test]
|
||||
* arg[1] = [1]
|
||||
* arg[2] = [2]
|
||||
* arg[3] = [foo bar]
|
||||
* arg[4] = [Sat Mar 20 18:36:22 CET 2010]
|
||||
* arg[5] = [Sat]
|
||||
* arg[6] = [Mar]
|
||||
* arg[7] = [20]
|
||||
* arg[8] = [18:36:22]
|
||||
* arg[9] = [CET]
|
||||
* arg[10] = [2010]
|
||||
* $
|
||||
*/
|
||||
|
||||
Bool GetCommandForPid(int pid, char ***argv, int *argc);
|
||||
extern char *__progname;
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
|
||||
char **nargv;
|
||||
int i, nargc;
|
||||
|
||||
if (argc < 2) {
|
||||
printf("Usage: %s arg arg arg ...\n", __progname);
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (GetCommandForPid(getpid(), &nargv, &nargc) == False) {
|
||||
printf("GetCommandForPid() failed\n");
|
||||
} else {
|
||||
printf("nargv = %d\n", nargc);
|
||||
for(i = 0; i < nargc; i++)
|
||||
printf("arg[%d] = [%s]\n", i, nargv[i]);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
Reference in New Issue
Block a user