Skip to content

Commit 75d910c

Browse files
committed
Issues 527, 528: AGW protocol compatible with NET/ROM.
1 parent 6be6f68 commit 75d910c

File tree

3 files changed

+90
-31
lines changed

3 files changed

+90
-31
lines changed

src/ax25_pad.c

+53
Original file line numberDiff line numberDiff line change
@@ -2579,6 +2579,59 @@ int ax25_get_c2 (packet_t this_p)
25792579
}
25802580

25812581

2582+
/*------------------------------------------------------------------
2583+
*
2584+
* Function: ax25_set_pid
2585+
*
2586+
* Purpose: Set protocol ID in packet.
2587+
*
2588+
* Inputs: this_p - pointer to packet object.
2589+
*
2590+
* pid - usually 0xF0 for APRS or 0xCF for NET/ROM.
2591+
*
2592+
* AX.25: "The Protocol Identifier (PID) field appears in information
2593+
* frames (I and UI) only. It identifies which kind of
2594+
* Layer 3 protocol, if any, is in use."
2595+
*
2596+
*------------------------------------------------------------------*/
2597+
2598+
void ax25_set_pid (packet_t this_p, int pid)
2599+
{
2600+
assert (this_p->magic1 == MAGIC);
2601+
assert (this_p->magic2 == MAGIC);
2602+
2603+
// Some applications set this to 0 which is an error.
2604+
// Change 0 to 0xF0 meaning no layer 3 protocol.
2605+
2606+
if (pid == 0) {
2607+
pid = AX25_PID_NO_LAYER_3;
2608+
}
2609+
2610+
// Sanity check: is it I or UI frame?
2611+
2612+
if (this_p->frame_len == 0) return;
2613+
2614+
ax25_frame_type_t frame_type;
2615+
cmdres_t cr; // command or response.
2616+
char description[64];
2617+
int pf; // Poll/Final.
2618+
int nr, ns; // Sequence numbers.
2619+
2620+
frame_type = ax25_frame_type (this_p, &cr, description, &pf, &nr, &ns);
2621+
2622+
if (frame_type != frame_type_I && frame_type != frame_type_U_UI) {
2623+
text_color_set(DW_COLOR_ERROR);
2624+
dw_printf ("ax25_set_pid(0x%2x): Packet type is not I or UI.\n", pid);
2625+
return;
2626+
}
2627+
2628+
// TODO: handle 2 control byte case.
2629+
if (this_p->num_addr >= 2) {
2630+
this_p->frame_data[ax25_get_pid_offset(this_p)] = pid;
2631+
}
2632+
}
2633+
2634+
25822635
/*------------------------------------------------------------------
25832636
*
25842637
* Function: ax25_get_pid

src/ax25_pad.h

+2
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,7 @@
6666
#define AX25_UI_FRAME 3 /* Control field value. */
6767

6868
#define AX25_PID_NO_LAYER_3 0xf0 /* protocol ID used for APRS */
69+
#define AX25_PID_NETROM 0xcf /* protocol ID used for NET/ROM */
6970
#define AX25_PID_SEGMENTATION_FRAGMENT 0x08
7071
#define AX25_PID_ESCAPE_CHARACTER 0xff
7172

@@ -427,6 +428,7 @@ extern int ax25_is_null_frame (packet_t this_p);
427428
extern int ax25_get_control (packet_t this_p);
428429
extern int ax25_get_c2 (packet_t this_p);
429430

431+
extern void ax25_set_pid (packet_t this_p, int pid);
430432
extern int ax25_get_pid (packet_t this_p);
431433

432434
extern int ax25_get_frame_len (packet_t this_p);

src/server.c

+35-31
Original file line numberDiff line numberDiff line change
@@ -379,7 +379,7 @@ static void debug_print (fromto_t fromto, int client, struct agwpe_s *pmsg, int
379379
case 'C': strlcpy (datakind, "AX.25 Connection Received", sizeof(datakind)); break;
380380
case 'D': strlcpy (datakind, "Connected AX.25 Data", sizeof(datakind)); break;
381381
case 'd': strlcpy (datakind, "Disconnected", sizeof(datakind)); break;
382-
case 'M': strlcpy (datakind, "Monitored Connected Information", sizeof(datakind)); break;
382+
case 'I': strlcpy (datakind, "Monitored Connected Information", sizeof(datakind)); break;
383383
case 'S': strlcpy (datakind, "Monitored Supervisory Information", sizeof(datakind)); break;
384384
case 'U': strlcpy (datakind, "Monitored Unproto Information", sizeof(datakind)); break;
385385
case 'T': strlcpy (datakind, "Monitoring Own Information", sizeof(datakind)); break;
@@ -1717,6 +1717,7 @@ static THREAD_F cmd_listen_thread (void *arg)
17171717

17181718
packet_t pp;
17191719

1720+
int pid = cmd.hdr.pid;
17201721
strlcpy (stemp, cmd.hdr.call_from, sizeof(stemp));
17211722
strlcat (stemp, ">", sizeof(stemp));
17221723
strlcat (stemp, cmd.hdr.call_to, sizeof(stemp));
@@ -1730,33 +1731,41 @@ static THREAD_F cmd_listen_thread (void *arg)
17301731
strlcat (stemp, p, sizeof(stemp));
17311732
p += 10;
17321733
}
1734+
// At this point, p now points to info part after digipeaters.
1735+
1736+
// Issue 527: NET/ROM routing broadcasts are binary info so we can't treat as string.
1737+
// Originally, I just appended the information part.
1738+
// That was fine until NET/ROM, with binary data, came along.
1739+
// Now we set the information field after creating the packet object.
1740+
17331741
strlcat (stemp, ":", sizeof(stemp));
1734-
strlcat (stemp, p, sizeof(stemp));
1742+
strlcat (stemp, " ", sizeof(stemp));
17351743

