patty/examples/daemon.c
XANTRONIX Development 5cb98de5d0 Allow setting baud, flow control for TNCs
Changes:

    * Add argument to patty_kiss_tnc_new(), patty_kiss_tnc_new_fd(),
      a pointer to patty_kiss_tnc_info, to allow specifying baud and
      flow control

    * Rework patty_ax25_if_kiss_tnc_info to hold patty_kiss_tnc_info as
      a hack to allow instantiating a KISS TNC-based network interface
      with desired baud and flow control
2024-03-01 00:20:47 -05:00

139 lines
3.4 KiB
C

#define _GNU_SOURCE
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdarg.h>
#include <termios.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
#include <patty/ax25.h>
#include <patty/daemon.h>
static void usage(int argc, char **argv, const char *message, ...) {
if (message != NULL) {
va_list args;
va_start(args, message);
vfprintf(stderr, message, args);
fprintf(stderr, "\n");
va_end(args);
}
fprintf(stderr, "usage: %s /dev/ttyXX|kiss.cap [path.sock] callsign\n", argv[0]);
exit(1);
}
int main(int argc, char **argv) {
patty_daemon *daemon;
patty_ax25_if_kiss_tnc_info info = {
.type = PATTY_AX25_IF_KISS_TNC_INFO_FD,
.flags = PATTY_KISS_TNC_BAUD
| PATTY_KISS_TNC_FLOW,
.baud = B9600,
.flow = PATTY_KISS_TNC_FLOW_CRTSCTS
};
if (argc < 2) {
usage(argc, argv, "No TNC device or KISS dump file provided");
} else if (argc < 3) {
usage(argc, argv, "No socket path provided");
} else if (argc < 4) {
usage(argc, argv, "No station callsign provided");
} else if (argc > 4) {
usage(argc, argv, "Too many arguments provided");
}
if ((info.fd = open(argv[1], O_RDWR)) < 0) {
fprintf(stderr, "%s: %s: %s: %s\n",
argv[0], "open()", argv[1], strerror(errno));
goto error_open;
}
if (isatty(info.fd) && grantpt(info.fd) == 0 && unlockpt(info.fd) == 0) {
char *pts;
if ((pts = ptsname(info.fd)) == NULL) {
fprintf(stderr, "%s: %s: %s: %s\n",
argv[0], argv[1], "ptsname()", strerror(errno));
goto error_open;
}
fprintf(stderr, "pts %s\n", pts);
}
errno = 0;
if ((daemon = patty_daemon_new()) == NULL) {
fprintf(stderr, "%s: %s: %s\n",
argv[0], "patty_daemon_new()", strerror(errno));
goto error_daemon_new;
}
if (argc >= 3) {
if (patty_daemon_set_sock_path(daemon, argv[2]) < 0) {
fprintf(stderr, "%s: %s: %s\n",
argv[0], "patty_daemon_set_sock_path()", strerror(errno));
goto error_daemon_set_sock_path;
}
}
if (patty_daemon_init(daemon) < 0) {
fprintf(stderr, "%s: %s: %s\n",
argv[0], "patty_daemon_init()", strerror(errno));
goto error_daemon_init;
}
if (patty_daemon_if_add(daemon,
PATTY_AX25_IF_KISS_TNC,
(patty_ax25_if_info *)&info,
argv[3]) < 0) {
fprintf(stderr, "%s: %s: %s\n",
argv[0], "patty_daemon_if_add()", strerror(errno));
goto error_daemon_if_add;
}
if (patty_daemon_route_add_default(daemon, "kiss0") < 0) {
fprintf(stderr, "%s: %s: %s\n",
argv[0], "patty_daemon_route_add_default()", strerror(errno));
goto error_daemon_route_add_default;
}
if (patty_daemon_run(daemon) < 0) {
fprintf(stderr, "%s: %s: %s: %s\n",
argv[0], argv[1], "patty_daemon_run()", strerror(errno));
goto error_daemon_run;
}
patty_daemon_destroy(daemon);
return 0;
error_daemon_run:
error_daemon_route_add_default:
error_daemon_if_add:
error_daemon_set_sock_path:
error_daemon_init:
patty_daemon_destroy(daemon);
error_daemon_new:
close(info.fd);
error_open:
return 1;
}