Skip to content

Commit f6c0049

Browse files
committed
Full Duplex.
1 parent 59d6708 commit f6c0049

File tree

7 files changed

+78
-16
lines changed

7 files changed

+78
-16
lines changed

CHANGES.md

+2
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,8 @@ This is a snapshot of ongoing development towards version of 1.5. Some features
1313

1414
- PTT using GPIO pin of CM108/CM119 (e.g. DMK URI, RB-USB RIM)
1515

16+
- Full Duplex operation. (Put "FULLDUP ON" in channel section of configuration file.)
17+
1618

1719
### Bugs Fixed: ###
1820

audio.h

+5-2
Original file line numberDiff line numberDiff line change
@@ -281,7 +281,10 @@ struct audio_s {
281281
/* are done sending the data. This is to avoid */
282282
/* dropping PTT too soon and chopping off the end */
283283
/* of the frame. Again 10 mS units. */
284-
/* At this point, I'm thinking of 10 as the default. */
284+
/* At this point, I'm thinking of 10 (= 100 mS) as the default */
285+
/* because we're not quite sure when the soundcard audio stops. */
286+
287+
int fulldup; /* Full Duplex. */
285288

286289
} achan[MAX_CHANS];
287290

@@ -368,7 +371,7 @@ struct audio_s {
368371
#define DEFAULT_PERSIST 63
369372
#define DEFAULT_TXDELAY 30
370373
#define DEFAULT_TXTAIL 10
371-
374+
#define DEFAULT_FULLDUP 0
372375

373376
/*
374377
* Note that we have two versions of these in audio.c and audio_win.c.

config.c

+32-7
Original file line numberDiff line numberDiff line change
@@ -794,6 +794,7 @@ void config_init (char *fname, struct audio_s *p_audio_config,
794794
p_audio_config->achan[channel].persist = DEFAULT_PERSIST;
795795
p_audio_config->achan[channel].txdelay = DEFAULT_TXDELAY;
796796
p_audio_config->achan[channel].txtail = DEFAULT_TXTAIL;
797+
p_audio_config->achan[channel].fulldup = DEFAULT_FULLDUP;
797798
}
798799

799800
/* First channel should always be valid. */
@@ -1778,13 +1779,13 @@ void config_init (char *fname, struct audio_s *p_audio_config,
17781779
// User can override for special cases.
17791780
p_audio_config->achan[channel].octrl[ot].ptt_invert = 0; // High for transmit.
17801781
strcpy (p_audio_config->achan[channel].octrl[ot].ptt_device, "");
1781-
1782+
17821783
// Try to find PTT device for audio output device.
17831784
// Simplifiying assumption is that we have one radio per USB Audio Adapter.
17841785
// Failure at this point is not an error.
17851786
// See if config file sets it explicitly before complaining.
17861787

1787-
cm108_find_ptt (p_audio_config->adev[ACHAN2ADEV(channel)].adevice_out,
1788+
cm108_find_ptt (p_audio_config->adev[ACHAN2ADEV(channel)].adevice_out,
17881789
p_audio_config->achan[channel].octrl[ot].ptt_device,
17891790
(int)sizeof(p_audio_config->achan[channel].octrl[ot].ptt_device));
17901791

@@ -1923,7 +1924,7 @@ void config_init (char *fname, struct audio_s *p_audio_config,
19231924
*
19241925
* TXINH - TX holdoff input
19251926
*
1926-
* xxx GPIO [-]gpio-num (only type supported yet)
1927+
* TXINH GPIO [-]gpio-num (only type supported so far)
19271928
*/
19281929

19291930
else if (strcasecmp(t, "TXINH") == 0) {
@@ -1966,7 +1967,7 @@ void config_init (char *fname, struct audio_s *p_audio_config,
19661967

19671968

19681969
/*
1969-
* DWAIT - Extra delay for receiver squelch.
1970+
* DWAIT n - Extra delay for receiver squelch. n = 10 mS units.
19701971
*/
19711972

19721973
else if (strcasecmp(t, "DWAIT") == 0) {
@@ -1990,7 +1991,7 @@ void config_init (char *fname, struct audio_s *p_audio_config,
19901991
}
19911992

19921993
/*
1993-
* SLOTTIME - For non-digipeat transmit delay timing.
1994+
* SLOTTIME n - For non-digipeat transmit delay timing. n = 10 mS units.
19941995
*/
19951996

19961997
else if (strcasecmp(t, "SLOTTIME") == 0) {
@@ -2038,7 +2039,7 @@ void config_init (char *fname, struct audio_s *p_audio_config,
20382039
}
20392040

20402041
/*
2041-
* TXDELAY - For transmit delay timing.
2042+
* TXDELAY n - For transmit delay timing. n = 10 mS units.
20422043
*/
20432044

20442045
else if (strcasecmp(t, "TXDELAY") == 0) {
@@ -2062,7 +2063,7 @@ void config_init (char *fname, struct audio_s *p_audio_config,
20622063
}
20632064

20642065
/*
2065-
* TXTAIL - For transmit timing.
2066+
* TXTAIL n - For transmit timing. n = 10 mS units.
20662067
*/
20672068

20682069
else if (strcasecmp(t, "TXTAIL") == 0) {
@@ -2085,6 +2086,30 @@ void config_init (char *fname, struct audio_s *p_audio_config,
20852086
}
20862087
}
20872088

2089+
/*
2090+
* FULLDUP {on|off} - Full Duplex
2091+
*/
2092+
else if (strcasecmp(t, "FULLDUP") == 0) {
2093+
2094+
t = split(NULL,0);
2095+
if (t == NULL) {
2096+
text_color_set(DW_COLOR_ERROR);
2097+
dw_printf ("Line %d: Missing parameter for FULLDUP command. Expecting ON or OFF.\n", line);
2098+
continue;
2099+
}
2100+
if (strcasecmp(t, "ON") == 0) {
2101+
p_audio_config->achan[channel].fulldup = 1;
2102+
}
2103+
else if (strcasecmp(t, "OFF") == 0) {
2104+
p_audio_config->achan[channel].fulldup = 0;
2105+
}
2106+
else {
2107+
p_audio_config->achan[channel].fulldup = 0;
2108+
text_color_set(DW_COLOR_ERROR);
2109+
dw_printf ("Line %d: Expected ON or OFF for FULLDUP.\n", line);
2110+
}
2111+
}
2112+
20882113
/*
20892114
* SPEECH script
20902115
*

doc/User-Guide.pdf

71.2 KB
Binary file not shown.

kiss_frame.c

+4-2
Original file line numberDiff line numberDiff line change
@@ -58,7 +58,8 @@
5858
* Spec says it is obsolete but Xastir
5959
* sends it and we respect it.
6060
*
61-
* _5 FullDuplex Ignored.
61+
* _5 FullDuplex Full Duplex. Transmit immediately without
62+
* waiting for channel to be clear.
6263
*
6364
* _6 SetHardware TNC specific.
6465
*
@@ -644,7 +645,8 @@ void kiss_process_msg (unsigned char *kiss_msg, int kiss_len, int debug, int cli
644645
case KISS_CMD_FULLDUPLEX: /* 5 = FullDuplex */
645646

646647
text_color_set(DW_COLOR_INFO);
647-
dw_printf ("KISS protocol set FullDuplex = %d, port %d - Ignored\n", kiss_msg[1], port);
648+
dw_printf ("KISS protocol set FullDuplex = %d, port %d\n", kiss_msg[1], port);
649+
xmit_set_fulldup (port, kiss_msg[1]);
648650
break;
649651

650652
case KISS_CMD_SET_HARDWARE: /* 6 = TNC specific */

xmit.c

+33-5
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
//
33
// This file is part of Dire Wolf, an amateur radio packet TNC.
44
//
5-
// Copyright (C) 2011, 2013, 2014, 2015, 2016 John Langner, WB2OSZ
5+
// Copyright (C) 2011, 2013, 2014, 2015, 2016, 2017 John Langner, WB2OSZ
66
//
77
// This program is free software: you can redistribute it and/or modify
88
// it under the terms of the GNU General Public License as published by
@@ -101,6 +101,8 @@ static int xmit_txtail[MAX_CHANS]; /* Amount of time to keep transmitting after
101101
/* dropping PTT too soon and chopping off the end */
102102
/* of the frame. Again 10 mS units. */
103103

104+
static int xmit_fulldup[MAX_CHANS]; /* Full duplex if non-zero. */
105+
104106
static int xmit_bits_per_sec[MAX_CHANS]; /* Data transmission rate. */
105107
/* Often called baud rate which is equivalent in */
106108
/* this case but could be different with other */
@@ -137,7 +139,7 @@ static dw_mutex_t audio_out_dev_mutex[MAX_ADEVS];
137139

138140

139141

140-
static int wait_for_clear_channel (int channel, int slotttime, int persist);
142+
static int wait_for_clear_channel (int channel, int slotttime, int persist, int fulldup);
141143
static void xmit_ax25_frames (int c, int p, packet_t pp, int max_bundle);
142144
static int send_one_frame (int c, int p, packet_t pp);
143145
static void xmit_speech (int c, packet_t pp);
@@ -218,6 +220,7 @@ void xmit_init (struct audio_s *p_modem, int debug_xmit_packet)
218220
xmit_persist[j] = p_modem->achan[j].persist;
219221
xmit_txdelay[j] = p_modem->achan[j].txdelay;
220222
xmit_txtail[j] = p_modem->achan[j].txtail;
223+
xmit_fulldup[j] = p_modem->achan[j].fulldup;
221224
}
222225

223226
#if DEBUG
@@ -306,6 +309,7 @@ void xmit_init (struct audio_s *p_modem, int debug_xmit_packet)
306309
* xmit_set_persist
307310
* xmit_set_slottime
308311
* xmit_set_txtail
312+
* xmit_set_fulldup
309313
*
310314
*
311315
* Purpose: The KISS protocol, and maybe others, can specify
@@ -355,6 +359,13 @@ void xmit_set_txtail (int channel, int value)
355359
}
356360
}
357361

362+
void xmit_set_fulldup (int channel, int value)
363+
{
364+
if (channel >= 0 && channel < MAX_CHANS) {
365+
xmit_fulldup[channel] = value;
366+
}
367+
}
368+
358369

359370
/*-------------------------------------------------------------------
360371
*
@@ -490,7 +501,7 @@ static void * xmit_thread (void *arg)
490501
* If there is something in the high priority queue, begin transmitting immediately.
491502
* Otherwise, wait a random amount of time, in hopes of minimizing collisions.
492503
*/
493-
ok = wait_for_clear_channel (chan, xmit_slottime[chan], xmit_persist[chan]);
504+
ok = wait_for_clear_channel (chan, xmit_slottime[chan], xmit_persist[chan], xmit_fulldup[chan]);
494505

495506
prio = TQ_PRIO_1_LO;
496507
pp = tq_remove (chan, TQ_PRIO_0_HI);
@@ -670,6 +681,8 @@ static void * xmit_thread (void *arg)
670681
* Once we have control of the channel, we might as well keep going.
671682
* [High] Priority frames will always go to head of the line,
672683
*
684+
* Version 1.5: Add full duplex option.
685+
*
673686
*--------------------------------------------------------------------*/
674687

675688

@@ -1231,12 +1244,19 @@ static void xmit_dtmf (int c, packet_t pp, int speed)
12311244
* slottime - Amount of time to wait for each iteration
12321245
* of the waiting algorithm. 10 mSec units.
12331246
*
1234-
* persist - Probability of transmitting
1247+
* persist - Probability of transmitting.
1248+
*
1249+
* fulldup - Full duplex. Just start sending immediately.
12351250
*
12361251
* Returns: 1 for OK. 0 for timeout.
12371252
*
12381253
* Description: New in version 1.2: also obtain a lock on audio out device.
12391254
*
1255+
* New in version 1.5: full duplex.
1256+
* Just start transmitting rather than waiting for clear channel.
1257+
* This would only be appropriate when transmit and receive are
1258+
* using different radio freqencies. e.g. VHF up, UHF down satellite.
1259+
*
12401260
* Transmit delay algorithm:
12411261
*
12421262
* Wait for channel to be clear.
@@ -1271,10 +1291,17 @@ static void xmit_dtmf (int c, packet_t pp, int speed)
12711291
#define WAIT_TIMEOUT_MS (60 * 1000)
12721292
#define WAIT_CHECK_EVERY_MS 10
12731293

1274-
static int wait_for_clear_channel (int chan, int slottime, int persist)
1294+
static int wait_for_clear_channel (int chan, int slottime, int persist, int fulldup)
12751295
{
12761296
int n = 0;
12771297

1298+
/*
1299+
* For dull duplex we skip the channel busy check and random wait.
1300+
* We still need to wait if operating in stereo and the other audio
1301+
* half is busy.
1302+
*/
1303+
if ( ! fulldup) {
1304+
12781305
start_over_again:
12791306

12801307
while (hdlc_rec_data_detect_any(chan)) {
@@ -1318,6 +1345,7 @@ static int wait_for_clear_channel (int chan, int slottime, int persist)
13181345
break;
13191346
}
13201347
}
1348+
}
13211349

13221350
/*
13231351
* This is to prevent two channels from transmitting at the same time

xmit.h

+2
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,8 @@ extern void xmit_set_slottime (int channel, int value);
1616

1717
extern void xmit_set_txtail (int channel, int value);
1818

19+
extern void xmit_set_fulldup (int channel, int value);
20+
1921

2022
extern int xmit_speak_it (char *script, int c, char *msg);
2123

0 commit comments

Comments
 (0)