1
1
//
2
2
// This file is part of Dire Wolf, an amateur radio packet TNC.
3
3
//
4
- // Copyright (C) 2011, 2013, 2014, 2015, 2016 John Langner, WB2OSZ
4
+ // Copyright (C) 2011, 2013, 2014, 2015, 2016, 2017 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
@@ -141,6 +141,7 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo);
141
141
void beacon_init (struct audio_s * pmodem , struct misc_config_s * pconfig , struct igate_config_s * pigate )
142
142
{
143
143
time_t now ;
144
+ struct tm tm ;
144
145
int j ;
145
146
int count ;
146
147
#if __WIN32__
@@ -270,21 +271,71 @@ void beacon_init (struct audio_s *pmodem, struct misc_config_s *pconfig, struct
270
271
}
271
272
272
273
/*
273
- * Calculate first time for each beacon from the 'delay' value.
274
+ * Calculate first time for each beacon from the 'slot' or ' delay' value.
274
275
*/
275
276
276
277
now = time (NULL );
278
+ localtime_r (& now , & tm );
277
279
278
280
for (j = 0 ; j < g_misc_config_p -> num_beacons ; j ++ ) {
281
+ struct beacon_s * bp = & (g_misc_config_p -> beacon [j ]);
279
282
#if DEBUG
280
283
281
284
text_color_set (DW_COLOR_DEBUG );
282
- dw_printf ("beacon[%d] chan=%d, delay=%d, every=%d\n" ,
285
+ dw_printf ("beacon[%d] chan=%d, delay=%d, slot=%d, every=%d\n" ,
283
286
j ,
284
- g_misc_config_p -> beacon [j ].sendto_chan ,
285
- g_misc_config_p -> beacon [j ].delay ,
286
- g_misc_config_p -> beacon [j ].every );
287
+ bp -> sendto_chan ,
288
+ bp -> delay ,
289
+ bp -> slot ,
290
+ bp -> every );
287
291
#endif
292
+
293
+ /*
294
+ * If timeslots, there must be a full number of beacon intervals per hour.
295
+ */
296
+ #define IS_GOOD (x ) ((3600/(x))*(x) == 3600)
297
+
298
+ if (bp -> slot != G_UNKNOWN ) {
299
+
300
+ if ( ! IS_GOOD (bp -> every )) {
301
+ text_color_set (DW_COLOR_ERROR );
302
+ dw_printf ("Config file, line %d: When using timeslots, there must be a whole number of beacon intervals per hour.\n" , bp -> lineno );
303
+
304
+ // Try to make it valid by adjusting up or down.
305
+
306
+ int n ;
307
+ for (n = 1 ; ; n ++ ) {
308
+ int e ;
309
+ e = bp -> every + n ;
310
+ if (e > 3600 ) {
311
+ bp -> every = 3600 ;
312
+ break ;
313
+ }
314
+ if (IS_GOOD (e )) {
315
+ bp -> every = e ;
316
+ break ;
317
+ }
318
+ e = bp -> every - n ;
319
+ if (e < 1 ) {
320
+ bp -> every = 1 ; // Impose a larger minimum?
321
+ break ;
322
+ }
323
+ if (IS_GOOD (e )) {
324
+ bp -> every = e ;
325
+ break ;
326
+ }
327
+ }
328
+ text_color_set (DW_COLOR_ERROR );
329
+ dw_printf ("Config file, line %d: Time between slotted beacons has been adjusted to %d seconds.\n" , bp -> lineno , bp -> every );
330
+ }
331
+ /*
332
+ * Determine when next slot time will arrive.
333
+ */
334
+ bp -> delay = bp -> slot - (tm .tm_min * 60 + tm .tm_sec );
335
+ while (bp -> delay > bp -> every ) bp -> delay -= bp -> every ;
336
+ while (bp -> delay < 5 ) bp -> delay += bp -> every ;
337
+ }
338
+
288
339
g_misc_config_p -> beacon [j ].next = now + g_misc_config_p -> beacon [j ].delay ;
289
340
}
290
341
@@ -491,10 +542,12 @@ static void * beacon_thread (void *arg)
491
542
*/
492
543
for (j = 0 ; j < g_misc_config_p -> num_beacons ; j ++ ) {
493
544
494
- if (g_misc_config_p -> beacon [j ].btype == BEACON_IGNORE )
545
+ struct beacon_s * bp = & (g_misc_config_p -> beacon [j ]);
546
+
547
+ if (bp -> btype == BEACON_IGNORE )
495
548
continue ;
496
549
497
- if (g_misc_config_p -> beacon [ j ]. next <= now ) {
550
+ if (bp -> next <= now ) {
498
551
499
552
/* Send the beacon. */
500
553
@@ -503,13 +556,20 @@ static void * beacon_thread (void *arg)
503
556
/* Calculate when the next one should be sent. */
504
557
/* Easy for fixed interval. SmartBeaconing takes more effort. */
505
558
506
- if (g_misc_config_p -> beacon [ j ]. btype == BEACON_TRACKER ) {
559
+ if (bp -> btype == BEACON_TRACKER ) {
507
560
508
561
if (gpsinfo .fix < DWFIX_2D ) {
509
562
/* Fix not available so beacon was not sent. */
510
- /* Try again in a couple seconds. */
511
563
512
- g_misc_config_p -> beacon [j ].next = now + 2 ;
564
+ if (g_misc_config_p -> sb_configured ) {
565
+ /* Try again in a couple seconds. */
566
+ bp -> next = now + 2 ;
567
+ }
568
+ else {
569
+ /* Stay with the schedule. */
570
+ /* Important for slotted. Might reconsider otherwise. */
571
+ bp -> next += bp -> every ;
572
+ }
513
573
}
514
574
else if (g_misc_config_p -> sb_configured ) {
515
575
@@ -519,18 +579,21 @@ static void * beacon_thread (void *arg)
519
579
sb_prev_time = now ;
520
580
sb_prev_course = gpsinfo .track ;
521
581
522
- g_misc_config_p -> beacon [ j ]. next = sb_calculate_next_time (now ,
582
+ bp -> next = sb_calculate_next_time (now ,
523
583
DW_KNOTS_TO_MPH (gpsinfo .speed_knots ), gpsinfo .track ,
524
584
sb_prev_time , sb_prev_course );
525
585
}
526
586
else {
527
- g_misc_config_p -> beacon [j ].next = now + g_misc_config_p -> beacon [j ].every ;
587
+ /* Tracker beacon, fixed spacing. */
588
+ bp -> next += bp -> every ;
528
589
}
529
590
}
530
591
else {
531
- /* non-tracker beacons are at fixed spacing. */
592
+ /* Non-tracker beacon, fixed spacing. */
593
+ /* Increment by 'every' so slotted times come out right. */
594
+ /* i.e. Don't take relative to now in case there was some delay. */
532
595
533
- g_misc_config_p -> beacon [ j ]. next = now + g_misc_config_p -> beacon [ j ]. every ;
596
+ bp -> next += bp -> every ;
534
597
}
535
598
536
599
} /* if time to send it */
@@ -682,6 +745,7 @@ static time_t sb_calculate_next_time (time_t now,
682
745
static void beacon_send (int j , dwgps_info_t * gpsinfo )
683
746
{
684
747
748
+ struct beacon_s * bp = & (g_misc_config_p -> beacon [j ]);
685
749
686
750
int strict = 1 ; /* Strict packet checking because they will go over air. */
687
751
char stemp [20 ];
@@ -704,13 +768,13 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo)
704
768
*/
705
769
strlcpy (mycall , "NOCALL" , sizeof (mycall ));
706
770
707
- assert (g_misc_config_p -> beacon [ j ]. sendto_chan >= 0 );
771
+ assert (bp -> sendto_chan >= 0 );
708
772
709
- strlcpy (mycall , g_modem_config_p -> achan [g_misc_config_p -> beacon [ j ]. sendto_chan ].mycall , sizeof (mycall ));
773
+ strlcpy (mycall , g_modem_config_p -> achan [bp -> sendto_chan ].mycall , sizeof (mycall ));
710
774
711
775
if (strlen (mycall ) == 0 || strcmp (mycall , "NOCALL" ) == 0 ) {
712
776
text_color_set (DW_COLOR_ERROR );
713
- dw_printf ("MYCALL not set for beacon in config file line %d.\n" , g_misc_config_p -> beacon [ j ]. lineno );
777
+ dw_printf ("MYCALL not set for beacon in config file line %d.\n" , bp -> lineno );
714
778
return ;
715
779
}
716
780
@@ -723,17 +787,17 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo)
723
787
strlcpy (beacon_text , mycall , sizeof (beacon_text ));
724
788
strlcat (beacon_text , ">" , sizeof (beacon_text ));
725
789
726
- if (g_misc_config_p -> beacon [ j ]. dest != NULL ) {
727
- strlcat (beacon_text , g_misc_config_p -> beacon [ j ]. dest , sizeof (beacon_text ));
790
+ if (bp -> dest != NULL ) {
791
+ strlcat (beacon_text , bp -> dest , sizeof (beacon_text ));
728
792
}
729
793
else {
730
794
snprintf (stemp , sizeof (stemp ), "%s%1d%1d" , APP_TOCALL , MAJOR_VERSION , MINOR_VERSION );
731
795
strlcat (beacon_text , stemp , sizeof (beacon_text ));
732
796
}
733
797
734
- if (g_misc_config_p -> beacon [ j ]. via != NULL ) {
798
+ if (bp -> via != NULL ) {
735
799
strlcat (beacon_text , "," , sizeof (beacon_text ));
736
- strlcat (beacon_text , g_misc_config_p -> beacon [ j ]. via , sizeof (beacon_text ));
800
+ strlcat (beacon_text , bp -> via , sizeof (beacon_text ));
737
801
}
738
802
strlcat (beacon_text , ":" , sizeof (beacon_text ));
739
803
@@ -746,53 +810,53 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo)
746
810
// TODO: test & document.
747
811
748
812
strlcpy (super_comment , "" , sizeof (super_comment ));
749
- if (g_misc_config_p -> beacon [ j ]. comment != NULL ) {
750
- strlcpy (super_comment , g_misc_config_p -> beacon [ j ]. comment , sizeof (super_comment ));
813
+ if (bp -> comment != NULL ) {
814
+ strlcpy (super_comment , bp -> comment , sizeof (super_comment ));
751
815
}
752
816
753
- if (g_misc_config_p -> beacon [ j ]. commentcmd != NULL ) {
817
+ if (bp -> commentcmd != NULL ) {
754
818
char var_comment [AX25_MAX_INFO_LEN ];
755
819
int k ;
756
820
757
821
/* Run given command to get variable part of comment. */
758
822
759
- k = dw_run_cmd (g_misc_config_p -> beacon [ j ]. commentcmd , 2 , var_comment , sizeof (var_comment ));
823
+ k = dw_run_cmd (bp -> commentcmd , 2 , var_comment , sizeof (var_comment ));
760
824
if (k > 0 ) {
761
825
strlcat (super_comment , var_comment , sizeof (super_comment ));
762
826
}
763
827
else {
764
828
text_color_set (DW_COLOR_ERROR );
765
- dw_printf ("xBEACON, config file line %d, COMMENTCMD failure.\n" , g_misc_config_p -> beacon [ j ]. lineno );
829
+ dw_printf ("xBEACON, config file line %d, COMMENTCMD failure.\n" , bp -> lineno );
766
830
}
767
831
}
768
832
769
833
770
834
/*
771
835
* Add the info part depending on beacon type.
772
836
*/
773
- switch (g_misc_config_p -> beacon [ j ]. btype ) {
837
+ switch (bp -> btype ) {
774
838
775
839
case BEACON_POSITION :
776
840
777
- encode_position (g_misc_config_p -> beacon [ j ]. messaging , g_misc_config_p -> beacon [ j ]. compress ,
778
- g_misc_config_p -> beacon [ j ]. lat , g_misc_config_p -> beacon [ j ]. lon , 0 ,
779
- (int )roundf (DW_METERS_TO_FEET (g_misc_config_p -> beacon [ j ]. alt_m )),
780
- g_misc_config_p -> beacon [ j ]. symtab , g_misc_config_p -> beacon [ j ]. symbol ,
781
- g_misc_config_p -> beacon [ j ]. power , g_misc_config_p -> beacon [ j ]. height , g_misc_config_p -> beacon [ j ]. gain , g_misc_config_p -> beacon [ j ]. dir ,
841
+ encode_position (bp -> messaging , bp -> compress ,
842
+ bp -> lat , bp -> lon , 0 ,
843
+ (int )roundf (DW_METERS_TO_FEET (bp -> alt_m )),
844
+ bp -> symtab , bp -> symbol ,
845
+ bp -> power , bp -> height , bp -> gain , bp -> dir ,
782
846
G_UNKNOWN , G_UNKNOWN , /* course, speed */
783
- g_misc_config_p -> beacon [ j ]. freq , g_misc_config_p -> beacon [ j ]. tone , g_misc_config_p -> beacon [ j ]. offset ,
847
+ bp -> freq , bp -> tone , bp -> offset ,
784
848
super_comment ,
785
849
info , sizeof (info ));
786
850
strlcat (beacon_text , info , sizeof (beacon_text ));
787
851
break ;
788
852
789
853
case BEACON_OBJECT :
790
854
791
- encode_object (g_misc_config_p -> beacon [ j ]. objname , g_misc_config_p -> beacon [ j ]. compress , 0 , g_misc_config_p -> beacon [ j ]. lat , g_misc_config_p -> beacon [ j ]. lon , 0 ,
792
- g_misc_config_p -> beacon [ j ]. symtab , g_misc_config_p -> beacon [ j ]. symbol ,
793
- g_misc_config_p -> beacon [ j ]. power , g_misc_config_p -> beacon [ j ]. height , g_misc_config_p -> beacon [ j ]. gain , g_misc_config_p -> beacon [ j ]. dir ,
855
+ encode_object (bp -> objname , bp -> compress , 0 , bp -> lat , bp -> lon , 0 ,
856
+ bp -> symtab , bp -> symbol ,
857
+ bp -> power , bp -> height , bp -> gain , bp -> dir ,
794
858
G_UNKNOWN , G_UNKNOWN , /* course, speed */
795
- g_misc_config_p -> beacon [ j ]. freq , g_misc_config_p -> beacon [ j ]. tone , g_misc_config_p -> beacon [ j ]. offset , super_comment ,
859
+ bp -> freq , bp -> tone , bp -> offset , super_comment ,
796
860
info , sizeof (info ));
797
861
strlcat (beacon_text , info , sizeof (beacon_text ));
798
862
break ;
@@ -809,7 +873,7 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo)
809
873
/* transmission of altitude from GPS. */
810
874
811
875
my_alt_ft = G_UNKNOWN ;
812
- if (gpsinfo -> fix >= 3 && gpsinfo -> altitude != G_UNKNOWN && g_misc_config_p -> beacon [ j ]. alt_m > 0 ) {
876
+ if (gpsinfo -> fix >= 3 && gpsinfo -> altitude != G_UNKNOWN && bp -> alt_m > 0 ) {
813
877
my_alt_ft = (int )roundf (DW_METERS_TO_FEET (gpsinfo -> altitude ));
814
878
}
815
879
@@ -818,12 +882,12 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo)
818
882
coarse = (int )roundf (gpsinfo -> track );
819
883
}
820
884
821
- encode_position (g_misc_config_p -> beacon [ j ]. messaging , g_misc_config_p -> beacon [ j ]. compress ,
885
+ encode_position (bp -> messaging , bp -> compress ,
822
886
gpsinfo -> dlat , gpsinfo -> dlon , 0 , my_alt_ft ,
823
- g_misc_config_p -> beacon [ j ]. symtab , g_misc_config_p -> beacon [ j ]. symbol ,
824
- g_misc_config_p -> beacon [ j ]. power , g_misc_config_p -> beacon [ j ]. height , g_misc_config_p -> beacon [ j ]. gain , g_misc_config_p -> beacon [ j ]. dir ,
887
+ bp -> symtab , bp -> symbol ,
888
+ bp -> power , bp -> height , bp -> gain , bp -> dir ,
825
889
coarse , (int )roundf (gpsinfo -> speed_knots ),
826
- g_misc_config_p -> beacon [ j ]. freq , g_misc_config_p -> beacon [ j ]. tone , g_misc_config_p -> beacon [ j ]. offset ,
890
+ bp -> freq , bp -> tone , bp -> offset ,
827
891
super_comment ,
828
892
info , sizeof (info ));
829
893
strlcat (beacon_text , info , sizeof (beacon_text ));
@@ -845,8 +909,8 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo)
845
909
A .g_dcs = G_UNKNOWN ;
846
910
847
911
strlcpy (A .g_src , mycall , sizeof (A .g_src ));
848
- A .g_symbol_table = g_misc_config_p -> beacon [ j ]. symtab ;
849
- A .g_symbol_code = g_misc_config_p -> beacon [ j ]. symbol ;
912
+ A .g_symbol_table = bp -> symtab ;
913
+ A .g_symbol_code = bp -> symbol ;
850
914
A .g_lat = gpsinfo -> dlat ;
851
915
A .g_lon = gpsinfo -> dlon ;
852
916
A .g_speed_mph = DW_KNOTS_TO_MPH (gpsinfo -> speed_knots );
@@ -865,25 +929,25 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo)
865
929
866
930
case BEACON_CUSTOM :
867
931
868
- if (g_misc_config_p -> beacon [ j ]. custom_info != NULL ) {
932
+ if (bp -> custom_info != NULL ) {
869
933
870
934
/* Fixed handcrafted text. */
871
935
872
- strlcat (beacon_text , g_misc_config_p -> beacon [ j ]. custom_info , sizeof (beacon_text ));
936
+ strlcat (beacon_text , bp -> custom_info , sizeof (beacon_text ));
873
937
}
874
- else if (g_misc_config_p -> beacon [ j ]. custom_infocmd != NULL ) {
938
+ else if (bp -> custom_infocmd != NULL ) {
875
939
char info_part [AX25_MAX_INFO_LEN ];
876
940
int k ;
877
941
878
942
/* Run given command to obtain the info part for packet. */
879
943
880
- k = dw_run_cmd (g_misc_config_p -> beacon [ j ]. custom_infocmd , 2 , info_part , sizeof (info_part ));
944
+ k = dw_run_cmd (bp -> custom_infocmd , 2 , info_part , sizeof (info_part ));
881
945
if (k > 0 ) {
882
946
strlcat (beacon_text , info_part , sizeof (beacon_text ));
883
947
}
884
948
else {
885
949
text_color_set (DW_COLOR_ERROR );
886
- dw_printf ("CBEACON, config file line %d, INFOCMD failure.\n" , g_misc_config_p -> beacon [ j ]. lineno );
950
+ dw_printf ("CBEACON, config file line %d, INFOCMD failure.\n" , bp -> lineno );
887
951
strlcpy (beacon_text , "" , sizeof (beacon_text )); // abort!
888
952
}
889
953
}
@@ -935,7 +999,7 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo)
935
999
alevel_t alevel ;
936
1000
937
1001
938
- switch (g_misc_config_p -> beacon [ j ]. sendto_type ) {
1002
+ switch (bp -> sendto_type ) {
939
1003
940
1004
case SENDTO_IGATE :
941
1005
@@ -949,21 +1013,21 @@ static void beacon_send (int j, dwgps_info_t *gpsinfo)
949
1013
case SENDTO_XMIT :
950
1014
default :
951
1015
952
- tq_append (g_misc_config_p -> beacon [ j ]. sendto_chan , TQ_PRIO_1_LO , pp );
1016
+ tq_append (bp -> sendto_chan , TQ_PRIO_1_LO , pp );
953
1017
break ;
954
1018
955
1019
case SENDTO_RECV :
956
1020
957
1021
/* Simulated reception from radio. */
958
1022
959
1023
memset (& alevel , 0xff , sizeof (alevel ));
960
- dlq_rec_frame (g_misc_config_p -> beacon [ j ]. sendto_chan , 0 , 0 , pp , alevel , 0 , "" );
1024
+ dlq_rec_frame (bp -> sendto_chan , 0 , 0 , pp , alevel , 0 , "" );
961
1025
break ;
962
1026
}
963
1027
}
964
1028
else {
965
1029
text_color_set (DW_COLOR_ERROR );
966
- dw_printf ("Config file: Failed to parse packet constructed from line %d.\n" , g_misc_config_p -> beacon [ j ]. lineno );
1030
+ dw_printf ("Config file: Failed to parse packet constructed from line %d.\n" , bp -> lineno );
967
1031
dw_printf ("%s\n" , beacon_text );
968
1032
}
969
1033
0 commit comments