Recommend to set boot mode explicitly
A future release will change the default boot mode from legacy BIOS to UEFI. The default boot mode can be set to [deploy]/default_boot_mode option for hardware types which support setting boot mode. Otherwise, the default boot mode is hard-coded as legacy BIOS. This patch recommends to set boot mode explicitly in a help message of [deploy]/default_boot_mode option. A warning message is also logged when a default hard-coded boot mode is used. This message is logged once even if there are multiple nodes whose boot modes are configured explicitly for not emitting too many messages. Change-Id: Ib90ebf59ba72d49cb757e44f3741b5373a411ddf Story: 2003936 Task: 27475
This commit is contained in:
parent
82190ae484
commit
60767aee61
@ -279,7 +279,7 @@ def create_pxe_config(task, pxe_options, template=None, ipxe_enabled=False):
|
|||||||
pxe_config_file_path = get_pxe_config_file_path(
|
pxe_config_file_path = get_pxe_config_file_path(
|
||||||
task.node.uuid,
|
task.node.uuid,
|
||||||
ipxe_enabled=ipxe_enabled)
|
ipxe_enabled=ipxe_enabled)
|
||||||
is_uefi_boot_mode = (boot_mode_utils.get_boot_mode_for_deploy(task.node)
|
is_uefi_boot_mode = (boot_mode_utils.get_boot_mode(task.node)
|
||||||
== 'uefi')
|
== 'uefi')
|
||||||
|
|
||||||
# grub bootloader panics with '{}' around any of its tags in its
|
# grub bootloader panics with '{}' around any of its tags in its
|
||||||
@ -351,8 +351,7 @@ def clean_up_pxe_config(task, ipxe_enabled=False):
|
|||||||
"""
|
"""
|
||||||
LOG.debug("Cleaning up PXE config for node %s", task.node.uuid)
|
LOG.debug("Cleaning up PXE config for node %s", task.node.uuid)
|
||||||
|
|
||||||
is_uefi_boot_mode = (boot_mode_utils.get_boot_mode_for_deploy(task.node)
|
is_uefi_boot_mode = (boot_mode_utils.get_boot_mode(task.node) == 'uefi')
|
||||||
== 'uefi')
|
|
||||||
|
|
||||||
if is_uefi_boot_mode and not ipxe_enabled:
|
if is_uefi_boot_mode and not ipxe_enabled:
|
||||||
api = dhcp_factory.DHCPFactory().provider
|
api = dhcp_factory.DHCPFactory().provider
|
||||||
@ -777,7 +776,7 @@ def build_service_pxe_config(task, instance_image_info,
|
|||||||
iwdi = node.driver_internal_info.get('is_whole_disk_image')
|
iwdi = node.driver_internal_info.get('is_whole_disk_image')
|
||||||
deploy_utils.switch_pxe_config(
|
deploy_utils.switch_pxe_config(
|
||||||
pxe_config_path, root_uuid_or_disk_id,
|
pxe_config_path, root_uuid_or_disk_id,
|
||||||
boot_mode_utils.get_boot_mode_for_deploy(node),
|
boot_mode_utils.get_boot_mode(node),
|
||||||
iwdi, deploy_utils.is_trusted_boot_requested(node),
|
iwdi, deploy_utils.is_trusted_boot_requested(node),
|
||||||
deploy_utils.is_iscsi_boot(task), ramdisk_boot,
|
deploy_utils.is_iscsi_boot(task), ramdisk_boot,
|
||||||
ipxe_enabled=ipxe_enabled)
|
ipxe_enabled=ipxe_enabled)
|
||||||
@ -852,7 +851,7 @@ def get_volume_pxe_options(task):
|
|||||||
|
|
||||||
def validate_boot_parameters_for_trusted_boot(node):
|
def validate_boot_parameters_for_trusted_boot(node):
|
||||||
"""Check if boot parameters are valid for trusted boot."""
|
"""Check if boot parameters are valid for trusted boot."""
|
||||||
boot_mode = boot_mode_utils.get_boot_mode_for_deploy(node)
|
boot_mode = boot_mode_utils.get_boot_mode(node)
|
||||||
boot_option = deploy_utils.get_boot_option(node)
|
boot_option = deploy_utils.get_boot_option(node)
|
||||||
is_whole_disk_image = node.driver_internal_info.get('is_whole_disk_image')
|
is_whole_disk_image = node.driver_internal_info.get('is_whole_disk_image')
|
||||||
# 'is_whole_disk_image' is not supported by trusted boot, because there is
|
# 'is_whole_disk_image' is not supported by trusted boot, because there is
|
||||||
@ -906,7 +905,7 @@ def prepare_instance_pxe_config(task, image_info,
|
|||||||
ipxe_enabled=ipxe_enabled)
|
ipxe_enabled=ipxe_enabled)
|
||||||
deploy_utils.switch_pxe_config(
|
deploy_utils.switch_pxe_config(
|
||||||
pxe_config_path, None,
|
pxe_config_path, None,
|
||||||
boot_mode_utils.get_boot_mode_for_deploy(node), False,
|
boot_mode_utils.get_boot_mode(node), False,
|
||||||
iscsi_boot=iscsi_boot, ramdisk_boot=ramdisk_boot,
|
iscsi_boot=iscsi_boot, ramdisk_boot=ramdisk_boot,
|
||||||
ipxe_enabled=ipxe_enabled)
|
ipxe_enabled=ipxe_enabled)
|
||||||
|
|
||||||
|
@ -85,10 +85,13 @@ opts = [
|
|||||||
help=_('Default boot mode to use when no boot mode is '
|
help=_('Default boot mode to use when no boot mode is '
|
||||||
'requested in node\'s driver_info, capabilities or '
|
'requested in node\'s driver_info, capabilities or '
|
||||||
'in the `instance_info` configuration. Currently the '
|
'in the `instance_info` configuration. Currently the '
|
||||||
'default boot mode is "%(bios)s". This option only '
|
'default boot mode is "%(bios)s", but it will be '
|
||||||
'has effect when management interface supports boot '
|
'changed to "%(uefi)s in the future. It is recommended '
|
||||||
'mode management') % {
|
'to set an explicit value for this option. This option '
|
||||||
'bios': boot_modes.LEGACY_BIOS}),
|
'only has effect when management interface supports '
|
||||||
|
'boot mode management') % {
|
||||||
|
'bios': boot_modes.LEGACY_BIOS,
|
||||||
|
'uefi': boot_modes.UEFI}),
|
||||||
cfg.BoolOpt('configdrive_use_object_store',
|
cfg.BoolOpt('configdrive_use_object_store',
|
||||||
default=False,
|
default=False,
|
||||||
deprecated_group='conductor',
|
deprecated_group='conductor',
|
||||||
|
@ -237,11 +237,8 @@ class AgentDeployMixin(agent_base_vendor.AgentDeployMixin):
|
|||||||
for label in PARTITION_IMAGE_LABELS:
|
for label in PARTITION_IMAGE_LABELS:
|
||||||
image_info[label] = node.instance_info.get(label)
|
image_info[label] = node.instance_info.get(label)
|
||||||
boot_option = deploy_utils.get_boot_option(node)
|
boot_option = deploy_utils.get_boot_option(node)
|
||||||
boot_mode = boot_mode_utils.get_boot_mode_for_deploy(node)
|
image_info['deploy_boot_mode'] = (
|
||||||
if boot_mode:
|
boot_mode_utils.get_boot_mode(node))
|
||||||
image_info['deploy_boot_mode'] = boot_mode
|
|
||||||
else:
|
|
||||||
image_info['deploy_boot_mode'] = 'bios'
|
|
||||||
image_info['boot_option'] = boot_option
|
image_info['boot_option'] = boot_option
|
||||||
disk_label = deploy_utils.get_disk_label(node)
|
disk_label = deploy_utils.get_disk_label(node)
|
||||||
if disk_label is not None:
|
if disk_label is not None:
|
||||||
@ -336,7 +333,7 @@ class AgentDeployMixin(agent_base_vendor.AgentDeployMixin):
|
|||||||
|
|
||||||
efi_sys_uuid = None
|
efi_sys_uuid = None
|
||||||
if not iwdi:
|
if not iwdi:
|
||||||
if boot_mode_utils.get_boot_mode_for_deploy(node) == 'uefi':
|
if boot_mode_utils.get_boot_mode(node) == 'uefi':
|
||||||
efi_sys_uuid = (self._get_uuid_from_result(task,
|
efi_sys_uuid = (self._get_uuid_from_result(task,
|
||||||
'efi_system_partition_uuid'))
|
'efi_system_partition_uuid'))
|
||||||
|
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
|
|
||||||
from oslo_log import log as logging
|
from oslo_log import log as logging
|
||||||
|
|
||||||
|
from ironic.common import boot_modes
|
||||||
from ironic.common import exception
|
from ironic.common import exception
|
||||||
from ironic.common.i18n import _
|
from ironic.common.i18n import _
|
||||||
from ironic.common import utils as common_utils
|
from ironic.common import utils as common_utils
|
||||||
@ -24,6 +25,8 @@ from ironic.drivers import utils as driver_utils
|
|||||||
|
|
||||||
LOG = logging.getLogger(__name__)
|
LOG = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
warn_about_default_boot_mode = False
|
||||||
|
|
||||||
|
|
||||||
def _set_boot_mode_on_bm(task, ironic_boot_mode, fail_if_unsupported=False):
|
def _set_boot_mode_on_bm(task, ironic_boot_mode, fail_if_unsupported=False):
|
||||||
try:
|
try:
|
||||||
@ -266,3 +269,29 @@ def get_boot_mode_for_deploy(node):
|
|||||||
{'boot_mode': boot_mode, 'node': node.uuid})
|
{'boot_mode': boot_mode, 'node': node.uuid})
|
||||||
|
|
||||||
return boot_mode
|
return boot_mode
|
||||||
|
|
||||||
|
|
||||||
|
def get_boot_mode(node):
|
||||||
|
"""Returns the boot mode.
|
||||||
|
|
||||||
|
:param node: an ironic node object.
|
||||||
|
:returns: 'bios' or 'uefi'
|
||||||
|
:raises: InvalidParameterValue, if the node boot mode disagrees with
|
||||||
|
the boot mode set to node properties/capabilities
|
||||||
|
"""
|
||||||
|
boot_mode = get_boot_mode_for_deploy(node)
|
||||||
|
if boot_mode:
|
||||||
|
return boot_mode
|
||||||
|
# TODO(hshiina): The default boot mode will be changed to UEFI.
|
||||||
|
global warn_about_default_boot_mode
|
||||||
|
if not warn_about_default_boot_mode:
|
||||||
|
warn_about_default_boot_mode = True
|
||||||
|
LOG.warning('Boot mode is not configured for node %(node_uuid)s '
|
||||||
|
'explicitly. The default boot mode is "%(bios)s", but, '
|
||||||
|
'the default will be changed to "%(uefi)s" in the future. '
|
||||||
|
'It is recommended to set the boot option into '
|
||||||
|
'properties/capabilities/boot_mode for all nodes.',
|
||||||
|
{'node_uuid': node.uuid,
|
||||||
|
'bios': boot_modes.LEGACY_BIOS,
|
||||||
|
'uefi': boot_modes.UEFI})
|
||||||
|
return boot_modes.LEGACY_BIOS
|
||||||
|
@ -690,7 +690,7 @@ def try_set_boot_device(task, device, persistent=True):
|
|||||||
persistent=persistent)
|
persistent=persistent)
|
||||||
except exception.IPMIFailure:
|
except exception.IPMIFailure:
|
||||||
with excutils.save_and_reraise_exception() as ctxt:
|
with excutils.save_and_reraise_exception() as ctxt:
|
||||||
if get_boot_mode_for_deploy(task.node) == 'uefi':
|
if boot_mode_utils.get_boot_mode(task.node) == 'uefi':
|
||||||
ctxt.reraise = False
|
ctxt.reraise = False
|
||||||
LOG.warning("ipmitool is unable to set boot device while "
|
LOG.warning("ipmitool is unable to set boot device while "
|
||||||
"the node %s is in UEFI boot mode. Please set "
|
"the node %s is in UEFI boot mode. Please set "
|
||||||
@ -722,7 +722,7 @@ def get_pxe_boot_file(node):
|
|||||||
cpu_arch = node.properties.get('cpu_arch')
|
cpu_arch = node.properties.get('cpu_arch')
|
||||||
boot_file = CONF.pxe.pxe_bootfile_name_by_arch.get(cpu_arch)
|
boot_file = CONF.pxe.pxe_bootfile_name_by_arch.get(cpu_arch)
|
||||||
if boot_file is None:
|
if boot_file is None:
|
||||||
if get_boot_mode_for_deploy(node) == 'uefi':
|
if boot_mode_utils.get_boot_mode(node) == 'uefi':
|
||||||
boot_file = CONF.pxe.uefi_pxe_bootfile_name
|
boot_file = CONF.pxe.uefi_pxe_bootfile_name
|
||||||
else:
|
else:
|
||||||
boot_file = CONF.pxe.pxe_bootfile_name
|
boot_file = CONF.pxe.pxe_bootfile_name
|
||||||
@ -743,7 +743,7 @@ def get_pxe_config_template(node):
|
|||||||
cpu_arch = node.properties.get('cpu_arch')
|
cpu_arch = node.properties.get('cpu_arch')
|
||||||
config_template = CONF.pxe.pxe_config_template_by_arch.get(cpu_arch)
|
config_template = CONF.pxe.pxe_config_template_by_arch.get(cpu_arch)
|
||||||
if config_template is None:
|
if config_template is None:
|
||||||
if get_boot_mode_for_deploy(node) == 'uefi':
|
if boot_mode_utils.get_boot_mode(node) == 'uefi':
|
||||||
config_template = CONF.pxe.uefi_pxe_config_template
|
config_template = CONF.pxe.uefi_pxe_config_template
|
||||||
else:
|
else:
|
||||||
config_template = CONF.pxe.pxe_config_template
|
config_template = CONF.pxe.pxe_config_template
|
||||||
|
@ -185,7 +185,7 @@ def _get_boot_iso(task, root_uuid):
|
|||||||
# Option 3 - Create boot_iso from kernel/ramdisk, upload to Swift
|
# Option 3 - Create boot_iso from kernel/ramdisk, upload to Swift
|
||||||
# or web server and provide its name.
|
# or web server and provide its name.
|
||||||
deploy_iso_uuid = deploy_info['ilo_deploy_iso']
|
deploy_iso_uuid = deploy_info['ilo_deploy_iso']
|
||||||
boot_mode = boot_mode_utils.get_boot_mode_for_deploy(task.node)
|
boot_mode = boot_mode_utils.get_boot_mode(task.node)
|
||||||
boot_iso_object_name = _get_boot_iso_object_name(task.node)
|
boot_iso_object_name = _get_boot_iso_object_name(task.node)
|
||||||
kernel_params = ""
|
kernel_params = ""
|
||||||
if deploy_utils.get_boot_option(task.node) == "ramdisk":
|
if deploy_utils.get_boot_option(task.node) == "ramdisk":
|
||||||
@ -525,7 +525,7 @@ class IloVirtualMediaBoot(base.BootInterface):
|
|||||||
"""
|
"""
|
||||||
ilo_common.cleanup_vmedia_boot(task)
|
ilo_common.cleanup_vmedia_boot(task)
|
||||||
|
|
||||||
boot_mode = boot_mode_utils.get_boot_mode_for_deploy(task.node)
|
boot_mode = boot_mode_utils.get_boot_mode(task.node)
|
||||||
boot_option = deploy_utils.get_boot_option(task.node)
|
boot_option = deploy_utils.get_boot_option(task.node)
|
||||||
|
|
||||||
if deploy_utils.is_iscsi_boot(task):
|
if deploy_utils.is_iscsi_boot(task):
|
||||||
@ -700,7 +700,7 @@ class IloPXEBoot(pxe.PXEBoot):
|
|||||||
# Need to enable secure boot, if being requested
|
# Need to enable secure boot, if being requested
|
||||||
ilo_common.update_secure_boot_mode(task, True)
|
ilo_common.update_secure_boot_mode(task, True)
|
||||||
|
|
||||||
boot_mode = boot_mode_utils.get_boot_mode_for_deploy(task.node)
|
boot_mode = boot_mode_utils.get_boot_mode(task.node)
|
||||||
|
|
||||||
if deploy_utils.is_iscsi_boot(task) and boot_mode == 'uefi':
|
if deploy_utils.is_iscsi_boot(task) and boot_mode == 'uefi':
|
||||||
# Need to set 'ilo_uefi_iscsi_boot' param for clean up
|
# Need to set 'ilo_uefi_iscsi_boot' param for clean up
|
||||||
|
@ -938,7 +938,7 @@ class IPMIManagement(base.ManagementInterface):
|
|||||||
# uefi mode, this will work with newer and older versions of the
|
# uefi mode, this will work with newer and older versions of the
|
||||||
# ipmitool utility. Also see:
|
# ipmitool utility. Also see:
|
||||||
# https://bugs.launchpad.net/ironic/+bug/1611306
|
# https://bugs.launchpad.net/ironic/+bug/1611306
|
||||||
boot_mode = boot_mode_utils.get_boot_mode_for_deploy(task.node)
|
boot_mode = boot_mode_utils.get_boot_mode(task.node)
|
||||||
if persistent and boot_mode == 'uefi':
|
if persistent and boot_mode == 'uefi':
|
||||||
raw_cmd = ('0x00 0x08 0x05 0xe0 %s 0x00 0x00 0x00' %
|
raw_cmd = ('0x00 0x08 0x05 0xe0 %s 0x00 0x00 0x00' %
|
||||||
BOOT_DEVICE_HEXA_MAP[device])
|
BOOT_DEVICE_HEXA_MAP[device])
|
||||||
|
@ -305,7 +305,7 @@ def _prepare_boot_iso(task, root_uuid):
|
|||||||
or image_properties['ramdisk_id'])
|
or image_properties['ramdisk_id'])
|
||||||
|
|
||||||
deploy_iso_href = deploy_info['irmc_deploy_iso']
|
deploy_iso_href = deploy_info['irmc_deploy_iso']
|
||||||
boot_mode = boot_mode_utils.get_boot_mode_for_deploy(task.node)
|
boot_mode = boot_mode_utils.get_boot_mode(task.node)
|
||||||
kernel_params = CONF.pxe.pxe_append_params
|
kernel_params = CONF.pxe.pxe_append_params
|
||||||
|
|
||||||
boot_iso_filename = _get_iso_name(task.node, label='boot')
|
boot_iso_filename = _get_iso_name(task.node, label='boot')
|
||||||
|
@ -117,7 +117,7 @@ def get_deploy_info(node, address, iqn, port=None, lun='1', conv_flags=None):
|
|||||||
'ephemeral_mb': i_info['ephemeral_mb'],
|
'ephemeral_mb': i_info['ephemeral_mb'],
|
||||||
'preserve_ephemeral': i_info['preserve_ephemeral'],
|
'preserve_ephemeral': i_info['preserve_ephemeral'],
|
||||||
'boot_option': deploy_utils.get_boot_option(node),
|
'boot_option': deploy_utils.get_boot_option(node),
|
||||||
'boot_mode': _get_boot_mode(node),
|
'boot_mode': boot_mode_utils.get_boot_mode(node),
|
||||||
'cpu_arch': node.properties.get('cpu_arch')})
|
'cpu_arch': node.properties.get('cpu_arch')})
|
||||||
|
|
||||||
# Append disk label if specified
|
# Append disk label if specified
|
||||||
@ -299,18 +299,6 @@ def do_agent_iscsi_deploy(task, agent_client):
|
|||||||
return uuid_dict_returned
|
return uuid_dict_returned
|
||||||
|
|
||||||
|
|
||||||
def _get_boot_mode(node):
|
|
||||||
"""Gets the boot mode.
|
|
||||||
|
|
||||||
:param node: A single Node.
|
|
||||||
:returns: A string representing the boot mode type. Defaults to 'bios'.
|
|
||||||
"""
|
|
||||||
boot_mode = boot_mode_utils.get_boot_mode_for_deploy(node)
|
|
||||||
if boot_mode:
|
|
||||||
return boot_mode
|
|
||||||
return "bios"
|
|
||||||
|
|
||||||
|
|
||||||
@METRICS.timer('validate')
|
@METRICS.timer('validate')
|
||||||
def validate(task):
|
def validate(task):
|
||||||
"""Validates the pre-requisites for iSCSI deploy.
|
"""Validates the pre-requisites for iSCSI deploy.
|
||||||
|
55
ironic/tests/unit/drivers/modules/test_boot_mode_utils.py
Normal file
55
ironic/tests/unit/drivers/modules/test_boot_mode_utils.py
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
# Copyright 2018 FUJITSU LIMITED.
|
||||||
|
# All Rights Reserved.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
import mock
|
||||||
|
|
||||||
|
from ironic.common import boot_modes
|
||||||
|
from ironic.drivers.modules import boot_mode_utils
|
||||||
|
from ironic.tests import base as tests_base
|
||||||
|
from ironic.tests.unit.objects import utils as obj_utils
|
||||||
|
|
||||||
|
|
||||||
|
class GetBootModeTestCase(tests_base.TestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(GetBootModeTestCase, self).setUp()
|
||||||
|
self.node = obj_utils.get_test_node(self.context,
|
||||||
|
driver='fake-hardware')
|
||||||
|
|
||||||
|
@mock.patch.object(boot_mode_utils, 'get_boot_mode_for_deploy',
|
||||||
|
autospect=True)
|
||||||
|
def test_get_boot_mode_bios(self, mock_for_deploy):
|
||||||
|
mock_for_deploy.return_value = boot_modes.LEGACY_BIOS
|
||||||
|
boot_mode = boot_mode_utils.get_boot_mode(self.node)
|
||||||
|
self.assertEqual(boot_modes.LEGACY_BIOS, boot_mode)
|
||||||
|
|
||||||
|
@mock.patch.object(boot_mode_utils, 'get_boot_mode_for_deploy',
|
||||||
|
autospec=True)
|
||||||
|
def test_get_boot_mode_uefi(self, mock_for_deploy):
|
||||||
|
mock_for_deploy.return_value = boot_modes.UEFI
|
||||||
|
boot_mode = boot_mode_utils.get_boot_mode(self.node)
|
||||||
|
self.assertEqual(boot_modes.UEFI, boot_mode)
|
||||||
|
|
||||||
|
@mock.patch.object(boot_mode_utils, 'LOG', autospect=True)
|
||||||
|
@mock.patch.object(boot_mode_utils, 'get_boot_mode_for_deploy',
|
||||||
|
autospect=True)
|
||||||
|
def test_get_boot_mode_default(self, mock_for_deploy, mock_log):
|
||||||
|
boot_mode_utils.warn_about_default_boot_mode = False
|
||||||
|
mock_for_deploy.return_value = None
|
||||||
|
boot_mode = boot_mode_utils.get_boot_mode(self.node)
|
||||||
|
self.assertEqual(boot_modes.LEGACY_BIOS, boot_mode)
|
||||||
|
boot_mode = boot_mode_utils.get_boot_mode(self.node)
|
||||||
|
self.assertEqual(boot_modes.LEGACY_BIOS, boot_mode)
|
||||||
|
self.assertEqual(1, mock_log.warning.call_count)
|
10
releasenotes/notes/uefi-first-prepare-e7fa1e2a78b4af99.yaml
Normal file
10
releasenotes/notes/uefi-first-prepare-e7fa1e2a78b4af99.yaml
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
---
|
||||||
|
upgrade:
|
||||||
|
- |
|
||||||
|
A future release will change the default value of
|
||||||
|
``[deploy]/default_boot_mode`` from "bios" to "uefi". It is recommended to
|
||||||
|
set an explicit value for this option. For hardware types which don't
|
||||||
|
support setting boot mode, a future release will assume boot mode is set
|
||||||
|
to UEFI if no boot mode is set to node's capabilities. It is also
|
||||||
|
recommended to set ``boot_mode`` into ``properties/capabilities``
|
||||||
|
of a node.
|
Loading…
x
Reference in New Issue
Block a user