2015-09-17 21:52:09 -05:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include <patty/kiss.h>
|
|
|
|
#include <patty/ax25.h>
|
|
|
|
|
|
|
|
static void usage(int argc, char **argv, const char *message, ...) {
|
|
|
|
if (message) {
|
|
|
|
va_list args;
|
|
|
|
|
|
|
|
va_start(args, message);
|
|
|
|
vfprintf(stderr, message, args);
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
va_end(args);
|
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(stderr, "usage: %s kiss.cap\n", argv[0]);
|
|
|
|
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int callsign_fprint(FILE *stream, const patty_ax25_address *address) {
|
|
|
|
fprintf(stream, "%s", address->callsign);
|
|
|
|
|
|
|
|
if (address->ssid) {
|
|
|
|
fprintf(stream, "/%d", address->ssid);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-05-22 22:31:04 -04:00
|
|
|
#define printable(c) \
|
|
|
|
(c >= 0x20 && c < 0x7f)
|
|
|
|
|
2020-05-22 21:14:22 -04:00
|
|
|
static int hexdump_fprint(FILE *stream, void *data, size_t len) {
|
|
|
|
size_t i;
|
|
|
|
|
2020-05-22 22:31:04 -04:00
|
|
|
for (i=0; i<len; i+=16) {
|
|
|
|
size_t x;
|
|
|
|
|
|
|
|
if (fprintf(stream, "%08lx:", i) < 0) {
|
|
|
|
goto error_io;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (x=0; x<16; x++) {
|
|
|
|
if (!(x & 1)) {
|
|
|
|
if (fprintf(stream, " ") < 0) {
|
|
|
|
goto error_io;
|
2020-05-22 21:14:22 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-22 22:31:04 -04:00
|
|
|
if (i+x < len) {
|
|
|
|
if (fprintf(stream, "%02x", ((uint8_t *)data)[i+x]) < 0) {
|
|
|
|
goto error_io;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (fprintf(stream, " ") < 0) {
|
|
|
|
goto error_io;
|
|
|
|
}
|
2020-05-22 21:14:22 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-22 22:31:04 -04:00
|
|
|
if (fprintf(stream, " ") < 0) {
|
|
|
|
goto error_io;
|
2020-05-22 21:14:22 -04:00
|
|
|
}
|
|
|
|
|
2020-05-22 22:31:04 -04:00
|
|
|
for (x=0; x<16 && i+x<len; x++) {
|
|
|
|
uint8_t c = ((uint8_t *)data)[i+x];
|
|
|
|
|
|
|
|
if (fputc(printable(c)? c: '.', stream) < 0) {
|
|
|
|
goto error_io;
|
2020-05-22 21:14:22 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-22 22:31:04 -04:00
|
|
|
if (fprintf(stream, "\n") < 0) {
|
|
|
|
goto error_io;
|
2020-05-22 21:14:22 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
2020-05-22 22:31:04 -04:00
|
|
|
error_io:
|
2020-05-22 21:14:22 -04:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-09-17 21:52:09 -05:00
|
|
|
static int address_fprint(FILE *stream, const patty_ax25_frame *frame) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
callsign_fprint(stream, &frame->src);
|
2020-05-22 20:18:59 -04:00
|
|
|
|
2015-09-17 21:52:09 -05:00
|
|
|
fprintf(stream, " > ");
|
|
|
|
|
|
|
|
for (i=0; i<frame->hops; i++) {
|
|
|
|
callsign_fprint(stream, &frame->repeaters[i]);
|
2020-05-22 20:18:59 -04:00
|
|
|
fprintf(stream, " > ");
|
2015-09-17 21:52:09 -05:00
|
|
|
}
|
|
|
|
|
2020-05-22 20:18:59 -04:00
|
|
|
callsign_fprint(stream, &frame->dest);
|
|
|
|
|
2015-09-17 21:52:09 -05:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-05-23 11:05:32 -04:00
|
|
|
static int frame_fprint(FILE *stream,
|
|
|
|
const patty_ax25_frame *frame,
|
|
|
|
void *data,
|
|
|
|
size_t len) {
|
|
|
|
if (address_fprint(stream, frame) < 0) {
|
|
|
|
goto error_io;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (frame->type == PATTY_AX25_FRAME_INFO) {
|
|
|
|
if (fprintf(stream, " poll %d ns %d nr %d",
|
|
|
|
PATTY_AX25_CONTROL_POLL(frame->control),
|
|
|
|
PATTY_AX25_CONTROL_SEQ_SEND(frame->control),
|
|
|
|
PATTY_AX25_CONTROL_SEQ_RECV(frame->control)) < 0) {
|
|
|
|
goto error_io;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fprintf(stream, " proto %02x %ld bytes\n", frame->proto, len) < 0) {
|
|
|
|
goto error_io;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hexdump_fprint(stream, data, len) < 0) {
|
|
|
|
goto error_io;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error_io:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-09-17 21:52:09 -05:00
|
|
|
int main(int argc, char **argv) {
|
|
|
|
patty_kiss_tnc *tnc;
|
|
|
|
ssize_t len;
|
|
|
|
void *data;
|
|
|
|
int port;
|
|
|
|
|
|
|
|
if (argc != 2) {
|
|
|
|
usage(argc, argv, "No TNC device provided");
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((tnc = patty_kiss_tnc_open(argv[1], 330)) == NULL) {
|
|
|
|
perror("Unable to open TNC");
|
|
|
|
|
2020-05-23 11:05:32 -04:00
|
|
|
goto error_kiss_tnc_open;
|
2015-09-17 21:52:09 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
while ((len = patty_kiss_tnc_recv(tnc, &data, &port)) > 0) {
|
|
|
|
patty_ax25_frame frame;
|
|
|
|
|
2020-05-22 21:40:37 -04:00
|
|
|
if (patty_ax25_frame_decode(&frame, data, len) < 0) {
|
2015-09-17 21:52:09 -05:00
|
|
|
perror("Unable to decode frame");
|
|
|
|
|
2020-05-23 11:05:32 -04:00
|
|
|
goto error_ax25_frame_decode;
|
2015-09-17 21:52:09 -05:00
|
|
|
}
|
|
|
|
|
2020-05-23 11:09:19 -04:00
|
|
|
if (frame_fprint(stdout, &frame, data, len) < 0) {
|
2020-05-23 11:05:32 -04:00
|
|
|
perror("Unable to print frame");
|
2015-09-17 21:52:09 -05:00
|
|
|
|
2020-05-23 11:05:32 -04:00
|
|
|
goto error_frame_fprint;
|
2015-09-17 21:52:09 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
patty_kiss_tnc_close(tnc);
|
|
|
|
|
|
|
|
return 0;
|
2020-05-23 11:05:32 -04:00
|
|
|
|
|
|
|
error_frame_fprint:
|
|
|
|
error_ax25_frame_decode:
|
|
|
|
patty_kiss_tnc_close(tnc);
|
|
|
|
|
|
|
|
error_kiss_tnc_open:
|
|
|
|
return 127;
|
2015-09-17 21:52:09 -05:00
|
|
|
}
|