Skip to content

Commit 8b9c6fc

Browse files
committed
Fix two problems with KISS pseudo terminal interface.
(1) kissattach: Error setting line discipline: TIOCSETD: Device or resource busy (2) Hang, and other resulting problems, when -p command line option used but there was no application reading from the pseudo terminal.
1 parent b35a674 commit 8b9c6fc

File tree

1 file changed

+98
-53
lines changed

1 file changed

+98
-53
lines changed

kiss.c

+98-53
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
//
22
// This file is part of Dire Wolf, an amateur radio packet TNC.
33
//
4-
// Copyright (C) 2011, 2013, 2014 John Langner, WB2OSZ
4+
// Copyright (C) 2011, 2013, 2014, 2016 John Langner, WB2OSZ
55
//
66
// This program is free software: you can redistribute it and/or modify
77
// it under the terms of the GNU General Public License as published by
@@ -24,14 +24,13 @@
2424
* Module: kiss.c
2525
*
2626
* Purpose: Act as a virtual KISS TNC for use by other packet radio applications.
27+
* On Windows, it is a serial port. On Linux, a pseudo terminal.
2728
*
2829
* Input:
2930
*
3031
* Outputs:
3132
*
32-
* Description: This provides a pseudo terminal for communication with a client application.
33-
*
34-
* It implements the KISS TNC protocol as described in:
33+
* Description: It implements the KISS TNC protocol as described in:
3534
* http://www.ka9q.net/papers/kiss.html
3635
*
3736
* Briefly, a frame is composed of
@@ -355,7 +354,7 @@ static MYFDTYPE kiss_open_pt (void)
355354
char *pts;
356355
struct termios ts;
357356
int e;
358-
//int flags;
357+
359358

360359
#if DEBUG
361360
text_color_set(DW_COLOR_DEBUG);
@@ -397,39 +396,31 @@ static MYFDTYPE kiss_open_pt (void)
397396
}
398397

399398
/*
400-
* After running for a while on Linux, the write eventually
401-
* blocks if no one is reading from the other side of
402-
* the pseudo terminal. We get stuck on the kiss data
403-
* write and reception stops.
404-
*
405-
* I tried using ioctl(,TIOCOUTQ,) to see how much was in
406-
* the queue but that always returned zero. (Ubuntu)
407-
*
408-
* Let's try using non-blocking writes and see if we get
409-
* the EWOULDBLOCK status instead of hanging.
399+
* We had a problem here since the beginning.
400+
* If no one was reading from the other end of the pseudo
401+
* terminal, the buffer space would eventually fill up,
402+
* the write here would block, and the receive decode
403+
* thread would get stuck.
404+
*
405+
* March 2016 - A "select" was put before the read to
406+
* solve a different problem. With that in place, we can
407+
* now use non-blocking I/O and detect the buffer full
408+
* condition here.
410409
*/
411410

412-
#if 0 // this is worse. all writes fail. errno = 0 bad file descriptor
413-
flags = fcntl(fd, F_GETFL, 0);
411+
// text_color_set(DW_COLOR_DEBUG);
412+
// dw_printf("Debug: Try using non-blocking mode for pseudo terminal.\n");
413+
414+
int flags = fcntl(fd, F_GETFL, 0);
414415
e = fcntl (fd, F_SETFL, flags | O_NONBLOCK);
415416
if (e != 0) {
416417
text_color_set(DW_COLOR_ERROR);
417418
dw_printf ("Can't set pseudo terminal to nonblocking, fcntl returns %d, errno = %d\n", e, errno);
418419
perror ("pt fcntl");
419420
}
420-
#endif
421-
#if 0 // same
422-
flags = 1;
423-
e = ioctl (fd, FIONBIO, &flags);
424-
if (e != 0) {
425-
text_color_set(DW_COLOR_ERROR);
426-
dw_printf ("Can't set pseudo terminal to nonblocking, ioctl returns %d, errno = %d\n", e, errno);
427-
perror ("pt ioctl");
428-
}
429-
#endif
421+
430422
text_color_set(DW_COLOR_INFO);
431423
dw_printf("Virtual KISS TNC is available on %s\n", pt_slave_name);
432-
dw_printf("WARNING - Dire Wolf will hang eventually if nothing is reading from it.\n");
433424

