diff --git a/ipmi-config/Makefile.am b/ipmi-config/Makefile.am index 1bc633b03..fb47649ae 100644 --- a/ipmi-config/Makefile.am +++ b/ipmi-config/Makefile.am @@ -53,6 +53,8 @@ ipmi_config_SOURCES = \ ipmi-config-category-core-lan-conf-security-keys-section.h \ ipmi-config-category-core-lan-conf-user-security-section.c \ ipmi-config-category-core-lan-conf-user-security-section.h \ + ipmi-config-category-core-lan6-conf-section.c \ + ipmi-config-category-core-lan6-conf-section.h \ ipmi-config-category-core-misc-section.c \ ipmi-config-category-core-misc-section.h \ ipmi-config-category-core-pef-conf-section.c \ diff --git a/ipmi-config/ipmi-config-category-core-lan6-conf-section.c b/ipmi-config/ipmi-config-category-core-lan6-conf-section.c new file mode 100644 index 000000000..cd5c79916 --- /dev/null +++ b/ipmi-config/ipmi-config-category-core-lan6-conf-section.c @@ -0,0 +1,845 @@ +/* + * Copyright (C) 2003-2015 FreeIPMI Core Team + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + * + */ + +#if HAVE_CONFIG_H +#include "config.h" +#endif /* HAVE_CONFIG_H */ + +#include +#include +#if STDC_HEADERS +#include +#endif /* STDC_HEADERS */ +#include +#include + +#include "ipmi-config.h" +#include "ipmi-config-map.h" +#include "ipmi-config-section.h" +#include "ipmi-config-utils.h" +#include "ipmi-config-validate.h" + +#include "freeipmi-portability.h" +#include "pstdout.h" + +#define BMC_MAXIPADDRLEN 16 +#define BMC_MAXIPV6ADDRLEN 45 +#define BMC_MAXMACADDRLEN 24 + +static ipmi_config_err_t +_get_ipv6_ipv4_support (ipmi_config_state_data_t *state_data, + const char *section_name) +{ + fiid_obj_t obj_cmd_rs = NULL; + uint64_t val; + ipmi_config_err_t rv = IPMI_CONFIG_ERR_FATAL_ERROR; + ipmi_config_err_t ret; + uint8_t channel_number; + + assert (state_data); + assert (section_name); + + if ((ret = get_lan_channel_number (state_data, section_name, &channel_number)) != IPMI_CONFIG_ERR_SUCCESS) + { + rv = ret; + goto cleanup; + } + + if (state_data->ipv6_ipv4_support_initialized + && state_data->ipv6_ipv4_support_channel_number == channel_number) + goto out; + + state_data->ipv6_ipv4_support_initialized = 0; + state_data->ipv6_ipv4_support_channel_number = 0; + + if (!(obj_cmd_rs = fiid_obj_create (tmpl_cmd_get_lan_configuration_parameters_ipv6_ipv4_support_rs))) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_create: %s\n", + strerror (errno)); + goto cleanup; + } + + if (ipmi_cmd_get_lan_configuration_parameters_ipv6_ipv4_support (state_data->ipmi_ctx, + channel_number, + IPMI_GET_LAN_PARAMETER, + IPMI_LAN_CONFIGURATION_PARAMETERS_NO_SET_SELECTOR, + IPMI_LAN_CONFIGURATION_PARAMETERS_NO_BLOCK_SELECTOR, + obj_cmd_rs) < 0) + { + if (ipmi_config_param_errnum_is_non_fatal (state_data, + obj_cmd_rs, + &ret)) + rv = ret; + + if (rv == IPMI_CONFIG_ERR_NON_FATAL_ERROR_NOT_SUPPORTED) + { + state_data->ipv6_ipv4_support_supports_ipv6_only = 0; + state_data->ipv6_ipv4_support_supports_ipv6_and_ipv4_simultaneously = 0; + state_data->ipv6_ipv4_support_supports_ipv6_destination_address_for_lan_alert = 0; + state_data->ipv6_ipv4_support_channel_number = channel_number; + state_data->ipv6_ipv4_support_initialized++; + goto out; + } + + if (rv == IPMI_CONFIG_ERR_FATAL_ERROR + || state_data->prog_data->args->common_args.debug) + pstdout_fprintf (state_data->pstate, + stderr, + "ipmi_cmd_get_lan_configuration_parameters_ipv6_ipv4_support: %s\n", + ipmi_ctx_errormsg (state_data->ipmi_ctx)); + + goto cleanup; + } + + if (FIID_OBJ_GET (obj_cmd_rs, "supports_ipv6_only", &val) < 0) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_get_data: 'supports_ipv6_only': %s\n", + fiid_obj_errormsg (obj_cmd_rs)); + goto cleanup; + } + state_data->ipv6_ipv4_support_supports_ipv6_only = val; + + if (FIID_OBJ_GET (obj_cmd_rs, "supports_ipv6_and_ipv4_simultaneously", &val) < 0) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_get_data: 'supports_ipv6_and_ipv4_simultaneously': %s\n", + fiid_obj_errormsg (obj_cmd_rs)); + goto cleanup; + } + state_data->ipv6_ipv4_support_supports_ipv6_and_ipv4_simultaneously = val; + + if (FIID_OBJ_GET (obj_cmd_rs, "supports_ipv6_destination_address_for_lan_alert", &val) < 0) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_get_data: 'supports_ipv6_destination_address_for_lan_alert': %s\n", + fiid_obj_errormsg (obj_cmd_rs)); + goto cleanup; + } + state_data->ipv6_ipv4_support_supports_ipv6_destination_address_for_lan_alert = val; + + state_data->ipv6_ipv4_support_channel_number = channel_number; + state_data->ipv6_ipv4_support_initialized++; + out: + rv = IPMI_CONFIG_ERR_SUCCESS; + cleanup: + fiid_obj_destroy (obj_cmd_rs); + return (rv); +} + +static ipmi_config_err_t +ipv6_ipv4_support_ipv6_only_checkout (ipmi_config_state_data_t *state_data, + const char *section_name, + struct ipmi_config_keyvalue *kv) +{ + ipmi_config_err_t rv = IPMI_CONFIG_ERR_FATAL_ERROR; + ipmi_config_err_t ret; + + ret = _get_ipv6_ipv4_support (state_data, + section_name); + + if (ret != IPMI_CONFIG_ERR_SUCCESS) + { + rv = ret; + goto cleanup; + } + + if (ipmi_config_section_update_keyvalue_output (state_data, + kv, + state_data->ipv6_ipv4_support_supports_ipv6_only ? "Yes" : "No") < 0) + return (IPMI_CONFIG_ERR_FATAL_ERROR); + + rv = IPMI_CONFIG_ERR_SUCCESS; + cleanup: + return (rv); +} + +static ipmi_config_err_t +ipv6_ipv4_support_ipv6_and_ipv4_simultaneously_checkout (ipmi_config_state_data_t *state_data, + const char *section_name, + struct ipmi_config_keyvalue *kv) +{ + ipmi_config_err_t rv = IPMI_CONFIG_ERR_FATAL_ERROR; + ipmi_config_err_t ret; + + ret = _get_ipv6_ipv4_support (state_data, + section_name); + + if (ret != IPMI_CONFIG_ERR_SUCCESS) + { + rv = ret; + goto cleanup; + } + + if (ipmi_config_section_update_keyvalue_output (state_data, + kv, + state_data->ipv6_ipv4_support_supports_ipv6_and_ipv4_simultaneously ? "Yes" : "No") < 0) + return (IPMI_CONFIG_ERR_FATAL_ERROR); + + rv = IPMI_CONFIG_ERR_SUCCESS; + cleanup: + return (rv); +} + +static ipmi_config_err_t +ipv6_ipv4_support_ipv6_destination_address_for_lan_alert_checkout (ipmi_config_state_data_t *state_data, + const char *section_name, + struct ipmi_config_keyvalue *kv) +{ + ipmi_config_err_t rv = IPMI_CONFIG_ERR_FATAL_ERROR; + ipmi_config_err_t ret; + + ret = _get_ipv6_ipv4_support (state_data, + section_name); + + if (ret != IPMI_CONFIG_ERR_SUCCESS) + { + rv = ret; + goto cleanup; + } + + if (ipmi_config_section_update_keyvalue_output (state_data, + kv, + state_data->ipv6_ipv4_support_supports_ipv6_destination_address_for_lan_alert ? "Yes" : "No") < 0) + return (IPMI_CONFIG_ERR_FATAL_ERROR); + + rv = IPMI_CONFIG_ERR_SUCCESS; + cleanup: + return (rv); +} + +static ipmi_config_err_t +ipv6_ipv4_addressing_enables_checkout (ipmi_config_state_data_t *state_data, + const char *section_name, + struct ipmi_config_keyvalue *kv) +{ + fiid_obj_t obj_cmd_rs = NULL; + ipmi_config_err_t rv = IPMI_CONFIG_ERR_FATAL_ERROR; + ipmi_config_err_t ret; + uint8_t channel_number; + uint64_t val; + + assert (state_data); + assert (section_name); + assert (kv); + + if (!(obj_cmd_rs = fiid_obj_create (tmpl_cmd_get_lan_configuration_parameters_ipv6_ipv4_addressing_enables_rs))) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_create: %s\n", + strerror (errno)); + goto cleanup; + } + + if ((ret = get_lan_channel_number (state_data, + section_name, + &channel_number)) != IPMI_CONFIG_ERR_SUCCESS) + { + rv = ret; + goto cleanup; + } + + if (ipmi_cmd_get_lan_configuration_parameters_ipv6_ipv4_addressing_enables (state_data->ipmi_ctx, + channel_number, + IPMI_GET_LAN_PARAMETER, + IPMI_LAN_CONFIGURATION_PARAMETERS_NO_SET_SELECTOR, + IPMI_LAN_CONFIGURATION_PARAMETERS_NO_BLOCK_SELECTOR, + obj_cmd_rs) < 0) + { + if (ipmi_config_param_errnum_is_non_fatal (state_data, + obj_cmd_rs, + &ret)) + rv = ret; + + if (rv == IPMI_CONFIG_ERR_FATAL_ERROR + || state_data->prog_data->args->common_args.debug) + pstdout_fprintf (state_data->pstate, + stderr, + "ipmi_cmd_get_lan_configuration_parameters_ipv6_ipv4_addressing_enables: %s\n", + ipmi_ctx_errormsg (state_data->ipmi_ctx)); + + goto cleanup; + } + + if (fiid_obj_get_data (obj_cmd_rs, + "enables", + &val, + 1) < 0) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_get_data: 'enables': %s\n", + fiid_obj_errormsg (obj_cmd_rs)); + goto cleanup; + } + + if (ipmi_config_section_update_keyvalue_output (state_data, + kv, + ipv6_ipv4_addressing_enables_string (val)) < 0) + return (IPMI_CONFIG_ERR_FATAL_ERROR); + + rv = IPMI_CONFIG_ERR_SUCCESS; + cleanup: + fiid_obj_destroy (obj_cmd_rs); + return (rv); +} + +static ipmi_config_err_t +ipv6_ipv4_addressing_enables_commit (ipmi_config_state_data_t *state_data, + const char *section_name, + const struct ipmi_config_keyvalue *kv) +{ + /* XXX TODO lamont -- Not needed for Read-Only. */ + return (IPMI_CONFIG_ERR_FATAL_ERROR); +} + +static ipmi_config_err_t +ipv6_static_addresses_checkout (ipmi_config_state_data_t *state_data, + const char *section_name, + struct ipmi_config_keyvalue *kv) +{ + fiid_obj_t obj_cmd_rs = NULL; + char ipv6_address_str[BMC_MAXIPV6ADDRLEN + 1]; + char *ipv6_addresses_str = NULL; + int ipv6_addresses_len; + int ipv6_addresses_pos = 0; + uint8_t ipv6_ip_address_bytes[16]; + ipmi_config_err_t rv = IPMI_CONFIG_ERR_FATAL_ERROR; + ipmi_config_err_t ret; + uint8_t channel_number; + uint8_t address_max; + uint8_t set; + uint64_t val; + + assert (state_data); + assert (section_name); + assert (kv); + + if (!(obj_cmd_rs = fiid_obj_create (tmpl_cmd_get_lan_configuration_parameters_ipv6_status_rs))) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_create: %s\n", + strerror (errno)); + goto cleanup; + } + + if ((ret = get_lan_channel_number (state_data, + section_name, + &channel_number)) != IPMI_CONFIG_ERR_SUCCESS) + { + rv = ret; + goto cleanup; + } + + if (ipmi_cmd_get_lan_configuration_parameters_ipv6_status (state_data->ipmi_ctx, + channel_number, + IPMI_GET_LAN_PARAMETER, + IPMI_LAN_CONFIGURATION_PARAMETERS_NO_SET_SELECTOR, + IPMI_LAN_CONFIGURATION_PARAMETERS_NO_BLOCK_SELECTOR, + obj_cmd_rs) < 0) + { + if (ipmi_config_param_errnum_is_non_fatal (state_data, + obj_cmd_rs, + &ret)) + rv = ret; + + if (rv == IPMI_CONFIG_ERR_FATAL_ERROR + || state_data->prog_data->args->common_args.debug) + pstdout_fprintf (state_data->pstate, + stderr, + "ipmi_cmd_get_lan_configuration_parameters_ipv6_static_ip_address: %s\n", + ipmi_ctx_errormsg (state_data->ipmi_ctx)); + + goto cleanup; + } + + if (FIID_OBJ_GET (obj_cmd_rs, "static_address_max", &val) < 0) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_get_data: 'supports_ipv6_only': %s\n", + fiid_obj_errormsg (obj_cmd_rs)); + goto cleanup; + } + address_max = val; + fiid_obj_destroy (obj_cmd_rs); + obj_cmd_rs = NULL; + + if (address_max) + { + ipv6_addresses_len = (BMC_MAXIPV6ADDRLEN+1)*address_max; + ipv6_addresses_str = malloc(ipv6_addresses_len); + memset (ipv6_addresses_str, '\0', ipv6_addresses_len); + for (set = 0; set < address_max; set++) + { + if (!(obj_cmd_rs = fiid_obj_create (tmpl_cmd_get_lan_configuration_parameters_ipv6_static_addresses_rs))) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_create: %s\n", + strerror (errno)); + goto cleanup; + } + + if ((ret = get_lan_channel_number (state_data, + section_name, + &channel_number)) != IPMI_CONFIG_ERR_SUCCESS) + { + rv = ret; + goto cleanup; + } + + if (ipmi_cmd_get_lan_configuration_parameters_ipv6_static_addresses (state_data->ipmi_ctx, + channel_number, + IPMI_GET_LAN_PARAMETER, + set, + IPMI_LAN_CONFIGURATION_PARAMETERS_NO_BLOCK_SELECTOR, + obj_cmd_rs) < 0) + { + if (ipmi_config_param_errnum_is_non_fatal (state_data, + obj_cmd_rs, + &ret)) + rv = ret; + + if (rv == IPMI_CONFIG_ERR_FATAL_ERROR + || state_data->prog_data->args->common_args.debug) + pstdout_fprintf (state_data->pstate, + stderr, + "ipmi_cmd_get_lan_configuration_parameters_ipv6_static_ip_address: %s\n", + ipmi_ctx_errormsg (state_data->ipmi_ctx)); + + goto cleanup; + } + + if (fiid_obj_get_data (obj_cmd_rs, + "address", + ipv6_ip_address_bytes, + 16) < 0) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_get_data: 'address': %s\n", + fiid_obj_errormsg (obj_cmd_rs)); + goto cleanup; + } + + memset (ipv6_address_str, '\0', BMC_MAXIPV6ADDRLEN+1); + if (NULL == inet_ntop(AF_INET6, ipv6_ip_address_bytes, ipv6_address_str, BMC_MAXIPV6ADDRLEN)) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_get_data: 'address': %s\n", + strerror (errno)); + goto cleanup; + + } + if (!same (ipv6_address_str, "::")) + { + sprintf(ipv6_addresses_str+ipv6_addresses_pos, "%s ", ipv6_address_str); + ipv6_addresses_pos += strlen(ipv6_address_str) + 1; + } + fiid_obj_destroy (obj_cmd_rs); + obj_cmd_rs = NULL; + } + if (ipv6_addresses_pos) + ipv6_addresses_str[ipv6_addresses_pos - 1] = '\0'; + + if (ipmi_config_section_update_keyvalue_output (state_data, + kv, + ipv6_addresses_str) < 0) + return (IPMI_CONFIG_ERR_FATAL_ERROR); + } + + rv = IPMI_CONFIG_ERR_SUCCESS; + cleanup: + if (ipv6_addresses_str) + free(ipv6_addresses_str); + fiid_obj_destroy (obj_cmd_rs); + return (rv); +} + +static ipmi_config_err_t +ipv6_static_addresses_commit (ipmi_config_state_data_t *state_data, + const char *section_name, + const struct ipmi_config_keyvalue *kv) +{ + fiid_obj_t obj_cmd_rs = NULL; + uint8_t ipv6_address_bytes[16]; + ipmi_config_err_t rv = IPMI_CONFIG_ERR_FATAL_ERROR; + ipmi_config_err_t ret; + uint8_t channel_number; + + assert (state_data); + assert (section_name); + assert (kv); + +#if 0 + /* XXX TODO lamont -- Not needed for Read-Only. + * -- kv->value_input is a collection of IP addresess. We need to: + * 1. break it up. + * 2. make sure that static_address_max is sufficent + * 3. iterate through the values setting them one at a time. + */ + if (inet_pton (AF_INET6, + kv->value_input, + ipv6_address_bytes) != 1) + goto cleanup; + + if (!(obj_cmd_rs = fiid_obj_create (tmpl_cmd_set_lan_configuration_parameters_ipv6_rs))) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_create: %s\n", + strerror (errno)); + goto cleanup; + } + + if ((ret = get_lan_channel_number (state_data, + section_name, + &channel_number)) != IPMI_CONFIG_ERR_SUCCESS) + { + rv = ret; + goto cleanup; + } + + if (ipmi_cmd_set_lan_configuration_parameters_ipv6_static_addresses (state_data->ipmi_ctx, + channel_number, + ip_address_val, + obj_cmd_rs) < 0) + { + if (ipmi_config_param_errnum_is_non_fatal (state_data, + obj_cmd_rs, + &ret)) + rv = ret; + + if (rv == IPMI_CONFIG_ERR_FATAL_ERROR + || state_data->prog_data->args->common_args.debug) + pstdout_fprintf (state_data->pstate, + stderr, + "ipmi_cmd_set_lan_configuration_parameters_ip_address: %s\n", + ipmi_ctx_errormsg (state_data->ipmi_ctx)); + + goto cleanup; + } + + rv = IPMI_CONFIG_ERR_SUCCESS; + cleanup: + fiid_obj_destroy (obj_cmd_rs); +#endif + return (rv); +} + +static ipmi_config_err_t +ipv6_dynamic_address_checkout (ipmi_config_state_data_t *state_data, + const char *section_name, + struct ipmi_config_keyvalue *kv) +{ + fiid_obj_t obj_cmd_rs = NULL; + char ipv6_address_str[BMC_MAXIPV6ADDRLEN + 1]; + char *ipv6_addresses_str = NULL; + int ipv6_addresses_len; + int ipv6_addresses_pos = 0; + uint8_t ipv6_ip_address_bytes[16]; + ipmi_config_err_t rv = IPMI_CONFIG_ERR_FATAL_ERROR; + ipmi_config_err_t ret; + uint8_t channel_number; + uint8_t address_max; + uint8_t set; + uint64_t val; + + assert (state_data); + assert (section_name); + assert (kv); + + if (!(obj_cmd_rs = fiid_obj_create (tmpl_cmd_get_lan_configuration_parameters_ipv6_status_rs))) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_create: %s\n", + strerror (errno)); + goto cleanup; + } + + if ((ret = get_lan_channel_number (state_data, + section_name, + &channel_number)) != IPMI_CONFIG_ERR_SUCCESS) + { + rv = ret; + goto cleanup; + } + + if (ipmi_cmd_get_lan_configuration_parameters_ipv6_status (state_data->ipmi_ctx, + channel_number, + IPMI_GET_LAN_PARAMETER, + IPMI_LAN_CONFIGURATION_PARAMETERS_NO_SET_SELECTOR, + IPMI_LAN_CONFIGURATION_PARAMETERS_NO_BLOCK_SELECTOR, + obj_cmd_rs) < 0) + { + if (ipmi_config_param_errnum_is_non_fatal (state_data, + obj_cmd_rs, + &ret)) + rv = ret; + + if (rv == IPMI_CONFIG_ERR_FATAL_ERROR + || state_data->prog_data->args->common_args.debug) + pstdout_fprintf (state_data->pstate, + stderr, + "ipmi_cmd_get_lan_configuration_parameters_ipv6_dynamic_ip_address: %s\n", + ipmi_ctx_errormsg (state_data->ipmi_ctx)); + + goto cleanup; + } + + if (FIID_OBJ_GET (obj_cmd_rs, "dynamic_address_max", &val) < 0) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_get_data: 'supports_ipv6_only': %s\n", + fiid_obj_errormsg (obj_cmd_rs)); + goto cleanup; + } + + address_max = val; + fiid_obj_destroy (obj_cmd_rs); + obj_cmd_rs = NULL; + + if (address_max) + { + ipv6_addresses_len = (BMC_MAXIPV6ADDRLEN+1)*address_max; + ipv6_addresses_str = malloc(ipv6_addresses_len); + memset (ipv6_addresses_str, '\0', ipv6_addresses_len); + for (set = 0; set < address_max; set++) + { + if (!(obj_cmd_rs = fiid_obj_create (tmpl_cmd_get_lan_configuration_parameters_ipv6_dynamic_address_rs))) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_create: %s\n", + strerror (errno)); + goto cleanup; + } + + if ((ret = get_lan_channel_number (state_data, + section_name, + &channel_number)) != IPMI_CONFIG_ERR_SUCCESS) + { + rv = ret; + goto cleanup; + } + + if (ipmi_cmd_get_lan_configuration_parameters_ipv6_dynamic_address (state_data->ipmi_ctx, + channel_number, + IPMI_GET_LAN_PARAMETER, + set, + IPMI_LAN_CONFIGURATION_PARAMETERS_NO_BLOCK_SELECTOR, + obj_cmd_rs) < 0) + { + if (ipmi_config_param_errnum_is_non_fatal (state_data, + obj_cmd_rs, + &ret)) + rv = ret; + + if (rv == IPMI_CONFIG_ERR_FATAL_ERROR + || state_data->prog_data->args->common_args.debug) + pstdout_fprintf (state_data->pstate, + stderr, + "ipmi_cmd_get_lan_configuration_parameters_ipv6_dynamic_ip_address: %s\n", + ipmi_ctx_errormsg (state_data->ipmi_ctx)); + + goto cleanup; + } + + if (FIID_OBJ_GET (obj_cmd_rs, "address_status", &val) < 0) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_get_data: 'address_status': %s\n", + fiid_obj_errormsg (obj_cmd_rs)); + goto cleanup; + } + + if (val == IPMI_IPV6_ADDRESS_STATUS_ACTIVE) + { + if (fiid_obj_get_data (obj_cmd_rs, + "address", + ipv6_ip_address_bytes, + 16) < 0) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_get_data: 'address': %s\n", + fiid_obj_errormsg (obj_cmd_rs)); + goto cleanup; + } + + memset (ipv6_address_str, '\0', BMC_MAXIPV6ADDRLEN+1); + if (NULL == inet_ntop(AF_INET6, ipv6_ip_address_bytes, ipv6_address_str, BMC_MAXIPV6ADDRLEN)) + { + pstdout_fprintf (state_data->pstate, + stderr, + "fiid_obj_get_data: 'address': %s\n", + strerror (errno)); + goto cleanup; + + } + if (!same (ipv6_address_str, "::")) + { + sprintf(ipv6_addresses_str+ipv6_addresses_pos, "%s ", ipv6_address_str); + ipv6_addresses_pos += strlen(ipv6_address_str) + 1; + } + } + fiid_obj_destroy (obj_cmd_rs); + obj_cmd_rs = NULL; + } + if (ipv6_addresses_pos) + ipv6_addresses_str[ipv6_addresses_pos - 1] = '\0'; + + if (ipmi_config_section_update_keyvalue_output (state_data, + kv, + ipv6_addresses_str) < 0) + return (IPMI_CONFIG_ERR_FATAL_ERROR); + } + + rv = IPMI_CONFIG_ERR_SUCCESS; + cleanup: + if (ipv6_addresses_str) + free(ipv6_addresses_str); + fiid_obj_destroy (obj_cmd_rs); + return (rv); +} + +static ipmi_config_err_t +ipv6_dynamic_address_commit (ipmi_config_state_data_t *state_data, + const char *section_name, + const struct ipmi_config_keyvalue *kv) +{ + ipmi_config_err_t rv = IPMI_CONFIG_ERR_FATAL_ERROR; + + assert (state_data); + assert (section_name); + assert (kv); + + /* Dynamic address field is not writable. */ + return (rv); +} + +struct ipmi_config_section * +ipmi_config_core_lan6_conf_section_get (ipmi_config_state_data_t *state_data, + unsigned int config_flags, + int channel_index) +{ + struct ipmi_config_section *section = NULL; + char *section_comment = + "In the Lan6_Conf section, typical networking configuration is setup. " + "Most users will choose to set an address in \"IPv6_Static_Addresses\" " + "and set the appropriate routing for the machine."; + char *section_name_base_str = "Lan6_Conf"; + unsigned int verbose_option_config_flags = 0; + + assert (state_data); + + if (!state_data->prog_data->args->verbose_count) + verbose_option_config_flags = IPMI_CONFIG_DO_NOT_CHECKOUT; + + if (!(section = ipmi_config_section_multi_channel_create (state_data, + section_name_base_str, + section_comment, + NULL, + NULL, + config_flags, + channel_index, + state_data->lan_channel_numbers, + state_data->lan_channel_numbers_count))) + goto cleanup; + + + if (ipmi_config_section_add_key (state_data, + section, + "Supports_IPv6_Only", + "READ-ONLY: Supports IPv6-only", + IPMI_CONFIG_CHECKOUT_KEY_COMMENTED_OUT | IPMI_CONFIG_READABLE_ONLY, + ipv6_ipv4_support_ipv6_only_checkout, + read_only_commit, + yes_no_validate) < 0) + goto cleanup; + + if (ipmi_config_section_add_key (state_data, + section, + "Supports_IPv6_And_IPv4_Simultaneously", + "READ-ONLY: Supports IPv6 And IPv4 Simultaneously", + IPMI_CONFIG_CHECKOUT_KEY_COMMENTED_OUT | IPMI_CONFIG_READABLE_ONLY, + ipv6_ipv4_support_ipv6_and_ipv4_simultaneously_checkout, + read_only_commit, + yes_no_validate) < 0) + goto cleanup; + + if (ipmi_config_section_add_key (state_data, + section, + "Supports_IPv6_Destination_Address_For_Lan_Alert", + "READ-ONLY: Supports IPv6 Destination Address For Lan Alert", + verbose_option_config_flags | IPMI_CONFIG_CHECKOUT_KEY_COMMENTED_OUT | IPMI_CONFIG_READABLE_ONLY, + ipv6_ipv4_support_ipv6_destination_address_for_lan_alert_checkout, + read_only_commit, + yes_no_validate) < 0) + goto cleanup; + + if (ipmi_config_section_add_key (state_data, + section, + "IPv6_IPv4_Addressing_Enables", + "Possible values: IPv4-Only/IPv6-Only/IPv4-and-IPv6", + IPMI_CONFIG_CHECKOUT_KEY_COMMENTED_OUT | IPMI_CONFIG_READABLE_ONLY, /* TODO: make this read-write. */ + ipv6_ipv4_addressing_enables_checkout, + ipv6_ipv4_addressing_enables_commit, + ipv6_ipv4_addressing_enables_validate) < 0) + goto cleanup; + + if (ipmi_config_section_add_key (state_data, + section, + "IPv6_Static_Addresses", + "Give valid IPv6 address", + IPMI_CONFIG_CHECKOUT_KEY_COMMENTED_OUT | IPMI_CONFIG_READABLE_ONLY, /* TODO: make this read-write. */ + ipv6_static_addresses_checkout, + ipv6_static_addresses_commit, + ipv6_address_validate) < 0) + goto cleanup; + + if (ipmi_config_section_add_key (state_data, + section, + "IPv6_Dynamic_Address", + "READ-ONLY: IPv6 dynamic address", + IPMI_CONFIG_CHECKOUT_KEY_COMMENTED_OUT | IPMI_CONFIG_READABLE_ONLY, + ipv6_dynamic_address_checkout, + read_only_commit, + ipv6_address_validate) < 0) + goto cleanup; + + return (section); + + cleanup: + if (section) + ipmi_config_section_destroy (section); + return (NULL); +} + diff --git a/ipmi-config/ipmi-config-category-core-lan6-conf-section.h b/ipmi-config/ipmi-config-category-core-lan6-conf-section.h new file mode 100644 index 000000000..e84aa1f5c --- /dev/null +++ b/ipmi-config/ipmi-config-category-core-lan6-conf-section.h @@ -0,0 +1,28 @@ +/* + * Copyright (C) 2003-2015 FreeIPMI Core Team + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + * + */ + +#ifndef IPMI_CONFIG_CATEGORY_CORE_LAN6_CONF_SECTION_H +#define IPMI_CONFIG_CATEGORY_CORE_LAN6_CONF_SECTION_H + +#include "ipmi-config.h" + +struct ipmi_config_section * ipmi_config_core_lan6_conf_section_get (ipmi_config_state_data_t *state_data, + unsigned int config_flags, + int channel_index); + +#endif /* IPMI_CONFIG_CATEGORY_CORE_LAN6_CONF_SECTION_H */ diff --git a/ipmi-config/ipmi-config-category-core-sections.c b/ipmi-config/ipmi-config-category-core-sections.c index 832a06552..1efaec1da 100644 --- a/ipmi-config/ipmi-config-category-core-sections.c +++ b/ipmi-config/ipmi-config-category-core-sections.c @@ -39,6 +39,7 @@ #include "ipmi-config-category-core-lan-conf-security-keys-section.h" #include "ipmi-config-category-core-lan-conf-misc-section.h" #include "ipmi-config-category-core-lan-conf-user-security-section.h" +#include "ipmi-config-category-core-lan6-conf-section.h" #include "ipmi-config-category-core-misc-section.h" #include "ipmi-config-category-core-pef-conf-section.h" #include "ipmi-config-category-core-rmcpplus-conf-privilege-section.h" @@ -230,6 +231,28 @@ ipmi_config_core_sections_create (ipmi_config_state_data_t *state_data) } } + /* Lan6_Conf Section(s) */ + + if (!(section = ipmi_config_core_lan6_conf_section_get (state_data, + state_data->lan_base_config_flags, + -1))) + goto cleanup; + if (ipmi_config_section_append (§ions, section) < 0) + goto cleanup; + + if (state_data->lan_channel_numbers_count > 1) + { + for (channelindex = 0; channelindex < state_data->lan_channel_numbers_count; channelindex++) + { + if (!(section = ipmi_config_core_lan6_conf_section_get (state_data, + state_data->lan_channel_config_flags, + channelindex))) + goto cleanup; + if (ipmi_config_section_append (§ions, section) < 0) + goto cleanup; + } + } + /* Lan_Conf_Auth Section(s) */ if (!(section = ipmi_config_core_lan_conf_auth_section_get (state_data, diff --git a/ipmi-config/ipmi-config-map.c b/ipmi-config/ipmi-config-map.c index 48f1b23a9..df6c9c63c 100644 --- a/ipmi-config/ipmi-config-map.c +++ b/ipmi-config/ipmi-config-map.c @@ -221,6 +221,35 @@ ip_address_source_string (uint8_t value) return ""; } +int +ipv6_ipv4_addressing_enables_number (const char *string) +{ + assert (string); + + if (same (string, "IPv6_Disabled")) + return (IPMI_IPV6_IPV4_ADDRESSING_ENABLES_IPV6_DISABLED); + if (same (string, "IPv6_Only")) + return (IPMI_IPV6_IPV4_ADDRESSING_ENABLES_IPV6_ONLY); + if (same (string, "IPv6_and_IPv4")) + return (IPMI_IPV6_IPV4_ADDRESSING_ENABLES_IPV6_AND_IPV4); + return (-1); +} + +char * +ipv6_ipv4_addressing_enables_string (uint8_t value) +{ + switch (value) + { + case IPMI_IPV6_IPV4_ADDRESSING_ENABLES_IPV6_DISABLED: + return "IPv6_Disabled"; + case IPMI_IPV6_IPV4_ADDRESSING_ENABLES_IPV6_ONLY: + return "IPv6_Only"; + case IPMI_IPV6_IPV4_ADDRESSING_ENABLES_IPV6_AND_IPV4: + return "IPv6_and_IPv4"; + } + return ""; +} + int power_restore_policy_number (const char *string) { diff --git a/ipmi-config/ipmi-config-map.h b/ipmi-config/ipmi-config-map.h index 1df1ea9d1..0fa6704ff 100644 --- a/ipmi-config/ipmi-config-map.h +++ b/ipmi-config/ipmi-config-map.h @@ -41,6 +41,10 @@ int ip_address_source_number (const char *source); char *ip_address_source_string (uint8_t source); +int ipv6_ipv4_addressing_enables_number (const char *source); + +char *ipv6_ipv4_addressing_enables_string (uint8_t source); + int power_restore_policy_number (const char *string); char *power_restore_policy_string (uint8_t value); diff --git a/ipmi-config/ipmi-config-utils.c b/ipmi-config/ipmi-config-utils.c index 33a00e4d0..546642666 100644 --- a/ipmi-config/ipmi-config-utils.c +++ b/ipmi-config/ipmi-config-utils.c @@ -645,3 +645,15 @@ get_sol_channel_number (ipmi_config_state_data_t *state_data, return (rv); } +ipmi_config_err_t +read_only_commit (ipmi_config_state_data_t *state_data, + const char *section_name, + const struct ipmi_config_keyvalue *kv) +{ + /* Read only parameter */ + pstdout_fprintf (state_data->pstate, + stderr, + "Ignoring attempt to set read-only configuration variable.\n"); + return (IPMI_CONFIG_ERR_SUCCESS); +} + diff --git a/ipmi-config/ipmi-config-utils.h b/ipmi-config/ipmi-config-utils.h index 17713dbab..adad018ee 100644 --- a/ipmi-config/ipmi-config-utils.h +++ b/ipmi-config/ipmi-config-utils.h @@ -69,4 +69,8 @@ ipmi_config_err_t get_sol_channel_number (ipmi_config_state_data_t *state_data, const char *section_name, uint8_t *channel_number); +ipmi_config_err_t read_only_commit (ipmi_config_state_data_t *state_data, + const char *section_name, + const struct ipmi_config_keyvalue *kv); + #endif /* IPMI_CONFIG_UTILS_H */ diff --git a/ipmi-config/ipmi-config-validate.c b/ipmi-config/ipmi-config-validate.c index 60345196e..9c3b84209 100644 --- a/ipmi-config/ipmi-config-validate.c +++ b/ipmi-config/ipmi-config-validate.c @@ -230,6 +230,24 @@ ip_address_validate (ipmi_config_state_data_t *state_data, return (IPMI_CONFIG_VALIDATE_INVALID_VALUE); } +ipmi_config_validate_t +ipv6_address_validate (ipmi_config_state_data_t *state_data, + const char *section_name, + const char *key_name, + const char *value) +{ + struct in6_addr a; + + assert (state_data); + assert (section_name); + assert (key_name); + assert (value); + + if (inet_pton (AF_INET6, value, &a)) + return (IPMI_CONFIG_VALIDATE_VALID_VALUE); + return (IPMI_CONFIG_VALIDATE_INVALID_VALUE); +} + ipmi_config_validate_t mac_address_validate (ipmi_config_state_data_t *state_data, const char *section_name, @@ -335,6 +353,22 @@ ip_address_source_number_validate (ipmi_config_state_data_t *state_data, return (IPMI_CONFIG_VALIDATE_INVALID_VALUE); } +ipmi_config_validate_t +ipv6_ipv4_addressing_enables_validate (ipmi_config_state_data_t *state_data, + const char *section_name, + const char *key_name, + const char *value) +{ + assert (state_data); + assert (section_name); + assert (key_name); + assert (value); + + if (ipv6_ipv4_addressing_enables_number (value) >= 0) + return (IPMI_CONFIG_VALIDATE_VALID_VALUE); + return (IPMI_CONFIG_VALIDATE_INVALID_VALUE); +} + ipmi_config_validate_t power_restore_policy_number_validate (ipmi_config_state_data_t *state_data, const char *section_name, diff --git a/ipmi-config/ipmi-config-validate.h b/ipmi-config/ipmi-config-validate.h index 7875772e8..528d89763 100644 --- a/ipmi-config/ipmi-config-validate.h +++ b/ipmi-config/ipmi-config-validate.h @@ -79,6 +79,11 @@ ipmi_config_validate_t ip_address_validate (ipmi_config_state_data_t *state_data const char *key_name, const char *value); +ipmi_config_validate_t ipv6_address_validate (ipmi_config_state_data_t *state_data, + const char *section_name, + const char *key_name, + const char *value); + ipmi_config_validate_t mac_address_validate (ipmi_config_state_data_t *state_data, const char *section_name, const char *key_name, @@ -109,6 +114,11 @@ ipmi_config_validate_t ip_address_source_number_validate (ipmi_config_state_data const char *key_name, const char *value); +ipmi_config_validate_t ipv6_ipv4_addressing_enables_validate (ipmi_config_state_data_t *state_data, + const char *section_name, + const char *key_name, + const char *value); + ipmi_config_validate_t power_restore_policy_number_validate (ipmi_config_state_data_t *state_data, const char *section_name, const char *key_name, diff --git a/ipmi-config/ipmi-config.h b/ipmi-config/ipmi-config.h index 83a8e31f2..ead434b61 100644 --- a/ipmi-config/ipmi-config.h +++ b/ipmi-config/ipmi-config.h @@ -276,6 +276,13 @@ struct ipmi_config_state_data int enable_user_after_password_len; ipmi_config_enable_user_after_password_t *enable_user_after_password; + /* lamont: caching to make ipv6 go faster */ + int ipv6_ipv4_support_initialized; + uint8_t ipv6_ipv4_support_channel_number; + uint8_t ipv6_ipv4_support_supports_ipv6_only; + uint8_t ipv6_ipv4_support_supports_ipv6_and_ipv4_simultaneously; + uint8_t ipv6_ipv4_support_supports_ipv6_destination_address_for_lan_alert; + /* achu: caching to make lan authentication enables go faster */ int authentication_type_initialized; uint8_t authentication_type_channel_number;