The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
fd_config.c
Go to the documentation of this file.
1/*
2 * This program is is free software; you can redistribute it and/or modify
3 * it under the terms of the GNU General Public License as published by
4 * the Free Software Foundation; either version 2 of the License, or (at
5 * your option) any later version.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 *
12 * You should have received a copy of the GNU General Public License
13 * along with this program; if not, write to the Free Software
14 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
15 */
16
17/**
18 * $Id: 2385740290d9179add001dbdda7f5158fa34de7e $
19 * @file lib/bio/fd_config.c
20 * @brief BIO abstractions for configuring file descriptors.
21 *
22 * @copyright 2024 Network RADIUS SAS (legal@networkradius.com)
23 */
24
25#include <freeradius-devel/server/cf_parse.h>
26#include <freeradius-devel/server/tmpl.h>
27#include <freeradius-devel/util/perm.h>
28
29#include <freeradius-devel/bio/fd_priv.h>
30
32 { L("read-only"), O_RDONLY },
33 { L("read-write"), O_RDWR },
34 { L("ro"), O_RDONLY },
35 { L("rw"), O_RDWR },
36};
38
39static int mode_parse(UNUSED TALLOC_CTX *ctx, void *out, UNUSED void *parent, CONF_ITEM *ci, UNUSED conf_parser_t const *rule)
40{
41 int mode;
42 char const *name = cf_pair_value(cf_item_to_pair(ci));
43
45 if (mode < 0) {
46 cf_log_err(ci, "Invalid mode name \"%s\"", name);
47 return -1;
48 }
49
50 *(int *) out = mode;
51
52 return 0;
53}
54
55/** Parse "transport" and then set the subconfig
56 *
57 */
58static int common_transport_parse(UNUSED TALLOC_CTX *ctx, void *out, void *parent, CONF_ITEM *ci, UNUSED conf_parser_t const *rule, fr_table_ptr_sorted_t const *transport_table, size_t transport_table_len)
59{
60 int socket_type = SOCK_STREAM;
61 conf_parser_t const *rules;
62 char const *name = cf_pair_value(cf_item_to_pair(ci));
64 CONF_SECTION *cs, *subcs;
65
66 rules = fr_table_value_by_str(transport_table, name, NULL);
67 if (!rules) {
68 cf_log_err(ci, "Invalid transport name \"%s\"", name);
69 return -1;
70 }
71
73
74 /*
75 * Find the relevant subsection. Note that we don't do anything with it, as we push a parse
76 * rule in the parent which then points to the subsection.
77 */
78 subcs = cf_section_find(cs, name, NULL);
79 if (!subcs) {
80 cf_log_perr(ci, "Failed finding transport configuration section %s { ... }", name);
81 return -1;
82 }
83
84 /*
85 * Note that these offsets will get interpreted as being offsets from base of the subsection.
86 * i.e. the parent section and the subsection have to be parsed with the same base pointer.
87 */
88 if (cf_section_rules_push(cs, rules) < 0) {
89 cf_log_perr(ci, "Failed updating parse rules");
90 return -1;
91 }
92
93 if (strcmp(name, "udp") == 0) socket_type = SOCK_DGRAM;
94
95 /*
96 * Client sockets are always connected.
97 */
98 fd_config->socket_type = socket_type;
99 *(char const **) out = name;
100
101 return 0;
102}
103
106 { FR_CONF_OFFSET_TYPE_FLAGS("ipv4addr", FR_TYPE_IPV4_ADDR, 0, fr_bio_fd_config_t, dst_ipaddr) },
107 { FR_CONF_OFFSET_TYPE_FLAGS("ipv6addr", FR_TYPE_IPV6_ADDR, 0, fr_bio_fd_config_t, dst_ipaddr) },
108
109 { FR_CONF_OFFSET("port", fr_bio_fd_config_t, dst_port) },
110
111 { FR_CONF_OFFSET_TYPE_FLAGS("src_ipaddr", FR_TYPE_COMBO_IP_ADDR, 0, fr_bio_fd_config_t, src_ipaddr) },
112 { FR_CONF_OFFSET_TYPE_FLAGS("src_ipv4addr", FR_TYPE_IPV4_ADDR, 0, fr_bio_fd_config_t, src_ipaddr) },
113 { FR_CONF_OFFSET_TYPE_FLAGS("src_ipv6addr", FR_TYPE_IPV6_ADDR, 0, fr_bio_fd_config_t, src_ipaddr) },
114
115 { FR_CONF_OFFSET("src_port", fr_bio_fd_config_t, src_port) },
116 { FR_CONF_OFFSET("src_port_start", fr_bio_fd_config_t, src_port_start) },
117 { FR_CONF_OFFSET("src_port_end", fr_bio_fd_config_t, src_port_end) },
118
119 { FR_CONF_OFFSET("interface", fr_bio_fd_config_t, interface) },
120
121 { FR_CONF_OFFSET_IS_SET("recv_buff", FR_TYPE_UINT32, 0, fr_bio_fd_config_t, recv_buff) },
122 { FR_CONF_OFFSET_IS_SET("send_buff", FR_TYPE_UINT32, 0, fr_bio_fd_config_t, send_buff) },
123
125};
126
128 { FR_CONF_POINTER("udp", 0, CONF_FLAG_SUBSECTION, NULL), .subcs = (void const *) client_udp_sub_config },
129
131};
132
133
135 { FR_CONF_OFFSET_TYPE_FLAGS("src_ipaddr", FR_TYPE_COMBO_IP_ADDR, 0, fr_bio_fd_config_t, src_ipaddr) },
136 { FR_CONF_OFFSET_TYPE_FLAGS("src_ipv4addr", FR_TYPE_IPV4_ADDR, 0, fr_bio_fd_config_t, src_ipaddr) },
137 { FR_CONF_OFFSET_TYPE_FLAGS("src_ipv6addr", FR_TYPE_IPV6_ADDR, 0, fr_bio_fd_config_t, src_ipaddr) },
138
139 { FR_CONF_OFFSET("interface", fr_bio_fd_config_t, interface) },
140
141 { FR_CONF_OFFSET("src_port_start", fr_bio_fd_config_t, src_port_start) },
142 { FR_CONF_OFFSET("src_port_end", fr_bio_fd_config_t, src_port_end) },
143
144 { FR_CONF_OFFSET_IS_SET("recv_buff", FR_TYPE_UINT32, 0, fr_bio_fd_config_t, recv_buff) },
145 { FR_CONF_OFFSET_IS_SET("send_buff", FR_TYPE_UINT32, 0, fr_bio_fd_config_t, send_buff) },
146
148};
149
155
156
159 { FR_CONF_OFFSET_TYPE_FLAGS("ipv4addr", FR_TYPE_IPV4_ADDR, 0, fr_bio_fd_config_t, dst_ipaddr) },
160 { FR_CONF_OFFSET_TYPE_FLAGS("ipv6addr", FR_TYPE_IPV6_ADDR, 0, fr_bio_fd_config_t, dst_ipaddr) },
161
162 { FR_CONF_OFFSET("port", fr_bio_fd_config_t, dst_port) },
163
164 { FR_CONF_OFFSET_TYPE_FLAGS("src_ipaddr", FR_TYPE_COMBO_IP_ADDR, 0, fr_bio_fd_config_t, src_ipaddr) },
165 { FR_CONF_OFFSET_TYPE_FLAGS("src_ipv4addr", FR_TYPE_IPV4_ADDR, 0, fr_bio_fd_config_t, src_ipaddr) },
166 { FR_CONF_OFFSET_TYPE_FLAGS("src_ipv6addr", FR_TYPE_IPV6_ADDR, 0, fr_bio_fd_config_t, src_ipaddr) },
167
168 { FR_CONF_OFFSET("src_port", fr_bio_fd_config_t, src_port) },
169 { FR_CONF_OFFSET("src_port_start", fr_bio_fd_config_t, src_port_start) },
170 { FR_CONF_OFFSET("src_port_end", fr_bio_fd_config_t, src_port_end) },
171
172 { FR_CONF_OFFSET("interface", fr_bio_fd_config_t, interface) },
173
174 { FR_CONF_OFFSET_IS_SET("recv_buff", FR_TYPE_UINT32, 0, fr_bio_fd_config_t, recv_buff) },
175 { FR_CONF_OFFSET_IS_SET("send_buff", FR_TYPE_UINT32, 0, fr_bio_fd_config_t, send_buff) },
176
177 { FR_CONF_OFFSET("delay_tcp_writes", fr_bio_fd_config_t, tcp_delay) },
178
180};
181
183 { FR_CONF_POINTER("tcp", 0, CONF_FLAG_SUBSECTION, NULL), .subcs = (void const *) client_tcp_sub_config },
184
186};
187
190
191 { FR_CONF_OFFSET("permissions", fr_bio_fd_config_t, perm), .dflt = "0600", .func = cf_parse_permissions },
192
193 { FR_CONF_OFFSET("mode", fr_bio_fd_config_t, flags), .dflt = "read-write", .func = mode_parse },
194
196};
197
199 { FR_CONF_POINTER("file", 0, CONF_FLAG_SUBSECTION, NULL), .subcs = (void const *) client_file_sub_config },
200
202};
203
209
211 { FR_CONF_POINTER("unix", 0, CONF_FLAG_SUBSECTION, NULL), .subcs = (void const *) client_unix_sub_config },
212
214};
215
217 { L("file"), client_file_config },
218 { L("tcp"), client_tcp_config },
219 { L("udp"), client_udp_config },
220 { L("unix"), client_unix_config },
221};
223
224/** Parse "transport" and then set the subconfig
225 *
226 */
227static int client_transport_parse(TALLOC_CTX *ctx, void *out, void *parent, CONF_ITEM *ci, conf_parser_t const *rule)
228{
230
231 /*
232 * Unconnected UDP sockets can only take src_ipaddr, but not port, and not dst_ipaddr.
233 */
235 char const *name = cf_pair_value(cf_item_to_pair(ci));
237
238 if (strcmp(name, "udp") != 0) {
239 cf_log_err(ci, "Invalid transport for unconnected UDP socket");
240 return -1;
241 }
242
244 cf_log_perr(ci, "Failed updating parse rules");
245 return -1;
246 }
247
248 fd_config->socket_type = SOCK_DGRAM;
249 *(char const **) out = name;
250
251 return 0;
252 }
253
255
256 return common_transport_parse(ctx, out, parent, ci, rule,
258}
259
260/*
261 * Client uses src_ipaddr for our address, and ipaddr for their address.
262 */
264 { FR_CONF_OFFSET("transport", fr_bio_fd_config_t, transport), .func = client_transport_parse },
265
266 { FR_CONF_OFFSET("async", fr_bio_fd_config_t, async), .dflt = "true" },
267
269};
270
271/*
272 * Server configuration
273 *
274 * "ipaddr" is src_ipaddr
275 * There's no "dst_ipaddr" or "src_ipaddr" in the config.
276 *
277 * Files have permissions which can be set.
278 */
279
282 { FR_CONF_OFFSET_TYPE_FLAGS("ipv4addr", FR_TYPE_IPV4_ADDR, 0, fr_bio_fd_config_t, src_ipaddr) },
283 { FR_CONF_OFFSET_TYPE_FLAGS("ipv6addr", FR_TYPE_IPV6_ADDR, 0, fr_bio_fd_config_t, src_ipaddr) },
284
285 { FR_CONF_OFFSET("port", fr_bio_fd_config_t, src_port) },
286
287 { FR_CONF_OFFSET("interface", fr_bio_fd_config_t, interface) },
288
289 { FR_CONF_OFFSET_IS_SET("recv_buff", FR_TYPE_UINT32, 0, fr_bio_fd_config_t, recv_buff) },
290 { FR_CONF_OFFSET_IS_SET("send_buff", FR_TYPE_UINT32, 0, fr_bio_fd_config_t, send_buff) },
291
293};
294
296 { FR_CONF_POINTER("udp", 0, CONF_FLAG_SUBSECTION, NULL), .subcs = (void const *) server_udp_sub_config },
297
299};
300
303 { FR_CONF_OFFSET_TYPE_FLAGS("ipv4addr", FR_TYPE_IPV4_ADDR, 0, fr_bio_fd_config_t, src_ipaddr) },
304 { FR_CONF_OFFSET_TYPE_FLAGS("ipv6addr", FR_TYPE_IPV6_ADDR, 0, fr_bio_fd_config_t, src_ipaddr) },
305
306 { FR_CONF_OFFSET("port", fr_bio_fd_config_t, src_port) },
307
308 { FR_CONF_OFFSET("interface", fr_bio_fd_config_t, interface) },
309
310 { FR_CONF_OFFSET_IS_SET("recv_buff", FR_TYPE_UINT32, 0, fr_bio_fd_config_t, recv_buff) },
311 { FR_CONF_OFFSET_IS_SET("send_buff", FR_TYPE_UINT32, 0, fr_bio_fd_config_t, send_buff) },
312
313 { FR_CONF_OFFSET("delay_tcp_writes", fr_bio_fd_config_t, tcp_delay) },
314
316};
317
319 { FR_CONF_POINTER("tcp", 0, CONF_FLAG_SUBSECTION, NULL), .subcs = (void const *) server_tcp_sub_config },
320
322};
323
326
327 { FR_CONF_OFFSET("permissions", fr_bio_fd_config_t, perm), .dflt = "0600", .func = cf_parse_permissions },
328
329 { FR_CONF_OFFSET("mkdir", fr_bio_fd_config_t, mkdir) },
330
332};
333
335 { FR_CONF_POINTER("file", 0, CONF_FLAG_SUBSECTION, NULL), .subcs = (void const *) server_file_sub_config },
336
338};
339
346
349
350 { FR_CONF_OFFSET("permissions", fr_bio_fd_config_t, perm), .dflt = "0600", .func = cf_parse_permissions },
351
352 { FR_CONF_OFFSET("mode", fr_bio_fd_config_t, flags), .dflt = "read-only", .func = mode_parse },
353
354 { FR_CONF_OFFSET("mkdir", fr_bio_fd_config_t, mkdir) },
355
356 { FR_CONF_POINTER("peercred", 0, CONF_FLAG_SUBSECTION, NULL), .subcs = (void const *) server_peercred_config },
357
359};
360
362 { FR_CONF_POINTER("unix", 0, CONF_FLAG_SUBSECTION, NULL), .subcs = (void const *) server_unix_sub_config },
363
365};
366
367/*
368 * @todo - move this to client/server config in the same struct?
369 */
371 { L("file"), server_file_config },
372 { L("tcp"), server_tcp_config },
373 { L("udp"), server_udp_config },
374 { L("unix"), server_unix_config },
375};
377
378/** Parse "transport" and then set the subconfig
379 *
380 */
381static int server_transport_parse(TALLOC_CTX *ctx, void *out, void *parent, CONF_ITEM *ci, conf_parser_t const *rule)
382{
383 int rcode;
385
386 fd_config->server = true;
387
388 rcode = common_transport_parse(ctx, out, parent, ci, rule,
390 if (rcode < 0) return rcode;
391
392 /*
393 * Automatically set the BIO type, too.
394 */
395 if (fd_config->socket_type == SOCK_DGRAM) {
397 } else {
399 }
400
401 return 0;
402}
403
404/*
405 * Server uses ipaddr for our address, and doesn't use src_ipaddr.
406 */
408 { FR_CONF_OFFSET("transport", fr_bio_fd_config_t, transport), .func = server_transport_parse },
409
410 { FR_CONF_OFFSET("async", fr_bio_fd_config_t, async), .dflt = "true" },
411
413};
#define L(_str)
Helper for initialising arrays of string literals.
Definition build.h:209
#define UNUSED
Definition build.h:317
#define NUM_ELEMENTS(_t)
Definition build.h:339
int cf_parse_gid(TALLOC_CTX *ctx, void *out, UNUSED void *parent, CONF_ITEM *ci, UNUSED conf_parser_t const *rule)
Generic function for resolving GID strings to uid_t values.
Definition cf_parse.c:1661
int cf_parse_permissions(UNUSED TALLOC_CTX *ctx, void *out, UNUSED void *parent, CONF_ITEM *ci, UNUSED conf_parser_t const *rule)
Generic function for resolving permissions to a mode-t.
Definition cf_parse.c:1676
int cf_parse_uid(TALLOC_CTX *ctx, void *out, UNUSED void *parent, CONF_ITEM *ci, UNUSED conf_parser_t const *rule)
Generic function for resolving UID strings to uid_t values.
Definition cf_parse.c:1646
#define CONF_PARSER_TERMINATOR
Definition cf_parse.h:658
cf_parse_t func
Override default parsing behaviour for the specified type with a custom parsing function.
Definition cf_parse.h:612
#define FR_CONF_OFFSET(_name, _struct, _field)
conf_parser_t which parses a single CONF_PAIR, writing the result to a field in a struct
Definition cf_parse.h:284
#define cf_section_rules_push(_cs, _rule)
Definition cf_parse.h:690
#define FR_CONF_POINTER(_name, _type, _flags, _res_p)
conf_parser_t which parses a single CONF_PAIR producing a single global result
Definition cf_parse.h:339
#define FR_CONF_OFFSET_IS_SET(_name, _type, _flags, _struct, _field)
conf_parser_t which parses a single CONF_PAIR, writing the result to a field in a struct,...
Definition cf_parse.h:298
#define FR_CONF_OFFSET_FLAGS(_name, _flags, _struct, _field)
conf_parser_t which parses a single CONF_PAIR, writing the result to a field in a struct
Definition cf_parse.h:272
@ CONF_FLAG_REQUIRED
Error out if no matching CONF_PAIR is found, and no dflt value is set.
Definition cf_parse.h:434
@ CONF_FLAG_SUBSECTION
Instead of putting the information into a configuration structure, the configuration file routines MA...
Definition cf_parse.h:428
#define FR_CONF_OFFSET_TYPE_FLAGS(_name, _type, _flags, _struct, _field)
conf_parser_t which parses a single CONF_PAIR, writing the result to a field in a struct
Definition cf_parse.h:241
Defines a CONF_PAIR to C data type mapping.
Definition cf_parse.h:595
Common header for all CONF_* types.
Definition cf_priv.h:49
A section grouping multiple CONF_PAIR.
Definition cf_priv.h:101
CONF_SECTION * cf_section_find(CONF_SECTION const *cs, char const *name1, char const *name2)
Find a CONF_SECTION with name1 and optionally name2.
Definition cf_util.c:1028
CONF_SECTION * cf_item_to_section(CONF_ITEM const *ci)
Cast a CONF_ITEM to a CONF_SECTION.
Definition cf_util.c:684
CONF_PAIR * cf_item_to_pair(CONF_ITEM const *ci)
Cast a CONF_ITEM to a CONF_PAIR.
Definition cf_util.c:664
char const * cf_pair_value(CONF_PAIR const *pair)
Return the value of a CONF_PAIR.
Definition cf_util.c:1594
#define cf_log_err(_cf, _fmt,...)
Definition cf_util.h:289
#define cf_parent(_cf)
Definition cf_util.h:101
#define cf_log_perr(_cf, _fmt,...)
Definition cf_util.h:296
@ FR_BIO_FD_CONNECTED
connected client sockets (UDP or TCP)
Definition fd.h:68
@ FR_BIO_FD_INVALID
not set
Definition fd.h:64
@ FR_BIO_FD_UNCONNECTED
unconnected UDP / datagram only
Definition fd.h:65
@ FR_BIO_FD_LISTEN
returns new fd in buffer on fr_bio_read() or fr_bio_fd_accept()
Definition fd.h:69
fr_bio_fd_type_t type
accept, connected, unconnected, etc.
Definition fd.h:82
bool server
is this a client or a server?
Definition fd.h:86
int socket_type
SOCK_STREAM or SOCK_DGRAM.
Definition fd.h:84
Configuration for sockets.
Definition fd.h:81
const conf_parser_t fr_bio_fd_server_config[]
Definition fd_config.c:407
static conf_parser_t const client_udp_unconnected_config[]
Definition fd_config.c:150
static conf_parser_t const client_file_config[]
Definition fd_config.c:198
static const conf_parser_t server_unix_sub_config[]
Definition fd_config.c:347
static size_t server_transport_names_len
Definition fd_config.c:376
static conf_parser_t const server_unix_config[]
Definition fd_config.c:361
static const conf_parser_t server_udp_sub_config[]
Definition fd_config.c:280
static int mode_parse(UNUSED TALLOC_CTX *ctx, void *out, UNUSED void *parent, CONF_ITEM *ci, UNUSED conf_parser_t const *rule)
Definition fd_config.c:39
static const conf_parser_t client_udp_sub_config[]
Definition fd_config.c:104
static const conf_parser_t server_file_sub_config[]
Definition fd_config.c:324
static const conf_parser_t client_unix_sub_config[]
Definition fd_config.c:204
static int common_transport_parse(UNUSED TALLOC_CTX *ctx, void *out, void *parent, CONF_ITEM *ci, UNUSED conf_parser_t const *rule, fr_table_ptr_sorted_t const *transport_table, size_t transport_table_len)
Parse "transport" and then set the subconfig.
Definition fd_config.c:58
static const conf_parser_t client_udp_unconnected_sub_config[]
Definition fd_config.c:134
const conf_parser_t fr_bio_fd_client_config[]
Definition fd_config.c:263
static conf_parser_t const client_udp_config[]
Definition fd_config.c:127
static int server_transport_parse(TALLOC_CTX *ctx, void *out, void *parent, CONF_ITEM *ci, conf_parser_t const *rule)
Parse "transport" and then set the subconfig.
Definition fd_config.c:381
static fr_table_ptr_sorted_t client_transport_names[]
Definition fd_config.c:216
static size_t client_transport_names_len
Definition fd_config.c:222
static const conf_parser_t client_file_sub_config[]
Definition fd_config.c:188
static int client_transport_parse(TALLOC_CTX *ctx, void *out, void *parent, CONF_ITEM *ci, conf_parser_t const *rule)
Parse "transport" and then set the subconfig.
Definition fd_config.c:227
static conf_parser_t const client_unix_config[]
Definition fd_config.c:210
static const conf_parser_t client_tcp_sub_config[]
Definition fd_config.c:157
static conf_parser_t const server_file_config[]
Definition fd_config.c:334
static conf_parser_t const server_udp_config[]
Definition fd_config.c:295
static const conf_parser_t server_peercred_config[]
Definition fd_config.c:340
static const conf_parser_t server_tcp_sub_config[]
Definition fd_config.c:301
static size_t mode_names_len
Definition fd_config.c:37
static fr_table_num_sorted_t mode_names[]
Definition fd_config.c:31
static conf_parser_t const server_tcp_config[]
Definition fd_config.c:318
static fr_table_ptr_sorted_t server_transport_names[]
Definition fd_config.c:370
static conf_parser_t const client_tcp_config[]
Definition fd_config.c:182
@ FR_TYPE_IPV4_ADDR
32 Bit IPv4 Address.
@ FR_TYPE_UINT32
32 Bit unsigned integer.
@ FR_TYPE_IPV6_ADDR
128 Bit IPv6 Address.
@ FR_TYPE_COMBO_IP_ADDR
IPv4 or IPv6 address depending on length.
static fr_bio_fd_config_t fd_config
static char const * name
#define fr_table_value_by_str(_table, _name, _def)
Convert a string to a value using a sorted or ordered table.
Definition table.h:653
An element in a lexicographically sorted array of name to num mappings.
Definition table.h:49
An element in a lexicographically sorted array of name to ptr mappings.
Definition table.h:65
static fr_slen_t parent
Definition pair.h:845
static size_t char ** out
Definition value.h:1012