[sw/dif] Add FW override support to entropy src.
The entropy src allows the firmware to disconnect the raw entropy from
the pre-conditioner block at the output of the health checks. This
allows firmware to access raw entropy.
The firmware can also write data into the pre-conditioner block to
provide firmware-controlled entropy at the output of the entropy src.
This functionality can be useful to implement KAT tests of the
pre-conditioner block.
Signed-off-by: Miguel Osorio <miguelosorio@google.com>
diff --git a/sw/device/lib/dif/dif_entropy_src.c b/sw/device/lib/dif/dif_entropy_src.c
index c6cdee6..90afed8 100644
--- a/sw/device/lib/dif/dif_entropy_src.c
+++ b/sw/device/lib/dif/dif_entropy_src.c
@@ -16,7 +16,7 @@
* Sets the `entropy` source configuration register with the settings
* derived from `config`.
*/
-static void set_config_register(const dif_entropy_src_t *entropy_src,
+static void config_register_set(const dif_entropy_src_t *entropy_src,
const dif_entropy_src_config_t *config) {
// TODO: Make this configurable at the API level.
uint32_t reg = bitfield_field32_write(
@@ -26,6 +26,10 @@
reg = bitfield_field32_write(reg, ENTROPY_SRC_CONF_HEALTH_TEST_CLR_FIELD,
health_clr_sel);
+ reg = bitfield_field32_write(reg,
+ ENTROPY_SRC_CONF_ENTROPY_DATA_REG_ENABLE_FIELD,
+ config->route_to_firmware ? 0xa : 0x5);
+
// Configure single RNG bit mode
uint32_t rng_bit_en =
(config->single_bit_mode == kDifEntropySrcSingleBitModeDisabled) ? 0x5
@@ -47,6 +51,35 @@
mmio_region_write32(entropy_src->base_addr, ENTROPY_SRC_CONF_REG_OFFSET, reg);
}
+/**
+ * Sets the `entropy` source firmware override register with the settings
+ * derived from `config`.
+ */
+static dif_result_t fw_override_set(
+ const dif_entropy_src_t *entropy_src,
+ const dif_entropy_src_fw_override_config_t *config) {
+ if (config->buffer_threshold > kDifEntropySrcFifoMaxCapacity) {
+ return kDifBadArg;
+ }
+
+ if (config->entropy_insert_enable && !config->enable) {
+ return kDifBadArg;
+ }
+ mmio_region_write32(entropy_src->base_addr,
+ ENTROPY_SRC_OBSERVE_FIFO_THRESH_REG_OFFSET,
+ config->buffer_threshold);
+
+ uint32_t reg =
+ bitfield_field32_write(0, ENTROPY_SRC_FW_OV_CONTROL_FW_OV_MODE_FIELD,
+ config->enable ? 0xa : 0x5);
+ reg = bitfield_field32_write(
+ reg, ENTROPY_SRC_FW_OV_CONTROL_FW_OV_ENTROPY_INSERT_FIELD,
+ config->entropy_insert_enable ? 0xa : 0x5);
+ mmio_region_write32(entropy_src->base_addr,
+ ENTROPY_SRC_FW_OV_CONTROL_REG_OFFSET, reg);
+ return kDifOk;
+}
+
dif_result_t dif_entropy_src_init(mmio_region_t base_addr,
dif_entropy_src_t *entropy_src) {
if (entropy_src == NULL) {
@@ -62,8 +95,15 @@
return kDifBadArg;
}
- // Conditioning bypass is hardcoded to enabled. Bypass is not intended as
- // a regular mode of operation.
+ dif_result_t result = fw_override_set(entropy_src, &config.fw_override);
+ if (result != kDifOk) {
+ return result;
+ }
+
+ // TODO: Add test configuration parameters.
+
+ // Conditioning bypass is hardcoded to disabled. Conditioning bypass is not
+ // intended as a regular mode of operation.
uint32_t es_route_val = config.route_to_firmware ? 0xa : 0x5;
uint32_t reg = bitfield_field32_write(
0, ENTROPY_SRC_ENTROPY_CONTROL_ES_ROUTE_FIELD, es_route_val);
@@ -71,14 +111,7 @@
0x5);
mmio_region_write32(entropy_src->base_addr,
ENTROPY_SRC_ENTROPY_CONTROL_REG_OFFSET, reg);
-
- // TODO: Add test configuration parameters.
-
- // TODO: Add support for FIFO mode.
- mmio_region_write32(entropy_src->base_addr,
- ENTROPY_SRC_FW_OV_CONTROL_REG_OFFSET, 0x55);
-
- set_config_register(entropy_src, &config);
+ config_register_set(entropy_src, &config);
return kDifOk;
}
@@ -119,9 +152,80 @@
return kDifOk;
}
+dif_result_t dif_entropy_src_fifo_read(const dif_entropy_src_t *entropy_src,
+ uint32_t *buf, size_t len) {
+ if (entropy_src == NULL) {
+ return kDifBadArg;
+ }
+
+ uint32_t reg = mmio_region_read32(entropy_src->base_addr,
+ ENTROPY_SRC_FW_OV_CONTROL_REG_OFFSET);
+ if (bitfield_field32_read(reg, ENTROPY_SRC_FW_OV_CONTROL_FW_OV_MODE_FIELD) !=
+ 0xa) {
+ return kDifError;
+ }
+
+ reg = mmio_region_read32(entropy_src->base_addr,
+ ENTROPY_SRC_OBSERVE_FIFO_THRESH_REG_OFFSET);
+ if (reg < len) {
+ return kDifBadArg;
+ }
+
+ do {
+ reg = mmio_region_read32(entropy_src->base_addr,
+ ENTROPY_SRC_INTR_STATE_REG_OFFSET);
+ } while (!bitfield_bit32_read(
+ reg, ENTROPY_SRC_INTR_STATE_ES_OBSERVE_FIFO_READY_BIT));
+
+ for (size_t i = 0; i < len; ++i) {
+ reg = mmio_region_read32(entropy_src->base_addr,
+ ENTROPY_SRC_FW_OV_RD_DATA_REG_OFFSET);
+ if (buf != NULL) {
+ buf[i] = reg;
+ }
+ }
+
+ // Clear the status bit.
+ reg = bitfield_bit32_write(
+ 0, ENTROPY_SRC_INTR_STATE_ES_OBSERVE_FIFO_READY_BIT, true);
+ mmio_region_write32(entropy_src->base_addr, ENTROPY_SRC_INTR_STATE_REG_OFFSET,
+ reg);
+ return kDifOk;
+}
+
+dif_result_t dif_entropy_src_fifo_write(const dif_entropy_src_t *entropy_src,
+ const uint32_t *buf, size_t len) {
+ if (entropy_src == NULL || buf == NULL) {
+ return kDifBadArg;
+ }
+
+ uint32_t reg = mmio_region_read32(entropy_src->base_addr,
+ ENTROPY_SRC_FW_OV_CONTROL_REG_OFFSET);
+ if (bitfield_field32_read(reg, ENTROPY_SRC_FW_OV_CONTROL_FW_OV_MODE_FIELD) !=
+ 0xa ||
+ bitfield_field32_read(
+ reg, ENTROPY_SRC_FW_OV_CONTROL_FW_OV_ENTROPY_INSERT_FIELD) != 0xa) {
+ return kDifBadArg;
+ }
+
+ for (size_t i = 0; i < len; ++i) {
+ mmio_region_write32(entropy_src->base_addr,
+ ENTROPY_SRC_FW_OV_WR_DATA_REG_OFFSET, buf[i]);
+ }
+ return kDifOk;
+}
+
dif_result_t dif_entropy_src_disable(const dif_entropy_src_t *entropy_src) {
+ if (entropy_src == NULL) {
+ return kDifBadArg;
+ }
// TODO: should first check if entropy is locked and return error if it is.
mmio_region_write32(entropy_src->base_addr, ENTROPY_SRC_CONF_REG_OFFSET, 0);
- return kDifOk;
+ const dif_entropy_src_fw_override_config_t kDefaultFwOverrideConfig = {
+ .enable = false,
+ .entropy_insert_enable = false,
+ .buffer_threshold = kDifEntropyFifoIntDefaultThreshold,
+ };
+ return fw_override_set(entropy_src, &kDefaultFwOverrideConfig);
}
diff --git a/sw/device/lib/dif/dif_entropy_src.h b/sw/device/lib/dif/dif_entropy_src.h
index 691e1eb..8e2387f 100644
--- a/sw/device/lib/dif/dif_entropy_src.h
+++ b/sw/device/lib/dif/dif_entropy_src.h
@@ -29,6 +29,14 @@
*/
// TODO: Synchronize value with hardware.
kDifEntropySrcFifoMaxCapacity = 64,
+
+ /**
+ * Default firmware observe FIFO threshold.
+ *
+ * Default value used to trigger the `kDifEntropySrcIrqEsObserveFifoReady`
+ * interrupt when enabled.
+ */
+ kDifEntropyFifoIntDefaultThreshold = 32,
};
/**
@@ -175,6 +183,33 @@
} dif_entropy_src_test_config_t;
/**
+ * Firmware override parameters for an entropy source.
+ */
+typedef struct dif_entropy_src_fw_override_config {
+ /**
+ * Enables firmware monitoring of the post-health test entropy via
+ * `dif_entropy_fifo_read()` calls.
+ */
+ bool enable;
+
+ /**
+ * Enables fimrware to insert entropy bits back into the pre-coditioner block
+ * via `dif_entropy_fifo_write()` calls. This feature is useful when the
+ * firmware is required to implement additional health checks, and to perform
+ * known answer tests of the preconditioner function.
+ *
+ * This field requires `fw_override_enable` to be set.
+ */
+ bool entropy_insert_enable;
+
+ /**
+ * This field sets the depth of the observe FIFO hardware buffer used when
+ * `fw_override_enable` is set to true.
+ */
+ uint8_t buffer_threshold;
+} dif_entropy_src_fw_override_config_t;
+
+/**
* Runtime configuration for an entropy source.
*
* This struct describes runtime information for one-time configuration of the
@@ -232,6 +267,10 @@
*/
dif_entropy_src_test_config_t test_config;
+ /**
+ * Configuration parameters for firmware override buffer.
+ */
+ dif_entropy_src_fw_override_config_t fw_override;
} dif_entropy_src_config_t;
/**
@@ -407,9 +446,9 @@
/**
* Performs an override read from the entropy pipeline.
*
- * This function pauses entropy flow out of the pre-conditioner FIFO and
- * instead flows words into `buf`. Normal operation of the entropy pipeline
- * will not resume until `dif_entropy_src_fifo_reconnect()` is called.
+ * Entropy source must be configured with firmware override mode enabled, and
+ * the `len` parameter must be strictly less than the FIFO threshold set in the
+ * firware override parameters.
*
* `buf` may be `NULL`; in this case, reads will be discarded.
*
@@ -425,9 +464,9 @@
/**
* Performs an override write to the entropy pipeline.
*
- * This function pauses entropy flow into the pre-conditioner FIFO and
- * instead flows words out of `buf`. Normal operation of the entropy pipeline
- * will not resume until `dif_entropy_src_fifo_reconnect()` is called.
+ * Entropy source must be configured with firmware override and insert mode
+ * enabled, otherwise the function will return
+ * `kDifBadArg`.
*
* @param entropy An entropy source handle.
* @param buf A buffer to push words from into the pipeline.
@@ -439,58 +478,6 @@
const uint32_t *buf, size_t len);
/**
- * Gets the current number of entries in the pre-conditioner FIFO.
- *
- * This function pauses the flow through the FIFO.
- *
- * @param entropy An entropy source handle.
- * @param[out] len The number of words in the FIFO.
- * @return The result of the operation.
- */
-OT_WARN_UNUSED_RESULT
-dif_result_t dif_entropy_src_fifo_get_len(const dif_entropy_src_t *entropy_src,
- uint8_t *len);
-
-/**
- * Gets the current capacity of the pre-conditioner FIFO.
- *
- * @param entropy An entropy source handle.
- * @param[out] capacity The number of words of capacity in the FIFO.
- * @return The result of the operation.
- */
-OT_WARN_UNUSED_RESULT
-dif_result_t dif_entropy_src_fifo_get_capacity(
- const dif_entropy_src_t *entropy_src, uint8_t *capacity);
-
-/**
- * Sets the current capacity of the pre-conditioner FIFO.
- *
- * The `capacity` value must be less or equal to the physical capacity
- * of the fifo, defined as `kDifEntropySrcFifoMaxCapacity`.
- *
- * @param entropy An entropy source handle.
- * @param capacity The new capacity for the FIFO.
- * @return The result of the operation.
- */
-OT_WARN_UNUSED_RESULT
-dif_result_t dif_entropy_src_fifo_set_capacity(
- const dif_entropy_src_t *entropy_src, uint8_t capacity);
-
-/**
- * Reconnects the entropy pipeline after an operation that pauses it.
- *
- * This is a separate function call to avoid races between software and hardware
- * when performing multiple such operations, such as getting the length followed
- * by a read.
- *
- * @param entropy An entropy source handle.
- * @return The result of the operation.
- */
-OT_WARN_UNUSED_RESULT
-dif_result_t dif_entropy_src_fifo_reconnect(
- const dif_entropy_src_t *entropy_src);
-
-/**
* Disables the entropy module
*
* @param entropy An entropy source handle.
diff --git a/sw/device/lib/dif/dif_entropy_src_unittest.cc b/sw/device/lib/dif/dif_entropy_src_unittest.cc
index 14a2354..cd173f2 100644
--- a/sw/device/lib/dif/dif_entropy_src_unittest.cc
+++ b/sw/device/lib/dif/dif_entropy_src_unittest.cc
@@ -40,6 +40,12 @@
.route_to_firmware = false,
.fips_mode = false,
.test_config = {0},
+ .fw_override =
+ {
+ .enable = false,
+ .entropy_insert_enable = false,
+ .buffer_threshold = kDifEntropyFifoIntDefaultThreshold,
+ },
};
};
@@ -47,6 +53,17 @@
EXPECT_EQ(dif_entropy_src_configure(nullptr, {}), kDifBadArg);
}
+TEST_F(ConfigTest, InvalidFifoThreshold) {
+ config_.fw_override.buffer_threshold = 65;
+ EXPECT_EQ(dif_entropy_src_configure(&entropy_src_, config_), kDifBadArg);
+}
+
+TEST_F(ConfigTest, InvalidFwOverrideSettings) {
+ config_.fw_override.enable = false;
+ config_.fw_override.entropy_insert_enable = true;
+ EXPECT_EQ(dif_entropy_src_configure(&entropy_src_, config_), kDifBadArg);
+}
+
struct ConfigParams {
dif_entropy_src_mode_t mode;
dif_entropy_src_single_bit_mode_t single_bit_mode;
@@ -69,13 +86,23 @@
config_.route_to_firmware = test_param.route_to_firmware;
config_.reset_health_test_registers = test_param.reset_health_test_registers;
+ EXPECT_WRITE32(ENTROPY_SRC_OBSERVE_FIFO_THRESH_REG_OFFSET,
+ config_.fw_override.buffer_threshold);
+ EXPECT_WRITE32(
+ ENTROPY_SRC_FW_OV_CONTROL_REG_OFFSET,
+ {
+ {ENTROPY_SRC_FW_OV_CONTROL_FW_OV_MODE_OFFSET,
+ (uint32_t)(config_.fw_override.enable ? 0xa : 0x5)},
+ {ENTROPY_SRC_FW_OV_CONTROL_FW_OV_ENTROPY_INSERT_OFFSET,
+ (uint32_t)(config_.fw_override.entropy_insert_enable ? 0xa : 0x5)},
+ });
+
EXPECT_WRITE32(ENTROPY_SRC_ENTROPY_CONTROL_REG_OFFSET,
{
{ENTROPY_SRC_ENTROPY_CONTROL_ES_ROUTE_OFFSET,
(uint32_t)(test_param.route_to_firmware ? 0xa : 0x5)},
{ENTROPY_SRC_ENTROPY_CONTROL_ES_TYPE_OFFSET, 0x5},
});
- EXPECT_WRITE32(ENTROPY_SRC_FW_OV_CONTROL_REG_OFFSET, 0x55);
// Current dif does not perform a read modified write
// EXPECT_READ32(ENTROPY_SRC_CONF_REG_OFFSET, 0);
@@ -84,6 +111,7 @@
uint32_t route_to_fw = test_param.route_to_firmware ? 0xa : 0x5;
uint32_t enable =
test_param.expected_mode != kDifEntropySrcModeDisabled ? 0xa : 0x5;
+
uint32_t reset_ht = test_param.reset_health_test_registers ? 0xa : 0x5;
EXPECT_WRITE32(
ENTROPY_SRC_CONF_REG_OFFSET,
diff --git a/sw/device/lib/testing/entropy_testutils.c b/sw/device/lib/testing/entropy_testutils.c
index 1d1939a..ae7fb89 100644
--- a/sw/device/lib/testing/entropy_testutils.c
+++ b/sw/device/lib/testing/entropy_testutils.c
@@ -33,7 +33,11 @@
.reset_health_test_registers = false,
.single_bit_mode = kDifEntropySrcSingleBitModeDisabled,
.route_to_firmware = false,
- };
+ .fw_override = {
+ .enable = false,
+ .entropy_insert_enable = false,
+ .buffer_threshold = kDifEntropyFifoIntDefaultThreshold,
+ }};
CHECK_DIF_OK(dif_entropy_src_configure(&entropy_src, config));
}
diff --git a/sw/device/tests/entropy_src_fw_ovr_test.c b/sw/device/tests/entropy_src_fw_ovr_test.c
new file mode 100644
index 0000000..0bca986
--- /dev/null
+++ b/sw/device/tests/entropy_src_fw_ovr_test.c
@@ -0,0 +1,123 @@
+// Copyright lowRISC contributors.
+// Licensed under the Apache License, Version 2.0, see LICENSE for details.
+// SPDX-License-Identifier: Apache-2.0
+
+#include "sw/device/lib/base/memory.h"
+#include "sw/device/lib/base/mmio.h"
+#include "sw/device/lib/dif/dif_entropy_src.h"
+#include "sw/device/lib/runtime/log.h"
+#include "sw/device/lib/testing/check.h"
+#include "sw/device/lib/testing/test_framework/test_main.h"
+
+#include "hw/top_earlgrey/sw/autogen/top_earlgrey.h" // Generated.
+
+const test_config_t kTestConfig;
+
+enum {
+ /**
+ * The size of the buffer used in firmware to process the entropy bits in
+ * firmware override mode.
+ */
+ kEntropyFifoBufferSize = 32,
+};
+
+/**
+ * Flushes the data entropy buffer until there is no data available to read.
+ *
+ * Asserts test error if any of the returned words is equal to zero. Logs the
+ * number of entropy words flushed in a single call.
+ *
+ * @param entropy An entropy source instance.
+ */
+static void entropy_data_flush(dif_entropy_src_t *entropy_src) {
+ uint32_t entropy_bits;
+ uint32_t read_count = 0;
+
+ // TODO: Remove this limit. Entropy source should block if there is no entropy
+ // available in FW override mode.
+ const uint32_t kMaxReadCount = 12;
+
+ while (dif_entropy_src_avail(entropy_src) == kDifOk) {
+ CHECK_DIF_OK(dif_entropy_src_read(entropy_src, &entropy_bits));
+ CHECK(entropy_bits != 0);
+ read_count++;
+ if (read_count >= kMaxReadCount) {
+ break;
+ }
+ }
+ LOG_INFO("Flushed %d entropy words.", read_count);
+}
+
+/**
+ * Configures the entropy source module in firmware override mode.
+ *
+ * Output is routed to firmware, and the fw_override mode is enabled to get data
+ * post-health tests and before the pre conditioner block.
+ *
+ * @param entropy An entropy source instance.
+ */
+static void entropy_with_fw_override_enable(dif_entropy_src_t *entropy_src) {
+ const dif_entropy_src_config_t config = {
+ .mode = kDifEntropySrcModePtrng,
+ .tests =
+ {
+ [kDifEntropySrcTestRepCount] = false,
+ [kDifEntropySrcTestAdaptiveProportion] = false,
+ [kDifEntropySrcTestBucket] = false,
+ [kDifEntropySrcTestMarkov] = false,
+ [kDifEntropySrcTestMailbox] = false,
+ [kDifEntropySrcTestVendorSpecific] = false,
+ },
+ .reset_health_test_registers = false,
+ .single_bit_mode = kDifEntropySrcSingleBitModeDisabled,
+ .route_to_firmware = true,
+ .fw_override =
+ {
+ .enable = true,
+ .entropy_insert_enable = true,
+ .buffer_threshold = kEntropyFifoBufferSize,
+ },
+ };
+ CHECK_DIF_OK(dif_entropy_src_configure(entropy_src, config));
+}
+
+/**
+ * Test the firmware override path.
+ *
+ * This tests disconnects the observation buffer from the entropy src
+ * pre-conditioner block to read the raw entropy data after the hardware
+ * health tests. It then feeds the data back into the conditioner block until
+ * there is data available in the output FIFO.
+ *
+ * @param entropy An Entropy handle.
+ */
+void test_firmware_override(dif_entropy_src_t *entropy) {
+ CHECK_DIF_OK(dif_entropy_src_disable(entropy));
+ entropy_with_fw_override_enable(entropy);
+ entropy_data_flush(entropy);
+
+ // Read data from the obeservation and write it back into the pre conditioner
+ // until there is data available in the output buffer.
+ uint32_t buf[kEntropyFifoBufferSize] = {0};
+ uint32_t word_count = 0;
+ do {
+ CHECK_DIF_OK(
+ dif_entropy_src_fifo_read(entropy, buf, kEntropyFifoBufferSize));
+ for (size_t i = 0; i < kEntropyFifoBufferSize; ++i) {
+ CHECK(buf[i] != 0);
+ }
+ CHECK_DIF_OK(
+ dif_entropy_src_fifo_write(entropy, buf, kEntropyFifoBufferSize));
+ word_count += kEntropyFifoBufferSize;
+ } while (dif_entropy_src_avail(entropy) == kDifUnavailable);
+ LOG_INFO("Processed %d words via FIFO_OVR buffer.", word_count);
+ entropy_data_flush(entropy);
+}
+
+bool test_main() {
+ dif_entropy_src_t entropy_src;
+ CHECK_DIF_OK(dif_entropy_src_init(
+ mmio_region_from_addr(TOP_EARLGREY_ENTROPY_SRC_BASE_ADDR), &entropy_src));
+ test_firmware_override(&entropy_src);
+ return true;
+}
diff --git a/sw/device/tests/entropy_src_smoketest.c b/sw/device/tests/entropy_src_smoketest.c
index 55e6bdb..c5d3baf 100644
--- a/sw/device/tests/entropy_src_smoketest.c
+++ b/sw/device/tests/entropy_src_smoketest.c
@@ -42,7 +42,12 @@
// this field needs to manually toggled by software. Disable for now
.reset_health_test_registers = false,
.single_bit_mode = kDifEntropySrcSingleBitModeDisabled,
- .route_to_firmware = true};
+ .route_to_firmware = true,
+ .fw_override = {
+ .enable = false,
+ .entropy_insert_enable = false,
+ .buffer_threshold = kDifEntropyFifoIntDefaultThreshold,
+ }};
CHECK_DIF_OK(dif_entropy_src_configure(&entropy_src, config));
uint32_t entropy_data[kEntropyDataNumWords];
diff --git a/sw/device/tests/meson.build b/sw/device/tests/meson.build
index 230bbb6..eb67901 100644
--- a/sw/device/tests/meson.build
+++ b/sw/device/tests/meson.build
@@ -252,6 +252,23 @@
}
}
+entropy_src_fw_ovr_test_lib = declare_dependency(
+ link_with: static_library(
+ 'entropy_src_fw_ovr_test_lib',
+ sources: ['entropy_src_fw_ovr_test.c'],
+ dependencies: [
+ sw_lib_dif_entropy_src,
+ sw_lib_mmio,
+ sw_lib_runtime_log,
+ ],
+ ),
+)
+sw_tests += {
+ 'entropy_src_fw_ovr_test': {
+ 'library': entropy_src_fw_ovr_test_lib,
+ }
+}
+
entropy_src_smoketest_lib = declare_dependency(
link_with: static_library(
'entropy_src_smoketest_lib',