17361744
//text_color_set(DW_COLOR_DEBUG);
17371745
//dw_printf ("Transmit '%s'\n", stemp);
17381746

17391747
pp = ax25_from_text (stemp, 1);
17401748

1741-
17421749
if (pp == NULL) {
17431750
text_color_set(DW_COLOR_ERROR);
17441751
dw_printf ("Failed to create frame from AGW 'V' message.\n");
1752+
break;
17451753
}
1746-
else {
17471754

1748-
/* This goes into the low priority queue because it is an original. */
1755+
ax25_set_info (pp, (unsigned char*)p, data_len - ndigi * 10);
1756+
// Issue 527: NET/ROM routing broadcasts use PID 0xCF which was not preserved here.
1757+
ax25_set_pid (pp, pid);
17491758

1750-
/* Note that the protocol has no way to set the "has been used" */
1751-
/* bits in the digipeater fields. */
1759+
/* This goes into the low priority queue because it is an original. */
17521760

1753-
/* This explains why the digipeating option is grayed out in */
1754-
/* xastir when using the AGW interface. */
1755-
/* The current version uses only the 'V' message, not 'K' for transmitting. */
1761+
/* Note that the protocol has no way to set the "has been used" */
1762+
/* bits in the digipeater fields. */
17561763

1757-
tq_append (cmd.hdr.portx, TQ_PRIO_1_LO, pp);
1764+
/* This explains why the digipeating option is grayed out in */
1765+
/* xastir when using the AGW interface. */
1766+
/* The current version uses only the 'V' message, not 'K' for transmitting. */
17581767

1759-
}
1768+
tq_append (cmd.hdr.portx, TQ_PRIO_1_LO, pp);
17601769
}
17611770

17621771
break;
@@ -1890,15 +1899,14 @@ static THREAD_F cmd_listen_thread (void *arg)
18901899
unsigned char num_digi; /* Expect to be in range 1 to 7. Why not up to 8? */
18911900
char dcall[7][10];
18921901
}
1893-
#if 1
1902+
18941903
// October 2017. gcc ??? complained:
18951904
// warning: dereferencing pointer 'v' does break strict-aliasing rules
18961905
// Try adding this attribute to get rid of the warning.
18971906
// If this upsets your compiler, take it out.
18981907
// Let me know. Maybe we could put in a compiler version check here.
18991908

19001909
__attribute__((__may_alias__))
1901-
#endif
19021910
*v = (struct via_info *)cmd.data;
19031911

19041912
char callsigns[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN];
@@ -2007,41 +2015,37 @@ static THREAD_F cmd_listen_thread (void *arg)
20072015
{
20082016

20092017
int pid = cmd.hdr.pid;
2010-
(void)(pid);
2011-
/* The AGW protocol spec says, */
2012-
/* "AX.25 PID 0x00 or 0xF0 for AX.25 0xCF NETROM and others" */
2013-
2014-
/* BUG: In theory, the AX.25 PID octet should be set from this. */
2015-
/* All examples seen (above) have 0. */
2016-
/* The AX.25 protocol spec doesn't list 0 as a valid value. */
2017-
/* We always send 0xf0, meaning no layer 3. */
2018-
/* Maybe we should have an ax25_set_pid function for cases when */
2019-
/* it is neither 0 nor 0xf0. */
2020-
20212018
char stemp[AX25_MAX_PACKET_LEN];
2022-
packet_t pp;
20232019

20242020
strlcpy (stemp, cmd.hdr.call_from, sizeof(stemp));
20252021
strlcat (stemp, ">", sizeof(stemp));
20262022
strlcat (stemp, cmd.hdr.call_to, sizeof(stemp));
20272023

20282024
cmd.data[data_len] = '\0';
20292025

2026+
// Issue 527: NET/ROM routing broadcasts are binary info so we can't treat as string.
2027+
// Originally, I just appended the information part as a text string.
2028+
// That was fine until NET/ROM, with binary data, came along.
2029+
// Now we set the information field after creating the packet object.
2030+
20302031
strlcat (stemp, ":", sizeof(stemp));
2031-
strlcat (stemp, cmd.data, sizeof(stemp));
2032+
strlcat (stemp, " ", sizeof(stemp));
20322033

20332034
//text_color_set(DW_COLOR_DEBUG);
20342035
//dw_printf ("Transmit '%s'\n", stemp);
20352036

2036-
pp = ax25_from_text (stemp, 1);
2037+
packet_t pp = ax25_from_text (stemp, 1);
20372038

20382039
if (pp == NULL) {
20392040
text_color_set(DW_COLOR_ERROR);
20402041
dw_printf ("Failed to create frame from AGW 'M' message.\n");
20412042
}
2042-
else {
2043-
tq_append (cmd.hdr.portx, TQ_PRIO_1_LO, pp);
2044-
}
2043+
2044+
ax25_set_info (pp, (unsigned char*)cmd.data, data_len);
2045+
// Issue 527: NET/ROM routing broadcasts use PID 0xCF which was not preserved here.
2046+
ax25_set_pid (pp, pid);
2047+
2048+
tq_append (cmd.hdr.portx, TQ_PRIO_1_LO, pp);
20452049
}
20462050
break;
20472051

0 commit comments

Comments
 (0)