Skip to content

Commit c15903e

Browse files
committed
Issue 252 - Direwolf segments large frames when sending AX.25 version 2.0 packets.
Instead, it should simply divide the data into packets without segment headers, on an AX.25 version 2.0 connection.
1 parent e654ca4 commit c15903e

File tree

1 file changed

+45
-5
lines changed

1 file changed

+45
-5
lines changed

src/ax25_link.c

+45-5
Original file line numberDiff line numberDiff line change
@@ -1157,6 +1157,15 @@ void dl_disconnect_request (dlq_item_t *E)
11571157
*
11581158
* Erratum: Not sure how to interpret that. See example below for how it was implemented.
11591159
*
1160+
* Version 1.6: Bug 252. Segmentation was occuring for a V2.0 link. From the spec:
1161+
* "The receipt of an XID response from the other station establishes that both
1162+
* stations are using AX.25 version 2.2 or higher and enables the use of the
1163+
* segmenter/reassembler and selective reject."
1164+
* "The segmenter/reassembler procedure is only enabled if both stations on the
1165+
* link are using AX.25 version 2.2 or higher."
1166+
*
1167+
* The Segmenter Ready State SDL has no decision based on protocol version.
1168+
*
11601169
*------------------------------------------------------------------------------*/
11611170

11621171
static void data_request_good_size (ax25_dlsm_t *S, cdata_t *txdata);
@@ -1166,8 +1175,6 @@ void dl_data_request (dlq_item_t *E)
11661175
{
11671176
ax25_dlsm_t *S;
11681177
int ok_to_create = 1;
1169-
int nseg_to_follow;
1170-
int orig_offset, remaining_len;
11711178

11721179

11731180
S = get_link_handle (E->addrs, E->num_addr, E->chan, E->client, ok_to_create);
@@ -1185,6 +1192,39 @@ void dl_data_request (dlq_item_t *E)
11851192
return;
11861193
}
11871194

1195+
#define DIVROUNDUP(a,b) (((a)+(b)-1) / (b))
1196+
1197+
// Erratum: Don't do V2.2 segmentation for a V2.0 link.
1198+
// In this case, we can just split it into multiple frames not exceeding the specified max size.
1199+
// Hopefully the receiving end treats it like a stream and doesn't care about length of each frame.
1200+
1201+
if (S->modulo == 8) {
1202+
1203+
int num_frames = 0;
1204+
int remaining_len = E->txdata->len;
1205+
int offset = 0;
1206+
1207+
while (remaining_len > 0) {
1208+
int this_len = MIN(remaining_len, S->n1_paclen);
1209+
1210+
cdata_t *new_txdata = cdata_new(E->txdata->pid, E->txdata->data + offset, this_len);
1211+
data_request_good_size (S, new_txdata);
1212+
1213+
offset += this_len;
1214+
remaining_len -= this_len;
1215+
num_frames++;
1216+
}
1217+
1218+
if (num_frames != DIVROUNDUP(E->txdata->len, S->n1_paclen) || remaining_len != 0) {
1219+
text_color_set(DW_COLOR_ERROR);
1220+
dw_printf ("INTERNAL ERROR, Segmentation line %d, data length = %d, N1 = %d, num frames = %d, remaining len = %d\n",
1221+
__LINE__, E->txdata->len, S->n1_paclen, num_frames, remaining_len);
1222+
}
1223+
cdata_delete (E->txdata);
1224+
E->txdata = NULL;
1225+
return;
1226+
}
1227+
11881228
// More interesting case.
11891229
// It is too large to fit in one frame so we segment it.
11901230

@@ -1251,7 +1291,7 @@ void dl_data_request (dlq_item_t *E)
12511291
// We will decrement this before putting it in the frame so the first
12521292
// will have one less than this number.
12531293

1254-
nseg_to_follow = DIVROUNDUP(E->txdata->len + 1, S->n1_paclen - 1);
1294+
int nseg_to_follow = DIVROUNDUP(E->txdata->len + 1, S->n1_paclen - 1);
12551295

12561296
if (nseg_to_follow < 2 || nseg_to_follow > 128) {
12571297
text_color_set(DW_COLOR_ERROR);
@@ -1262,8 +1302,8 @@ void dl_data_request (dlq_item_t *E)
12621302
return;
12631303
}
12641304

1265-
orig_offset = 0;
1266-
remaining_len = E->txdata->len;
1305+
int orig_offset = 0;
1306+
int remaining_len = E->txdata->len;
12671307

12681308
// First segment.
12691309

0 commit comments

Comments
 (0)