Skip to content

Commit 220e7dd

Browse files
committed
Issue 132 continued. If a client app tried to send connected data when
the transmitter was already on, the packet would get stuck in the outgoing data queue.
1 parent 4ecaf47 commit 220e7dd

File tree

7 files changed

+157
-35
lines changed

7 files changed

+157
-35
lines changed

ax25_link.c

+34-31
Original file line numberDiff line numberDiff line change
@@ -156,6 +156,8 @@
156156
* Implemented Multi Selective Reject.
157157
* More efficient generation of SREJ frames.
158158
* Reduced number of duplicate I frames sent for both REJ and SREJ cases.
159+
* Avoided unnecessary RR when I frame could take care of the ack.
160+
* (This led to issue 132 where outgoing data sometimes got stuck in the queue.)
159161
*
160162
*------------------------------------------------------------------*/
161163

@@ -591,8 +593,6 @@ static int AX25MODULO(int n, int m, const char *file, const char *func, int line
591593

592594
static void dl_data_indication (ax25_dlsm_t *S, int pid, char *data, int len);
593595

594-
static void lm_seize_confirm (ax25_dlsm_t *S);
595-
596596
static void i_frame (ax25_dlsm_t *S, cmdres_t cr, int p, int nr, int ns, int pid, char *info_ptr, int info_len);
597597
static void i_frame_continued (ax25_dlsm_t *S, int p, int ns, int pid, char *info_ptr, int info_len);
598598
static int is_ns_in_window (ax25_dlsm_t *S, int ns);
@@ -1801,8 +1801,6 @@ static void dl_data_indication (ax25_dlsm_t *S, int pid, char *data, int len)
18011801
*
18021802
* Description: We need to pause the timers when the channel is busy.
18031803
*
1804-
* Signal lm_seize_confirm when we have started to transmit.
1805-
*
18061804
*------------------------------------------------------------------------------*/
18071805

18081806
static int dcd_status[MAX_CHANS];
@@ -1859,14 +1857,6 @@ void lm_channel_busy (dlq_item_t *E)
18591857
S->radio_channel_busy = 1;
18601858
PAUSE_T1;
18611859
PAUSE_TM201;
1862-
1863-
// Did channel become busy due to PTT turning on?
1864-
1865-
if ( E->activity == OCTYPE_PTT && E->status == 1) {
1866-
1867-
lm_seize_confirm (S); // C4.2. "This primitive indicates, to the Data-link State
1868-
// machine, that the transmission opportunity has arrived."
1869-
}
18701860
}
18711861
else if ( ! busy && S->radio_channel_busy) {
18721862
S->radio_channel_busy = 0;
@@ -1901,32 +1891,43 @@ void lm_channel_busy (dlq_item_t *E)
19011891
*
19021892
*------------------------------------------------------------------------------*/
19031893

1904-
static void lm_seize_confirm (ax25_dlsm_t *S)
1894+
void lm_seize_confirm (dlq_item_t *E)
19051895
{
19061896

1907-
switch (S->state) {
1897+
assert (E->chan >= 0 && E->chan < MAX_CHANS);
19081898

1909-
case state_0_disconnected:
1910-
case state_1_awaiting_connection:
1911-
case state_2_awaiting_release:
1912-
case state_5_awaiting_v22_connection:
1899+
ax25_dlsm_t *S;
19131900

1914-
break;
1901+
for (S = list_head; S != NULL; S = S->next) {
19151902

1916-
case state_3_connected:
1917-
case state_4_timer_recovery:
1903+
if (E->chan == S->chan) {
19181904

1919-
// v1.5 change in strategy.
1920-
// New I frames, not sent yet, are delayed until after processing anything in the received transmission.
1921-
// Previously we started sending new frames, from the client app, as soon as they arrived.
1922-
// Now, we first take care of those in progress before throwing more into the mix.
19231905

1924-
i_frame_pop_off_queue(S);
1906+
switch (S->state) {
19251907

1926-
if (S->acknowledge_pending) {
1927-
S->acknowledge_pending = 0;
1928-
enquiry_response (S, frame_not_AX25, 0);
1929-
}
1908+
case state_0_disconnected:
1909+
case state_1_awaiting_connection:
1910+
case state_2_awaiting_release:
1911+
case state_5_awaiting_v22_connection:
1912+
1913+
break;
1914+
1915+
case state_3_connected:
1916+
case state_4_timer_recovery:
1917+
1918+
// v1.5 change in strategy.
1919+
// New I frames, not sent yet, are delayed until after processing anything in the received transmission.
1920+
// Previously we started sending new frames, from the client app, as soon as they arrived.
1921+
// Now, we first take care of those in progress before throwing more into the mix.
1922+
1923+
i_frame_pop_off_queue(S);
1924+
1925+
// Need an RR if we didn't have I frame send the necessary ack.
1926+
1927+
if (S->acknowledge_pending) {
1928+
S->acknowledge_pending = 0;
1929+
enquiry_response (S, frame_not_AX25, 0);
1930+
}
19301931

19311932
// Implementation difference: The flow chart for state 3 has LM-RELEASE Request here.
19321933
// I don't think I need it because the transmitter will turn off
@@ -1935,7 +1936,9 @@ static void lm_seize_confirm (ax25_dlsm_t *S)
19351936
// Erratum: The original spec had LM-SEIZE request here, for state 4, which didn't seem right.
19361937
// The 2006 revision has LM-RELEASE Request so states 3 & 4 are the same.
19371938

1938-
break;
1939+
break;
1940+
}
1941+
}
19391942
}
19401943

19411944
} /* lm_seize_confirm */

ax25_link.h

+4
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,8 @@ void ax25_link_init (struct misc_config_s *pconfig);
5252
// These functions must be called on a single thread, one at a time.
5353
// The Data Link Queue (DLQ) is used to serialize events from multiple sources.
5454

55+
// Maybe the dispatch switch should be moved to ax25_link.c so they can all
56+
// be made static and they can't be called from the wrong place accidentally.
5557

5658
void dl_connect_request (dlq_item_t *E);
5759

@@ -68,6 +70,8 @@ void dl_client_cleanup (dlq_item_t *E);
6870

6971
void lm_data_indication (dlq_item_t *E);
7072

73+
void lm_seize_confirm (dlq_item_t *E);
74+
7175
void lm_channel_busy (dlq_item_t *E);
7276

7377

direwolf.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -262,7 +262,7 @@ int main (int argc, char *argv[])
262262

263263
text_color_init(t_opt);
264264
text_color_set(DW_COLOR_INFO);
265-
dw_printf ("Dire Wolf version %d.%d (%s) Beta Test 3\n", MAJOR_VERSION, MINOR_VERSION, __DATE__);
265+
dw_printf ("Dire Wolf version %d.%d (%s) Beta Test 4\n", MAJOR_VERSION, MINOR_VERSION, __DATE__);
266266
//dw_printf ("Dire Wolf DEVELOPMENT version %d.%d %s (%s)\n", MAJOR_VERSION, MINOR_VERSION, "C", __DATE__);
267267
//dw_printf ("Dire Wolf version %d.%d\n", MAJOR_VERSION, MINOR_VERSION);
268268

dlq.c

+47
Original file line numberDiff line numberDiff line change
@@ -767,6 +767,53 @@ void dlq_channel_busy (int chan, int activity, int status)
767767

768768

769769

770+
771+
772+
/*-------------------------------------------------------------------
773+
*
774+
* Name: dlq_seize_confirm
775+
*
776+
* Purpose: Inform data link state machine that the transmitter is on.
777+
* This is in reponse to lm_seize_request.
778+
*
779+
* Inputs: chan - Radio channel number.
780+
*
781+
* Outputs: Request is appended to queue for processing by
782+
* the data link state machine.
783+
*
784+
* Description: When removed from the data link state machine queue, this
785+
* becomes lm_seize_confirm.
786+
*
787+
*--------------------------------------------------------------------*/
788+
789+
void dlq_seize_confirm (int chan)
790+
{
791+
struct dlq_item_s *pnew;
792+
793+
#if DEBUG
794+
text_color_set(DW_COLOR_DEBUG);
795+
dw_printf ("dlq_seize_confirm (chan=%d)\n", chan);
796+
#endif
797+
798+
799+
/* Allocate a new queue item. */
800+
801+
pnew = (struct dlq_item_s *) calloc (sizeof(struct dlq_item_s), 1);
802+
s_new_count++;
803+
804+
pnew->type = DLQ_SEIZE_CONFIRM;
805+
pnew->chan = chan;
806+
807+
/* Put it into queue. */
808+
809+
append_to_queue (pnew);
810+
811+
812+
} /* end dlq_seize_confirm */
813+
814+
815+
816+
770817
/*-------------------------------------------------------------------
771818
*
772819
* Name: dlq_client_cleanup

dlq.h

+3-1
Original file line numberDiff line numberDiff line change
@@ -35,7 +35,7 @@ typedef struct cdata_s {
3535

3636
/* Types of things that can be in queue. */
3737

38-
typedef enum dlq_type_e {DLQ_REC_FRAME, DLQ_CONNECT_REQUEST, DLQ_DISCONNECT_REQUEST, DLQ_XMIT_DATA_REQUEST, DLQ_REGISTER_CALLSIGN, DLQ_UNREGISTER_CALLSIGN, DLQ_CHANNEL_BUSY, DLQ_CLIENT_CLEANUP} dlq_type_t;
38+
typedef enum dlq_type_e {DLQ_REC_FRAME, DLQ_CONNECT_REQUEST, DLQ_DISCONNECT_REQUEST, DLQ_XMIT_DATA_REQUEST, DLQ_REGISTER_CALLSIGN, DLQ_UNREGISTER_CALLSIGN, DLQ_CHANNEL_BUSY, DLQ_SEIZE_CONFIRM, DLQ_CLIENT_CLEANUP} dlq_type_t;
3939

4040

4141
/* A queue item. */
@@ -116,6 +116,8 @@ void dlq_unregister_callsign (char addr[AX25_MAX_ADDR_LEN], int chan, int client
116116

117117
void dlq_channel_busy (int chan, int activity, int status);
118118

119+
void dlq_seize_confirm (int chan);
120+
119121
void dlq_client_cleanup (int client);
120122

121123

recv.c

+5
Original file line numberDiff line numberDiff line change
@@ -378,6 +378,11 @@ void recv_process (void)
378378
lm_channel_busy (pitem);
379379
break;
380380

381+
case DLQ_SEIZE_CONFIRM:
382+
383+
lm_seize_confirm (pitem);
384+
break;
385+
381386
case DLQ_CLIENT_CLEANUP:
382387

383388
dl_client_cleanup (pitem);

xmit.c

+63-2
Original file line numberDiff line numberDiff line change
@@ -74,6 +74,7 @@
7474
#include "morse.h"
7575
#include "dtmf.h"
7676
#include "xid.h"
77+
#include "dlq.h"
7778

7879

7980

@@ -104,8 +105,8 @@ static int xmit_txtail[MAX_CHANS]; /* Amount of time to keep transmitting after
104105
static int xmit_fulldup[MAX_CHANS]; /* Full duplex if non-zero. */
105106

106107
static int xmit_bits_per_sec[MAX_CHANS]; /* Data transmission rate. */
107-
/* Often called baud rate which is equivalent in */
108-
/* this case but could be different with other */
108+
/* Often called baud rate which is equivalent for */
109+
/* 1200 & 9600 cases but could be different with other */
109110
/* modulation techniques. */
110111

111112
static int g_debug_xmit_packet; /* print packet in hexadecimal form for debugging. */
@@ -114,11 +115,42 @@ static int g_debug_xmit_packet; /* print packet in hexadecimal form for debuggi
114115
// TODO: When this was first written, bits/sec was same as baud.
115116
// Need to revisit this for PSK modes where they are not the same.
116117

118+
#if 0 // Added during 1.5 beta test
119+
120+
static int BITS_TO_MS (int b, int ch) {
121+
122+
int bits_per_symbol;
123+
124+
switch (save_audio_config_p->achan[ch].modem_type) {
125+
case MODEM_QPSK: bits_per_symbol = 2; break;
126+
case MODEM_8PSK: bits_per_symbol = 3; break;
127+
case default: bits_per_symbol = 1; break;
128+
}
129+
130+
return ( (b * 1000) / (xmit_bits_per_sec[(ch)] * bits_per_symbol) );
131+
}
132+
133+
static int MS_TO_BITS (int ms, int ch) {
134+
135+
int bits_per_symbol;
136+
137+
switch (save_audio_config_p->achan[ch].modem_type) {
138+
case MODEM_QPSK: bits_per_symbol = 2; break;
139+
case MODEM_8PSK: bits_per_symbol = 3; break;
140+
case default: bits_per_symbol = 1; break;
141+
}
142+
143+
return ( (ms * xmit_bits_per_sec[(ch)] * bits_per_symbol) / 1000 ); TODO...
144+
}
145+
146+
#else // OK for 1200, 9600 but wrong for PSK
117147

118148
#define BITS_TO_MS(b,ch) (((b)*1000)/xmit_bits_per_sec[(ch)])
119149

120150
#define MS_TO_BITS(ms,ch) (((ms)*xmit_bits_per_sec[(ch)])/1000)
121151

152+
#endif
153+
122154
#define MAXX(a,b) (((a)>(b)) ? (a) : (b))
123155

124156

@@ -723,13 +755,24 @@ static void xmit_ax25_frames (int chan, int prio, packet_t pp, int max_bundle)
723755
#endif
724756
ptt_set (OCTYPE_PTT, chan, 1);
725757

758+
759+
// Inform data link state machine that we are now transmitting.
760+
761+
dlq_seize_confirm (chan); // C4.2. "This primitive indicates, to the Data-link State
762+
// machine, that the transmission opportunity has arrived."
763+
726764
pre_flags = MS_TO_BITS(xmit_txdelay[chan] * 10, chan) / 8;
727765
num_bits = hdlc_send_flags (chan, pre_flags, 0);
728766
#if DEBUG
729767
text_color_set(DW_COLOR_DEBUG);
730768
dw_printf ("xmit_thread: txdelay=%d [*10], pre_flags=%d, num_bits=%d\n", xmit_txdelay[chan], pre_flags, num_bits);
731769
#endif
732770

771+
SLEEP_MS (10); // Give data link state machine a chance to
772+
// to stuff more frames into the transmit queue,
773+
// in response to dlq_seize_confirm, so
774+
// we don't run off the end too soon.
775+
733776

734777
/*
735778
* Transmit the frame.
@@ -913,6 +956,24 @@ static int send_one_frame (int c, int p, packet_t pp)
913956

914957

915958
if (ax25_is_null_frame(pp)) {
959+
960+
// Issue 132 - We could end up in a situation where:
961+
// Transmitter is already on.
962+
// Application wants to send a frame.
963+
// dl_seize_request turns into this null frame.
964+
// It was being ignored here so the data got stuck in the queue.
965+
// I think the solution is to send back a seize confirm here.
966+
// It shouldn't hurt if we send it redundantly.
967+
// Added for 1.5 beta test 4.
968+
969+
dlq_seize_confirm (c); // C4.2. "This primitive indicates, to the Data-link State
970+
// machine, that the transmission opportunity has arrived."
971+
972+
SLEEP_MS (10); // Give data link state machine a chance to
973+
// to stuff more frames into the transmit queue,
974+
// in response to dlq_seize_confirm, so
975+
// we don't run off the end too soon.
976+
916977
return(0);
917978
}
918979

0 commit comments

Comments
 (0)