www.nico.schottelius.org/software/cinit/browse_source/cinit-0.3pre7/client/exec_svc.c
Nico Schottelius 423ba10303 import cinit from unix.schottelius.org
Signed-off-by: Nico Schottelius <nico@ikn.schottelius.org>
2009-09-16 12:53:45 +02:00

264 lines
6.1 KiB
C

/*
* (c) 2005 Nico Schottelius (nico-linux at schottelius.org)
* run_svc
* part of cinit
*/
#include <errno.h>
#include <fcntl.h>
#include <limits.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
#include "cinit.h"
/***********************************************************************
* exec_svc: exec service abspath C_ON or C_OFF
*/
inline pid_t exec_svc(char *abspath, int action)
{
int tmp;
char *p, pathtmp[PATH_MAX];
char *sbuf = NULL, **nargv = NULL, **nenv = NULL;
int fd, argc;
struct stat buf;
cpid = fork();
if( cpid == -1 ) {
perror(MSG_ERR_FORK);
return 0;
}
/***** PARENT ******/
if(cpid > 0) {
waitpid(cpid,&tmp,0);
if( WIFEXITED(tmp) ) {
if( ! WEXITSTATUS(tmp) ) {
return cpid;
}
}
SERVICE_LOG(abspath,LOG_SVC_FAIL);
return 0;
}
/*********** CHILD EXECUTION ***********/
strcpy(pathtmp,abspath);
strcat(pathtmp,SLASH);
if(action == CMD_START_SVC) {
strcat(pathtmp,C_ON);
} else {
strcat(pathtmp,C_OFF);
}
/* readlink retrieves real name, if it is a symlink */
if ( ( tmp = readlink(pathtmp,pathtmp,PATH_MAX) ) == -1) {
/* nothing there? fine, exit */
if(errno == ENOENT) {
_exit(0);
} else if (errno != EINVAL) {
perror(MSG_READLINK);
_exit(1);
}
/* restore original name, file is not a link */
strcpy(pathtmp,abspath);
strcat(pathtmp,SLASH);
if(action == CMD_START_SVC) {
strcat(pathtmp,C_ON);
} else {
strcat(pathtmp,C_OFF);
}
tmp = strlen(pathtmp);
} else {
/* add '\0', readlink forgets it */
pathtmp[tmp] = '\0';
}
++tmp; /* the byte to add to memory for \0;
neither readlink nor strlen count the \0 */
/********** build argv0 ********/
nargv = (char **) malloc( sizeof(char *) );
if(nargv == NULL) _exit(1);
*nargv = (char *) malloc( tmp );
if(*nargv == NULL) _exit(1);
strcpy(*nargv,pathtmp);
/********************** read params *********************/
strcpy(pathtmp,abspath);
strcat(pathtmp,SLASH);
if(action == CMD_START_SVC) {
strcat(pathtmp,C_ONARG);
} else {
strcat(pathtmp,C_OFFARG);
}
/* open params file */
if( !stat(pathtmp,&buf) ) {
fd = open(pathtmp,O_RDONLY);
if(fd == -1) {
perror(MSG_ERR_OPEN);
_exit(1);
}
argc = 0;
/* fill sbuf with content */
while ( (tmp = read(fd,pathtmp,PATH_MAX) ) != 0 ) {
if(tmp == -1) {
perror(MSG_ERR_READ);
_exit(1);
}
sbuf = realloc(sbuf,argc + tmp + 1);
strncpy(&sbuf[argc],pathtmp,tmp);
argc += tmp;
}
close(fd);
if(argc) {
sbuf[argc] = '\0'; /* terminate string */
}
} else {
sbuf = NULL;
}
/************** build string **************/
/* The idea is to reuse the allocated memory. We'll simply
* replace the \n with \0 and set a char* to it.
* We'll lose the last byte (the initial \0), but we
* don't care, as we die some seconds later with execv()
* oh, we'll only lose it, if the file is terminated by \n */
argc = 1; /* argv0 */
while( sbuf != NULL) {
p = strchr(sbuf,'\n');
nargv = realloc(nargv, sizeof(char *) * (argc + 1));
if(nargv == NULL) {
LOG(MSG_ERR_ALLOC);
_exit(1);
}
/* set to the beginning */
nargv[argc] = sbuf;
/* if we found \n */
if(p != NULL) {
*p = '\0';
} else {
/* set to the end of sbuf, not to the \0, but one before */
p = sbuf + (strlen(sbuf)-1);
}
/* if next byte is 0, the end of string is found */
if( *(p+1) == '\0') {
sbuf = NULL;
} else {
sbuf = p+1;
}
++argc;
}
/************ close argv list **************/
nargv = realloc(nargv, sizeof(char *) * (argc + 1)); /* 1: NULL-pointer */
if(nargv == NULL) {
LOG(MSG_ERR_ALLOC);
_exit(1);
}
nargv[argc] = NULL; /* terminate argv list */
/********************** read environment *********************/
strcpy(pathtmp,abspath);
strcat(pathtmp,SLASH);
if(action == CMD_START_SVC) {
strcat(pathtmp,C_ONENV);
} else {
strcat(pathtmp,C_OFFENV);
}
argc = 0;
sbuf = NULL;
if( !stat(pathtmp,&buf) ) {
fd = open(pathtmp,O_RDONLY);
/* file exists, failing to open it is an error */
if(fd == -1) {
perror(MSG_ERR_OPEN);
_exit(1);
}
while ( (tmp = read(fd,pathtmp,PATH_MAX) ) != 0 ) {
if(tmp == -1) {
perror(MSG_ERR_READ);
_exit(1);
}
sbuf = realloc(sbuf,argc + tmp + 1);
strncpy(&sbuf[argc],pathtmp,tmp);
argc += tmp;
}
close(fd);
if(argc)
sbuf[argc] = '\0'; /* terminate string */
}
/************** build environment string **************/
argc = 0;
while( sbuf != NULL ) {
p = strchr(sbuf,'\n');
nenv = realloc(nenv, sizeof(char *) * (argc + 1));
if(nenv == NULL) {
LOG(MSG_ERR_ALLOC);
_exit(1);
}
nenv[argc] = sbuf;
/* if we found \n */
if(p != NULL) {
*p = '\0';
} else {
/* set to the end of sbuf, not to the \0, but one before */
p = sbuf + (strlen(sbuf)-1);
}
if( *(p+1) == '\0') {
sbuf = NULL;
} else {
sbuf = p+1;
}
++argc;
}
/************ close env list **************/
nenv = realloc(nenv, sizeof(char *) * (argc + 1));
if(nenv == NULL) {
LOG(MSG_ERR_ALLOC);
_exit(1);
}
nenv[argc] = NULL;
/****************** EXECUTE ********************/
execve(nargv[0],nargv,nenv);
/* we should never reach this block */
P_EXEC_FAILED(nargv[0]);
perror(MSG_ERR_EXECVE);
/* FIXME: remove debug, remove double printing without loosing info */
sleep(3); /* debug */
_exit(1);
}