Skip to content

Commit 41a85d8

Browse files
committed
Issue 169 - Fix AGW protocol 'Y' command.
1 parent e1a4716 commit 41a85d8

File tree

5 files changed

+233
-31
lines changed

5 files changed

+233
-31
lines changed

ax25_link.c

+75
Original file line numberDiff line numberDiff line change
@@ -887,6 +887,7 @@ static ax25_dlsm_t *get_link_handle (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LE
887887
//
888888
// dl_connect_request
889889
// dl_disconnect_request
890+
// dl_outstanding_frames_request - (mine) Ask about outgoing queue for a link.
890891
// dl_data_request - send connected data
891892
// dl_unit_data_request - not implemented. APRS & KISS bypass this
892893
// dl_flow_off - not implemented. Not in AGW API.
@@ -1505,6 +1506,80 @@ void dl_unregister_callsign (dlq_item_t *E)
15051506

15061507

15071508

1509+
/*------------------------------------------------------------------------------
1510+
*
1511+
* Name: dl_outstanding_frames_request
1512+
*
1513+
* Purpose: Client app wants to know how many frames are still on their way
1514+
* to other station. This is handy for flow control. We would like
1515+
* to keep the pipeline filled sufficiently to take advantage of a
1516+
* large window size (MAXFRAMES). It is also good to know that the
1517+
* the last packet sent was actually received before we commence
1518+
* the disconnect.
1519+
*
1520+
* Inputs: E - Event from the queue.
1521+
* The caller will free it.
1522+
*
1523+
* Outputs: This gets back to the AGW server which sends the 'Y' reply.
1524+
*
1525+
* Description: This is the sum of:
1526+
* - Incoming connected data, from application still in the queue.
1527+
* - I frames which have been transmitted but not yet acknowleged.
1528+
*
1529+
*------------------------------------------------------------------------------*/
1530+
1531+
void dl_outstanding_frames_request (dlq_item_t *E)
1532+
{
1533+
ax25_dlsm_t *S;
1534+
int ok_to_create = 0; // must exist already.
1535+
1536+
1537+
if (s_debug_client_app) {
1538+
text_color_set(DW_COLOR_DEBUG);
1539+
dw_printf ("dl_outstanding_frames_request ( to %s )\n", E->addrs[PEERCALL]);
1540+
}
1541+
1542+
S = get_link_handle (E->addrs, E->num_addr, E->chan, E->client, ok_to_create);
1543+
1544+
if (S == NULL) {
1545+
text_color_set(DW_COLOR_ERROR);
1546+
dw_printf ("Can't get outstanding frames for %s -> %s, chan %d\n", E->addrs[OWNCALL], E->addrs[PEERCALL], E->chan);
1547+
server_outstanding_frames_reply (E->chan, E->client, E->addrs[OWNCALL], E->addrs[PEERCALL], 0);
1548+
return;
1549+
}
1550+
1551+
// Add up these
1552+
//
1553+
// cdata_t *i_frame_queue; // Connected data from client which has not been transmitted yet.
1554+
// // Linked list.
1555+
// // The name is misleading because these are just blocks of
1556+
// // data, not "I frames" at this point. The name comes from
1557+
// // the protocol specification.
1558+
//
1559+
// cdata_t *txdata_by_ns[128]; // Data which has already been transmitted.
1560+
// // Indexed by N(S) in case it gets lost and needs to be sent again.
1561+
// // Cleared out when we get ACK for it.
1562+
1563+
int count1 = 0;
1564+
cdata_t *incoming;
1565+
for (incoming = S->i_frame_queue; incoming != NULL; incoming = incoming->next) {
1566+
count1++;
1567+
}
1568+
1569+
int count2 = 0;
1570+
int k;
1571+
for (k = 0; k < S->modulo; k++) {
1572+
if (S->txdata_by_ns[k] != NULL) {
1573+
count2++;
1574+
}
1575+
}
1576+
1577+
server_outstanding_frames_reply (S->chan, S->client, S->addrs[OWNCALL], S->addrs[PEERCALL], count1 + count2);
1578+
1579+
} // end dl_outstanding_frames_request
1580+
1581+
1582+
15081583
/*------------------------------------------------------------------------------
15091584
*
15101585
* Name: dl_client_cleanup

dlq.c

+64-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
//
22
// This file is part of Dire Wolf, an amateur radio packet TNC.
33
//
4-
// Copyright (C) 2014, 2015, 2016 John Langner, WB2OSZ
4+
// Copyright (C) 2014, 2015, 2016, 2018 John Langner, WB2OSZ
55
//
66
// This program is free software: you can redistribute it and/or modify
77
// it under the terms of the GNU General Public License as published by
@@ -553,7 +553,69 @@ void dlq_disconnect_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int
553553

554554
append_to_queue (pnew);
555555

556-
} /* end dlq_connect_request */
556+
} /* end dlq_disconnect_request */
557+
558+
559+
560+
/*-------------------------------------------------------------------
561+
*
562+
* Name: dlq_outstanding_frames_request
563+
*
564+
* Purpose: Client application wants to know number of outstanding information
565+
* frames supplied, supplied by the client, that have not yet been
566+
* delivered to the remote station.
567+
*
568+
* Inputs: addrs - Source (owncall), destination (peercall)
569+
*
570+
* num_addr - Number of addresses. Should be 2.
571+
* If more they will be ignored.
572+
*
573+
* chan - Channel, 0 is first.
574+
*
575+
* client - Client application instance. We could have multiple
576+
* applications, all on the same channel, connecting
577+
* to different stations. We need to know which one
578+
* should get the results.
579+
*
580+
* Outputs: Request is appended to queue for processing by
581+
* the data link state machine.
582+
*
583+
* Description: The data link state machine will count up all information frames
584+
* for the given source(mycall) / destination(remote) / channel link.
585+
* A 'Y' response will be sent back to the client application.
586+
*
587+
*--------------------------------------------------------------------*/
588+
589+
void dlq_outstanding_frames_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, int chan, int client)
590+
{
591+
struct dlq_item_s *pnew;
592+
#if DEBUG
593+
text_color_set(DW_COLOR_DEBUG);
594+
dw_printf ("dlq_outstanding_frames_request (...)\n");
595+
#endif
596+
597+
assert (chan >= 0 && chan < MAX_CHANS);
598+
599+
/* Allocate a new queue item. */
600+
601+
pnew = (struct dlq_item_s *) calloc (sizeof(struct dlq_item_s), 1);
602+
s_new_count++;
603+
604+
pnew->type = DLQ_OUTSTANDING_FRAMES_REQUEST;
605+
pnew->chan = chan;
606+
memcpy (pnew->addrs, addrs, sizeof(pnew->addrs));
607+
pnew->num_addr = num_addr;
608+
pnew->client = client;
609+
610+
/* Put it into queue. */
611+
612+
append_to_queue (pnew);
613+
614+
} /* end dlq_outstanding_frames_request */
615+
616+
617+
618+
557619

