2020-09-13 16:56:10 -04:00
|
|
|
#define _GNU_SOURCE
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <inttypes.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <termios.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/un.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
2020-09-14 23:24:11 -05:00
|
|
|
#include <patty/ax25.h>
|
2020-09-13 16:56:10 -04:00
|
|
|
#include <patty/kiss.h>
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
enum tnc_opts {
|
|
|
|
TNC_NONE = 0,
|
|
|
|
TNC_CLOSE_ON_DESTROY = 1 << 0
|
|
|
|
};
|
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
enum state {
|
|
|
|
KISS_NONE,
|
2020-09-22 16:30:05 -04:00
|
|
|
KISS_FRAME_COMMAND,
|
2020-09-17 01:10:34 -05:00
|
|
|
KISS_FRAME_BODY,
|
|
|
|
KISS_FRAME_ESCAPE
|
|
|
|
};
|
|
|
|
|
2020-09-13 16:56:10 -04:00
|
|
|
struct _patty_kiss_tnc {
|
2020-09-20 12:59:56 -05:00
|
|
|
patty_ax25_if_stats stats;
|
|
|
|
|
2020-09-13 16:56:10 -04:00
|
|
|
struct termios attrs,
|
|
|
|
attrs_old;
|
|
|
|
|
|
|
|
int fd,
|
|
|
|
opts;
|
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
void *buf;
|
|
|
|
|
|
|
|
enum state state;
|
2020-09-22 16:30:05 -04:00
|
|
|
enum patty_kiss_command command;
|
2020-10-05 00:19:11 -04:00
|
|
|
int port;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
|
|
|
size_t bufsz,
|
2020-09-17 01:10:34 -05:00
|
|
|
readlen,
|
|
|
|
offset_i,
|
2020-09-20 12:59:56 -05:00
|
|
|
offset_o;
|
2020-09-13 16:56:10 -04:00
|
|
|
};
|
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
static int init_sock(patty_kiss_tnc *tnc, patty_kiss_tnc_info *info) {
|
|
|
|
struct sockaddr_un addr;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
if (strlen(info->device) > sizeof(addr.sun_path)) {
|
|
|
|
errno = EOVERFLOW;
|
|
|
|
|
|
|
|
goto error_overflow;
|
2020-09-13 16:56:10 -04:00
|
|
|
}
|
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
if ((tnc->fd = socket(PF_UNIX, SOCK_STREAM, 0)) < 0) {
|
|
|
|
goto error_socket;
|
2020-09-13 16:56:10 -04:00
|
|
|
}
|
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
memset(&addr, '\0', sizeof(addr));
|
|
|
|
addr.sun_family = AF_UNIX;
|
|
|
|
strncpy(addr.sun_path, info->device, sizeof(addr.sun_path)-1);
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
if (connect(tnc->fd, (struct sockaddr *)&addr, sizeof(addr)) < 0) {
|
|
|
|
goto error_connect;
|
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
error_connect:
|
|
|
|
(void)close(tnc->fd);
|
|
|
|
|
|
|
|
error_socket:
|
|
|
|
error_overflow:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int init_termios(patty_kiss_tnc *tnc, patty_kiss_tnc_info *info) {
|
|
|
|
if (tcgetattr(tnc->fd, &tnc->attrs) < 0) {
|
|
|
|
goto error_tcgetattr;
|
2020-09-13 16:56:10 -04:00
|
|
|
}
|
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
memcpy(&tnc->attrs_old, &tnc->attrs, sizeof(tnc->attrs_old));
|
|
|
|
|
|
|
|
cfmakeraw(&tnc->attrs);
|
|
|
|
|
|
|
|
if (info->flags & PATTY_KISS_TNC_BAUD) {
|
|
|
|
cfsetspeed(&tnc->attrs, info->baud);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (info->flags & PATTY_KISS_TNC_FLOW) {
|
|
|
|
switch (info->flow) {
|
|
|
|
case PATTY_KISS_TNC_FLOW_NONE:
|
|
|
|
break;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
case PATTY_KISS_TNC_FLOW_CRTSCTS:
|
|
|
|
tnc->attrs.c_cflag |= CRTSCTS;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
break;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
case PATTY_KISS_TNC_FLOW_XONXOFF:
|
|
|
|
tnc->attrs.c_iflag |= IXON | IXOFF;
|
|
|
|
|
|
|
|
break;
|
2020-09-13 16:56:10 -04:00
|
|
|
}
|
2020-10-09 13:52:38 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (tcflush(tnc->fd, TCIOFLUSH) < 0) {
|
|
|
|
goto error_tcflush;
|
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
if (tcsetattr(tnc->fd, TCSANOW, &tnc->attrs) < 0) {
|
|
|
|
goto error_tcsetattr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
error_tcsetattr:
|
|
|
|
error_tcflush:
|
|
|
|
error_tcgetattr:
|
|
|
|
return -1;
|
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
static int init_device(patty_kiss_tnc *tnc, patty_kiss_tnc_info *info) {
|
|
|
|
struct stat st;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
if (strcmp(info->device, "/dev/ptmx") == 0) {
|
|
|
|
int ptysub;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
if (openpty(&tnc->fd, &ptysub, NULL, NULL, NULL) < 0) {
|
|
|
|
goto error;
|
2020-09-13 16:56:10 -04:00
|
|
|
}
|
2020-10-09 13:52:38 -04:00
|
|
|
} else if (stat(info->device, &st) < 0) {
|
|
|
|
goto error;
|
|
|
|
} else if ((st.st_mode & S_IFMT) == S_IFSOCK) {
|
|
|
|
if (init_sock(tnc, info) < 0) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ((tnc->fd = open(info->device, O_RDWR | O_NOCTTY)) < 0) {
|
|
|
|
goto error;
|
2020-09-13 16:56:10 -04:00
|
|
|
}
|
2020-10-09 13:52:38 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
tnc->opts |= TNC_CLOSE_ON_DESTROY;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
return -1;
|
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
patty_kiss_tnc *patty_kiss_tnc_new(patty_kiss_tnc_info *info) {
|
|
|
|
patty_kiss_tnc *tnc;
|
|
|
|
|
|
|
|
if ((tnc = malloc(sizeof(*tnc))) == NULL) {
|
|
|
|
goto error_malloc_tnc;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((tnc->buf = malloc(PATTY_KISS_TNC_BUFSZ)) == NULL) {
|
|
|
|
goto error_malloc_buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (info->flags & PATTY_KISS_TNC_DEVICE) {
|
|
|
|
if (init_device(tnc, info) < 0) {
|
|
|
|
goto error_init_device;
|
2020-09-13 16:56:10 -04:00
|
|
|
}
|
2020-10-09 13:52:38 -04:00
|
|
|
} else if (info->flags & PATTY_KISS_TNC_FD) {
|
|
|
|
tnc->fd = info->fd;
|
2020-09-13 16:56:10 -04:00
|
|
|
} else {
|
2020-10-09 13:52:38 -04:00
|
|
|
errno = EINVAL;
|
|
|
|
|
|
|
|
goto error_invalid;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isatty(tnc->fd)) {
|
|
|
|
if (init_termios(tnc, info) < 0) {
|
|
|
|
goto error_init_termios;
|
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
}
|
|
|
|
|
2020-09-20 12:59:56 -05:00
|
|
|
memset(&tnc->stats, '\0', sizeof(tnc->stats));
|
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
tnc->opts = TNC_NONE;
|
|
|
|
tnc->state = KISS_NONE;
|
2020-09-22 16:30:05 -04:00
|
|
|
tnc->command = PATTY_KISS_RETURN;
|
2020-10-05 00:19:11 -04:00
|
|
|
tnc->port = 0;
|
2020-09-17 01:10:34 -05:00
|
|
|
tnc->bufsz = PATTY_KISS_TNC_BUFSZ;
|
|
|
|
tnc->offset_i = 0;
|
|
|
|
tnc->offset_o = 0;
|
|
|
|
tnc->readlen = 0;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
|
|
|
return tnc;
|
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
error_init_termios:
|
2020-09-13 16:56:10 -04:00
|
|
|
if (info->flags & PATTY_KISS_TNC_DEVICE) {
|
|
|
|
(void)close(tnc->fd);
|
|
|
|
}
|
|
|
|
|
2020-10-09 13:52:38 -04:00
|
|
|
error_init_device:
|
|
|
|
error_invalid:
|
2020-09-13 16:56:10 -04:00
|
|
|
free(tnc->buf);
|
|
|
|
|
|
|
|
error_malloc_buf:
|
|
|
|
free(tnc);
|
|
|
|
|
|
|
|
error_malloc_tnc:
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void patty_kiss_tnc_destroy(patty_kiss_tnc *tnc) {
|
|
|
|
if (isatty(tnc->fd) && ptsname(tnc->fd) == NULL) {
|
|
|
|
(void)tcsetattr(tnc->fd, TCSANOW, &tnc->attrs_old);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tnc->opts & TNC_CLOSE_ON_DESTROY) {
|
|
|
|
(void)close(tnc->fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
free(tnc->buf);
|
|
|
|
free(tnc);
|
|
|
|
}
|
|
|
|
|
2020-09-20 12:59:56 -05:00
|
|
|
patty_ax25_if_stats *patty_kiss_tnc_stats(patty_kiss_tnc *tnc) {
|
|
|
|
return &tnc->stats;
|
|
|
|
}
|
|
|
|
|
2020-09-13 16:56:10 -04:00
|
|
|
int patty_kiss_tnc_fd(patty_kiss_tnc *tnc) {
|
|
|
|
return tnc->fd;
|
|
|
|
}
|
|
|
|
|
2020-10-03 14:56:58 -04:00
|
|
|
int patty_kiss_tnc_ready(patty_kiss_tnc *tnc, fd_set *fds) {
|
|
|
|
return FD_ISSET(tnc->fd, fds);
|
|
|
|
}
|
|
|
|
|
2020-09-13 16:56:10 -04:00
|
|
|
static void tnc_drop(patty_kiss_tnc *tnc) {
|
2020-09-17 01:10:34 -05:00
|
|
|
tnc->state = KISS_NONE;
|
2020-10-05 00:19:11 -04:00
|
|
|
tnc->command = PATTY_KISS_RETURN;
|
|
|
|
tnc->port = 0;
|
2020-09-17 01:10:34 -05:00
|
|
|
tnc->offset_i = 0;
|
|
|
|
tnc->offset_o = 0;
|
|
|
|
tnc->readlen = 0;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-20 12:59:56 -05:00
|
|
|
tnc->stats.dropped++;
|
2020-09-13 16:56:10 -04:00
|
|
|
}
|
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
ssize_t patty_kiss_tnc_fill(patty_kiss_tnc *tnc) {
|
|
|
|
if ((tnc->readlen = read(tnc->fd, tnc->buf, tnc->bufsz)) < 0) {
|
|
|
|
goto error_read;
|
2020-09-13 16:56:10 -04:00
|
|
|
}
|
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
tnc->offset_i = 0;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
return tnc->readlen;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
error_read:
|
|
|
|
return -1;
|
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
ssize_t patty_kiss_tnc_drain(patty_kiss_tnc *tnc, void *buf, size_t len) {
|
|
|
|
size_t offset_start = tnc->offset_i;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
while (tnc->offset_i < tnc->readlen) {
|
2020-09-13 16:56:10 -04:00
|
|
|
uint8_t c;
|
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
if (tnc->offset_o == len) {
|
2020-09-13 16:56:10 -04:00
|
|
|
tnc_drop(tnc);
|
|
|
|
}
|
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
c = ((uint8_t *)tnc->buf)[tnc->offset_i++];
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
switch (tnc->state) {
|
|
|
|
case KISS_NONE:
|
|
|
|
if (c == PATTY_KISS_FEND) {
|
2020-09-22 16:30:05 -04:00
|
|
|
tnc->state = KISS_FRAME_COMMAND;
|
2020-09-17 01:10:34 -05:00
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
break;
|
|
|
|
|
2020-10-05 00:19:11 -04:00
|
|
|
case KISS_FRAME_COMMAND: {
|
|
|
|
uint8_t command = PATTY_KISS_COMMAND(c),
|
|
|
|
port = PATTY_KISS_COMMAND_PORT(c);
|
|
|
|
|
|
|
|
if (command == PATTY_KISS_FEND) {
|
2020-09-17 01:10:34 -05:00
|
|
|
break;
|
2020-09-22 16:30:05 -04:00
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-05 00:19:11 -04:00
|
|
|
switch (command) {
|
2020-09-22 16:30:05 -04:00
|
|
|
case PATTY_KISS_DATA:
|
|
|
|
case PATTY_KISS_TXDELAY:
|
|
|
|
case PATTY_KISS_PERSISTENCE:
|
|
|
|
case PATTY_KISS_SLOT_TIME:
|
|
|
|
case PATTY_KISS_TX_TAIL:
|
|
|
|
case PATTY_KISS_FULL_DUPLEX:
|
|
|
|
case PATTY_KISS_HW_SET:
|
|
|
|
case PATTY_KISS_RETURN:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
errno = EIO;
|
|
|
|
|
|
|
|
goto error_io;
|
2020-09-17 01:10:34 -05:00
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-22 16:30:05 -04:00
|
|
|
tnc->state = KISS_FRAME_BODY;
|
2020-10-05 00:19:11 -04:00
|
|
|
tnc->command = command;
|
|
|
|
tnc->port = port;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
break;
|
2020-10-05 00:19:11 -04:00
|
|
|
}
|
2020-09-17 01:10:34 -05:00
|
|
|
|
|
|
|
case KISS_FRAME_BODY:
|
|
|
|
if (c == PATTY_KISS_FESC) {
|
|
|
|
tnc->state = KISS_FRAME_ESCAPE;
|
|
|
|
} else if (c == PATTY_KISS_FEND) {
|
2020-10-05 00:19:11 -04:00
|
|
|
tnc->state = KISS_FRAME_COMMAND;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
|
|
|
goto done;
|
2020-09-17 01:10:34 -05:00
|
|
|
} else {
|
2020-09-22 16:30:05 -04:00
|
|
|
switch (tnc->command) {
|
|
|
|
case PATTY_KISS_DATA:
|
|
|
|
((uint8_t *)buf)[tnc->offset_o++] = c;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
}
|
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
break;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
case KISS_FRAME_ESCAPE:
|
|
|
|
if (c == PATTY_KISS_TFEND) {
|
|
|
|
((uint8_t *)buf)[tnc->offset_o++] = PATTY_KISS_FEND;
|
|
|
|
} else if (c == PATTY_KISS_TFESC) {
|
|
|
|
((uint8_t *)buf)[tnc->offset_o++] = PATTY_KISS_FESC;
|
|
|
|
} else {
|
|
|
|
errno = EIO;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
goto error_io;
|
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
tnc->state = KISS_FRAME_BODY;
|
2020-09-13 16:56:10 -04:00
|
|
|
}
|
2020-09-17 01:10:34 -05:00
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
done:
|
|
|
|
return tnc->offset_i - offset_start;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
error_io:
|
|
|
|
return -1;
|
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-03 13:06:27 -04:00
|
|
|
int patty_kiss_tnc_pending(patty_kiss_tnc *tnc) {
|
2020-10-05 00:19:11 -04:00
|
|
|
return tnc->state == KISS_FRAME_COMMAND
|
|
|
|
&& tnc->command == PATTY_KISS_DATA
|
|
|
|
&& tnc->port == 0
|
|
|
|
&& tnc->offset_o > 0? 1: 0;
|
2020-09-17 01:10:34 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t patty_kiss_tnc_flush(patty_kiss_tnc *tnc) {
|
2020-10-05 12:02:14 -04:00
|
|
|
ssize_t ret = tnc->offset_o;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-10-04 23:21:15 -04:00
|
|
|
tnc->state = KISS_NONE;
|
|
|
|
tnc->command = PATTY_KISS_RETURN;
|
|
|
|
tnc->offset_o = 0;
|
|
|
|
|
|
|
|
return ret;
|
2020-09-17 01:10:34 -05:00
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
ssize_t patty_kiss_tnc_recv(patty_kiss_tnc *tnc, void *buf, size_t len) {
|
2020-10-03 13:06:27 -04:00
|
|
|
while (!patty_kiss_tnc_pending(tnc)) {
|
2020-09-17 01:10:34 -05:00
|
|
|
ssize_t drained;
|
|
|
|
|
|
|
|
if ((drained = patty_kiss_tnc_drain(tnc, buf, len)) < 0) {
|
|
|
|
goto error_drain;
|
|
|
|
} else if (drained == 0) {
|
|
|
|
ssize_t filled;
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
if ((filled = patty_kiss_tnc_fill(tnc)) < 0) {
|
|
|
|
goto error_fill;
|
|
|
|
} else if (filled == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2020-09-13 16:56:10 -04:00
|
|
|
}
|
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
return patty_kiss_tnc_flush(tnc);
|
2020-09-13 16:56:10 -04:00
|
|
|
|
2020-09-17 01:10:34 -05:00
|
|
|
error_drain:
|
|
|
|
error_fill:
|
2020-09-13 16:56:10 -04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t patty_kiss_tnc_send(patty_kiss_tnc *tnc,
|
|
|
|
const void *buf,
|
|
|
|
size_t len) {
|
|
|
|
return patty_kiss_frame_send(tnc->fd, buf, len, PATTY_KISS_TNC_PORT);
|
|
|
|
}
|
|
|
|
|
|
|
|
patty_ax25_if_driver *patty_kiss_tnc_driver() {
|
|
|
|
static patty_ax25_if_driver driver = {
|
2020-09-16 23:08:19 -05:00
|
|
|
.create = (patty_ax25_if_driver_create *)patty_kiss_tnc_new,
|
2020-09-13 16:56:10 -04:00
|
|
|
.destroy = (patty_ax25_if_driver_destroy *)patty_kiss_tnc_destroy,
|
2020-09-20 12:59:56 -05:00
|
|
|
.stats = (patty_ax25_if_driver_stats *)patty_kiss_tnc_stats,
|
2020-09-13 16:56:10 -04:00
|
|
|
.fd = (patty_ax25_if_driver_fd *)patty_kiss_tnc_fd,
|
2020-10-03 14:56:58 -04:00
|
|
|
.ready = (patty_ax25_if_driver_ready *)patty_kiss_tnc_ready,
|
2020-09-17 01:10:34 -05:00
|
|
|
.fill = (patty_ax25_if_driver_fill *)patty_kiss_tnc_fill,
|
|
|
|
.drain = (patty_ax25_if_driver_drain *)patty_kiss_tnc_drain,
|
2020-10-03 13:06:27 -04:00
|
|
|
.pending = (patty_ax25_if_driver_pending *)patty_kiss_tnc_pending,
|
2020-09-17 01:10:34 -05:00
|
|
|
.flush = (patty_ax25_if_driver_flush *)patty_kiss_tnc_flush,
|
2020-09-13 16:56:10 -04:00
|
|
|
.send = (patty_ax25_if_driver_send *)patty_kiss_tnc_send
|
|
|
|
};
|
|
|
|
|
|
|
|
return &driver;
|
|
|
|
}
|