make passwd/group functions safe against cancellation in stdio
[musl] / src / passwd / getgrent_a.c
1 #include "pwf.h"
2 #include <pthread.h>
3
4 struct group *__getgrent_a(FILE *f, struct group *gr, char **line, size_t *size, char ***mem, size_t *nmem)
5 {
6         ssize_t l;
7         char *s, *mems;
8         size_t i;
9         int cs;
10         pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, &cs);
11         for (;;) {
12                 if ((l=getline(line, size, f)) < 0) {
13                         free(*line);
14                         *line = 0;
15                         gr = 0;
16                         goto end;
17                 }
18                 line[0][l-1] = 0;
19
20                 s = line[0];
21                 gr->gr_name = s++;
22                 if (!(s = strchr(s, ':'))) continue;
23
24                 *s++ = 0; gr->gr_passwd = s;
25                 if (!(s = strchr(s, ':'))) continue;
26
27                 *s++ = 0; gr->gr_gid = atoi(s);
28                 if (!(s = strchr(s, ':'))) continue;
29
30                 *s++ = 0; mems = s;
31                 break;
32         }
33
34         for (*nmem=!!*s; *s; s++)
35                 if (*s==',') ++*nmem;
36         free(*mem);
37         *mem = calloc(sizeof(char *), *nmem+1);
38         if (!*mem) {
39                 free(*line);
40                 *line = 0;
41                 return 0;
42         }
43         if (*mems) {
44                 mem[0][0] = mems;
45                 for (s=mems, i=0; *s; s++)
46                         if (*s==',') *s++ = 0, mem[0][++i] = s;
47                 mem[0][++i] = 0;
48         } else {
49                 mem[0][0] = 0;
50         }
51         gr->gr_mem = *mem;
52 end:
53         pthread_setcancelstate(cs, 0);
54         return gr;
55 }