558620

559621
/*-------------------------------------------------------------------

dlq.h

+3-1
Original file line numberDiff line numberDiff line change
@@ -35,7 +35,7 @@ typedef struct cdata_s {
3535

3636
/* Types of things that can be in queue. */
3737

38-
typedef enum dlq_type_e {DLQ_REC_FRAME, DLQ_CONNECT_REQUEST, DLQ_DISCONNECT_REQUEST, DLQ_XMIT_DATA_REQUEST, DLQ_REGISTER_CALLSIGN, DLQ_UNREGISTER_CALLSIGN, DLQ_CHANNEL_BUSY, DLQ_SEIZE_CONFIRM, DLQ_CLIENT_CLEANUP} dlq_type_t;
38+
typedef enum dlq_type_e {DLQ_REC_FRAME, DLQ_CONNECT_REQUEST, DLQ_DISCONNECT_REQUEST, DLQ_XMIT_DATA_REQUEST, DLQ_REGISTER_CALLSIGN, DLQ_UNREGISTER_CALLSIGN, DLQ_OUTSTANDING_FRAMES_REQUEST, DLQ_CHANNEL_BUSY, DLQ_SEIZE_CONFIRM, DLQ_CLIENT_CLEANUP} dlq_type_t;
3939

4040

4141
/* A queue item. */
@@ -108,6 +108,8 @@ void dlq_connect_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num
108108

