@@ -1157,6 +1157,15 @@ void dl_disconnect_request (dlq_item_t *E)
1157
1157
*
1158
1158
* Erratum: Not sure how to interpret that. See example below for how it was implemented.
1159
1159
*
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
+ *
1160
1169
*------------------------------------------------------------------------------*/
1161
1170
1162
1171
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)
1166
1175
{
1167
1176
ax25_dlsm_t * S ;
1168
1177
int ok_to_create = 1 ;
1169
- int nseg_to_follow ;
1170
- int orig_offset , remaining_len ;
1171
1178
1172
1179
1173
1180
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)
1185
1192
return ;
1186
1193
}
1187
1194
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
+
1188
1228
// More interesting case.
1189
1229
// It is too large to fit in one frame so we segment it.
1190
1230
@@ -1251,7 +1291,7 @@ void dl_data_request (dlq_item_t *E)
1251
1291
// We will decrement this before putting it in the frame so the first
1252
1292
// will have one less than this number.
1253
1293
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 );
1255
1295
1256
1296
if (nseg_to_follow < 2 || nseg_to_follow > 128 ) {
1257
1297
text_color_set (DW_COLOR_ERROR );
@@ -1262,8 +1302,8 @@ void dl_data_request (dlq_item_t *E)
1262
1302
return ;
1263
1303
}
1264
1304
1265
- orig_offset = 0 ;
1266
- remaining_len = E -> txdata -> len ;
1305
+ int orig_offset = 0 ;
1306
+ int remaining_len = E -> txdata -> len ;
1267
1307
1268
1308
// First segment.
1269
1309
0 commit comments