/mandos/release

To get this branch, use:
bzr branch http://bzr.recompile.se/loggerhead/mandos/release

« back to all changes in this revision

Viewing changes to plugin-runner.c

  • Committer: Teddy Hogeborn
  • Date: 2008-08-29 05:53:59 UTC
  • Revision ID: teddy@fukt.bsnet.se-20080829055359-wkdasnyxtylmnxus
* mandos.xml (EXAMPLE): Replaced all occurences of command name with
                        "&COMMANDNAME;".

* plugins.d/password-prompt.c (main): Improved some documentation
                                      strings.  Do perror() of
                                      tcgetattr() fails.  Add debug
                                      output if interrupted by signal.
                                      Loop over write() instead of
                                      using fwrite() when outputting
                                      password.  Add debug output if
                                      getline() returns 0, unless it
                                      was caused by a signal.  Add
                                      exit status code to debug
                                      output.

* plugins.d/password-prompt.xml: Changed all single quotes to double
                                 quotes for consistency.  Removed
                                 <?xml-stylesheet>.
  (ENTITY TIMESTAMP): New.  Automatically updated by Emacs time-stamp
                      by using Emacs local variables.
  (/refentry/refentryinfo/title): Changed to "Mandos Manual".
  (/refentry/refentryinfo/productname): Changed to "Mandos".
  (/refentry/refentryinfo/date): New; set to "&TIMESTAMP;".
  (/refentry/refentryinfo/copyright): Split copyright holders.
  (/refentry/refnamediv/refpurpose): Improved wording.
  (SYNOPSIS): Fix to use correct markup.  Add short options.
  (DESCRIPTION, OPTIONS): Improved wording.
  (OPTIONS): Improved wording.  Use more correct markup.  Document
             short options.
  (EXIT STATUS): Add text.
  (ENVIRONMENT): Document use of "cryptsource" and "crypttarget".
  (FILES): REMOVED.
  (BUGS): Add text.
  (EXAMPLE): Added some examples.
  (SECURITY): Added text.
  (SEE ALSO): Remove reference to mandos(8).  Add reference to
              crypttab(5).

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
/*  -*- coding: utf-8; mode: c; mode: orgtbl -*- */
 
1
/*  -*- coding: utf-8 -*- */
2
2
/*
3
3
 * Mandos plugin runner - Run Mandos plugins
4
4
 *
5
 
 * Copyright © 2008-2018 Teddy Hogeborn
6
 
 * Copyright © 2008-2018 Björn Påhlsson
7
 
 * 
8
 
 * This file is part of Mandos.
9
 
 * 
10
 
 * Mandos is free software: you can redistribute it and/or modify it
11
 
 * under the terms of the GNU General Public License as published by
12
 
 * the Free Software Foundation, either version 3 of the License, or
13
 
 * (at your option) any later version.
14
 
 * 
15
 
 * Mandos is distributed in the hope that it will be useful, but
 
5
 * Copyright © 2007-2008 Teddy Hogeborn & Björn Påhlsson
 
6
 * 
 
7
 * This program is free software: you can redistribute it and/or
 
8
 * modify it under the terms of the GNU General Public License as
 
9
 * published by the Free Software Foundation, either version 3 of the
 
10
 * License, or (at your option) any later version.
 
11
 * 
 
12
 * This program is distributed in the hope that it will be useful, but
16
13
 * WITHOUT ANY WARRANTY; without even the implied warranty of
17
14
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
18
15
 * General Public License for more details.
19
16
 * 
20
17
 * You should have received a copy of the GNU General Public License
21
 
 * along with Mandos.  If not, see <http://www.gnu.org/licenses/>.
 
18
 * along with this program.  If not, see
 
19
 * <http://www.gnu.org/licenses/>.
22
20
 * 
23
 
 * Contact the authors at <mandos@recompile.se>.
 
21
 * Contact the authors at <mandos@fukt.bsnet.se>.
24
22
 */
25
23
 
26
24
#define _GNU_SOURCE             /* TEMP_FAILURE_RETRY(), getline(),
27
 
                                   O_CLOEXEC, pipe2() */
 
25
                                   asprintf() */
28
26
#include <stddef.h>             /* size_t, NULL */
29
 
#include <stdlib.h>             /* malloc(), exit(), EXIT_SUCCESS,
30
 
                                   realloc() */
 
27
#include <stdlib.h>             /* malloc(), exit(), EXIT_FAILURE,
 
28
                                   EXIT_SUCCESS, realloc() */
31
29
#include <stdbool.h>            /* bool, true, false */
32
 
#include <stdio.h>              /* fileno(), fprintf(),
33
 
                                   stderr, STDOUT_FILENO, fclose() */
34
 
#include <sys/types.h>          /* fstat(), struct stat, waitpid(),
35
 
                                   WIFEXITED(), WEXITSTATUS(), wait(),
36
 
                                   pid_t, uid_t, gid_t, getuid(),
37
 
                                   getgid() */
 
30
#include <stdio.h>              /* perror, popen(), fileno(),
 
31
                                   fprintf(), stderr, STDOUT_FILENO */
 
32
#include <sys/types.h>          /* DIR, opendir(), stat(), struct
 
33
                                   stat, waitpid(), WIFEXITED(),
 
34
                                   WEXITSTATUS(), wait(), pid_t,
 
35
                                   uid_t, gid_t, getuid(), getgid(),
 
36
                                   dirfd() */
38
37
#include <sys/select.h>         /* fd_set, select(), FD_ZERO(),
39
38
                                   FD_SET(), FD_ISSET(), FD_CLR */
40
39
#include <sys/wait.h>           /* wait(), waitpid(), WIFEXITED(),
41
 
                                   WEXITSTATUS(), WTERMSIG() */
42
 
#include <sys/stat.h>           /* struct stat, fstat(), S_ISREG() */
 
40
                                   WEXITSTATUS() */
 
41
#include <sys/stat.h>           /* struct stat, stat(), S_ISREG() */
43
42
#include <iso646.h>             /* and, or, not */
44
 
#include <dirent.h>             /* struct dirent, scandirat() */
45
 
#include <unistd.h>             /* fcntl(), F_GETFD, F_SETFD,
46
 
                                   FD_CLOEXEC, write(), STDOUT_FILENO,
47
 
                                   struct stat, fstat(), close(),
48
 
                                   setgid(), setuid(), S_ISREG(),
49
 
                                   faccessat() pipe2(), fork(),
50
 
                                   _exit(), dup2(), fexecve(), read()
51
 
                                */
 
43
#include <dirent.h>             /* DIR, struct dirent, opendir(),
 
44
                                   readdir(), closedir(), dirfd() */
 
45
#include <unistd.h>             /* struct stat, stat(), S_ISREG(),
 
46
                                   fcntl(), setuid(), setgid(),
 
47
                                   F_GETFD, F_SETFD, FD_CLOEXEC,
 
48
                                   access(), pipe(), fork(), close()
 
49
                                   dup2, STDOUT_FILENO, _exit(),
 
50
                                   execv(), write(), read(),
 
51
                                   close() */
52
52
#include <fcntl.h>              /* fcntl(), F_GETFD, F_SETFD,
53
 
                                   FD_CLOEXEC, openat(), scandirat(),
54
 
                                   pipe2() */
55
 
#include <string.h>             /* strsep, strlen(), strsignal(),
56
 
                                   strcmp(), strncmp() */
 
53
                                   FD_CLOEXEC */
 
54
#include <string.h>             /* strsep, strlen(), asprintf() */
57
55
#include <errno.h>              /* errno */
58
56
#include <argp.h>               /* struct argp_option, struct
59
57
                                   argp_state, struct argp,
63
61
#include <signal.h>             /* struct sigaction, sigemptyset(),
64
62
                                   sigaddset(), sigaction(),
65
63
                                   sigprocmask(), SIG_BLOCK, SIGCHLD,
66
 
                                   SIG_UNBLOCK, kill(), sig_atomic_t
67
 
                                */
 
64
                                   SIG_UNBLOCK, kill() */
68
65
#include <errno.h>              /* errno, EBADF */
69
 
#include <inttypes.h>           /* intmax_t, PRIdMAX, strtoimax() */
70
 
#include <sysexits.h>           /* EX_OSERR, EX_USAGE, EX_IOERR,
71
 
                                   EX_CONFIG, EX_UNAVAILABLE, EX_OK */
72
 
#include <errno.h>              /* errno */
73
 
#include <error.h>              /* error() */
74
 
#include <fnmatch.h>            /* fnmatch() */
75
66
 
76
67
#define BUFFER_SIZE 256
77
68
 
78
69
#define PDIR "/lib/mandos/plugins.d"
79
 
#define PHDIR "/lib/mandos/plugin-helpers"
80
70
#define AFILE "/conf/conf.d/mandos/plugin-runner.conf"
81
71
 
82
 
const char *argp_program_version = "plugin-runner " VERSION;
83
 
const char *argp_program_bug_address = "<mandos@recompile.se>";
 
72
const char *argp_program_version = "plugin-runner 1.0";
 
73
const char *argp_program_bug_address = "<mandos@fukt.bsnet.se>";
 
74
 
 
75
struct process;
 
76
 
 
77
typedef struct process{
 
78
  pid_t pid;
 
79
  int fd;
 
80
  char *buffer;
 
81
  size_t buffer_size;
 
82
  size_t buffer_length;
 
83
  bool eof;
 
84
  volatile bool completed;
 
85
  volatile int status;
 
86
  struct process *next;
 
87
} process;
84
88
 
85
89
typedef struct plugin{
86
90
  char *name;                   /* can be NULL or any plugin name */
89
93
  char **environ;
90
94
  int envc;
91
95
  bool disabled;
92
 
  
93
 
  /* Variables used for running processes*/
94
 
  pid_t pid;
95
 
  int fd;
96
 
  char *buffer;
97
 
  size_t buffer_size;
98
 
  size_t buffer_length;
99
 
  bool eof;
100
 
  volatile sig_atomic_t completed;
101
 
  int status;
102
96
  struct plugin *next;
103
97
} plugin;
104
98
 