109109
void dlq_disconnect_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, int chan, int client);
110110

111+
void dlq_outstanding_frames_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, int chan, int client);
112+
111113
void dlq_xmit_data_request (char addrs[AX25_MAX_ADDRS][AX25_MAX_ADDR_LEN], int num_addr, int chan, int client, int pid, char *xdata_ptr, int xdata_len);
112114

113115
void dlq_register_callsign (char addr[AX25_MAX_ADDR_LEN], int chan, int client);

server.c

+89-28
Original file line numberDiff line numberDiff line change
@@ -136,11 +136,7 @@
136136
#include <sys/ioctl.h>
137137
#include <sys/socket.h>
138138
#include <netinet/in.h>
139-
#ifdef __OpenBSD__
140139
#include <errno.h>
141-
#else
142-
#include <sys/errno.h>
143-
#endif
144140
#endif
145141

146142
#include <unistd.h>
@@ -1121,6 +1117,50 @@ void server_rec_conn_data (int chan, int client, char *remote_call, char *own_ca
11211117
} /* end server_rec_conn_data */
11221118

11231119

1120+
/*-------------------------------------------------------------------
1121+
*
1122+
* Name: server_outstanding_frames_reply
1123+
*
1124+
* Purpose: Send 'Y' Outstanding frames for connected data to the application.
1125+
*
1126+
* Inputs: chan - Which radio channel.
1127+
*
1128+
* client - Which one of potentially several clients.
1129+
*
1130+
* own_call - Callsign[-ssid] of my end.
1131+
*
1132+
* remote_call - Callsign[-ssid] of remote station.
1133+
*
1134+
* count - Number of frames sent from the application but
1135+
* not yet received by the other station.
1136+
*
1137+
*--------------------------------------------------------------------*/
1138+
1139+
void server_outstanding_frames_reply (int chan, int client, char *own_call, char *remote_call, int count)
1140+
{
1141+
1142+
struct {
1143+
struct agwpe_s hdr;
1144+
int count_NETLE;
1145+
} reply;
1146+
1147+
1148+
memset (&reply.hdr, 0, sizeof(reply.hdr));
1149+
1150+
reply.hdr.portx = chan;
1151+
reply.hdr.datakind = 'Y';
1152+
1153+
strlcpy (reply.hdr.call_from, own_call, sizeof(reply.hdr.call_from));
1154+
strlcpy (reply.hdr.call_to, remote_call, sizeof(reply.hdr.call_to));
1155+
1156+
reply.hdr.data_len_NETLE = host2netle(4);
1157+
reply.count_NETLE = host2netle(count);
1158+
1159+
send_to_client (client, &reply);
1160+
1161+
} /* end server_outstanding_frames_reply */
1162+
1163+
11241164
/*-------------------------------------------------------------------
11251165
*
11261166
* Name: read_from_socket
@@ -1220,8 +1260,9 @@ static THREAD_F cmd_listen_thread (void *arg)
12201260
struct {
12211261
struct agwpe_s hdr; /* Command header. */
12221262

1223-
char data[512]; /* Additional data used by some commands. */
1263+
char data[AX25_MAX_PACKET_LEN]; /* Additional data used by some commands. */
12241264
/* Maximum for 'V': 1 + 8*10 + 256 */
1265+
/* Maximum for 'D': Info part length + 1 */
12251266
} cmd;
12261267

12271268
int client = (int)(long)arg;
@@ -1864,6 +1905,7 @@ static THREAD_F cmd_listen_thread (void *arg)
18641905

18651906
int n = 0;
18661907
if (cmd.hdr.portx >= 0 && cmd.hdr.portx < MAX_CHANS) {
1908+
// Count both normal and expedited in transmit queue for given channel.
18671909
n = tq_count (cmd.hdr.portx, -1, "", "", 0);
18681910
}
18691911
reply.data_NETLE = host2netle(n);
@@ -1874,34 +1916,53 @@ static THREAD_F cmd_listen_thread (void *arg)
18741916

18751917
case 'Y': /* How Many Outstanding frames wait for tx for a particular station */
18761918

