diff options
Diffstat (limited to 'src/libsystemd-network/sd-dhcp-client.c')
| -rw-r--r-- | src/libsystemd-network/sd-dhcp-client.c | 105 | 
1 files changed, 71 insertions, 34 deletions
| diff --git a/src/libsystemd-network/sd-dhcp-client.c b/src/libsystemd-network/sd-dhcp-client.c index 137537253a..7deb00af9c 100644 --- a/src/libsystemd-network/sd-dhcp-client.c +++ b/src/libsystemd-network/sd-dhcp-client.c @@ -34,6 +34,8 @@  #include "dhcp-internal.h"  #include "dhcp-lease-internal.h"  #include "dhcp-protocol.h" +#include "dns-domain.h" +#include "hostname-util.h"  #include "random-util.h"  #include "string-util.h"  #include "util.h" @@ -298,6 +300,9 @@ int sd_dhcp_client_set_hostname(sd_dhcp_client *client,          assert_return(client, -EINVAL); +        if (!hostname_is_valid(hostname, false) && !dns_name_is_valid(hostname)) +                return -EINVAL; +          if (streq_ptr(client->hostname, hostname))                  return 0; @@ -539,6 +544,24 @@ static int client_message_init(sd_dhcp_client *client, DHCPPacket **ret,          return 0;  } +static int client_append_fqdn_option(DHCPMessage *message, size_t optlen, size_t *optoffset, +                                     const char *fqdn) { +        uint8_t buffer[3 + DHCP_MAX_FQDN_LENGTH]; +        int r; + +        buffer[0] = DHCP_FQDN_FLAG_S | /* Request server to perform A RR DNS updates */ +                    DHCP_FQDN_FLAG_E;  /* Canonical wire format */ +        buffer[1] = 0;                 /* RCODE1 (deprecated) */ +        buffer[2] = 0;                 /* RCODE2 (deprecated) */ + +        r = dns_name_to_wire_format(fqdn, buffer + 3, sizeof(buffer) - 3); +        if (r > 0) +                r = dhcp_option_append(message, optlen, optoffset, 0, +                                       DHCP_OPTION_FQDN, 3 + r, buffer); + +        return r; +} +  static int dhcp_client_send_raw(sd_dhcp_client *client, DHCPPacket *packet,                                  size_t len) {          dhcp_packet_append_ip_headers(packet, INADDR_ANY, DHCP_PORT_CLIENT, @@ -576,13 +599,21 @@ static int client_send_discover(sd_dhcp_client *client) {                          return r;          } -        /* it is unclear from RFC 2131 if client should send hostname in -           DHCPDISCOVER but dhclient does and so we do as well -        */          if (client->hostname) { -                r = dhcp_option_append(&discover->dhcp, optlen, &optoffset, 0, -                                       DHCP_OPTION_HOST_NAME, -                                       strlen(client->hostname), client->hostname); +                /* According to RFC 4702 "clients that send the Client FQDN option in +                   their messages MUST NOT also send the Host Name option". Just send +                   one of the two depending on the hostname type. +                */ +                if (dns_name_is_single_label(client->hostname)) { +                        /* it is unclear from RFC 2131 if client should send hostname in +                           DHCPDISCOVER but dhclient does and so we do as well +                        */ +                        r = dhcp_option_append(&discover->dhcp, optlen, &optoffset, 0, +                                               DHCP_OPTION_HOST_NAME, +                                               strlen(client->hostname), client->hostname); +                } else +                        r = client_append_fqdn_option(&discover->dhcp, optlen, &optoffset, +                                                      client->hostname);                  if (r < 0)                          return r;          } @@ -688,9 +719,13 @@ static int client_send_request(sd_dhcp_client *client) {          }          if (client->hostname) { -                r = dhcp_option_append(&request->dhcp, optlen, &optoffset, 0, -                                       DHCP_OPTION_HOST_NAME, -                                       strlen(client->hostname), client->hostname); +                if (dns_name_is_single_label(client->hostname)) +                        r = dhcp_option_append(&request->dhcp, optlen, &optoffset, 0, +                                               DHCP_OPTION_HOST_NAME, +                                               strlen(client->hostname), client->hostname); +                else +                        r = client_append_fqdn_option(&request->dhcp, optlen, &optoffset, +                                                      client->hostname);                  if (r < 0)                          return r;          } @@ -1047,7 +1082,7 @@ static int client_handle_offer(sd_dhcp_client *client, DHCPMessage *offer,                          return r;          } -        r = dhcp_option_parse(offer, len, dhcp_lease_parse_options, lease); +        r = dhcp_option_parse(offer, len, dhcp_lease_parse_options, lease, NULL);          if (r != DHCP_OFFER) {                  log_dhcp_client(client, "received message was not an OFFER, ignoring");                  return -ENOMSG; @@ -1086,7 +1121,7 @@ static int client_handle_forcerenew(sd_dhcp_client *client, DHCPMessage *force,                                      size_t len) {          int r; -        r = dhcp_option_parse(force, len, NULL, NULL); +        r = dhcp_option_parse(force, len, NULL, NULL, NULL);          if (r != DHCP_FORCERENEW)                  return -ENOMSG; @@ -1098,6 +1133,7 @@ static int client_handle_forcerenew(sd_dhcp_client *client, DHCPMessage *force,  static int client_handle_ack(sd_dhcp_client *client, DHCPMessage *ack,                               size_t len) {          _cleanup_dhcp_lease_unref_ sd_dhcp_lease *lease = NULL; +        _cleanup_free_ char *error_message = NULL;          int r;          r = dhcp_lease_new(&lease); @@ -1112,9 +1148,9 @@ static int client_handle_ack(sd_dhcp_client *client, DHCPMessage *ack,                          return r;          } -        r = dhcp_option_parse(ack, len, dhcp_lease_parse_options, lease); +        r = dhcp_option_parse(ack, len, dhcp_lease_parse_options, lease, &error_message);          if (r == DHCP_NAK) { -                log_dhcp_client(client, "NAK"); +                log_dhcp_client(client, "NAK: %s", strna(error_message));                  return -EADDRNOTAVAIL;          } @@ -1478,9 +1514,8 @@ static int client_receive_message_udp(sd_event_source *s, int fd,          r = ioctl(fd, FIONREAD, &buflen);          if (r < 0) -                return r; - -        if (buflen < 0) +                return -errno; +        else if (buflen < 0)                  /* this can't be right */                  return -EIO; @@ -1490,26 +1525,28 @@ static int client_receive_message_udp(sd_event_source *s, int fd,          len = read(fd, message, buflen);          if (len < 0) { -                log_dhcp_client(client, "could not receive message from UDP " -                                "socket: %m"); -                return 0; +                if (errno == EAGAIN || errno == EINTR) +                        return 0; + +                log_dhcp_client(client, "Could not receive message from UDP socket: %m"); +                return -errno;          } else if ((size_t)len < sizeof(DHCPMessage)) { -                log_dhcp_client(client, "too small to be a DHCP message: ignoring"); +                log_dhcp_client(client, "Too small to be a DHCP message: ignoring");                  return 0;          }          if (be32toh(message->magic) != DHCP_MAGIC_COOKIE) { -                log_dhcp_client(client, "not a DHCP message: ignoring"); +                log_dhcp_client(client, "Not a DHCP message: ignoring");                  return 0;          }          if (message->op != BOOTREPLY) { -                log_dhcp_client(client, "not a BOOTREPLY message: ignoring"); +                log_dhcp_client(client, "Not a BOOTREPLY message: ignoring");                  return 0;          }          if (message->htype != client->arp_type) { -                log_dhcp_client(client, "packet type does not match client type"); +                log_dhcp_client(client, "Packet type does not match client type");                  return 0;          } @@ -1523,13 +1560,12 @@ static int client_receive_message_udp(sd_event_source *s, int fd,          }          if (message->hlen != expected_hlen) { -                log_dhcp_client(client, "unexpected packet hlen %d", message->hlen); +                log_dhcp_client(client, "Unexpected packet hlen %d", message->hlen);                  return 0;          }          if (memcmp(&message->chaddr[0], expected_chaddr, ETH_ALEN)) { -                log_dhcp_client(client, "received chaddr does not match " -                                "expected: ignoring"); +                log_dhcp_client(client, "Received chaddr does not match expected: ignoring");                  return 0;          } @@ -1537,8 +1573,7 @@ static int client_receive_message_udp(sd_event_source *s, int fd,              be32toh(message->xid) != client->xid) {                  /* in BOUND state, we may receive FORCERENEW with xid set by server,                     so ignore the xid in this case */ -                log_dhcp_client(client, "received xid (%u) does not match " -                                "expected (%u): ignoring", +                log_dhcp_client(client, "Received xid (%u) does not match expected (%u): ignoring",                                  be32toh(message->xid), client->xid);                  return 0;          } @@ -1567,9 +1602,8 @@ static int client_receive_message_raw(sd_event_source *s, int fd,          r = ioctl(fd, FIONREAD, &buflen);          if (r < 0) -                return r; - -        if (buflen < 0) +                return -errno; +        else if (buflen < 0)                  /* this can't be right */                  return -EIO; @@ -1582,9 +1616,12 @@ static int client_receive_message_raw(sd_event_source *s, int fd,          len = recvmsg(fd, &msg, 0);          if (len < 0) { -                log_dhcp_client(client, "could not receive message from raw " -                                "socket: %m"); -                return 0; +                if (errno == EAGAIN || errno == EINTR) +                        return 0; + +                log_dhcp_client(client, "Could not receive message from raw socket: %m"); + +                return -errno;          } else if ((size_t)len < sizeof(DHCPPacket))                  return 0; | 
