Skip to content

Commit ecf5fd1

Browse files
committed
Issue 186 - Copy KISS frames between TCP KISS clients.
1 parent c13375a commit ecf5fd1

File tree

5 files changed

+140
-2
lines changed

5 files changed

+140
-2
lines changed

config.c

+13
Original file line numberDiff line numberDiff line change
@@ -854,6 +854,7 @@ void config_init (char *fname, struct audio_s *p_audio_config,
854854
p_misc_config->agwpe_port = DEFAULT_AGWPE_PORT;
855855
p_misc_config->kiss_port = DEFAULT_KISS_PORT;
856856
p_misc_config->enable_kiss_pt = 0; /* -p option */
857+
p_misc_config->kiss_copy = 0;
857858

858859
/* Defaults from http://info.aprs.net/index.php?title=SmartBeaconing */
859860

@@ -4332,6 +4333,16 @@ void config_init (char *fname, struct audio_s *p_audio_config,
43324333
}
43334334
}
43344335

4336+
4337+
/*
4338+
* KISSCOPY - Data from network KISS client is copied to all others.
4339+
*/
4340+
4341+
else if (strcasecmp(t, "KISSCOPY") == 0) {
4342+
p_misc_config->kiss_copy = 1;
4343+
}
4344+
4345+
43354346
/*
43364347
* GPSNMEA - Device name for reading from GPS receiver.
43374348
*/
@@ -4712,6 +4723,7 @@ void config_init (char *fname, struct audio_s *p_audio_config,
47124723
p_misc_config->maxframe_basic = n;
47134724
}
47144725
else {
4726+
p_misc_config->maxframe_basic = AX25_K_MAXFRAME_BASIC_DEFAULT;
47154727
text_color_set(DW_COLOR_ERROR);
47164728
dw_printf ("Line %d: Invalid MAXFRAME value outside range of %d to %d. Using default %d.\n",
47174729
line, AX25_K_MAXFRAME_BASIC_MIN, AX25_K_MAXFRAME_BASIC_MAX, p_misc_config->maxframe_basic);
@@ -4737,6 +4749,7 @@ void config_init (char *fname, struct audio_s *p_audio_config,
47374749
p_misc_config->maxframe_extended = n;
47384750
}
47394751
else {
4752+
p_misc_config->maxframe_extended = AX25_K_MAXFRAME_EXTENDED_DEFAULT;
47404753
text_color_set(DW_COLOR_ERROR);
47414754
dw_printf ("Line %d: Invalid EMAXFRAME value outside of range %d to %d. Using default %d.\n",
47424755
line, AX25_K_MAXFRAME_EXTENDED_MIN, AX25_K_MAXFRAME_EXTENDED_MAX, p_misc_config->maxframe_extended);

config.h

+1
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,7 @@ struct misc_config_s {
3535

3636
int agwpe_port; /* Port number for the "AGW TCPIP Socket Interface" */
3737
int kiss_port; /* Port number for the "TCP KISS" protocol. */
38+
int kiss_copy; /* Data from network KISS client is copied to all others. */
3839
int enable_kiss_pt; /* Enable pseudo terminal for KISS. */
3940
/* Want this to be off by default because it hangs */
4041
/* after a while if nothing is reading from other end. */

kiss_frame.c

+37-2
Original file line numberDiff line numberDiff line change
@@ -93,6 +93,7 @@
9393
#include "tq.h"
9494
#include "xmit.h"
9595
#include "version.h"
96+
#include "kissnet.h"
9697

9798

9899
/* In server.c. Should probably move to some misc. function file. */
@@ -506,7 +507,7 @@ void kiss_rec_byte (kiss_frame_t *kf, unsigned char ch, int debug, int client, v
506507
* debug - Debug option is selected.
507508
*
508509
* client - Client app number for TCP KISS.
509-
* Ignored for pseudo termal and serial port.
510+
* Should be -1 for pseudo termal and serial port.
510511
*
511512
* sendfun - Function to send something to the client application.
512513
* "Set Hardware" can send a response.
@@ -522,7 +523,7 @@ void kiss_rec_byte (kiss_frame_t *kf, unsigned char ch, int debug, int client, v
522523

523524
void kiss_process_msg (unsigned char *kiss_msg, int kiss_len, int debug, int client, void (*sendfun)(int,int,unsigned char*,int,int))
524525
{
525-
int port;
526+
int port; // Should rename to chan because that's what we use everywhere else.
526527
int cmd;
527528
packet_t pp;
528529
alevel_t alevel;
@@ -534,6 +535,10 @@ void kiss_process_msg (unsigned char *kiss_msg, int kiss_len, int debug, int cli
534535
{
535536
case KISS_CMD_DATA_FRAME: /* 0 = Data Frame */
536537

538+
if (client >= 0) {
539+
kissnet_copy (kiss_msg, kiss_len, port, cmd, client);
540+
}
541+
537542
/* Special hack - Discard apparently bad data from Linux AX25. */
538543

539544
/* Note July 2017: There is a variant of of KISS, called SMACK, that assumes */
@@ -545,6 +550,36 @@ void kiss_process_msg (unsigned char *kiss_msg, int kiss_len, int debug, int cli
545550
/* Our current default is a maximum of 6 channels but it is easily */
546551
/* increased by changing one number and recompiling. */
547552

553+
// Additional information, from Mike Playle, December 2018, for Issue #42
554+
//
555+
// I came across this the other day with Xastir, and took a quick look.
556+
// The problem is fixable without the kiss_frame.c hack, which doesn't help with Xastir anyway.
557+
//
558+
// Workaround
559+
//
560+
// After the kissattach command, put the interface into CRC mode "none" with a command like this:
561+
//
562+
// # kissparms -c 1 -p radio
563+
//
564+
// Analysis
565+
//
566+
// The source of this behaviour is the kernel's KISS implementation:
567+
//
568+
// https://elixir.bootlin.com/linux/v4.9/source/drivers/net/hamradio/mkiss.c#L489
569+
//
570+
// It defaults to starting in state CRC_MODE_SMACK_TEST and ending up in mode CRC_NONE
571+
// after the first two packets, which have their framing byte modified by this code in the process.
572+
// It looks to me like deliberate behaviour on the kernel's part.
573+
//
574+
// Setting the CRC mode explicitly before sending any packets stops this state machine from running.
575+
//
576+
// Is this a bug? I don't know - that's up to you! Maybe it would make sense for Direwolf to set
577+
// the CRC mode itself, or to expect this behaviour and ignore these flags on the first packets
578+
// received from the Linux pty.
579+
//
580+
// This workaround seems sound to me, though, so perhaps this is just a documentation issue.
581+
582+
548583
if (kiss_len > 16 &&
549584
(port == 2 || port == 8) &&
550585
kiss_msg[1] == 'Q' << 1 &&

kissnet.c

+87
Original file line numberDiff line numberDiff line change
@@ -159,6 +159,7 @@ static THREAD_F connect_listen_thread (void *arg);
159159
static THREAD_F kissnet_listen_thread (void *arg);
160160

161161

162+
static struct misc_config_s *s_misc_config_p;
162163

163164
static int kiss_debug = 0; /* Print information flowing from and to client. */
164165

@@ -204,6 +205,8 @@ void kissnet_init (struct misc_config_s *mc)
204205
pthread_t cmd_listen_tid[MAX_NET_CLIENTS];
205206
int e;
206207
#endif
208+
s_misc_config_p = mc;
209+
207210
int kiss_port = mc->kiss_port; /* default 8001 but easily changed. */
208211

209212

@@ -650,6 +653,90 @@ void kissnet_send_rec_packet (int chan, int kiss_cmd, unsigned char *fbuf, int f
650653
} /* end kissnet_send_rec_packet */
651654

652655

656+
/*-------------------------------------------------------------------
657+
*
658+
* Name: kissnet_copy
659+
*
660+
* Purpose: Send data from one network KISS client to all others.
661+
*
662+
* Inputs: in_msg - KISS frame data without the framing or escapes.
663+
* The first byte is channel (port) and command (should be data).
664+
*
665+
* in_len - Number of bytes in above.
666+
*
667+
* chan - Channel. Redundant because it is also in first byte of kiss_msg.
668+
* Not currently used.
669+
*
670+
* cmd - KISS command nybble. Redundant because it is in first byte.
671+
* Should be 0 because I'm expecting this only for data.
672+
*
673+
* from_client - Number of network (TCP) client instance.
674+
* Should be 0, 1, 2, ...
675+
*
676+
*
677+
* Global In: kiss_copy - From misc. configuration.
678+
* This enables the feature.
679+
*
680+
*
681+
* Description: Send message to any attached network KISS clients, other than the one where it came from.
682+
* Enable this by putting KISSCOPY in the configuration file.
683+
* Note that this applies only to network (TCP) KISS clients, not serial port, or pseudo terminal.
684+
*
685+
*
686+
*--------------------------------------------------------------------*/
687+
688+
689+
void kissnet_copy (unsigned char *in_msg, int in_len, int chan, int cmd, int from_client)
690+
{
691+
unsigned char kiss_buff[2 * AX25_MAX_PACKET_LEN];
692+
int kiss_len;
693+
int err;
694+
int send_to;
695+
696+
(void) chan;
697+
(void) cmd;
698+
699+
if (s_misc_config_p->kiss_copy) {
700+
701+
for (send_to = 0; send_to < MAX_NET_CLIENTS; send_to++) {
702+
703+
if (send_to != from_client && client_sock[send_to] != -1) {
704+
705+
kiss_len = kiss_encapsulate (in_msg, in_len, kiss_buff);
706+
707+
/* This has the escapes and the surrounding FENDs. */
708+
709+
if (kiss_debug) {
710+
kiss_debug_print (TO_CLIENT, NULL, kiss_buff, kiss_len);
711+
}
712+
713+
#if __WIN32__
714+
err = SOCK_SEND(client_sock[send_to], (char*)kiss_buff, kiss_len);
715+
if (err == SOCKET_ERROR)
716+
{
717+
text_color_set(DW_COLOR_ERROR);
718+
dw_printf ("\nError %d copying message to KISS client %d application. Closing connection.\n\n", WSAGetLastError(), send_to);
719+
closesocket (client_sock[send_to]);
720+
client_sock[send_to] = -1;
721+
WSACleanup();
722+
}
723+
#else
724+
err = SOCK_SEND (client_sock[send_to], kiss_buff, kiss_len);
725+
if (err <= 0)
726+
{
727+
text_color_set(DW_COLOR_ERROR);
728+
dw_printf ("\nError copying message to KISS client %d application. Closing connection.\n\n", send_to);
729+
close (client_sock[send_to]);
730+
client_sock[send_to] = -1;
731+
}
732+
#endif
733+
} // if origin and destination different.
734+
} // loop over all KISS network clients.
735+
} // Feature enabled.
736+
737+
} /* end kissnet_copy */
738+
739+
653740

654741
/*-------------------------------------------------------------------
655742
*

kissnet.h

+2
Original file line numberDiff line numberDiff line change
@@ -17,5 +17,7 @@ void kissnet_send_rec_packet (int chan, int kiss_cmd, unsigned char *fbuf, int
1717

1818
void kiss_net_set_debug (int n);
1919

20+
void kissnet_copy (unsigned char *kiss_msg, int kiss_len, int chan, int cmd, int from_client);
21+
2022

2123
/* end kissnet.h */

0 commit comments

Comments
 (0)