blob: 7357d43a25205edaa84174bb3699804322ac7d77 [file] [log] [blame]
Jeremy Kerr3d36ee22019-05-30 11:15:37 +08001/* SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later */
2
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +08003#define _GNU_SOURCE
4
Matt Johnston3ef47782024-12-11 15:19:06 +08005#ifdef HAVE_CONFIG_H
Andrew Jefferyd4103f82021-06-16 14:39:36 +09306#include "config.h"
Matt Johnston3ef47782024-12-11 15:19:06 +08007#endif
Andrew Jeffery5ab78252022-02-17 21:04:59 +10308
9#define SD_LISTEN_FDS_START 3
10
11#include "compiler.h"
12#include "libmctp.h"
13#include "libmctp-serial.h"
14#include "libmctp-astlpc.h"
Andrew Jefferycad47302021-08-20 21:37:57 +093015#include "utils/mctp-capture.h"
Andrew Jefferyd4103f82021-06-16 14:39:36 +093016
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +080017#include <assert.h>
18#include <err.h>
Andrew Jeffery6896d412020-03-11 09:25:32 +103019#include <errno.h>
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +080020#include <getopt.h>
Andrew Jefferyb93b6112020-06-05 14:13:44 +093021#include <limits.h>
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +080022#include <poll.h>
Andrew Jeffery490e3872021-08-24 21:11:30 +093023#include <signal.h>
Andrew Jeffery04b81fc2020-02-05 13:07:29 +103024#include <stdbool.h>
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +080025#include <stdio.h>
26#include <stdlib.h>
27#include <string.h>
28#include <unistd.h>
29
Andrew Jeffery490e3872021-08-24 21:11:30 +093030#include <sys/signalfd.h>
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +080031#include <sys/socket.h>
32#include <sys/un.h>
33
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +080034#define ARRAY_SIZE(a) (sizeof(a) / sizeof(a[0]))
35
Andrew Jefferyd4103f82021-06-16 14:39:36 +093036#if HAVE_SYSTEMD_SD_DAEMON_H
37#include <systemd/sd-daemon.h>
38#else
39static inline int sd_listen_fds(int i __unused)
40{
41 return -1;
42}
43#endif
44
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +080045static const mctp_eid_t local_eid_default = 8;
46static char sockname[] = "\0mctp-mux";
47
48struct binding {
Andrew Jeffery8f53d632022-07-25 20:57:14 +093049 const char *name;
50 int (*init)(struct mctp *mctp, struct binding *binding, mctp_eid_t eid,
51 int n_params, char *const *params);
52 void (*destroy)(struct mctp *mctp, struct binding *binding);
53 int (*init_pollfd)(struct binding *binding, struct pollfd *pollfd);
54 int (*process)(struct binding *binding);
55 void *data;
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +080056};
57
58struct client {
Patrick Williamsa721c2d2022-12-04 14:30:26 -060059 bool active;
60 int sock;
61 uint8_t type;
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +080062};
63
64struct ctx {
Patrick Williamsa721c2d2022-12-04 14:30:26 -060065 struct mctp *mctp;
66 struct binding *binding;
67 bool verbose;
68 int local_eid;
Matt Johnston3ef47782024-12-11 15:19:06 +080069 uint8_t *buf;
Patrick Williamsa721c2d2022-12-04 14:30:26 -060070 size_t buf_size;
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +080071
Patrick Williamsa721c2d2022-12-04 14:30:26 -060072 int sock;
73 struct pollfd *pollfds;
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +080074
Patrick Williamsa721c2d2022-12-04 14:30:26 -060075 struct client *clients;
76 int n_clients;
Andrew Jefferycad47302021-08-20 21:37:57 +093077
78 struct {
79 struct capture binding;
80 struct capture socket;
81 } pcap;
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +080082};
83
84static void tx_message(struct ctx *ctx, mctp_eid_t eid, void *msg, size_t len)
85{
Andrew Jeffery06735052021-01-27 23:50:55 +103086 int rc;
87
Sumanth Bhatf39c3852022-01-10 17:04:10 +053088 rc = mctp_message_tx(ctx->mctp, eid, MCTP_MESSAGE_TO_SRC, 0, msg, len);
Andrew Jeffery06735052021-01-27 23:50:55 +103089 if (rc)
90 warnx("Failed to send message: %d", rc);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +080091}
92
93static void client_remove_inactive(struct ctx *ctx)
94{
95 int i;
96
97 for (i = 0; i < ctx->n_clients; i++) {
98 struct client *client = &ctx->clients[i];
99 if (client->active)
100 continue;
101 close(client->sock);
102
103 ctx->n_clients--;
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600104 memmove(&ctx->clients[i], &ctx->clients[i + 1],
105 (ctx->n_clients - i) * sizeof(*ctx->clients));
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800106 ctx->clients = realloc(ctx->clients,
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600107 ctx->n_clients * sizeof(*ctx->clients));
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800108 }
109}
110
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600111static void rx_message(uint8_t eid, bool tag_owner __unused,
112 uint8_t msg_tag __unused, void *data, void *msg,
113 size_t len)
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800114{
115 struct ctx *ctx = data;
116 struct iovec iov[2];
117 struct msghdr msghdr;
118 bool removed;
119 uint8_t type;
120 int i, rc;
121
122 if (len < 2)
123 return;
124
125 type = *(uint8_t *)msg;
126
127 if (ctx->verbose)
128 fprintf(stderr, "MCTP message received: len %zd, type %d\n",
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600129 len, type);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800130
131 memset(&msghdr, 0, sizeof(msghdr));
132 msghdr.msg_iov = iov;
133 msghdr.msg_iovlen = 2;
134 iov[0].iov_base = &eid;
135 iov[0].iov_len = 1;
136 iov[1].iov_base = msg;
137 iov[1].iov_len = len;
138
139 for (i = 0; i < ctx->n_clients; i++) {
140 struct client *client = &ctx->clients[i];
141
142 if (client->type != type)
143 continue;
144
145 if (ctx->verbose)
146 fprintf(stderr, " forwarding to client %d\n", i);
147
148 rc = sendmsg(client->sock, &msghdr, 0);
149 if (rc != (ssize_t)(len + 1)) {
150 client->active = false;
151 removed = true;
152 }
153 }
154
155 if (removed)
156 client_remove_inactive(ctx);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800157}
158
Jeremy Kerr34b9b3d2019-11-27 11:27:05 +0800159static int binding_null_init(struct mctp *mctp __unused,
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600160 struct binding *binding __unused,
161 mctp_eid_t eid __unused, int n_params,
162 char *const *params __unused)
Jeremy Kerr34b9b3d2019-11-27 11:27:05 +0800163{
164 if (n_params != 0) {
165 warnx("null binding doesn't accept parameters");
166 return -1;
167 }
168 return 0;
169}
170
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800171static int binding_serial_init(struct mctp *mctp, struct binding *binding,
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600172 mctp_eid_t eid, int n_params,
173 char *const *params)
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800174{
175 struct mctp_binding_serial *serial;
176 const char *path;
177 int rc;
178
179 if (n_params != 1) {
180 warnx("serial binding requires device param");
181 return -1;
182 }
183
184 path = params[0];
185
186 serial = mctp_serial_init();
187 assert(serial);
188
189 rc = mctp_serial_open_path(serial, path);
190 if (rc)
191 return -1;
192
Jeremy Kerr3b36d172019-09-04 11:56:09 +0800193 mctp_register_bus(mctp, mctp_binding_serial_core(serial), eid);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800194
195 binding->data = serial;
196
197 return 0;
198}
199
Andrew Jeffery8f53d632022-07-25 20:57:14 +0930200static int binding_serial_init_pollfd(struct binding *binding,
201 struct pollfd *pollfd)
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800202{
Andrew Jeffery8f53d632022-07-25 20:57:14 +0930203 return mctp_serial_init_pollfd(binding->data, pollfd);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800204}
205
206static int binding_serial_process(struct binding *binding)
207{
208 return mctp_serial_read(binding->data);
209}
210
211static int binding_astlpc_init(struct mctp *mctp, struct binding *binding,
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600212 mctp_eid_t eid, int n_params,
213 char *const *params __attribute__((unused)))
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800214{
215 struct mctp_binding_astlpc *astlpc;
Konstantin Aladyshev400766f2023-09-07 15:57:25 +0300216 const char *path;
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800217
Konstantin Aladyshev400766f2023-09-07 15:57:25 +0300218 if (n_params != 1) {
219 warnx("astlpc binding requires kcs device param");
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800220 return -1;
221 }
222
Konstantin Aladyshev400766f2023-09-07 15:57:25 +0300223 path = params[0];
224
225 astlpc = mctp_astlpc_init_fileio(path);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800226 if (!astlpc) {
227 warnx("could not initialise astlpc binding");
228 return -1;
229 }
230
Jeremy Kerr3b36d172019-09-04 11:56:09 +0800231 mctp_register_bus(mctp, mctp_binding_astlpc_core(astlpc), eid);
232
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800233 binding->data = astlpc;
234 return 0;
235}
236
Andrew Jefferyedebe162021-08-26 12:35:00 +0930237static void binding_astlpc_destroy(struct mctp *mctp, struct binding *binding)
238{
239 struct mctp_binding_astlpc *astlpc = binding->data;
240
241 mctp_unregister_bus(mctp, mctp_binding_astlpc_core(astlpc));
242
243 mctp_astlpc_destroy(astlpc);
244}
245
Andrew Jeffery8f53d632022-07-25 20:57:14 +0930246static int binding_astlpc_init_pollfd(struct binding *binding,
247 struct pollfd *pollfd)
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800248{
Andrew Jeffery8f53d632022-07-25 20:57:14 +0930249 return mctp_astlpc_init_pollfd(binding->data, pollfd);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800250}
251
252static int binding_astlpc_process(struct binding *binding)
253{
254 return mctp_astlpc_poll(binding->data);
255}
256
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600257struct binding bindings[] = { {
258 .name = "null",
259 .init = binding_null_init,
260 },
261 {
262 .name = "serial",
263 .init = binding_serial_init,
264 .destroy = NULL,
265 .init_pollfd = binding_serial_init_pollfd,
266 .process = binding_serial_process,
267 },
268 {
269 .name = "astlpc",
270 .init = binding_astlpc_init,
271 .destroy = binding_astlpc_destroy,
272 .init_pollfd = binding_astlpc_init_pollfd,
273 .process = binding_astlpc_process,
274 } };
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800275
276struct binding *binding_lookup(const char *name)
277{
278 struct binding *binding;
279 unsigned int i;
280
281 for (i = 0; i < ARRAY_SIZE(bindings); i++) {
282 binding = &bindings[i];
283
284 if (!strcmp(binding->name, name))
285 return binding;
286 }
287
288 return NULL;
289}
290
291static int socket_init(struct ctx *ctx)
292{
293 struct sockaddr_un addr;
294 int namelen, rc;
295
296 namelen = sizeof(sockname) - 1;
297 addr.sun_family = AF_UNIX;
298 memcpy(addr.sun_path, sockname, namelen);
299
300 ctx->sock = socket(AF_UNIX, SOCK_SEQPACKET, 0);
301 if (ctx->sock < 0) {
302 warn("can't create socket");
303 return -1;
304 }
305
306 rc = bind(ctx->sock, (struct sockaddr *)&addr,
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600307 sizeof(addr.sun_family) + namelen);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800308 if (rc) {
309 warn("can't bind socket");
310 goto err_close;
311 }
312
313 rc = listen(ctx->sock, 1);
314 if (rc) {
315 warn("can't listen on socket");
316 goto err_close;
317 }
318
319 return 0;
320
321err_close:
322 close(ctx->sock);
323 return -1;
324}
325
326static int socket_process(struct ctx *ctx)
327{
328 struct client *client;
329 int fd;
330
331 fd = accept4(ctx->sock, NULL, 0, SOCK_NONBLOCK);
332 if (fd < 0)
333 return -1;
334
335 ctx->n_clients++;
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600336 ctx->clients =
337 realloc(ctx->clients, ctx->n_clients * sizeof(struct client));
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800338
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600339 client = &ctx->clients[ctx->n_clients - 1];
Andrew Jeffery8676c932020-01-24 12:22:21 +1030340 memset(client, 0, sizeof(*client));
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800341 client->active = true;
342 client->sock = fd;
343
344 return 0;
345}
346
347static int client_process_recv(struct ctx *ctx, int idx)
348{
349 struct client *client = &ctx->clients[idx];
Jeremy Kerrf49b2ac2019-08-02 15:10:51 +0800350 uint8_t eid;
351 ssize_t len;
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800352 int rc;
353
354 /* are we waiting for a type message? */
355 if (!client->type) {
356 uint8_t type;
357 rc = read(client->sock, &type, 1);
358 if (rc <= 0)
359 goto out_close;
360
361 if (type == 0) {
362 rc = -1;
363 goto out_close;
364 }
365 if (ctx->verbose)
366 fprintf(stderr, "client[%d] registered for type %u\n",
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600367 idx, type);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800368 client->type = type;
369 return 0;
370 }
371
Jeremy Kerrf49b2ac2019-08-02 15:10:51 +0800372 len = recv(client->sock, NULL, 0, MSG_PEEK | MSG_TRUNC);
373 if (len < 0) {
Andrew Jeffery6896d412020-03-11 09:25:32 +1030374 if (errno != ECONNRESET)
375 warn("can't receive (peek) from client");
376
Andrew Jeffery13a40412020-02-07 17:05:20 +1030377 rc = -1;
Jeremy Kerrf49b2ac2019-08-02 15:10:51 +0800378 goto out_close;
379 }
380
Andrew Jefferyb93b6112020-06-05 14:13:44 +0930381 if ((size_t)len > ctx->buf_size) {
Jeremy Kerrf49b2ac2019-08-02 15:10:51 +0800382 void *tmp;
383
384 tmp = realloc(ctx->buf, len);
385 if (!tmp) {
386 warn("can't allocate for incoming message");
Andrew Jeffery13a40412020-02-07 17:05:20 +1030387 rc = -1;
Jeremy Kerrf49b2ac2019-08-02 15:10:51 +0800388 goto out_close;
389 }
390 ctx->buf = tmp;
391 ctx->buf_size = len;
392 }
393
394 rc = recv(client->sock, ctx->buf, ctx->buf_size, 0);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800395 if (rc < 0) {
Andrew Jeffery6896d412020-03-11 09:25:32 +1030396 if (errno != ECONNRESET)
397 warn("can't receive from client");
Andrew Jeffery13a40412020-02-07 17:05:20 +1030398 rc = -1;
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800399 goto out_close;
400 }
401
Jeremy Kerr195a7c52019-06-24 14:24:56 +0800402 if (rc <= 0) {
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800403 rc = -1;
404 goto out_close;
405 }
406
Jeremy Kerrf49b2ac2019-08-02 15:10:51 +0800407 eid = *(uint8_t *)ctx->buf;
Jeremy Kerrd690d8e2019-08-01 21:16:58 +0800408
Rashmica Guptaf2988972022-11-09 12:26:44 +1100409 if (ctx->pcap.socket.path)
410 capture_socket(ctx->pcap.socket.dumper, ctx->buf, rc,
411 MCTP_MESSAGE_CAPTURE_OUTGOING, eid);
412
Jeremy Kerr195a7c52019-06-24 14:24:56 +0800413 if (ctx->verbose)
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600414 fprintf(stderr, "client[%d] sent message: dest 0x%02x len %d\n",
Jeremy Kerr195a7c52019-06-24 14:24:56 +0800415 idx, eid, rc - 1);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800416
Jeremy Kerr195a7c52019-06-24 14:24:56 +0800417 if (eid == ctx->local_eid)
Sumanth Bhatf39c3852022-01-10 17:04:10 +0530418 rx_message(eid, MCTP_MESSAGE_TO_DST, 0, ctx, ctx->buf + 1,
419 rc - 1);
Jeremy Kerr195a7c52019-06-24 14:24:56 +0800420 else
Jeremy Kerrf49b2ac2019-08-02 15:10:51 +0800421 tx_message(ctx, eid, ctx->buf + 1, rc - 1);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800422
423 return 0;
424
425out_close:
426 client->active = false;
427 return rc;
428}
429
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600430static int binding_init(struct ctx *ctx, const char *name, int argc,
431 char *const *argv)
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800432{
433 int rc;
434
435 ctx->binding = binding_lookup(name);
436 if (!ctx->binding) {
437 warnx("no such binding '%s'", name);
438 return -1;
439 }
440
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600441 rc = ctx->binding->init(ctx->mctp, ctx->binding, ctx->local_eid, argc,
442 argv);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800443 return rc;
444}
445
Andrew Jefferyedebe162021-08-26 12:35:00 +0930446static void binding_destroy(struct ctx *ctx)
447{
448 if (ctx->binding->destroy)
449 ctx->binding->destroy(ctx->mctp, ctx->binding);
450}
451
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800452enum {
453 FD_BINDING = 0,
454 FD_SOCKET,
Andrew Jeffery490e3872021-08-24 21:11:30 +0930455 FD_SIGNAL,
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800456 FD_NR,
457};
458
459static int run_daemon(struct ctx *ctx)
460{
461 bool clients_changed = false;
Andrew Jeffery490e3872021-08-24 21:11:30 +0930462 sigset_t mask;
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800463 int rc, i;
464
465 ctx->pollfds = malloc(FD_NR * sizeof(struct pollfd));
466
Andrew Jeffery8f53d632022-07-25 20:57:14 +0930467 if (!ctx->binding->init_pollfd) {
Jeremy Kerr34b9b3d2019-11-27 11:27:05 +0800468 ctx->pollfds[FD_BINDING].fd = -1;
469 ctx->pollfds[FD_BINDING].events = 0;
470 }
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800471
Andrew Jeffery490e3872021-08-24 21:11:30 +0930472 sigemptyset(&mask);
473 sigaddset(&mask, SIGINT);
474 sigaddset(&mask, SIGTERM);
475 sigaddset(&mask, SIGQUIT);
476
477 if ((rc = sigprocmask(SIG_BLOCK, &mask, NULL)) == -1) {
478 warn("sigprocmask");
479 return rc;
480 }
481
482 ctx->pollfds[FD_SIGNAL].fd = signalfd(-1, &mask, 0);
483 ctx->pollfds[FD_SIGNAL].events = POLLIN;
484
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800485 ctx->pollfds[FD_SOCKET].fd = ctx->sock;
486 ctx->pollfds[FD_SOCKET].events = POLLIN;
487
488 mctp_set_rx_all(ctx->mctp, rx_message, ctx);
489
490 for (;;) {
491 if (clients_changed) {
492 int i;
493
494 ctx->pollfds = realloc(ctx->pollfds,
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600495 (ctx->n_clients + FD_NR) *
496 sizeof(struct pollfd));
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800497
498 for (i = 0; i < ctx->n_clients; i++) {
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600499 ctx->pollfds[FD_NR + i].fd =
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800500 ctx->clients[i].sock;
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600501 ctx->pollfds[FD_NR + i].events = POLLIN;
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800502 }
503 clients_changed = false;
504 }
505
Andrew Jeffery8f53d632022-07-25 20:57:14 +0930506 if (ctx->binding->init_pollfd)
507 ctx->binding->init_pollfd(ctx->binding,
508 &ctx->pollfds[FD_BINDING]);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800509 rc = poll(ctx->pollfds, ctx->n_clients + FD_NR, -1);
510 if (rc < 0) {
511 warn("poll failed");
512 break;
513 }
514
515 if (!rc)
516 continue;
517
Andrew Jeffery490e3872021-08-24 21:11:30 +0930518 if (ctx->pollfds[FD_SIGNAL].revents) {
519 struct signalfd_siginfo si;
520 ssize_t got;
521
522 got = read(ctx->pollfds[FD_SIGNAL].fd, &si, sizeof(si));
523 if (got == sizeof(si)) {
524 warnx("Received %s, quitting",
525 strsignal(si.ssi_signo));
526 rc = 0;
527 break;
528 } else {
529 warnx("Unexpected read result for signalfd: %d",
530 rc);
531 warnx("Quitting on the basis that signalfd became ready");
532 rc = -1;
533 break;
534 }
535 }
536
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800537 if (ctx->pollfds[FD_BINDING].revents) {
Jeremy Kerr34b9b3d2019-11-27 11:27:05 +0800538 rc = 0;
539 if (ctx->binding->process)
540 rc = ctx->binding->process(ctx->binding);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800541 if (rc)
542 break;
543 }
544
545 for (i = 0; i < ctx->n_clients; i++) {
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600546 if (!ctx->pollfds[FD_NR + i].revents)
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800547 continue;
548
549 rc = client_process_recv(ctx, i);
550 if (rc)
551 clients_changed = true;
552 }
553
554 if (ctx->pollfds[FD_SOCKET].revents) {
555 rc = socket_process(ctx);
556 if (rc)
557 break;
558 clients_changed = true;
559 }
560
561 if (clients_changed)
562 client_remove_inactive(ctx);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800563 }
564
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800565 free(ctx->pollfds);
566
567 return rc;
568}
569
570static const struct option options[] = {
Andrew Jefferycad47302021-08-20 21:37:57 +0930571 { "capture-binding", required_argument, 0, 'b' },
572 { "capture-socket", required_argument, 0, 's' },
573 { "binding-linktype", required_argument, 0, 'B' },
574 { "socket-linktype", required_argument, 0, 'S' },
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800575 { "verbose", no_argument, 0, 'v' },
576 { "eid", required_argument, 0, 'e' },
577 { 0 },
578};
579
580static void usage(const char *progname)
581{
582 unsigned int i;
583
584 fprintf(stderr, "usage: %s <binding> [params]\n", progname);
585 fprintf(stderr, "Available bindings:\n");
586 for (i = 0; i < ARRAY_SIZE(bindings); i++)
587 fprintf(stderr, " %s\n", bindings[i].name);
588}
589
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600590int main(int argc, char *const *argv)
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800591{
592 struct ctx *ctx, _ctx;
593 int rc;
594
595 ctx = &_ctx;
596 ctx->clients = NULL;
597 ctx->n_clients = 0;
598 ctx->local_eid = local_eid_default;
Andrew Jeffery04b81fc2020-02-05 13:07:29 +1030599 ctx->verbose = false;
Andrew Jefferycad47302021-08-20 21:37:57 +0930600 ctx->pcap.binding.path = NULL;
601 ctx->pcap.socket.path = NULL;
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800602
603 for (;;) {
Andrew Jefferycad47302021-08-20 21:37:57 +0930604 rc = getopt_long(argc, argv, "b:es::v", options, NULL);
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800605 if (rc == -1)
606 break;
607 switch (rc) {
Andrew Jefferycad47302021-08-20 21:37:57 +0930608 case 'b':
609 ctx->pcap.binding.path = optarg;
610 break;
611 case 's':
612 ctx->pcap.socket.path = optarg;
613 break;
614 case 'B':
Rashmica Guptaf2988972022-11-09 12:26:44 +1100615 fprintf(stderr,
616 "binding-linktype argument is deprecated\n");
Andrew Jefferycad47302021-08-20 21:37:57 +0930617 break;
618 case 'S':
Rashmica Guptaf2988972022-11-09 12:26:44 +1100619 fprintf(stderr,
620 "socket-linktype argument is deprecated\n");
Andrew Jefferycad47302021-08-20 21:37:57 +0930621 break;
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800622 case 'v':
623 ctx->verbose = true;
624 break;
625 case 'e':
626 ctx->local_eid = atoi(optarg);
627 break;
628 default:
629 fprintf(stderr, "Invalid argument\n");
630 return EXIT_FAILURE;
631 }
632 }
633
634 if (optind >= argc) {
635 fprintf(stderr, "missing binding argument\n");
636 usage(argv[0]);
637 return EXIT_FAILURE;
638 }
639
Jeremy Kerrf49b2ac2019-08-02 15:10:51 +0800640 /* setup initial buffer */
641 ctx->buf_size = 4096;
642 ctx->buf = malloc(ctx->buf_size);
643
Andrew Jefferydca82592023-06-23 15:02:05 +0930644 mctp_set_log_stdio(ctx->verbose ? MCTP_LOG_DEBUG : MCTP_LOG_NOTICE);
Jeremy Kerr0b278a62019-05-30 21:27:21 +0800645
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800646 ctx->mctp = mctp_init();
647 assert(ctx->mctp);
648
Andrew Jefferycad47302021-08-20 21:37:57 +0930649 if (ctx->pcap.binding.path || ctx->pcap.socket.path) {
650 if (capture_init()) {
651 rc = EXIT_FAILURE;
652 goto cleanup_mctp;
653 }
654 }
655
656 if (ctx->pcap.binding.path) {
657 rc = capture_prepare(&ctx->pcap.binding);
658 if (rc == -1) {
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600659 fprintf(stderr, "Failed to initialise capture: %d\n",
660 rc);
Andrew Jefferycad47302021-08-20 21:37:57 +0930661 rc = EXIT_FAILURE;
662 goto cleanup_mctp;
663 }
664
665 mctp_set_capture_handler(ctx->mctp, capture_binding,
666 ctx->pcap.binding.dumper);
667 }
668
669 if (ctx->pcap.socket.path) {
670 rc = capture_prepare(&ctx->pcap.socket);
671 if (rc == -1) {
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600672 fprintf(stderr, "Failed to initialise capture: %d\n",
673 rc);
Andrew Jefferycad47302021-08-20 21:37:57 +0930674 rc = EXIT_FAILURE;
675 goto cleanup_pcap_binding;
676 }
677 }
678
Patrick Williamsa721c2d2022-12-04 14:30:26 -0600679 rc = binding_init(ctx, argv[optind], argc - optind - 1,
680 argv + optind + 1);
Andrew Jefferycad47302021-08-20 21:37:57 +0930681 if (rc) {
682 fprintf(stderr, "Failed to initialise binding: %d\n", rc);
683 rc = EXIT_FAILURE;
684 goto cleanup_pcap_socket;
685 }
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800686
Andrew Jefferyd4103f82021-06-16 14:39:36 +0930687 rc = sd_listen_fds(true);
688 if (rc <= 0) {
689 rc = socket_init(ctx);
Andrew Jefferycad47302021-08-20 21:37:57 +0930690 if (rc) {
691 fprintf(stderr, "Failed to initialse socket: %d\n", rc);
Andrew Jefferyedebe162021-08-26 12:35:00 +0930692 goto cleanup_binding;
Andrew Jefferycad47302021-08-20 21:37:57 +0930693 }
Andrew Jefferyd4103f82021-06-16 14:39:36 +0930694 } else {
695 ctx->sock = SD_LISTEN_FDS_START;
696 }
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800697
698 rc = run_daemon(ctx);
699
Andrew Jefferyedebe162021-08-26 12:35:00 +0930700cleanup_binding:
701 binding_destroy(ctx);
702
Andrew Jefferycad47302021-08-20 21:37:57 +0930703cleanup_pcap_socket:
704 if (ctx->pcap.socket.path)
705 capture_close(&ctx->pcap.socket);
706
707cleanup_pcap_binding:
708 if (ctx->pcap.binding.path)
709 capture_close(&ctx->pcap.binding);
710
711 rc = rc ? EXIT_FAILURE : EXIT_SUCCESS;
712cleanup_mctp:
713
714 return rc;
Jeremy Kerr8e31cfd2019-05-07 17:13:51 +0800715}