105
 
static plugin *plugin_list = NULL;
106
 
 
107
 
/* Gets an existing plugin based on name,
108
 
   or if none is found, creates a new one */
109
 
__attribute__((warn_unused_result))
110
 
static plugin *getplugin(char *name){
111
 
  /* Check for existing plugin with that name */
112
 
  for(plugin *p = plugin_list; p != NULL; p = p->next){
113
 
    if((p->name == name)
114
 
       or (p->name and name and (strcmp(p->name, name) == 0))){
 
99
static plugin *getplugin(char *name, plugin **plugin_list){
 
100
  for (plugin *p = *plugin_list; p != NULL; p = p->next){
 
101
    if ((p->name == name)
 
102
        or (p->name and name and (strcmp(p->name, name) == 0))){
115
103
      return p;
116
104
    }
117
105
  }
118
106
  /* Create a new plugin */
119
 
  plugin *new_plugin = NULL;
120
 
  do {
121
 
    new_plugin = malloc(sizeof(plugin));
122
 
  } while(new_plugin == NULL and errno == EINTR);
123
 
  if(new_plugin == NULL){
 
107
  plugin *new_plugin = malloc(sizeof(plugin));
 
108
  if (new_plugin == NULL){
124
109
    return NULL;
125
110
  }
126
111
  char *copy_name = NULL;
127
112
  if(name != NULL){
128
 
    do {
129
 
      copy_name = strdup(name);
130
 
    } while(copy_name == NULL and errno == EINTR);
 
113
    copy_name = strdup(name);
131
114
    if(copy_name == NULL){
132
 
      int e = errno;
133
 
      free(new_plugin);
134
 
      errno = e;
135
115
      return NULL;
136
116
    }
137
117
  }
138
118
  
139
 
  *new_plugin = (plugin){ .name = copy_name,
140
 
                          .argc = 1,
141
 
                          .disabled = false,
142
 
                          .next = plugin_list };
 
119
  *new_plugin = (plugin) { .name = copy_name,
 
120
                           .argc = 1,
 
121
                           .envc = 0,
 
122
                           .disabled = false,
 
123
                           .next = *plugin_list };
143
124
  
144
 
  do {
145
 
    new_plugin->argv = malloc(sizeof(char *) * 2);
146
 
  } while(new_plugin->argv == NULL and errno == EINTR);
147
 
  if(new_plugin->argv == NULL){
148
 
    int e = errno;
 
125
  new_plugin->argv = malloc(sizeof(char *) * 2);
 
126
  if (new_plugin->argv == NULL){
149
127
    free(copy_name);
150
128
    free(new_plugin);
151
 
    errno = e;
152
129
    return NULL;
153
130
  }
154
131
  new_plugin->argv[0] = copy_name;
155
132
  new_plugin->argv[1] = NULL;
156
 
  
157
 
  do {
158
 
    new_plugin->environ = malloc(sizeof(char *));
159
 
  } while(new_plugin->environ == NULL and errno == EINTR);
 
133
 
 
134
  new_plugin->environ = malloc(sizeof(char *));
160
135
  if(new_plugin->environ == NULL){
161
 
    int e = errno;
162
136
    free(copy_name);
163
137
    free(new_plugin->argv);
164
138
    free(new_plugin);
165
 
    errno = e;
166
139
    return NULL;
167
140
  }
168
141
  new_plugin->environ[0] = NULL;
169
 
  
170
142
  /* Append the new plugin to the list */
171
 
  plugin_list = new_plugin;
 
143
  *plugin_list = new_plugin;
172
144
  return new_plugin;
173
145
}
174
146
 
175
147
/* Helper function for add_argument and add_environment */
176
 
__attribute__((nonnull, warn_unused_result))
177
148
static bool add_to_char_array(const char *new, char ***array,
178
149
                              int *len){
179
150
  /* Resize the pointed-to array to hold one more pointer */
180
 
  char **new_array = NULL;
181
 
  do {
182
 
    new_array = realloc(*array, sizeof(char *)
183
 
                        * (size_t) ((*len) + 2));
184
 
  } while(new_array == NULL and errno == EINTR);
 
151
  *array = realloc(*array, sizeof(char *)
 
152
                   * (size_t) ((*len) + 2));
185
153
  /* Malloc check */
186
 
  if(new_array == NULL){
 
154
  if(*array == NULL){
187
155
    return false;
188
156
  }
189
 
  *array = new_array;
190
157
  /* Make a copy of the new string */
191
 
  char *copy;
192
 
  do {
193
 
    copy = strdup(new);
194
 
  } while(copy == NULL and errno == EINTR);
 
158
  char *copy = strdup(new);
195
159
  if(copy == NULL){
196
160
    return false;
197
161
  }
204
168
}
205
169
 
206
170
/* Add to a plugin's argument vector */
207
 
__attribute__((nonnull(2), warn_unused_result))
208
171
static bool add_argument(plugin *p, const char *arg){
209
172
  if(p == NULL){
210
173
    return false;
213
176
}
214
177
 
215
178
/* Add to a plugin's environment */
216
 
__attribute__((nonnull(2), warn_unused_result))
217
 
static bool add_environment(plugin *p, const char *def, bool replace){
 
179
static bool add_environment(plugin *p, const char *def){
218
180
  if(p == NULL){
219
181
    return false;
220
182
  }
221
 
  /* namelen = length of name of environment variable */
222
 
  size_t namelen = (size_t)(strchrnul(def, '=') - def);
223
 
  /* Search for this environment variable */
224
 
  for(char **envdef = p->environ; *envdef != NULL; envdef++){
225
 
    if(strncmp(*envdef, def, namelen + 1) == 0){
226
 
      /* It already exists */
227
 
      if(replace){
228
 
        char *new_envdef;
229
 
        do {
230
 
          new_envdef = realloc(*envdef, strlen(def) + 1);
231
 
        } while(new_envdef == NULL and errno == EINTR);
232
 
        if(new_envdef == NULL){
233
 
          return false;
234
 
        }
235
 
        *envdef = new_envdef;
236
 
        strcpy(*envdef, def);
237
 
      }
238
 
      return true;
239
 
    }
240
 
  }
241
183
  return add_to_char_array(def, &(p->environ), &(p->envc));
242
184
}
243
185
 
244
 
#ifndef O_CLOEXEC
 
186
 
245
187
/*
246
188
 * Based on the example in the GNU LibC manual chapter 13.13 "File
247
189
 * Descriptor Flags".
248
 
 | [[info:libc:Descriptor%20Flags][File Descriptor Flags]] |
 
190
 * *Note File Descriptor Flags:(libc)Descriptor Flags.
249
191
 */
250
 
__attribute__((warn_unused_result))
251
 
static int set_cloexec_flag(int fd){
252
 
  int ret = (int)TEMP_FAILURE_RETRY(fcntl(fd, F_GETFD, 0));
 
192
static int set_cloexec_flag(int fd)
 
193
{
 
194
  int ret = fcntl(fd, F_GETFD, 0);
253
195
  /* If reading the flags failed, return error indication now. */
254
196
  if(ret < 0){
255
197
    return ret;
256
198
  }
257
199
  /* Store modified flag word in the descriptor. */
258
 
  return (int)TEMP_FAILURE_RETRY(fcntl(fd, F_SETFD,
259
 
                                       ret | FD_CLOEXEC));
 
200
  return fcntl(fd, F_SETFD, ret | FD_CLOEXEC);
260
201
}
261
 
#endif  /* not O_CLOEXEC */
262
202
 
 
203
process *process_list = NULL;
263
204
 
264
205
/* Mark processes as completed when they exit, and save their exit
265
206
   status. */
266
 
static void handle_sigchld(__attribute__((unused)) int sig){
267
 
  int old_errno = errno;
 
207
void handle_sigchld(__attribute__((unused)) int sig){
268
208
  while(true){
269
 
    plugin *proc = plugin_list;
 
209
    process *proc = process_list;
270
210
    int status;
271
211
    pid_t pid = waitpid(-1, &status, WNOHANG);
272
212
    if(pid == 0){
274
214
      break;
275
215
    }
276
216
    if(pid == -1){
277
 
      if(errno == ECHILD){
278
 
        /* No child processes */
279
 
        break;
 
217
      if (errno != ECHILD){
 
218
        perror("waitpid");
280
219
      }
281
 
      error(0, errno, "waitpid");
 
220
      /* No child processes */
 
221
      break;
282
222
    }
283
 
    
 
223
 
284
224
    /* A child exited, find it in process_list */
285
225
    while(proc != NULL and proc->pid != pid){
286
226
      proc = proc->next;
290
230
      continue;
291
231
    }
292
232
    proc->status = status;
293
 
    proc->completed = 1;
 
233
    proc->completed = true;
294
234
  }
295
 
  errno = old_errno;
296
235
}
297
236
 
298
 
/* Prints out a password to stdout */
299
 
__attribute__((nonnull, warn_unused_result))
300
 
static bool print_out_password(const char *buffer, size_t length){
 
237
bool print_out_password(const char *buffer, size_t length){
301
238
  ssize_t ret;
 
239
  if(length>0 and buffer[length-1] == '\n'){
 
240
    length--;
 
241
  }
302
242
  for(size_t written = 0; written < length; written += (size_t)ret){
303
243
    ret = TEMP_FAILURE_RETRY(write(STDOUT_FILENO, buffer + written,
304
244
                                   length - written));
309
249
  return true;
310
250
}
311
251
 
312
 
/* Removes and free a plugin from the plugin list */
313
 
__attribute__((nonnull))
314
 
static void free_plugin(plugin *plugin_node){
315
 
  
316
 
  for(char **arg = plugin_node->argv; *arg != NULL; arg++){
317
 
    free(*arg);
318
 
  }
319
 
  free(plugin_node->argv);
320
 
  for(char **env = plugin_node->environ; *env != NULL; env++){
321
 
    free(*env);
322
 
  }
323
 
  free(plugin_node->environ);
324
 
  free(plugin_node->buffer);
325
 
  
326
 
  /* Removes the plugin from the singly-linked list */
327
 
  if(plugin_node == plugin_list){
328
 
    /* First one - simple */
329
 
    plugin_list = plugin_list->next;
330
 
  } else {
331
 
    /* Second one or later */
332
 
    for(plugin *p = plugin_list; p != NULL; p = p->next){
333
 
      if(p->next == plugin_node){
334
 
        p->next = plugin_node->next;
335
 
        break;
336
 
      }
337
 
    }
338
 
  }
339
 
  
340
 
  free(plugin_node);
341
 
}
342
 
 
343
 
static void free_plugin_list(void){
344
 
  while(plugin_list != NULL){
345
 
    free_plugin(plugin_list);
 
252
static void free_plugin_list(plugin *plugin_list){
 
253
  for(plugin *next; plugin_list != NULL; plugin_list = next){
 
254
    next = plugin_list->next;
 
255
    for(char **arg = plugin_list->argv; *arg != NULL; arg++){
 
256
      free(*arg);
 
257
    }
 
258
    free(plugin_list->argv);
 
259
    for(char **env = plugin_list->environ; *env != NULL; env++){
 
260
      free(*env);
 
261
    }
 
262
    free(plugin_list->environ);
 
263
    free(plugin_list);
346
264
  }
347
265
}
348
266
 
349
267
int main(int argc, char *argv[]){
350
268
  char *plugindir = NULL;
351
 
  char *pluginhelperdir = NULL;
352
269
  char *argfile = NULL;
353
270
  FILE *conffp;
354
 
  struct dirent **direntries = NULL;
 
271
  size_t d_name_len;
 
272
  DIR *dir = NULL;
 
273
  struct dirent *dirst;
355
274
  struct stat st;
356
275
  fd_set rfds_all;
357
276
  int ret, maxfd = 0;
358
 
  ssize_t sret;
359
277
  uid_t uid = 65534;
360
278
  gid_t gid = 65534;
361
279
  bool debug = false;
365
283
                                      .sa_flags = SA_NOCLDSTOP };
366
284
  char **custom_argv = NULL;
367
285
  int custom_argc = 0;
368
 
  int dir_fd = -1;
369
286
  
370
287
  /* Establish a signal handler */
371
288
  sigemptyset(&sigchld_action.sa_mask);
372
289
  ret = sigaddset(&sigchld_action.sa_mask, SIGCHLD);
373
290
  if(ret == -1){
374
 
    error(0, errno, "sigaddset");
375
 
    exitstatus = EX_OSERR;
 
291
    perror("sigaddset");
 
292
    exitstatus = EXIT_FAILURE;
376
293
    goto fallback;
377
294
  }
378
295
  ret = sigaction(SIGCHLD, &sigchld_action, &old_sigchld_action);
379
296
  if(ret == -1){
380
 
    error(0, errno, "sigaction");
381
 
    exitstatus = EX_OSERR;
 
297
    perror("sigaction");
 
298
    exitstatus = EXIT_FAILURE;
382
299
    goto fallback;
383
300
  }
384
301
  
387
304
    { .name = "global-options", .key = 'g',
388
305
      .arg = "OPTION[,OPTION[,...]]",
389
306
      .doc = "Options passed to all plugins" },
390
 
    { .name = "global-env", .key = 'G',
 
307
    { .name = "global-envs", .key = 'e',
391
308
      .arg = "VAR=value",
392
309
      .doc = "Environment variable passed to all plugins" },
393
310
    { .name = "options-for", .key = 'o',
394
311
      .arg = "PLUGIN:OPTION[,OPTION[,...]]",
395
312
      .doc = "Options passed only to specified plugin" },
396
 
    { .name = "env-for", .key = 'E',
 
313
    { .name = "envs-for", .key = 'f',
397
314
      .arg = "PLUGIN:ENV=value",
398
315
      .doc = "Environment variable passed to specified plugin" },
399
316
    { .name = "disable", .key = 'd',
400
317
      .arg = "PLUGIN",
401
318
      .doc = "Disable a specific plugin", .group = 1 },
402
 
    { .name = "enable", .key = 'e',
403
 
      .arg = "PLUGIN",
404
 
      .doc = "Enable a specific plugin", .group = 1 },
405
319
    { .name = "plugin-dir", .key = 128,
406
320
      .arg = "DIRECTORY",
407
321
      .doc = "Specify a different plugin directory", .group = 2 },
416
330
      .doc = "Group ID the plugins will run as", .group = 3 },
417
331
    { .name = "debug", .key = 132,
418
332
      .doc = "Debug mode", .group = 4 },
419
 
    { .name = "plugin-helper-dir", .key = 133,
420
 
      .arg = "DIRECTORY",
421
 
      .doc = "Specify a different plugin helper directory",
422
 
      .group = 2 },
423
 
    /*
424
 
     * These reproduce what we would get without ARGP_NO_HELP
425
 
     */
426
 
    { .name = "help", .key = '?',
427
 
      .doc = "Give this help list", .group = -1 },
428
 
    { .name = "usage", .key = -3,
429
 
      .doc = "Give a short usage message", .group = -1 },
430
 
    { .name = "version", .key = 'V',
431
 
      .doc = "Print program version", .group = -1 },
432
333
    { .name = NULL }
433
334
  };
434
335
  
435
 
  __attribute__((nonnull(3)))
436
 
  error_t parse_opt(int key, char *arg, struct argp_state *state){
437
 
    errno = 0;
438
 
    switch(key){
439
 
      char *tmp;
440
 
      intmax_t tmp_id;
441
 
    case 'g':                   /* --global-options */
442
 
      {
443
 
        char *plugin_option;
444
 
        while((plugin_option = strsep(&arg, ",")) != NULL){
445
 
          if(not add_argument(getplugin(NULL), plugin_option)){
446
 
            break;
447
 
          }
448
 
        }
449
 
        errno = 0;
450
 
      }
451
 
      break;
452
 
    case 'G':                   /* --global-env */
453
 
      if(add_environment(getplugin(NULL), arg, true)){
454
 
        errno = 0;
455
 
      }
456
 
      break;
457
 
    case 'o':                   /* --options-for */
458
 
      {
459
 
        char *option_list = strchr(arg, ':');
460
 
        if(option_list == NULL){
461
 
          argp_error(state, "No colon in \"%s\"", arg);
462
 
          errno = EINVAL;
463
 
          break;
464
 
        }
465
 
        *option_list = '\0';
466
 
        option_list++;
467
 
        if(arg[0] == '\0'){
468
 
          argp_error(state, "Empty plugin name");
469
 
          errno = EINVAL;
470
 
          break;
471
 
        }
472
 
        char *option;
473
 
        while((option = strsep(&option_list, ",")) != NULL){
474
 
          if(not add_argument(getplugin(arg), option)){
475
 
            break;
476
 
          }
477
 
        }
478
 
        errno = 0;
479
 
      }
480
 
      break;
481
 
    case 'E':                   /* --env-for */
 
336
  error_t parse_opt (int key, char *arg, struct argp_state *state) {
 
337
    /* Get the INPUT argument from `argp_parse', which we know is a
 
338
       pointer to our plugin list pointer. */
 
339
    plugin **plugins = state->input;
 
340
    switch (key) {
 
341
    case 'g':
 
342
      if (arg != NULL){
 
343
        char *p;
 
344
        while((p = strsep(&arg, ",")) != NULL){
 
345
          if(p[0] == '\0'){
 
346
            continue;
 
347
          }
 
348
          if(not add_argument(getplugin(NULL, plugins), p)){
 
349
            perror("add_argument");
 
350
            return ARGP_ERR_UNKNOWN;
 
351
          }
 
352
        }
 
353
      }
 
354
      break;
 
355
    case 'e':
 
356
      if(arg == NULL){
 
357
        break;
 
358
      }
 
359
      {
 
360
        char *envdef = strdup(arg);
 
361
        if(envdef == NULL){
 
362
          break;
 
363
        }
 
364
        if(not add_environment(getplugin(NULL, plugins), envdef)){
 
365
          perror("add_environment");
 
366
        }
 
367
      }
 
368
      break;
 
369
    case 'o':
 
370
      if (arg != NULL){
 
371
        char *p_name = strsep(&arg, ":");
 
372
        if(p_name[0] == '\0'){
 
373
          break;
 
374
        }
 
375
        char *opt = strsep(&arg, ":");
 
376
        if(opt[0] == '\0'){
 
377
          break;
 
378
        }
 
379
        if(opt != NULL){
 
380
          char *p;
 
381
          while((p = strsep(&opt, ",")) != NULL){
 
382
            if(p[0] == '\0'){
 
383
              continue;
 
384
            }
 
385
            if(not add_argument(getplugin(p_name, plugins), p)){
 
386
              perror("add_argument");
 
387
              return ARGP_ERR_UNKNOWN;
 
388
            }
 
389
          }
 
390
        }
 
391
      }
 
392
      break;
 
393
    case 'f':
 
394
      if(arg == NULL){
 
395
        break;
 
396
      }
482
397
      {
483
398
        char *envdef = strchr(arg, ':');
484
399
        if(envdef == NULL){
485
 
          argp_error(state, "No colon in \"%s\"", arg);
486
 
          errno = EINVAL;
487
 
          break;
488
 
        }
489
 
        *envdef = '\0';
 
400
          break;
 
401
        }
 
402
        char *p_name = strndup(arg, (size_t) (envdef-arg));
 
403
        if(p_name == NULL){
 
404
          break;
 
405
        }
490
406
        envdef++;
491
 
        if(arg[0] == '\0'){
492
 
          argp_error(state, "Empty plugin name");
493
 
          errno = EINVAL;
494
 
          break;
495
 
        }
496
 
        if(add_environment(getplugin(arg), envdef, true)){
497
 
          errno = 0;
498
 
        }
499
 
      }
500
 
      break;
501
 
    case 'd':                   /* --disable */
502
 
      {
503
 
        plugin *p = getplugin(arg);
504
 
        if(p != NULL){
505
 
          p->disabled = true;
506
 
          errno = 0;
507
 
        }
508
 
      }
509
 
      break;
510
 
    case 'e':                   /* --enable */
511
 
      {
512
 
        plugin *p = getplugin(arg);
513
 
        if(p != NULL){
514
 
          p->disabled = false;
515
 
          errno = 0;
516
 
        }
517
 
      }
518
 
      break;
519
 
    case 128:                   /* --plugin-dir */
520
 
      free(plugindir);
 
407
        if(not add_environment(getplugin(p_name, plugins), envdef)){
 
408
          perror("add_environment");
 
409
        }
 
410
      }
 
411
      break;
 
412
    case 'd':
 
413
      if (arg != NULL){
 
414
        plugin *p = getplugin(arg, plugins);
 
415
        if(p == NULL){
 
416
          return ARGP_ERR_UNKNOWN;
 
417
        }
 
418
        p->disabled = true;
 
419
      }
 
420
      break;
 
421
    case 128:
521
422
      plugindir = strdup(arg);
522
 
      if(plugindir != NULL){
523
 
        errno = 0;
524
 
      }
525
 
      break;
526
 
    case 129:                   /* --config-file */
527
 
      /* This is already done by parse_opt_config_file() */
528
 
      break;
529
 
    case 130:                   /* --userid */
530
 
      tmp_id = strtoimax(arg, &tmp, 10);
531
 
      if(errno != 0 or tmp == arg or *tmp != '\0'
532
 
         or tmp_id != (uid_t)tmp_id){
533
 
        argp_error(state, "Bad user ID number: \"%s\", using %"
534
 
                   PRIdMAX, arg, (intmax_t)uid);
535
 
        break;
536
 
      }
537
 
      uid = (uid_t)tmp_id;
538
 
      errno = 0;
539
 
      break;
540
 
    case 131:                   /* --groupid */
541
 
      tmp_id = strtoimax(arg, &tmp, 10);
542
 
      if(errno != 0 or tmp == arg or *tmp != '\0'
543
 
         or tmp_id != (gid_t)tmp_id){
544
 
        argp_error(state, "Bad group ID number: \"%s\", using %"
545
 
                   PRIdMAX, arg, (intmax_t)gid);
546
 
        break;
547
 
      }
548
 
      gid = (gid_t)tmp_id;
549
 
      errno = 0;
550
 
      break;
551
 
    case 132:                   /* --debug */
 
423
      if(plugindir == NULL){
 
424
        perror("strdup");
 
425
      }      
 
426
      break;
 
427
    case 129:
 
428
      argfile = strdup(arg);
 
429
      if(argfile == NULL){
 
430
        perror("strdup");
 
431
      }
 
432
      break;      
 
433
    case 130:
 
434
      uid = (uid_t)strtol(arg, NULL, 10);
 
435
      break;
 
436
    case 131:
 
437
      gid = (gid_t)strtol(arg, NULL, 10);
 
438
      break;
 
439
    case 132:
552
440
      debug = true;
553
441
      break;
554
 
    case 133:                   /* --plugin-helper-dir */
555
 
      free(pluginhelperdir);
556
 
      pluginhelperdir = strdup(arg);
557
 
      if(pluginhelperdir != NULL){
558
 
        errno = 0;
559
 
      }
560
 
      break;
561
 
      /*
562
 
       * These reproduce what we would get without ARGP_NO_HELP
563
 
       */
564
 
    case '?':                   /* --help */
565
 
      state->flags &= ~(unsigned int)ARGP_NO_EXIT; /* force exit */
566
 
      argp_state_help(state, state->out_stream, ARGP_HELP_STD_HELP);
567
 
      __builtin_unreachable();
568
 
    case -3:                    /* --usage */
569
 
      state->flags &= ~(unsigned int)ARGP_NO_EXIT; /* force exit */
570
 
      argp_state_help(state, state->out_stream,
571
 
                      ARGP_HELP_USAGE | ARGP_HELP_EXIT_OK);
572
 
      __builtin_unreachable();
573
 
    case 'V':                   /* --version */
574
 
      fprintf(state->out_stream, "%s\n", argp_program_version);
575
 
      exit(EXIT_SUCCESS);
576
 
      break;
577
 
/*
578
 
 * When adding more options before this line, remember to also add a
579
 
 * "case" to the "parse_opt_config_file" function below.
580
 
 */
581
 
    case ARGP_KEY_ARG:
582
 
      /* Cryptsetup always passes an argument, which is an empty
583
 
         string if "none" was specified in /etc/crypttab.  So if
584
 
         argument was empty, we ignore it silently. */
585
 
      if(arg[0] == '\0'){
586
 
        break;
587
 
      }
588
 
      /* FALLTHROUGH */
589
 
    default:
590
 
      return ARGP_ERR_UNKNOWN;
591
 
    }
592
 
    return errno;               /* Set to 0 at start */
593
 
  }
594
 
  
595
 
  /* This option parser is the same as parse_opt() above, except it
596
 
     ignores everything but the --config-file option. */
597
 
  error_t parse_opt_config_file(int key, char *arg,
598
 
                                __attribute__((unused))
599
 
                                struct argp_state *state){
600
 
    errno = 0;
601
 
    switch(key){
602
 
    case 'g':                   /* --global-options */
603
 
    case 'G':                   /* --global-env */
604
 
    case 'o':                   /* --options-for */
605
 
    case 'E':                   /* --env-for */
606
 
    case 'd':                   /* --disable */
607
 
    case 'e':                   /* --enable */
608
 
    case 128:                   /* --plugin-dir */
609
 
      break;
610
 
    case 129:                   /* --config-file */
611
 
      free(argfile);
612
 
      argfile = strdup(arg);
613
 
      if(argfile != NULL){
614
 
        errno = 0;
615
 
      }
616
 
      break;
617
 
    case 130:                   /* --userid */
618
 
    case 131:                   /* --groupid */
619
 
    case 132:                   /* --debug */
620
 
    case 133:                   /* --plugin-helper-dir */
621
 
    case '?':                   /* --help */
622
 
    case -3:                    /* --usage */
623
 
    case 'V':                   /* --version */
624
 
    case ARGP_KEY_ARG:
625
 
      break;
626
 
    default:
627
 
      return ARGP_ERR_UNKNOWN;
628
 
    }
629
 
    return errno;
630
 
  }
631
 
  
632
 
  struct argp argp = { .options = options,
633
 
                       .parser = parse_opt_config_file,
634
 
                       .args_doc = "",
 
442
    case ARGP_KEY_ARG:
 
443
      fprintf(stderr, "Ignoring unknown argument \"%s\"\n", arg);
 
444
      break;
 
445
    case ARGP_KEY_END:
 
446
      break;
 
447
    default:
 
448
      return ARGP_ERR_UNKNOWN;
 
449
    }
 
450
    return 0;
 
451
  }
 
452
  
 
453
  plugin *plugin_list = NULL;
 
454
  
 
455
  struct argp argp = { .options = options, .parser = parse_opt,
 
456
                       .args_doc = "[+PLUS_SEPARATED_OPTIONS]",
635
457
                       .doc = "Mandos plugin runner -- Run plugins" };
636
458
  
637
 
  /* Parse using parse_opt_config_file() in order to get the custom
638
 
     config file location, if any. */
639
 
  ret = argp_parse(&argp, argc, argv,
640
 
                   ARGP_IN_ORDER | ARGP_NO_EXIT | ARGP_NO_HELP,
641
 
                   NULL, NULL);
642
 
  switch(ret){
643
 
  case 0:
644
 
    break;
645
 
  case ENOMEM:
646
 
  default:
647
 
    errno = ret;
648
 
    error(0, errno, "argp_parse");
649
 
    exitstatus = EX_OSERR;
650
 
    goto fallback;
651
 
  case EINVAL:
652
 
    exitstatus = EX_USAGE;
 
459
  ret = argp_parse (&argp, argc, argv, 0, 0, &plugin_list);
 
460
  if (ret == ARGP_ERR_UNKNOWN){
 
461
    fprintf(stderr, "Unknown error while parsing arguments\n");
 
462
    exitstatus = EXIT_FAILURE;
653
463
    goto fallback;
654
464
  }
655
 
  
656
 
  /* Reset to the normal argument parser */
657
 
  argp.parser = parse_opt;
658
 
  
659
 
  /* Open the configfile if available */
660
 
  if(argfile == NULL){
 
465
 
 
466
  if (argfile == NULL){
661
467
    conffp = fopen(AFILE, "r");
662
468
  } else {
663
469
    conffp = fopen(argfile, "r");
664
470
  }
 
471
  
665
472
  if(conffp != NULL){
666
473
    char *org_line = NULL;
667
474
    char *p, *arg, *new_arg, *line;
668
475
    size_t size = 0;
 
476
    ssize_t sret;
669
477
    const char whitespace_delims[] = " \r\t\f\v\n";
670
478
    const char comment_delim[] = "#";
671
 
    
 
479
 
672
480
    custom_argc = 1;
673
481
    custom_argv = malloc(sizeof(char*) * 2);
674
482
    if(custom_argv == NULL){
675
 
      error(0, errno, "malloc");
676
 
      exitstatus = EX_OSERR;
 
483
      perror("malloc");
 
484
      exitstatus = EXIT_FAILURE;
677
485
      goto fallback;
678
486
    }
679
487
    custom_argv[0] = argv[0];
680
488
    custom_argv[1] = NULL;
681
489
    
682
 
    /* for each line in the config file, strip whitespace and ignore
683
 
       commented text */
684
490
    while(true){
685
491
      sret = getline(&org_line, &size, conffp);
686
492
      if(sret == -1){
687
493
        break;
688
494
      }
689
 
      
 
495
 
690
496
      line = org_line;
691
497
      arg = strsep(&line, comment_delim);
692
498
      while((p = strsep(&arg, whitespace_delims)) != NULL){
695
501
        }
696
502
        new_arg = strdup(p);
697
503
        if(new_arg == NULL){
698
 
          error(0, errno, "strdup");
699
 
          exitstatus = EX_OSERR;
 
504
          perror("strdup");
 
505
          exitstatus = EXIT_FAILURE;
700
506
          free(org_line);
701
507
          goto fallback;
702
508
        }
703
509
        
704
510
        custom_argc += 1;
705
 
        {
706
 
          char **new_argv = realloc(custom_argv, sizeof(char *)
707
 
                                    * ((size_t)custom_argc + 1));
708
 
          if(new_argv == NULL){
709
 
            error(0, errno, "realloc");
710
 
            exitstatus = EX_OSERR;
711
 
            free(new_arg);
712
 
            free(org_line);
713
 
            goto fallback;
714
 
          } else {
715
 
            custom_argv = new_argv;
716
 
          }
 
511
        custom_argv = realloc(custom_argv, sizeof(char *)
 
512
                              * ((unsigned int) custom_argc + 1));
 
513
        if(custom_argv == NULL){
 
514
          perror("realloc");
 
515
          exitstatus = EXIT_FAILURE;
 
516
          free(org_line);
 
517
          goto fallback;
717
518
        }
718
519
        custom_argv[custom_argc-1] = new_arg;
719
 
        custom_argv[custom_argc] = NULL;
 
520
        custom_argv[custom_argc] = NULL;        
720
521
      }
721
522
    }
722
 
    do {
723
 
      ret = fclose(conffp);
724
 
    } while(ret == EOF and errno == EINTR);
725
 
    if(ret == EOF){
726
 
      error(0, errno, "fclose");
727
 
      exitstatus = EX_IOERR;
728
 
      goto fallback;
729
 
    }
730
523
    free(org_line);
731
 
  } else {
 
524
  } else{
732
525
    /* Check for harmful errors and go to fallback. Other errors might
733
526
       not affect opening plugins */
734
 
    if(errno == EMFILE or errno == ENFILE or errno == ENOMEM){
735
 
      error(0, errno, "fopen");
736
 
      exitstatus = EX_OSERR;
 
527
    if (errno == EMFILE or errno == ENFILE or errno == ENOMEM){
 
528
      perror("fopen");
 
529
      exitstatus = EXIT_FAILURE;
737
530
      goto fallback;
738
531
    }
739
532
  }
740
 
  /* If there were any arguments from the configuration file, pass
741
 
     them to parser as command line arguments */
 
533
 
742
534
  if(custom_argv != NULL){
743
 
    ret = argp_parse(&argp, custom_argc, custom_argv,
744
 
                     ARGP_IN_ORDER | ARGP_NO_EXIT | ARGP_NO_HELP,
745
 
                     NULL, NULL);
746
 
    switch(ret){
747
 
    case 0:
748
 
      break;
749
 
    case ENOMEM:
750
 
    default:
751
 
      errno = ret;
752
 
      error(0, errno, "argp_parse");
753
 
      exitstatus = EX_OSERR;
754
 
      goto fallback;
755
 
    case EINVAL:
756
 
      exitstatus = EX_CONFIG;
757
 
      goto fallback;
758
 
    }
759
 
  }
760
 
  
761
 
  /* Parse actual command line arguments, to let them override the
762
 
     config file */
763
 
  ret = argp_parse(&argp, argc, argv,
764
 
                   ARGP_IN_ORDER | ARGP_NO_EXIT | ARGP_NO_HELP,
765
 
                   NULL, NULL);
766
 
  switch(ret){
767
 
  case 0:
768
 
    break;
769
 
  case ENOMEM:
770
 
  default:
771
 
    errno = ret;
772
 
    error(0, errno, "argp_parse");
773
 
    exitstatus = EX_OSERR;
774
 
    goto fallback;
775
 
  case EINVAL:
776
 
    exitstatus = EX_USAGE;
777
 
    goto fallback;
778
 
  }
779
 
  
780
 
  {
781
 
    char *pluginhelperenv;
782
 
    bool bret = true;
783
 
    ret = asprintf(&pluginhelperenv, "MANDOSPLUGINHELPERDIR=%s",
784
 
                   pluginhelperdir != NULL ? pluginhelperdir : PHDIR);
785
 
    if(ret != -1){
786
 
      bret = add_environment(getplugin(NULL), pluginhelperenv, true);
787
 
    }
788
 
    if(ret == -1 or not bret){
789
 
      error(0, errno, "Failed to set MANDOSPLUGINHELPERDIR"
790
 
            " environment variable to \"%s\" for all plugins\n",
791
 
            pluginhelperdir != NULL ? pluginhelperdir : PHDIR);
792
 
    }
793
 
    if(ret != -1){
794
 
      free(pluginhelperenv);
 
535
    ret = argp_parse (&argp, custom_argc, custom_argv, 0, 0, &plugin_list);
 
536
    if (ret == ARGP_ERR_UNKNOWN){
 
537
      fprintf(stderr, "Unknown error while parsing arguments\n");
 
538
      exitstatus = EXIT_FAILURE;
 
539
      goto fallback;
795
540
    }
796
541
  }
797
542
  
798
543
  if(debug){
799
 
    for(plugin *p = plugin_list; p != NULL; p = p->next){
 
544
    for(plugin *p = plugin_list; p != NULL; p=p->next){
800
545
      fprintf(stderr, "Plugin: %s has %d arguments\n",
801
546
              p->name ? p->name : "Global", p->argc - 1);
802
547
      for(char **a = p->argv; *a != NULL; a++){
803
548
        fprintf(stderr, "\tArg: %s\n", *a);
804
549
      }
805
 
      fprintf(stderr, "...and %d environment variables\n", p->envc);
 
550
      fprintf(stderr, "...and %u environment variables\n", p->envc);
806
551
      for(char **a = p->environ; *a != NULL; a++){
807
552
        fprintf(stderr, "\t%s\n", *a);
808
553
      }
809
554
    }
810
555
  }
811
556
  
812
 
  if(getuid() == 0){
813
 
    /* Work around Debian bug #633582:
814
 
       <https://bugs.debian.org/633582> */
815
 
    int plugindir_fd = open(/* plugindir or */ PDIR, O_RDONLY);
816
 
    if(plugindir_fd == -1){
817
 
      if(errno != ENOENT){
818
 
        error(0, errno, "open(\"" PDIR "\")");
819
 
      }
820
 
    } else {
821
 
      ret = (int)TEMP_FAILURE_RETRY(fstat(plugindir_fd, &st));
822
 
      if(ret == -1){
823
 
        error(0, errno, "fstat");
824
 
      } else {
825
 
        if(S_ISDIR(st.st_mode) and st.st_uid == 0 and st.st_gid == 0){
826
 
          ret = fchown(plugindir_fd, uid, gid);
827
 
          if(ret == -1){
828
 
            error(0, errno, "fchown");
829
 
          }
830
 
        }
831
 
      }
832
 
      close(plugindir_fd);
833
 
    }
834
 
  }
835
 
  
836
 
  /* Lower permissions */
837
 
  ret = setgid(gid);
838
 
  if(ret == -1){
839
 
    error(0, errno, "setgid");
840
 
  }
841
557
  ret = setuid(uid);
842
 
  if(ret == -1){
843
 
    error(0, errno, "setuid");
844
 
  }
845
 
  
846
 
  /* Open plugin directory with close_on_exec flag */
 
558
  if (ret == -1){
 
559
    perror("setuid");
 
560
  }
 
561
  
 
562
  setgid(gid);
 
563
  if (ret == -1){
 
564
    perror("setgid");
 
565
  }
 
566
 
 
567
  if (plugindir == NULL){
 
568
    dir = opendir(PDIR);
 
569
  } else {
 
570
    dir = opendir(plugindir);
 
571
  }
 
572
  
 
573
  if(dir == NULL){
 
574
    perror("Could not open plugin dir");
 
575
    exitstatus = EXIT_FAILURE;
 
576
    goto fallback;
 
577
  }
 
578
  
 
579
  /* Set the FD_CLOEXEC flag on the directory, if possible */
847
580
  {
848
 
    dir_fd = open(plugindir != NULL ? plugindir : PDIR, O_RDONLY |
849
 
#ifdef O_CLOEXEC
850
 
                  O_CLOEXEC
851
 
#else  /* not O_CLOEXEC */
852
 
                  0
853
 
#endif  /* not O_CLOEXEC */
854
 
                  );
855
 
    if(dir_fd == -1){
856
 
      error(0, errno, "Could not open plugin dir");
857
 
      exitstatus = EX_UNAVAILABLE;
858
 
      goto fallback;
859
 
    }
860
 
    
861
 
#ifndef O_CLOEXEC
862
 
  /* Set the FD_CLOEXEC flag on the directory */
863
 
    ret = set_cloexec_flag(dir_fd);
864
 
    if(ret < 0){
865
 
      error(0, errno, "set_cloexec_flag");
866
 
      exitstatus = EX_OSERR;
867
 
      goto fallback;
868
 
    }
869
 
#endif  /* O_CLOEXEC */
870
 
  }
871
 
  
872
 
  int good_name(const struct dirent * const dirent){
873
 
    const char * const patterns[] = { ".*", "#*#", "*~", "*.dpkg-new",
874
 
                                      "*.dpkg-old", "*.dpkg-bak",
875
 
                                      "*.dpkg-divert", NULL };
876
 
#ifdef __GNUC__
877
 
#pragma GCC diagnostic push
878
 
#pragma GCC diagnostic ignored "-Wcast-qual"
879
 
#endif
880
 
    for(const char **pat = (const char **)patterns;
881
 
        *pat != NULL; pat++){
882
 
#ifdef __GNUC__
883
 
#pragma GCC diagnostic pop
884
 
#endif
885
 
      if(fnmatch(*pat, dirent->d_name, FNM_FILE_NAME | FNM_PERIOD)
886
 
         != FNM_NOMATCH){
887
 
        if(debug){
888
 
            fprintf(stderr, "Ignoring plugin dir entry \"%s\""
889
 
                    " matching pattern %s\n", dirent->d_name, *pat);
890
 
        }
891
 
        return 0;
 
581
    int dir_fd = dirfd(dir);
 
582
    if(dir_fd >= 0){
 
583
      ret = set_cloexec_flag(dir_fd);
 
584
      if(ret < 0){
 
585
        perror("set_cloexec_flag");
 
586
        exitstatus = EXIT_FAILURE;
 
587
        goto fallback;
892
588
      }
893
589
    }
894
 
    return 1;
895
 
  }
896
 
  
897
 
  int numplugins = scandirat(dir_fd, ".", &direntries, good_name,
898
 
                             alphasort);
899
 
  if(numplugins == -1){
900
 
    error(0, errno, "Could not scan plugin dir");
901
 
    direntries = NULL;
902
 
    exitstatus = EX_OSERR;
903
 
    goto fallback;
904
590
  }
905
591
  
906
592
  FD_ZERO(&rfds_all);
907
593
  
908
 
  /* Read and execute any executable in the plugin directory*/
909
 
  for(int i = 0; i < numplugins; i++){
910
 
    
911
 
    int plugin_fd = openat(dir_fd, direntries[i]->d_name, O_RDONLY);
912
 
    if(plugin_fd == -1){
913
 
      error(0, errno, "Could not open plugin");
914
 
      free(direntries[i]);
915
 
      continue;
916
 
    }
917
 
    ret = (int)TEMP_FAILURE_RETRY(fstat(plugin_fd, &st));
918
 
    if(ret == -1){
919
 
      error(0, errno, "stat");
920
 
      close(plugin_fd);
921
 
      free(direntries[i]);
922
 
      continue;
923
 
    }
924
 
    
925
 
    /* Ignore non-executable files */
926
 
    if(not S_ISREG(st.st_mode)
927
 
       or (TEMP_FAILURE_RETRY(faccessat(dir_fd, direntries[i]->d_name,
928
 
                                        X_OK, 0)) != 0)){
 
594
  while(true){
 
595
    dirst = readdir(dir);
 
596
    
 
597
    // All directory entries have been processed
 
598
    if(dirst == NULL){
 
599
      if (errno == EBADF){
 
600
        perror("readdir");
 
601
        exitstatus = EXIT_FAILURE;
 
602
        goto fallback;
 
603
      }
 
604
      break;
 
605
    }
 
606
    
 
607
    d_name_len = strlen(dirst->d_name);
 
608
    
 
609
    // Ignore dotfiles, backup files and other junk
 
610
    {
 
611
      bool bad_name = false;
 
612
      
 
613
      const char const *bad_prefixes[] = { ".", "#", NULL };
 
614
      
 
615
      const char const *bad_suffixes[] = { "~", "#", ".dpkg-new",
 
616
                                           ".dpkg-old",
 
617
                                           ".dpkg-divert", NULL };
 
618
      for(const char **pre = bad_prefixes; *pre != NULL; pre++){
 
619
        size_t pre_len = strlen(*pre);
 
620
        if((d_name_len >= pre_len)
 
621
           and strncmp((dirst->d_name), *pre, pre_len) == 0){
 
622
          if(debug){
 
623
            fprintf(stderr, "Ignoring plugin dir entry \"%s\""
 
624
                    " with bad prefix %s\n", dirst->d_name, *pre);
 
625
          }
 
626
          bad_name = true;
 
627
          break;
 
628
        }
 
629
      }
 
630
      
 
631
      if(bad_name){
 
632
        continue;
 
633
      }
 
634
      
 
635
      for(const char **suf = bad_suffixes; *suf != NULL; suf++){
 
636
        size_t suf_len = strlen(*suf);
 
637
        if((d_name_len >= suf_len)
 
638
           and (strcmp((dirst->d_name)+d_name_len-suf_len, *suf)
 
639
                == 0)){
 
640
          if(debug){
 
641
            fprintf(stderr, "Ignoring plugin dir entry \"%s\""
 
642
                    " with bad suffix %s\n", dirst->d_name, *suf);
 
643
          }
 
644
          bad_name = true;
 
645
          break;
 
646
        }
 
647
      }
 
648
      
 
649
      if(bad_name){
 
650
        continue;
 
651
      }
 
652
    }
 
653
 
 
654
    char *filename;
 
655
    ret = asprintf(&filename, "%s/%s", plugindir, dirst->d_name);
 
656
    if(ret < 0){
 
657
      perror("asprintf");
 
658
      continue;
 
659
    }
 
660
    
 
661
    ret = stat(filename, &st);
 
662
    if (ret == -1){
 
663
      perror("stat");
 
664
      free(filename);
 
665
      continue;
 
666
    }
 
667
    
 
668
    if (not S_ISREG(st.st_mode) or (access(filename, X_OK) != 0)){
929
669
      if(debug){
930
 
        fprintf(stderr, "Ignoring plugin dir entry \"%s/%s\""
931
 
                " with bad type or mode\n",
932
 
                plugindir != NULL ? plugindir : PDIR,
933
 
                direntries[i]->d_name);
 
670
        fprintf(stderr, "Ignoring plugin dir entry \"%s\""
 
671
                " with bad type or mode\n", filename);
934
672
      }
935
 
      close(plugin_fd);
936
 
      free(direntries[i]);
 
673
      free(filename);
937
674
      continue;
938
675
    }
939
 
    
940
 
    plugin *p = getplugin(direntries[i]->d_name);
 
676
    plugin *p = getplugin(dirst->d_name, &plugin_list);
941
677
    if(p == NULL){
942
 
      error(0, errno, "getplugin");
943
 
      close(plugin_fd);
944
 
      free(direntries[i]);
 
678
      perror("getplugin");
 
679
      free(filename);
945
680
      continue;
946
681
    }
947
682
    if(p->disabled){
948
683
      if(debug){
949
684
        fprintf(stderr, "Ignoring disabled plugin \"%s\"\n",
950
 
                direntries[i]->d_name);
 
685
                dirst->d_name);
951
686
      }
952
 
      close(plugin_fd);
953
 
      free(direntries[i]);
 
687
      free(filename);
954
688
      continue;
955
689
    }
956
690
    {
957
691
      /* Add global arguments to argument list for this plugin */
958
 
      plugin *g = getplugin(NULL);
 
692
      plugin *g = getplugin(NULL, &plugin_list);
959
693
      if(g != NULL){
960
694
        for(char **a = g->argv + 1; *a != NULL; a++){
961
695
          if(not add_argument(p, *a)){
962
 
            error(0, errno, "add_argument");
 
696
            perror("add_argument");
963
697
          }
964
698
        }
965
699
        /* Add global environment variables */
966
700
        for(char **e = g->environ; *e != NULL; e++){
967
 
          if(not add_environment(p, *e, false)){
968
 
            error(0, errno, "add_environment");
 
701
          if(not add_environment(p, *e)){
 
702
            perror("add_environment");
969
703
          }
970
704
        }
971
705
      }
972
706
    }
973
 
    /* If this plugin has any environment variables, we need to
974
 
       duplicate the environment from this process, too. */
 
707
    /* If this plugin has any environment variables, we will call
 
708
       using execve and need to duplicate the environment from this
 
709
       process, too. */
975
710
    if(p->environ[0] != NULL){
976
711
      for(char **e = environ; *e != NULL; e++){
977
 
        if(not add_environment(p, *e, false)){
978
 
          error(0, errno, "add_environment");
 
712
        char *copy = strdup(*e);
 
713
        if(copy == NULL){
 
714
          perror("strdup");
 
715
          continue;
 
716
        }
 
717
        if(not add_environment(p, copy)){
 
718
          perror("add_environment");
979
719
        }
980
720
      }
981
721
    }
982
722
    
983
723
    int pipefd[2];
984
 
#ifndef O_CLOEXEC
985
 
    ret = (int)TEMP_FAILURE_RETRY(pipe(pipefd));
986
 
#else  /* O_CLOEXEC */
987
 
    ret = (int)TEMP_FAILURE_RETRY(pipe2(pipefd, O_CLOEXEC));
988
 
#endif  /* O_CLOEXEC */
989
 
    if(ret == -1){
990
 
      error(0, errno, "pipe");
991
 
      exitstatus = EX_OSERR;
992
 
      free(direntries[i]);
993
 
      goto fallback;
994
 
    }
995
 
    if(pipefd[0] >= FD_SETSIZE){
996
 
      fprintf(stderr, "pipe()[0] (%d) >= FD_SETSIZE (%d)", pipefd[0],
997
 
              FD_SETSIZE);
998
 
      close(pipefd[0]);
999
 
      close(pipefd[1]);
1000
 
      exitstatus = EX_OSERR;
1001
 
      free(direntries[i]);
1002
 
      goto fallback;
1003
 
    }
1004
 
#ifndef O_CLOEXEC
1005
 
    /* Ask OS to automatic close the pipe on exec */
 
724
    ret = pipe(pipefd);
 
725
    if (ret == -1){
 
726
      perror("pipe");
 
727
      exitstatus = EXIT_FAILURE;
 
728
      goto fallback;
 
729
    }
1006
730
    ret = set_cloexec_flag(pipefd[0]);
1007
731
    if(ret < 0){
1008
 
      error(0, errno, "set_cloexec_flag");
1009
 
      close(pipefd[0]);
1010
 
      close(pipefd[1]);
1011
 
      exitstatus = EX_OSERR;
1012
 
      free(direntries[i]);
 
732
      perror("set_cloexec_flag");
 
733
      exitstatus = EXIT_FAILURE;
1013
734
      goto fallback;
1014
735
    }
1015
736
    ret = set_cloexec_flag(pipefd[1]);
1016
737
    if(ret < 0){
1017
 
      error(0, errno, "set_cloexec_flag");
1018
 
      close(pipefd[0]);
1019
 
      close(pipefd[1]);
1020
 
      exitstatus = EX_OSERR;
1021
 
      free(direntries[i]);
 
738
      perror("set_cloexec_flag");
 
739
      exitstatus = EXIT_FAILURE;
1022
740
      goto fallback;
1023
741
    }
1024
 
#endif  /* not O_CLOEXEC */
1025
742
    /* Block SIGCHLD until process is safely in process list */
1026
 
    ret = (int)TEMP_FAILURE_RETRY(sigprocmask(SIG_BLOCK,
1027
 
                                              &sigchld_action.sa_mask,
1028
 
                                              NULL));
 
743
    ret = sigprocmask (SIG_BLOCK, &sigchld_action.sa_mask, NULL);
1029
744
    if(ret < 0){
1030
 
      error(0, errno, "sigprocmask");
1031
 
      exitstatus = EX_OSERR;
1032
 
      free(direntries[i]);
 
745
      perror("sigprocmask");
 
746
      exitstatus = EXIT_FAILURE;
1033
747
      goto fallback;
1034
748
    }
1035
 
    /* Starting a new process to be watched */
1036
 
    pid_t pid;
1037
 
    do {
1038
 
      pid = fork();
1039
 
    } while(pid == -1 and errno == EINTR);
 
749
    // Starting a new process to be watched
 
750
    pid_t pid = fork();
1040
751
    if(pid == -1){
1041
 
      error(0, errno, "fork");
1042
 
      TEMP_FAILURE_RETRY(sigprocmask(SIG_UNBLOCK,
1043
 
                                     &sigchld_action.sa_mask, NULL));
1044
 
      close(pipefd[0]);
1045
 
      close(pipefd[1]);
1046
 
      exitstatus = EX_OSERR;
1047
 
      free(direntries[i]);
 
752
      perror("fork");
 
753
      exitstatus = EXIT_FAILURE;
1048
754
      goto fallback;
1049
755
    }
1050
756
    if(pid == 0){
1051
757
      /* this is the child process */
1052
758
      ret = sigaction(SIGCHLD, &old_sigchld_action, NULL);
1053
759
      if(ret < 0){
1054
 
        error(0, errno, "sigaction");
1055
 
        _exit(EX_OSERR);
 
760
        perror("sigaction");
 
761
        _exit(EXIT_FAILURE);
1056
762
      }
1057
 
      ret = sigprocmask(SIG_UNBLOCK, &sigchld_action.sa_mask, NULL);
 
763
      ret = sigprocmask (SIG_UNBLOCK, &sigchld_action.sa_mask, NULL);
1058
764
      if(ret < 0){
1059
 
        error(0, errno, "sigprocmask");
1060
 
        _exit(EX_OSERR);
 
765
        perror("sigprocmask");
 
766
        _exit(EXIT_FAILURE);
1061
767
      }
1062
 
      
 
768
 
1063
769
      ret = dup2(pipefd[1], STDOUT_FILENO); /* replace our stdout */
1064
770
      if(ret == -1){
1065
 
        error(0, errno, "dup2");
1066
 
        _exit(EX_OSERR);
 
771
        perror("dup2");
 
772
        _exit(EXIT_FAILURE);
1067
773
      }
1068
774
      
1069
 
      if(fexecve(plugin_fd, p->argv,
1070
 
                (p->environ[0] != NULL) ? p->environ : environ) < 0){
1071
 
        error(0, errno, "fexecve for %s/%s",
1072
 
              plugindir != NULL ? plugindir : PDIR,
1073
 
              direntries[i]->d_name);
1074
 
        _exit(EX_OSERR);
 
775
      if(dirfd(dir) < 0){
 
776
        /* If dir has no file descriptor, we could not set FD_CLOEXEC
 
777
           above and must now close it manually here. */
 
778
        closedir(dir);
 
779
      }
 
780
      if(p->environ[0] == NULL){
 
781
        if(execv(filename, p->argv) < 0){
 
782
          perror("execv");
 
783
          _exit(EXIT_FAILURE);
 
784
        }
 
785
      } else {
 
786
        if(execve(filename, p->argv, p->environ) < 0){
 
787
          perror("execve");
 
788
          _exit(EXIT_FAILURE);
 
789
        }
1075
790
      }
1076
791
      /* no return */
1077
792
    }
1078
 
    /* Parent process */
1079
 
    close(pipefd[1]);           /* Close unused write end of pipe */
1080
 
    close(plugin_fd);
1081
 
    plugin *new_plugin = getplugin(direntries[i]->d_name);
1082
 
    if(new_plugin == NULL){
1083
 
      error(0, errno, "getplugin");
1084
 
      ret = (int)(TEMP_FAILURE_RETRY
1085
 
                  (sigprocmask(SIG_UNBLOCK, &sigchld_action.sa_mask,
1086
 
                               NULL)));
 
793
    /* parent process */
 
794
    free(filename);
 
795
    close(pipefd[1]);           /* close unused write end of pipe */
 
796
    process *new_process = malloc(sizeof(process));
 
797
    if (new_process == NULL){
 
798
      perror("malloc");
 
799
      ret = sigprocmask (SIG_UNBLOCK, &sigchld_action.sa_mask, NULL);
1087
800
      if(ret < 0){
1088
 
        error(0, errno, "sigprocmask");
 
801
        perror("sigprocmask");
1089
802
      }
1090
 
      exitstatus = EX_OSERR;
1091
 
      free(direntries[i]);
 
803
      exitstatus = EXIT_FAILURE;
1092
804
      goto fallback;
1093
805
    }
1094
 
    free(direntries[i]);
1095
806
    
1096
 
    new_plugin->pid = pid;
1097
 
    new_plugin->fd = pipefd[0];
1098
 
 
1099
 
    if(debug){
1100
 
      fprintf(stderr, "Plugin %s started (PID %" PRIdMAX ")\n",
1101
 
              new_plugin->name, (intmax_t) (new_plugin->pid));
1102
 
    }
1103
 
 
 
807
    *new_process = (struct process){ .pid = pid,
 
808
                                     .fd = pipefd[0],
 
809
                                     .next = process_list };
 
810
    // List handling
 
811
    process_list = new_process;
1104
812
    /* Unblock SIGCHLD so signal handler can be run if this process
1105
813
       has already completed */
1106
 
    ret = (int)TEMP_FAILURE_RETRY(sigprocmask(SIG_UNBLOCK,
1107
 
                                              &sigchld_action.sa_mask,
1108
 
                                              NULL));
 
814
    ret = sigprocmask (SIG_UNBLOCK, &sigchld_action.sa_mask, NULL);
1109
815
    if(ret < 0){
1110
 
      error(0, errno, "sigprocmask");
1111
 
      exitstatus = EX_OSERR;
 
816
      perror("sigprocmask");
 
817
      exitstatus = EXIT_FAILURE;
1112
818
      goto fallback;
1113
819
    }
1114
820
    
1115
 
    FD_SET(new_plugin->fd, &rfds_all);
1116
 
    
1117
 
    if(maxfd < new_plugin->fd){
1118
 
      maxfd = new_plugin->fd;
1119
 
    }
1120
 
  }
1121
 
  
1122
 
  free(direntries);
1123
 
  direntries = NULL;
1124
 
  close(dir_fd);
1125
 
  dir_fd = -1;
1126
 
  free_plugin(getplugin(NULL));
1127
 
  
1128
 
  for(plugin *p = plugin_list; p != NULL; p = p->next){
1129
 
    if(p->pid != 0){
1130
 
      break;
1131
 
    }
1132
 
    if(p->next == NULL){
1133
 
      fprintf(stderr, "No plugin processes started. Incorrect plugin"
1134
 
              " directory?\n");
1135
 
      free_plugin_list();
1136
 
    }
1137
 
  }
1138
 
  
1139
 
  /* Main loop while running plugins exist */
1140
 
  while(plugin_list){
 
821
    FD_SET(new_process->fd, &rfds_all);
 
822
    
 
823
    if (maxfd < new_process->fd){
 
824
      maxfd = new_process->fd;
 
825
    }
 
826
    
 
827
  }
 
828
 
 
829
  free_plugin_list(plugin_list);
 
830
  plugin_list = NULL;
 
831
  
 
832
  closedir(dir);
 
833
  dir = NULL;
 
834
    
 
835
  if (process_list == NULL){
 
836
    fprintf(stderr, "No plugin processes started. Incorrect plugin"
 
837
            " directory?\n");
 
838
    process_list = NULL;
 
839
  }
 
840
  while(process_list){
1141
841
    fd_set rfds = rfds_all;
1142
842
    int select_ret = select(maxfd+1, &rfds, NULL, NULL, NULL);
1143
 
    if(select_ret == -1 and errno != EINTR){
1144
 
      error(0, errno, "select");
1145
 
      exitstatus = EX_OSERR;
 
843
    if (select_ret == -1){
 
844
      perror("select");
 
845
      exitstatus = EXIT_FAILURE;
1146
846
      goto fallback;
1147
847
    }
1148
848
    /* OK, now either a process completed, or something can be read
1149
849
       from one of them */
1150
 
    for(plugin *proc = plugin_list; proc != NULL;){
 
850
    for(process *proc = process_list; proc ; proc = proc->next){
1151
851
      /* Is this process completely done? */
1152
 
      if(proc->completed and proc->eof){
 
852
      if(proc->eof and proc->completed){
1153
853
        /* Only accept the plugin output if it exited cleanly */
1154
854
        if(not WIFEXITED(proc->status)
1155
855
           or WEXITSTATUS(proc->status) != 0){
1156
856
          /* Bad exit by plugin */
1157
 
          
1158
857
          if(debug){
1159
858
            if(WIFEXITED(proc->status)){
1160
 
              fprintf(stderr, "Plugin %s [%" PRIdMAX "] exited with"
1161
 
                      " status %d\n", proc->name,
1162
 
                      (intmax_t) (proc->pid),
 
859
              fprintf(stderr, "Plugin %u exited with status %d\n",
 
860
                      (unsigned int) (proc->pid),
1163
861
                      WEXITSTATUS(proc->status));
1164
 
            } else if(WIFSIGNALED(proc->status)){
1165
 
              fprintf(stderr, "Plugin %s [%" PRIdMAX "] killed by"
1166
 
                      " signal %d: %s\n", proc->name,
1167
 
                      (intmax_t) (proc->pid),
1168
 
                      WTERMSIG(proc->status),
1169
 
                      strsignal(WTERMSIG(proc->status)));
 
862
            } else if(WIFSIGNALED(proc->status)) {
 
863
              fprintf(stderr, "Plugin %u killed by signal %d\n",
 
864
                      (unsigned int) (proc->pid),
 
865
                      WTERMSIG(proc->status));
 
866
            } else if(WCOREDUMP(proc->status)){
 
867
              fprintf(stderr, "Plugin %d dumped core\n",
 
868
                      (unsigned int) (proc->pid));
1170
869
            }
1171
870
          }
1172
 
          
1173
871
          /* Remove the plugin */
1174
872
          FD_CLR(proc->fd, &rfds_all);
1175
 
          
1176
873
          /* Block signal while modifying process_list */
1177
 
          ret = (int)TEMP_FAILURE_RETRY(sigprocmask
1178
 
                                        (SIG_BLOCK,
1179
 
                                         &sigchld_action.sa_mask,
1180
 
                                         NULL));
 
874
          ret = sigprocmask(SIG_BLOCK, &sigchld_action.sa_mask, NULL);
1181
875
          if(ret < 0){
1182
 
            error(0, errno, "sigprocmask");
1183
 
            exitstatus = EX_OSERR;
 
876
            perror("sigprocmask");
 
877
            exitstatus = EXIT_FAILURE;
1184
878
            goto fallback;
1185
879
          }
1186
 
          
1187
 
          plugin *next_plugin = proc->next;
1188
 
          free_plugin(proc);
1189
 
          proc = next_plugin;
1190
 
          
 
880
          /* Delete this process entry from the list */
 
881
          if(process_list == proc){
 
882
            /* First one - simple */
 
883
            process_list = proc->next;
 
884
          } else {
 
885
            /* Second one or later */
 
886
            for(process *p = process_list; p != NULL; p = p->next){
 
887
              if(p->next == proc){
 
888
                p->next = proc->next;
 
889
                break;
 
890
              }
 
891
            }
 
892
          }
1191
893
          /* We are done modifying process list, so unblock signal */
1192
 
          ret = (int)(TEMP_FAILURE_RETRY
1193
 
                      (sigprocmask(SIG_UNBLOCK,
1194
 
                                   &sigchld_action.sa_mask, NULL)));
 
894
          ret = sigprocmask (SIG_UNBLOCK, &sigchld_action.sa_mask,
 
895
                             NULL);
1195
896
          if(ret < 0){
1196
 
            error(0, errno, "sigprocmask");
1197
 
            exitstatus = EX_OSERR;
1198
 
            goto fallback;
1199
 
          }
1200
 
          
1201
 
          if(plugin_list == NULL){
1202
 
            break;
1203
 
          }
1204
 
          
1205
 
          continue;
 
897
            perror("sigprocmask");
 
898
          }
 
899
          free(proc->buffer);
 
900
          free(proc);
 
901
          /* We deleted this process from the list, so we can't go
 
902
             proc->next.  Therefore, start over from the beginning of
 
903
             the process list */
 
904
          break;
1206
905
        }
1207
 
        
1208
906
        /* This process exited nicely, so print its buffer */
1209
 
        
 
907
 
1210
908
        bool bret = print_out_password(proc->buffer,
1211
909
                                       proc->buffer_length);
1212
910
        if(not bret){
1213
 
          error(0, errno, "print_out_password");
1214
 
          exitstatus = EX_IOERR;
 
911
          perror("print_out_password");
 
912
          exitstatus = EXIT_FAILURE;
1215
913
        }
1216
914
        goto fallback;
1217
915
      }
1218
 
      
1219
916
      /* This process has not completed.  Does it have any output? */
1220
917
      if(proc->eof or not FD_ISSET(proc->fd, &rfds)){
1221
918
        /* This process had nothing to say at this time */
1222
 
        proc = proc->next;
1223
919
        continue;
1224
920
      }
1225
921
      /* Before reading, make the process' data buffer large enough */
1226
922
      if(proc->buffer_length + BUFFER_SIZE > proc->buffer_size){
1227
 
        char *new_buffer = realloc(proc->buffer, proc->buffer_size
1228
 
                                   + (size_t) BUFFER_SIZE);
1229
 
        if(new_buffer == NULL){
1230
 
          error(0, errno, "malloc");
1231
 
          exitstatus = EX_OSERR;
 
923
        proc->buffer = realloc(proc->buffer, proc->buffer_size
 
924
                               + (size_t) BUFFER_SIZE);
 
925
        if (proc->buffer == NULL){
 
926
          perror("malloc");
 
927
          exitstatus = EXIT_FAILURE;
1232
928
          goto fallback;
1233
929
        }
1234
 
        proc->buffer = new_buffer;
1235
930
        proc->buffer_size += BUFFER_SIZE;
1236
931
      }
1237
932
      /* Read from the process */
1238
 
      sret = TEMP_FAILURE_RETRY(read(proc->fd,
1239
 
                                     proc->buffer
1240
 
                                     + proc->buffer_length,
1241
 
                                     BUFFER_SIZE));
1242
 
      if(sret < 0){
 
933
      ret = read(proc->fd, proc->buffer + proc->buffer_length,
 
934
                 BUFFER_SIZE);
 
935
      if(ret < 0){
1243
936
        /* Read error from this process; ignore the error */
1244
 
        proc = proc->next;
1245
937
        continue;
1246
938
      }
1247
 
      if(sret == 0){
 
939
      if(ret == 0){
1248
940
        /* got EOF */
1249
941
        proc->eof = true;
1250
942
      } else {
1251
 
        proc->buffer_length += (size_t) sret;
 
943
        proc->buffer_length += (size_t) ret;
1252
944
      }
1253
945
    }
1254
946
  }
1255
 
  
1256
 
  
 
947
 
 
948
 
1257
949
 fallback:
1258
950
  
1259
 
  if(plugin_list == NULL or (exitstatus != EXIT_SUCCESS
1260
 
                             and exitstatus != EX_OK)){
 
951
  if(process_list == NULL or exitstatus != EXIT_SUCCESS){
1261
952
    /* Fallback if all plugins failed, none are found or an error
1262
953
       occured */
1263
954
    bool bret;
1264
955
    fprintf(stderr, "Going to fallback mode using getpass(3)\n");
1265
956
    char *passwordbuffer = getpass("Password: ");
1266
 
    size_t len = strlen(passwordbuffer);
1267
 
    /* Strip trailing newline */
1268
 
    if(len > 0 and passwordbuffer[len-1] == '\n'){
1269
 
      passwordbuffer[len-1] = '\0'; /* not strictly necessary */
1270
 
      len--;
1271
 
    }
1272
 
    bret = print_out_password(passwordbuffer, len);
 
957
    bret = print_out_password(passwordbuffer, strlen(passwordbuffer));
1273
958
    if(not bret){
1274
 
      error(0, errno, "print_out_password");
1275
 
      exitstatus = EX_IOERR;
 
959
      perror("print_out_password");
 
960
      exitstatus = EXIT_FAILURE;
1276
961
    }
1277
962
  }
1278
963
  
1279
964
  /* Restore old signal handler */
1280
965
  ret = sigaction(SIGCHLD, &old_sigchld_action, NULL);
1281
966
  if(ret == -1){
1282
 
    error(0, errno, "sigaction");
1283
 
    exitstatus = EX_OSERR;
 
967
    perror("sigaction");
 
968
    exitstatus = EXIT_FAILURE;
1284
969
  }
1285
 
  
 
970
 
1286
971
  if(custom_argv != NULL){
1287
972
    for(char **arg = custom_argv+1; *arg != NULL; arg++){
1288
973
      free(*arg);
1289
974
    }
1290
975
    free(custom_argv);
1291
976
  }
1292
 
  
1293
 
  free(direntries);
1294
 
  
1295
 
  if(dir_fd != -1){
1296
 
    close(dir_fd);
 
977
  free_plugin_list(plugin_list);
 
978
  
 
979
  if(dir != NULL){
 
980
    closedir(dir);
1297
981
  }
1298
982
  
1299
 
  /* Kill the processes */
1300
 
  for(plugin *p = plugin_list; p != NULL; p = p->next){
1301
 
    if(p->pid != 0){
1302
 
      close(p->fd);
1303
 
      ret = kill(p->pid, SIGTERM);
1304
 
      if(ret == -1 and errno != ESRCH){
1305
 
        /* Set-uid proccesses might not get closed */
1306
 
        error(0, errno, "kill");
1307
 
      }
 
983
  /* Free the process list and kill the processes */
 
984
  for(process *next; process_list != NULL; process_list = next){
 
985
    next = process_list->next;
 
986
    close(process_list->fd);
 
987
    ret = kill(process_list->pid, SIGTERM);
 
988
    if(ret == -1 and errno != ESRCH){
 
989
      /* set-uid proccesses migth not get closed */
 
990
      perror("kill");
1308
991
    }
 
992
    free(process_list->buffer);
 
993
    free(process_list);
1309
994
  }
1310
995
  
1311
996
  /* Wait for any remaining child processes to terminate */
1312
 
  do {
 
997
  do{
1313
998
    ret = wait(NULL);
1314
999
  } while(ret >= 0);
1315
1000
  if(errno != ECHILD){
1316
 
    error(0, errno, "wait");
 
1001
    perror("wait");
1317
1002
  }
1318
 
  
1319
 
  free_plugin_list();
1320
 
  
 
1003
 
1321
1004
  free(plugindir);
1322
 
  free(pluginhelperdir);
1323
1005
  free(argfile);
1324
1006
  
1325
1007
  return exitstatus;