mirror of
https://github.com/systemd/systemd-stable.git
synced 2025-01-18 06:03:42 +03:00
sd-dhcp-lease: fix reading unaligned memory
The destination address was read twice, one is for prefixlen, and other is for destination address itself. And for prefixlen, the address might be read from unaligned buffer. This also modernizes the code. (cherry picked from commit 7b868543072bb9073174a4ae46032fdb6eb24c92)
This commit is contained in:
parent
a5fc827b3a
commit
2b04d3b3fc
@ -468,41 +468,48 @@ static int lease_parse_sip_server(const uint8_t *option, size_t len, struct in_a
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int lease_parse_routes(
|
static int lease_parse_routes(
|
||||||
const uint8_t *option, size_t len,
|
const uint8_t *option,
|
||||||
struct sd_dhcp_route **routes, size_t *routes_size) {
|
size_t len,
|
||||||
|
struct sd_dhcp_route **routes,
|
||||||
|
size_t *routes_size) {
|
||||||
|
|
||||||
struct in_addr addr;
|
int r;
|
||||||
|
|
||||||
assert(option || len <= 0);
|
assert(option || len <= 0);
|
||||||
assert(routes);
|
assert(routes);
|
||||||
assert(routes_size);
|
assert(routes_size);
|
||||||
|
|
||||||
if (len <= 0)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
if (len % 8 != 0)
|
if (len % 8 != 0)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
if (!GREEDY_REALLOC(*routes, *routes_size + (len / 8)))
|
|
||||||
return -ENOMEM;
|
|
||||||
|
|
||||||
while (len >= 8) {
|
while (len >= 8) {
|
||||||
struct sd_dhcp_route *route = *routes + *routes_size;
|
struct in_addr dst, gw;
|
||||||
int r;
|
uint8_t prefixlen;
|
||||||
|
|
||||||
route->option = SD_DHCP_OPTION_STATIC_ROUTE;
|
assert_se(lease_parse_be32(option, 4, &dst.s_addr) >= 0);
|
||||||
r = in4_addr_default_prefixlen((struct in_addr*) option, &route->dst_prefixlen);
|
|
||||||
if (r < 0)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
assert_se(lease_parse_be32(option, 4, &addr.s_addr) >= 0);
|
|
||||||
route->dst_addr = inet_makeaddr(inet_netof(addr), 0);
|
|
||||||
option += 4;
|
option += 4;
|
||||||
|
|
||||||
assert_se(lease_parse_be32(option, 4, &route->gw_addr.s_addr) >= 0);
|
assert_se(lease_parse_be32(option, 4, &gw.s_addr) >= 0);
|
||||||
option += 4;
|
option += 4;
|
||||||
|
|
||||||
len -= 8;
|
len -= 8;
|
||||||
|
|
||||||
|
r = in4_addr_default_prefixlen(&dst, &prefixlen);
|
||||||
|
if (r < 0)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
(void) in4_addr_mask(&dst, prefixlen);
|
||||||
|
|
||||||
|
if (!GREEDY_REALLOC(*routes, *routes_size + 1))
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
(*routes)[*routes_size] = (struct sd_dhcp_route) {
|
||||||
|
.dst_addr = dst,
|
||||||
|
.gw_addr = gw,
|
||||||
|
.dst_prefixlen = prefixlen,
|
||||||
|
.option = SD_DHCP_OPTION_STATIC_ROUTE,
|
||||||
|
};
|
||||||
|
|
||||||
(*routes_size)++;
|
(*routes_size)++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user