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
27NSS_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
34struct 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))
40LINE_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
47enum nss_status
48_nss_hesiod_setservent (int stayopen)
49{
50 return NSS_STATUS_SUCCESS;
51}
52
53enum nss_status
54_nss_hesiod_endservent (void)
55{
56 return NSS_STATUS_SUCCESS;
57}
58
59static enum nss_status
60lookup (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: &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 (s: *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 (dest: data->linebuffer, src: *item, n: len);
100
101 parse_res = parse_line (line: buffer, generic_result: serv, data, datalen: 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 (s1: serv->s_proto, s2: 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
128enum 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, type: "service", protocol, serv, buffer, buflen, errnop);
134}
135
136enum 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 (s: portstr, maxlen: sizeof portstr, format: "%d", ntohs (port));
144
145 return lookup (name: portstr, type: "port", protocol, serv, buffer, buflen, errnop);
146}
147

source code of glibc/hesiod/nss_hesiod/hesiod-service.c