1
1
//
2
2
// This file is part of Dire Wolf, an amateur radio packet TNC.
3
3
//
4
- // Copyright (C) 2011, 2013, 2014 John Langner, WB2OSZ
4
+ // Copyright (C) 2011, 2013, 2014, 2016 John Langner, WB2OSZ
5
5
//
6
6
// This program is free software: you can redistribute it and/or modify
7
7
// it under the terms of the GNU General Public License as published by
24
24
* Module: kiss.c
25
25
*
26
26
* 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.
27
28
*
28
29
* Input:
29
30
*
30
31
* Outputs:
31
32
*
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:
35
34
* http://www.ka9q.net/papers/kiss.html
36
35
*
37
36
* Briefly, a frame is composed of
@@ -355,7 +354,7 @@ static MYFDTYPE kiss_open_pt (void)
355
354
char * pts ;
356
355
struct termios ts ;
357
356
int e ;
358
- //int flags;
357
+
359
358
360
359
#if DEBUG
361
360
text_color_set (DW_COLOR_DEBUG );
@@ -397,39 +396,31 @@ static MYFDTYPE kiss_open_pt (void)
397
396
}
398
397
399
398
/*
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 .
410
409
*/
411
410
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 );
414
415
e = fcntl (fd , F_SETFL , flags | O_NONBLOCK );
415
416
if (e != 0 ) {
416
417
text_color_set (DW_COLOR_ERROR );
417
418
dw_printf ("Can't set pseudo terminal to nonblocking, fcntl returns %d, errno = %d\n" , e , errno );
418
419
perror ("pt fcntl" );
419
420
}
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
+
430
422
text_color_set (DW_COLOR_INFO );
431
423
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" );
433
424
434
425
435
426
#if 1
@@ -632,13 +623,11 @@ static MYFDTYPE kiss_open_nullmodem (char *devicename)
632
623
*
633
624
* flen - Length of raw received frame not including the FCS
634
625
* or -1 for a text string.
635
- *
636
626
*
637
627
* Description: Send message to client.
638
628
* We really don't care if anyone is listening or not.
639
629
* I don't even know if we can find out.
640
630
*
641
- *
642
631
*--------------------------------------------------------------------*/
643
632
644
633
@@ -702,14 +691,12 @@ void kiss_send_rec_packet (int chan, unsigned char *fbuf, int flen)
702
691
703
692
/* Pseudo terminal for Cygwin and Linux. */
704
693
705
-
706
694
err = write (pt_master_fd , kiss_buff , (size_t )kiss_len );
707
695
708
696
if (err == -1 && errno == EWOULDBLOCK ) {
709
- #if DEBUG
710
697
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" );
713
700
}
714
701
else if (err != kiss_len )
715
702
{
@@ -772,13 +759,6 @@ void kiss_send_rec_packet (int chan, unsigned char *fbuf, int flen)
772
759
//nullmodem_fd = MYFDERROR;
773
760
}
774
761
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
-
782
762
#else
783
763
err = write (nullmodem_fd , kiss_buf , (size_t )kiss_len );
784
764
if (err != len )
@@ -796,20 +776,25 @@ void kiss_send_rec_packet (int chan, unsigned char *fbuf, int flen)
796
776
797
777
798
778
779
+
799
780
/*-------------------------------------------------------------------
800
781
*
801
- * Name: kiss_listen_thread
782
+ * Name: kiss_get
783
+ *
784
+ * Purpose: Read one byte from the KISS client app.
802
785
*
803
- * Purpose: Wait for messages from an application.
786
+ * Global In: nullmodem_fd (Windows) or pt_master_fd (Linux)
804
787
*
805
- * Global In: nullmodem_fd or pt_master_fd
788
+ * Returns: one byte (value 0 - 255) or terminate thread on error.
806
789
*
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.
808
795
*
809
796
*--------------------------------------------------------------------*/
810
797
811
- /* Return one byte (value 0 - 255) or terminate thread on error. */
812
-
813
798
814
799
static int kiss_get (/* MYFDTYPE fd*/ void )
815
800
{
@@ -884,26 +869,74 @@ static int kiss_get (/* MYFDTYPE fd*/ void )
884
869
#else /* Linux/Cygwin version */
885
870
886
871
int n = 0 ;
887
- fd_set fd_in ;
872
+ fd_set fd_in , fd_ex ;
888
873
int rc ;
889
874
890
875
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
+
893
916
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
895
928
896
929
if (rc == 0 )
897
930
{
898
- continue ;
931
+ continue ; // When could we get a 0?
899
932
}
900
933
901
934
if (rc == MYFDERROR
902
935
|| (n = read (pt_master_fd , & ch , (size_t )1 )) != 1 )
903
936
{
904
937
905
938
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 );
907
940
perror ("" );
908
941
909
942
close (pt_master_fd );
@@ -938,6 +971,18 @@ static int kiss_get (/* MYFDTYPE fd*/ void )
938
971
}
939
972
940
973
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
+ *--------------------------------------------------------------------*/
941
986
942
987
943
988
static THREAD_F kiss_listen_thread (void * arg )
0 commit comments