mirror of
https://github.com/lwip-tcpip/lwip.git
synced 2025-03-28 19:21:20 +00:00
lwip_sendmsg: implement EMSGSIZE checks
This commit is contained in:
parent
c9d0192b4a
commit
c797222407
@ -1134,14 +1134,16 @@ lwip_sendmsg(int s, const struct msghdr *msg, int flags)
|
|||||||
|
|
||||||
LWIP_ERROR("lwip_sendmsg: invalid msghdr", msg != NULL,
|
LWIP_ERROR("lwip_sendmsg: invalid msghdr", msg != NULL,
|
||||||
sock_set_errno(sock, err_to_errno(ERR_ARG)); return -1;);
|
sock_set_errno(sock, err_to_errno(ERR_ARG)); return -1;);
|
||||||
LWIP_ERROR("lwip_sendmsg: maximum iovs exceeded", (msg->msg_iovlen <= IOV_MAX),
|
LWIP_ERROR("lwip_sendmsg: invalid msghdr iov", msg->msg_iov != NULL,
|
||||||
sock_set_errno(sock, err_to_errno(ERR_ARG)); return -1;);
|
sock_set_errno(sock, err_to_errno(ERR_ARG)); return -1;);
|
||||||
|
LWIP_ERROR("lwip_sendmsg: maximum iovs exceeded", (msg->msg_iovlen > 0) && (msg->msg_iovlen <= IOV_MAX),
|
||||||
|
sock_set_errno(sock, EMSGSIZE); return -1;);
|
||||||
|
LWIP_ERROR("lwip_sendmsg: unsupported flags", ((flags == 0) || (flags == MSG_NOSIGNAL)),
|
||||||
|
sock_set_errno(sock, EOPNOTSUPP); return -1;);
|
||||||
|
|
||||||
LWIP_UNUSED_ARG(msg->msg_control);
|
LWIP_UNUSED_ARG(msg->msg_control);
|
||||||
LWIP_UNUSED_ARG(msg->msg_controllen);
|
LWIP_UNUSED_ARG(msg->msg_controllen);
|
||||||
LWIP_UNUSED_ARG(msg->msg_flags);
|
LWIP_UNUSED_ARG(msg->msg_flags);
|
||||||
LWIP_ERROR("lwip_sendmsg: invalid msghdr iov", (msg->msg_iov != NULL && msg->msg_iovlen != 0),
|
|
||||||
sock_set_errno(sock, err_to_errno(ERR_ARG)); return -1;);
|
|
||||||
|
|
||||||
if (NETCONNTYPE_GROUP(netconn_type(sock->conn)) == NETCONN_TCP) {
|
if (NETCONNTYPE_GROUP(netconn_type(sock->conn)) == NETCONN_TCP) {
|
||||||
#if LWIP_TCP
|
#if LWIP_TCP
|
||||||
@ -1180,6 +1182,14 @@ lwip_sendmsg(int s, const struct msghdr *msg, int flags)
|
|||||||
#if LWIP_NETIF_TX_SINGLE_PBUF
|
#if LWIP_NETIF_TX_SINGLE_PBUF
|
||||||
for (i = 0; i < msg->msg_iovlen; i++) {
|
for (i = 0; i < msg->msg_iovlen; i++) {
|
||||||
size += msg->msg_iov[i].iov_len;
|
size += msg->msg_iov[i].iov_len;
|
||||||
|
if ((msg->msg_iov[i].iov_len > INT_MAX) || (size < (int)msg->msg_iov[i].iov_len)) {
|
||||||
|
/* overflow */
|
||||||
|
goto sendmsg_emsgsize;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (size > 0xFFFF) {
|
||||||
|
/* overflow */
|
||||||
|
goto sendmsg_emsgsize;
|
||||||
}
|
}
|
||||||
/* Allocate a new netbuf and copy the data into it. */
|
/* Allocate a new netbuf and copy the data into it. */
|
||||||
if (netbuf_alloc(&chain_buf, (u16_t)size) == NULL) {
|
if (netbuf_alloc(&chain_buf, (u16_t)size) == NULL) {
|
||||||
@ -1204,19 +1214,28 @@ lwip_sendmsg(int s, const struct msghdr *msg, int flags)
|
|||||||
/* create a chained netbuf from the IO vectors. NOTE: we assemble a pbuf chain
|
/* create a chained netbuf from the IO vectors. NOTE: we assemble a pbuf chain
|
||||||
manually to avoid having to allocate, chain, and delete a netbuf for each iov */
|
manually to avoid having to allocate, chain, and delete a netbuf for each iov */
|
||||||
for (i = 0; i < msg->msg_iovlen; i++) {
|
for (i = 0; i < msg->msg_iovlen; i++) {
|
||||||
struct pbuf *p = pbuf_alloc(PBUF_TRANSPORT, 0, PBUF_REF);
|
struct pbuf *p;
|
||||||
|
if (msg->msg_iov[i].iov_len > 0xFFFF) {
|
||||||
|
/* overflow */
|
||||||
|
goto sendmsg_emsgsize;
|
||||||
|
}
|
||||||
|
p = pbuf_alloc(PBUF_TRANSPORT, 0, PBUF_REF);
|
||||||
if (p == NULL) {
|
if (p == NULL) {
|
||||||
err = ERR_MEM; /* let netbuf_delete() cleanup chain_buf */
|
err = ERR_MEM; /* let netbuf_delete() cleanup chain_buf */
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
p->payload = msg->msg_iov[i].iov_base;
|
p->payload = msg->msg_iov[i].iov_base;
|
||||||
LWIP_ASSERT("iov_len < u16_t", msg->msg_iov[i].iov_len <= 0xFFFF);
|
|
||||||
p->len = p->tot_len = (u16_t)msg->msg_iov[i].iov_len;
|
p->len = p->tot_len = (u16_t)msg->msg_iov[i].iov_len;
|
||||||
/* netbuf empty, add new pbuf */
|
/* netbuf empty, add new pbuf */
|
||||||
if (chain_buf.p == NULL) {
|
if (chain_buf.p == NULL) {
|
||||||
chain_buf.p = chain_buf.ptr = p;
|
chain_buf.p = chain_buf.ptr = p;
|
||||||
/* add pbuf to existing pbuf chain */
|
/* add pbuf to existing pbuf chain */
|
||||||
} else {
|
} else {
|
||||||
|
if (chain_buf.p->tot_len + p->len > 0xffff) {
|
||||||
|
/* overflow */
|
||||||
|
pbuf_free(p);
|
||||||
|
goto sendmsg_emsgsize;
|
||||||
|
}
|
||||||
pbuf_cat(chain_buf.p, p);
|
pbuf_cat(chain_buf.p, p);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1245,6 +1264,11 @@ lwip_sendmsg(int s, const struct msghdr *msg, int flags)
|
|||||||
sock_set_errno(sock, err_to_errno(err));
|
sock_set_errno(sock, err_to_errno(err));
|
||||||
done_socket(sock);
|
done_socket(sock);
|
||||||
return (err == ERR_OK ? size : -1);
|
return (err == ERR_OK ? size : -1);
|
||||||
|
sendmsg_emsgsize:
|
||||||
|
sock_set_errno(sock, EMSGSIZE);
|
||||||
|
netbuf_free(&chain_buf);
|
||||||
|
done_socket(sock);
|
||||||
|
return -1;
|
||||||
}
|
}
|
||||||
#else /* LWIP_UDP || LWIP_RAW */
|
#else /* LWIP_UDP || LWIP_RAW */
|
||||||
sock_set_errno(sock, err_to_errno(ERR_ARG));
|
sock_set_errno(sock, err_to_errno(ERR_ARG));
|
||||||
|
Loading…
x
Reference in New Issue
Block a user