2015-10-07 09:16:18 +00:00
|
|
|
/* See LICENSE file for copyright and license details. */
|
|
|
|
#include <sys/file.h>
|
|
|
|
#include <sys/wait.h>
|
|
|
|
|
|
|
|
#include <errno.h>
|
2015-10-07 14:20:09 +00:00
|
|
|
#include <fcntl.h>
|
2015-10-07 09:16:18 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include "util.h"
|
|
|
|
|
|
|
|
static void
|
|
|
|
usage(void)
|
|
|
|
{
|
2015-10-08 15:43:09 +00:00
|
|
|
eprintf("usage: %s [-nosux] file cmd [arg ...]\n", argv0);
|
2015-10-07 09:16:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
main(int argc, char *argv[])
|
|
|
|
{
|
2015-10-08 15:43:09 +00:00
|
|
|
int fd, status, savederrno, flags = LOCK_EX, nonblk = 0, oflag = 0;
|
2015-10-07 09:16:18 +00:00
|
|
|
pid_t pid;
|
|
|
|
|
|
|
|
ARGBEGIN {
|
|
|
|
case 'n':
|
|
|
|
nonblk = LOCK_NB;
|
|
|
|
break;
|
2015-10-08 15:43:09 +00:00
|
|
|
case 'o':
|
|
|
|
oflag = 1;
|
|
|
|
break;
|
2015-10-07 09:16:18 +00:00
|
|
|
case 's':
|
|
|
|
flags = LOCK_SH;
|
|
|
|
break;
|
|
|
|
case 'u':
|
|
|
|
flags = LOCK_UN;
|
|
|
|
break;
|
|
|
|
case 'x':
|
2015-10-07 09:26:04 +00:00
|
|
|
flags = LOCK_EX;
|
2015-10-07 09:16:18 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
usage();
|
2015-11-01 10:16:49 +00:00
|
|
|
} ARGEND
|
2015-10-07 09:16:18 +00:00
|
|
|
|
|
|
|
if (argc < 2)
|
|
|
|
usage();
|
|
|
|
|
2015-10-07 09:22:46 +00:00
|
|
|
if ((fd = open(*argv, O_RDONLY | O_CREAT, 0644)) < 0)
|
2015-10-07 09:16:18 +00:00
|
|
|
eprintf("open %s:", *argv);
|
|
|
|
|
|
|
|
if (flock(fd, flags | nonblk)) {
|
|
|
|
if (nonblk && errno == EWOULDBLOCK)
|
|
|
|
return 1;
|
|
|
|
eprintf("flock:");
|
|
|
|
}
|
|
|
|
|
|
|
|
switch ((pid = fork())) {
|
|
|
|
case -1:
|
|
|
|
eprintf("fork:");
|
|
|
|
case 0:
|
2015-10-08 15:43:09 +00:00
|
|
|
if (oflag && close(fd) < 0)
|
|
|
|
eprintf("close:");
|
2015-10-07 09:16:18 +00:00
|
|
|
argv++;
|
|
|
|
execvp(*argv, argv);
|
|
|
|
savederrno = errno;
|
|
|
|
weprintf("execvp %s:", *argv);
|
|
|
|
_exit(126 + (savederrno == ENOENT));
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
waitpid(pid, &status, 0);
|
|
|
|
|
2015-10-08 15:15:51 +00:00
|
|
|
if (WIFSIGNALED(status))
|
|
|
|
return 128 + WTERMSIG(status);
|
|
|
|
if (WIFEXITED(status))
|
2015-10-07 09:16:18 +00:00
|
|
|
return WEXITSTATUS(status);
|
|
|
|
|
|
|
|
if (close(fd) < 0)
|
|
|
|
eprintf("close:");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|