The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
Loading...
Searching...
No Matches
proto_radius_udp.c
Go to the documentation of this file.
1/*
2 * This program 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
5 * (at 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: 3434ad944e5f96b036ca81cc79254eb193d624f4 $
19 * @file proto_radius_udp.c
20 * @brief RADIUS handler for UDP.
21 *
22 * @copyright 2016 The FreeRADIUS server project.
23 * @copyright 2016 Alan DeKok (aland@deployingradius.com)
24 */
25#include <netdb.h>
26#include <freeradius-devel/server/protocol.h>
27#include <freeradius-devel/util/udp.h>
28#include <freeradius-devel/util/trie.h>
29#include <freeradius-devel/radius/radius.h>
30#include <freeradius-devel/io/application.h>
31#include <freeradius-devel/io/listen.h>
32#include <freeradius-devel/io/schedule.h>
33
34#include "proto_radius.h"
35
37
38typedef struct {
39 char const *name; //!< socket name
40 int sockfd;
41
42 fr_io_address_t *connection; //!< for connected sockets.
43
44 fr_stats_t stats; //!< statistics for this socket
45
47
48typedef struct {
49 CONF_SECTION *cs; //!< our configuration
50
51 fr_ipaddr_t ipaddr; //!< IP address to listen on.
52 fr_ipaddr_t src_ipaddr; //!< IP to use for responses, which may be
53 ///< different from the ipaddr we're listening on
54 ///< with L3 load balancing setups.
55 bool src_ipaddr_is_set; //!< Use specified a source ip address.
56
57 char const *interface; //!< Interface to bind to.
58 char const *port_name; //!< Name of the port for getservent().
59
60 uint32_t recv_buff; //!< How big the kernel's receive buffer should be.
61 uint32_t send_buff; //!< How big the kernel's send buffer should be.
62
63 uint32_t max_packet_size; //!< for message ring buffer.
64 uint32_t max_attributes; //!< Limit maximum decodable attributes.
65
66 uint16_t port; //!< Port to listen on.
67
68 bool recv_buff_is_set; //!< Whether we were provided with a recv_buff
69 bool send_buff_is_set; //!< Whether we were provided with a send_buff
70 bool dynamic_clients; //!< whether we have dynamic clients
71
72 fr_client_list_t *clients; //!< local clients
73
74 fr_trie_t *trie; //!< for parsed networks
75 fr_ipaddr_t *allow; //!< allowed networks for dynamic clients
76 fr_ipaddr_t *deny; //!< denied networks for dynamic clients
77
78 bool read_hexdump; //!< Do we debug hexdump read packets.
79 bool write_hexdump; //!< Do we debug hexdump write packets.
81
82
89
90
95
96 { FR_CONF_OFFSET_IS_SET("src_ipaddr", FR_TYPE_COMBO_IP_ADDR, 0, proto_radius_udp_t, src_ipaddr) },
97
98 { FR_CONF_OFFSET("interface", proto_radius_udp_t, interface) },
99 { FR_CONF_OFFSET("port_name", proto_radius_udp_t, port_name) },
100
101 { FR_CONF_OFFSET("port", proto_radius_udp_t, port) },
102
103 { FR_CONF_OFFSET_IS_SET("recv_buff", FR_TYPE_UINT32, 0, proto_radius_udp_t, recv_buff) },
104 { FR_CONF_OFFSET_IS_SET("send_buff", FR_TYPE_UINT32, 0, proto_radius_udp_t, send_buff) },
105
106 { FR_CONF_OFFSET("dynamic_clients", proto_radius_udp_t, dynamic_clients) } ,
107 { FR_CONF_POINTER("networks", 0, CONF_FLAG_SUBSECTION, NULL), .subcs = (void const *) networks_config },
108
109 { FR_CONF_OFFSET("max_packet_size", proto_radius_udp_t, max_packet_size), .dflt = "4096" } ,
110 { FR_CONF_OFFSET("max_attributes", proto_radius_udp_t, max_attributes), .dflt = STRINGIFY(RADIUS_MAX_ATTRIBUTES) } ,
111
112 { FR_CONF_OFFSET("read_hexdump", proto_radius_udp_t, read_hexdump) },
113 { FR_CONF_OFFSET("write_hexdump", proto_radius_udp_t, write_hexdump) },
114
116};
117
118
119static ssize_t mod_read(fr_listen_t *li, void **packet_ctx, fr_time_t *recv_time_p, uint8_t *buffer, size_t buffer_len,
120 size_t *leftover)
121{
123 proto_radius_udp_thread_t *thread = talloc_get_type_abort(li->thread_instance, proto_radius_udp_thread_t);
124 fr_io_address_t *address, **address_p;
125
126 int flags;
127 ssize_t data_size;
128 size_t packet_len;
130
131 *leftover = 0; /* always for UDP */
132
133 /*
134 * Where the addresses should go. This is a special case
135 * for proto_radius.
136 */
137 address_p = (fr_io_address_t **)packet_ctx;
138 address = *address_p;
139
140 /*
141 * Tell udp_recv if we're connected or not.
142 */
143 flags = UDP_FLAGS_CONNECTED * (thread->connection != NULL);
144
145 data_size = udp_recv(thread->sockfd, flags, &address->socket, buffer, buffer_len, recv_time_p);
146 if (data_size < 0) {
148 "%s", fr_strerror());
149 return data_size;
150 }
151
152 if (!data_size) {
154 "Received no data");
155 return 0;
156 }
157
158 packet_len = data_size;
159
160 if (data_size < 20) {
162 "Received packet length %zu", packet_len);
164 return 0;
165 }
166
167 if (packet_len > inst->max_packet_size) {
169 "Received packet length %zu");
171 return 0;
172 }
173
174 if ((buffer[0] == 0) || (buffer[0] >= FR_RADIUS_CODE_MAX)) {
176 "Received packet code %u", buffer[0]);
177 thread->stats.total_unknown_types++;
178 return 0;
179 }
180
181 /*
182 * If it's not well-formed, discard it.
183 */
184 if (!fr_radius_ok(buffer, &packet_len, inst->max_attributes, false, &reason)) {
185 proto_radius_log(li, thread->name, reason, &address->socket,
186 "Received invalid packet");
188 return 0;
189 }
190
191 /*
192 * proto_radius sets the priority
193 */
194 thread->stats.total_requests++;
195
196 /*
197 * Print out what we received.
198 */
199 DEBUG2("proto_radius_udp - Received %s ID %d length %d %s",
201 (int) packet_len, thread->name);
202
203 return packet_len;
204}
205
206static ssize_t mod_write(fr_listen_t *li, void *packet_ctx, UNUSED fr_time_t request_time,
207 uint8_t *buffer, size_t buffer_len, UNUSED size_t written)
208{
210 proto_radius_udp_thread_t *thread = talloc_get_type_abort(li->thread_instance, proto_radius_udp_thread_t);
211
212 fr_io_track_t *track = talloc_get_type_abort(packet_ctx, fr_io_track_t);
213 fr_socket_t socket;
214
215 int flags;
216 ssize_t data_size;
217
218 /*
219 * @todo - share a stats interface with the parent? or
220 * put the stats in the listener, so that proto_radius
221 * can update them, too.. <sigh>
222 */
223 thread->stats.total_responses++;
224
225 flags = UDP_FLAGS_CONNECTED * (thread->connection != NULL);
226
227 /*
228 * Swap src/dst address so we send the response to
229 * the client, not ourselves.
230 */
231 fr_socket_addr_swap(&socket, &track->address->socket);
232
233 /*
234 * Override the source ip address in the outbound
235 * packet.
236 */
237 if (inst->src_ipaddr_is_set) socket.inet.src_ipaddr = inst->src_ipaddr;
238
239 /*
240 * This handles the race condition where we get a DUP,
241 * but the original packet replies before we're run.
242 * i.e. this packet isn't marked DUP, so we have to
243 * discover it's a dup later...
244 *
245 * As such, if there's already a reply, then we ignore
246 * the encoded reply (which is probably going to be a
247 * NAK), and instead reply with the cached reply.
248 */
249 if (track->reply_len) {
250 if (track->reply_len >= 20) {
251 char *packet;
252
253 memcpy(&packet, &track->reply, sizeof(packet)); /* const issues */
254
255 return udp_send(&socket, flags, packet, track->reply_len);
256 }
257
258 return buffer_len;
259 }
260
261 /*
262 * We only write RADIUS packets.
263 */
264 fr_assert(buffer_len >= 20);
265
266 /*
267 * Only write replies if they're RADIUS packets.
268 * sometimes we want to NOT send a reply...
269 */
270 data_size = udp_send(&socket, flags, buffer, buffer_len);
271
272 /*
273 * This socket is dead. That's an error...
274 */
275 if (data_size <= 0) return data_size;
276
277 /*
278 * Root through the reply to determine any
279 * connection-level negotiation data.
280 */
281 if (track->packet[0] == FR_RADIUS_CODE_STATUS_SERVER) {
282// status_check_reply(inst, buffer, buffer_len);
283 }
284
285 return data_size;
286}
287
288
290{
291 proto_radius_udp_thread_t *thread = talloc_get_type_abort(li->thread_instance, proto_radius_udp_thread_t);
292
293 thread->connection = connection;
294 return 0;
295}
296
297
298static void mod_network_get(int *ipproto, bool *dynamic_clients, fr_trie_t const **trie, void *instance)
299{
300 proto_radius_udp_t *inst = talloc_get_type_abort(instance, proto_radius_udp_t);
301
302 *ipproto = IPPROTO_UDP;
303 *dynamic_clients = inst->dynamic_clients;
304 *trie = inst->trie;
305}
306
307/** Open a UDP listener for RADIUS
308 *
309 */
310static int mod_open(fr_listen_t *li)
311{
313 proto_radius_udp_thread_t *thread = talloc_get_type_abort(li->thread_instance, proto_radius_udp_thread_t);
314
315 int sockfd;
316 fr_ipaddr_t ipaddr = inst->ipaddr;
317 uint16_t port = inst->port;
318
319 li->fd = sockfd = fr_socket_server_udp(&inst->ipaddr, &port, inst->port_name, true);
320 if (sockfd < 0) {
321 PERROR("Failed opening UDP socket");
322 error:
323 return -1;
324 }
325
326 li->app_io_addr = fr_socket_addr_alloc_inet_src(li, IPPROTO_UDP, 0, &inst->ipaddr, port);
327
328 /*
329 * Set SO_REUSEPORT before bind, so that all packets can
330 * listen on the same destination IP address.
331 */
332 if (1) {
333 int on = 1;
334
335 if (setsockopt(sockfd, SOL_SOCKET, SO_REUSEPORT, &on, sizeof(on)) < 0) {
336 ERROR("Failed to set socket 'reuseport': %s", fr_syserror(errno));
337 return -1;
338 }
339 }
340
341#ifdef SO_RCVBUF
342 if (inst->recv_buff_is_set) {
343 int opt;
344
345 opt = inst->recv_buff;
346 if (setsockopt(sockfd, SOL_SOCKET, SO_RCVBUF, &opt, sizeof(int)) < 0) {
347 WARN("Failed setting 'recv_buf': %s", fr_syserror(errno));
348 }
349 }
350#endif
351
352#ifdef SO_SNDBUF
353 if (inst->send_buff_is_set) {
354 int opt;
355
356 opt = inst->send_buff;
357 if (setsockopt(sockfd, SOL_SOCKET, SO_SNDBUF, &opt, sizeof(int)) < 0) {
358 WARN("Failed setting 'send_buf': %s", fr_syserror(errno));
359 }
360 }
361#endif
362
363 if (fr_socket_bind(sockfd, inst->interface, &ipaddr, &port) < 0) {
364 close(sockfd);
365 PERROR("Failed binding socket");
366 goto error;
367 }
368
369 thread->sockfd = sockfd;
370
371 fr_assert((cf_parent(inst->cs) != NULL) && (cf_parent(cf_parent(inst->cs)) != NULL)); /* listen { ... } */
372
374 NULL, 0,
375 &inst->ipaddr, inst->port,
376 inst->interface);
377
378 return 0;
379}
380
381/** Set the file descriptor for this socket.
382 *
383 */
384static int mod_fd_set(fr_listen_t *li, int fd)
385{
387 proto_radius_udp_thread_t *thread = talloc_get_type_abort(li->thread_instance, proto_radius_udp_thread_t);
388
389 thread->sockfd = fd;
390
392 &thread->connection->socket.inet.src_ipaddr, thread->connection->socket.inet.src_port,
393 &inst->ipaddr, inst->port,
394 inst->interface);
395
396 return 0;
397}
398
399static void *mod_track_create(UNUSED void const *instance, UNUSED void *thread_instance, UNUSED fr_client_t *client,
400 fr_io_track_t *track, uint8_t const *packet, UNUSED size_t packet_len)
401{
402 return talloc_memdup(track, packet, RADIUS_HEADER_LENGTH);
403}
404
405static int mod_track_compare(UNUSED void const *instance, UNUSED void *thread_instance, UNUSED fr_client_t *client,
406 void const *one, void const *two)
407{
408 int ret;
409 uint8_t const *a = one;
410 uint8_t const *b = two;
411
412 /*
413 * The tree is ordered by IDs, which are (hopefully)
414 * pseudo-randomly distributed.
415 */
416 ret = (a[1] < b[1]) - (a[1] > b[1]);
417 if (ret != 0) return ret;
418
419 /*
420 * Then ordered by code, which is usually the same.
421 */
422 return (a[0] < b[0]) - (a[0] > b[0]);
423}
424
425
426static char const *mod_name(fr_listen_t *li)
427{
428 proto_radius_udp_thread_t *thread = talloc_get_type_abort(li->thread_instance, proto_radius_udp_thread_t);
429
430 return thread->name;
431}
432
433static void mod_hexdump_set(fr_listen_t *li, void *data)
434{
435 proto_radius_udp_t *inst = talloc_get_type_abort(data, proto_radius_udp_t);
436 li->read_hexdump = inst->read_hexdump;
437 li->write_hexdump = inst->write_hexdump;
438}
439
440static int mod_instantiate(module_inst_ctx_t const *mctx)
441{
442 proto_radius_udp_t *inst = talloc_get_type_abort(mctx->mi->data, proto_radius_udp_t);
443 CONF_SECTION *conf = mctx->mi->conf;
444 size_t num;
445 CONF_ITEM *ci;
446 CONF_SECTION *server_cs;
447
448 inst->cs = conf;
449
450 /*
451 * Complain if no "ipaddr" is set.
452 */
453 if (inst->ipaddr.af == AF_UNSPEC) {
454 cf_log_err(conf, "No 'ipaddr' was specified in the 'udp' section");
455 return -1;
456 }
457
458 if (inst->recv_buff_is_set) {
459 FR_INTEGER_BOUND_CHECK("recv_buff", inst->recv_buff, >=, 32);
460 FR_INTEGER_BOUND_CHECK("recv_buff", inst->recv_buff, <=, INT_MAX);
461 }
462
463 if (inst->send_buff_is_set) {
464 FR_INTEGER_BOUND_CHECK("send_buff", inst->send_buff, >=, inst->max_packet_size);
465 FR_INTEGER_BOUND_CHECK("send_buff", inst->send_buff, <=, (1 << 30));
466 }
467
468 FR_INTEGER_BOUND_CHECK("max_packet_size", inst->max_packet_size, >=, 20);
469 FR_INTEGER_BOUND_CHECK("max_packet_size", inst->max_packet_size, <=, 65536);
470
471 if (!inst->port) {
472 struct servent *s;
473
474 if (!inst->port_name) {
475 cf_log_err(conf, "No 'port' was specified in the 'udp' section");
476 return -1;
477 }
478
479 s = getservbyname(inst->port_name, "udp");
480 if (!s) {
481 cf_log_err(conf, "Unknown value for 'port_name = %s", inst->port_name);
482 return -1;
483 }
484
485 inst->port = ntohl(s->s_port);
486 }
487
488 /*
489 * Parse and create the trie for dynamic clients, even if
490 * there's no dynamic clients.
491 */
492 num = talloc_array_length(inst->allow);
493 if (!num) {
494 if (inst->dynamic_clients) {
495 cf_log_err(conf, "The 'allow' subsection MUST contain at least one 'network' entry when 'dynamic_clients = true'.");
496 return -1;
497 }
498 } else {
499 inst->trie = fr_master_io_network(inst, inst->ipaddr.af, inst->allow, inst->deny);
500 if (!inst->trie) {
501 cf_log_perr(conf, "Failed creating list of networks");
502 return -1;
503 }
504 }
505
506 ci = cf_section_to_item(mctx->mi->parent->conf); /* listen { ... } */
507 fr_assert(ci != NULL);
508 ci = cf_parent(ci);
509 fr_assert(ci != NULL);
510
511 server_cs = cf_item_to_section(ci);
512
513 /*
514 * Look up local clients, if they exist.
515 *
516 * @todo - ensure that we only parse clients which are
517 * for IPPROTO_UDP, and require a "secret".
518 */
519 if (cf_section_find_next(server_cs, NULL, "client", CF_IDENT_ANY)) {
520 inst->clients = client_list_parse_section(server_cs, IPPROTO_UDP, false);
521 if (!inst->clients) {
522 cf_log_err(conf, "Failed creating local clients");
523 return -1;
524 }
525 }
526
527 return 0;
528}
529
531{
533 fr_client_t *client;
534
535 /*
536 * Prefer local clients.
537 */
538 if (inst->clients) {
539 client = client_find(inst->clients, ipaddr, ipproto);
540 if (client) return client;
541 }
542
543 return client_find(NULL, ipaddr, ipproto);
544}
545
547 .common = {
548 .magic = MODULE_MAGIC_INIT,
549 .name = "radius_udp",
551 .inst_size = sizeof(proto_radius_udp_t),
552 .thread_inst_size = sizeof(proto_radius_udp_thread_t),
553 .instantiate = mod_instantiate
554 },
555 .default_message_size = 4096,
556 .track_duplicates = true,
557
558 .open = mod_open,
559 .read = mod_read,
560 .write = mod_write,
561 .fd_set = mod_fd_set,
562 .track_create = mod_track_create,
563 .track_compare = mod_track_compare,
564 .connection_set = mod_connection_set,
565 .network_get = mod_network_get,
566 .client_find = mod_client_find,
567 .get_name = mod_name,
568 .hexdump_set = mod_hexdump_set,
569};
static int const char char buffer[256]
Definition acutest.h:578
char const * fr_app_io_socket_name(TALLOC_CTX *ctx, fr_app_io_t const *app_io, fr_ipaddr_t const *src_ipaddr, int src_port, fr_ipaddr_t const *dst_ipaddr, int dst_port, char const *interface)
Definition app_io.c:32
module_t common
Common fields to all loadable modules.
Definition app_io.h:34
Public structure describing an I/O path for a protocol.
Definition app_io.h:33
#define STRINGIFY(x)
Definition build.h:197
#define UNUSED
Definition build.h:317
#define CONF_PARSER_TERMINATOR
Definition cf_parse.h:660
#define FR_INTEGER_BOUND_CHECK(_name, _var, _op, _bound)
Definition cf_parse.h:520
#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:283
#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:337
#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:297
@ CONF_FLAG_MULTI
CONF_PAIR can have multiple copies.
Definition cf_parse.h:449
@ CONF_FLAG_SUBSECTION
Instead of putting the information into a configuration structure, the configuration file routines MA...
Definition cf_parse.h:426
#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:597
Common header for all CONF_* types.
Definition cf_priv.h:49
A section grouping multiple CONF_PAIR.
Definition cf_priv.h:101
CONF_ITEM * cf_section_to_item(CONF_SECTION const *cs)
Cast a CONF_SECTION to a CONF_ITEM.
Definition cf_util.c:737
CONF_SECTION * cf_item_to_section(CONF_ITEM const *ci)
Cast a CONF_ITEM to a CONF_SECTION.
Definition cf_util.c:683
CONF_SECTION * cf_section_find_next(CONF_SECTION const *cs, CONF_SECTION const *prev, char const *name1, char const *name2)
Return the next matching section.
Definition cf_util.c:1048
#define cf_log_err(_cf, _fmt,...)
Definition cf_util.h:288
#define cf_parent(_cf)
Definition cf_util.h:101
#define cf_log_perr(_cf, _fmt,...)
Definition cf_util.h:295
#define CF_IDENT_ANY
Definition cf_util.h:78
@ FR_RADIUS_CODE_MAX
Maximum possible protocol code.
Definition defs.h:53
@ FR_RADIUS_CODE_STATUS_SERVER
RFC2865/RFC5997 - Status Server (request)
Definition defs.h:44
#define ERROR(fmt,...)
Definition dhcpclient.c:41
static int sockfd
Definition dhcpclient.c:56
#define MODULE_MAGIC_INIT
Stop people using different module/library/server versions together.
Definition dl_module.h:63
IPv4/6 prefix.
fr_socket_t socket
src/dst ip and port.
Definition base.h:336
bool read_hexdump
Do we debug hexdump packets as they're read.
Definition listen.h:52
fr_socket_t * app_io_addr
for tracking duplicate sockets
Definition listen.h:36
bool write_hexdump
Do we debug hexdump packets as they're written.
Definition listen.h:53
void const * app_io_instance
I/O path configuration context.
Definition listen.h:33
void * thread_instance
thread / socket context
Definition listen.h:34
int fd
file descriptor for this socket - set by open
Definition listen.h:28
Describes a host allowed to send packets to the server.
Definition client.h:80
#define PERROR(_fmt,...)
Definition log.h:228
uint64_t total_responses
Definition stats.h:38
uint64_t total_unknown_types
Definition stats.h:46
uint64_t total_requests
Definition stats.h:35
uint64_t total_malformed_requests
Definition stats.h:42
ssize_t udp_recv(int sockfd, int flags, fr_socket_t *socket_out, void *data, size_t data_len, fr_time_t *when)
Read a UDP packet.
Definition udp.c:144
int udp_send(fr_socket_t const *sock, int flags, void *data, size_t data_len)
Send a packet via a UDP socket.
Definition udp.c:42
fr_trie_t * fr_master_io_network(TALLOC_CTX *ctx, int af, fr_ipaddr_t *allow, fr_ipaddr_t *deny)
Create a trie from arrays of allow / deny IP addresses.
Definition master.c:2952
uint8_t * reply
reply packet (if any)
Definition master.h:46
fr_io_address_t const * address
of this packet.. shared between multiple packets
Definition master.h:54
uint8_t * packet
really a tracking structure, not a packet
Definition master.h:56
size_t reply_len
length of reply, or 1 for "do not reply"
Definition master.h:47
bool fr_radius_ok(uint8_t const *packet, size_t *packet_len_p, uint32_t max_attributes, bool require_message_authenticator, decode_fail_t *reason)
unsigned short uint16_t
@ FR_TYPE_IPV4_ADDR
32 Bit IPv4 Address.
@ FR_TYPE_COMBO_IP_PREFIX
IPv4 or IPv6 address prefix depending on length.
@ 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.
unsigned int uint32_t
long int ssize_t
unsigned char uint8_t
module_instance_t * mi
Instance of the module being instantiated.
Definition module_ctx.h:51
Temporary structure to hold arguments for instantiation calls.
Definition module_ctx.h:50
#define RADIUS_HEADER_LENGTH
Definition net.h:80
void proto_radius_log(fr_listen_t *li, char const *name, fr_radius_decode_fail_t reason, fr_socket_t const *sock, char const *fmt,...)
Log a message in a canonical format.
char const * name
socket name
fr_ipaddr_t * allow
allowed networks for dynamic clients
fr_ipaddr_t * deny
denied networks for dynamic clients
static ssize_t mod_read(fr_listen_t *li, void **packet_ctx, fr_time_t *recv_time_p, uint8_t *buffer, size_t buffer_len, size_t *leftover)
fr_io_address_t * connection
for connected sockets.
bool read_hexdump
Do we debug hexdump read packets.
bool recv_buff_is_set
Whether we were provided with a recv_buff.
static void * mod_track_create(UNUSED void const *instance, UNUSED void *thread_instance, UNUSED fr_client_t *client, fr_io_track_t *track, uint8_t const *packet, UNUSED size_t packet_len)
char const * port_name
Name of the port for getservent().
fr_client_list_t * clients
local clients
static void mod_hexdump_set(fr_listen_t *li, void *data)
bool src_ipaddr_is_set
Use specified a source ip address.
uint32_t send_buff
How big the kernel's send buffer should be.
fr_ipaddr_t ipaddr
IP address to listen on.
CONF_SECTION * cs
our configuration
bool send_buff_is_set
Whether we were provided with a send_buff.
static fr_client_t * mod_client_find(fr_listen_t *li, fr_ipaddr_t const *ipaddr, int ipproto)
static int mod_open(fr_listen_t *li)
Open a UDP listener for RADIUS.
static const conf_parser_t udp_listen_config[]
static void mod_network_get(int *ipproto, bool *dynamic_clients, fr_trie_t const **trie, void *instance)
static const conf_parser_t networks_config[]
bool dynamic_clients
whether we have dynamic clients
uint16_t port
Port to listen on.
uint32_t recv_buff
How big the kernel's receive buffer should be.
static int mod_track_compare(UNUSED void const *instance, UNUSED void *thread_instance, UNUSED fr_client_t *client, void const *one, void const *two)
fr_app_io_t proto_radius_udp
uint32_t max_attributes
Limit maximum decodable attributes.
static int mod_connection_set(fr_listen_t *li, fr_io_address_t *connection)
uint32_t max_packet_size
for message ring buffer.
fr_trie_t * trie
for parsed networks
bool write_hexdump
Do we debug hexdump write packets.
static int mod_fd_set(fr_listen_t *li, int fd)
Set the file descriptor for this socket.
static char const * mod_name(fr_listen_t *li)
static ssize_t mod_write(fr_listen_t *li, void *packet_ctx, UNUSED fr_time_t request_time, uint8_t *buffer, size_t buffer_len, UNUSED size_t written)
static int mod_instantiate(module_inst_ctx_t const *mctx)
char const * interface
Interface to bind to.
fr_ipaddr_t src_ipaddr
IP to use for responses, which may be different from the ipaddr we're listening on with L3 load balan...
fr_stats_t stats
statistics for this socket
char const * fr_radius_packet_name[FR_RADIUS_CODE_MAX]
Definition base.c:116
#define fr_assert(_expr)
Definition rad_assert.h:38
static int ipproto
#define DEBUG2(fmt,...)
Definition radclient.h:43
#define WARN(fmt,...)
Definition radclient.h:47
fr_radius_decode_fail_t
Failure reasons.
Definition radius.h:90
@ FR_RADIUS_FAIL_IO_ERROR
Definition radius.h:115
@ FR_RADIUS_FAIL_UNKNOWN_PACKET_CODE
Definition radius.h:96
@ FR_RADIUS_FAIL_MIN_LENGTH_PACKET
Definition radius.h:92
#define RADIUS_MAX_ATTRIBUTES
Definition radius.h:40
static rs_t * conf
Definition radsniff.c:53
CONF_SECTION * conf
Module's instance configuration.
Definition module.h:349
void * data
Module's instance data.
Definition module.h:291
module_instance_t const * parent
Parent module's instance (if any).
Definition module.h:357
conf_parser_t const * config
How to convert a CONF_SECTION to a module instance.
Definition module.h:206
int fr_socket_server_udp(fr_ipaddr_t const *src_ipaddr, uint16_t *src_port, char const *port_name, bool async)
Open an IPv4/IPv6 unconnected UDP socket.
Definition socket.c:873
int fr_socket_bind(int sockfd, char const *ifname, fr_ipaddr_t *src_ipaddr, uint16_t *src_port)
Bind a UDP/TCP v4/v6 socket to a given ipaddr src port, and interface.
Definition socket.c:227
fr_client_t * client_find(fr_client_list_t const *clients, fr_ipaddr_t const *ipaddr, int proto)
Definition client.c:373
fr_client_list_t * client_list_parse_section(CONF_SECTION *section, int proto, TLS_UNUSED bool tls_required)
Definition client.c:478
Group of clients.
Definition client.c:50
eap_aka_sim_process_conf_t * inst
char const * fr_syserror(int num)
Guaranteed to be thread-safe version of strerror.
Definition syserror.c:243
#define talloc_get_type_abort_const
Definition talloc.h:245
"server local" time.
Definition time.h:69
#define UDP_FLAGS_CONNECTED
Definition udp.h:38
static fr_socket_t * fr_socket_addr_alloc_inet_src(TALLOC_CTX *ctx, int proto, int ifindex, fr_ipaddr_t const *ipaddr, int port)
A variant of fr_socket_addr_init_inet_src will also allocates a fr_socket_t.
Definition socket.h:244
static void fr_socket_addr_swap(fr_socket_t *dst, fr_socket_t const *src)
Swap src/dst information of a fr_socket_t.
Definition socket.h:121
Holds information necessary for binding or connecting to a socket.
Definition socket.h:63
char const * fr_strerror(void)
Get the last library error.
Definition strerror.c:553
static fr_slen_t data
Definition value.h:1334