]> git.ipfire.org Git - thirdparty/glibc.git/blob - nis/nss_nis/nis-rpc.c
8c1a3d4860738f56cd67207e864d01965b21ae8e
[thirdparty/glibc.git] / nis / nss_nis / nis-rpc.c
1 /* Copyright (C) 1996, 1997, 1998, 2000 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
3 Contributed by Thorsten Kukuk <kukuk@suse.de>, 1996.
4
5 The GNU C Library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
9
10 The GNU C Library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
14
15 You should have received a copy of the GNU Lesser General Public
16 License along with the GNU C Library; if not, write to the Free
17 Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
18 02111-1307 USA. */
19
20 #include <nss.h>
21 #include <netdb.h>
22 #include <ctype.h>
23 #include <errno.h>
24 #include <string.h>
25 #include <bits/libc-lock.h>
26 #include <rpcsvc/yp.h>
27 #include <rpcsvc/ypclnt.h>
28
29 #include "nss-nis.h"
30
31 /* Get the declaration of the parser function. */
32 #define ENTNAME rpcent
33 #define EXTERN_PARSER
34 #include <nss/nss_files/files-parse.c>
35
36 __libc_lock_define_initialized (static, lock)
37
38 struct response_t
39 {
40 char *val;
41 struct response_t *next;
42 };
43
44 struct intern_t
45 {
46 struct response_t *start;
47 struct response_t *next;
48 };
49 typedef struct intern_t intern_t;
50
51 static intern_t intern = {NULL, NULL};
52
53 static int
54 saveit (int instatus, char *inkey, int inkeylen, char *inval,
55 int invallen, char *indata)
56 {
57 intern_t *intern = (intern_t *)indata;
58
59 if (instatus != YP_TRUE)
60 return instatus;
61
62 if (inkey && inkeylen > 0 && inval && invallen > 0)
63 {
64 if (intern->start == NULL)
65 {
66 intern->start = malloc (sizeof (struct response_t));
67 if (intern->start == NULL)
68 return YP_FALSE;
69 intern->next = intern->start;
70 }
71 else
72 {
73 intern->next->next = malloc (sizeof (struct response_t));
74 if (intern->next->next == NULL)
75 return YP_FALSE;
76 intern->next = intern->next->next;
77 }
78 intern->next->next = NULL;
79 intern->next->val = malloc (invallen + 1);
80 if (intern->next->val == NULL)
81 return YP_FALSE;
82 strncpy (intern->next->val, inval, invallen);
83 intern->next->val[invallen] = '\0';
84 }
85
86 return 0;
87 }
88
89 static enum nss_status
90 internal_nis_setrpcent (intern_t *intern)
91 {
92 char *domainname;
93 struct ypall_callback ypcb;
94 enum nss_status status;
95
96 if (yp_get_default_domain (&domainname))
97 return NSS_STATUS_UNAVAIL;
98
99 while (intern->start != NULL)
100 {
101 if (intern->start->val != NULL)
102 free (intern->start->val);
103 intern->next = intern->start;
104 intern->start = intern->start->next;
105 free (intern->next);
106 }
107 intern->start = NULL;
108
109 ypcb.foreach = saveit;
110 ypcb.data = (char *)intern;
111 status = yperr2nss (yp_all(domainname, "rpc.bynumber", &ypcb));
112 intern->next = intern->start;
113
114 return status;
115 }
116
117 enum nss_status
118 _nss_nis_setrpcent (int stayopen)
119 {
120 enum nss_status status;
121
122 __libc_lock_lock (lock);
123
124 status = internal_nis_setrpcent (&intern);
125
126 __libc_lock_unlock (lock);
127
128 return status;
129 }
130
131 static enum nss_status
132 internal_nis_endrpcent (intern_t *intern)
133 {
134 while (intern->start != NULL)
135 {
136 if (intern->start->val != NULL)
137 free (intern->start->val);
138 intern->next = intern->start;
139 intern->start = intern->start->next;
140 free (intern->next);
141 }
142 intern->start = NULL;
143
144 return NSS_STATUS_SUCCESS;
145 }
146
147 enum nss_status
148 _nss_nis_endrpcent (void)
149 {
150 enum nss_status status;
151
152 __libc_lock_lock (lock);
153
154 status = internal_nis_endrpcent (&intern);
155
156 __libc_lock_unlock (lock);
157
158 return status;
159 }
160
161 static enum nss_status
162 internal_nis_getrpcent_r (struct rpcent *rpc, char *buffer, size_t buflen,
163 int *errnop, intern_t *data)
164 {
165 struct parser_data *pdata = (void *) buffer;
166 int parse_res;
167 char *p;
168
169 if (data->start == NULL)
170 internal_nis_setrpcent (data);
171
172 /* Get the next entry until we found a correct one. */
173 do
174 {
175 if (data->next == NULL)
176 {
177 *errnop = ENOENT;
178 return NSS_STATUS_NOTFOUND;
179 }
180 p = strncpy (buffer, data->next->val, buflen);
181 while (isspace (*p))
182 ++p;
183
184 parse_res = _nss_files_parse_rpcent (p, rpc, pdata, buflen, errnop);
185 if (parse_res == -1)
186 return NSS_STATUS_TRYAGAIN;
187 data->next = data->next->next;
188 }
189 while (!parse_res);
190
191 return NSS_STATUS_SUCCESS;
192 }
193
194 enum nss_status
195 _nss_nis_getrpcent_r (struct rpcent *rpc, char *buffer, size_t buflen,
196 int *errnop)
197 {
198 enum nss_status status;
199
200 __libc_lock_lock (lock);
201
202 status = internal_nis_getrpcent_r (rpc, buffer, buflen, errnop, &intern);
203
204 __libc_lock_unlock (lock);
205
206 return status;
207 }
208
209 enum nss_status
210 _nss_nis_getrpcbyname_r (const char *name, struct rpcent *rpc,
211 char *buffer, size_t buflen, int *errnop)
212 {
213 intern_t data = {NULL, NULL};
214 enum nss_status status;
215 int found;
216
217 if (name == NULL)
218 {
219 *errnop = EINVAL;
220 return NSS_STATUS_UNAVAIL;
221 }
222
223 status = internal_nis_setrpcent (&data);
224 if (status != NSS_STATUS_SUCCESS)
225 return status;
226
227 found = 0;
228 while (!found &&
229 ((status = internal_nis_getrpcent_r (rpc, buffer, buflen, errnop,
230 &data)) == NSS_STATUS_SUCCESS))
231 {
232 if (strcmp (rpc->r_name, name) == 0)
233 found = 1;
234 else
235 {
236 int i = 0;
237
238 while (rpc->r_aliases[i] != NULL)
239 {
240 if (strcmp (rpc->r_aliases[i], name) == 0)
241 {
242 found = 1;
243 break;
244 }
245 else
246 ++i;
247 }
248 }
249 }
250
251 internal_nis_endrpcent (&data);
252
253 if (!found && status == NSS_STATUS_SUCCESS)
254 {
255 *errnop = ENOENT;
256 return NSS_STATUS_NOTFOUND;
257 }
258 else
259 return status;
260 }
261
262 enum nss_status
263 _nss_nis_getrpcbynumber_r (int number, struct rpcent *rpc,
264 char *buffer, size_t buflen, int *errnop)
265 {
266 struct parser_data *data = (void *) buffer;
267 enum nss_status retval;
268 char *domain, *result, *p;
269 int len, nlen, parse_res;
270 char buf[32];
271
272 if (yp_get_default_domain (&domain))
273 return NSS_STATUS_UNAVAIL;
274
275 nlen = sprintf (buf, "%d", number);
276
277 retval = yperr2nss (yp_match (domain, "rpc.bynumber", buf,
278 nlen, &result, &len));
279
280 if (retval != NSS_STATUS_SUCCESS)
281 {
282 if (retval == NSS_STATUS_NOTFOUND)
283 *errnop = ENOENT;
284 else if (retval == NSS_STATUS_TRYAGAIN)
285 *errnop = errno;
286 return retval;
287 }
288
289 if ((size_t) (len + 1) > buflen)
290 {
291 free (result);
292 *errnop = ERANGE;
293 return NSS_STATUS_TRYAGAIN;
294 }
295
296 p = strncpy (buffer, result, len);
297 buffer[len] = '\0';
298 while (isspace (*p))
299 ++p;
300 free (result);
301
302 parse_res = _nss_files_parse_rpcent (p, rpc, data, buflen, errnop);
303
304 if (parse_res < 1)
305 {
306 if (parse_res == -1)
307 return NSS_STATUS_TRYAGAIN;
308 else
309 {
310 *errnop = ENOENT;
311 return NSS_STATUS_NOTFOUND;
312 }
313 }
314 else
315 return NSS_STATUS_SUCCESS;
316 }