extmod/modlwip: Fix close and clean up of UDP and raw sockets.
The correct callback-deregister functions must be called dependent on the socket type, otherwise resources may not be freed correctly. Signed-off-by: Damien George <damien@micropython.org>
This commit is contained in:
parent
90d47ee34d
commit
8fcdb5490c
@ -1502,16 +1502,16 @@ STATIC mp_uint_t lwip_socket_ioctl(mp_obj_t self_in, mp_uint_t request, uintptr_
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Deregister callback (pcb.tcp is set to NULL below so must deregister now)
|
|
||||||
tcp_arg(socket->pcb.tcp, NULL);
|
|
||||||
tcp_err(socket->pcb.tcp, NULL);
|
|
||||||
tcp_recv(socket->pcb.tcp, NULL);
|
|
||||||
|
|
||||||
// Free any incoming buffers or connections that are stored
|
// Free any incoming buffers or connections that are stored
|
||||||
lwip_socket_free_incoming(socket);
|
lwip_socket_free_incoming(socket);
|
||||||
|
|
||||||
switch (socket->type) {
|
switch (socket->type) {
|
||||||
case MOD_NETWORK_SOCK_STREAM: {
|
case MOD_NETWORK_SOCK_STREAM: {
|
||||||
|
// Deregister callback (pcb.tcp is set to NULL below so must deregister now)
|
||||||
|
tcp_arg(socket->pcb.tcp, NULL);
|
||||||
|
tcp_err(socket->pcb.tcp, NULL);
|
||||||
|
tcp_recv(socket->pcb.tcp, NULL);
|
||||||
|
|
||||||
if (socket->pcb.tcp->state != LISTEN) {
|
if (socket->pcb.tcp->state != LISTEN) {
|
||||||
// Schedule a callback to abort the connection if it's not cleanly closed after
|
// Schedule a callback to abort the connection if it's not cleanly closed after
|
||||||
// the given timeout. The callback must be set before calling tcp_close since
|
// the given timeout. The callback must be set before calling tcp_close since
|
||||||
@ -1525,10 +1525,12 @@ STATIC mp_uint_t lwip_socket_ioctl(mp_obj_t self_in, mp_uint_t request, uintptr_
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case MOD_NETWORK_SOCK_DGRAM:
|
case MOD_NETWORK_SOCK_DGRAM:
|
||||||
|
udp_recv(socket->pcb.udp, NULL, NULL);
|
||||||
udp_remove(socket->pcb.udp);
|
udp_remove(socket->pcb.udp);
|
||||||
break;
|
break;
|
||||||
#if MICROPY_PY_LWIP_SOCK_RAW
|
#if MICROPY_PY_LWIP_SOCK_RAW
|
||||||
case MOD_NETWORK_SOCK_RAW:
|
case MOD_NETWORK_SOCK_RAW:
|
||||||
|
raw_recv(socket->pcb.raw, NULL, NULL);
|
||||||
raw_remove(socket->pcb.raw);
|
raw_remove(socket->pcb.raw);
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
|
35
tests/multi_net/udp_data.py
Normal file
35
tests/multi_net/udp_data.py
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
# Simple test of a UDP server and client transferring data
|
||||||
|
|
||||||
|
import socket
|
||||||
|
|
||||||
|
NUM_NEW_SOCKETS = 4
|
||||||
|
NUM_TRANSFERS = 4
|
||||||
|
PORT = 8000
|
||||||
|
|
||||||
|
# Server
|
||||||
|
def instance0():
|
||||||
|
multitest.globals(IP=multitest.get_network_ip())
|
||||||
|
multitest.next()
|
||||||
|
for i in range(NUM_NEW_SOCKETS):
|
||||||
|
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
|
||||||
|
s.bind(socket.getaddrinfo("0.0.0.0", PORT)[0][-1])
|
||||||
|
multitest.broadcast("server ready")
|
||||||
|
for j in range(NUM_TRANSFERS):
|
||||||
|
data, addr = s.recvfrom(1000)
|
||||||
|
print(data)
|
||||||
|
s.sendto(b"server to client %d %d" % (i, j), addr)
|
||||||
|
s.close()
|
||||||
|
|
||||||
|
|
||||||
|
# Client
|
||||||
|
def instance1():
|
||||||
|
multitest.next()
|
||||||
|
ai = socket.getaddrinfo(IP, PORT)[0][-1]
|
||||||
|
for i in range(NUM_NEW_SOCKETS):
|
||||||
|
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
|
||||||
|
multitest.wait("server ready")
|
||||||
|
for j in range(NUM_TRANSFERS):
|
||||||
|
s.sendto(b"client to server %d %d" % (i, j), ai)
|
||||||
|
data, addr = s.recvfrom(1000)
|
||||||
|
print(data)
|
||||||
|
s.close()
|
Loading…
x
Reference in New Issue
Block a user