OLD | NEW |
1 #define _GNU_SOURCE | 1 #define _GNU_SOURCE |
2 #include "pwf.h" | 2 #include "pwf.h" |
3 #include <grp.h> | 3 #include <grp.h> |
4 #include <string.h> | 4 #include <string.h> |
5 #include <limits.h> | 5 #include <limits.h> |
6 #include <stdio.h> | 6 #include <stdio.h> |
7 #include <stdlib.h> | 7 #include <stdlib.h> |
8 #include <byteswap.h> | 8 #include <byteswap.h> |
9 #include <errno.h> | 9 #include <errno.h> |
10 #include "nscd.h" | 10 #include "nscd.h" |
11 | 11 |
12 int getgrouplist(const char *user, gid_t gid, gid_t *groups, int *ngroups) | 12 int getgrouplist(const char* user, gid_t gid, gid_t* groups, int* ngroups) { |
13 { | 13 int rv, nlim, ret = -1; |
14 » int rv, nlim, ret = -1; | 14 ssize_t i, n = 1; |
15 » ssize_t i, n = 1; | 15 struct group gr; |
16 » struct group gr; | 16 struct group* res; |
17 » struct group *res; | 17 FILE* f; |
18 » FILE *f; | 18 int swap = 0; |
19 » int swap = 0; | 19 int32_t resp[INITGR_LEN]; |
20 » int32_t resp[INITGR_LEN]; | 20 uint32_t* nscdbuf = 0; |
21 » uint32_t *nscdbuf = 0; | 21 char* buf = 0; |
22 » char *buf = 0; | 22 char** mem = 0; |
23 » char **mem = 0; | 23 size_t nmem = 0; |
24 » size_t nmem = 0; | 24 size_t size; |
25 » size_t size; | 25 nlim = *ngroups; |
26 » nlim = *ngroups; | 26 if (nlim >= 1) |
27 » if (nlim >= 1) *groups++ = gid; | 27 *groups++ = gid; |
28 | 28 |
29 » f = __nscd_query(GETINITGR, user, resp, sizeof resp, &swap); | 29 f = __nscd_query(GETINITGR, user, resp, sizeof resp, &swap); |
30 » if (!f) goto cleanup; | 30 if (!f) |
31 » if (resp[INITGRFOUND]) { | 31 goto cleanup; |
32 » » nscdbuf = calloc(resp[INITGRNGRPS], sizeof(uint32_t)); | 32 if (resp[INITGRFOUND]) { |
33 » » if (!nscdbuf) goto cleanup; | 33 nscdbuf = calloc(resp[INITGRNGRPS], sizeof(uint32_t)); |
34 » » if (!fread(nscdbuf, sizeof(*nscdbuf)*resp[INITGRNGRPS], 1, f)) { | 34 if (!nscdbuf) |
35 » » » if (!ferror(f)) errno = EIO; | 35 goto cleanup; |
36 » » » goto cleanup; | 36 if (!fread(nscdbuf, sizeof(*nscdbuf) * resp[INITGRNGRPS], 1, f)) { |
37 » » } | 37 if (!ferror(f)) |
38 » » if (swap) { | 38 errno = EIO; |
39 » » » for (i = 0; i < resp[INITGRNGRPS]; i++) | 39 goto cleanup; |
40 » » » » nscdbuf[i] = bswap_32(nscdbuf[i]); | 40 } |
41 » » } | 41 if (swap) { |
42 » } | 42 for (i = 0; i < resp[INITGRNGRPS]; i++) |
43 » fclose(f); | 43 nscdbuf[i] = bswap_32(nscdbuf[i]); |
| 44 } |
| 45 } |
| 46 fclose(f); |
44 | 47 |
45 » f = fopen("/etc/group", "rbe"); | 48 f = fopen("/etc/group", "rbe"); |
46 » if (!f && errno != ENOENT && errno != ENOTDIR) | 49 if (!f && errno != ENOENT && errno != ENOTDIR) |
47 » » goto cleanup; | 50 goto cleanup; |
48 | 51 |
49 » if (f) { | 52 if (f) { |
50 » » while (!(rv = __getgrent_a(f, &gr, &buf, &size, &mem, &nmem, &re
s)) && res) { | 53 while (!(rv = __getgrent_a(f, &gr, &buf, &size, &mem, &nmem, &res)) && |
51 » » » if (nscdbuf) | 54 res) { |
52 » » » » for (i=0; i < resp[INITGRNGRPS]; i++) { | 55 if (nscdbuf) |
53 » » » » » if (nscdbuf[i] == gr.gr_gid) nscdbuf[i]
= gid; | 56 for (i = 0; i < resp[INITGRNGRPS]; i++) { |
54 » » » » } | 57 if (nscdbuf[i] == gr.gr_gid) |
55 » » » for (i=0; gr.gr_mem[i] && strcmp(user, gr.gr_mem[i]); i+
+); | 58 nscdbuf[i] = gid; |
56 » » » if (!gr.gr_mem[i]) continue; | 59 } |
57 » » » if (++n <= nlim) *groups++ = gr.gr_gid; | 60 for (i = 0; gr.gr_mem[i] && strcmp(user, gr.gr_mem[i]); i++) |
58 » » } | 61 ; |
59 » » if (rv) { | 62 if (!gr.gr_mem[i]) |
60 » » » errno = rv; | 63 continue; |
61 » » » goto cleanup; | 64 if (++n <= nlim) |
62 » » } | 65 *groups++ = gr.gr_gid; |
63 » } | 66 } |
64 » if (nscdbuf) { | 67 if (rv) { |
65 » » for(i=0; i < resp[INITGRNGRPS]; i++) { | 68 errno = rv; |
66 » » » if (nscdbuf[i] != gid) | 69 goto cleanup; |
67 » » » » if(++n <= nlim) *groups++ = nscdbuf[i]; | 70 } |
68 » » } | 71 } |
69 » } | 72 if (nscdbuf) { |
| 73 for (i = 0; i < resp[INITGRNGRPS]; i++) { |
| 74 if (nscdbuf[i] != gid) |
| 75 if (++n <= nlim) |
| 76 *groups++ = nscdbuf[i]; |
| 77 } |
| 78 } |
70 | 79 |
71 » ret = n > nlim ? -1 : n; | 80 ret = n > nlim ? -1 : n; |
72 » *ngroups = n; | 81 *ngroups = n; |
73 | 82 |
74 cleanup: | 83 cleanup: |
75 » if (f) fclose(f); | 84 if (f) |
76 » free(nscdbuf); | 85 fclose(f); |
77 » free(buf); | 86 free(nscdbuf); |
78 » free(mem); | 87 free(buf); |
79 » return ret; | 88 free(mem); |
| 89 return ret; |
80 } | 90 } |
OLD | NEW |