initial commit
[glibc.git] / hesiod / nss_hesiod / hesiod-service.c
1 /* Copyright (C) 1997-2022 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
3
4 The GNU C Library is free software; you can redistribute it and/or
5 modify it under the terms of the GNU Lesser General Public
6 License as published by the Free Software Foundation; either
7 version 2.1 of the License, or (at your option) any later version.
8
9 The GNU C Library is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 Lesser General Public License for more details.
13
14 You should have received a copy of the GNU Lesser General Public
15 License along with the GNU C Library; if not, see
16 <https://www.gnu.org/licenses/>. */
17
18 #include <errno.h>
19 #include <hesiod.h>
20 #include <netdb.h>
21 #include <netinet/in.h>
22 #include <nss.h>
23 #include <stdio.h>
24 #include <stdlib.h>
25 #include <string.h>
26
27 NSS_DECLARE_MODULE_FUNCTIONS (hesiod)
28
29 /* Hesiod uses a format for service entries that differs from the
30 traditional format. We therefore declare our own parser. */
31
32 #define ENTNAME servent
33
34 struct servent_data {};
35
36 #define TRAILING_LIST_MEMBER s_aliases
37 #define TRAILING_LIST_SEPARATOR_P isspace
38 #include <nss/nss_files/files-parse.c>
39 #define ISSC_OR_SPACE(c) ((c) == ';' || isspace (c))
40 LINE_PARSER
41 ("#",
42 STRING_FIELD (result->s_name, ISSC_OR_SPACE, 1);
43 STRING_FIELD (result->s_proto, ISSC_OR_SPACE, 1);
44 INT_FIELD (result->s_port, ISSC_OR_SPACE, 10, 0, htons);
45 )
46
47 enum nss_status
48 _nss_hesiod_setservent (int stayopen)
49 {
50 return NSS_STATUS_SUCCESS;
51 }
52
53 enum nss_status
54 _nss_hesiod_endservent (void)
55 {
56 return NSS_STATUS_SUCCESS;
57 }
58
59 static enum nss_status
60 lookup (const char *name, const char *type, const char *protocol,
61 struct servent *serv, char *buffer, size_t buflen, int *errnop)
62 {
63 struct parser_data *data = (void *) buffer;
64 size_t linebuflen;
65 void *context;
66 char **list, **item;
67 int parse_res;
68 int found;
69 int olderr = errno;
70
71 if (hesiod_init (&context) < 0)
72 return NSS_STATUS_UNAVAIL;
73
74 list = hesiod_resolve (context, name, type);
75 if (list == NULL)
76 {
77 int err = errno;
78 hesiod_end (context);
79 __set_errno (olderr);
80 return err == ENOENT ? NSS_STATUS_NOTFOUND : NSS_STATUS_UNAVAIL;
81 }
82
83 linebuflen = buffer + buflen - data->linebuffer;
84
85 item = list;
86 found = 0;
87 do
88 {
89 size_t len = strlen (*item) + 1;
90
91 if (linebuflen < len)
92 {
93 hesiod_free_list (context, list);
94 hesiod_end (context);
95 *errnop = ERANGE;
96 return NSS_STATUS_TRYAGAIN;
97 }
98
99 memcpy (data->linebuffer, *item, len);
100
101 parse_res = parse_line (buffer, serv, data, buflen, errnop);
102 if (parse_res == -1)
103 {
104 hesiod_free_list (context, list);
105 hesiod_end (context);
106 return NSS_STATUS_TRYAGAIN;
107 }
108
109 if (parse_res > 0)
110 found = protocol == NULL || strcasecmp (serv->s_proto, protocol) == 0;
111
112 ++item;
113 }
114 while (*item != NULL && !found);
115
116 hesiod_free_list (context, list);
117 hesiod_end (context);
118
119 if (found == 0)
120 {
121 __set_errno (olderr);
122 return NSS_STATUS_NOTFOUND;
123 }
124
125 return NSS_STATUS_SUCCESS;
126 }
127
128 enum nss_status
129 _nss_hesiod_getservbyname_r (const char *name, const char *protocol,
130 struct servent *serv,
131 char *buffer, size_t buflen, int *errnop)
132 {
133 return lookup (name, "service", protocol, serv, buffer, buflen, errnop);
134 }
135
136 enum nss_status
137 _nss_hesiod_getservbyport_r (const int port, const char *protocol,
138 struct servent *serv,
139 char *buffer, size_t buflen, int *errnop)
140 {
141 char portstr[6]; /* Port numbers are restricted to 16 bits. */
142
143 snprintf (portstr, sizeof portstr, "%d", ntohs (port));
144
145 return lookup (portstr, "port", protocol, serv, buffer, buflen, errnop);
146 }