TCP listening support
This commit is contained in:
parent
f4e6e2909b
commit
c410299b05
2 changed files with 308 additions and 147 deletions
|
@ -74,6 +74,19 @@ struct grub_net_tcp_socket
|
|||
grub_priority_queue_t pq;
|
||||
};
|
||||
|
||||
struct grub_net_tcp_listen
|
||||
{
|
||||
struct grub_net_tcp_listen *next;
|
||||
|
||||
grub_uint16_t port;
|
||||
const struct grub_net_network_level_interface *inf;
|
||||
|
||||
grub_err_t (*listen_hook) (grub_net_tcp_listen_t listen,
|
||||
grub_net_tcp_socket_t sock,
|
||||
void *data);
|
||||
void *hook_data;
|
||||
};
|
||||
|
||||
struct tcphdr
|
||||
{
|
||||
grub_uint16_t src;
|
||||
|
@ -95,9 +108,38 @@ struct tcp_pseudohdr
|
|||
grub_uint16_t tcp_length;
|
||||
} __attribute__ ((packed));
|
||||
|
||||
struct grub_net_tcp_socket *tcp_sockets;
|
||||
static struct grub_net_tcp_socket *tcp_sockets;
|
||||
static struct grub_net_tcp_listen *tcp_listens;
|
||||
|
||||
#define FOR_TCP_SOCKETS(var) for (var = tcp_sockets; var; var = var->next)
|
||||
#define FOR_TCP_SOCKETS(var) FOR_LIST_ELEMENTS (var, tcp_sockets)
|
||||
#define FOR_TCP_LISTENS(var) FOR_LIST_ELEMENTS (var, tcp_listens)
|
||||
|
||||
grub_net_tcp_listen_t
|
||||
grub_net_tcp_listen (grub_uint16_t port,
|
||||
const struct grub_net_network_level_interface *inf,
|
||||
grub_err_t (*listen_hook) (grub_net_tcp_listen_t listen,
|
||||
grub_net_tcp_socket_t sock,
|
||||
void *data),
|
||||
void *hook_data)
|
||||
{
|
||||
grub_net_tcp_listen_t ret;
|
||||
ret = grub_malloc (sizeof (*ret));
|
||||
if (!ret)
|
||||
return NULL;
|
||||
ret->listen_hook = listen_hook;
|
||||
ret->hook_data = hook_data;
|
||||
ret->port = port;
|
||||
ret->inf = inf;
|
||||
grub_list_push (GRUB_AS_LIST_P (&tcp_listens), GRUB_AS_LIST (ret));
|
||||
return ret;
|
||||
}
|
||||
|
||||
void
|
||||
grub_net_tcp_stop_listen (grub_net_tcp_listen_t listen)
|
||||
{
|
||||
grub_list_remove (GRUB_AS_LIST_P (&tcp_listens),
|
||||
GRUB_AS_LIST (listen));
|
||||
}
|
||||
|
||||
static inline void
|
||||
tcp_socket_register (grub_net_tcp_socket_t sock)
|
||||
|
@ -106,6 +148,25 @@ tcp_socket_register (grub_net_tcp_socket_t sock)
|
|||
GRUB_AS_LIST (sock));
|
||||
}
|
||||
|
||||
static void
|
||||
error (grub_net_tcp_socket_t sock)
|
||||
{
|
||||
struct unacked *unack, *next;
|
||||
|
||||
if (sock->established && sock->error_hook)
|
||||
sock->error_hook (sock, sock->hook_data);
|
||||
|
||||
for (unack = sock->unack_first; unack; unack = next)
|
||||
{
|
||||
next = unack->next;
|
||||
grub_netbuff_free (unack->nb);
|
||||
grub_free (unack);
|
||||
}
|
||||
|
||||
sock->unack_first = NULL;
|
||||
sock->unack_last = NULL;
|
||||
}
|
||||
|
||||
static grub_err_t
|
||||
tcp_send (struct grub_net_buff *nb, grub_net_tcp_socket_t socket)
|
||||
{
|
||||
|
@ -164,10 +225,13 @@ grub_net_tcp_close (grub_net_tcp_socket_t sock)
|
|||
|
||||
sock->i_closed = 1;
|
||||
|
||||
nb_fin = grub_netbuff_alloc (sizeof (*tcph_fin) + 128);
|
||||
nb_fin = grub_netbuff_alloc (sizeof (*tcph_fin)
|
||||
+ GRUB_NET_OUR_IPV4_HEADER_SIZE
|
||||
+ GRUB_NET_MAX_LINK_HEADER_SIZE);
|
||||
if (!nb_fin)
|
||||
return;
|
||||
err = grub_netbuff_reserve (nb_fin, 128);
|
||||
err = grub_netbuff_reserve (nb_fin, GRUB_NET_OUR_IPV4_HEADER_SIZE
|
||||
+ GRUB_NET_MAX_LINK_HEADER_SIZE);
|
||||
if (err)
|
||||
{
|
||||
grub_netbuff_free (nb_fin);
|
||||
|
@ -261,9 +325,7 @@ grub_net_tcp_retransmit (void)
|
|||
|
||||
if (unack->try_count > TCP_RETRANSMISSION_COUNT)
|
||||
{
|
||||
if (sock->error_hook)
|
||||
sock->error_hook (sock, sock->hook_data);
|
||||
grub_net_tcp_close (sock);
|
||||
error (sock);
|
||||
break;
|
||||
}
|
||||
unack->try_count++;
|
||||
|
@ -332,6 +394,54 @@ destroy_pq (grub_net_tcp_socket_t sock)
|
|||
grub_priority_queue_destroy (sock->pq);
|
||||
}
|
||||
|
||||
grub_err_t
|
||||
grub_net_tcp_accept (grub_net_tcp_socket_t sock,
|
||||
grub_err_t (*recv_hook) (grub_net_tcp_socket_t sock,
|
||||
struct grub_net_buff *nb,
|
||||
void *data),
|
||||
void (*error_hook) (grub_net_tcp_socket_t sock,
|
||||
void *data),
|
||||
void *hook_data)
|
||||
{
|
||||
struct grub_net_buff *nb_ack;
|
||||
struct tcphdr *tcph;
|
||||
grub_err_t err;
|
||||
|
||||
sock->recv_hook = recv_hook;
|
||||
sock->error_hook = error_hook;
|
||||
sock->hook_data = hook_data;
|
||||
nb_ack = grub_netbuff_alloc (sizeof (*tcph)
|
||||
+ GRUB_NET_OUR_IPV4_HEADER_SIZE
|
||||
+ GRUB_NET_MAX_LINK_HEADER_SIZE);
|
||||
if (!nb_ack)
|
||||
return grub_errno;
|
||||
err = grub_netbuff_reserve (nb_ack, GRUB_NET_OUR_IPV4_HEADER_SIZE
|
||||
+ GRUB_NET_MAX_LINK_HEADER_SIZE);
|
||||
if (err)
|
||||
{
|
||||
grub_netbuff_free (nb_ack);
|
||||
return err;
|
||||
}
|
||||
|
||||
err = grub_netbuff_put (nb_ack, sizeof (*tcph));
|
||||
if (err)
|
||||
{
|
||||
grub_netbuff_free (nb_ack);
|
||||
return err;
|
||||
}
|
||||
tcph = (void *) nb_ack->data;
|
||||
tcph->ack = grub_cpu_to_be32 (sock->their_cur_seq);
|
||||
tcph->flags = grub_cpu_to_be16 ((5 << 12) | TCP_SYN | TCP_ACK);
|
||||
tcph->window = grub_cpu_to_be16 (sock->my_window);
|
||||
tcph->urgent = 0;
|
||||
tcp_socket_register (sock);
|
||||
err = tcp_send (nb_ack, sock);
|
||||
if (err)
|
||||
return err;
|
||||
sock->my_cur_seq++;
|
||||
return GRUB_ERR_NONE;
|
||||
}
|
||||
|
||||
grub_net_tcp_socket_t
|
||||
grub_net_tcp_open (char *server,
|
||||
grub_uint16_t out_port,
|
||||
|
@ -545,12 +655,12 @@ grub_net_recv_tcp_packet (struct grub_net_buff *nb,
|
|||
|
||||
FOR_TCP_SOCKETS (sock)
|
||||
{
|
||||
if (grub_be_to_cpu16 (tcph->dst) == sock->in_port
|
||||
if (!(grub_be_to_cpu16 (tcph->dst) == sock->in_port
|
||||
&& grub_be_to_cpu16 (tcph->src) == sock->out_port
|
||||
&& inf == sock->inf
|
||||
&& source->type == GRUB_NET_NETWORK_LEVEL_PROTOCOL_IPV4
|
||||
&& source->ipv4 == sock->out_nla.ipv4)
|
||||
{
|
||||
&& source->ipv4 == sock->out_nla.ipv4))
|
||||
continue;
|
||||
if (tcph->checksum)
|
||||
{
|
||||
grub_uint16_t chk, expected;
|
||||
|
@ -582,17 +692,9 @@ grub_net_recv_tcp_packet (struct grub_net_buff *nb,
|
|||
|
||||
if (grub_be_to_cpu16 (tcph->flags) & TCP_RST)
|
||||
{
|
||||
struct unacked *unack, *next;
|
||||
sock->reseted = 1;
|
||||
for (unack = sock->unack_first; unack; unack = next)
|
||||
{
|
||||
next = unack->next;
|
||||
grub_netbuff_free (unack->nb);
|
||||
grub_free (unack);
|
||||
}
|
||||
|
||||
sock->unack_first = NULL;
|
||||
sock->unack_last = NULL;
|
||||
error (sock);
|
||||
|
||||
grub_netbuff_free (nb);
|
||||
|
||||
|
@ -697,6 +799,42 @@ grub_net_recv_tcp_packet (struct grub_net_buff *nb,
|
|||
|
||||
return GRUB_ERR_NONE;
|
||||
}
|
||||
if (grub_be_to_cpu16 (tcph->flags) & TCP_SYN)
|
||||
{
|
||||
grub_net_tcp_listen_t listen;
|
||||
|
||||
FOR_TCP_LISTENS (listen)
|
||||
{
|
||||
if (!(grub_be_to_cpu16 (tcph->dst) == listen->port
|
||||
&& (inf == listen->inf || listen->inf == NULL)))
|
||||
continue;
|
||||
sock = grub_zalloc (sizeof (*sock));
|
||||
if (sock == NULL)
|
||||
return grub_errno;
|
||||
|
||||
sock->out_port = grub_be_to_cpu16 (tcph->src);
|
||||
sock->in_port = grub_be_to_cpu16 (tcph->dst);
|
||||
sock->inf = inf;
|
||||
sock->out_nla = *source;
|
||||
sock->their_start_seq = grub_be_to_cpu32 (tcph->seqnr);
|
||||
sock->their_cur_seq = sock->their_start_seq + 1;
|
||||
sock->my_cur_seq = sock->my_start_seq = grub_get_time_ms ();
|
||||
sock->my_window = 8192;
|
||||
|
||||
sock->pq = grub_priority_queue_new (sizeof (struct grub_net_buff *),
|
||||
cmp);
|
||||
if (!sock->pq)
|
||||
{
|
||||
grub_netbuff_free (nb);
|
||||
return grub_errno;
|
||||
}
|
||||
|
||||
err = listen->listen_hook (listen, sock, listen->hook_data);
|
||||
|
||||
grub_netbuff_free (nb);
|
||||
return err;
|
||||
|
||||
}
|
||||
}
|
||||
grub_netbuff_free (nb);
|
||||
return GRUB_ERR_NONE;
|
||||
|
|
|
@ -24,6 +24,9 @@
|
|||
struct grub_net_tcp_socket;
|
||||
typedef struct grub_net_tcp_socket *grub_net_tcp_socket_t;
|
||||
|
||||
struct grub_net_tcp_listen;
|
||||
typedef struct grub_net_tcp_listen *grub_net_tcp_listen_t;
|
||||
|
||||
grub_net_tcp_socket_t
|
||||
grub_net_tcp_open (char *server,
|
||||
grub_uint16_t out_port,
|
||||
|
@ -34,6 +37,17 @@ grub_net_tcp_open (char *server,
|
|||
void *data),
|
||||
void *hook_data);
|
||||
|
||||
grub_net_tcp_listen_t
|
||||
grub_net_tcp_listen (grub_uint16_t port,
|
||||
const struct grub_net_network_level_interface *inf,
|
||||
grub_err_t (*listen_hook) (grub_net_tcp_listen_t listen,
|
||||
grub_net_tcp_socket_t sock,
|
||||
void *data),
|
||||
void *hook_data);
|
||||
|
||||
void
|
||||
grub_net_tcp_stop_listen (grub_net_tcp_listen_t listen);
|
||||
|
||||
grub_err_t
|
||||
grub_net_send_tcp_packet (const grub_net_tcp_socket_t socket,
|
||||
struct grub_net_buff *nb,
|
||||
|
@ -42,4 +56,13 @@ grub_net_send_tcp_packet (const grub_net_tcp_socket_t socket,
|
|||
void
|
||||
grub_net_tcp_close (grub_net_tcp_socket_t sock);
|
||||
|
||||
grub_err_t
|
||||
grub_net_tcp_accept (grub_net_tcp_socket_t sock,
|
||||
grub_err_t (*recv_hook) (grub_net_tcp_socket_t sock,
|
||||
struct grub_net_buff *nb,
|
||||
void *data),
|
||||
void (*error_hook) (grub_net_tcp_socket_t sock,
|
||||
void *data),
|
||||
void *hook_data);
|
||||
|
||||
#endif
|
||||
|
|
Loading…
Reference in a new issue