136
136
#include <sys/ioctl.h>
137
137
#include <sys/socket.h>
138
138
#include <netinet/in.h>
139
- #ifdef __OpenBSD__
140
139
#include <errno.h>
141
- #else
142
- #include <sys/errno.h>
143
- #endif
144
140
#endif
145
141
146
142
#include <unistd.h>
@@ -1121,6 +1117,50 @@ void server_rec_conn_data (int chan, int client, char *remote_call, char *own_ca
1121
1117
} /* end server_rec_conn_data */
1122
1118
1123
1119
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
+
1124
1164
/*-------------------------------------------------------------------
1125
1165
*
1126
1166
* Name: read_from_socket
@@ -1220,8 +1260,9 @@ static THREAD_F cmd_listen_thread (void *arg)
1220
1260
struct {
1221
1261
struct agwpe_s hdr ; /* Command header. */
1222
1262
1223
- char data [512 ]; /* Additional data used by some commands. */
1263
+ char data [AX25_MAX_PACKET_LEN ]; /* Additional data used by some commands. */
1224
1264
/* Maximum for 'V': 1 + 8*10 + 256 */
1265
+ /* Maximum for 'D': Info part length + 1 */
1225
1266
} cmd ;
1226
1267
1227
1268
int client = (int )(long )arg ;
@@ -1864,6 +1905,7 @@ static THREAD_F cmd_listen_thread (void *arg)
1864
1905
1865
1906
int n = 0 ;
1866
1907
if (cmd .hdr .portx >= 0 && cmd .hdr .portx < MAX_CHANS ) {
1908
+ // Count both normal and expedited in transmit queue for given channel.
1867
1909
n = tq_count (cmd .hdr .portx , -1 , "" , "" , 0 );
1868
1910
}
1869
1911
reply .data_NETLE = host2netle (n );
@@ -1874,34 +1916,53 @@ static THREAD_F cmd_listen_thread (void *arg)
1874
1916
1875
1917
case 'Y' : /* How Many Outstanding frames wait for tx for a particular station */
1876
1918
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.
1882
1954
1883
- struct {
1884
- struct agwpe_s hdr ;
1885
- int data_NETLE ; // Little endian order.
1886
- } reply ;
1955
+ {
1887
1956
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 ;
1890
1959
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 ]));
1897
1962
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 );
1903
1965
1904
- send_to_client (client , & reply );
1905
1966
}
1906
1967
break ;
1907
1968
0 commit comments