1
1
mirror of https://github.com/systemd/systemd-stable.git synced 2025-01-18 06:03:42 +03:00

dhcp: fix potential buffer overflow

Fixes a bug introduced by 324f818781a250b60f2fcfa74ff1c9101d2d1315.

This also renames several macros for DHCP packet size.

(cherry picked from commit 4473cd7f61b9eb0860f2daab81491ad2145d554b)
(cherry picked from commit 037b1a8acc50cbeeebb82f95594a4909375577c2)
This commit is contained in:
Yu Watanabe 2022-08-06 13:05:59 +09:00 committed by Zbigniew Jędrzejewski-Szmek
parent 647c44c21a
commit 887837a5a9
3 changed files with 12 additions and 12 deletions

View File

@ -43,9 +43,10 @@ typedef struct DHCPPacket DHCPPacket;
#define DHCP_IP_SIZE (int32_t)(sizeof(struct iphdr)) #define DHCP_IP_SIZE (int32_t)(sizeof(struct iphdr))
#define DHCP_IP_UDP_SIZE (int32_t)(sizeof(struct udphdr) + DHCP_IP_SIZE) #define DHCP_IP_UDP_SIZE (int32_t)(sizeof(struct udphdr) + DHCP_IP_SIZE)
#define DHCP_MESSAGE_SIZE (int32_t)(sizeof(DHCPMessage)) #define DHCP_HEADER_SIZE (int32_t)(sizeof(DHCPMessage))
#define DHCP_DEFAULT_MIN_SIZE 576 /* the minimum internet hosts must be able to receive */ #define DHCP_MIN_MESSAGE_SIZE 576 /* the minimum internet hosts must be able to receive, see RFC 2132 Section 9.10 */
#define DHCP_MIN_OPTIONS_SIZE (DHCP_DEFAULT_MIN_SIZE - DHCP_IP_UDP_SIZE - DHCP_MESSAGE_SIZE) #define DHCP_MIN_OPTIONS_SIZE (DHCP_MIN_MESSAGE_SIZE - DHCP_HEADER_SIZE)
#define DHCP_MIN_PACKET_SIZE (DHCP_MIN_MESSAGE_SIZE + DHCP_IP_UDP_SIZE)
#define DHCP_MAGIC_COOKIE (uint32_t)(0x63825363) #define DHCP_MAGIC_COOKIE (uint32_t)(0x63825363)
enum { enum {

View File

@ -645,7 +645,7 @@ int sd_dhcp_client_set_client_port(
int sd_dhcp_client_set_mtu(sd_dhcp_client *client, uint32_t mtu) { int sd_dhcp_client_set_mtu(sd_dhcp_client *client, uint32_t mtu) {
assert_return(client, -EINVAL); assert_return(client, -EINVAL);
assert_return(mtu >= DHCP_DEFAULT_MIN_SIZE, -ERANGE); assert_return(mtu >= DHCP_MIN_PACKET_SIZE, -ERANGE);
client->mtu = mtu; client->mtu = mtu;
@ -812,7 +812,6 @@ static int client_message_init(
_cleanup_free_ DHCPPacket *packet = NULL; _cleanup_free_ DHCPPacket *packet = NULL;
size_t optlen, optoffset, size; size_t optlen, optoffset, size;
be16_t max_size;
usec_t time_now; usec_t time_now;
uint16_t secs; uint16_t secs;
int r; int r;
@ -963,9 +962,9 @@ static int client_message_init(
*/ */
/* RFC7844 section 3: /* RFC7844 section 3:
SHOULD NOT contain any other option. */ SHOULD NOT contain any other option. */
if (!client->anonymize && type != DHCP_RELEASE) { if (!client->anonymize && IN_SET(type, DHCP_DISCOVER, DHCP_REQUEST)) {
max_size = htobe16(size); be16_t max_size = htobe16(MIN(client->mtu - DHCP_IP_UDP_SIZE, (uint32_t) UINT16_MAX));
r = dhcp_option_append(&packet->dhcp, client->mtu, &optoffset, 0, r = dhcp_option_append(&packet->dhcp, optlen, &optoffset, 0,
SD_DHCP_OPTION_MAXIMUM_MESSAGE_SIZE, SD_DHCP_OPTION_MAXIMUM_MESSAGE_SIZE,
2, &max_size); 2, &max_size);
if (r < 0) if (r < 0)
@ -2279,7 +2278,7 @@ int sd_dhcp_client_new(sd_dhcp_client **ret, int anonymize) {
.state = DHCP_STATE_INIT, .state = DHCP_STATE_INIT,
.ifindex = -1, .ifindex = -1,
.fd = -1, .fd = -1,
.mtu = DHCP_DEFAULT_MIN_SIZE, .mtu = DHCP_MIN_PACKET_SIZE,
.port = DHCP_PORT_CLIENT, .port = DHCP_PORT_CLIENT,
.anonymize = !!anonymize, .anonymize = !!anonymize,
.max_attempts = UINT64_MAX, .max_attempts = UINT64_MAX,

View File

@ -712,9 +712,9 @@ int dhcp_lease_parse_options(uint8_t code, uint8_t len, const void *option, void
r = lease_parse_u16(option, len, &lease->mtu, 68); r = lease_parse_u16(option, len, &lease->mtu, 68);
if (r < 0) if (r < 0)
log_debug_errno(r, "Failed to parse MTU, ignoring: %m"); log_debug_errno(r, "Failed to parse MTU, ignoring: %m");
if (lease->mtu < DHCP_DEFAULT_MIN_SIZE) { if (lease->mtu < DHCP_MIN_PACKET_SIZE) {
log_debug("MTU value of %" PRIu16 " too small. Using default MTU value of %d instead.", lease->mtu, DHCP_DEFAULT_MIN_SIZE); log_debug("MTU value of %" PRIu16 " too small. Using default MTU value of %d instead.", lease->mtu, DHCP_MIN_PACKET_SIZE);
lease->mtu = DHCP_DEFAULT_MIN_SIZE; lease->mtu = DHCP_MIN_PACKET_SIZE;
} }
break; break;