1877-
/* Number of frames sitting in transmit queue for given channel, */
1878-
/* source (optional) and destination addresses. */
1879-
{
1880-
char source[AX25_MAX_ADDR_LEN];
1881-
char dest[AX25_MAX_ADDR_LEN];
1919+
// This is different than the above 'y' because this refers to a specific
1920+
// link in connected mode.
1921+
1922+
// This would be useful for a couple different purposes.
1923+
1924+
// When sending bulk data, we want to keep a fair amount queued up to take
1925+
// advantage of large window sizes (MAXFRAME, EMAXFRAME). On the other
1926+
// hand we don't want to get TOO far ahead when transferring a large file.
1927+
1928+
// Before disconnecting from another station, it would be good to know
1929+
// that it actually received the last message we sent. For this reason,
1930+
// I think it would be good for this to include information frames that were
1931+
// transmitted but not yet acknowleged.
1932+
// You could say that a particular frame is still waiting to be sent even
1933+
// if was already sent because it could be sent again if lost previously.
1934+
1935+
// The documentation is inconsistent about the address order.
1936+
// One place says "callfrom" is my callsign and "callto" is the other guy.
1937+
// That would make sense. We are asking about frames going to the other guy.
1938+
1939+
// But another place says it depends on who initiated the connection.
1940+
//
1941+
// "If we started the connection CallFrom=US and CallTo=THEM
1942+
// If the other end started the connection CallFrom=THEM and CallTo=US"
1943+
//
1944+
// The response description says nothing about the order; it just mentions two addresses.
1945+
// If you are writing a client or server application, the order would
1946+
// be clear but right here it could be either case.
1947+
//
1948+
// Another version of the documentation mentioned the source address being optional.
1949+
//
1950+
1951+
// The only way to get this information is from inside the data link state machine.
1952+
// We will send a request to it and the result coming out will be used to
1953+
// send the reply back to the client application.
18821954

1883-
struct {
1884-
struct agwpe_s hdr;
1885-
int data_NETLE; // Little endian order.
1886-
} reply;
1955+
{
18871956

1888-
strlcpy (source, cmd.hdr.call_from, sizeof(source));
1889-
strlcpy (dest, cmd.hdr.call_to, sizeof(dest));
1957+
char callsigns[2][AX25_MAX_ADDR_LEN];
1958+
const int num_calls = 2;
18901959

1891-
memset (&reply, 0, sizeof(reply));
1892-
reply.hdr.portx = cmd.hdr.portx; /* Reply with same port number, addresses. */
1893-
reply.hdr.datakind = 'Y';
1894-
strlcpy (reply.hdr.call_from, source, sizeof(reply.hdr.call_from));
1895-
strlcpy (reply.hdr.call_to, dest, sizeof(reply.hdr.call_to));
1896-
reply.hdr.data_len_NETLE = host2netle(4);
1960+
strlcpy (callsigns[AX25_SOURCE], cmd.hdr.call_from, sizeof(callsigns[AX25_SOURCE]));
1961+
strlcpy (callsigns[AX25_DESTINATION], cmd.hdr.call_to, sizeof(callsigns[AX25_SOURCE]));
18971962

1898-
int n = 0;
1899-
if (cmd.hdr.portx >= 0 && cmd.hdr.portx < MAX_CHANS) {
1900-
n = tq_count (cmd.hdr.portx, -1, source, dest, 0);
1901-
}
1902-
reply.data_NETLE = host2netle(n);
1963+
// Issue 169. Proper implementation for 'Y'.
1964+
dlq_outstanding_frames_request (callsigns, num_calls, cmd.hdr.portx, client);
19031965

1904-
send_to_client (client, &reply);
19051966
}
19061967
break;
19071968

server.h

+2
Original file line numberDiff line numberDiff line change
@@ -24,5 +24,7 @@ void server_link_terminated (int chan, int client, char *remote_call, char *own_
2424

2525
void server_rec_conn_data (int chan, int client, char *remote_call, char *own_call, int pid, char *data_ptr, int data_len);
2626

27+
void server_outstanding_frames_reply (int chan, int client, char *own_call, char *remote_call, int count);
28+
2729

2830
/* end server.h */

0 commit comments

Comments
 (0)