1
0
mirror of https://github.com/gryf/wmaker.git synced 2025-12-19 04:20:27 +01:00
Files
wmaker/src/osdep_linux.c
Tamas TEVESZ 71aa4f2884 Switch file ops to stdio
- Does away with the O_BINARY abomination
- as a byproduct, plugs an fd leak in wcolorpanel.c:fetchFile()
- sprinkle some fsync()s to files that have been written to (this
  needs to be done everywhere)

+ fix brown paper bag thinko in configure.ac
2010-03-27 10:31:13 +01:00

85 lines
1.7 KiB
C

#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <fcntl.h>
#include <limits.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <WINGs/WUtil.h>
#include "wconfig.h"
#define RETRY( x ) do { \
x; \
} while (errno == EINTR);
/*
* 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
*/
Bool GetCommandForPid(int pid, char ***argv, int *argc)
{
static char buf[_POSIX_ARG_MAX];
int fd, i, j;
ssize_t count;
*argv = NULL;
*argc = 0;
/* cmdline is a flattened series of null-terminated strings */
snprintf(buf, sizeof(buf), "/proc/%d/cmdline", pid);
while (1) {
/* not switching this to stdio yet, as this does not need
* to be portable, and i'm lazy */
if ((fd = open(buf, O_RDONLY)) != -1)
break;
if (errno == EINTR)
continue;
return False;
}
while (1) {
if ((count = read(fd, buf, sizeof(buf))) != -1)
break;
if (errno == EINTR)
continue;
RETRY( close(fd) )
return False;
}
RETRY( close(fd) )
/* count args */
for (i = 0; i < count; i++)
if (buf[i] == '\0')
(*argc)++;
if (*argc == 0)
return False;
*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 */
for (i = 0, j = 1; i < count; i++) {
if (buf[i] != '\0')
continue;
if (i < count - 1)
(*argv)[j++] = &buf[i + 1];
if (j == *argc)
break;
}
/* the list of arguments must be terminated by a null pointer */
(*argv)[j] = NULL;
return True;
}