434425

435426
#if 1
@@ -632,13 +623,11 @@ static MYFDTYPE kiss_open_nullmodem (char *devicename)
632623
*
633624
* flen - Length of raw received frame not including the FCS
634625
* or -1 for a text string.
635-
*
636626
*
637627
* Description: Send message to client.
638628
* We really don't care if anyone is listening or not.
639629
* I don't even know if we can find out.
640630
*
641-
*
642631
*--------------------------------------------------------------------*/
643632

644633

@@ -702,14 +691,12 @@ void kiss_send_rec_packet (int chan, unsigned char *fbuf, int flen)
702691

703692
/* Pseudo terminal for Cygwin and Linux. */
704693

705-
706694
err = write (pt_master_fd, kiss_buff, (size_t)kiss_len);
707695

708696
if (err == -1 && errno == EWOULDBLOCK) {
709-
#if DEBUG
710697
text_color_set (DW_COLOR_INFO);
711-
dw_printf ("KISS SEND - discarding message because write would block.\n");
712-
#endif
698+
dw_printf ("KISS SEND - Discarding message because no one is listening.\n");
699+
dw_printf ("This happens when you use the -p option and don't read from the pseudo terminal.\n");
713700
}
714701
else if (err != kiss_len)
715702
{
@@ -772,13 +759,6 @@ void kiss_send_rec_packet (int chan, unsigned char *fbuf, int flen)
772759
//nullmodem_fd = MYFDERROR;
773760
}
774761

775-
#if DEBUG
776-
/* Could wait with GetOverlappedResult but we never */
777-
/* have an issues in this direction. */
778-
//text_color_set(DW_COLOR_DEBUG);
779-
//dw_printf ("KISS SEND completed. wrote %d / %d\n", nwritten, kiss_len);
780-
#endif
781-
782762
#else
783763
err = write (nullmodem_fd, kiss_buf, (size_t)kiss_len);
784764
if (err != len)
@@ -796,20 +776,25 @@ void kiss_send_rec_packet (int chan, unsigned char *fbuf, int flen)
796776

797777

798778

779+
799780
/*-------------------------------------------------------------------
800781
*
801-
* Name: kiss_listen_thread
782+
* Name: kiss_get
783+
*
784+
* Purpose: Read one byte from the KISS client app.
802785
*
803-
* Purpose: Wait for messages from an application.
786+
* Global In: nullmodem_fd (Windows) or pt_master_fd (Linux)
804787
*
805-
* Global In: nullmodem_fd or pt_master_fd
788+
* Returns: one byte (value 0 - 255) or terminate thread on error.
806789
*
807-
* Description: Process messages from the client application.
790+
* Description: There is room for improvment here. Reading one byte
791+
* at a time is inefficient. We could read a large block
792+
* into a local buffer and return a byte from that most of the time.
793+
* Is it worth the effort? I don't know. With GHz processors and
794+
* the low data rate here it might not make a noticable difference.
808795
*
809796
*--------------------------------------------------------------------*/
810797

811-
/* Return one byte (value 0 - 255) or terminate thread on error. */
812-
813798

