Skip to content

Commit 04ecdbc

Browse files
committed
Complete the new ICHANNEL feature.
1 parent 031c937 commit 04ecdbc

8 files changed

+249
-55
lines changed

CHANGES.md

+3
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,9 @@
77

88
### New Features: ###
99

10+
- Additional documentation location to slow down growth of main repository. [https://github.com/wb2osz/direwolf-doc](https://github.com/wb2osz/direwolf-doc)
11+
12+
- New ICHANNEL configuration option to map a KISS client application channel to APRS-IS. Packets from APRS-IS will be presented to client applications as the specified channel. Packets sent, by client applications, to that channel will go to APRS-IS rather than a radio channel. Details in ***Internal-Packet-Routing.pdf***.
1013

1114
- New variable speed option for gen_packets. For example, "-v 5,0.1" would generate packets from 5% too slow to 5% too fast with increments of 0.1. Some implementations might have imprecise timing. Use this to test how well TNCs tolerate sloppy timing.
1215

src/beacon.c

+11-3
Original file line numberDiff line numberDiff line change
@@ -162,6 +162,7 @@ void beacon_init (struct audio_s *pmodem, struct misc_config_s *pconfig, struct
162162
int chan = g_misc_config_p->beacon[j].sendto_chan;
163163

164164
if (chan < 0) chan = 0; /* For IGate, use channel 0 call. */
165+
if (chan >= MAX_CHANS) chan = 0; // For ICHANNEL, use channel 0 call.
165166

166167
if (g_modem_config_p->chan_medium[chan] == MEDIUM_RADIO ||
167168
g_modem_config_p->chan_medium[chan] == MEDIUM_NETTNC) {
@@ -621,6 +622,7 @@ static void * beacon_thread (void *arg)
621622
// On reboot, the time is in the past.
622623
// After time gets set from GPS, all beacons from that interval are sent.
623624
// FIXME: This will surely break time slotted scheduling.
625+
// TODO: The correct fix will be using monotonic, rather than clock, time.
624626

625627
/* craigerl: if next beacon is scheduled in the past, then set next beacon relative to now (happens when NTP pushes clock AHEAD) */
626628
/* fixme: if NTP sets clock BACK an hour, this thread will sleep for that hour */
@@ -805,11 +807,17 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo)
805807

806808
assert (bp->sendto_chan >= 0);
807809

808-
strlcpy (mycall, g_modem_config_p->achan[bp->sendto_chan].mycall, sizeof(mycall));
810+
if (g_modem_config_p->chan_medium[bp->sendto_chan] == MEDIUM_IGATE) { // ICHANNEL uses chan 0 mycall.
811+
// TODO: Maybe it should be allowed to have own.
812+
strlcpy (mycall, g_modem_config_p->achan[0].mycall, sizeof(mycall));
813+
}
814+
else {
815+
strlcpy (mycall, g_modem_config_p->achan[bp->sendto_chan].mycall, sizeof(mycall));
816+
}
809817

810818
if (strlen(mycall) == 0 || strcmp(mycall, "NOCALL") == 0) {
811819
text_color_set(DW_COLOR_ERROR);
812-
dw_printf ("MYCALL not set for beacon in config file line %d.\n", bp->lineno);
820+
dw_printf ("MYCALL not set for beacon to chan %d in config file line %d.\n", bp->sendto_chan, bp->lineno);
813821
return;
814822
}
815823

@@ -1046,7 +1054,7 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo)
10461054
text_color_set(DW_COLOR_XMIT);
10471055
dw_printf ("[ig] %s\n", beacon_text);
10481056

1049-
igate_send_rec_packet (0, pp);
1057+
igate_send_rec_packet (-1, pp); // Channel -1 to avoid RF>IS filtering.
10501058
ax25_delete (pp);
10511059
break;
10521060

src/config.c

+26-10
Original file line numberDiff line numberDiff line change
@@ -5594,7 +5594,8 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_
55945594
}
55955595
else if (value[0] == 'r' || value[0] == 'R') {
55965596
int n = atoi(value+1);
5597-
if ( n < 0 || n >= MAX_CHANS || p_audio_config->chan_medium[n] == MEDIUM_NONE) {
5597+
if (( n < 0 || n >= MAX_CHANS || p_audio_config->chan_medium[n] == MEDIUM_NONE)
5598+
&& p_audio_config->chan_medium[n] != MEDIUM_IGATE) {
55985599
text_color_set(DW_COLOR_ERROR);
55995600
dw_printf ("Config file, line %d: Simulated receive on channel %d is not valid.\n", line, n);
56005601
continue;
@@ -5604,7 +5605,8 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_
56045605
}
56055606
else if (value[0] == 't' || value[0] == 'T' || value[0] == 'x' || value[0] == 'X') {
56065607
int n = atoi(value+1);
5607-
if ( n < 0 || n >= MAX_CHANS || p_audio_config->chan_medium[n] == MEDIUM_NONE) {
5608+
if (( n < 0 || n >= MAX_CHANS || p_audio_config->chan_medium[n] == MEDIUM_NONE)
5609+
&& p_audio_config->chan_medium[n] != MEDIUM_IGATE) {
56085610
text_color_set(DW_COLOR_ERROR);
56095611
dw_printf ("Config file, line %d: Send to channel %d is not valid.\n", line, n);
56105612
continue;
@@ -5615,7 +5617,8 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_
56155617
}
56165618
else {
56175619
int n = atoi(value);
5618-
if ( n < 0 || n >= MAX_CHANS || p_audio_config->chan_medium[n] == MEDIUM_NONE) {
5620+
if (( n < 0 || n >= MAX_CHANS || p_audio_config->chan_medium[n] == MEDIUM_NONE)
5621+
&& p_audio_config->chan_medium[n] != MEDIUM_IGATE) {
56195622
text_color_set(DW_COLOR_ERROR);
56205623
dw_printf ("Config file, line %d: Send to channel %d is not valid.\n", line, n);
56215624
continue;
@@ -5864,19 +5867,32 @@ static int beacon_options(char *cmd, struct beacon_s *b, int line, struct audio_
58645867

58655868
if (b->sendto_type == SENDTO_XMIT) {
58665869

5867-
if ( b->sendto_chan < 0 || b->sendto_chan >= MAX_CHANS || p_audio_config->chan_medium[b->sendto_chan] == MEDIUM_NONE) {
5870+
if (( b->sendto_chan < 0 || b->sendto_chan >= MAX_CHANS || p_audio_config->chan_medium[b->sendto_chan] == MEDIUM_NONE)
5871+
&& p_audio_config->chan_medium[b->sendto_chan] != MEDIUM_IGATE) {
58685872
text_color_set(DW_COLOR_ERROR);
58695873
dw_printf ("Config file, line %d: Send to channel %d is not valid.\n", line, b->sendto_chan);
58705874
return (0);
58715875
}
58725876

5873-
if ( strcmp(p_audio_config->achan[b->sendto_chan].mycall, "") == 0 ||
5874-
strcmp(p_audio_config->achan[b->sendto_chan].mycall, "NOCALL") == 0 ||
5875-
strcmp(p_audio_config->achan[b->sendto_chan].mycall, "N0CALL") == 0 ) {
5877+
if (p_audio_config->chan_medium[b->sendto_chan] == MEDIUM_IGATE) { // Prevent subscript out of bounds.
5878+
// Will be using call from chan 0 later.
5879+
if ( strcmp(p_audio_config->achan[0].mycall, "") == 0 ||
5880+
strcmp(p_audio_config->achan[0].mycall, "NOCALL") == 0 ||
5881+
strcmp(p_audio_config->achan[0].mycall, "N0CALL") == 0 ) {
58765882

5877-
text_color_set(DW_COLOR_ERROR);
5878-
dw_printf ("Config file: MYCALL must be set for channel %d before beaconing is allowed.\n", b->sendto_chan);
5879-
return (0);
5883+
text_color_set(DW_COLOR_ERROR);
5884+
dw_printf ("Config file: MYCALL must be set for channel %d before beaconing is allowed.\n", 0);
5885+
return (0);
5886+
}
5887+
} else {
5888+
if ( strcmp(p_audio_config->achan[b->sendto_chan].mycall, "") == 0 ||
5889+
strcmp(p_audio_config->achan[b->sendto_chan].mycall, "NOCALL") == 0 ||
5890+
strcmp(p_audio_config->achan[b->sendto_chan].mycall, "N0CALL") == 0 ) {
5891+
5892+
text_color_set(DW_COLOR_ERROR);
5893+
dw_printf ("Config file: MYCALL must be set for channel %d before beaconing is allowed.\n", b->sendto_chan);
5894+
return (0);
5895+
}
58805896
}
58815897
}
58825898

src/dtime_now.c

+103-8
Original file line numberDiff line numberDiff line change
@@ -25,9 +25,9 @@
2525

2626
/*------------------------------------------------------------------
2727
*
28-
* Name: dtime_now
28+
* Name: dtime_realtime
2929
*
30-
* Purpose: Return current time as double precision.
30+
* Purpose: Return current wall clock time as double precision.
3131
*
3232
* Input: none
3333
*
@@ -40,15 +40,99 @@
4040
* part of a second, and having extra calculations everywhere,
4141
* simply use double precision floating point to make usage
4242
* easier.
43-
*
43+
*
44+
* NOTE: This is not a good way to calculate elapsed time because
45+
* it can jump forward or backware via NTP or other manual setting.
46+
*
47+
* Use the monotonic version for measuring elapsed time.
48+
*
49+
* History: Originally I called this dtime_now. We ran into issues where
50+
* we really cared about elapsed time, rather than wall clock time.
51+
* The wall clock time could be wrong at start up time if there
52+
* is no realtime clock or Internet access. It can then jump
53+
* when GPS time or Internet access becomes available.
54+
* All instances of dtime_now should be replaced by dtime_realtime
55+
* if we want wall clock time, or dtime_monotonic if it is to be
56+
* used for measuring elapsed time, such as between becons.
57+
*
4458
*---------------------------------------------------------------*/
4559

60+
double dtime_realtime (void)
61+
{
62+
double result;
63+
64+
#if __WIN32__
65+
/* 64 bit integer is number of 100 nanosecond intervals from Jan 1, 1601. */
66+
67+
FILETIME ft;
68+
69+
GetSystemTimeAsFileTime (&ft);
70+
71+
result = ((( (double)ft.dwHighDateTime * (256. * 256. * 256. * 256.) +
72+
(double)ft.dwLowDateTime ) / 10000000.) - 11644473600.);
73+
#else
74+
/* tv_sec is seconds from Jan 1, 1970. */
75+
76+
struct timespec ts;
77+
78+
#ifdef __APPLE__
79+
80+
// Why didn't I use clock_gettime?
81+
// Not available before Max OSX 10.12? https://github.com/gambit/gambit/issues/293
82+
83+
struct timeval tp;
84+
gettimeofday(&tp, NULL);
85+
ts.tv_nsec = tp.tv_usec * 1000;
86+
ts.tv_sec = tp.tv_sec;
87+
#else
88+
clock_gettime (CLOCK_REALTIME, &ts);
89+
#endif
90+
91+
result = ((double)(ts.tv_sec) + (double)(ts.tv_nsec) * 0.000000001);
92+
93+
#endif
94+
95+
#if DEBUG
96+
text_color_set(DW_COLOR_DEBUG);
97+
dw_printf ("dtime_realtime() returns %.3f\n", result );
98+
#endif
99+
100+
return (result);
101+
}
102+
46103

47-
double dtime_now (void)
104+
/*------------------------------------------------------------------
105+
*
106+
* Name: dtime_monotonic
107+
*
108+
* Purpose: Return montonically increasing time, which is not influenced
109+
* by the wall clock changing. e.g. leap seconds, NTP adjustments.
110+
*
111+
* Input: none
112+
*
113+
* Returns: Time as double precision, so we can get resolution
114+
* finer than one second.
115+
*
116+
* Description: Use this when calculating elapsed time.
117+
*
118+
*---------------------------------------------------------------*/
119+
120+
double dtime_monotonic (void)
48121
{
49122
double result;
50123

51124
#if __WIN32__
125+
126+
// FIXME:
127+
// This is still returning wall clock time.
128+
// https://docs.microsoft.com/en-us/windows/win32/api/sysinfoapi/nf-sysinfoapi-gettickcount64
129+
// GetTickCount64 would be ideal but it requires Vista or Server 2008.
130+
// As far as I know, the current version of direwolf still works on XP.
131+
//
132+
// As a work-around, GetTickCount could be used if we add extra code to deal
133+
// with the wrap around after about 49.7 days.
134+
// Resolution is only about 10 or 16 milliseconds. Is that good enough?
135+
52136
/* 64 bit integer is number of 100 nanosecond intervals from Jan 1, 1601. */
53137

54138
FILETIME ft;
@@ -63,12 +147,22 @@ double dtime_now (void)
63147
struct timespec ts;
64148

65149
#ifdef __APPLE__
150+
151+
// FIXME: Does MacOS have a monotonically increasing time?
152+
// https://stackoverflow.com/questions/41509505/clock-gettime-on-macos
153+
66154
struct timeval tp;
67155
gettimeofday(&tp, NULL);
68156
ts.tv_nsec = tp.tv_usec * 1000;
69157
ts.tv_sec = tp.tv_sec;
70158
#else
71-
clock_gettime (CLOCK_REALTIME, &ts);
159+
160+
// This is the only case handled properly.
161+
// Probably the only one that matters.
162+
// It is common to have a Raspberry Pi, without Internet,
163+
// starting up direwolf before GPS/NTP adjusts the time.
164+
165+
clock_gettime (CLOCK_MONOTONIC, &ts);
72166
#endif
73167

74168
result = ((double)(ts.tv_sec) + (double)(ts.tv_nsec) * 0.000000001);
@@ -84,6 +178,7 @@ double dtime_now (void)
84178
}
85179

86180

181+
87182
/*------------------------------------------------------------------
88183
*
89184
* Name: timestamp_now
@@ -104,7 +199,7 @@ double dtime_now (void)
104199

105200
void timestamp_now (char *result, int result_size, int show_ms)
106201
{
107-
double now = dtime_now();
202+
double now = dtime_realtime();
108203
time_t t = (int)now;
109204
struct tm tm;
110205

@@ -150,7 +245,7 @@ void timestamp_now (char *result, int result_size, int show_ms)
150245

151246
void timestamp_user_format (char *result, int result_size, char *user_format)
152247
{
153-
double now = dtime_now();
248+
double now = dtime_realtime();
154249
time_t t = (int)now;
155250
struct tm tm;
156251

@@ -191,7 +286,7 @@ void timestamp_user_format (char *result, int result_size, char *user_format)
191286

192287
void timestamp_filename (char *result, int result_size)
193288
{
194-
double now = dtime_now();
289+
double now = dtime_realtime();
195290
time_t t = (int)now;
196291
struct tm tm;
197292

src/dtime_now.h

+11-2
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,18 @@
11

22

3-
extern double dtime_now (void);
3+
extern double dtime_realtime (void);
4+
5+
extern double dtime_monotonic (void);
6+
47

58
void timestamp_now (char *result, int result_size, int show_ms);
69

710
void timestamp_user_format (char *result, int result_size, char *user_format);
811

9-
void timestamp_filename (char *result, int result_size);
12+
void timestamp_filename (char *result, int result_size);
13+
14+
15+
// FIXME: remove temp workaround.
16+
// Needs many scattered updates.
17+
18+
#define dtime_now dtime_realtime

0 commit comments

Comments
 (0)