12#include "hw/top/otp_ctrl_regs.h"
23static bool checks_are_locked(
const dif_otp_ctrl_t *otp,
bool check_config) {
24 ptrdiff_t reg_offset = check_config
25 ? OTP_CTRL_CHECK_REGWEN_REG_OFFSET
26 : OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET;
28 check_config ? OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT
29 : OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT;
30 uint32_t locked = mmio_region_read32(otp->
base_addr, reg_offset);
31 return !bitfield_bit32_read(locked, regwen_bit);
35 uint32_t partition_number,
39 .mask = OTP_CTRL_ERR_CODE_0_ERR_CODE_0_MASK,
40 .index = OTP_CTRL_ERR_CODE_0_ERR_CODE_0_OFFSET,
43 ptrdiff_t err_code_address =
44 OTP_CTRL_ERR_CODE_0_REG_OFFSET +
45 (ptrdiff_t)partition_number * (ptrdiff_t)
sizeof(uint32_t);
46 uint32_t error_code = mmio_region_read32(otp->
base_addr, err_code_address);
48 switch (bitfield_field32_read(error_code, field)) {
49 case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_NO_ERROR:
52 case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_ERROR:
55 case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_ECC_CORR_ERROR:
58 case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_ECC_UNCORR_ERROR:
61 case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_WRITE_BLANK_ERROR:
64 case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_ACCESS_ERROR:
67 case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_CHECK_FAIL_ERROR:
70 case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_FSM_STATE_ERROR:
84 if (checks_are_locked(otp,
true)) {
88 mmio_region_write32(otp->
base_addr, OTP_CTRL_CHECK_TIMEOUT_REG_OFFSET,
91 OTP_CTRL_INTEGRITY_CHECK_PERIOD_REG_OFFSET,
94 OTP_CTRL_CONSISTENCY_CHECK_PERIOD_REG_OFFSET,
104 if (checks_are_locked(otp,
false)) {
109 bitfield_bit32_write(0, OTP_CTRL_CHECK_TRIGGER_INTEGRITY_BIT,
true);
110 mmio_region_write32(otp->
base_addr, OTP_CTRL_CHECK_TRIGGER_REG_OFFSET, reg);
119 if (checks_are_locked(otp,
false)) {
124 bitfield_bit32_write(0, OTP_CTRL_CHECK_TRIGGER_CONSISTENCY_BIT,
true);
125 mmio_region_write32(otp->
base_addr, OTP_CTRL_CHECK_TRIGGER_REG_OFFSET, reg);
135 uint32_t reg = bitfield_bit32_write(
136 0, OTP_CTRL_DIRECT_ACCESS_REGWEN_DIRECT_ACCESS_REGWEN_BIT,
false);
137 mmio_region_write32(otp->
base_addr, OTP_CTRL_DIRECT_ACCESS_REGWEN_REG_OFFSET,
145 if (otp == NULL || is_locked == NULL) {
149 uint32_t reg = mmio_region_read32(otp->
base_addr,
150 OTP_CTRL_DIRECT_ACCESS_REGWEN_REG_OFFSET);
151 *is_locked = !bitfield_bit32_read(
152 reg, OTP_CTRL_DIRECT_ACCESS_REGWEN_DIRECT_ACCESS_REGWEN_BIT);
163 bitfield_bit32_write(0, OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT,
false);
164 mmio_region_write32(otp->
base_addr, OTP_CTRL_CHECK_REGWEN_REG_OFFSET, reg);
171 if (otp == NULL || is_locked == NULL) {
175 *is_locked = checks_are_locked(otp,
true);
184 uint32_t reg = bitfield_bit32_write(
185 0, OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT,
false);
186 mmio_region_write32(otp->
base_addr, OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET,
194 if (otp == NULL || is_locked == NULL) {
198 *is_locked = checks_are_locked(otp,
false);
203 ptrdiff_t *reg_offset,
207 *reg_offset = OTP_CTRL_VENDOR_TEST_READ_LOCK_REG_OFFSET;
208 *index = OTP_CTRL_VENDOR_TEST_READ_LOCK_VENDOR_TEST_READ_LOCK_BIT;
211 *reg_offset = OTP_CTRL_CREATOR_SW_CFG_READ_LOCK_REG_OFFSET;
212 *index = OTP_CTRL_CREATOR_SW_CFG_READ_LOCK_CREATOR_SW_CFG_READ_LOCK_BIT;
215 *reg_offset = OTP_CTRL_OWNER_SW_CFG_READ_LOCK_REG_OFFSET;
216 *index = OTP_CTRL_OWNER_SW_CFG_READ_LOCK_OWNER_SW_CFG_READ_LOCK_BIT;
218#if defined(OPENTITAN_IS_EGRET)
219 case kDifOtpCtrlPartitionRotCreatorAuthCodesign:
220 *reg_offset = OTP_CTRL_ROT_CREATOR_AUTH_CODESIGN_READ_LOCK_REG_OFFSET;
222 OTP_CTRL_ROT_CREATOR_AUTH_CODESIGN_READ_LOCK_ROT_CREATOR_AUTH_CODESIGN_READ_LOCK_BIT;
224 case kDifOtpCtrlPartitionRotCreatorAuthState:
225 *reg_offset = OTP_CTRL_ROT_CREATOR_AUTH_STATE_READ_LOCK_REG_OFFSET;
227 OTP_CTRL_ROT_CREATOR_AUTH_STATE_READ_LOCK_ROT_CREATOR_AUTH_STATE_READ_LOCK_BIT;
229#elif defined(OPENTITAN_IS_DRAGONFLY)
230 case kDifOtpCtrlPartitionOwnershipSlotState:
231 *reg_offset = OTP_CTRL_OWNERSHIP_SLOT_STATE_READ_LOCK_REG_OFFSET;
233 OTP_CTRL_OWNERSHIP_SLOT_STATE_READ_LOCK_OWNERSHIP_SLOT_STATE_READ_LOCK_BIT;
235 case kDifOtpCtrlPartitionRotCreatorIdentity:
236 *reg_offset = OTP_CTRL_ROT_CREATOR_IDENTITY_READ_LOCK_REG_OFFSET;
238 OTP_CTRL_ROT_CREATOR_IDENTITY_READ_LOCK_ROT_CREATOR_IDENTITY_READ_LOCK_BIT;
240 case kDifOtpCtrlPartitionRotOwnerAuthSlot0:
241 *reg_offset = OTP_CTRL_ROT_OWNER_AUTH_SLOT0_READ_LOCK_REG_OFFSET;
243 OTP_CTRL_ROT_OWNER_AUTH_SLOT0_READ_LOCK_ROT_OWNER_AUTH_SLOT0_READ_LOCK_BIT;
245 case kDifOtpCtrlPartitionRotOwnerAuthSlot1:
246 *reg_offset = OTP_CTRL_ROT_OWNER_AUTH_SLOT1_READ_LOCK_REG_OFFSET;
248 OTP_CTRL_ROT_OWNER_AUTH_SLOT1_READ_LOCK_ROT_OWNER_AUTH_SLOT1_READ_LOCK_BIT;
250 case kDifOtpCtrlPartitionPlatIntegAuthSlot0:
251 *reg_offset = OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_READ_LOCK_REG_OFFSET;
253 OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_READ_LOCK_PLAT_INTEG_AUTH_SLOT0_READ_LOCK_BIT;
255 case kDifOtpCtrlPartitionPlatIntegAuthSlot1:
256 *reg_offset = OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_READ_LOCK_REG_OFFSET;
258 OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_READ_LOCK_PLAT_INTEG_AUTH_SLOT1_READ_LOCK_BIT;
260 case kDifOtpCtrlPartitionPlatOwnerAuthSlot0:
261 *reg_offset = OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_READ_LOCK_REG_OFFSET;
263 OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_READ_LOCK_PLAT_OWNER_AUTH_SLOT0_READ_LOCK_BIT;
265 case kDifOtpCtrlPartitionPlatOwnerAuthSlot1:
266 *reg_offset = OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_READ_LOCK_REG_OFFSET;
268 OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_READ_LOCK_PLAT_OWNER_AUTH_SLOT1_READ_LOCK_BIT;
270 case kDifOtpCtrlPartitionPlatOwnerAuthSlot2:
271 *reg_offset = OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_READ_LOCK_REG_OFFSET;
273 OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_READ_LOCK_PLAT_OWNER_AUTH_SLOT2_READ_LOCK_BIT;
275 case kDifOtpCtrlPartitionPlatOwnerAuthSlot3:
276 *reg_offset = OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_READ_LOCK_REG_OFFSET;
278 OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_READ_LOCK_PLAT_OWNER_AUTH_SLOT3_READ_LOCK_BIT;
280 case kDifOtpCtrlPartitionExtNvm:
281 *reg_offset = OTP_CTRL_EXT_NVM_READ_LOCK_REG_OFFSET;
282 *index = OTP_CTRL_EXT_NVM_READ_LOCK_EXT_NVM_READ_LOCK_BIT;
284 case kDifOtpCtrlPartitionRomPatch:
285 *reg_offset = OTP_CTRL_ROM_PATCH_READ_LOCK_REG_OFFSET;
286 *index = OTP_CTRL_ROM_PATCH_READ_LOCK_ROM_PATCH_READ_LOCK_BIT;
288 case kDifOtpCtrlPartitionSocFusesCp:
289 *reg_offset = OTP_CTRL_SOC_FUSES_CP_READ_LOCK_REG_OFFSET;
290 *index = OTP_CTRL_SOC_FUSES_CP_READ_LOCK_SOC_FUSES_CP_READ_LOCK_BIT;
292 case kDifOtpCtrlPartitionSocFusesFt:
293 *reg_offset = OTP_CTRL_SOC_FUSES_FT_READ_LOCK_REG_OFFSET;
294 *index = OTP_CTRL_SOC_FUSES_FT_READ_LOCK_SOC_FUSES_FT_READ_LOCK_BIT;
296 case kDifOtpCtrlPartitionScratchFuses:
297 *reg_offset = OTP_CTRL_SCRATCH_FUSES_READ_LOCK_REG_OFFSET;
298 *index = OTP_CTRL_SCRATCH_FUSES_READ_LOCK_SCRATCH_FUSES_READ_LOCK_BIT;
301#error "dif_otp_ctrl does not support this top"
317 if (!sw_read_lock_reg_offset(partition, &offset, &index)) {
321 uint32_t reg = bitfield_bit32_write(0, index,
false);
322 mmio_region_write32(otp->
base_addr, offset, reg);
330 if (otp == NULL || is_locked == NULL) {
336 if (!sw_read_lock_reg_offset(partition, &offset, &index)) {
340 uint32_t reg = mmio_region_read32(otp->
base_addr, offset);
341 *is_locked = !bitfield_bit32_read(reg, index);
347 if (otp == NULL ||
status == NULL) {
354 uint32_t status_code_reg =
355 mmio_region_read32(otp->
base_addr, OTP_CTRL_STATUS_REG_OFFSET);
358 if (bitfield_bit32_read(status_code_reg,
359 OTP_CTRL_STATUS_PARTITION_ERROR_BIT)) {
361 for (
int status_reg_num = 0; status_reg_num < num_part_status_regs;
363 uint32_t partition_status_reg = mmio_region_read32(
365 (ptrdiff_t)(OTP_CTRL_PARTITION_STATUS_0_REG_OFFSET +
366 (ptrdiff_t)
sizeof(uint32_t) * status_reg_num));
368 for (
int status_idx = 0; status_idx < 32; ++status_idx) {
369 uint32_t partition_number =
370 (uint32_t)status_reg_num * 32 + (uint32_t)status_idx;
376 if (!bitfield_bit32_read(partition_status_reg,
383 status->codes = bitfield_bit32_write(
389 if (get_error_code(otp, partition_number, &err) !=
kDifOk) {
392 status->causes[partition_number] = err;
414 if (get_error_code(otp, err_code_index, &err) !=
kDifOk) {
419 status->causes[err_code_index] = err;
476static const partition_info_t kPartitions[] = {
478 .start_addr = OTP_CTRL_PARAM_VENDOR_TEST_OFFSET,
479 .len = OTP_CTRL_PARAM_VENDOR_TEST_SIZE,
483 .is_lifecycle =
false},
485 .start_addr = OTP_CTRL_PARAM_CREATOR_SW_CFG_OFFSET,
486 .len = OTP_CTRL_PARAM_CREATOR_SW_CFG_SIZE,
490 .is_lifecycle =
false},
492 .start_addr = OTP_CTRL_PARAM_OWNER_SW_CFG_OFFSET,
493 .len = OTP_CTRL_PARAM_OWNER_SW_CFG_SIZE,
497 .is_lifecycle =
false},
498#if defined(OPENTITAN_IS_EGRET)
499 [kDifOtpCtrlPartitionRotCreatorAuthCodesign] = {
500 .start_addr = OTP_CTRL_PARAM_ROT_CREATOR_AUTH_CODESIGN_OFFSET,
501 .len = OTP_CTRL_PARAM_ROT_CREATOR_AUTH_CODESIGN_SIZE,
505 .is_lifecycle =
false},
506 [kDifOtpCtrlPartitionRotCreatorAuthState] = {
507 .start_addr = OTP_CTRL_PARAM_ROT_CREATOR_AUTH_STATE_OFFSET,
508 .len = OTP_CTRL_PARAM_ROT_CREATOR_AUTH_STATE_SIZE,
512 .is_lifecycle =
false},
513#elif defined(OPENTITAN_IS_DRAGONFLY)
514 [kDifOtpCtrlPartitionOwnershipSlotState] = {
515 .start_addr = OTP_CTRL_PARAM_OWNERSHIP_SLOT_STATE_OFFSET,
516 .len = OTP_CTRL_PARAM_OWNERSHIP_SLOT_STATE_SIZE,
520 .is_lifecycle =
false},
521 [kDifOtpCtrlPartitionRotCreatorIdentity] = {
522 .start_addr = OTP_CTRL_PARAM_ROT_CREATOR_IDENTITY_OFFSET,
523 .len = OTP_CTRL_PARAM_ROT_CREATOR_IDENTITY_SIZE,
527 .is_lifecycle =
false},
528 [kDifOtpCtrlPartitionRotOwnerAuthSlot0] = {
529 .start_addr = OTP_CTRL_PARAM_ROT_OWNER_AUTH_SLOT0_OFFSET,
530 .len = OTP_CTRL_PARAM_ROT_OWNER_AUTH_SLOT0_SIZE,
534 .is_lifecycle =
false},
535 [kDifOtpCtrlPartitionRotOwnerAuthSlot1] = {
536 .start_addr = OTP_CTRL_PARAM_ROT_OWNER_AUTH_SLOT1_OFFSET,
537 .len = OTP_CTRL_PARAM_ROT_OWNER_AUTH_SLOT1_SIZE,
541 .is_lifecycle =
false},
542 [kDifOtpCtrlPartitionPlatIntegAuthSlot0] = {
543 .start_addr = OTP_CTRL_PARAM_PLAT_INTEG_AUTH_SLOT0_OFFSET,
544 .len = OTP_CTRL_PARAM_PLAT_INTEG_AUTH_SLOT0_SIZE,
548 .is_lifecycle =
false},
549 [kDifOtpCtrlPartitionPlatIntegAuthSlot1] = {
550 .start_addr = OTP_CTRL_PARAM_PLAT_INTEG_AUTH_SLOT1_OFFSET,
551 .len = OTP_CTRL_PARAM_PLAT_INTEG_AUTH_SLOT1_SIZE,
555 .is_lifecycle =
false},
556 [kDifOtpCtrlPartitionPlatOwnerAuthSlot0] = {
557 .start_addr = OTP_CTRL_PARAM_PLAT_OWNER_AUTH_SLOT0_OFFSET,
558 .len = OTP_CTRL_PARAM_PLAT_OWNER_AUTH_SLOT0_SIZE,
562 .is_lifecycle =
false},
563 [kDifOtpCtrlPartitionPlatOwnerAuthSlot1] = {
564 .start_addr = OTP_CTRL_PARAM_PLAT_OWNER_AUTH_SLOT1_OFFSET,
565 .len = OTP_CTRL_PARAM_PLAT_OWNER_AUTH_SLOT1_SIZE,
569 .is_lifecycle =
false},
570 [kDifOtpCtrlPartitionPlatOwnerAuthSlot2] = {
571 .start_addr = OTP_CTRL_PARAM_PLAT_OWNER_AUTH_SLOT2_OFFSET,
572 .len = OTP_CTRL_PARAM_PLAT_OWNER_AUTH_SLOT2_SIZE,
576 .is_lifecycle =
false},
577 [kDifOtpCtrlPartitionPlatOwnerAuthSlot3] = {
578 .start_addr = OTP_CTRL_PARAM_PLAT_OWNER_AUTH_SLOT3_OFFSET,
579 .len = OTP_CTRL_PARAM_PLAT_OWNER_AUTH_SLOT3_SIZE,
583 .is_lifecycle =
false},
584 [kDifOtpCtrlPartitionExtNvm] = {
585 .start_addr = OTP_CTRL_PARAM_EXT_NVM_OFFSET,
586 .len = OTP_CTRL_PARAM_EXT_NVM_SIZE,
590 .is_lifecycle =
false},
591 [kDifOtpCtrlPartitionRomPatch] = {
592 .start_addr = OTP_CTRL_PARAM_ROM_PATCH_OFFSET,
593 .len = OTP_CTRL_PARAM_ROM_PATCH_SIZE,
597 .is_lifecycle =
false},
599#error "dif_otp_ctrl does not support this top"
602 .start_addr = OTP_CTRL_PARAM_HW_CFG0_OFFSET,
603 .len = OTP_CTRL_PARAM_HW_CFG0_SIZE,
605 .is_software =
false,
607 .is_lifecycle =
false},
609 .start_addr = OTP_CTRL_PARAM_HW_CFG1_OFFSET,
610 .len = OTP_CTRL_PARAM_HW_CFG1_SIZE,
612 .is_software =
false,
614 .is_lifecycle =
false},
616 .start_addr = OTP_CTRL_PARAM_SECRET0_OFFSET,
617 .len = OTP_CTRL_PARAM_SECRET0_SIZE,
619 .is_software =
false,
621 .is_lifecycle =
false},
623 .start_addr = OTP_CTRL_PARAM_SECRET1_OFFSET,
624 .len = OTP_CTRL_PARAM_SECRET1_SIZE,
626 .is_software =
false,
628 .is_lifecycle =
false},
630 .start_addr = OTP_CTRL_PARAM_SECRET2_OFFSET,
631 .len = OTP_CTRL_PARAM_SECRET2_SIZE,
633 .is_software =
false,
635 .is_lifecycle =
false},
636#if defined(OPENTITAN_IS_DRAGONFLY)
637 [kDifOtpCtrlPartitionSecret3] = {
638 .start_addr = OTP_CTRL_PARAM_SECRET3_OFFSET,
639 .len = OTP_CTRL_PARAM_SECRET3_SIZE,
641 .is_software =
false,
643 .is_lifecycle =
false},
644#elif defined(OPENTITAN_IS_EGRET)
647#error "dif_otp_ctrl does not support this top"
650 .start_addr = OTP_CTRL_PARAM_LIFE_CYCLE_OFFSET,
651 .len = OTP_CTRL_PARAM_LIFE_CYCLE_SIZE,
653 .is_software =
false,
655 .is_lifecycle =
true},
660 uint32_t abs_address,
661 uint32_t *relative_address) {
662 *relative_address = 0;
664 if (partition >=
ARRAYSIZE(kPartitions)) {
668 if ((abs_address & kPartitions[partition].align_mask) != 0) {
672 if (abs_address < kPartitions[partition].start_addr) {
676 *relative_address = abs_address - kPartitions[partition].start_addr;
677 if (*relative_address >= kPartitions[partition].len) {
678 *relative_address = 0;
688 if (otp == NULL || partition >=
ARRAYSIZE(kPartitions)) {
692 if ((address & kPartitions[partition].align_mask) != 0) {
696 if (address >= kPartitions[partition].len) {
700 address += kPartitions[partition].start_addr;
701 mmio_region_write32(otp->
base_addr, OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET,
705 bitfield_bit32_write(0, OTP_CTRL_DIRECT_ACCESS_CMD_RD_BIT,
true);
706 mmio_region_write32(otp->
base_addr, OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET,
714 if (otp == NULL || value == NULL) {
718 *value = mmio_region_read32(otp->
base_addr,
719 OTP_CTRL_DIRECT_ACCESS_RDATA_0_REG_OFFSET);
725 if (otp == NULL || value == NULL) {
729 *value = mmio_region_read32(otp->
base_addr,
730 OTP_CTRL_DIRECT_ACCESS_RDATA_1_REG_OFFSET);
732 *value |= mmio_region_read32(otp->
base_addr,
733 OTP_CTRL_DIRECT_ACCESS_RDATA_0_REG_OFFSET);
739 uint32_t address, uint32_t value) {
740 if (otp == NULL || partition >=
ARRAYSIZE(kPartitions)) {
749 if (kPartitions[partition].align_mask != 0x3 ||
750 kPartitions[partition].is_lifecycle) {
754 if ((address & kPartitions[partition].align_mask) != 0) {
761 size_t digest_size = kPartitions[partition].has_digest *
sizeof(uint64_t);
762 if (address >= kPartitions[partition].len - digest_size) {
766 address += kPartitions[partition].start_addr;
767 mmio_region_write32(otp->
base_addr, OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET,
770 mmio_region_write32(otp->
base_addr, OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET,
774 bitfield_bit32_write(0, OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT,
true);
775 mmio_region_write32(otp->
base_addr, OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET,
783 uint32_t address, uint64_t value) {
784 if (otp == NULL || partition >=
ARRAYSIZE(kPartitions)) {
790 if (kPartitions[partition].align_mask != 0x7) {
794 if ((address & kPartitions[partition].align_mask) != 0) {
800 size_t digest_size =
sizeof(uint64_t);
801 if (address >= kPartitions[partition].len - digest_size) {
805 address += kPartitions[partition].start_addr;
806 mmio_region_write32(otp->
base_addr, OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET,
809 mmio_region_write32(otp->
base_addr, OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET,
811 mmio_region_write32(otp->
base_addr, OTP_CTRL_DIRECT_ACCESS_WDATA_1_REG_OFFSET,
815 bitfield_bit32_write(0, OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT,
true);
816 mmio_region_write32(otp->
base_addr, OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET,
825 if (otp == NULL || partition >=
ARRAYSIZE(kPartitions)) {
830 if (!kPartitions[partition].has_digest) {
836 bool is_sw = kPartitions[partition].is_software;
837 if (is_sw == (digest == 0)) {
841 uint32_t address = kPartitions[partition].start_addr;
843 address += kPartitions[partition].len -
sizeof(digest);
845 mmio_region_write32(otp->
base_addr, OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET,
850 OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET,
851 digest & 0xffffffff);
853 OTP_CTRL_DIRECT_ACCESS_WDATA_1_REG_OFFSET,
858 ? OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT
859 : OTP_CTRL_DIRECT_ACCESS_CMD_DIGEST_BIT;
860 uint32_t cmd = bitfield_bit32_write(0, cmd_bit,
true);
861 mmio_region_write32(otp->
base_addr, OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET,
871 *reg0 = OTP_CTRL_VENDOR_TEST_DIGEST_0_REG_OFFSET;
872 *reg1 = OTP_CTRL_VENDOR_TEST_DIGEST_1_REG_OFFSET;
875 *reg0 = OTP_CTRL_CREATOR_SW_CFG_DIGEST_0_REG_OFFSET;
876 *reg1 = OTP_CTRL_CREATOR_SW_CFG_DIGEST_1_REG_OFFSET;
879 *reg0 = OTP_CTRL_OWNER_SW_CFG_DIGEST_0_REG_OFFSET;
880 *reg1 = OTP_CTRL_OWNER_SW_CFG_DIGEST_1_REG_OFFSET;
882#if defined(OPENTITAN_IS_EGRET)
883 case kDifOtpCtrlPartitionRotCreatorAuthCodesign:
884 *reg0 = OTP_CTRL_ROT_CREATOR_AUTH_CODESIGN_DIGEST_0_REG_OFFSET;
885 *reg1 = OTP_CTRL_ROT_CREATOR_AUTH_CODESIGN_DIGEST_1_REG_OFFSET;
887 case kDifOtpCtrlPartitionRotCreatorAuthState:
888 *reg0 = OTP_CTRL_ROT_CREATOR_AUTH_STATE_DIGEST_0_REG_OFFSET;
889 *reg1 = OTP_CTRL_ROT_CREATOR_AUTH_STATE_DIGEST_1_REG_OFFSET;
891#elif defined(OPENTITAN_IS_DRAGONFLY)
892 case kDifOtpCtrlPartitionRotCreatorIdentity:
893 *reg0 = OTP_CTRL_ROT_CREATOR_IDENTITY_DIGEST_0_REG_OFFSET;
894 *reg1 = OTP_CTRL_ROT_CREATOR_IDENTITY_DIGEST_1_REG_OFFSET;
896 case kDifOtpCtrlPartitionRotOwnerAuthSlot0:
897 *reg0 = OTP_CTRL_ROT_OWNER_AUTH_SLOT0_DIGEST_0_REG_OFFSET;
898 *reg1 = OTP_CTRL_ROT_OWNER_AUTH_SLOT0_DIGEST_1_REG_OFFSET;
900 case kDifOtpCtrlPartitionRotOwnerAuthSlot1:
901 *reg0 = OTP_CTRL_ROT_OWNER_AUTH_SLOT1_DIGEST_0_REG_OFFSET;
902 *reg1 = OTP_CTRL_ROT_OWNER_AUTH_SLOT1_DIGEST_1_REG_OFFSET;
904 case kDifOtpCtrlPartitionPlatIntegAuthSlot0:
905 *reg0 = OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_DIGEST_0_REG_OFFSET;
906 *reg1 = OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_DIGEST_1_REG_OFFSET;
908 case kDifOtpCtrlPartitionPlatIntegAuthSlot1:
909 *reg0 = OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_DIGEST_0_REG_OFFSET;
910 *reg1 = OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_DIGEST_1_REG_OFFSET;
912 case kDifOtpCtrlPartitionPlatOwnerAuthSlot0:
913 *reg0 = OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_DIGEST_0_REG_OFFSET;
914 *reg1 = OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_DIGEST_1_REG_OFFSET;
916 case kDifOtpCtrlPartitionPlatOwnerAuthSlot1:
917 *reg0 = OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_DIGEST_0_REG_OFFSET;
918 *reg1 = OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_DIGEST_1_REG_OFFSET;
920 case kDifOtpCtrlPartitionPlatOwnerAuthSlot2:
921 *reg0 = OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_DIGEST_0_REG_OFFSET;
922 *reg1 = OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_DIGEST_1_REG_OFFSET;
924 case kDifOtpCtrlPartitionPlatOwnerAuthSlot3:
925 *reg0 = OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_DIGEST_0_REG_OFFSET;
926 *reg1 = OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_DIGEST_1_REG_OFFSET;
928 case kDifOtpCtrlPartitionRomPatch:
929 *reg0 = OTP_CTRL_ROM_PATCH_DIGEST_0_REG_OFFSET;
930 *reg1 = OTP_CTRL_ROM_PATCH_DIGEST_1_REG_OFFSET;
933#error "dif_otp_ctrl does not support this top"
936 *reg0 = OTP_CTRL_HW_CFG0_DIGEST_0_REG_OFFSET;
937 *reg1 = OTP_CTRL_HW_CFG0_DIGEST_1_REG_OFFSET;
940 *reg0 = OTP_CTRL_HW_CFG1_DIGEST_0_REG_OFFSET;
941 *reg1 = OTP_CTRL_HW_CFG1_DIGEST_1_REG_OFFSET;
944 *reg0 = OTP_CTRL_SECRET0_DIGEST_0_REG_OFFSET;
945 *reg1 = OTP_CTRL_SECRET0_DIGEST_1_REG_OFFSET;
948 *reg0 = OTP_CTRL_SECRET1_DIGEST_0_REG_OFFSET;
949 *reg1 = OTP_CTRL_SECRET1_DIGEST_1_REG_OFFSET;
952 *reg0 = OTP_CTRL_SECRET2_DIGEST_0_REG_OFFSET;
953 *reg1 = OTP_CTRL_SECRET2_DIGEST_1_REG_OFFSET;
955#if defined(OPENTITAN_IS_DRAGONFLY)
956 case kDifOtpCtrlPartitionSecret3:
957 *reg0 = OTP_CTRL_SECRET3_DIGEST_0_REG_OFFSET;
958 *reg1 = OTP_CTRL_SECRET3_DIGEST_1_REG_OFFSET;
960#elif defined(OPENTITAN_IS_EGRET)
963#error "dif_otp_ctrl does not support this top"
975 if (otp == NULL || is_computed == NULL) {
979 ptrdiff_t reg0, reg1;
980 if (!get_digest_regs(partition, ®0, ®1)) {
984 uint64_t value = mmio_region_read32(otp->
base_addr, reg1);
986 value |= mmio_region_read32(otp->
base_addr, reg0);
988 *is_computed = value != 0;
996 if (otp == NULL || digest == NULL) {
1000 ptrdiff_t reg0, reg1;
1001 if (!get_digest_regs(partition, ®0, ®1)) {
1005 uint64_t value = mmio_region_read32(otp->
base_addr, reg1);
1007 value |= mmio_region_read32(otp->
base_addr, reg0);
1019 uint32_t address, uint32_t *buf,
1021 if (otp == NULL || partition >=
ARRAYSIZE(kPartitions) || buf == NULL) {
1025 if (!kPartitions[partition].is_software) {
1029 if ((address & kPartitions[partition].align_mask) != 0) {
1033 if (address + len >= kPartitions[partition].len) {
1037 uint32_t reg_offset = OTP_CTRL_SW_CFG_WINDOW_REG_OFFSET +
1038 kPartitions[partition].start_addr + address;
1039 mmio_region_memcpy_from_mmio32(otp->
base_addr, reg_offset, buf,
1040 len *
sizeof(uint32_t));