814799
static int kiss_get (/* MYFDTYPE fd*/ void )
815800
{
@@ -884,26 +869,74 @@ static int kiss_get (/* MYFDTYPE fd*/ void )
884869
#else /* Linux/Cygwin version */
885870

886871
int n = 0;
887-
fd_set fd_in;
872+
fd_set fd_in, fd_ex;
888873
int rc;
889874

890875
while ( n == 0 ) {
891-
/* Reading from master fd of the pty before the client has connected leads to trouble with kissattach. */
892-
/* Use select to check if the slave has sent any data before trying to read from it. */
876+
877+
/*
878+
* Since the beginning we've always had a couple annoying problems with
879+
* the pseudo terminal KISS interface.
880+
* When using "kissattach" we would sometimes get the error message:
881+
*
882+
* kissattach: Error setting line discipline: TIOCSETD: Device or resource busy
883+
* Are you sure you have enabled MKISS support in the kernel
884+
* or, if you made it a module, that the module is loaded?
885+
*
886+
* martinhpedersen came up with the interesting idea of putting in a "select"
887+
* before the "read" and explained it like this:
888+
*
889+
* "Reading from master fd of the pty before the client has connected leads
890+
* to trouble with kissattach. Use select to check if the slave has sent
891+
* any data before trying to read from it."
892+
*
893+
* "This fix resolves the issue by not reading from the pty's master fd, until
894+
* kissattach has opened and configured the slave. This is implemented using
895+
* select() to wait for data before reading from the master fd."
896+
*
897+
* The submitted code looked like this:
898+
*
899+
* FD_ZERO(&fd_in);
900+
* rc = select(pt_master_fd + 1, &fd_in, NULL, &fd_in, NULL);
901+
*
902+
* That doesn't look right to me for a couple reasons.
903+
* First, I would expect to use FD_SET for the fd.
904+
* Second, using the same bit mask for two arguments doesn't seem
905+
* like a good idea because select modifies them.
906+
* When I tried running it, we don't get the failure message
907+
* anymore but the select never returns so we can't read data from
908+
* the KISS client app.
909+
*
910+
* I think this is what we want.
911+
*
912+
* Tested on Raspian (ARM) and Ubuntu (x86_64).
913+
* We don't get the error from kissattach anymore.
914+
*/
915+
893916
FD_ZERO(&fd_in);
894-
rc = select(pt_master_fd + 1, &fd_in, NULL, &fd_in, NULL);
917+
FD_SET(pt_master_fd, &fd_in);
918+
919+
FD_ZERO(&fd_ex);
920+
FD_SET(pt_master_fd, &fd_ex);
921+
922+
rc = select(pt_master_fd + 1, &fd_in, NULL, &fd_ex, NULL);
923+
924+
#if 0
925+
text_color_set(DW_COLOR_DEBUG);
926+
dw_printf ("select returns %d, errno=%d, fd=%d, fd_in=%08x, fd_ex=%08x\n", rc, errno, pt_master_fd, *((int*)(&fd_in)), *((int*)(&fd_in)));
927+
#endif
895928

896929
if (rc == 0)
897930
{
898-
continue;
931+
continue; // When could we get a 0?
899932
}
900933

901934
if (rc == MYFDERROR
902935
|| (n = read(pt_master_fd, &ch, (size_t)1)) != 1)
903936
{
904937

905938
text_color_set(DW_COLOR_ERROR);
906-
dw_printf ("\nError receiving kiss message from client application. Closing %s.\n\n", pt_slave_name);
939+
dw_printf ("\nError receiving KISS message from client application. Closing %s.\n\n", pt_slave_name);
907940
perror ("");
908941

909942
close (pt_master_fd);
@@ -938,6 +971,18 @@ static int kiss_get (/* MYFDTYPE fd*/ void )
938971
}
939972

940973

974+
/*-------------------------------------------------------------------
975+
*
976+
* Name: kiss_listen_thread
977+
*
978+
* Purpose: Read messages from serial port KISS client application.
979+
*
980+
* Global In: nullmodem_fd (Windows) or pt_master_fd (Linux)
981+
*
982+
* Description: Reads bytes from the KISS client app and
983+
* sends them to kiss_rec_byte for processing.
984+
*
985+
*--------------------------------------------------------------------*/
941986

942987

943988
static THREAD_F kiss_listen_thread (void *arg)

0 commit comments

Comments
 (0)