1
1
mirror of https://github.com/systemd/systemd-stable.git synced 2025-01-10 01:17:44 +03:00

sd-dhcp-server: set DHCPLease::server before hashmap_put()

Otherwise, if the second push is failed, then the first hashmap contains
dirty entry.

Also, this makes hashmap_remove_value() used when removing leases to
make not wrong lease is removed from the hashmap.

Note, this just hide the root cause of the issue #22253, which will be
fixed in later commit.

Fixes #22253.
This commit is contained in:
Yu Watanabe 2022-01-27 00:17:23 +09:00
parent 37e219800f
commit 8a7d048d1d

View File

@ -30,19 +30,10 @@ static DHCPLease *dhcp_lease_free(DHCPLease *lease) {
return NULL; return NULL;
if (lease->server) { if (lease->server) {
DHCPLease *e; hashmap_remove_value(lease->server->bound_leases_by_address, UINT32_TO_PTR(lease->address), lease);
hashmap_remove_value(lease->server->bound_leases_by_client_id, &lease->client_id, lease);
e = hashmap_get(lease->server->bound_leases_by_client_id, &lease->client_id); hashmap_remove_value(lease->server->static_leases_by_address, UINT32_TO_PTR(lease->address), lease);
if (e == lease) { hashmap_remove_value(lease->server->static_leases_by_client_id, &lease->client_id, lease);
hashmap_remove(lease->server->bound_leases_by_address, UINT32_TO_PTR(lease->address));
hashmap_remove(lease->server->bound_leases_by_client_id, &lease->client_id);
}
e = hashmap_get(lease->server->static_leases_by_client_id, &lease->client_id);
if (e == lease) {
hashmap_remove(lease->server->static_leases_by_address, UINT32_TO_PTR(lease->address));
hashmap_remove(lease->server->static_leases_by_client_id, &lease->client_id);
}
} }
free(lease->client_id.data); free(lease->client_id.data);
@ -1118,7 +1109,10 @@ int dhcp_server_handle_message(sd_dhcp_server *server, DHCPMessage *message, siz
log_dhcp_server(server, "ACK (0x%x)", be32toh(req->message->xid)); log_dhcp_server(server, "ACK (0x%x)", be32toh(req->message->xid));
dhcp_lease_free(hashmap_remove(server->bound_leases_by_client_id, &lease->client_id)); dhcp_lease_free(hashmap_get(server->bound_leases_by_client_id, &lease->client_id));
lease->server = server; /* This must be set just before hashmap_put(). */
r = hashmap_ensure_put(&server->bound_leases_by_client_id, &dhcp_lease_hash_ops, &lease->client_id, lease); r = hashmap_ensure_put(&server->bound_leases_by_client_id, &dhcp_lease_hash_ops, &lease->client_id, lease);
if (r < 0) if (r < 0)
return log_dhcp_server_errno(server, r, "Could not save lease: %m"); return log_dhcp_server_errno(server, r, "Could not save lease: %m");
@ -1127,7 +1121,6 @@ int dhcp_server_handle_message(sd_dhcp_server *server, DHCPMessage *message, siz
if (r < 0) if (r < 0)
return log_dhcp_server_errno(server, r, "Could not save lease: %m"); return log_dhcp_server_errno(server, r, "Could not save lease: %m");
lease->server = server;
TAKE_PTR(lease); TAKE_PTR(lease);
if (server->callback) if (server->callback)
@ -1168,6 +1161,8 @@ int dhcp_server_handle_message(sd_dhcp_server *server, DHCPMessage *message, siz
log_dhcp_server(server, "ACK (0x%x)", be32toh(req->message->xid)); log_dhcp_server(server, "ACK (0x%x)", be32toh(req->message->xid));
lease->server = server; /* This must be set just before hashmap_put(). */
r = hashmap_ensure_put(&server->bound_leases_by_client_id, &dhcp_lease_hash_ops, &lease->client_id, lease); r = hashmap_ensure_put(&server->bound_leases_by_client_id, &dhcp_lease_hash_ops, &lease->client_id, lease);
if (r < 0) if (r < 0)
return log_dhcp_server_errno(server, r, "Could not save lease: %m"); return log_dhcp_server_errno(server, r, "Could not save lease: %m");
@ -1175,7 +1170,6 @@ int dhcp_server_handle_message(sd_dhcp_server *server, DHCPMessage *message, siz
if (r < 0) if (r < 0)
return log_dhcp_server_errno(server, r, "Could not save lease: %m"); return log_dhcp_server_errno(server, r, "Could not save lease: %m");
lease->server = server;
TAKE_PTR(new_lease); TAKE_PTR(new_lease);
if (server->callback) if (server->callback)
@ -1599,13 +1593,10 @@ int sd_dhcp_server_set_static_lease(
.data = client_id, .data = client_id,
}; };
dhcp_lease_free(hashmap_remove(server->static_leases_by_client_id, &c)); dhcp_lease_free(hashmap_get(server->static_leases_by_client_id, &c));
return 0; return 0;
} }
if (hashmap_contains(server->static_leases_by_address, UINT32_TO_PTR(address->s_addr)))
return -EEXIST;
lease = new(DHCPLease, 1); lease = new(DHCPLease, 1);
if (!lease) if (!lease)
return -ENOMEM; return -ENOMEM;
@ -1618,6 +1609,8 @@ int sd_dhcp_server_set_static_lease(
if (!lease->client_id.data) if (!lease->client_id.data)
return -ENOMEM; return -ENOMEM;
lease->server = server; /* This must be set just before hashmap_put(). */
r = hashmap_ensure_put(&server->static_leases_by_client_id, &dhcp_lease_hash_ops, &lease->client_id, lease); r = hashmap_ensure_put(&server->static_leases_by_client_id, &dhcp_lease_hash_ops, &lease->client_id, lease);
if (r < 0) if (r < 0)
return r; return r;
@ -1625,7 +1618,6 @@ int sd_dhcp_server_set_static_lease(
if (r < 0) if (r < 0)
return r; return r;
lease->server = server;
TAKE_PTR(lease); TAKE_PTR(lease);
return 0; return 0;
} }