74
72
struct plugin *next;
77
static plugin *getplugin(char *name, plugin **plugin_list){
75
plugin *getplugin(char *name, plugin **plugin_list){
78
76
for (plugin *p = *plugin_list; p != NULL; p = p->next){
79
77
if ((p->name == name)
80
78
or (p->name and name and (strcmp(p->name, name) == 0))){
103
101
return new_plugin;
106
static void addargument(plugin *p, char *arg){
104
void addargument(plugin *p, char *arg){
107
105
p->argv[p->argc] = arg;
108
106
p->argv = realloc(p->argv, sizeof(char *) * (size_t)(p->argc + 2));
109
107
if (p->argv == NULL){
119
117
* Descriptor Flags".
120
118
* *Note File Descriptor Flags:(libc)Descriptor Flags.
122
static int set_cloexec_flag(int fd)
120
int set_cloexec_flag(int fd)
124
122
int ret = fcntl(fd, F_GETFD, 0);
125
123
/* If reading the flags failed, return error indication now. */
162
162
int ret, maxfd = 0;
165
163
bool debug = false;
166
164
int exitstatus = EXIT_SUCCESS;
167
struct sigaction old_sigchld_action;
168
struct sigaction sigchld_action = { .sa_handler = handle_sigchld,
169
.sa_flags = SA_NOCLDSTOP };
170
char *plus_options = NULL;
171
char **plus_argv = NULL;
173
166
/* Establish a signal handler */
167
struct sigaction old_sigchld_action,
168
sigchld_action = { .sa_handler = handle_sigchld,
169
.sa_flags = SA_NOCLDSTOP };
174
170
sigemptyset(&sigchld_action.sa_mask);
175
171
ret = sigaddset(&sigchld_action.sa_mask, SIGCHLD);
197
193
{ .name = "plugin-dir", .key = 128,
198
194
.arg = "DIRECTORY",
199
195
.doc = "Specify a different plugin directory", .group = 2 },
200
{ .name = "userid", .key = 129,
201
.arg = "ID", .flags = 0,
202
.doc = "User ID the plugins will run as", .group = 2 },
203
{ .name = "groupid", .key = 130,
204
.arg = "ID", .flags = 0,
205
.doc = "Group ID the plugins will run as", .group = 2 },
206
{ .name = "debug", .key = 131,
196
{ .name = "debug", .key = 129,
207
197
.doc = "Debug mode", .group = 3 },
220
210
addargument(getplugin(NULL, plugins), p);
221
211
p = strtok(NULL, ",");
226
216
if (arg != NULL){
227
217
char *name = strtok(arg, ":");
228
218
char *p = strtok(NULL, ":");
230
220
p = strtok(p, ",");
232
222
addargument(getplugin(name, plugins), p);
233
223
p = strtok(NULL, ",");
247
uid = (uid_t)strtol(arg, NULL, 10);
250
gid = (gid_t)strtol(arg, NULL, 10);
255
239
case ARGP_KEY_ARG:
256
if(plus_options != NULL or arg == NULL or arg[0] != '+'){
261
242
case ARGP_KEY_END:
269
250
plugin *plugin_list = NULL;
271
252
struct argp argp = { .options = options, .parser = parse_opt,
272
.args_doc = "[+PLUS_SEPARATED_OPTIONS]",
273
254
.doc = "Mandos plugin runner -- Run plugins" };
275
argp_parse (&argp, argc, argv, 0, 0, &plugin_list);
278
/* This is a mangled argument in the form of
279
"+--option+--other-option=parameter+--yet-another-option", etc */
280
/* Make new argc and argv vars, and call argp_parse() again. */
281
plus_options++; /* skip the first '+' character */
282
const char delims[] = "+";
285
plus_argv = malloc(sizeof(char*) * 2);
286
if(plus_argv == NULL){
288
exitstatus = EXIT_FAILURE;
291
plus_argv[0] = argv[0];
293
arg = strtok(plus_options, delims); /* Get first argument */
296
plus_argv = realloc(plus_argv, sizeof(char *)
297
* ((unsigned int) new_argc + 1));
298
if(plus_argv == NULL){
300
exitstatus = EXIT_FAILURE;
303
plus_argv[new_argc-1] = arg;
304
plus_argv[new_argc] = NULL;
305
arg = strtok(NULL, delims); /* Get next argument */
307
argp_parse (&argp, new_argc, plus_argv, 0, 0, &plugin_list);
256
argp_parse (&argp, argc, argv, 0, 0, &plugin_list);
311
259
for(plugin *p = plugin_list; p != NULL; p=p->next){
662
600
/* Restore old signal handler */
663
601
sigaction(SIGCHLD, &old_sigchld_action, NULL);
667
603
/* Free the plugin list */
668
604
for(plugin *next; plugin_list != NULL; plugin_list = next){
669
605
next = plugin_list->next;