diff options
Diffstat (limited to 'src/libsystemd-network')
| -rw-r--r-- | src/libsystemd-network/dhcp-lease-internal.h | 2 | ||||
| -rw-r--r-- | src/libsystemd-network/dhcp-protocol.h | 1 | ||||
| -rw-r--r-- | src/libsystemd-network/sd-dhcp-lease.c | 68 | ||||
| -rw-r--r-- | src/libsystemd-network/sd-lldp.c | 2 | 
4 files changed, 62 insertions, 11 deletions
| diff --git a/src/libsystemd-network/dhcp-lease-internal.h b/src/libsystemd-network/dhcp-lease-internal.h index 9e184ac4b5..6e00b1ad30 100644 --- a/src/libsystemd-network/dhcp-lease-internal.h +++ b/src/libsystemd-network/dhcp-lease-internal.h @@ -72,6 +72,8 @@ struct sd_dhcp_lease {          char *root_path;          uint8_t *client_id;          size_t client_id_len; +        uint8_t *vendor_specific; +        size_t vendor_specific_len;  };  int dhcp_lease_new(sd_dhcp_lease **ret); diff --git a/src/libsystemd-network/dhcp-protocol.h b/src/libsystemd-network/dhcp-protocol.h index abca9422c5..aa37e9b0b5 100644 --- a/src/libsystemd-network/dhcp-protocol.h +++ b/src/libsystemd-network/dhcp-protocol.h @@ -125,6 +125,7 @@ enum {          DHCP_OPTION_BROADCAST                   = 28,          DHCP_OPTION_STATIC_ROUTE                = 33,          DHCP_OPTION_NTP_SERVER                  = 42, +        DHCP_OPTION_VENDOR_SPECIFIC             = 43,          DHCP_OPTION_REQUESTED_IP_ADDRESS        = 50,          DHCP_OPTION_IP_ADDRESS_LEASE_TIME       = 51,          DHCP_OPTION_OVERLOAD                    = 52, diff --git a/src/libsystemd-network/sd-dhcp-lease.c b/src/libsystemd-network/sd-dhcp-lease.c index d8bc76edda..54417b3af3 100644 --- a/src/libsystemd-network/sd-dhcp-lease.c +++ b/src/libsystemd-network/sd-dhcp-lease.c @@ -179,6 +179,21 @@ int sd_dhcp_lease_get_routes(sd_dhcp_lease *lease, struct sd_dhcp_route **routes          return 0;  } +int sd_dhcp_lease_get_vendor_specific(sd_dhcp_lease *lease, const uint8_t **data, +                                      size_t *data_len) { +        assert_return(lease, -EINVAL); +        assert_return(data, -EINVAL); +        assert_return(data_len, -EINVAL); + +        if (!lease->vendor_specific) +                return -ENOENT; + +        *data = lease->vendor_specific; +        *data_len = lease->vendor_specific_len; + +        return 0; +} +  sd_dhcp_lease *sd_dhcp_lease_ref(sd_dhcp_lease *lease) {          if (lease)                  assert_se(REFCNT_INC(lease->n_ref) >= 2); @@ -194,6 +209,7 @@ sd_dhcp_lease *sd_dhcp_lease_unref(sd_dhcp_lease *lease) {                  free(lease->ntp);                  free(lease->static_route);                  free(lease->client_id); +                free(lease->vendor_specific);                  free(lease);          } @@ -435,7 +451,8 @@ int dhcp_lease_parse_options(uint8_t code, uint8_t len, const uint8_t *option,                  break;          case DHCP_OPTION_ROUTER: -                lease_parse_be32(option, len, &lease->router); +                if(len >= 4) +                        lease_parse_be32(option, 4, &lease->router);                  break; @@ -579,6 +596,17 @@ int dhcp_lease_parse_options(uint8_t code, uint8_t len, const uint8_t *option,                          return r;                  break; + +        case DHCP_OPTION_VENDOR_SPECIFIC: +                if (len >= 1) { +                        free(lease->vendor_specific); +                        lease->vendor_specific = memdup(option, len); +                        if (!lease->vendor_specific) +                                return -ENOMEM; +                        lease->vendor_specific_len = len; +                } + +               break;          }          return 0; @@ -603,8 +631,8 @@ int sd_dhcp_lease_save(sd_dhcp_lease *lease, const char *lease_file) {          _cleanup_fclose_ FILE *f = NULL;          struct in_addr address;          const struct in_addr *addresses; -        const uint8_t *client_id; -        size_t client_id_len; +        const uint8_t *client_id, *data; +        size_t client_id_len, data_len;          const char *string;          uint16_t mtu;          struct sd_dhcp_route *routes; @@ -690,6 +718,18 @@ int sd_dhcp_lease_save(sd_dhcp_lease *lease, const char *lease_file) {                  fprintf(f, "CLIENTID=%s\n", client_id_hex);          } +        r = sd_dhcp_lease_get_vendor_specific(lease, &data, &data_len); +        if (r >= 0) { +                _cleanup_free_ char *option_hex = NULL; + +                option_hex = hexmem(data, data_len); +                if (!option_hex) { +                        r = -ENOMEM; +                        goto finish; +                } +                fprintf(f, "VENDOR_SPECIFIC=%s\n", option_hex); +        } +          r = 0;          fflush(f); @@ -712,7 +752,8 @@ int sd_dhcp_lease_load(sd_dhcp_lease **ret, const char *lease_file) {          _cleanup_free_ char *address = NULL, *router = NULL, *netmask = NULL,                              *server_address = NULL, *next_server = NULL,                              *dns = NULL, *ntp = NULL, *mtu = NULL, -                            *routes = NULL, *client_id_hex = NULL; +                            *routes = NULL, *client_id_hex = NULL, +                            *vendor_specific_hex = NULL;          struct in_addr addr;          int r; @@ -737,6 +778,7 @@ int sd_dhcp_lease_load(sd_dhcp_lease **ret, const char *lease_file) {                             "ROOT_PATH", &lease->root_path,                             "ROUTES", &routes,                             "CLIENTID", &client_id_hex, +                           "VENDOR_SPECIFIC", &vendor_specific_hex,                             NULL);          if (r < 0) {                  if (r == -ENOENT) @@ -811,13 +853,21 @@ int sd_dhcp_lease_load(sd_dhcp_lease **ret, const char *lease_file) {          }          if (client_id_hex) { -                if (strlen (client_id_hex) % 2) +                if (strlen(client_id_hex) % 2)                          return -EINVAL; -                lease->client_id = unhexmem (client_id_hex, strlen (client_id_hex)); -                if (!lease->client_id) -                        return -ENOMEM; -                lease->client_id_len = strlen (client_id_hex) / 2; +                r = unhexmem(client_id_hex, strlen(client_id_hex), (void**) &lease->client_id, &lease->client_id_len); +                if (r < 0) +                        return r; +        } + +        if (vendor_specific_hex) { +                if (strlen(vendor_specific_hex) % 2) +                        return -EINVAL; + +                r = unhexmem(vendor_specific_hex, strlen(vendor_specific_hex), (void**) &lease->vendor_specific, &lease->vendor_specific_len); +                if (r < 0) +                        return r;          }          *ret = lease; diff --git a/src/libsystemd-network/sd-lldp.c b/src/libsystemd-network/sd-lldp.c index fddda97f52..6a2c05185d 100644 --- a/src/libsystemd-network/sd-lldp.c +++ b/src/libsystemd-network/sd-lldp.c @@ -133,8 +133,6 @@ static int lldp_receive_frame(sd_lldp *lldp, tlv_packet *tlv) {          lldp->statistics.stats_frames_in_total ++; -        return 0; -   out:          if (r < 0)                  log_lldp("Receive frame failed: %s", strerror(-r)); | 
