From 80495aa658e723bb0688f61ba5fcb462d00f674c Mon Sep 17 00:00:00 2001 From: Guillermo Maturana Date: Fri, 10 Jan 2025 09:48:53 +0000 Subject: [PATCH] [ipgen,otp_ctrl] Change otp_ctrl to ipgen This commit is similar to other ipgen conversions, but changes a few extra things because it needs to change the generation of otp memory images. It merges the following 32 individual commits: - Fix OtpMemMap exception message - Improve handling template parameters by ipgen - Add doc and avoid redundant declaration. - Move otp_ctrl images and mmap hjson out of hw/ip - The files that are top-specific moved to hw/top_earlgrey/data/otp - Adjust lots of paths to match this change - Move otp_ctrl_img.c.tpl to util/design/data since it is generic - Move the documentation of otp image generation to util/design/README.md - Add otp_ctrl ipgen tpldesc hjson file - Copy all hw/ip/otp_ctrl files to ip_templates - This is a straight copy with no changes to the files. - Remove generated files - There is no use for files in ip_templates that will end up being generated. - Move template files to their specific directories - Turn a few more files into templates and fix paths - The template files are almost identical to the previous ones, except - they use ${topname} instead of a specific name - consecutive '#' are escaped so they are not considered hugo comments; for example, ### is replaced by ${"###"} - text bracketed by CMDGEN is removed - Adjust some relative paths to account for the change in the file locations. - Use {self_dir} to simplify some paths: it points to the location of the hjson file used to process the contents of a given file. - Fix some incorrect links to docs. - Move top-specific items out of otp_ctrl_pkg - Move top-specific and implementation items to otp_ctrl_top_specific_pkg. - Create separate core files for both packages. - Adjust multiple sv files to use the right package. - Remove unused lc_otp_token_{req,rsp}_t. - Fix core files - Change most core file into templates, use instance_vlnv, and declare a virtual target. - Change otp_ctrl_sim_cfg.hjson into a template since it uses a template fusesoc core. - Change templates to use plain dictionaries - In practice this just removes redundant config token. - Block building template targets - Add otp_ctrl in hw/ip_templates/BUILD all_files. - Template files are only intended for code generation, so it makes no sense to use them to build targets. - Remove gen_comment from templates - The comment regarding generation for ipgen artifacts will be dealt with uniformly in the PR that fixes #25444. - Remove gen_comment from the tpldesc file. - Add code in OtpMemMap to support ipgen - Create packages for design.lib and design.mubi. - gen-flash-img depends on modules that import topgen modules so it needs to have topgen in sys.path. This was not needed for some reason, perhaps the introduction of packages ends up requiring it. - Place darjeeling mmap and img files under hw/top_darjeeling - This is similar to the new placement of the corresponding earlgrey files. - Fix the copyright banner. - Block generation of dif artifacts - These artifacts need to be placed under top-specific directories. - When the sw side is ready for that, change the flow to create the per top generated files in the right place. - Create otp_ctrl with ipgen - Create both earlgrey and darjeeling otp_ctrl with topgen.py via ipgen. - As a side effect some random constants are changed, since there are more requests for random numbers in topgen. - Make util/design/gen-otp-mmap.py compatible with otp_ctrl ipgen - This tool may become obsolete, but it is best to upgrade it per the ipgen flow. - Comment out dif generation, and add a TODO with an explicit issue. - Add mmap_def argument in bazel's otp_image_consts - Remove unnecessary otp link in md file - This avoids the confusion about which top to refer to. - Generate otp_ctrl difs via ipgen - Fix top level core files - Fix BUILD files - Use ip_autogen path for otp_ctrl_c_regs. - Use ip_autogen path for rust artifacts. - Change a few hw BUILD files for ipgen changes. - Fix reference to data/otp_ctrl.hjson in BUILD.tpl. - Fix image handling in dvsim sim.mk - Fix hjson files - Fix md files - Describe references to top-specific otp_ctrl as examples. - Fix BLOCKFILE - Fix documentation collateral - Fix englishbreakfast build - Remove otp_ctrl from top_englishbreakfast because it is not instantiated, but keep otp_ctrl_pkg since it shows in types of unconnected ports. - Remove obsolete otp-mmap check from ci - Remove legacy files - Update hw/ip/README.md file Fixes #25019 Signed-off-by: Guillermo Maturana --- BLOCKFILE | 29 +- SUMMARY.md | 29 +- ci/scripts/check-generated.sh | 1 - .../run-english-breakfast-verilator-tests.sh | 2 +- doc/getting_started/setup_fpga.md | 2 +- doc/getting_started/setup_verilator.md | 4 +- doc/security/specs/secure_boot/README.md | 1 - hw/bitstream/README.md | 4 +- hw/bitstream/vivado/BUILD | 2 +- hw/dv/sv/mem_bkdr_util/mem_bkdr_util.core | 1 + hw/dv/sv/mem_bkdr_util/otp_scrambler_pkg.sv | 1 - hw/dv/tools/dvsim/sim.mk | 10 +- hw/ip/BUILD | 1 - hw/ip/README.md | 2 - hw/ip/lc_ctrl/doc/interfaces.md | 2 +- hw/ip/lc_ctrl/doc/theory_of_operation.md | 18 +- hw/ip/lc_ctrl/dv/env/lc_ctrl_if.sv | 1 - hw/ip/otbn/doc/theory_of_operation.md | 6 +- hw/ip/otbn/dv/README.md | 2 +- hw/ip/otp_ctrl/data/README.md | 72 - hw/ip/otp_ctrl/doc/checklist.md | 271 - hw/ip/prim/dv/prim_secded/secded_enc.c | 135 +- hw/ip/rom_ctrl/util/gen_vivado_mem_image.py | 5 +- hw/ip/sram_ctrl/doc/interfaces.md | 6 +- hw/ip/sram_ctrl/doc/theory_of_operation.md | 2 +- hw/ip_templates/BUILD | 1 + .../flash_ctrl/doc/programmers_guide.md | 2 +- .../flash_ctrl/doc/theory_of_operation.md | 2 +- hw/{ip => ip_templates}/otp_ctrl/BUILD | 4 +- hw/ip_templates/otp_ctrl/BUILD.tpl | 10 + hw/ip_templates/otp_ctrl/README.md.tpl | 63 + .../otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet} | 3 +- .../otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet} | 7 +- .../dif_otp_ctrl_unittest.cc.tpl.not_yet} | 3 +- .../otp_ctrl/data/otp_ctrl.hjson.tpl | 23 +- .../otp_ctrl/data/otp_ctrl.tpldesc.hjson | 19 + .../data/otp_ctrl_sec_cm_testplan.hjson | 0 .../otp_ctrl/data/otp_ctrl_testplan.hjson | 0 hw/ip_templates/otp_ctrl/defs.bzl.tpl | 9 + hw/ip_templates/otp_ctrl/doc/checklist.md | 271 + .../otp_ctrl/doc/interfaces.md.tpl | 192 + .../doc/otp_ctrl_behavioral_model.svg | 0 .../otp_ctrl/doc/otp_ctrl_blockdiag.svg | 0 .../otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg | 0 .../otp_ctrl/doc/otp_ctrl_dai_fsm.svg | 0 .../doc/otp_ctrl_digest_mechanism.svg | 0 .../otp_ctrl/doc/otp_ctrl_digests.md | 0 .../doc/otp_ctrl_field_descriptions.md | 0 .../otp_ctrl/doc/otp_ctrl_kdi_fsm.svg | 0 .../otp_ctrl/doc/otp_ctrl_key_req_ack.svg | 0 .../otp_ctrl/doc/otp_ctrl_lci_fsm.svg | 0 .../otp_ctrl/doc/otp_ctrl_mmap.md | 0 .../otp_ctrl/doc/otp_ctrl_overview.svg | 0 .../otp_ctrl/doc/otp_ctrl_partitions.md | 0 .../otp_ctrl/doc/otp_ctrl_prim_otp.svg | 0 .../otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg | 0 .../otp_ctrl/doc/programmers_guide.md | 6 +- hw/ip_templates/otp_ctrl/doc/registers.md.tpl | 4 + .../otp_ctrl/doc/theory_of_operation.md | 30 +- hw/ip_templates/otp_ctrl/dv/README.md.tpl | 146 + .../otp_ctrl/dv/cov/otp_ctrl_cov.core.tpl | 28 + .../otp_ctrl/dv/cov}/otp_ctrl_cov_bind.sv.tpl | 5 +- .../dv/cov/otp_ctrl_cov_fsm_unr_excl.el | 0 .../otp_ctrl/dv/cov/otp_ctrl_cov_if.sv | 0 .../otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el | 0 .../otp_ctrl/dv/cov/otp_ctrl_cover.cfg | 0 .../otp_ctrl/dv/doc/tb.svg | 0 .../dv/env/otp_ctrl_ast_inputs_cfg.sv | 0 .../otp_ctrl/dv/env/otp_ctrl_env.core.tpl | 62 + .../otp_ctrl/dv/env/otp_ctrl_env.sv | 0 .../otp_ctrl/dv/env/otp_ctrl_env_cfg.sv | 0 .../otp_ctrl/dv/env}/otp_ctrl_env_cov.sv.tpl | 13 +- .../otp_ctrl/dv/env}/otp_ctrl_env_pkg.sv.tpl | 8 +- .../otp_ctrl/dv/env}/otp_ctrl_if.sv.tpl | 7 +- .../dv/env}/otp_ctrl_scoreboard.sv.tpl | 13 +- .../dv/env/otp_ctrl_virtual_sequencer.sv | 0 .../seq_lib/otp_ctrl_background_chks_vseq.sv | 0 .../dv/env/seq_lib}/otp_ctrl_base_vseq.sv.tpl | 18 +- .../dv/env/seq_lib/otp_ctrl_callback_vseq.sv | 0 .../env/seq_lib/otp_ctrl_check_fail_vseq.sv | 0 .../dv/env/seq_lib/otp_ctrl_common_vseq.sv | 0 .../dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv | 0 .../seq_lib}/otp_ctrl_dai_lock_vseq.sv.tpl | 5 +- .../dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv | 0 .../seq_lib/otp_ctrl_low_freq_read_vseq.sv | 0 .../env/seq_lib/otp_ctrl_macro_errs_vseq.sv | 0 .../seq_lib/otp_ctrl_parallel_base_vseq.sv | 0 .../seq_lib/otp_ctrl_parallel_key_req_vseq.sv | 0 .../seq_lib/otp_ctrl_parallel_lc_esc_vseq.sv | 0 .../seq_lib/otp_ctrl_parallel_lc_req_vseq.sv | 0 .../seq_lib/otp_ctrl_partition_walk_vseq.sv | 0 .../dv/env/seq_lib/otp_ctrl_regwen_vseq.sv | 0 .../env/seq_lib}/otp_ctrl_smoke_vseq.sv.tpl | 3 +- .../env/seq_lib/otp_ctrl_stress_all_vseq.sv | 0 .../env/seq_lib/otp_ctrl_test_access_vseq.sv | 0 .../dv/env/seq_lib/otp_ctrl_vseq_list.sv | 0 .../dv/env/seq_lib/otp_ctrl_wake_up_vseq.sv | 0 .../otp_ctrl/dv/otp_ctrl_sim.core.tpl | 37 + .../otp_ctrl/dv/otp_ctrl_sim_cfg.hjson.tpl | 179 + .../otp_ctrl/dv/sva/otp_ctrl_bind.sv | 0 .../otp_ctrl/dv/sva/otp_ctrl_sva.core.tpl | 41 + hw/{ip => ip_templates}/otp_ctrl/dv/tb.sv | 0 .../otp_ctrl/dv/tests/otp_ctrl_base_test.sv | 0 .../otp_ctrl/dv/tests/otp_ctrl_test.core.tpl | 22 + .../otp_ctrl/dv/tests/otp_ctrl_test_pkg.sv | 0 .../otp_ctrl/lint/otp_ctrl.vbl | 0 .../otp_ctrl/lint/otp_ctrl.vlt | 0 .../otp_ctrl/lint/otp_ctrl.waiver | 0 .../otp_ctrl/lint/otp_ctrl_pkg.vbl | 5 + .../otp_ctrl/lint/otp_ctrl_pkg.vlt | 2 +- .../otp_ctrl/lint/otp_ctrl_pkg.waiver | 5 + .../lint/otp_ctrl_top_specific_pkg.vbl | 8 + .../lint/otp_ctrl_top_specific_pkg.vlt | 5 + .../lint/otp_ctrl_top_specific_pkg.waiver} | 2 +- hw/ip_templates/otp_ctrl/otp_ctrl.core.tpl | 106 + .../otp_ctrl/otp_ctrl_pkg.core | 6 +- .../otp_ctrl/otp_ctrl_prim_reg_top.core | 5 +- .../otp_ctrl_top_specific_pkg.core.tpl | 75 + .../otp_ctrl/rtl/otp_ctrl.sv | 4 +- .../otp_ctrl/rtl/otp_ctrl_dai.sv | 1 + .../otp_ctrl/rtl/otp_ctrl_ecc_reg.sv | 0 .../otp_ctrl/rtl/otp_ctrl_kdi.sv | 1 + .../otp_ctrl/rtl/otp_ctrl_lci.sv | 1 + .../otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv | 1 + .../otp_ctrl/rtl/otp_ctrl_part_buf.sv | 1 + .../otp_ctrl/rtl}/otp_ctrl_part_pkg.sv.tpl | 50 +- .../otp_ctrl/rtl/otp_ctrl_part_unbuf.sv | 1 + .../otp_ctrl/rtl/otp_ctrl_pkg.sv | 99 +- .../otp_ctrl/rtl/otp_ctrl_scrmbl.sv | 14 +- .../otp_ctrl/rtl/otp_ctrl_token_const.sv | 2 +- .../otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv | 108 + .../otp_ctrl/syn/constraints.sdc | 0 .../syn/otp_ctrl_gtech_syn_cfg.hjson.tpl | 24 + .../otp_ctrl/syn/otp_ctrl_syn_cfg.hjson.tpl | 26 + .../otp_ctrl/syn/post_elab_gtech.tcl | 0 .../pwrmgr/doc/theory_of_operation.md | 2 +- hw/top_darjeeling/data/autogen/defs.bzl | 2 +- .../data/autogen/top_darjeeling.gen.hjson | 119 +- .../otp/otp_ctrl_img_creator_sw_cfg.hjson | 154 + .../data/otp}/otp_ctrl_img_dev.hjson | 0 .../data/otp/otp_ctrl_img_hw_cfg.hjson | 25 + .../data/otp/otp_ctrl_img_owner_sw_cfg.hjson | 245 + .../data/otp}/otp_ctrl_img_prod.hjson | 0 .../data/otp}/otp_ctrl_img_raw.hjson | 0 .../data/otp}/otp_ctrl_img_rma.hjson | 0 .../data/otp}/otp_ctrl_img_test_locked0.hjson | 0 .../data/otp}/otp_ctrl_img_test_locked1.hjson | 0 .../otp}/otp_ctrl_img_test_unlocked0.hjson | 0 .../otp}/otp_ctrl_img_test_unlocked1.hjson | 0 .../otp}/otp_ctrl_img_test_unlocked2.hjson | 0 .../data/otp/otp_ctrl_mmap.hjson | 1102 +++ hw/top_darjeeling/data/top_darjeeling.hjson | 1 + .../dv/autogen/xbar_env_pkg__params.sv | 2 +- .../dv/autogen/xbar_tgl_excl.cfg | 2 +- .../dv/top_darjeeling_sim_cfgs.hjson | 2 +- .../data/autogen/xbar_peri.gen.hjson | 2 +- .../ip/xbar_peri/dv/autogen/xbar_cover.cfg | 2 +- .../dv/autogen/xbar_env_pkg__params.sv | 2 +- .../ip/xbar_peri/rtl/autogen/tl_peri_pkg.sv | 2 +- hw/top_darjeeling/ip_autogen/otp_ctrl/BUILD | 10 + .../ip_autogen/otp_ctrl/README.md | 63 + .../otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet | 684 ++ .../otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet | 614 ++ .../data/dif_otp_ctrl_unittest.cc.tpl.not_yet | 789 +++ .../ip_autogen/otp_ctrl/data/otp_ctrl.hjson | 3927 +++++++++++ .../data/otp_ctrl_sec_cm_testplan.hjson | 303 + .../otp_ctrl/data/otp_ctrl_testplan.hjson | 404 ++ .../top_darjeeling_otp_ctrl.ipconfig.hjson | 1992 ++++++ .../ip_autogen/otp_ctrl/defs.bzl | 9 + .../ip_autogen/otp_ctrl/doc/checklist.md | 271 + .../ip_autogen/otp_ctrl/doc/interfaces.md | 301 + .../doc/otp_ctrl_behavioral_model.svg | 1 + .../otp_ctrl/doc/otp_ctrl_blockdiag.svg | 1 + .../otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg | 1 + .../otp_ctrl/doc/otp_ctrl_dai_fsm.svg | 1 + .../doc/otp_ctrl_digest_mechanism.svg | 1 + .../otp_ctrl/doc/otp_ctrl_digests.md | 17 + .../doc/otp_ctrl_field_descriptions.md | 101 + .../otp_ctrl/doc/otp_ctrl_kdi_fsm.svg | 1 + .../otp_ctrl/doc/otp_ctrl_key_req_ack.svg | 1 + .../otp_ctrl/doc/otp_ctrl_lci_fsm.svg | 1 + .../ip_autogen/otp_ctrl/doc/otp_ctrl_mmap.md | 122 + .../otp_ctrl/doc/otp_ctrl_overview.svg | 1 + .../otp_ctrl/doc/otp_ctrl_partitions.md | 57 + .../otp_ctrl/doc/otp_ctrl_prim_otp.svg | 1 + .../otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg | 1 + .../otp_ctrl/doc/programmers_guide.md | 261 + .../ip_autogen/otp_ctrl/doc/registers.md | 1525 ++++ .../otp_ctrl/doc/theory_of_operation.md | 514 ++ .../ip_autogen/otp_ctrl/dv/README.md | 146 + .../otp_ctrl/dv/cov/otp_ctrl_cov.core | 28 + .../otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv | 112 + .../dv/cov/otp_ctrl_cov_fsm_unr_excl.el | 199 + .../otp_ctrl/dv/cov/otp_ctrl_cov_if.sv | 113 + .../otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el | 6133 +++++++++++++++++ .../otp_ctrl/dv/cov/otp_ctrl_cover.cfg | 14 + .../ip_autogen/otp_ctrl/dv/doc/tb.svg | 1 + .../dv/env/otp_ctrl_ast_inputs_cfg.sv | 35 + .../otp_ctrl/dv/env/otp_ctrl_env.core | 62 + .../otp_ctrl/dv/env/otp_ctrl_env.sv | 106 + .../otp_ctrl/dv/env/otp_ctrl_env_cfg.sv | 106 + .../otp_ctrl/dv/env/otp_ctrl_env_cov.sv | 470 ++ .../otp_ctrl/dv/env/otp_ctrl_env_pkg.sv | 297 + .../ip_autogen/otp_ctrl/dv/env/otp_ctrl_if.sv | 459 ++ .../otp_ctrl/dv/env/otp_ctrl_scoreboard.sv | 2108 ++++++ .../dv/env/otp_ctrl_virtual_sequencer.sv | 19 + .../seq_lib/otp_ctrl_background_chks_vseq.sv | 73 + .../dv/env/seq_lib/otp_ctrl_base_vseq.sv | 842 +++ .../dv/env/seq_lib/otp_ctrl_callback_vseq.sv | 22 + .../env/seq_lib/otp_ctrl_check_fail_vseq.sv | 32 + .../dv/env/seq_lib/otp_ctrl_common_vseq.sv | 232 + .../dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv | 51 + .../dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv | 146 + .../dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv | 235 + .../seq_lib/otp_ctrl_low_freq_read_vseq.sv | 68 + .../env/seq_lib/otp_ctrl_macro_errs_vseq.sv | 21 + .../seq_lib/otp_ctrl_parallel_base_vseq.sv | 42 + .../seq_lib/otp_ctrl_parallel_key_req_vseq.sv | 56 + .../seq_lib/otp_ctrl_parallel_lc_esc_vseq.sv | 110 + .../seq_lib/otp_ctrl_parallel_lc_req_vseq.sv | 51 + .../seq_lib/otp_ctrl_partition_walk_vseq.sv | 50 + .../dv/env/seq_lib/otp_ctrl_regwen_vseq.sv | 69 + .../dv/env/seq_lib/otp_ctrl_smoke_vseq.sv | 287 + .../env/seq_lib/otp_ctrl_stress_all_vseq.sv | 81 + .../env/seq_lib/otp_ctrl_test_access_vseq.sv | 44 + .../dv/env/seq_lib/otp_ctrl_vseq_list.sv | 24 + .../dv/env/seq_lib/otp_ctrl_wake_up_vseq.sv | 56 + .../ip_autogen/otp_ctrl/dv/otp_ctrl_sim.core | 37 + .../otp_ctrl/dv/otp_ctrl_sim_cfg.hjson | 14 +- .../otp_ctrl/dv/sva/otp_ctrl_bind.sv | 32 + .../otp_ctrl/dv/sva/otp_ctrl_sva.core | 41 + .../ip_autogen/otp_ctrl/dv/tb.sv | 238 + .../otp_ctrl/dv/tests/otp_ctrl_base_test.sv | 51 + .../otp_ctrl/dv/tests/otp_ctrl_test.core | 22 + .../otp_ctrl/dv/tests/otp_ctrl_test_pkg.sv | 22 + .../ip_autogen/otp_ctrl/lint/otp_ctrl.vbl} | 2 +- .../ip_autogen/otp_ctrl/lint/otp_ctrl.vlt | 21 + .../ip_autogen/otp_ctrl/lint/otp_ctrl.waiver | 26 + .../ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vbl | 5 + .../ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vlt | 5 + .../otp_ctrl/lint/otp_ctrl_pkg.waiver | 5 + .../lint/otp_ctrl_top_specific_pkg.vbl | 8 + .../lint/otp_ctrl_top_specific_pkg.vlt | 5 + .../lint/otp_ctrl_top_specific_pkg.waiver | 9 + .../ip_autogen/otp_ctrl/otp_ctrl.core | 106 + .../ip_autogen/otp_ctrl/otp_ctrl_pkg.core | 67 + .../otp_ctrl/otp_ctrl_prim_reg_top.core | 29 + .../otp_ctrl/otp_ctrl_top_specific_pkg.core | 75 + .../ip_autogen/otp_ctrl/rtl/otp_ctrl.sv | 1572 +++++ .../otp_ctrl/rtl/otp_ctrl_core_reg_top.sv | 3997 +++++++++++ .../ip_autogen/otp_ctrl/rtl/otp_ctrl_dai.sv | 859 +++ .../otp_ctrl/rtl/otp_ctrl_ecc_reg.sv | 105 + .../ip_autogen/otp_ctrl/rtl/otp_ctrl_kdi.sv | 604 ++ .../ip_autogen/otp_ctrl/rtl/otp_ctrl_lci.sv | 299 + .../otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv | 398 ++ .../otp_ctrl/rtl/otp_ctrl_part_buf.sv | 822 +++ .../otp_ctrl/rtl/otp_ctrl_part_pkg.sv | 997 +++ .../otp_ctrl/rtl/otp_ctrl_part_unbuf.sv | 532 ++ .../ip_autogen/otp_ctrl/rtl/otp_ctrl_pkg.sv | 232 + .../otp_ctrl/rtl/otp_ctrl_prim_reg_top.sv | 0 .../otp_ctrl/rtl/otp_ctrl_reg_pkg.sv | 1527 ++++ .../otp_ctrl/rtl/otp_ctrl_scrmbl.sv | 512 ++ .../otp_ctrl/rtl/otp_ctrl_token_const.sv | 65 + .../otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv | 108 + .../ip_autogen/otp_ctrl/syn/constraints.sdc | 50 + .../otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson | 24 + .../otp_ctrl/syn/otp_ctrl_syn_cfg.hjson | 26 + .../otp_ctrl/syn/post_elab_gtech.tcl | 9 + .../pwrmgr/doc/theory_of_operation.md | 2 +- .../lint/top_darjeeling_dv_lint_cfgs.hjson | 4 +- .../lint/top_darjeeling_lint_cfgs.hjson | 4 +- .../rtl/autogen/top_darjeeling_pkg.sv | 2 +- .../autogen/top_darjeeling_rnd_cnst_pkg.sv | 214 +- .../sw/autogen/chip/top_darjeeling.rs | 2 +- hw/top_darjeeling/sw/autogen/top_darjeeling.h | 2 +- .../sw/autogen/top_darjeeling_memory.h | 2 +- hw/top_darjeeling/top_darjeeling.core | 6 +- hw/top_earlgrey/BUILD | 26 +- hw/top_earlgrey/data/autogen/defs.bzl | 2 +- .../data/autogen/top_earlgrey.gen.hjson | 123 +- .../data => top_earlgrey/data/otp}/BUILD | 38 - .../otp}/otp_ctrl_img_creator_sw_cfg.hjson | 0 .../data/otp/otp_ctrl_img_dev.hjson | 77 + .../data/otp}/otp_ctrl_img_hw_cfg.hjson | 0 .../data/otp}/otp_ctrl_img_owner_sw_cfg.hjson | 0 .../data/otp/otp_ctrl_img_prod.hjson | 77 + .../data/otp/otp_ctrl_img_raw.hjson | 27 + .../data/otp/otp_ctrl_img_rma.hjson | 77 + .../data/otp/otp_ctrl_img_test_locked0.hjson | 37 + .../data/otp/otp_ctrl_img_test_locked1.hjson | 37 + .../otp/otp_ctrl_img_test_unlocked0.hjson | 37 + .../otp/otp_ctrl_img_test_unlocked1.hjson | 37 + .../otp/otp_ctrl_img_test_unlocked2.hjson | 37 + .../data/otp}/otp_ctrl_mmap.hjson | 0 .../data/otp/sival_skus}/BUILD | 27 +- hw/top_earlgrey/data/top_earlgrey.hjson | 1 + hw/top_earlgrey/dv/chip_sim_cfg.hjson | 2 +- .../dv/top_earlgrey_sim_cfgs.hjson | 2 +- .../dv/verilator/verilator_sim_cfg.hjson | 2 +- .../formal/top_earlgrey_fpv_sec_cm_cfgs.hjson | 4 +- hw/top_earlgrey/ip/BUILD | 1 + .../flash_ctrl/doc/programmers_guide.md | 2 +- .../flash_ctrl/doc/theory_of_operation.md | 2 +- hw/top_earlgrey/ip_autogen/otp_ctrl/BUILD | 10 + .../ip_autogen}/otp_ctrl/README.md | 8 +- .../otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet | 684 ++ .../otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet | 614 ++ .../data/dif_otp_ctrl_unittest.cc.tpl.not_yet | 789 +++ .../ip_autogen}/otp_ctrl/data/otp_ctrl.hjson | 9 +- .../data/otp_ctrl_sec_cm_testplan.hjson | 303 + .../otp_ctrl/data/otp_ctrl_testplan.hjson | 404 ++ .../data/top_earlgrey_otp_ctrl.ipconfig.hjson | 1990 ++++++ .../ip_autogen}/otp_ctrl/defs.bzl | 2 +- .../ip_autogen/otp_ctrl/doc/checklist.md | 271 + .../ip_autogen}/otp_ctrl/doc/interfaces.md | 14 +- .../doc/otp_ctrl_behavioral_model.svg | 1 + .../otp_ctrl/doc/otp_ctrl_blockdiag.svg | 1 + .../otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg | 1 + .../otp_ctrl/doc/otp_ctrl_dai_fsm.svg | 1 + .../doc/otp_ctrl_digest_mechanism.svg | 1 + .../otp_ctrl/doc/otp_ctrl_digests.md | 17 + .../doc/otp_ctrl_field_descriptions.md | 101 + .../otp_ctrl/doc/otp_ctrl_kdi_fsm.svg | 1 + .../otp_ctrl/doc/otp_ctrl_key_req_ack.svg | 1 + .../otp_ctrl/doc/otp_ctrl_lci_fsm.svg | 1 + .../ip_autogen/otp_ctrl/doc/otp_ctrl_mmap.md | 122 + .../otp_ctrl/doc/otp_ctrl_overview.svg | 1 + .../otp_ctrl/doc/otp_ctrl_partitions.md | 57 + .../otp_ctrl/doc/otp_ctrl_prim_otp.svg | 1 + .../otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg | 1 + .../otp_ctrl/doc/programmers_guide.md | 261 + .../ip_autogen}/otp_ctrl/doc/registers.md | 2 +- .../otp_ctrl/doc/theory_of_operation.md | 514 ++ .../ip_autogen}/otp_ctrl/dv/README.md | 42 +- .../otp_ctrl/dv/cov/otp_ctrl_cov.core | 8 +- .../otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv | 3 - .../dv/cov/otp_ctrl_cov_fsm_unr_excl.el | 199 + .../otp_ctrl/dv/cov/otp_ctrl_cov_if.sv | 113 + .../otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el | 6133 +++++++++++++++++ .../otp_ctrl/dv/cov/otp_ctrl_cover.cfg | 14 + .../ip_autogen/otp_ctrl/dv/doc/tb.svg | 1 + .../dv/env/otp_ctrl_ast_inputs_cfg.sv | 35 + .../otp_ctrl/dv/env/otp_ctrl_env.core | 5 +- .../otp_ctrl/dv/env/otp_ctrl_env.sv | 106 + .../otp_ctrl/dv/env/otp_ctrl_env_cfg.sv | 106 + .../otp_ctrl/dv/env/otp_ctrl_env_cov.sv | 3 - .../otp_ctrl/dv/env/otp_ctrl_env_pkg.sv | 4 +- .../otp_ctrl/dv/env/otp_ctrl_if.sv | 3 - .../otp_ctrl/dv/env/otp_ctrl_scoreboard.sv | 3 - .../dv/env/otp_ctrl_virtual_sequencer.sv | 19 + .../seq_lib/otp_ctrl_background_chks_vseq.sv | 73 + .../dv/env/seq_lib/otp_ctrl_base_vseq.sv | 9 +- .../dv/env/seq_lib/otp_ctrl_callback_vseq.sv | 22 + .../env/seq_lib/otp_ctrl_check_fail_vseq.sv | 32 + .../dv/env/seq_lib/otp_ctrl_common_vseq.sv | 232 + .../dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv | 51 + .../dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv | 3 - .../dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv | 235 + .../seq_lib/otp_ctrl_low_freq_read_vseq.sv | 68 + .../env/seq_lib/otp_ctrl_macro_errs_vseq.sv | 21 + .../seq_lib/otp_ctrl_parallel_base_vseq.sv | 42 + .../seq_lib/otp_ctrl_parallel_key_req_vseq.sv | 56 + .../seq_lib/otp_ctrl_parallel_lc_esc_vseq.sv | 110 + .../seq_lib/otp_ctrl_parallel_lc_req_vseq.sv | 51 + .../seq_lib/otp_ctrl_partition_walk_vseq.sv | 50 + .../dv/env/seq_lib/otp_ctrl_regwen_vseq.sv | 69 + .../dv/env/seq_lib/otp_ctrl_smoke_vseq.sv | 3 - .../env/seq_lib/otp_ctrl_stress_all_vseq.sv | 81 + .../env/seq_lib/otp_ctrl_test_access_vseq.sv | 44 + .../dv/env/seq_lib/otp_ctrl_vseq_list.sv | 24 + .../dv/env/seq_lib/otp_ctrl_wake_up_vseq.sv | 56 + .../ip_autogen}/otp_ctrl/dv/otp_ctrl_sim.core | 13 +- .../otp_ctrl/dv/otp_ctrl_sim_cfg.hjson | 179 + .../otp_ctrl/dv/sva/otp_ctrl_bind.sv | 32 + .../otp_ctrl/dv/sva/otp_ctrl_sva.core | 7 +- hw/top_earlgrey/ip_autogen/otp_ctrl/dv/tb.sv | 238 + .../otp_ctrl/dv/tests/otp_ctrl_base_test.sv | 51 + .../otp_ctrl/dv/tests/otp_ctrl_test.core | 7 +- .../otp_ctrl/dv/tests/otp_ctrl_test_pkg.sv | 22 + .../ip_autogen/otp_ctrl/lint/otp_ctrl.vbl | 8 + .../ip_autogen/otp_ctrl/lint/otp_ctrl.vlt | 21 + .../ip_autogen/otp_ctrl/lint/otp_ctrl.waiver | 26 + .../ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vbl | 5 + .../ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vlt | 5 + .../otp_ctrl/lint/otp_ctrl_pkg.waiver | 5 + .../lint/otp_ctrl_top_specific_pkg.vbl | 8 + .../lint/otp_ctrl_top_specific_pkg.vlt | 5 + .../lint/otp_ctrl_top_specific_pkg.waiver | 9 + .../ip_autogen}/otp_ctrl/otp_ctrl.core | 5 +- .../ip_autogen/otp_ctrl/otp_ctrl_pkg.core | 67 + .../otp_ctrl/otp_ctrl_prim_reg_top.core | 29 + .../otp_ctrl/otp_ctrl_top_specific_pkg.core | 75 + .../ip_autogen/otp_ctrl/rtl/otp_ctrl.sv | 1572 +++++ .../otp_ctrl/rtl/otp_ctrl_core_reg_top.sv | 0 .../ip_autogen/otp_ctrl/rtl/otp_ctrl_dai.sv | 859 +++ .../otp_ctrl/rtl/otp_ctrl_ecc_reg.sv | 105 + .../ip_autogen/otp_ctrl/rtl/otp_ctrl_kdi.sv | 604 ++ .../ip_autogen/otp_ctrl/rtl/otp_ctrl_lci.sv | 299 + .../otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv | 398 ++ .../otp_ctrl/rtl/otp_ctrl_part_buf.sv | 822 +++ .../otp_ctrl/rtl/otp_ctrl_part_pkg.sv | 4 +- .../otp_ctrl/rtl/otp_ctrl_part_unbuf.sv | 532 ++ .../ip_autogen/otp_ctrl/rtl/otp_ctrl_pkg.sv | 232 + .../otp_ctrl/rtl/otp_ctrl_prim_reg_top.sv | 1467 ++++ .../otp_ctrl/rtl/otp_ctrl_reg_pkg.sv | 0 .../otp_ctrl/rtl/otp_ctrl_scrmbl.sv | 512 ++ .../otp_ctrl/rtl/otp_ctrl_token_const.sv | 65 + .../otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv | 108 + .../ip_autogen/otp_ctrl/syn/constraints.sdc | 50 + .../otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson | 4 +- .../otp_ctrl/syn/otp_ctrl_syn_cfg.hjson | 4 +- .../otp_ctrl/syn/post_elab_gtech.tcl | 9 + .../pwrmgr/doc/theory_of_operation.md | 2 +- .../lint/top_earlgrey_dv_lint_cfgs.hjson | 4 +- .../lint/top_earlgrey_lint_cfgs.hjson | 4 +- .../rtl/autogen/top_earlgrey_rnd_cnst_pkg.sv | 222 +- .../syn/top_earlgrey_batch_syn_cfg.hjson | 2 +- .../top_earlgrey_gtech_batch_syn_cfg.hjson | 2 +- hw/top_earlgrey/top_earlgrey.core | 6 +- .../lint/top_englishbreakfast_lint_cfgs.hjson | 2 +- .../top_englishbreakfast.core | 1 - release/BUILD | 2 +- rules/lc.bzl | 2 +- rules/opentitan/legacy.bzl | 6 +- rules/otp.bzl | 84 +- sw/device/lib/dif/dif_otp_ctrl.c | 3 - sw/device/lib/dif/dif_otp_ctrl.h | 7 +- sw/device/lib/dif/dif_otp_ctrl.md | 2 +- sw/device/lib/dif/dif_otp_ctrl_unittest.cc | 3 - sw/device/silicon_creator/lib/drivers/BUILD | 2 +- sw/device/silicon_creator/manuf/README.md | 2 +- sw/device/silicon_creator/manuf/base/BUILD | 6 +- .../manuf/base/provisioning_inputs.bzl | 2 +- sw/device/silicon_creator/manuf/lib/BUILD | 6 +- sw/device/silicon_creator/manuf/tests/BUILD | 14 +- sw/device/silicon_creator/rom/e2e/BUILD | 2 +- .../rom/e2e/boot_data_recovery/BUILD | 2 +- .../rom/e2e/boot_policy_bad_manifest/BUILD | 2 +- .../rom/e2e/boot_policy_big_image/BUILD | 2 +- .../rom/e2e/boot_policy_flash_ecc_error/BUILD | 4 +- .../rom/e2e/boot_policy_newer/BUILD | 2 +- .../rom/e2e/boot_policy_rollback/BUILD | 2 +- .../rom/e2e/boot_policy_valid/BUILD | 2 +- .../silicon_creator/rom/e2e/bootstrap/BUILD | 4 +- .../silicon_creator/rom/e2e/epmp_init/BUILD | 4 +- .../rom/e2e/immutable_rom_ext_section/BUILD | 2 +- .../silicon_creator/rom/e2e/jtag_inject/BUILD | 4 +- .../silicon_creator/rom/e2e/keymgr/BUILD | 2 +- .../rom/e2e/presigned_images/BUILD | 2 +- .../silicon_creator/rom/e2e/release/BUILD | 2 +- .../rom/e2e/reset_reason/BUILD | 4 +- .../rom/e2e/retention_ram/BUILD | 2 +- .../rom/e2e/rom_e2e_bootstrap_entry/BUILD | 2 +- .../rom/e2e/rom_ext_upgrade_interrupt/BUILD | 2 +- .../rom/e2e/shutdown_alert/BUILD | 4 +- .../rom/e2e/shutdown_output/BUILD | 2 +- .../rom/e2e/shutdown_redact/BUILD | 2 +- .../rom/e2e/shutdown_watchdog/BUILD | 2 +- .../rom/e2e/sigverify_always/BUILD | 2 +- .../rom/e2e/sigverify_key_type/BUILD | 2 +- .../rom/e2e/sigverify_key_validity/BUILD | 2 +- .../rom/e2e/sigverify_spx/BUILD | 2 +- .../rom/e2e/sigverify_usage_constraints/BUILD | 2 +- sw/device/silicon_creator/rom/e2e/sram/BUILD | 2 +- .../silicon_creator/rom/e2e/watchdog/BUILD | 4 +- sw/device/silicon_creator/rom_ext/e2e/BUILD | 2 +- .../rom_ext/e2e/verified_boot/BUILD | 2 +- sw/device/tests/BUILD | 38 +- sw/device/tests/README.md | 4 +- sw/device/tests/doc/sival/README.md | 3 +- sw/device/tests/doc/sival/devguide.md | 4 +- sw/device/tock/BUILD | 1 + util/design/BUILD | 11 + util/design/README.md | 86 +- util/design/data/BUILD | 29 + .../design}/data/otp_ctrl_img.c.tpl | 2 +- util/design/gen-flash-img.py | 10 +- util/design/gen-otp-img.py | 4 +- util/design/gen-otp-mmap.py | 198 +- util/design/lib/BUILD | 2 +- util/design/lib/OtpMemMap.py | 42 +- util/design/lib/__init__.py | 8 + util/design/mubi/__init__.py | 3 + util/ipgen/lib.py | 17 +- util/mdbook_dashboard.py | 40 +- util/site/blocks.json | 4 +- util/site/fetch_block_stats.py | 2 +- util/topgen-fusesoc.py | 4 +- util/topgen.py | 26 +- 489 files changed, 65718 insertions(+), 1566 deletions(-) delete mode 100644 hw/ip/otp_ctrl/data/README.md delete mode 100644 hw/ip/otp_ctrl/doc/checklist.md rename hw/{ip => ip_templates}/otp_ctrl/BUILD (76%) create mode 100644 hw/ip_templates/otp_ctrl/BUILD.tpl create mode 100644 hw/ip_templates/otp_ctrl/README.md.tpl rename hw/{ip/otp_ctrl/data/dif_otp_ctrl.c.tpl => ip_templates/otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet} (99%) rename hw/{ip/otp_ctrl/data/dif_otp_ctrl.h.tpl => ip_templates/otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet} (99%) rename hw/{ip/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl => ip_templates/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl.not_yet} (99%) rename hw/{ip => ip_templates}/otp_ctrl/data/otp_ctrl.hjson.tpl (98%) create mode 100644 hw/ip_templates/otp_ctrl/data/otp_ctrl.tpldesc.hjson rename hw/{ip => ip_templates}/otp_ctrl/data/otp_ctrl_sec_cm_testplan.hjson (100%) rename hw/{ip => ip_templates}/otp_ctrl/data/otp_ctrl_testplan.hjson (100%) create mode 100644 hw/ip_templates/otp_ctrl/defs.bzl.tpl create mode 100644 hw/ip_templates/otp_ctrl/doc/checklist.md create mode 100644 hw/ip_templates/otp_ctrl/doc/interfaces.md.tpl rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_behavioral_model.svg (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_blockdiag.svg (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_dai_fsm.svg (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_digest_mechanism.svg (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_digests.md (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_field_descriptions.md (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_kdi_fsm.svg (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_key_req_ack.svg (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_lci_fsm.svg (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_mmap.md (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_overview.svg (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_partitions.md (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_prim_otp.svg (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg (100%) rename hw/{ip => ip_templates}/otp_ctrl/doc/programmers_guide.md (98%) create mode 100644 hw/ip_templates/otp_ctrl/doc/registers.md.tpl rename hw/{ip => ip_templates}/otp_ctrl/doc/theory_of_operation.md (96%) create mode 100644 hw/ip_templates/otp_ctrl/dv/README.md.tpl create mode 100644 hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cov.core.tpl rename hw/{ip/otp_ctrl/data => ip_templates/otp_ctrl/dv/cov}/otp_ctrl_cov_bind.sv.tpl (96%) rename hw/{ip => ip_templates}/otp_ctrl/dv/cov/otp_ctrl_cov_fsm_unr_excl.el (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/cov/otp_ctrl_cov_if.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/cov/otp_ctrl_cover.cfg (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/doc/tb.svg (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/otp_ctrl_ast_inputs_cfg.sv (100%) create mode 100644 hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_env.core.tpl rename hw/{ip => ip_templates}/otp_ctrl/dv/env/otp_ctrl_env.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/otp_ctrl_env_cfg.sv (100%) rename hw/{ip/otp_ctrl/data => ip_templates/otp_ctrl/dv/env}/otp_ctrl_env_cov.sv.tpl (97%) rename hw/{ip/otp_ctrl/data => ip_templates/otp_ctrl/dv/env}/otp_ctrl_env_pkg.sv.tpl (97%) rename hw/{ip/otp_ctrl/data => ip_templates/otp_ctrl/dv/env}/otp_ctrl_if.sv.tpl (98%) rename hw/{ip/otp_ctrl/data => ip_templates/otp_ctrl/dv/env}/otp_ctrl_scoreboard.sv.tpl (99%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/otp_ctrl_virtual_sequencer.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_background_chks_vseq.sv (100%) rename hw/{ip/otp_ctrl/data => ip_templates/otp_ctrl/dv/env/seq_lib}/otp_ctrl_base_vseq.sv.tpl (97%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_callback_vseq.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_check_fail_vseq.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_common_vseq.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv (100%) rename hw/{ip/otp_ctrl/data => ip_templates/otp_ctrl/dv/env/seq_lib}/otp_ctrl_dai_lock_vseq.sv.tpl (94%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_low_freq_read_vseq.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_macro_errs_vseq.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_base_vseq.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_key_req_vseq.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_lc_esc_vseq.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_lc_req_vseq.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_partition_walk_vseq.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_regwen_vseq.sv (100%) rename hw/{ip/otp_ctrl/data => ip_templates/otp_ctrl/dv/env/seq_lib}/otp_ctrl_smoke_vseq.sv.tpl (99%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_stress_all_vseq.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_test_access_vseq.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_vseq_list.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/env/seq_lib/otp_ctrl_wake_up_vseq.sv (100%) create mode 100644 hw/ip_templates/otp_ctrl/dv/otp_ctrl_sim.core.tpl create mode 100644 hw/ip_templates/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson.tpl rename hw/{ip => ip_templates}/otp_ctrl/dv/sva/otp_ctrl_bind.sv (100%) create mode 100644 hw/ip_templates/otp_ctrl/dv/sva/otp_ctrl_sva.core.tpl rename hw/{ip => ip_templates}/otp_ctrl/dv/tb.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/dv/tests/otp_ctrl_base_test.sv (100%) create mode 100644 hw/ip_templates/otp_ctrl/dv/tests/otp_ctrl_test.core.tpl rename hw/{ip => ip_templates}/otp_ctrl/dv/tests/otp_ctrl_test_pkg.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/lint/otp_ctrl.vbl (100%) rename hw/{ip => ip_templates}/otp_ctrl/lint/otp_ctrl.vlt (100%) rename hw/{ip => ip_templates}/otp_ctrl/lint/otp_ctrl.waiver (100%) create mode 100644 hw/ip_templates/otp_ctrl/lint/otp_ctrl_pkg.vbl rename hw/{ip => ip_templates}/otp_ctrl/lint/otp_ctrl_pkg.vlt (77%) create mode 100644 hw/ip_templates/otp_ctrl/lint/otp_ctrl_pkg.waiver create mode 100644 hw/ip_templates/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vbl create mode 100644 hw/ip_templates/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vlt rename hw/{ip/otp_ctrl/lint/otp_ctrl_pkg.waiver => ip_templates/otp_ctrl/lint/otp_ctrl_top_specific_pkg.waiver} (86%) create mode 100644 hw/ip_templates/otp_ctrl/otp_ctrl.core.tpl rename hw/{ip => ip_templates}/otp_ctrl/otp_ctrl_pkg.core (90%) rename hw/{ip => ip_templates}/otp_ctrl/otp_ctrl_prim_reg_top.core (71%) create mode 100644 hw/ip_templates/otp_ctrl/otp_ctrl_top_specific_pkg.core.tpl rename hw/{ip => ip_templates}/otp_ctrl/rtl/otp_ctrl.sv (99%) rename hw/{ip => ip_templates}/otp_ctrl/rtl/otp_ctrl_dai.sv (99%) rename hw/{ip => ip_templates}/otp_ctrl/rtl/otp_ctrl_ecc_reg.sv (100%) rename hw/{ip => ip_templates}/otp_ctrl/rtl/otp_ctrl_kdi.sv (99%) rename hw/{ip => ip_templates}/otp_ctrl/rtl/otp_ctrl_lci.sv (99%) rename hw/{ip => ip_templates}/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv (99%) rename hw/{ip => ip_templates}/otp_ctrl/rtl/otp_ctrl_part_buf.sv (99%) rename hw/{ip/otp_ctrl/data => ip_templates/otp_ctrl/rtl}/otp_ctrl_part_pkg.sv.tpl (87%) rename hw/{ip => ip_templates}/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv (99%) rename hw/{ip => ip_templates}/otp_ctrl/rtl/otp_ctrl_pkg.sv (70%) rename hw/{ip => ip_templates}/otp_ctrl/rtl/otp_ctrl_scrmbl.sv (97%) rename hw/{ip => ip_templates}/otp_ctrl/rtl/otp_ctrl_token_const.sv (96%) create mode 100644 hw/ip_templates/otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv rename hw/{ip => ip_templates}/otp_ctrl/syn/constraints.sdc (100%) create mode 100644 hw/ip_templates/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson.tpl create mode 100644 hw/ip_templates/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson.tpl rename hw/{ip => ip_templates}/otp_ctrl/syn/post_elab_gtech.tcl (100%) create mode 100644 hw/top_darjeeling/data/otp/otp_ctrl_img_creator_sw_cfg.hjson rename hw/{ip/otp_ctrl/data => top_darjeeling/data/otp}/otp_ctrl_img_dev.hjson (100%) create mode 100644 hw/top_darjeeling/data/otp/otp_ctrl_img_hw_cfg.hjson create mode 100644 hw/top_darjeeling/data/otp/otp_ctrl_img_owner_sw_cfg.hjson rename hw/{ip/otp_ctrl/data => top_darjeeling/data/otp}/otp_ctrl_img_prod.hjson (100%) rename hw/{ip/otp_ctrl/data => top_darjeeling/data/otp}/otp_ctrl_img_raw.hjson (100%) rename hw/{ip/otp_ctrl/data => top_darjeeling/data/otp}/otp_ctrl_img_rma.hjson (100%) rename hw/{ip/otp_ctrl/data => top_darjeeling/data/otp}/otp_ctrl_img_test_locked0.hjson (100%) rename hw/{ip/otp_ctrl/data => top_darjeeling/data/otp}/otp_ctrl_img_test_locked1.hjson (100%) rename hw/{ip/otp_ctrl/data => top_darjeeling/data/otp}/otp_ctrl_img_test_unlocked0.hjson (100%) rename hw/{ip/otp_ctrl/data => top_darjeeling/data/otp}/otp_ctrl_img_test_unlocked1.hjson (100%) rename hw/{ip/otp_ctrl/data => top_darjeeling/data/otp}/otp_ctrl_img_test_unlocked2.hjson (100%) create mode 100644 hw/top_darjeeling/data/otp/otp_ctrl_mmap.hjson create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/BUILD create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/README.md create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl.not_yet create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/data/otp_ctrl.hjson create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/data/otp_ctrl_sec_cm_testplan.hjson create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/data/otp_ctrl_testplan.hjson create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/data/top_darjeeling_otp_ctrl.ipconfig.hjson create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/defs.bzl create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/checklist.md create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/interfaces.md create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_behavioral_model.svg create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_blockdiag.svg create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_dai_fsm.svg create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_digest_mechanism.svg create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_digests.md create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_field_descriptions.md create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_kdi_fsm.svg create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_key_req_ack.svg create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_lci_fsm.svg create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_mmap.md create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_overview.svg create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_partitions.md create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_prim_otp.svg create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/programmers_guide.md create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/registers.md create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/doc/theory_of_operation.md create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/README.md create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov.core create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_fsm_unr_excl.el create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_if.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cover.cfg create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/doc/tb.svg create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_ast_inputs_cfg.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.core create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cfg.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cov.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_pkg.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_if.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_scoreboard.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_virtual_sequencer.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_background_chks_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_base_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_callback_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_check_fail_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_common_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_low_freq_read_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_macro_errs_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_base_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_key_req_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_lc_esc_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_lc_req_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_partition_walk_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_regwen_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_smoke_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_stress_all_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_test_access_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_vseq_list.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_wake_up_vseq.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/otp_ctrl_sim.core rename hw/{ip => top_darjeeling/ip_autogen}/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson (90%) create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/sva/otp_ctrl_bind.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/sva/otp_ctrl_sva.core create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tb.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tests/otp_ctrl_base_test.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tests/otp_ctrl_test.core create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tests/otp_ctrl_test_pkg.sv rename hw/{ip/otp_ctrl/lint/otp_ctrl_pkg.vbl => top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl.vbl} (87%) create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl.vlt create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl.waiver create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vbl create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vlt create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.waiver create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vbl create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vlt create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.waiver create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl.core create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl_pkg.core create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl_prim_reg_top.core create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl_top_specific_pkg.core create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_core_reg_top.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_dai.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_ecc_reg.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_kdi.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_lci.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_buf.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_pkg.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_pkg.sv rename hw/{ip => top_darjeeling/ip_autogen}/otp_ctrl/rtl/otp_ctrl_prim_reg_top.sv (100%) create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_reg_pkg.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_scrmbl.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_token_const.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/syn/constraints.sdc create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson create mode 100644 hw/top_darjeeling/ip_autogen/otp_ctrl/syn/post_elab_gtech.tcl rename hw/{ip/otp_ctrl/data => top_earlgrey/data/otp}/BUILD (94%) rename hw/{ip/otp_ctrl/data => top_earlgrey/data/otp}/otp_ctrl_img_creator_sw_cfg.hjson (100%) create mode 100644 hw/top_earlgrey/data/otp/otp_ctrl_img_dev.hjson rename hw/{ip/otp_ctrl/data => top_earlgrey/data/otp}/otp_ctrl_img_hw_cfg.hjson (100%) rename hw/{ip/otp_ctrl/data => top_earlgrey/data/otp}/otp_ctrl_img_owner_sw_cfg.hjson (100%) create mode 100644 hw/top_earlgrey/data/otp/otp_ctrl_img_prod.hjson create mode 100644 hw/top_earlgrey/data/otp/otp_ctrl_img_raw.hjson create mode 100644 hw/top_earlgrey/data/otp/otp_ctrl_img_rma.hjson create mode 100644 hw/top_earlgrey/data/otp/otp_ctrl_img_test_locked0.hjson create mode 100644 hw/top_earlgrey/data/otp/otp_ctrl_img_test_locked1.hjson create mode 100644 hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked0.hjson create mode 100644 hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked1.hjson create mode 100644 hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked2.hjson rename hw/{ip/otp_ctrl/data => top_earlgrey/data/otp}/otp_ctrl_mmap.hjson (100%) rename hw/{ip/otp_ctrl/data/earlgrey_skus/sival => top_earlgrey/data/otp/sival_skus}/BUILD (94%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/BUILD rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/README.md (87%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl.not_yet rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/data/otp_ctrl.hjson (99%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/data/otp_ctrl_sec_cm_testplan.hjson create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/data/otp_ctrl_testplan.hjson create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/data/top_earlgrey_otp_ctrl.ipconfig.hjson rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/defs.bzl (78%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/checklist.md rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/doc/interfaces.md (98%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_behavioral_model.svg create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_blockdiag.svg create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_dai_fsm.svg create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_digest_mechanism.svg create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_digests.md create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_field_descriptions.md create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_kdi_fsm.svg create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_key_req_ack.svg create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_lci_fsm.svg create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_mmap.md create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_overview.svg create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_partitions.md create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_prim_otp.svg create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/programmers_guide.md rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/doc/registers.md (99%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/doc/theory_of_operation.md rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/dv/README.md (76%) rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/dv/cov/otp_ctrl_cov.core (70%) rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv (97%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_fsm_unr_excl.el create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_if.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cover.cfg create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/doc/tb.svg create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_ast_inputs_cfg.sv rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/dv/env/otp_ctrl_env.core (96%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cfg.sv rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/dv/env/otp_ctrl_env_cov.sv (99%) rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/dv/env/otp_ctrl_env_pkg.sv (98%) rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/dv/env/otp_ctrl_if.sv (99%) rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/dv/env/otp_ctrl_scoreboard.sv (99%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_virtual_sequencer.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_background_chks_vseq.sv rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/dv/env/seq_lib/otp_ctrl_base_vseq.sv (98%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_callback_vseq.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_check_fail_vseq.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_common_vseq.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv (97%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_low_freq_read_vseq.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_macro_errs_vseq.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_base_vseq.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_key_req_vseq.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_lc_esc_vseq.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_lc_req_vseq.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_partition_walk_vseq.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_regwen_vseq.sv rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/dv/env/seq_lib/otp_ctrl_smoke_vseq.sv (99%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_stress_all_vseq.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_test_access_vseq.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_vseq_list.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_wake_up_vseq.sv rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/dv/otp_ctrl_sim.core (64%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/sva/otp_ctrl_bind.sv rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/dv/sva/otp_ctrl_sva.core (83%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/tb.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/tests/otp_ctrl_base_test.sv rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/dv/tests/otp_ctrl_test.core (73%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/dv/tests/otp_ctrl_test_pkg.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl.vbl create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl.vlt create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl.waiver create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vbl create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vlt create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.waiver create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vbl create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vlt create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.waiver rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/otp_ctrl.core (93%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl_pkg.core create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl_prim_reg_top.core create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl_top_specific_pkg.core create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl.sv rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/rtl/otp_ctrl_core_reg_top.sv (100%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_dai.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_ecc_reg.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_kdi.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_lci.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_buf.sv rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/rtl/otp_ctrl_part_pkg.sv (99%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_pkg.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_prim_reg_top.sv rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/rtl/otp_ctrl_reg_pkg.sv (100%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_scrmbl.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_token_const.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/syn/constraints.sdc rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson (81%) rename hw/{ip => top_earlgrey/ip_autogen}/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson (81%) create mode 100644 hw/top_earlgrey/ip_autogen/otp_ctrl/syn/post_elab_gtech.tcl rename {hw/ip/otp_ctrl => util/design}/data/otp_ctrl_img.c.tpl (98%) create mode 100644 util/design/lib/__init__.py create mode 100644 util/design/mubi/__init__.py diff --git a/BLOCKFILE b/BLOCKFILE index e81e83ed71eba7..d786ba5a4a156d 100644 --- a/BLOCKFILE +++ b/BLOCKFILE @@ -1,7 +1,7 @@ # If a PR changes a file that matches a pattern in this file it will be blocked # from merged by CI. The patterns as glob-like patterns matched using python # fnmatch. In particular note there is no special handling for '/' so '*' can -# match multiple directory levels e.g. 'this/is/*/a/path' matches both +# match multiple directory levels e.g. 'this/is/a/*/path' matches both # 'this/is/a/foo/path' and 'this/is/a/foo/bar/path'. # # Anyone on the COMMITTERS list can authorize a change by adding a comment @@ -49,7 +49,6 @@ hw/ip/keymgr/rtl/* hw/ip/kmac/rtl/* hw/ip/lc_ctrl/rtl/* hw/ip/otbn/rtl/* -hw/ip/otp_ctrl/rtl/* hw/ip/pattgen/rtl/* hw/ip/prim/rtl/* hw/ip/prim_generic/rtl/* @@ -92,19 +91,19 @@ hw/ip/otbn/data/otbn.hjson hw/ip/entropy_src/data/entropy_src.hjson hw/ip/aes/data/aes.hjson hw/ip/i2c/data/i2c.hjson -hw/ip/otp_ctrl/data/otp_ctrl.hjson -hw/ip/otp_ctrl/data/otp_ctrl_img_rma.hjson -hw/ip/otp_ctrl/data/otp_ctrl_img_dev.hjson -hw/ip/otp_ctrl/data/otp_ctrl_img_test_locked0.hjson -hw/ip/otp_ctrl/data/otp_ctrl_img_test_locked1.hjson -hw/ip/otp_ctrl/data/otp_ctrl_img_test_unlocked0.hjson -hw/ip/otp_ctrl/data/otp_ctrl_img_creator_sw_cfg.hjson -hw/ip/otp_ctrl/data/otp_ctrl_img_hw_cfg.hjson -hw/ip/otp_ctrl/data/otp_ctrl_img_raw.hjson -hw/ip/otp_ctrl/data/otp_ctrl_mmap.hjson -hw/ip/otp_ctrl/data/otp_ctrl_img_test_unlocked1.hjson -hw/ip/otp_ctrl/data/otp_ctrl_img_prod.hjson -hw/ip/otp_ctrl/data/otp_ctrl_img_test_unlocked2.hjson +hw/top_earlgrey/data/otp/otp_ctrl.hjson +hw/top_earlgrey/data/otp/otp_ctrl_img_rma.hjson +hw/top_earlgrey/data/otp/otp_ctrl_img_dev.hjson +hw/top_earlgrey/data/otp/otp_ctrl_img_test_locked0.hjson +hw/top_earlgrey/data/otp/otp_ctrl_img_test_locked1.hjson +hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked0.hjson +hw/top_earlgrey/data/otp/otp_ctrl_img_creator_sw_cfg.hjson +hw/top_earlgrey/data/otp/otp_ctrl_img_hw_cfg.hjson +hw/top_earlgrey/data/otp/otp_ctrl_img_raw.hjson +hw/top_earlgrey/data/otp/otp_ctrl_mmap.hjson +hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked1.hjson +hw/top_earlgrey/data/otp/otp_ctrl_img_prod.hjson +hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked2.hjson hw/ip/rv_core_ibex/data/rv_core_ibex.hjson hw/ip/pwm/data/pwm.hjson hw/ip/aon_timer/data/aon_timer.hjson diff --git a/SUMMARY.md b/SUMMARY.md index 912c167c6d1e4d..586bd37b8733fa 100644 --- a/SUMMARY.md +++ b/SUMMARY.md @@ -48,6 +48,15 @@ - [Interface and Registers](./hw/top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson) - [Device Interface Functions](./sw/device/lib/dif/dif_rv_plic.h) - [Checklist](./hw/top_earlgrey/ip_autogen/rv_plic/doc/checklist.md) + - [OTP Controller](./hw/top_earlgrey/ip_autogen/otp_ctrl/README.md) + - [Theory of Operation](./hw/top_earlgrey/ip_autogen/otp_ctrl/doc/theory_of_operation.md) + - [Design Verification](./hw/top_earlgrey/ip_autogen/otp_ctrl/dv/README.md) + - [Testplan](./hw/top_earlgrey/ip_autogen/otp_ctrl/data/otp_ctrl_testplan.hjson) + - [Programmer's Guide](./hw/top_earlgrey/ip_autogen/otp_ctrl/doc/programmers_guide.md) + - [Hardware Interfaces](./hw/top_earlgrey/ip_autogen/otp_ctrl/doc/interfaces.md) + - [Registers](./hw/top_earlgrey/ip_autogen/otp_ctrl/doc/registers.md) + - [Device Interface Functions](./sw/device/lib/dif/dif_otp_ctrl.h) + - [Checklist](./hw/top_earlgrey/ip_autogen/otp_ctrl/doc/checklist.md) - [Sensor Control](./hw/top_earlgrey/ip/sensor_ctrl/README.md) - [Theory of Operation](./hw/top_earlgrey/ip/sensor_ctrl/doc/theory_of_operation.md) - [Programmer's Guide](./hw/top_earlgrey/ip/sensor_ctrl/doc/programmers_guide.md) @@ -63,6 +72,15 @@ - [Top Darjeeling](./hw/top_darjeeling/README.md) - [Datasheet](./hw/top_darjeeling/doc/datasheet.md) + - [OTP Controller](./hw/top_darjeeling/ip_autogen/otp_ctrl/README.md) + - [Theory of Operation](./hw/top_darjeeling/ip_autogen/otp_ctrl/doc/theory_of_operation.md) + - [Design Verification](./hw/top_darjeeling/ip_autogen/otp_ctrl/dv/README.md) + - [Testplan](./hw/top_darjeeling/ip_autogen/otp_ctrl/data/otp_ctrl_testplan.hjson) + - [Programmer's Guide](./hw/top_darjeeling/ip_autogen/otp_ctrl/doc/programmers_guide.md) + - [Hardware Interfaces](./hw/top_darjeeling/ip_autogen/otp_ctrl/doc/interfaces.md) + - [Registers](./hw/top_darjeeling/ip_autogen/otp_ctrl/doc/registers.md) + - [Device Interface Functions](./sw/device/lib/dif/dif_otp_ctrl.h) + - [Checklist](./hw/top_darjeeling/ip_autogen/otp_ctrl/doc/checklist.md) - [Cores](./hw/doc/cores.md) - [Ibex RISC-V Core Wrapper](./hw/ip/rv_core_ibex/README.md) @@ -229,15 +247,6 @@ - [Registers](./hw/ip/lc_ctrl/doc/registers.md) - [Device Interface Functions](./sw/device/lib/dif/dif_lc_ctrl.h) - [Checklist](./hw/ip/lc_ctrl/doc/checklist.md) - - [OTP Controller](./hw/ip/otp_ctrl/README.md) - - [Theory of Operation](./hw/ip/otp_ctrl/doc/theory_of_operation.md) - - [Design Verification](./hw/ip/otp_ctrl/dv/README.md) - - [Testplan](./hw/ip/otp_ctrl/data/otp_ctrl_testplan.hjson) - - [Programmer's Guide](./hw/ip/otp_ctrl/doc/programmers_guide.md) - - [Hardware Interfaces](./hw/ip/otp_ctrl/doc/interfaces.md) - - [Registers](./hw/ip/otp_ctrl/doc/registers.md) - - [Device Interface Functions](./sw/device/lib/dif/dif_otp_ctrl.h) - - [Checklist](./hw/ip/otp_ctrl/doc/checklist.md) - [Pattern Generator](./hw/ip/pattgen/README.md) - [Theory of Operation](./hw/ip/pattgen/doc/theory_of_operation.md) - [Design Verification](./hw/ip/pattgen/dv/README.md) @@ -419,7 +428,7 @@ - [Device Software](./sw/device/README.md) - [Build & Test Rules](./rules/opentitan/README.md) - [FPGA Bitstreams](./hw/bitstream/README.md) - - [OTP Build and Test Infrastructure](./hw/ip/otp_ctrl/data/README.md) + - [OTP Preload Image Generator](./util/design/README.md#otp_preload_image_generator) - [Device Libraries](./sw/device/lib/README.md) - [DIF Library](./sw/device/lib/dif/README.md) - [ADC Checklist](sw/device/lib/dif/dif_adc_ctrl.md) diff --git a/ci/scripts/check-generated.sh b/ci/scripts/check-generated.sh index 6f26086bbc7bb9..ccca422d1931a1 100755 --- a/ci/scripts/check-generated.sh +++ b/ci/scripts/check-generated.sh @@ -67,7 +67,6 @@ fi destructive_cleanup gen_hw_and_check_clean "Register headers" regs || bad=1 -gen_hw_and_check_clean "OTP memory map" otp-mmap || bad=1 gen_hw_and_check_clean "LC state" lc-state-enc || bad=1 # This runs both top and cmdgen targets together since for ipgen diff --git a/ci/scripts/run-english-breakfast-verilator-tests.sh b/ci/scripts/run-english-breakfast-verilator-tests.sh index 733d89f6ae887b..09e056fc3317b0 100755 --- a/ci/scripts/run-english-breakfast-verilator-tests.sh +++ b/ci/scripts/run-english-breakfast-verilator-tests.sh @@ -21,7 +21,7 @@ set -e ./bazelisk.sh build \ --copt=-DOT_IS_ENGLISH_BREAKFAST_REDUCED_SUPPORT_FOR_INTERNAL_USE_ONLY_ \ --features=-rv32_bitmanip \ - //sw/host/opentitantool //hw/ip/otp_ctrl/data:img_rma + //sw/host/opentitantool //hw/top_earlgrey/data/otp:img_rma # Run the one test. # This needs to be run outside the bazel sandbox, so we do not use `bazel run` diff --git a/doc/getting_started/setup_fpga.md b/doc/getting_started/setup_fpga.md index 98e0136652b184..4bf872f1d3db20 100644 --- a/doc/getting_started/setup_fpga.md +++ b/doc/getting_started/setup_fpga.md @@ -69,7 +69,7 @@ Specifically, you can build the [`//hw/bitstream/universal:splice`](https://gith For example, to splice a CW310 bitstream with the mask ROM image and a specific OTP image, you can run ```sh bazel build \ - --//hw/bitstream/universal:otp=//hw/ip/otp_ctrl/data:img_dev \ + --//hw/bitstream/universal:otp=//hw/top_earlgrey/data/otp:img_dev \ --//hw/bitstream/universal:env=//hw/top_earlgrey:fpga_cw310_rom_with_fake_keys \ //hw/bitstream/universal:splice ``` diff --git a/doc/getting_started/setup_verilator.md b/doc/getting_started/setup_verilator.md index 10495dd3a66db1..3facc1250f8eab 100644 --- a/doc/getting_started/setup_verilator.md +++ b/doc/getting_started/setup_verilator.md @@ -74,8 +74,8 @@ bazel test --test_tag_filters=verilator --test_output=streamed //sw/device/tests You should expect to see something like: ```console -Invoking: sw/host/opentitantool/opentitantool --rcfile= --logging=info --interface=verilator --verilator-bin=hw/build.verilator_real/sim-verilator/Vchip_sim_tb --verilator-rom=sw/device/lib/testing/test_rom/test_rom_sim_verilator.scr.39.vmem --verilator-flash=sw/device/tests/uart_smoketest_prog_sim_verilator.64.scr.vmem --verilator-otp=hw/ip/otp_ctrl/data/img_rma.vmem console --exit-failure=(FAIL|FAULT).*\n --exit-success=PASS.*\n --timeout=3600s -[2022-06-09T08:08:16Z INFO opentitanlib::transport::verilator::subprocess] Spawning verilator: "hw/build.verilator_real/sim-verilator/Vchip_sim_tb" ["--meminit=rom,sw/device/lib/testing/test_rom/test_rom_sim_verilator.scr.39.vmem", "--meminit=flash,sw/device/tests/uart_smoketest_prog_sim_verilator.64.scr.vmem", "--meminit=otp,hw/ip/otp_ctrl/data/img_rma.vmem"] +Invoking: sw/host/opentitantool/opentitantool --rcfile= --logging=info --interface=verilator --verilator-bin=hw/build.verilator_real/sim-verilator/Vchip_sim_tb --verilator-rom=sw/device/lib/testing/test_rom/test_rom_sim_verilator.scr.39.vmem --verilator-flash=sw/device/tests/uart_smoketest_prog_sim_verilator.64.scr.vmem --verilator-otp=hw/top_earlgrey/data/otp/img_rma.vmem console --exit-failure=(FAIL|FAULT).*\n --exit-success=PASS.*\n --timeout=3600s +[2022-06-09T08:08:16Z INFO opentitanlib::transport::verilator::subprocess] Spawning verilator: "hw/build.verilator_real/sim-verilator/Vchip_sim_tb" ["--meminit=rom,sw/device/lib/testing/test_rom/test_rom_sim_verilator.scr.39.vmem", "--meminit=flash,sw/device/tests/uart_smoketest_prog_sim_verilator.64.scr.vmem", "--meminit=otp,hw/top_earlgrey/data/otp/img_rma.vmem"] [2022-06-09T08:08:16Z INFO opentitanlib::transport::verilator::stdout] Simulation of OpenTitan Earl Grey [2022-06-09T08:08:16Z INFO opentitanlib::transport::verilator::stdout] ================================= [2022-06-09T08:08:16Z INFO opentitanlib::transport::verilator::stdout] diff --git a/doc/security/specs/secure_boot/README.md b/doc/security/specs/secure_boot/README.md index d2cae046d14ada..2734afd32017b6 100644 --- a/doc/security/specs/secure_boot/README.md +++ b/doc/security/specs/secure_boot/README.md @@ -200,7 +200,6 @@ However, these are the requirements that the manifest format is required to supp [key-manager]: ../../../../hw/ip/keymgr/README.md [manifest-format]: ../../../../sw/device/silicon_creator/rom_ext/doc/manifest.md [rom-epmp]: ../../../../sw/device/silicon_creator/rom/doc/memory_protection.md -[otp-mmap]: ../../../../hw/ip/otp_ctrl/README.md#direct-access-memory-map [ot-flash]: # [ot-unlock-flow]: # [ownership-transfer]: ../ownership_transfer/README.md diff --git a/hw/bitstream/README.md b/hw/bitstream/README.md index 40452865e50313..45058b1168e696 100644 --- a/hw/bitstream/README.md +++ b/hw/bitstream/README.md @@ -35,7 +35,7 @@ opentitan_test( name = "individualize_sw_cfg_functest", srcs = ["individualize_sw_cfg_functest.c"], fpga = fpga_params( - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_test_unlocked0_manuf_initialized", + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_test_unlocked0_manuf_initialized", tags = ["manuf"], ), exec_env = { @@ -98,4 +98,4 @@ opentitan_test( ## Read More -* [OTP Build and Test Infrastructure](../ip/otp_ctrl/data/README.md) +* [OTP Preload Image Generator](../../util/design/README.md#otp_preload_image_generator) diff --git a/hw/bitstream/vivado/BUILD b/hw/bitstream/vivado/BUILD index bfe1e6e804af65..985b5dccae1c90 100644 --- a/hw/bitstream/vivado/BUILD +++ b/hw/bitstream/vivado/BUILD @@ -24,7 +24,7 @@ _CW310_TESTROM = "//sw/device/lib/testing/test_rom:test_rom_fpga_cw310_scr_vmem" _CW340_TESTROM = _CW310_TESTROM -_OTP_RMA = "//hw/ip/otp_ctrl/data:img_rma" +_OTP_RMA = "//hw/top_earlgrey/data/otp:img_rma" _CW310_TESTROM_PATH = "{}/$(location {})".format(_PREFIX, _CW310_TESTROM) diff --git a/hw/dv/sv/mem_bkdr_util/mem_bkdr_util.core b/hw/dv/sv/mem_bkdr_util/mem_bkdr_util.core index 63e06e97d6c3cb..553054a05aabcc 100644 --- a/hw/dv/sv/mem_bkdr_util/mem_bkdr_util.core +++ b/hw/dv/sv/mem_bkdr_util/mem_bkdr_util.core @@ -15,6 +15,7 @@ filesets: - lowrisc:prim:cipher_pkg:0.1 - lowrisc:prim:secded:0.1 - lowrisc:ip:otp_ctrl_pkg:1.0 + - lowrisc:ip_interfaces:otp_ctrl_top_specific_pkg - lowrisc:ip_interfaces:flash_ctrl_pkg - lowrisc:dv:digestpp_dpi - lowrisc:ip:kmac_pkg diff --git a/hw/dv/sv/mem_bkdr_util/otp_scrambler_pkg.sv b/hw/dv/sv/mem_bkdr_util/otp_scrambler_pkg.sv index 74a8cf6ced8619..91812a0f0e23d9 100644 --- a/hw/dv/sv/mem_bkdr_util/otp_scrambler_pkg.sv +++ b/hw/dv/sv/mem_bkdr_util/otp_scrambler_pkg.sv @@ -9,7 +9,6 @@ package otp_scrambler_pkg; import uvm_pkg::*; - import otp_ctrl_reg_pkg::*; import otp_ctrl_part_pkg::*; import bus_params_pkg::*; diff --git a/hw/dv/tools/dvsim/sim.mk b/hw/dv/tools/dvsim/sim.mk index 60ad58967d7610..3c730c98b9c45b 100644 --- a/hw/dv/tools/dvsim/sim.mk +++ b/hw/dv/tools/dvsim/sim.mk @@ -101,13 +101,13 @@ ifneq (${sw_images},) echo "Building SW image \"$${bazel_label}\"."; \ bazel_airgapped_opts=""; \ bazel_opts="${sw_build_opts} --define DISABLE_VERILATOR_BUILD=true"; \ - bazel_opts+=" --//hw/ip/otp_ctrl/data:img_seed=${seed}"; \ + bazel_opts+=" --//util/design/data:img_seed=${seed}"; \ if [[ "${build_seed}" != "None" ]]; then \ - bazel_opts+=" --//hw/ip/otp_ctrl/data:lc_seed=${build_seed}"; \ - bazel_opts+=" --//hw/ip/otp_ctrl/data:otp_seed=${build_seed}"; \ + bazel_opts+=" --//util/design/data:lc_seed=${build_seed}"; \ + bazel_opts+=" --//util/design/data:otp_seed=${build_seed}"; \ fi; \ if [[ -n $${BAZEL_OTP_DATA_PERM_FLAG} ]]; then \ - bazel_opts+=" --//hw/ip/otp_ctrl/data:data_perm=$${BAZEL_OTP_DATA_PERM_FLAG}"; \ + bazel_opts+=" --//util/design/data:data_perm=$${BAZEL_OTP_DATA_PERM_FLAG}"; \ fi; \ if [[ $${OT_AIRGAPPED} != true ]]; then \ echo "Building \"$${bazel_label}\" on network connected machine."; \ @@ -165,7 +165,7 @@ ifneq (${sw_images},) `# Bazel 6 cquery outputs repository targets in canonical format (@//blabla) whereas bazel 5 does not, ` \ `# so we use a custom starlark printer to remove in leading @ when needed.` \ --starlark:expr='str(target.label)[1:] if str(target.label).startswith("@//") else target.label'); do \ - if [[ $$dep == //hw/ip/otp_ctrl/data* ]] || \ + if [[ $$dep == //hw/top_*/ip_autogen/otp_ctrl/data* ]] || \ ([[ $$dep != //hw* ]] && [[ $$dep != //util* ]] && [[ $$dep != //sw/host* ]]); then \ for artifact in $$($${bazel_cmd} cquery $${bazel_airgapped_opts} $${dep} \ --ui_event_filters=-info \ diff --git a/hw/ip/BUILD b/hw/ip/BUILD index d3d407fe1cab1c..11cb8027231ece 100644 --- a/hw/ip/BUILD +++ b/hw/ip/BUILD @@ -20,7 +20,6 @@ filegroup( "//hw/ip/kmac:all_files", "//hw/ip/lc_ctrl:all_files", "//hw/ip/otbn:all_files", - "//hw/ip/otp_ctrl:all_files", "//hw/ip/pattgen:all_files", "//hw/ip/prim:all_files", "//hw/ip/prim_generic:all_files", diff --git a/hw/ip/README.md b/hw/ip/README.md index 44bed20ee58bab..05e3a671b99b2f 100644 --- a/hw/ip/README.md +++ b/hw/ip/README.md @@ -21,7 +21,6 @@ | [`lc_ctrl`] | Manages device life cycle states and transitions, and controls key manager, flash, OTP, and debug access | | [`mbx`] | DOE mailbox for use as an integrated OpenTitan communication channel. | | [`otbn`] | Programmable coprocessor for asymmetric cryptography with SCA and FI countermeasures | -| [`otp_ctrl`] | Interfaces integrated one-time programmable memory, supports scrambling, integrity and secure wipe | | [`pattgen`] | Transmission of short time-dependent data patterns on two clock-parallel output channels | | [`pwm`] | Transmission of pulse-width modulated output signals with adjustable duty cycle | | [`rom_ctrl`] | Interfaces scrambled boot ROM with system bus and KMAC for initial health check after reset | @@ -56,7 +55,6 @@ [`lc_ctrl`]: ./lc_ctrl/README.md [`mbx`]: ./mbx/README.md [`otbn`]: ./otbn/README.md -[`otp_ctrl`]: ./otp_ctrl/README.md [`pattgen`]: ./pattgen/README.md [`pwm`]: ./pwm/README.md [`rom_ctrl`]: ./rom_ctrl/README.md diff --git a/hw/ip/lc_ctrl/doc/interfaces.md b/hw/ip/lc_ctrl/doc/interfaces.md index b1b49b9dd7de90..5795705b6b4823 100644 --- a/hw/ip/lc_ctrl/doc/interfaces.md +++ b/hw/ip/lc_ctrl/doc/interfaces.md @@ -153,7 +153,7 @@ See also [power manager documentation](../../../top_earlgrey/ip_autogen/pwrmgr/R #### OTP Interfaces -All interfaces to and from OTP are explained in detail in the [OTP Specification Document](../../otp_ctrl/README.md#life-cycle-interfaces). +All interfaces to and from OTP are explained in detail in the OTP Specification Document (for example, see [earlgrey's OTP specification](../../../top_earlgrey/ip_autogen/otp_ctrl/README.md#life-cycle-interfaces)). #### KMAC Interface diff --git a/hw/ip/lc_ctrl/doc/theory_of_operation.md b/hw/ip/lc_ctrl/doc/theory_of_operation.md index 10e2589b26f99b..7968e0e14684f0 100644 --- a/hw/ip/lc_ctrl/doc/theory_of_operation.md +++ b/hw/ip/lc_ctrl/doc/theory_of_operation.md @@ -5,7 +5,7 @@ It begins with life cycle sensing at power up, progresses through how life cycle ## Power Up Sequence -Upon power up, the life cycle controller will default to "RAW" state and wait for the OTP controller to initialize and sense the contents of the [life cycle partition](../../otp_ctrl/README.md#logical-partitions). +Upon power up, the life cycle controller will default to "RAW" state and wait for the OTP controller to initialize and sense the contents of the life cycle partition (for example, see earlgrey's [life cycle partition doc](../../../top_earlgrey/ip_autogen/otp_ctrl/README.md#logical-partitions)). When the OTP is ready, the life cycle controller reads the contents of the life cycle partition, decodes the life cycle state and updates its internal state to match. This implies that unlike the life cycle definition diagram, there is a one-time "RAW to any state" logical transition that is implicit within the implementation. Note during OTP sensing, the life cycle controller does not perform any redundant checks upon the value it reads; instead that responsibility is allocated to the OTP controller. @@ -38,7 +38,7 @@ For conditional transitions, such as those that require a token (RAW_UNLOCK, TES Note that conditional transitions will only be allowed if the OTP partition holding the corresponding token has been provisioned and locked. Since unlock tokens are considered secret, they are not stored in their raw form. -Instead, the tokens are wrapped and unwrapped based on a global constant using a [PRESENT-based scrambling mechanism](../../otp_ctrl/README.md#secret-vs-non-secret-partitions). +Instead, the tokens are wrapped and unwrapped based on a global constant using a PRESENT-based scrambling mechanism (for example, see earlgrey's [PRESENT-based scrambling mechanism](../../../top_earlgrey/ip_autogen/otp_ctrl/README.md#secret-vs-non-secret-partitions)). This ensures that a breach of fuse physical security does not automatically expose all the relevant information without also breaking the constant key. RAW_UNLOCK is not exposed in the open source design, rather it is something provisioned by the silicon creators prior to tapeout. @@ -168,7 +168,8 @@ This signal is also unconditionally asserted in all INVALID and SCRAP states (in #### CHECK_BYP_EN -The CHECK_BYP_EN signal is used to disable the [background consistency checks](../../otp_ctrl/README.md#partition-checks) of the life cycle OTP partition during life cycle transitions to prevent spurious consistency check failures (the OTP contents and the buffer registers can get out of sync during state transitions). +The CHECK_BYP_EN signal is used to disable the background consistency checks of the life cycle OTP partition during life cycle transitions to prevent spurious consistency check failures (the OTP contents and the buffer registers can get out of sync during state transitions). +For example, see [earlgrey's background consistency checks](../../../top_earlgrey/ip_autogen/otp_ctrl/README.md#partition-checks). The CHECK_BYP_EN signal is only asserted when a transition command is issued. #### CLK_BYP_REQ @@ -226,7 +227,8 @@ The TOKENs and KEYS are considered secret data and are stored in [wrapped format Before use, the secrets are unwrapped. The SECRET0_DIGEST and SECRET2_DIGEST are the digest values computed over the secret partitions in OTP holding the tokens and root keys. -As described in more detail in the [OTP controller specification](../../otp_ctrl/README.md#direct-access-memory-map), these digests have a non-zero value once the partition has been provisioned and write/read access has been locked. +As described in more detail in the OTP controller specification, these digests have a non-zero value once the partition has been provisioned and write/read access has been locked. +For example, see earlgey's [OTP controller specification](../../../top_earlgrey/ip_autogen/otp_ctrl/README.md#direct-access-memory-map). ### ID State of the Device @@ -239,7 +241,7 @@ If SECRET2_DIGEST has a nonzero value, the CREATOR_SEED_SW_RW_EN signal will be ### Secret Collateral -Among the OTP life cycle collateral, the following are considered secrets (note there may be other secrets unrelated to life cycle, please see [OTP controller specification](../../otp_ctrl/README.md#partition-listing-and-description) for more details): +Among the OTP life cycle collateral, the following are considered secrets (note there may be other secrets unrelated to life cycle, for example please see earlgrey's [OTP controller specification](../../../top_earlgrey/ip_autogen/otp_ctrl/README.md#partition-listing-and-description) for more details): - *_TOKEN - CREATOR_ROOT_KEY* @@ -253,8 +255,7 @@ Thus the system cannot be abused to generate a large number of traces for inform Note also, a global key is used here because there is no other non-volatile location to store a secret key. If PUFs were available (either in memory form or fused form), it could become an appealing alternative to hold a device unique fuse key. - -See the [OTP controller](../../otp_ctrl/README.md#secret-vs-non-secret-partitions) for more details. +For example, see earlgrey's [OTP controller](../../../top_earlgrey/ip_autogen/otp_ctrl/README.md#secret-vs-non-secret-partitions) for more details. ### OTP Accessibility Summary and Impact of Life Cycle Signals @@ -369,8 +370,9 @@ However a decoded version of the manufacturing life cycle is exposed in the [`LC In order to guard against glitch attacks during OTP sense and readout, the OTP controller makes sure to read out the life cycle partition before releasing the state to the life cycle controller. I.e., the OTP controller senses and buffers the life cycle in registers in a first readout pass. -Then, as part of the [consistency check mechanism](../../otp_ctrl/README.md#storage-consistency), the OTP controller performs a second and third readout pass to verify whether the buffered life cycle state indeed corresponds to the values stored in OTP. +Then, as part of the consistency check mechanism, the OTP controller performs a second and third readout pass to verify whether the buffered life cycle state indeed corresponds to the values stored in OTP. The second readout pass uses a linearly increasing address sequence, whereas the third readout pass uses a linearly decreasing address sequence (i.e., reads in reverse order). +For example, see earlgrey's [consistency check mechanism](../../../top_earlgrey/ip_autogen/otp_ctrl/README.md#storage-consistency). ### Transition Counter Encoding diff --git a/hw/ip/lc_ctrl/dv/env/lc_ctrl_if.sv b/hw/ip/lc_ctrl/dv/env/lc_ctrl_if.sv index a2b9f64f5f2fcc..32b0da816c51a6 100644 --- a/hw/ip/lc_ctrl/dv/env/lc_ctrl_if.sv +++ b/hw/ip/lc_ctrl/dv/env/lc_ctrl_if.sv @@ -20,7 +20,6 @@ interface lc_ctrl_if #( import lc_ctrl_reg_pkg::*; import lc_ctrl_state_pkg::*; import otp_ctrl_pkg::*; - import otp_ctrl_part_pkg::*; `include "uvm_macros.svh" diff --git a/hw/ip/otbn/doc/theory_of_operation.md b/hw/ip/otbn/doc/theory_of_operation.md index 46051fa4e0ae0c..d9b5e447909c16 100644 --- a/hw/ip/otbn/doc/theory_of_operation.md +++ b/hw/ip/otbn/doc/theory_of_operation.md @@ -412,9 +412,9 @@ this code is not re-computed within the memory block. Before being stored in SRAM, the data word with the attached Integrity Protection Code, as well as the address are scrambled according to the [memory scrambling algorithm](#memory-scrambling). The scrambling is reversed on a read. -The ephemeral memory scrambling key and the nonce are provided by the [OTP block](../../otp_ctrl/README.md). +The ephemeral memory scrambling key and the nonce are provided by the OTP block. They are set once when OTBN block is reset, and changed whenever a [secure wipe](#data-memory-dmem-secure-wipe) of the data memory is performed. - +For example, see earlgrey's [OTP block](../../../top_earlgrey/ip_autogen/otp_ctrl/doc/theory_of_operation.md#scrambling-key-derivation). The Integrity Protection Code is checked on every memory read, even though the code remains attached to the data. A further check must be performed when the data is consumed. @@ -428,7 +428,7 @@ this code is not re-computed within the memory block. Before being stored in SRAM, the instruction word with the attached Integrity Protection Code, as well as the address are scrambled according to the [memory scrambling algorithm](#memory-scrambling). The scrambling is reversed on a read. -The ephemeral memory scrambling key and the nonce are provided by the [OTP block](../../otp_ctrl/README.md). +The ephemeral memory scrambling key and the nonce are provided by the OTP block (for example, see earlgrey's [scrambling specification](../../../top_earlgrey/ip_autogen/otp_ctrl/doc/theory_of_operation.md#scrambling-key-derivation)). They are set once when OTBN block is reset, and changed whenever a [secure wipe](#instruction-memory-imem-secure-wipe) of the instruction memory is performed. The Integrity Protection Code is checked on every memory read, even though the code remains attached to the data. diff --git a/hw/ip/otbn/dv/README.md b/hw/ip/otbn/dv/README.md index 9ea58d986dda97..1a1f04454eb4b8 100644 --- a/hw/ip/otbn/dv/README.md +++ b/hw/ip/otbn/dv/README.md @@ -45,7 +45,7 @@ OTBN has the following interfaces: - One interrupt - An [alert interface](../../../dv/sv/alert_esc_agent/README.md) - A life cycle escalation interface -- An [OTP](../../otp_ctrl/README.md) connection +- An OTP: for example, see earlgrey's [OTP](../../../top_earlgrey/ip_autogen/otp_ctrl/README.md) connection - Two [EDN](../../edn/README.md) connections - A RAM configuration interface, which is passed through to the SRAM macros diff --git a/hw/ip/otp_ctrl/data/README.md b/hw/ip/otp_ctrl/data/README.md deleted file mode 100644 index 2e4b56925e93b9..00000000000000 --- a/hw/ip/otp_ctrl/data/README.md +++ /dev/null @@ -1,72 +0,0 @@ -# OTP Build and Test Infrastructure - -OTP image configurations are defined using hjson objects. Currently there are -two ways to build images: - -1. Pre-built OTP image overlays defined in hjson. These is the approach - currently used in most DV test cases. -2. Dynamically built OTP image overlays defined in [Bazel](#bazel). This is the - approach currently used in FPGA and Silicon Validation (SiVal) targets. - -## Bazel - -See `//hw/ip/otp_ctrl/data/BUILD` for detailed references on the rules -described below. - -### OTP HJSON Map - -OTP image overlays are first defined using the `otp_json` Bazel rule. The -following example shows the definition of a `SECRET2` partition configuration: - -```python -otp_json( - name = "otp_json_secret2_unlocked", - partitions = [ - otp_partition( - name = "SECRET2", - items = { - "RMA_TOKEN": "", - "CREATOR_ROOT_KEY_SHARE0": "", - "CREATOR_ROOT_KEY_SHARE1": "", - }, - lock = False, - ), - ], -) -``` - -See `//rules/otp.bzl` for additional documentation on additional parameters -available in the `otp_json` rule. - -### OTP Image - -An OTP image is a collection of OTP JSON files used to create an OTP image. -An OTP can have multiple `otp_json` dependencies. Each dependency has the -ability of override the values of the previous dependency, so the order in -which these are listed is important. - -```python -# Represents a device in DEV state with the SECRET0 and SECRET1 partitions in -# locked state. SECRET2 partition is unlocked. -otp_image( - name = "img_dev_individualized", - src = ":otp_json_dev", - overlays = [ - ":otp_json_secret0", - ":otp_json_secret1", - ] + STD_OTP_OVERLAYS_WITHOUT_SECRET_PARTITIONS, -) -``` - -In this example, the `src` attribute points to the baseline OTP JSON -configuration, and the list of overlay dependencies are applied in order -of precedence in the `overlays` attribute. - -The `STD_OTP_OVERLAYS_WITHOUT_SECRET_PARTITIONS` definition imported from -`//rules:otp.bzl` declares a list of `otp_json` targets that are used -as overlays. There are other list of predefined overlays that are used -throughout the code base. - -### FPGA Integration - -See [FPGA bitstreams](../../../bitstream/README.md) documentation for more details. diff --git a/hw/ip/otp_ctrl/doc/checklist.md b/hw/ip/otp_ctrl/doc/checklist.md deleted file mode 100644 index 1fab63e8fa9b44..00000000000000 --- a/hw/ip/otp_ctrl/doc/checklist.md +++ /dev/null @@ -1,271 +0,0 @@ -# OTP_CTRL Checklist - - -This checklist is for [Hardware Stage](../../../../doc/project_governance/development_stages.md) transitions for the [OTP_CTRL peripheral.](../README.md) -All checklist items refer to the content in the [Checklist.](../../../../doc/project_governance/checklist/README.md) - -## Design Checklist - -### D1 - -Type | Item | Resolution | Note/Collaterals ---------------|--------------------------------|-------------|------------------ -Documentation | [SPEC_COMPLETE][] | Done | [OTP_CTRL Design Spec](../README.md) -Documentation | [CSR_DEFINED][] | Done | -RTL | [CLKRST_CONNECTED][] | Done | -RTL | [IP_TOP][] | Done | -RTL | [IP_INSTANTIABLE][] | Done | -RTL | [PHYSICAL_MACROS_DEFINED_80][] | Done | -RTL | [FUNC_IMPLEMENTED][] | Done | -RTL | [ASSERT_KNOWN_ADDED][] | Done | -Code Quality | [LINT_SETUP][] | Done | - -[SPEC_COMPLETE]: ../../../../doc/project_governance/checklist/README.md#spec_complete -[CSR_DEFINED]: ../../../../doc/project_governance/checklist/README.md#csr_defined -[CLKRST_CONNECTED]: ../../../../doc/project_governance/checklist/README.md#clkrst_connected -[IP_TOP]: ../../../../doc/project_governance/checklist/README.md#ip_top -[IP_INSTANTIABLE]: ../../../../doc/project_governance/checklist/README.md#ip_instantiable -[PHYSICAL_MACROS_DEFINED_80]: ../../../../doc/project_governance/checklist/README.md#mem_instanced_80 -[FUNC_IMPLEMENTED]: ../../../../doc/project_governance/checklist/README.md#func_implemented -[ASSERT_KNOWN_ADDED]: ../../../../doc/project_governance/checklist/README.md#assert_known_added -[LINT_SETUP]: ../../../../doc/project_governance/checklist/README.md#lint_setup - -### D2 - -Type | Item | Resolution | Note/Collaterals ---------------|---------------------------|-------------|------------------ -Documentation | [NEW_FEATURES][] | Done | -Documentation | [BLOCK_DIAGRAM][] | Done | -Documentation | [DOC_INTERFACE][] | Done | -Documentation | [DOC_INTEGRATION_GUIDE][] | Waived | This checklist item has been added retrospectively. -Documentation | [MISSING_FUNC][] | Done | -Documentation | [FEATURE_FROZEN][] | Done | -RTL | [FEATURE_COMPLETE][] | Done | -RTL | [PORT_FROZEN][] | Done | -RTL | [ARCHITECTURE_FROZEN][] | Done | -RTL | [REVIEW_TODO][] | Done | -RTL | [STYLE_X][] | Done | -RTL | [CDC_SYNCMACRO][] | N/A | -Code Quality | [LINT_PASS][] | Done | -Code Quality | [CDC_SETUP][] | Waived | No block-level flow available - waived to top-level signoff. -Code Quality | [RDC_SETUP][] | Waived | No block-level flow available - waived to top-level signoff. -Code Quality | [AREA_CHECK][] | Done | -Code Quality | [TIMING_CHECK][] | Done | -Security | [SEC_CM_DOCUMENTED][] | Done | - -[NEW_FEATURES]: ../../../../doc/project_governance/checklist/README.md#new_features -[BLOCK_DIAGRAM]: ../../../../doc/project_governance/checklist/README.md#block_diagram -[DOC_INTERFACE]: ../../../../doc/project_governance/checklist/README.md#doc_interface -[DOC_INTEGRATION_GUIDE]: ../../../../doc/project_governance/checklist/README.md#doc_integration_guide -[MISSING_FUNC]: ../../../../doc/project_governance/checklist/README.md#missing_func -[FEATURE_FROZEN]: ../../../../doc/project_governance/checklist/README.md#feature_frozen -[FEATURE_COMPLETE]: ../../../../doc/project_governance/checklist/README.md#feature_complete -[PORT_FROZEN]: ../../../../doc/project_governance/checklist/README.md#port_frozen -[ARCHITECTURE_FROZEN]: ../../../../doc/project_governance/checklist/README.md#architecture_frozen -[REVIEW_TODO]: ../../../../doc/project_governance/checklist/README.md#review_todo -[STYLE_X]: ../../../../doc/project_governance/checklist/README.md#style_x -[CDC_SYNCMACRO]: ../../../../doc/project_governance/checklist/README.md#cdc_syncmacro -[LINT_PASS]: ../../../../doc/project_governance/checklist/README.md#lint_pass -[CDC_SETUP]: ../../../../doc/project_governance/checklist/README.md#cdc_setup -[RDC_SETUP]: ../../../../doc/project_governance/checklist/README.md#rdc_setup -[AREA_CHECK]: ../../../../doc/project_governance/checklist/README.md#area_check -[TIMING_CHECK]: ../../../../doc/project_governance/checklist/README.md#timing_check -[SEC_CM_DOCUMENTED]: ../../../../doc/project_governance/checklist/README.md#sec_cm_documented - -### D2S - - Type | Item | Resolution | Note/Collaterals ---------------|------------------------------|-------------|------------------ -Security | [SEC_CM_ASSETS_LISTED][] | Done | -Security | [SEC_CM_IMPLEMENTED][] | Done | -Security | [SEC_CM_RND_CNST][] | Done | -Security | [SEC_CM_NON_RESET_FLOPS][] | Done | -Security | [SEC_CM_SHADOW_REGS][] | Done | -Security | [SEC_CM_RTL_REVIEWED][] | Done | -Security | [SEC_CM_COUNCIL_REVIEWED][] | Done | - -[SEC_CM_ASSETS_LISTED]: ../../../../doc/project_governance/checklist/README.md#sec_cm_assets_listed -[SEC_CM_IMPLEMENTED]: ../../../../doc/project_governance/checklist/README.md#sec_cm_implemented -[SEC_CM_RND_CNST]: ../../../../doc/project_governance/checklist/README.md#sec_cm_rnd_cnst -[SEC_CM_NON_RESET_FLOPS]: ../../../../doc/project_governance/checklist/README.md#sec_cm_non_reset_flops -[SEC_CM_SHADOW_REGS]: ../../../../doc/project_governance/checklist/README.md#sec_cm_shadow_regs -[SEC_CM_RTL_REVIEWED]: ../../../../doc/project_governance/checklist/README.md#sec_cm_rtl_reviewed -[SEC_CM_COUNCIL_REVIEWED]: ../../../../doc/project_governance/checklist/README.md#sec_cm_council_reviewed - -### D3 - - Type | Item | Resolution | Note/Collaterals ---------------|-------------------------|-------------|------------------ -Documentation | [NEW_FEATURES_D3][] | Done | -RTL | [TODO_COMPLETE][] | Done | -Code Quality | [LINT_COMPLETE][] | Done | Waiver files approved by TC on 2024-08-08. -Code Quality | [CDC_COMPLETE][] | Waived | No block-level flow available - waived to top-level signoff. -Code Quality | [RDC_COMPLETE][] | Waived | No block-level flow available - waived to top-level signoff. -Review | [REVIEW_RTL][] | Done | -Review | [REVIEW_DELETED_FF][] | Waived | No block-level flow available - waived to top-level signoff. -Review | [REVIEW_SW_CHANGE][] | Done | -Review | [REVIEW_SW_ERRATA][] | Done | -Review | Reviewer(s) | Done | adk@ vogelpi@ -Review | Signoff date | Done | 2024-08-08 - -[NEW_FEATURES_D3]: ../../../../doc/project_governance/checklist/README.md#new_features_d3 -[TODO_COMPLETE]: ../../../../doc/project_governance/checklist/README.md#todo_complete -[LINT_COMPLETE]: ../../../../doc/project_governance/checklist/README.md#lint_complete -[CDC_COMPLETE]: ../../../../doc/project_governance/checklist/README.md#cdc_complete -[RDC_COMPLETE]: ../../../../doc/project_governance/checklist/README.md#rdc_complete -[REVIEW_RTL]: ../../../../doc/project_governance/checklist/README.md#review_rtl -[REVIEW_DELETED_FF]: ../../../../doc/project_governance/checklist/README.md#review_deleted_ff -[REVIEW_SW_CHANGE]: ../../../../doc/project_governance/checklist/README.md#review_sw_change -[REVIEW_SW_ERRATA]: ../../../../doc/project_governance/checklist/README.md#review_sw_errata - -## Verification Checklist - -### V1 - - Type | Item | Resolution | Note/Collaterals ---------------|---------------------------------------|-------------|------------------ -Documentation | [DV_DOC_DRAFT_COMPLETED][] | Done | [OTP_CTRL DV document](../dv/README.md) -Documentation | [TESTPLAN_COMPLETED][] | Done | [OTP_CTRL Testplan](../dv/README.md#testplan) -Testbench | [TB_TOP_CREATED][] | Done | -Testbench | [PRELIMINARY_ASSERTION_CHECKS_ADDED][]| Done | -Testbench | [SIM_TB_ENV_CREATED][] | Done | -Testbench | [SIM_RAL_MODEL_GEN_AUTOMATED][] | Done | -Testbench | [CSR_CHECK_GEN_AUTOMATED][] | Done | -Testbench | [TB_GEN_AUTOMATED][] | Done | -Tests | [SIM_SMOKE_TEST_PASSING][] | Done | -Tests | [SIM_CSR_MEM_TEST_SUITE_PASSING][] | Done | -Tests | [FPV_MAIN_ASSERTIONS_PROVEN][] | N/A | -Tool Setup | [SIM_ALT_TOOL_SETUP][] | Done | -Regression | [SIM_SMOKE_REGRESSION_SETUP][] | Done | -Regression | [SIM_NIGHTLY_REGRESSION_SETUP][] | Done | -Regression | [FPV_REGRESSION_SETUP][] | N/A | -Coverage | [SIM_COVERAGE_MODEL_ADDED][] | Done | -Code Quality | [TB_LINT_SETUP][] | Done | -Integration | [PRE_VERIFIED_SUB_MODULES_V1][] | N/A | Exception for IP modules -Review | [DESIGN_SPEC_REVIEWED][] | Done | -Review | [TESTPLAN_REVIEWED][] | Done | -Review | [STD_TEST_CATEGORIES_PLANNED][] | Done | Exception (Security, Power, Debug) -Review | [V2_CHECKLIST_SCOPED][] | Done | - -[DV_DOC_DRAFT_COMPLETED]: ../../../../doc/project_governance/checklist/README.md#dv_doc_draft_completed -[TESTPLAN_COMPLETED]: ../../../../doc/project_governance/checklist/README.md#testplan_completed -[TB_TOP_CREATED]: ../../../../doc/project_governance/checklist/README.md#tb_top_created -[PRELIMINARY_ASSERTION_CHECKS_ADDED]: ../../../../doc/project_governance/checklist/README.md#preliminary_assertion_checks_added -[SIM_TB_ENV_CREATED]: ../../../../doc/project_governance/checklist/README.md#sim_tb_env_created -[SIM_RAL_MODEL_GEN_AUTOMATED]: ../../../../doc/project_governance/checklist/README.md#sim_ral_model_gen_automated -[CSR_CHECK_GEN_AUTOMATED]: ../../../../doc/project_governance/checklist/README.md#csr_check_gen_automated -[TB_GEN_AUTOMATED]: ../../../../doc/project_governance/checklist/README.md#tb_gen_automated -[SIM_SMOKE_TEST_PASSING]: ../../../../doc/project_governance/checklist/README.md#sim_smoke_test_passing -[SIM_CSR_MEM_TEST_SUITE_PASSING]: ../../../../doc/project_governance/checklist/README.md#sim_csr_mem_test_suite_passing -[FPV_MAIN_ASSERTIONS_PROVEN]: ../../../../doc/project_governance/checklist/README.md#fpv_main_assertions_proven -[SIM_ALT_TOOL_SETUP]: ../../../../doc/project_governance/checklist/README.md#sim_alt_tool_setup -[SIM_SMOKE_REGRESSION_SETUP]: ../../../../doc/project_governance/checklist/README.md#sim_smoke_regression_setup -[SIM_NIGHTLY_REGRESSION_SETUP]: ../../../../doc/project_governance/checklist/README.md#sim_nightly_regression_setup -[FPV_REGRESSION_SETUP]: ../../../../doc/project_governance/checklist/README.md#fpv_regression_setup -[SIM_COVERAGE_MODEL_ADDED]: ../../../../doc/project_governance/checklist/README.md#sim_coverage_model_added -[PRE_VERIFIED_SUB_MODULES_V1]: ../../../../doc/project_governance/checklist/README.md#pre_verified_sub_modules_v1 -[TB_LINT_SETUP]: ../../../../doc/project_governance/checklist/README.md#tb_lint_setup -[DESIGN_SPEC_REVIEWED]: ../../../../doc/project_governance/checklist/README.md#design_spec_reviewed -[TESTPLAN_REVIEWED]: ../../../../doc/project_governance/checklist/README.md#testplan_reviewed -[STD_TEST_CATEGORIES_PLANNED]: ../../../../doc/project_governance/checklist/README.md#std_test_categories_planned -[V2_CHECKLIST_SCOPED]: ../../../../doc/project_governance/checklist/README.md#v2_checklist_scoped - -### V2 - - Type | Item | Resolution | Note/Collaterals ---------------|-----------------------------------------|-------------|------------------ -Documentation | [DESIGN_DELTAS_CAPTURED_V2][] | Done | -Documentation | [DV_DOC_COMPLETED][] | Done | -Testbench | [FUNCTIONAL_COVERAGE_IMPLEMENTED][] | Done | -Testbench | [ALL_INTERFACES_EXERCISED][] | Done | `prim_tl_o/i` has a simple prim_tl_agent support, but need to be replaced with auto-generated tl_agent once reggen tool is optimized. -Testbench | [ALL_ASSERTION_CHECKS_ADDED][] | Done | -Testbench | [SIM_TB_ENV_COMPLETED][] | Done | -Tests | [SIM_ALL_TESTS_PASSING][] | Done | -Tests | [FPV_ALL_ASSERTIONS_WRITTEN][] | N/A | -Tests | [FPV_ALL_ASSUMPTIONS_REVIEWED][] | N/A | -Tests | [SIM_FW_SIMULATED][] | N/A | -Regression | [SIM_NIGHTLY_REGRESSION_V2][] | Done | -Coverage | [SIM_CODE_COVERAGE_V2][] | Done | -Coverage | [SIM_FUNCTIONAL_COVERAGE_V2][] | Done | -Coverage | [FPV_CODE_COVERAGE_V2][] | N/A | -Coverage | [FPV_COI_COVERAGE_V2][] | N/A | -Integration | [PRE_VERIFIED_SUB_MODULES_V2][] | Done | Waived `prim_alert_sender` and `prim_lfsr` -Issues | [NO_HIGH_PRIORITY_ISSUES_PENDING][] | Done | -Issues | [ALL_LOW_PRIORITY_ISSUES_ROOT_CAUSED][] | Done | -Review | [DV_DOC_TESTPLAN_REVIEWED][] | Done | Reviewed on 05/24/2021 -Review | [V3_CHECKLIST_SCOPED][] | Done | - -[DESIGN_DELTAS_CAPTURED_V2]: ../../../../doc/project_governance/checklist/README.md#design_deltas_captured_v2 -[DV_DOC_COMPLETED]: ../../../../doc/project_governance/checklist/README.md#dv_doc_completed -[FUNCTIONAL_COVERAGE_IMPLEMENTED]: ../../../../doc/project_governance/checklist/README.md#functional_coverage_implemented -[ALL_INTERFACES_EXERCISED]: ../../../../doc/project_governance/checklist/README.md#all_interfaces_exercised -[ALL_ASSERTION_CHECKS_ADDED]: ../../../../doc/project_governance/checklist/README.md#all_assertion_checks_added -[SIM_TB_ENV_COMPLETED]: ../../../../doc/project_governance/checklist/README.md#sim_tb_env_completed -[SIM_ALL_TESTS_PASSING]: ../../../../doc/project_governance/checklist/README.md#sim_all_tests_passing -[FPV_ALL_ASSERTIONS_WRITTEN]: ../../../../doc/project_governance/checklist/README.md#fpv_all_assertions_written -[FPV_ALL_ASSUMPTIONS_REVIEWED]: ../../../../doc/project_governance/checklist/README.md#fpv_all_assumptions_reviewed -[SIM_FW_SIMULATED]: ../../../../doc/project_governance/checklist/README.md#sim_fw_simulated -[SIM_NIGHTLY_REGRESSION_V2]: ../../../../doc/project_governance/checklist/README.md#sim_nightly_regression_v2 -[SIM_CODE_COVERAGE_V2]: ../../../../doc/project_governance/checklist/README.md#sim_code_coverage_v2 -[SIM_FUNCTIONAL_COVERAGE_V2]: ../../../../doc/project_governance/checklist/README.md#sim_functional_coverage_v2 -[FPV_CODE_COVERAGE_V2]: ../../../../doc/project_governance/checklist/README.md#fpv_code_coverage_v2 -[FPV_COI_COVERAGE_V2]: ../../../../doc/project_governance/checklist/README.md#fpv_coi_coverage_v2 -[PRE_VERIFIED_SUB_MODULES_V2]: ../../../../doc/project_governance/checklist/README.md#pre_verified_sub_modules_v2 -[NO_HIGH_PRIORITY_ISSUES_PENDING]: ../../../../doc/project_governance/checklist/README.md#no_high_priority_issues_pending -[ALL_LOW_PRIORITY_ISSUES_ROOT_CAUSED]:../../../../doc/project_governance/checklist/README.md#all_low_priority_issues_root_caused -[DV_DOC_TESTPLAN_REVIEWED]: ../../../../doc/project_governance/checklist/README.md#dv_doc_testplan_reviewed -[V3_CHECKLIST_SCOPED]: ../../../../doc/project_governance/checklist/README.md#v3_checklist_scoped - -### V2S - - Type | Item | Resolution | Note/Collaterals ---------------|-----------------------------------------|-------------|------------------ -Documentation | [SEC_CM_TESTPLAN_COMPLETED][] | Done | -Tests | [FPV_SEC_CM_VERIFIED][] | Done | -Tests | [SIM_SEC_CM_VERIFIED][] | Done | -Coverage | [SIM_COVERAGE_REVIEWED][] | Done | -Review | [SEC_CM_DV_REVIEWED][] | Done | - -[SEC_CM_TESTPLAN_COMPLETED]: ../../../../doc/project_governance/checklist/README.md#sec_cm_testplan_completed -[FPV_SEC_CM_VERIFIED]: ../../../../doc/project_governance/checklist/README.md#fpv_sec_cm_verified -[SIM_SEC_CM_VERIFIED]: ../../../../doc/project_governance/checklist/README.md#sim_sec_cm_verified -[SIM_COVERAGE_REVIEWED]: ../../../../doc/project_governance/checklist/README.md#sim_coverage_reviewed -[SEC_CM_DV_REVIEWED]: ../../../../doc/project_governance/checklist/README.md#sec_cm_dv_reviewed - -### V3 - - Type | Item | Resolution | Note/Collaterals ---------------|-----------------------------------|-------------|------------------ -Documentation | [DESIGN_DELTAS_CAPTURED_V3][] | Not Started | -Tests | [X_PROP_ANALYSIS_COMPLETED][] | Not Started | -Tests | [FPV_ASSERTIONS_PROVEN_AT_V3][] | Not Started | -Regression | [SIM_NIGHTLY_REGRESSION_AT_V3][] | Not Started | -Coverage | [SIM_CODE_COVERAGE_AT_100][] | Not Started | -Coverage | [SIM_FUNCTIONAL_COVERAGE_AT_100][]| Not Started | -Coverage | [FPV_CODE_COVERAGE_AT_100][] | Not Started | -Coverage | [FPV_COI_COVERAGE_AT_100][] | Not Started | -Code Quality | [ALL_TODOS_RESOLVED][] | Not Started | -Code Quality | [NO_TOOL_WARNINGS_THROWN][] | Not Started | -Code Quality | [TB_LINT_COMPLETE][] | Not Started | -Integration | [PRE_VERIFIED_SUB_MODULES_V3][] | Not Started | -Issues | [NO_ISSUES_PENDING][] | Not Started | -Review | Reviewer(s) | Not Started | -Review | Signoff date | Not Started | - -[DESIGN_DELTAS_CAPTURED_V3]: ../../../../doc/project_governance/checklist/README.md#design_deltas_captured_v3 -[X_PROP_ANALYSIS_COMPLETED]: ../../../../doc/project_governance/checklist/README.md#x_prop_analysis_completed -[FPV_ASSERTIONS_PROVEN_AT_V3]: ../../../../doc/project_governance/checklist/README.md#fpv_assertions_proven_at_v3 -[SIM_NIGHTLY_REGRESSION_AT_V3]: ../../../../doc/project_governance/checklist/README.md#sim_nightly_regression_at_v3 -[SIM_CODE_COVERAGE_AT_100]: ../../../../doc/project_governance/checklist/README.md#sim_code_coverage_at_100 -[SIM_FUNCTIONAL_COVERAGE_AT_100]:../../../../doc/project_governance/checklist/README.md#sim_functional_coverage_at_100 -[FPV_CODE_COVERAGE_AT_100]: ../../../../doc/project_governance/checklist/README.md#fpv_code_coverage_at_100 -[FPV_COI_COVERAGE_AT_100]: ../../../../doc/project_governance/checklist/README.md#fpv_coi_coverage_at_100 -[ALL_TODOS_RESOLVED]: ../../../../doc/project_governance/checklist/README.md#all_todos_resolved -[NO_TOOL_WARNINGS_THROWN]: ../../../../doc/project_governance/checklist/README.md#no_tool_warnings_thrown -[TB_LINT_COMPLETE]: ../../../../doc/project_governance/checklist/README.md#tb_lint_complete -[PRE_VERIFIED_SUB_MODULES_V3]: ../../../../doc/project_governance/checklist/README.md#pre_verified_sub_modules_v3 -[NO_ISSUES_PENDING]: ../../../../doc/project_governance/checklist/README.md#no_issues_pending diff --git a/hw/ip/prim/dv/prim_secded/secded_enc.c b/hw/ip/prim/dv/prim_secded/secded_enc.c index c4e447c954da5d..d2bd15f1a7755a 100644 --- a/hw/ip/prim/dv/prim_secded/secded_enc.c +++ b/hw/ip/prim/dv/prim_secded/secded_enc.c @@ -5,6 +5,7 @@ // SECDED encode code generated by // util/design/secded_gen.py from util/design/data/secded_cfg.hjson + #include "secded_enc.h" #include @@ -26,135 +27,61 @@ static uint8_t calc_parity(uint64_t word, bool invert) { } uint8_t enc_secded_22_16(const uint8_t bytes[2]) { - uint16_t word = ((uint16_t)bytes[0] << 0) | ((uint16_t)bytes[1] << 8); - - return (calc_parity(word & 0x496e, false) << 0) | - (calc_parity(word & 0xf20b, false) << 1) | - (calc_parity(word & 0x8ed8, false) << 2) | - (calc_parity(word & 0x7714, false) << 3) | - (calc_parity(word & 0xaca5, false) << 4) | - (calc_parity(word & 0x11f3, false) << 5); +uint16_t word = ((uint16_t)bytes[0] << 0) | ((uint16_t)bytes[1] << 8); + +return (calc_parity(word & 0x496e, false) << 0) | (calc_parity(word & 0xf20b, false) << 1) | (calc_parity(word & 0x8ed8, false) << 2) | (calc_parity(word & 0x7714, false) << 3) | (calc_parity(word & 0xaca5, false) << 4) | (calc_parity(word & 0x11f3, false) << 5); } uint8_t enc_secded_28_22(const uint8_t bytes[3]) { - uint32_t word = ((uint32_t)bytes[0] << 0) | ((uint32_t)bytes[1] << 8) | - ((uint32_t)bytes[2] << 16); - - return (calc_parity(word & 0x3003ff, false) << 0) | - (calc_parity(word & 0x10fc0f, false) << 1) | - (calc_parity(word & 0x271c71, false) << 2) | - (calc_parity(word & 0x3b6592, false) << 3) | - (calc_parity(word & 0x3daaa4, false) << 4) | - (calc_parity(word & 0x3ed348, false) << 5); +uint32_t word = ((uint32_t)bytes[0] << 0) | ((uint32_t)bytes[1] << 8) | ((uint32_t)bytes[2] << 16); + +return (calc_parity(word & 0x3003ff, false) << 0) | (calc_parity(word & 0x10fc0f, false) << 1) | (calc_parity(word & 0x271c71, false) << 2) | (calc_parity(word & 0x3b6592, false) << 3) | (calc_parity(word & 0x3daaa4, false) << 4) | (calc_parity(word & 0x3ed348, false) << 5); } uint8_t enc_secded_39_32(const uint8_t bytes[4]) { - uint32_t word = ((uint32_t)bytes[0] << 0) | ((uint32_t)bytes[1] << 8) | - ((uint32_t)bytes[2] << 16) | ((uint32_t)bytes[3] << 24); - - return (calc_parity(word & 0x2606bd25, false) << 0) | - (calc_parity(word & 0xdeba8050, false) << 1) | - (calc_parity(word & 0x413d89aa, false) << 2) | - (calc_parity(word & 0x31234ed1, false) << 3) | - (calc_parity(word & 0xc2c1323b, false) << 4) | - (calc_parity(word & 0x2dcc624c, false) << 5) | - (calc_parity(word & 0x98505586, false) << 6); +uint32_t word = ((uint32_t)bytes[0] << 0) | ((uint32_t)bytes[1] << 8) | ((uint32_t)bytes[2] << 16) | ((uint32_t)bytes[3] << 24); + +return (calc_parity(word & 0x2606bd25, false) << 0) | (calc_parity(word & 0xdeba8050, false) << 1) | (calc_parity(word & 0x413d89aa, false) << 2) | (calc_parity(word & 0x31234ed1, false) << 3) | (calc_parity(word & 0xc2c1323b, false) << 4) | (calc_parity(word & 0x2dcc624c, false) << 5) | (calc_parity(word & 0x98505586, false) << 6); } uint8_t enc_secded_64_57(const uint8_t bytes[8]) { - uint64_t word = ((uint64_t)bytes[0] << 0) | ((uint64_t)bytes[1] << 8) | - ((uint64_t)bytes[2] << 16) | ((uint64_t)bytes[3] << 24) | - ((uint64_t)bytes[4] << 32) | ((uint64_t)bytes[5] << 40) | - ((uint64_t)bytes[6] << 48) | ((uint64_t)bytes[7] << 56); - - return (calc_parity(word & 0x103fff800007fff, false) << 0) | - (calc_parity(word & 0x17c1ff801ff801f, false) << 1) | - (calc_parity(word & 0x1bde1f87e0781e1, false) << 2) | - (calc_parity(word & 0x1deee3b8e388e22, false) << 3) | - (calc_parity(word & 0x1ef76cdb2c93244, false) << 4) | - (calc_parity(word & 0x1f7bb56d5525488, false) << 5) | - (calc_parity(word & 0x1fbdda769a46910, false) << 6); +uint64_t word = ((uint64_t)bytes[0] << 0) | ((uint64_t)bytes[1] << 8) | ((uint64_t)bytes[2] << 16) | ((uint64_t)bytes[3] << 24) | ((uint64_t)bytes[4] << 32) | ((uint64_t)bytes[5] << 40) | ((uint64_t)bytes[6] << 48) | ((uint64_t)bytes[7] << 56); + +return (calc_parity(word & 0x103fff800007fff, false) << 0) | (calc_parity(word & 0x17c1ff801ff801f, false) << 1) | (calc_parity(word & 0x1bde1f87e0781e1, false) << 2) | (calc_parity(word & 0x1deee3b8e388e22, false) << 3) | (calc_parity(word & 0x1ef76cdb2c93244, false) << 4) | (calc_parity(word & 0x1f7bb56d5525488, false) << 5) | (calc_parity(word & 0x1fbdda769a46910, false) << 6); } uint8_t enc_secded_72_64(const uint8_t bytes[8]) { - uint64_t word = ((uint64_t)bytes[0] << 0) | ((uint64_t)bytes[1] << 8) | - ((uint64_t)bytes[2] << 16) | ((uint64_t)bytes[3] << 24) | - ((uint64_t)bytes[4] << 32) | ((uint64_t)bytes[5] << 40) | - ((uint64_t)bytes[6] << 48) | ((uint64_t)bytes[7] << 56); - - return (calc_parity(word & 0xb9000000001fffff, false) << 0) | - (calc_parity(word & 0x5e00000fffe0003f, false) << 1) | - (calc_parity(word & 0x67003ff003e007c1, false) << 2) | - (calc_parity(word & 0xcd0fc0f03c207842, false) << 3) | - (calc_parity(word & 0xb671c711c4438884, false) << 4) | - (calc_parity(word & 0xb5b65926488c9108, false) << 5) | - (calc_parity(word & 0xcbdaaa4a91152210, false) << 6) | - (calc_parity(word & 0x7aed348d221a4420, false) << 7); +uint64_t word = ((uint64_t)bytes[0] << 0) | ((uint64_t)bytes[1] << 8) | ((uint64_t)bytes[2] << 16) | ((uint64_t)bytes[3] << 24) | ((uint64_t)bytes[4] << 32) | ((uint64_t)bytes[5] << 40) | ((uint64_t)bytes[6] << 48) | ((uint64_t)bytes[7] << 56); + +return (calc_parity(word & 0xb9000000001fffff, false) << 0) | (calc_parity(word & 0x5e00000fffe0003f, false) << 1) | (calc_parity(word & 0x67003ff003e007c1, false) << 2) | (calc_parity(word & 0xcd0fc0f03c207842, false) << 3) | (calc_parity(word & 0xb671c711c4438884, false) << 4) | (calc_parity(word & 0xb5b65926488c9108, false) << 5) | (calc_parity(word & 0xcbdaaa4a91152210, false) << 6) | (calc_parity(word & 0x7aed348d221a4420, false) << 7); } uint8_t enc_secded_inv_22_16(const uint8_t bytes[2]) { - uint16_t word = ((uint16_t)bytes[0] << 0) | ((uint16_t)bytes[1] << 8); - - return (calc_parity(word & 0x496e, false) << 0) | - (calc_parity(word & 0xf20b, true) << 1) | - (calc_parity(word & 0x8ed8, false) << 2) | - (calc_parity(word & 0x7714, true) << 3) | - (calc_parity(word & 0xaca5, false) << 4) | - (calc_parity(word & 0x11f3, true) << 5); +uint16_t word = ((uint16_t)bytes[0] << 0) | ((uint16_t)bytes[1] << 8); + +return (calc_parity(word & 0x496e, false) << 0) | (calc_parity(word & 0xf20b, true) << 1) | (calc_parity(word & 0x8ed8, false) << 2) | (calc_parity(word & 0x7714, true) << 3) | (calc_parity(word & 0xaca5, false) << 4) | (calc_parity(word & 0x11f3, true) << 5); } uint8_t enc_secded_inv_28_22(const uint8_t bytes[3]) { - uint32_t word = ((uint32_t)bytes[0] << 0) | ((uint32_t)bytes[1] << 8) | - ((uint32_t)bytes[2] << 16); - - return (calc_parity(word & 0x3003ff, false) << 0) | - (calc_parity(word & 0x10fc0f, true) << 1) | - (calc_parity(word & 0x271c71, false) << 2) | - (calc_parity(word & 0x3b6592, true) << 3) | - (calc_parity(word & 0x3daaa4, false) << 4) | - (calc_parity(word & 0x3ed348, true) << 5); +uint32_t word = ((uint32_t)bytes[0] << 0) | ((uint32_t)bytes[1] << 8) | ((uint32_t)bytes[2] << 16); + +return (calc_parity(word & 0x3003ff, false) << 0) | (calc_parity(word & 0x10fc0f, true) << 1) | (calc_parity(word & 0x271c71, false) << 2) | (calc_parity(word & 0x3b6592, true) << 3) | (calc_parity(word & 0x3daaa4, false) << 4) | (calc_parity(word & 0x3ed348, true) << 5); } uint8_t enc_secded_inv_39_32(const uint8_t bytes[4]) { - uint32_t word = ((uint32_t)bytes[0] << 0) | ((uint32_t)bytes[1] << 8) | - ((uint32_t)bytes[2] << 16) | ((uint32_t)bytes[3] << 24); - - return (calc_parity(word & 0x2606bd25, false) << 0) | - (calc_parity(word & 0xdeba8050, true) << 1) | - (calc_parity(word & 0x413d89aa, false) << 2) | - (calc_parity(word & 0x31234ed1, true) << 3) | - (calc_parity(word & 0xc2c1323b, false) << 4) | - (calc_parity(word & 0x2dcc624c, true) << 5) | - (calc_parity(word & 0x98505586, false) << 6); +uint32_t word = ((uint32_t)bytes[0] << 0) | ((uint32_t)bytes[1] << 8) | ((uint32_t)bytes[2] << 16) | ((uint32_t)bytes[3] << 24); + +return (calc_parity(word & 0x2606bd25, false) << 0) | (calc_parity(word & 0xdeba8050, true) << 1) | (calc_parity(word & 0x413d89aa, false) << 2) | (calc_parity(word & 0x31234ed1, true) << 3) | (calc_parity(word & 0xc2c1323b, false) << 4) | (calc_parity(word & 0x2dcc624c, true) << 5) | (calc_parity(word & 0x98505586, false) << 6); } uint8_t enc_secded_inv_64_57(const uint8_t bytes[8]) { - uint64_t word = ((uint64_t)bytes[0] << 0) | ((uint64_t)bytes[1] << 8) | - ((uint64_t)bytes[2] << 16) | ((uint64_t)bytes[3] << 24) | - ((uint64_t)bytes[4] << 32) | ((uint64_t)bytes[5] << 40) | - ((uint64_t)bytes[6] << 48) | ((uint64_t)bytes[7] << 56); - - return (calc_parity(word & 0x103fff800007fff, false) << 0) | - (calc_parity(word & 0x17c1ff801ff801f, true) << 1) | - (calc_parity(word & 0x1bde1f87e0781e1, false) << 2) | - (calc_parity(word & 0x1deee3b8e388e22, true) << 3) | - (calc_parity(word & 0x1ef76cdb2c93244, false) << 4) | - (calc_parity(word & 0x1f7bb56d5525488, true) << 5) | - (calc_parity(word & 0x1fbdda769a46910, false) << 6); +uint64_t word = ((uint64_t)bytes[0] << 0) | ((uint64_t)bytes[1] << 8) | ((uint64_t)bytes[2] << 16) | ((uint64_t)bytes[3] << 24) | ((uint64_t)bytes[4] << 32) | ((uint64_t)bytes[5] << 40) | ((uint64_t)bytes[6] << 48) | ((uint64_t)bytes[7] << 56); + +return (calc_parity(word & 0x103fff800007fff, false) << 0) | (calc_parity(word & 0x17c1ff801ff801f, true) << 1) | (calc_parity(word & 0x1bde1f87e0781e1, false) << 2) | (calc_parity(word & 0x1deee3b8e388e22, true) << 3) | (calc_parity(word & 0x1ef76cdb2c93244, false) << 4) | (calc_parity(word & 0x1f7bb56d5525488, true) << 5) | (calc_parity(word & 0x1fbdda769a46910, false) << 6); } uint8_t enc_secded_inv_72_64(const uint8_t bytes[8]) { - uint64_t word = ((uint64_t)bytes[0] << 0) | ((uint64_t)bytes[1] << 8) | - ((uint64_t)bytes[2] << 16) | ((uint64_t)bytes[3] << 24) | - ((uint64_t)bytes[4] << 32) | ((uint64_t)bytes[5] << 40) | - ((uint64_t)bytes[6] << 48) | ((uint64_t)bytes[7] << 56); - - return (calc_parity(word & 0xb9000000001fffff, false) << 0) | - (calc_parity(word & 0x5e00000fffe0003f, true) << 1) | - (calc_parity(word & 0x67003ff003e007c1, false) << 2) | - (calc_parity(word & 0xcd0fc0f03c207842, true) << 3) | - (calc_parity(word & 0xb671c711c4438884, false) << 4) | - (calc_parity(word & 0xb5b65926488c9108, true) << 5) | - (calc_parity(word & 0xcbdaaa4a91152210, false) << 6) | - (calc_parity(word & 0x7aed348d221a4420, true) << 7); +uint64_t word = ((uint64_t)bytes[0] << 0) | ((uint64_t)bytes[1] << 8) | ((uint64_t)bytes[2] << 16) | ((uint64_t)bytes[3] << 24) | ((uint64_t)bytes[4] << 32) | ((uint64_t)bytes[5] << 40) | ((uint64_t)bytes[6] << 48) | ((uint64_t)bytes[7] << 56); + +return (calc_parity(word & 0xb9000000001fffff, false) << 0) | (calc_parity(word & 0x5e00000fffe0003f, true) << 1) | (calc_parity(word & 0x67003ff003e007c1, false) << 2) | (calc_parity(word & 0xcd0fc0f03c207842, true) << 3) | (calc_parity(word & 0xb671c711c4438884, false) << 4) | (calc_parity(word & 0xb5b65926488c9108, true) << 5) | (calc_parity(word & 0xcbdaaa4a91152210, false) << 6) | (calc_parity(word & 0x7aed348d221a4420, true) << 7); } diff --git a/hw/ip/rom_ctrl/util/gen_vivado_mem_image.py b/hw/ip/rom_ctrl/util/gen_vivado_mem_image.py index 9a04c52f543943..ec098cd2a611a6 100755 --- a/hw/ip/rom_ctrl/util/gen_vivado_mem_image.py +++ b/hw/ip/rom_ctrl/util/gen_vivado_mem_image.py @@ -77,8 +77,9 @@ def parse_otp_init_strings(init_line_groups: List[List[str]]) -> List[int]: """Parse a sequence of 22-bit OTP words from Vivado INIT_XX lines. The data layout was determined by running a full Vivado bitstream build and - comparing the OTP image (//hw/ip/otp_ctrl/data:img_rma) with the INIT_XX - strings that Vivado produces (see the otp_init_strings.txt artifact). + comparing the OTP image (//hw/top_earlgrey/data/otp:img_rma) with the + INIT_XX strings that Vivado produces (see the otp_init_strings.txt + artifact). """ out = [] diff --git a/hw/ip/sram_ctrl/doc/interfaces.md b/hw/ip/sram_ctrl/doc/interfaces.md index fedbd0898cc044..757f6e431bd962 100644 --- a/hw/ip/sram_ctrl/doc/interfaces.md +++ b/hw/ip/sram_ctrl/doc/interfaces.md @@ -80,13 +80,13 @@ Signal | Direction | Type `lc_escalate_en_i` | `input` | `lc_ctrl_pkg::lc_tx_t` | Multibit life cycle escalation enable signal coming from life cycle controller, asserted if an escalation has occurred. `sram_otp_key_o` | `output` | `otp_ctrl_pkg::sram_otp_key_req_t` | Key derivation request going to the key derivation interface of the OTP controller. `sram_otp_key_i` | `input` | `otp_ctrl_pkg::sram_otp_key_rsp_t` | Ephemeral scrambling key coming back from the key derivation interface of the OTP controller. -`otp_en_sram_ifetch_i` | `input` | `otp_ctrl_pkg::mubi8_t` | Multibit value coming from the OTP HW_CFG partition ([EN_SRAM_IFETCH](../../otp_ctrl/README.md#direct-access-memory-map)), set to kMuBi8True in order to enable the [`EXEC`](../data/sram_ctrl.hjson#exec) CSR. +`otp_en_sram_ifetch_i` | `input` | `otp_ctrl_pkg::mubi8_t` | Multibit value coming from the OTP HW_CFG partition EN_SRAM_IFETCH, set to kMuBi8True in order to enable the [`EXEC`](../data/sram_ctrl.hjson#exec) CSR. For example, see earlgrey's [OTP Field Descriptions](../../../top_earlgrey/ip_autogen/otp_ctrl/doc/programmers_guide.md#otp-field-descriptions)) `cfg_i` | `input` | `logic [CfgWidth-1:0]` | Attributes for physical memory macro. ### Interfaces to OTP and the SRAM Scrambling Primitive The interface to the key derivation interface inside the OTP controller follows a simple req / ack protocol, where the SRAM controller first requests an updated ephemeral key by asserting the `sram_otp_key_i.req`. -The OTP controller then fetches entropy from CSRNG and derives an ephemeral key using the SRAM_DATA_KEY_SEED and the PRESENT scrambling data path as described in the [OTP controller spec](../../otp_ctrl/README.md#scrambling-datapath). +The OTP controller then fetches entropy from CSRNG and derives an ephemeral key using the SRAM_DATA_KEY_SEED and the PRESENT scrambling data path as described in the earlgrey's [OTP controller spec](../../../top_earlgrey/ip_autogen/otp_ctrl/README.md#scrambling-datapath) for example. Finally, the OTP controller returns a fresh ephemeral key via the response channels (`sram_otp_key_o[*]`, `otbn_otp_key_o`), which complete the req / ack handshake. The key and nonce are made available to the scrambling primitive in the subsequent cycle. The wave diagram below illustrates this process. @@ -114,7 +114,7 @@ It should be noted that this mechanism requires the CSRNG and entropy distributi Note that the req/ack protocol runs on `clk_otp_i`. The SRAM controller synchronizes the data over via a req/ack handshake primitive `prim_sync_reqack.sv` primitive as shown below. -![OTP Key Req Ack](../../otp_ctrl/doc/otp_ctrl_key_req_ack.svg) +![OTP Key Req Ack](../../../top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_key_req_ack.svg) Note that the key and nonce output signals on the OTP controller side are guaranteed to remain stable for at least 62 OTP clock cycles after the `ack` signal is pulsed high, because the derivation of a 64bit half-key takes at least two passes through the 31-cycle PRESENT primitive. Hence, if the SRAM controller clock `clk_i` is faster or in the same order of magnitude as `clk_otp_i`, the data can be directly sampled upon assertion of `src_ack_o`. diff --git a/hw/ip/sram_ctrl/doc/theory_of_operation.md b/hw/ip/sram_ctrl/doc/theory_of_operation.md index 206d8345554294..9980c92604a145 100644 --- a/hw/ip/sram_ctrl/doc/theory_of_operation.md +++ b/hw/ip/sram_ctrl/doc/theory_of_operation.md @@ -57,7 +57,7 @@ Note however that the PRNG sequence does not have strong security guarantees, si ### Code Execution from SRAM The SRAM controller contains an access control mechanism for filtering instruction fetches from the processor. -As illustrated below, an OTP switch EN_SRAM_IFETCH (see [OTP memory map](../../otp_ctrl/README.md#direct-access-memory-map)) allows to either tie code execution from SRAM to the life cycle state via the HW_DEBUG_EN function (see [life cycle docs](../../lc_ctrl/README.md#hw_debug_en)), or it can be enabled / disabled via the [`EXEC`](registers.md#exec) CSR. +As illustrated below, an OTP switch EN_SRAM_IFETCH (for example, see earlgrey's [OTP memory map](../../../top_earlgrey/ip_autogen/otp_ctrl/README.md#direct-access-memory-map)) allows to either tie code execution from SRAM to the life cycle state via the HW_DEBUG_EN function (see [life cycle docs](../../lc_ctrl/README.md#hw_debug_en)), or it can be enabled / disabled via the [`EXEC`](registers.md#exec) CSR. ![SRAM Code Execution](../doc/sram_ctrl_sram_execution.svg) diff --git a/hw/ip_templates/BUILD b/hw/ip_templates/BUILD index 25ba0b61f4f6db..e467cbf939c513 100644 --- a/hw/ip_templates/BUILD +++ b/hw/ip_templates/BUILD @@ -9,6 +9,7 @@ filegroup( srcs = glob(["**"]) + [ "//hw/ip_templates/clkmgr:all_files", "//hw/ip_templates/flash_ctrl:all_files", + "//hw/ip_templates/otp_ctrl:all_files", "//hw/ip_templates/pinmux:all_files", "//hw/ip_templates/pwrmgr:all_files", "//hw/ip_templates/rstmgr:all_files", diff --git a/hw/ip_templates/flash_ctrl/doc/programmers_guide.md b/hw/ip_templates/flash_ctrl/doc/programmers_guide.md index 8f57609cf38640..c47051fbe75a08 100644 --- a/hw/ip_templates/flash_ctrl/doc/programmers_guide.md +++ b/hw/ip_templates/flash_ctrl/doc/programmers_guide.md @@ -43,7 +43,7 @@ Once the address is discovered, further steps can be taken to triage the issue. ### Hardware Initiated Reads -If the root secrets have been provisioned in OTP and the life cycle state is in either DEV, PROD* or RMA, the special info pages holding the creator and owner seeds will be read out automatically by the flash controller and sent to the keymanager upon flash initialization (see [life cycle controller documentation](../../../../ip/lc_ctrl/doc/theory_of_operation.md#life-cycle-access-control-signals) and [OTP controller documentation](../../../../ip/otp_ctrl/doc/theory_of_operation.md#life-cycle-interfaces) for more details). +If the root secrets have been provisioned in OTP and the life cycle state is in either DEV, PROD* or RMA, the special info pages holding the creator and owner seeds will be read out automatically by the flash controller and sent to the keymanager upon flash initialization (see [life cycle controller documentation](../../../../ip/lc_ctrl/doc/theory_of_operation.md#life-cycle-access-control-signals) and [OTP controller documentation](../../otp_ctrl/doc/theory_of_operation.md#life-cycle-interfaces) for more details). Hence, it is important that these pages are initialized with valid data, since otherwise the hardware will likely encounter ECC errors during the automatic readout. Note that by default, hardware assumes that scrambling and ECC is enabled on these special info pages. diff --git a/hw/ip_templates/flash_ctrl/doc/theory_of_operation.md b/hw/ip_templates/flash_ctrl/doc/theory_of_operation.md index be6f32e03c0b56..b95837cf3f98e1 100644 --- a/hw/ip_templates/flash_ctrl/doc/theory_of_operation.md +++ b/hw/ip_templates/flash_ctrl/doc/theory_of_operation.md @@ -39,7 +39,7 @@ It is expected that after an RMA transition, the entire system will be rebooted. #### Initialization The flash protocol controller is initialized through [`INIT`](registers.md#init). -When initialization is invoked, the flash controller requests the address and data scrambling keys from an external entity, [otp_ctrl](../../../../ip/otp_ctrl/README.md#interface-to-flash-scrambler) in this case. +When initialization is invoked, the flash controller requests the address and data scrambling keys from an external entity, [otp_ctrl](../../otp_ctrl/README.md#interface-to-flash-scrambler) in this case. After the scrambling keys are requested, the flash protocol controller reads the root seeds out of the [secret partitions](#secret-information-partitions) and sends them to the key manager. Once the above steps are completed, the read buffers in the flash physical controller are enabled for operation. diff --git a/hw/ip/otp_ctrl/BUILD b/hw/ip_templates/otp_ctrl/BUILD similarity index 76% rename from hw/ip/otp_ctrl/BUILD rename to hw/ip_templates/otp_ctrl/BUILD index a62d75c1b6c994..dda56de039e27f 100644 --- a/hw/ip/otp_ctrl/BUILD +++ b/hw/ip_templates/otp_ctrl/BUILD @@ -6,7 +6,5 @@ package(default_visibility = ["//visibility:public"]) filegroup( name = "all_files", - srcs = glob(["**"]) + [ - "//hw/ip/otp_ctrl/data:all_files", - ], + srcs = glob(["**"]), ) diff --git a/hw/ip_templates/otp_ctrl/BUILD.tpl b/hw/ip_templates/otp_ctrl/BUILD.tpl new file mode 100644 index 00000000000000..dda56de039e27f --- /dev/null +++ b/hw/ip_templates/otp_ctrl/BUILD.tpl @@ -0,0 +1,10 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 + +package(default_visibility = ["//visibility:public"]) + +filegroup( + name = "all_files", + srcs = glob(["**"]), +) diff --git a/hw/ip_templates/otp_ctrl/README.md.tpl b/hw/ip_templates/otp_ctrl/README.md.tpl new file mode 100644 index 00000000000000..f32e3e12e18f9d --- /dev/null +++ b/hw/ip_templates/otp_ctrl/README.md.tpl @@ -0,0 +1,63 @@ +# OTP Controller Technical Specification + +[`otp_ctrl`](https://reports.opentitan.org/hw/top_${topname}/ip_autogen/otp_ctrl/dv/latest/report.html): +![](https://dashboards.lowrisc.org/badges/dv/otp_ctrl/test.svg) +![](https://dashboards.lowrisc.org/badges/dv/otp_ctrl/passing.svg) +![](https://dashboards.lowrisc.org/badges/dv/otp_ctrl/functional.svg) +![](https://dashboards.lowrisc.org/badges/dv/otp_ctrl/code.svg) + +# Overview + +This document specifies the functionality of the one time programmable (OTP) memory controller. +The OTP controller is a module that is a peripheral on the chip interconnect bus, and thus follows the [Comportability Specification](../../../../doc/contributing/hw/comportability/README.md). + +The OTP is a module that provides a device with one-time-programming functionality. +The result of this programming is non-volatile, and unlike flash, cannot be reversed. +The OTP functionality is constructed through an open-source OTP controller and a proprietary OTP IP. + +The OTP controller provides: +- An open-source abstraction interface that software can use to interact with a proprietary OTP block underneath. +- An open-source abstraction interface that hardware components (for example [life cycle controller](../../../ip/lc_ctrl/README.md) and [key manager](../../../ip/keymgr/README.md)) can use to interact with a proprietary OTP block underneath. +- High level logical security protection, such as integrity checks and scrambling of sensitive content. +- Software isolation for when OTP contents are readable and programmable. + +The proprietary OTP IP provides: +- Reliable, non-volatile storage. +- Technology-specific redundancy or error correction mechanisms. +- Physical defensive features such as SCA and FI resistance. +- Visual and electrical probing resistance. + +Together, the OTP controller and IP provide secure one-time-programming functionality that is used throughout the life cycle (LC) of a device. + +${"##"} Features + +- Multiple logical partitions of the underlying OTP IP + - Each partition is lockable and integrity checked + - Integrity digests are stored alongside each logical bank +- Periodic / persistent checks of OTP values + - Periodic checks of shadowed content vs digests + - Periodic checks of OTP stored content and shadowed content + - Persistent checks for immediate errors +- Separate life cycle partition and interface to life cycle controller + - Supports life cycle functions, but cannot be integrity locked +- Lightweight scrambling of secret OTP partition using a global netlist constant +- Lightweight ephemeral key derivation function for RAM scrambling mechanisms +- Lightweight key derivation function for FLASH scrambling mechanism + +${"##"} OTP Controller Overview + +The functionality of OTP is split into an open-source and a closed-source part, with a clearly defined boundary in between, as illustrated in the simplified high-level block diagram below. + +![OTP Controller Overview](./doc/otp_ctrl_overview.svg) + +It is the task of the open-source controller to provide a common, non-technology specific interface to OTP users with a common register interface and a clearly defined I/O interface to hardware. +The open-source controller implements logical isolation and partitioning of OTP storage that enables users to separate different functions of the OTP into "partitions" with different properties. +Finally, the open-source controller provides a high level of security for specific partitions by provisioning integrity digests for each partition, and scrambling of partitions where required. + +The proprietary IP on the other hand translates a common access interface to the technology-specific OTP interface, both for functional and debug accesses (for example register accesses to the macro-internal control structure). + +This split implies that every proprietary OTP IP must implement a translation layer from a standardized OpenTitan interface to the module underneath. +It also implies that no matter how the OTP storage or word size may change underneath, the open-source controller must present a consistent and coherent software and hardware interface. +This standardized interface is defined further below, and the wrapper leverages the same [technology primitive mechanism](../../../ip/prim/README.md) that is employed in other parts of OpenTitan in order to wrap and abstract technology-specific macros (such as memories and clocking cells) that are potentially closed-source. + +In order to enable simulation and FPGA emulation of the OTP controller even without access to the proprietary OTP IP, a generalized and synthesizable model of the OTP IP is provided in the form of a [generic technology primitive](https://github.com/lowRISC/opentitan/blob/master/hw/ip/prim_generic/rtl/prim_generic_otp.sv). diff --git a/hw/ip/otp_ctrl/data/dif_otp_ctrl.c.tpl b/hw/ip_templates/otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet similarity index 99% rename from hw/ip/otp_ctrl/data/dif_otp_ctrl.c.tpl rename to hw/ip_templates/otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet index 24266ccbdaff18..766a1a333d5840 100644 --- a/hw/ip/otp_ctrl/data/dif_otp_ctrl.c.tpl +++ b/hw/ip_templates/otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet @@ -1,11 +1,10 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -${gen_comment} <% from topgen.lib import Name -parts = otp_mmap.config["partitions"] +parts = otp_mmap["partitions"] digest_parts = [part for part in parts if part["hw_digest"] or part["sw_digest"]] read_locked_csr_parts = [part for part in parts if part["read_lock"] == "CSR"] diff --git a/hw/ip/otp_ctrl/data/dif_otp_ctrl.h.tpl b/hw/ip_templates/otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet similarity index 99% rename from hw/ip/otp_ctrl/data/dif_otp_ctrl.h.tpl rename to hw/ip_templates/otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet index 7699db5f2912e6..fbab47b27d488d 100644 --- a/hw/ip/otp_ctrl/data/dif_otp_ctrl.h.tpl +++ b/hw/ip_templates/otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet @@ -1,11 +1,10 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -${gen_comment} <% from topgen.lib import Name -parts = otp_mmap.config["partitions"] +parts = otp_mmap["partitions"] digest_parts = [part for part in parts if part["hw_digest"] == "true" or part["sw_digest"] == "true"] read_locked_csr_parts = [part for part in parts if part["read_lock"] == "CSR"] @@ -16,8 +15,8 @@ secret_parts = [part for part in parts if part["secret"] == "true"] /** * @file - * @brief OTP Controller Device Interface - * Functions + * @brief + * OTP Controller Device Interface Functions */ #include diff --git a/hw/ip/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl b/hw/ip_templates/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl.not_yet similarity index 99% rename from hw/ip/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl rename to hw/ip_templates/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl.not_yet index bf269497441985..2063fd5971ebc5 100644 --- a/hw/ip/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl +++ b/hw/ip_templates/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl.not_yet @@ -1,11 +1,10 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -${gen_comment} <% from topgen.lib import Name -parts = otp_mmap.config["partitions"] +parts = otp_mmap["partitions"] digest_parts = [part for part in parts if part["hw_digest"] or part["sw_digest"]] hw_digest_parts = [part for part in parts if part["hw_digest"]] diff --git a/hw/ip/otp_ctrl/data/otp_ctrl.hjson.tpl b/hw/ip_templates/otp_ctrl/data/otp_ctrl.hjson.tpl similarity index 98% rename from hw/ip/otp_ctrl/data/otp_ctrl.hjson.tpl rename to hw/ip_templates/otp_ctrl/data/otp_ctrl.hjson.tpl index f9e96a5c05193d..04b6479eafab9e 100644 --- a/hw/ip/otp_ctrl/data/otp_ctrl.hjson.tpl +++ b/hw/ip_templates/otp_ctrl/data/otp_ctrl.hjson.tpl @@ -4,17 +4,16 @@ // // HJSON with partition metadata. // -${gen_comment} <% from topgen.lib import Name -num_part = len(otp_mmap.config["partitions"]) +num_part = len(otp_mmap["partitions"]) num_part_unbuf = 0 -for part in otp_mmap.config["partitions"]: +for part in otp_mmap["partitions"]: if part["variant"] == "Unbuffered": num_part_unbuf += 1 num_part_buf = num_part - num_part_unbuf -otp_size_as_bytes = 2 ** otp_mmap.config["otp"]["byte_addr_width"] +otp_size_as_bytes = 2 ** otp_mmap["otp"]["byte_addr_width"] otp_size_as_uint32 = otp_size_as_bytes // 4 %>\ { @@ -124,19 +123,19 @@ otp_size_as_uint32 = otp_size_as_bytes // 4 // Random netlist constants { name: "RndCnstLfsrSeed", desc: "Compile-time random bits for initial LFSR seed", - type: "otp_ctrl_pkg::lfsr_seed_t" + type: "otp_ctrl_top_specific_pkg::lfsr_seed_t" randcount: "40", randtype: "data", // randomize randcount databits } { name: "RndCnstLfsrPerm", desc: "Compile-time random permutation for LFSR output", - type: "otp_ctrl_pkg::lfsr_perm_t" + type: "otp_ctrl_top_specific_pkg::lfsr_perm_t" randcount: "40", randtype: "perm", // random permutation for randcount elements } { name: "RndCnstScrmblKeyInit", desc: "Compile-time random permutation for scrambling key/nonce register reset value", - type: "otp_ctrl_pkg::scrmbl_key_init_t" + type: "otp_ctrl_top_specific_pkg::scrmbl_key_init_t" randcount: "256", randtype: "data", // random permutation for randcount elements } @@ -150,7 +149,7 @@ otp_size_as_uint32 = otp_size_as_bytes // 4 { name: "OtpByteAddrWidth", desc: "Width of the OTP byte address.", type: "int", - default: "${otp_mmap.config["otp"]["byte_addr_width"]}", + default: "${otp_mmap["otp"]["byte_addr_width"]}", local: "true" }, { name: "NumErrorEntries", @@ -197,7 +196,7 @@ otp_size_as_uint32 = otp_size_as_bytes // 4 default: "${num_part_buf}", local: "true" }, -% for part in otp_mmap.config["partitions"]: +% for part in otp_mmap["partitions"]: <% part_name = Name.from_snake_case(part["name"]) part_name_camel = part_name.as_camel_case() @@ -713,7 +712,7 @@ otp_size_as_uint32 = otp_size_as_bytes // 4 tags: [ // OTP internal HW can modify status register "excl:CsrAllTests:CsrExclCheck"], fields: [ - % for k, part in enumerate(otp_mmap.config["partitions"]): + % for k, part in enumerate(otp_mmap["partitions"]): { bits: "${k}" name: "${part["name"]}_ERROR" desc: ''' @@ -1140,7 +1139,7 @@ otp_size_as_uint32 = otp_size_as_bytes // 4 //////////////////////////////////// // Dynamic Locks of SW Parititons // //////////////////////////////////// - % for part in otp_mmap.config["partitions"]: + % for part in otp_mmap["partitions"]: % if part["read_lock"].lower() == "csr": { name: "${part["name"]}_READ_LOCK", desc: ''' @@ -1169,7 +1168,7 @@ otp_size_as_uint32 = otp_size_as_bytes // 4 /////////////////////// // Integrity Digests // /////////////////////// - % for part in otp_mmap.config["partitions"]: + % for part in otp_mmap["partitions"]: % if part["sw_digest"]: { multireg: { name: "${part["name"]}_DIGEST", diff --git a/hw/ip_templates/otp_ctrl/data/otp_ctrl.tpldesc.hjson b/hw/ip_templates/otp_ctrl/data/otp_ctrl.tpldesc.hjson new file mode 100644 index 00000000000000..e0ff61c40848c7 --- /dev/null +++ b/hw/ip_templates/otp_ctrl/data/otp_ctrl.tpldesc.hjson @@ -0,0 +1,19 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +{ + template_param_list: [ + { + name: "topname" + desc: "Name of top-level design, e.g., 'darjeeling' or 'earlgrey'" + type: "string" + default: "" + } + { + name: "otp_mmap" + desc: "An object containing the memory map and all attributes" + type: "object" + default: {} + } + ] +} diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_sec_cm_testplan.hjson b/hw/ip_templates/otp_ctrl/data/otp_ctrl_sec_cm_testplan.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_sec_cm_testplan.hjson rename to hw/ip_templates/otp_ctrl/data/otp_ctrl_sec_cm_testplan.hjson diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_testplan.hjson b/hw/ip_templates/otp_ctrl/data/otp_ctrl_testplan.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_testplan.hjson rename to hw/ip_templates/otp_ctrl/data/otp_ctrl_testplan.hjson diff --git a/hw/ip_templates/otp_ctrl/defs.bzl.tpl b/hw/ip_templates/otp_ctrl/defs.bzl.tpl new file mode 100644 index 00000000000000..c9e903f3a76a60 --- /dev/null +++ b/hw/ip_templates/otp_ctrl/defs.bzl.tpl @@ -0,0 +1,9 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +load("//rules/opentitan:hw.bzl", "opentitan_ip") + +OTP_CTRL = opentitan_ip( + name = "otp_ctrl", + hjson = "//hw/top_${topname}/ip_autogen/otp_ctrl:data/otp_ctrl.hjson", +) diff --git a/hw/ip_templates/otp_ctrl/doc/checklist.md b/hw/ip_templates/otp_ctrl/doc/checklist.md new file mode 100644 index 00000000000000..117db7dd8d60bd --- /dev/null +++ b/hw/ip_templates/otp_ctrl/doc/checklist.md @@ -0,0 +1,271 @@ +# OTP_CTRL Checklist + + +This checklist is for [Hardware Stage](../../../../../doc/project_governance/development_stages.md) transitions for the [OTP_CTRL peripheral.](../README.md) +All checklist items refer to the content in the [Checklist.](../../../../../doc/project_governance/checklist/README.md) + +## Design Checklist + +### D1 + +Type | Item | Resolution | Note/Collaterals +--------------|--------------------------------|-------------|------------------ +Documentation | [SPEC_COMPLETE][] | Done | [OTP_CTRL Design Spec](../README.md) +Documentation | [CSR_DEFINED][] | Done | +RTL | [CLKRST_CONNECTED][] | Done | +RTL | [IP_TOP][] | Done | +RTL | [IP_INSTANTIABLE][] | Done | +RTL | [PHYSICAL_MACROS_DEFINED_80][] | Done | +RTL | [FUNC_IMPLEMENTED][] | Done | +RTL | [ASSERT_KNOWN_ADDED][] | Done | +Code Quality | [LINT_SETUP][] | Done | + +[SPEC_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#spec_complete +[CSR_DEFINED]: ../../../../../doc/project_governance/checklist/README.md#csr_defined +[CLKRST_CONNECTED]: ../../../../../doc/project_governance/checklist/README.md#clkrst_connected +[IP_TOP]: ../../../../../doc/project_governance/checklist/README.md#ip_top +[IP_INSTANTIABLE]: ../../../../../doc/project_governance/checklist/README.md#ip_instantiable +[PHYSICAL_MACROS_DEFINED_80]: ../../../../../doc/project_governance/checklist/README.md#mem_instanced_80 +[FUNC_IMPLEMENTED]: ../../../../../doc/project_governance/checklist/README.md#func_implemented +[ASSERT_KNOWN_ADDED]: ../../../../../doc/project_governance/checklist/README.md#assert_known_added +[LINT_SETUP]: ../../../../../doc/project_governance/checklist/README.md#lint_setup + +### D2 + +Type | Item | Resolution | Note/Collaterals +--------------|---------------------------|-------------|------------------ +Documentation | [NEW_FEATURES][] | Done | +Documentation | [BLOCK_DIAGRAM][] | Done | +Documentation | [DOC_INTERFACE][] | Done | +Documentation | [DOC_INTEGRATION_GUIDE][] | Waived | This checklist item has been added retrospectively. +Documentation | [MISSING_FUNC][] | Done | +Documentation | [FEATURE_FROZEN][] | Done | +RTL | [FEATURE_COMPLETE][] | Done | +RTL | [PORT_FROZEN][] | Done | +RTL | [ARCHITECTURE_FROZEN][] | Done | +RTL | [REVIEW_TODO][] | Done | +RTL | [STYLE_X][] | Done | +RTL | [CDC_SYNCMACRO][] | N/A | +Code Quality | [LINT_PASS][] | Done | +Code Quality | [CDC_SETUP][] | Waived | No block-level flow available - waived to top-level signoff. +Code Quality | [RDC_SETUP][] | Waived | No block-level flow available - waived to top-level signoff. +Code Quality | [AREA_CHECK][] | Done | +Code Quality | [TIMING_CHECK][] | Done | +Security | [SEC_CM_DOCUMENTED][] | Done | + +[NEW_FEATURES]: ../../../../../doc/project_governance/checklist/README.md#new_features +[BLOCK_DIAGRAM]: ../../../../../doc/project_governance/checklist/README.md#block_diagram +[DOC_INTERFACE]: ../../../../../doc/project_governance/checklist/README.md#doc_interface +[DOC_INTEGRATION_GUIDE]: ../../../../../doc/project_governance/checklist/README.md#doc_integration_guide +[MISSING_FUNC]: ../../../../../doc/project_governance/checklist/README.md#missing_func +[FEATURE_FROZEN]: ../../../../../doc/project_governance/checklist/README.md#feature_frozen +[FEATURE_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#feature_complete +[PORT_FROZEN]: ../../../../../doc/project_governance/checklist/README.md#port_frozen +[ARCHITECTURE_FROZEN]: ../../../../../doc/project_governance/checklist/README.md#architecture_frozen +[REVIEW_TODO]: ../../../../../doc/project_governance/checklist/README.md#review_todo +[STYLE_X]: ../../../../../doc/project_governance/checklist/README.md#style_x +[CDC_SYNCMACRO]: ../../../../../doc/project_governance/checklist/README.md#cdc_syncmacro +[LINT_PASS]: ../../../../../doc/project_governance/checklist/README.md#lint_pass +[CDC_SETUP]: ../../../../../doc/project_governance/checklist/README.md#cdc_setup +[RDC_SETUP]: ../../../../../doc/project_governance/checklist/README.md#rdc_setup +[AREA_CHECK]: ../../../../../doc/project_governance/checklist/README.md#area_check +[TIMING_CHECK]: ../../../../../doc/project_governance/checklist/README.md#timing_check +[SEC_CM_DOCUMENTED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_documented + +### D2S + + Type | Item | Resolution | Note/Collaterals +--------------|------------------------------|-------------|------------------ +Security | [SEC_CM_ASSETS_LISTED][] | Done | +Security | [SEC_CM_IMPLEMENTED][] | Done | +Security | [SEC_CM_RND_CNST][] | Done | +Security | [SEC_CM_NON_RESET_FLOPS][] | Done | +Security | [SEC_CM_SHADOW_REGS][] | Done | +Security | [SEC_CM_RTL_REVIEWED][] | Done | +Security | [SEC_CM_COUNCIL_REVIEWED][] | Done | + +[SEC_CM_ASSETS_LISTED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_assets_listed +[SEC_CM_IMPLEMENTED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_implemented +[SEC_CM_RND_CNST]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_rnd_cnst +[SEC_CM_NON_RESET_FLOPS]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_non_reset_flops +[SEC_CM_SHADOW_REGS]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_shadow_regs +[SEC_CM_RTL_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_rtl_reviewed +[SEC_CM_COUNCIL_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_council_reviewed + +### D3 + + Type | Item | Resolution | Note/Collaterals +--------------|-------------------------|-------------|------------------ +Documentation | [NEW_FEATURES_D3][] | Done | +RTL | [TODO_COMPLETE][] | Done | +Code Quality | [LINT_COMPLETE][] | Done | Waiver files approved by TC on 2024-08-08. +Code Quality | [CDC_COMPLETE][] | Waived | No block-level flow available - waived to top-level signoff. +Code Quality | [RDC_COMPLETE][] | Waived | No block-level flow available - waived to top-level signoff. +Review | [REVIEW_RTL][] | Done | +Review | [REVIEW_DELETED_FF][] | Waived | No block-level flow available - waived to top-level signoff. +Review | [REVIEW_SW_CHANGE][] | Done | +Review | [REVIEW_SW_ERRATA][] | Done | +Review | Reviewer(s) | Done | adk@ vogelpi@ +Review | Signoff date | Done | 2024-08-08 + +[NEW_FEATURES_D3]: ../../../../../doc/project_governance/checklist/README.md#new_features_d3 +[TODO_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#todo_complete +[LINT_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#lint_complete +[CDC_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#cdc_complete +[RDC_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#rdc_complete +[REVIEW_RTL]: ../../../../../doc/project_governance/checklist/README.md#review_rtl +[REVIEW_DELETED_FF]: ../../../../../doc/project_governance/checklist/README.md#review_deleted_ff +[REVIEW_SW_CHANGE]: ../../../../../doc/project_governance/checklist/README.md#review_sw_change +[REVIEW_SW_ERRATA]: ../../../../../doc/project_governance/checklist/README.md#review_sw_errata + +## Verification Checklist + +### V1 + + Type | Item | Resolution | Note/Collaterals +--------------|---------------------------------------|-------------|------------------ +Documentation | [DV_DOC_DRAFT_COMPLETED][] | Done | [OTP_CTRL DV document](../dv/README.md) +Documentation | [TESTPLAN_COMPLETED][] | Done | [OTP_CTRL Testplan](../dv/README.md#testplan) +Testbench | [TB_TOP_CREATED][] | Done | +Testbench | [PRELIMINARY_ASSERTION_CHECKS_ADDED][]| Done | +Testbench | [SIM_TB_ENV_CREATED][] | Done | +Testbench | [SIM_RAL_MODEL_GEN_AUTOMATED][] | Done | +Testbench | [CSR_CHECK_GEN_AUTOMATED][] | Done | +Testbench | [TB_GEN_AUTOMATED][] | Done | +Tests | [SIM_SMOKE_TEST_PASSING][] | Done | +Tests | [SIM_CSR_MEM_TEST_SUITE_PASSING][] | Done | +Tests | [FPV_MAIN_ASSERTIONS_PROVEN][] | N/A | +Tool Setup | [SIM_ALT_TOOL_SETUP][] | Done | +Regression | [SIM_SMOKE_REGRESSION_SETUP][] | Done | +Regression | [SIM_NIGHTLY_REGRESSION_SETUP][] | Done | +Regression | [FPV_REGRESSION_SETUP][] | N/A | +Coverage | [SIM_COVERAGE_MODEL_ADDED][] | Done | +Code Quality | [TB_LINT_SETUP][] | Done | +Integration | [PRE_VERIFIED_SUB_MODULES_V1][] | N/A | Exception for IP modules +Review | [DESIGN_SPEC_REVIEWED][] | Done | +Review | [TESTPLAN_REVIEWED][] | Done | +Review | [STD_TEST_CATEGORIES_PLANNED][] | Done | Exception (Security, Power, Debug) +Review | [V2_CHECKLIST_SCOPED][] | Done | + +[DV_DOC_DRAFT_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#dv_doc_draft_completed +[TESTPLAN_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#testplan_completed +[TB_TOP_CREATED]: ../../../../../doc/project_governance/checklist/README.md#tb_top_created +[PRELIMINARY_ASSERTION_CHECKS_ADDED]: ../../../../../doc/project_governance/checklist/README.md#preliminary_assertion_checks_added +[SIM_TB_ENV_CREATED]: ../../../../../doc/project_governance/checklist/README.md#sim_tb_env_created +[SIM_RAL_MODEL_GEN_AUTOMATED]: ../../../../../doc/project_governance/checklist/README.md#sim_ral_model_gen_automated +[CSR_CHECK_GEN_AUTOMATED]: ../../../../../doc/project_governance/checklist/README.md#csr_check_gen_automated +[TB_GEN_AUTOMATED]: ../../../../../doc/project_governance/checklist/README.md#tb_gen_automated +[SIM_SMOKE_TEST_PASSING]: ../../../../../doc/project_governance/checklist/README.md#sim_smoke_test_passing +[SIM_CSR_MEM_TEST_SUITE_PASSING]: ../../../../../doc/project_governance/checklist/README.md#sim_csr_mem_test_suite_passing +[FPV_MAIN_ASSERTIONS_PROVEN]: ../../../../../doc/project_governance/checklist/README.md#fpv_main_assertions_proven +[SIM_ALT_TOOL_SETUP]: ../../../../../doc/project_governance/checklist/README.md#sim_alt_tool_setup +[SIM_SMOKE_REGRESSION_SETUP]: ../../../../../doc/project_governance/checklist/README.md#sim_smoke_regression_setup +[SIM_NIGHTLY_REGRESSION_SETUP]: ../../../../../doc/project_governance/checklist/README.md#sim_nightly_regression_setup +[FPV_REGRESSION_SETUP]: ../../../../../doc/project_governance/checklist/README.md#fpv_regression_setup +[SIM_COVERAGE_MODEL_ADDED]: ../../../../../doc/project_governance/checklist/README.md#sim_coverage_model_added +[PRE_VERIFIED_SUB_MODULES_V1]: ../../../../../doc/project_governance/checklist/README.md#pre_verified_sub_modules_v1 +[TB_LINT_SETUP]: ../../../../../doc/project_governance/checklist/README.md#tb_lint_setup +[DESIGN_SPEC_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#design_spec_reviewed +[TESTPLAN_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#testplan_reviewed +[STD_TEST_CATEGORIES_PLANNED]: ../../../../../doc/project_governance/checklist/README.md#std_test_categories_planned +[V2_CHECKLIST_SCOPED]: ../../../../../doc/project_governance/checklist/README.md#v2_checklist_scoped + +### V2 + + Type | Item | Resolution | Note/Collaterals +--------------|-----------------------------------------|-------------|------------------ +Documentation | [DESIGN_DELTAS_CAPTURED_V2][] | Done | +Documentation | [DV_DOC_COMPLETED][] | Done | +Testbench | [FUNCTIONAL_COVERAGE_IMPLEMENTED][] | Done | +Testbench | [ALL_INTERFACES_EXERCISED][] | Done | `prim_tl_o/i` has a simple prim_tl_agent support, but need to be replaced with auto-generated tl_agent once reggen tool is optimized. +Testbench | [ALL_ASSERTION_CHECKS_ADDED][] | Done | +Testbench | [SIM_TB_ENV_COMPLETED][] | Done | +Tests | [SIM_ALL_TESTS_PASSING][] | Done | +Tests | [FPV_ALL_ASSERTIONS_WRITTEN][] | N/A | +Tests | [FPV_ALL_ASSUMPTIONS_REVIEWED][] | N/A | +Tests | [SIM_FW_SIMULATED][] | N/A | +Regression | [SIM_NIGHTLY_REGRESSION_V2][] | Done | +Coverage | [SIM_CODE_COVERAGE_V2][] | Done | +Coverage | [SIM_FUNCTIONAL_COVERAGE_V2][] | Done | +Coverage | [FPV_CODE_COVERAGE_V2][] | N/A | +Coverage | [FPV_COI_COVERAGE_V2][] | N/A | +Integration | [PRE_VERIFIED_SUB_MODULES_V2][] | Done | Waived `prim_alert_sender` and `prim_lfsr` +Issues | [NO_HIGH_PRIORITY_ISSUES_PENDING][] | Done | +Issues | [ALL_LOW_PRIORITY_ISSUES_ROOT_CAUSED][] | Done | +Review | [DV_DOC_TESTPLAN_REVIEWED][] | Done | Reviewed on 05/24/2021 +Review | [V3_CHECKLIST_SCOPED][] | Done | + +[DESIGN_DELTAS_CAPTURED_V2]: ../../../../../doc/project_governance/checklist/README.md#design_deltas_captured_v2 +[DV_DOC_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#dv_doc_completed +[FUNCTIONAL_COVERAGE_IMPLEMENTED]: ../../../../../doc/project_governance/checklist/README.md#functional_coverage_implemented +[ALL_INTERFACES_EXERCISED]: ../../../../../doc/project_governance/checklist/README.md#all_interfaces_exercised +[ALL_ASSERTION_CHECKS_ADDED]: ../../../../../doc/project_governance/checklist/README.md#all_assertion_checks_added +[SIM_TB_ENV_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#sim_tb_env_completed +[SIM_ALL_TESTS_PASSING]: ../../../../../doc/project_governance/checklist/README.md#sim_all_tests_passing +[FPV_ALL_ASSERTIONS_WRITTEN]: ../../../../../doc/project_governance/checklist/README.md#fpv_all_assertions_written +[FPV_ALL_ASSUMPTIONS_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#fpv_all_assumptions_reviewed +[SIM_FW_SIMULATED]: ../../../../../doc/project_governance/checklist/README.md#sim_fw_simulated +[SIM_NIGHTLY_REGRESSION_V2]: ../../../../../doc/project_governance/checklist/README.md#sim_nightly_regression_v2 +[SIM_CODE_COVERAGE_V2]: ../../../../../doc/project_governance/checklist/README.md#sim_code_coverage_v2 +[SIM_FUNCTIONAL_COVERAGE_V2]: ../../../../../doc/project_governance/checklist/README.md#sim_functional_coverage_v2 +[FPV_CODE_COVERAGE_V2]: ../../../../../doc/project_governance/checklist/README.md#fpv_code_coverage_v2 +[FPV_COI_COVERAGE_V2]: ../../../../../doc/project_governance/checklist/README.md#fpv_coi_coverage_v2 +[PRE_VERIFIED_SUB_MODULES_V2]: ../../../../../doc/project_governance/checklist/README.md#pre_verified_sub_modules_v2 +[NO_HIGH_PRIORITY_ISSUES_PENDING]: ../../../../../doc/project_governance/checklist/README.md#no_high_priority_issues_pending +[ALL_LOW_PRIORITY_ISSUES_ROOT_CAUSED]:../../../../../doc/project_governance/checklist/README.md#all_low_priority_issues_root_caused +[DV_DOC_TESTPLAN_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#dv_doc_testplan_reviewed +[V3_CHECKLIST_SCOPED]: ../../../../../doc/project_governance/checklist/README.md#v3_checklist_scoped + +### V2S + + Type | Item | Resolution | Note/Collaterals +--------------|-----------------------------------------|-------------|------------------ +Documentation | [SEC_CM_TESTPLAN_COMPLETED][] | Done | +Tests | [FPV_SEC_CM_VERIFIED][] | Done | +Tests | [SIM_SEC_CM_VERIFIED][] | Done | +Coverage | [SIM_COVERAGE_REVIEWED][] | Done | +Review | [SEC_CM_DV_REVIEWED][] | Done | + +[SEC_CM_TESTPLAN_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_testplan_completed +[FPV_SEC_CM_VERIFIED]: ../../../../../doc/project_governance/checklist/README.md#fpv_sec_cm_verified +[SIM_SEC_CM_VERIFIED]: ../../../../../doc/project_governance/checklist/README.md#sim_sec_cm_verified +[SIM_COVERAGE_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#sim_coverage_reviewed +[SEC_CM_DV_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_dv_reviewed + +### V3 + + Type | Item | Resolution | Note/Collaterals +--------------|-----------------------------------|-------------|------------------ +Documentation | [DESIGN_DELTAS_CAPTURED_V3][] | Not Started | +Tests | [X_PROP_ANALYSIS_COMPLETED][] | Not Started | +Tests | [FPV_ASSERTIONS_PROVEN_AT_V3][] | Not Started | +Regression | [SIM_NIGHTLY_REGRESSION_AT_V3][] | Not Started | +Coverage | [SIM_CODE_COVERAGE_AT_100][] | Not Started | +Coverage | [SIM_FUNCTIONAL_COVERAGE_AT_100][]| Not Started | +Coverage | [FPV_CODE_COVERAGE_AT_100][] | Not Started | +Coverage | [FPV_COI_COVERAGE_AT_100][] | Not Started | +Code Quality | [ALL_TODOS_RESOLVED][] | Not Started | +Code Quality | [NO_TOOL_WARNINGS_THROWN][] | Not Started | +Code Quality | [TB_LINT_COMPLETE][] | Not Started | +Integration | [PRE_VERIFIED_SUB_MODULES_V3][] | Not Started | +Issues | [NO_ISSUES_PENDING][] | Not Started | +Review | Reviewer(s) | Not Started | +Review | Signoff date | Not Started | + +[DESIGN_DELTAS_CAPTURED_V3]: ../../../../../doc/project_governance/checklist/README.md#design_deltas_captured_v3 +[X_PROP_ANALYSIS_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#x_prop_analysis_completed +[FPV_ASSERTIONS_PROVEN_AT_V3]: ../../../../../doc/project_governance/checklist/README.md#fpv_assertions_proven_at_v3 +[SIM_NIGHTLY_REGRESSION_AT_V3]: ../../../../../doc/project_governance/checklist/README.md#sim_nightly_regression_at_v3 +[SIM_CODE_COVERAGE_AT_100]: ../../../../../doc/project_governance/checklist/README.md#sim_code_coverage_at_100 +[SIM_FUNCTIONAL_COVERAGE_AT_100]:../../../../../doc/project_governance/checklist/README.md#sim_functional_coverage_at_100 +[FPV_CODE_COVERAGE_AT_100]: ../../../../../doc/project_governance/checklist/README.md#fpv_code_coverage_at_100 +[FPV_COI_COVERAGE_AT_100]: ../../../../../doc/project_governance/checklist/README.md#fpv_coi_coverage_at_100 +[ALL_TODOS_RESOLVED]: ../../../../../doc/project_governance/checklist/README.md#all_todos_resolved +[NO_TOOL_WARNINGS_THROWN]: ../../../../../doc/project_governance/checklist/README.md#no_tool_warnings_thrown +[TB_LINT_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#tb_lint_complete +[PRE_VERIFIED_SUB_MODULES_V3]: ../../../../../doc/project_governance/checklist/README.md#pre_verified_sub_modules_v3 +[NO_ISSUES_PENDING]: ../../../../../doc/project_governance/checklist/README.md#no_issues_pending diff --git a/hw/ip_templates/otp_ctrl/doc/interfaces.md.tpl b/hw/ip_templates/otp_ctrl/doc/interfaces.md.tpl new file mode 100644 index 00000000000000..159c54f05ebb54 --- /dev/null +++ b/hw/ip_templates/otp_ctrl/doc/interfaces.md.tpl @@ -0,0 +1,192 @@ +# Hardware Interfaces + +${"##"} Parameters + +The following table lists the instantiation parameters of OTP. +Note that parameters prefixed with `RndCnst` are random netlist constants that need to be regenerated via topgen before the tapeout (typically by the silicon creator). + +Parameter | Default (Max) | Top Earlgrey | Description +----------------------------|---------------|--------------|--------------- +`AlertAsyncOn` | 2'b11 | 2'b11 | +`RndCnstLfsrSeed` | (see RTL) | (see RTL) | Seed to be used for the internal 40bit partition check timer LFSR. This needs to be replaced by the silicon creator before the tapeout. +`RndCnstLfsrPerm` | (see RTL) | (see RTL) | Permutation to be used for the internal 40bit partition check timer LFSR. This needs to be replaced by the silicon creator before the tapeout. +`RndCnstKey` | (see RTL) | (see RTL) | Random scrambling keys for secret partitions, to be used in the [scrambling datapath](#scrambling-datapath). +`RndCnstDigestConst` | (see RTL) | (see RTL) | Random digest finalization constants, to be used in the [scrambling datapath](#scrambling-datapath). +`RndCnstDigestIV` | (see RTL) | (see RTL) | Random digest initialization vectors, to be used in the [scrambling datapath](#scrambling-datapath). +`RndCnstRawUnlockToken` | (see RTL) | (see RTL) | Global RAW unlock token to be used for the first life cycle transition. See also [conditional life cycle transitions](../../../../ip/lc_ctrl/README.md#conditional-transitions). + + + + +The OTP controller contains various interfaces that connect to other comportable IPs within OpenTitan, and these are briefly explained further below. + +${"###"} EDN Interface + +The entropy request interface that talks to EDN in order to fetch fresh entropy for ephemeral SRAM scrambling key derivation and the LFSR counters for background checks. +It is comprised of the `otp_edn_o` and `otp_edn_i` signals and follows a req / ack protocol. + +See also [EDN documentation](../../../../ip/edn/README.md). + +${"###"} Power Manager Interface + +The power manager interface is comprised of three signals overall: an initialization request (`pwr_otp_i.otp_init`), an initialization done response (`pwr_otp_o.otp_done`) and an idle indicator (`pwr_otp_o.otp_idle`). + +The power manager asserts `pwr_otp_i.otp_init` in order to signal to the OTP controller that it can start initialization, and the OTP controller signals completion of the initialization sequence by asserting `pwr_otp_o.otp_done` (the signal will remain high until reset). + +The idle indication signal `pwr_otp_o.otp_idle` indicates whether there is an ongoing write operation in the Direct Access Interface (DAI) or Life Cycle Interface (LCI), and the power manager uses that indication to determine whether a power down request needs to be aborted. + +Since the power manager may run in a different clock domain, the `pwr_otp_i.otp_init` signal is synchronized within the OTP controller. +The power manager is responsible for synchronizing the `pwr_otp_o.otp_done` and `pwr_otp_o.otp_idle` signals. + +See also [power manager documentation](../../pwrmgr/README.md). + +${"###"} Life Cycle Interfaces + +The interface to the life cycle controller can be split into three functional sub-interfaces (vendor test, state output, state transitions), and these are explained in more detail below. +Note that the OTP and life cycle controllers are supposed to be in the same clock domain, hence no additional signal synchronization is required. +See also [life cycle controller documentation](../../../../ip/lc_ctrl/README.md) for more details. + +${"####"} Vendor Test Signals + +The `lc_otp_vendor_test_i` and `lc_otp_vendor_test_o` signals are connected to a 32bit control and a 32bit status register in the life cycle TAP, respectively, and are directly routed to the `prim_otp` wrapper. +These control and status signals may be used by the silicon creator to exercise the OTP programming smoke checks on the VENDOR_TEST partition. +The signals are gated with the life cycle state inside the life cycle controller such that they do not have any effect in production life cycle states. + +${"####"} State, Counter and Token Output + +After initialization, the life cycle partition contents, as well as the tokens and personalization status is output to the life cycle controller via the `otp_lc_data_o` struct. +The life cycle controller uses this information to determine the life cycle state, and steer the appropriate qualifier signals. +Some of these qualifier signals (`lc_dft_en_i`, `lc_creator_seed_sw_rw_en_i`, `lc_seed_hw_rd_en_i` and `lc_escalate_en_i`) are fed back to the OTP controller in order to ungate testing logic to the OTP macro; enable SW write access to the `SECRET2` partition; enable hardware read access to the root key in the `SECRET2` partition; or to push the OTP controller into escalation state. + +A possible sequence for the signals described is illustrated below. +```wavejson +{signal: [ + {name: 'clk_i', wave: 'p.................'}, + {name: 'otp_lc_data_o.valid', wave: '0.|...|.1.|...|...'}, + {name: 'otp_lc_data_o.state', wave: '03|...|...|...|...'}, + {name: 'otp_lc_data_o.count', wave: '03|...|...|...|...'}, + {}, + {name: 'otp_lc_data_o.test_unlock_token', wave: '0.|...|.3.|...|...'}, + {name: 'otp_lc_data_o.test_exit_token', wave: '0.|...|.3.|...|...'}, + {name: 'otp_lc_data_o.test_tokens_valid', wave: '0.|...|.3.|...|...'}, + {}, + {name: 'otp_lc_data_o.rma_token', wave: '0.|.3.|...|...|...'}, + {name: 'otp_lc_data_o.rma_token_valid', wave: '0.|.3.|...|...|...'}, + {}, + {name: 'otp_lc_data_o.secrets_valid', wave: '0.|.3.|...|...|...'}, + {}, + {name: 'lc_creator_seed_sw_rw_en_i', wave: '0.|...|...|.4.|...'}, + {name: 'lc_seed_hw_rd_en_i', wave: '0.|...|...|.4.|...'}, + {name: 'lc_dft_en_i', wave: '0.|...|...|.4.|...'}, + {}, + {name: 'lc_escalate_en_i', wave: '0.|...|...|...|.5.'}, +]} +``` + +Note that the `otp_lc_data_o.valid` signal is only asserted after the `LIFE_CYCLE`, `SECRET0` and `SECRET2` partitions have successfully initialized, since the life cycle collateral contains information from all three partitions. +The `otp_lc_data_o.test_tokens_valid` and `otp_lc_data_o.rma_token_valid` signals are multibit valid signals indicating whether the corresponding tokens are valid. +The ``otp_lc_data_o.secrets_valid`` signal is a multibit valid signal that is set to `lc_ctrl_pkg::On` iff the `SECRET2` partition containing the root keys has been locked with a digest. + + +${"####"} State Transitions + +In order to perform life cycle state transitions, the life cycle controller can present the new value of the life cycle state and counter via the programming interface as shown below: + +```wavejson +{signal: [ + {name: 'clk_i', wave: 'p.......'}, + {name: 'lc_otp_program_i.req', wave: '01.|..0.'}, + {name: 'lc_otp_program_i.state', wave: '03.|..0.'}, + {name: 'lc_otp_program_i.count', wave: '03.|..0.'}, + {name: 'lc_otp_program_o.ack', wave: '0..|.10.'}, + {name: 'lc_otp_program_o.err', wave: '0..|.40.'}, +]} +``` + +The request must remain asserted until the life cycle controller has responded. +An error is fatal and indicates that the OTP programming operation has failed. + +Note that the new state must not clear any bits that have already been programmed to OTP - i.e., the new state must be incrementally programmable on top of the previous state. +There are hence some implications on the life cycle encoding due to the ECC employed, see [life cycle state encoding](../../../../ip/lc_ctrl/README.md#life-cycle-manufacturing-state-encodings) for details. + +Note that the behavior of the `lc_otp_program_i.otp_test_ctrl` signal is vendor-specific, and hence the signal is set to `x` in the timing diagram above. +The purpose of this signal is to control vendor-specific test mechanisms, and its value will only be forwarded to the OTP macro in RAW, TEST_* and RMA states. +In all other life cycle states this signal will be clamped to zero. + +${"###"} Interface to Key Manager + +The interface to the key manager is a simple struct that outputs the CREATOR_ROOT_KEY_SHARE0 and CREATOR_ROOT_KEY_SHARE1 keys via `otp_keymgr_key_o` if these secrets have been provisioned and locked (via CREATOR_KEY_LOCK). +Otherwise, this signal is tied to a random netlist constant. + +Since the key manager may run in a different clock domain, key manager is responsible for synchronizing the `otp_keymgr_key_o` signals. + +${"###"} Interface to Flash Scrambler + +The interface to the FLASH scrambling device is a simple req/ack interface that provides the flash controller with the two 128bit keys for data and address scrambling. + +The keys can be requested as illustrated below: + +```wavejson +{signal: [ + {name: 'clk_i', wave: 'p...........'}, + {name: 'flash_otp_key_i.data_req', wave: '01.|..0.|...'}, + {name: 'flash_otp_key_i.addr_req', wave: '01.|....|..0'}, + {name: 'flash_otp_key_o.data_ack', wave: '0..|.10.|...'}, + {name: 'flash_otp_key_o.addr_ack', wave: '0..|....|.10'}, + {name: 'flash_otp_key_o.key', wave: '0..|.30.|.40'}, + {name: 'flash_otp_key_o.seed_valid', wave: '0..|.10.|.10'}, +]} +``` + +The keys are derived from the FLASH_DATA_KEY_SEED and FLASH_ADDR_KEY_SEED values stored in the `SECRET1` partition using the [scrambling primitive](#scrambling-datapath). +If the key seeds have not yet been provisioned, the keys are derived from all-zero constants, and the `flash_otp_key_o.seed_valid` signal will be set to 0 in the response. +The resulting scrambling key is still ephemeral (i.e., it is derived using entropy from CSRNG) and okay to be used. + +Note that the req/ack protocol runs on the OTP clock. +It is the task of the scrambling device to synchronize the handshake protocol by instantiating the `prim_sync_reqack.sv` primitive as shown below. + +![OTP Key Req Ack](../doc/otp_ctrl_key_req_ack.svg) + +Note that the key and nonce output signals on the OTP controller side are guaranteed to remain stable for at least 62 OTP clock cycles after the `ack` signal is pulsed high, because the derivation of a 64bit half-key takes at least two passes through the 31-cycle PRESENT primitive. +Hence, if the scrambling device clock is faster or in the same order of magnitude as the OTP clock, the data can be directly sampled upon assertion of `src_ack_o`. +If the scrambling device runs on a significantly slower clock than OTP, an additional register (as indicated with dashed grey lines in the figure) has to be added. + +${"###"} Interfaces to SRAM and OTBN Scramblers + +The interfaces to the SRAM and OTBN scrambling devices follow a req / ack protocol, where the scrambling device first requests a new ephemeral key by asserting the request channel (`sram_otp_key_i[*]`, `otbn_otp_key_i`). +The OTP controller then fetches entropy from EDN and derives an ephemeral key using the SRAM_DATA_KEY_SEED and the [PRESENT scrambling data path](#scrambling-datapath). +Finally, the OTP controller returns a fresh ephemeral key via the response channels (`sram_otp_key_o[*]`, `otbn_otp_key_o`), which complete the req / ack handshake. +The wave diagram below illustrates this process for the OTBN scrambling device. + +```wavejson +{signal: [ + {name: 'clk_i', wave: 'p.......'}, + {name: 'otbn_otp_key_i.req', wave: '01.|..0.'}, + {name: 'otbn_otp_key_o.ack', wave: '0..|.10.'}, + {name: 'otbn_otp_key_o.nonce', wave: '0..|.30.'}, + {name: 'otbn_otp_key_o.key', wave: '0..|.30.'}, + {name: 'otbn_otp_key_o.seed_valid', wave: '0..|.10.'}, +]} +``` + +If the key seeds have not yet been provisioned, the keys are derived from all-zero constants, and the `*.seed_valid` signal will be set to 0 in the response. +The resulting scrambling key is still ephemeral (i.e., it is derived using entropy from CSRNG) and okay to be used. +It should be noted that this mechanism requires the EDN and entropy distribution network to be operational, and a key derivation request will block if they are not. + +Note that the req/ack protocol runs on the OTP clock. +It is the task of the scrambling device to perform the synchronization as described in the previous subsection on the [flash scrambler interface](#interface-to-flash-scrambler). + +${"###"} Hardware Config Bits + +The bits of the HW_CFG* partitions are output via the `otp_ctrl_otp_broadcast_o` struct. +IPs that consume collateral stored in this partition shall connect to this struct via the topgen feature, and break out the appropriate bits by either accessing the correct index or using the struct fields. +These fields are autogenerated from the memory map items allocated to the HW_CFG* partitions, and the autogenerated struct type can be found in the `otp_ctrl_part_pkg.sv` package. +Note that it is the task of the receiving IP to synchronize these bits accordingly to the local clock. +For convenience, a valid bit is also available in that struct. +The valid bit indicates that the HW_CFG* partitions have initialized. + +${"##"} Parameter and Memory Map Changes after D3/V3 + +Note that all instantiation parameters can be changed without affecting D3/V3 status of the module. +Similarly, it is permissible to change the contents (partition size, adding and removing items) of the `CREATOR_SW_CFG`, `OWNER_SW_CFG` and `HW_CFG*` partitions without affecting D3 status. +Note however that partition size changes may affect V3 coverage metrics, hence if the size any of the above three partitions is changed, V3 needs to be re-assessed. diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_behavioral_model.svg b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_behavioral_model.svg similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_behavioral_model.svg rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_behavioral_model.svg diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_blockdiag.svg b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_blockdiag.svg similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_blockdiag.svg rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_blockdiag.svg diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_dai_fsm.svg b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_dai_fsm.svg similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_dai_fsm.svg rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_dai_fsm.svg diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_digest_mechanism.svg b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_digest_mechanism.svg similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_digest_mechanism.svg rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_digest_mechanism.svg diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_digests.md b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_digests.md similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_digests.md rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_digests.md diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_field_descriptions.md b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_field_descriptions.md similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_field_descriptions.md rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_field_descriptions.md diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_kdi_fsm.svg b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_kdi_fsm.svg similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_kdi_fsm.svg rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_kdi_fsm.svg diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_key_req_ack.svg b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_key_req_ack.svg similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_key_req_ack.svg rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_key_req_ack.svg diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_lci_fsm.svg b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_lci_fsm.svg similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_lci_fsm.svg rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_lci_fsm.svg diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_mmap.md b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_mmap.md similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_mmap.md rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_mmap.md diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_overview.svg b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_overview.svg similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_overview.svg rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_overview.svg diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_partitions.md b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_partitions.md similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_partitions.md rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_partitions.md diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_prim_otp.svg b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_prim_otp.svg similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_prim_otp.svg rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_prim_otp.svg diff --git a/hw/ip/otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg b/hw/ip_templates/otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg similarity index 100% rename from hw/ip/otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg rename to hw/ip_templates/otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg diff --git a/hw/ip/otp_ctrl/doc/programmers_guide.md b/hw/ip_templates/otp_ctrl/doc/programmers_guide.md similarity index 98% rename from hw/ip/otp_ctrl/doc/programmers_guide.md rename to hw/ip_templates/otp_ctrl/doc/programmers_guide.md index 58bf230e2404f3..a3d00735f7668c 100644 --- a/hw/ip/otp_ctrl/doc/programmers_guide.md +++ b/hw/ip_templates/otp_ctrl/doc/programmers_guide.md @@ -39,7 +39,7 @@ Hence the OTP controller performs a blank check and returns an error if a write It should be noted that the locked status of the partition holding the creator root key (i.e., the value of the [`SECRET2_DIGEST_0`](registers.md#secret2_digest)) determines the ID_STATUS of the device, which in turn determines SW accessibility of creator seed material in flash and OTP. That means that creator-seed-related collateral needs to be provisioned to Flash **before** the OTP digest lockdown mechanism is triggered, since otherwise accessibility to the corresponding flash region is lost. -See the [life cycle controller documentation](../../lc_ctrl/README.md#id-state-of-the-device) for more details. +See the [life cycle controller documentation](../../../../ip/lc_ctrl/README.md#id-state-of-the-device) for more details. ## Direct Access Interface @@ -181,7 +181,7 @@ For the hardware partitions, hardware calculates this digest and uses it for [ba Digest calculation can be triggered via the DAI. Finally, it should be noted that the RMA_TOKEN and CREATOR_ROOT_KEY_SHARE0 / CREATOR_ROOT_KEY_SHARE1 items can only be programmed when the device is in the DEV, PROD, PROD_END and RMA stages. -Please consult the [life cycle controller documentation](../../lc_ctrl/README.md) documentation for more information. +Please consult the [life cycle controller documentation](../../../../ip/lc_ctrl/README.md) documentation for more information. ## OTP Field Descriptions @@ -226,7 +226,7 @@ This is however rare and should not happen after fabrication testing. ## Device Interface Functions (DIFs) -- [Device Interface Functions](../../../../sw/device/lib/dif/dif_otp_ctrl.h) +- [Device Interface Functions](../../../../../sw/device/lib/dif/dif_otp_ctrl.h) # Additional Notes diff --git a/hw/ip_templates/otp_ctrl/doc/registers.md.tpl b/hw/ip_templates/otp_ctrl/doc/registers.md.tpl new file mode 100644 index 00000000000000..93d24bd23953dc --- /dev/null +++ b/hw/ip_templates/otp_ctrl/doc/registers.md.tpl @@ -0,0 +1,4 @@ +# Registers + + + diff --git a/hw/ip/otp_ctrl/doc/theory_of_operation.md b/hw/ip_templates/otp_ctrl/doc/theory_of_operation.md similarity index 96% rename from hw/ip/otp_ctrl/doc/theory_of_operation.md rename to hw/ip_templates/otp_ctrl/doc/theory_of_operation.md index 540b9e04bd486a..e3c5de4040a111 100644 --- a/hw/ip/otp_ctrl/doc/theory_of_operation.md +++ b/hw/ip_templates/otp_ctrl/doc/theory_of_operation.md @@ -5,7 +5,7 @@ The "front-end" contains the logical partitions that feed the hardware and softw The "back-end" represents the programming interface used by hardware and software components to stage the upcoming values. The diagram below illustrates this behavioral model. -![OTP Controller Block Diagram](../doc/otp_ctrl_behavioral_model.svg) +![OTP Controller Block Diagram](otp_ctrl_behavioral_model.svg) Note that the front-end contains both buffered and unbuffered partitions. Buffered partitions are sensed once per power cycle and their contents are stored in registers, whereas unbuffered partitions are read on-demand. @@ -69,7 +69,7 @@ As part of injecting the final firmware, the stock-keeping-unit-specific hardwar The life cycle partition is active throughout all stages and hence it is the **ONLY** partition that cannot be locked. After the device finishes provisioning and goes into production, it must retain the ability to transition back to RMA in case of unexpected failures. -In order to support this transition, the [life cycle state](../../lc_ctrl/README.md) and counters must always be update-able. +In order to support this transition, the [life cycle state](../../../../ip/lc_ctrl/README.md) and counters must always be update-able. ## Locking a Partition @@ -122,7 +122,7 @@ These values are stored in OTP as plaintext. Secret partitions contain data that are critical to security, for example FLASH scrambling keys, device root secret and unlock tokens. These values are stored scrambled in OTP, and are descrambled upon read. -The currently employed cipher is PRESENT, as it lends itself well to iterative decomposition, and it is a proven lightweight block cipher (see also [PRESENT Scrambling Primitive](../../prim/doc/prim_present.md). +The currently employed cipher is PRESENT, as it lends itself well to iterative decomposition, and it is a proven lightweight block cipher (see also [PRESENT Scrambling Primitive](../../../../ip/prim/doc/prim_present.md). The usage of a block cipher however implies that the secret partitions can only be written in 64bit chunks. Further, the contents of a particular secret partition are not readable by software once locked (other than the digest which must be always readable); while non-secret partitions are always readable unless read accessibility is explicitly removed by software. @@ -225,7 +225,7 @@ The functional interface is used to update all partitions except for life cycle. As mentioned previously, any updates made during the current power cycle are **NOT** reflected in the buffered partitions until the next reboot. The life cycle interface is used to update the life cycle state and transition counter only. -The commands are issued from the [life cycle controller](../../lc_ctrl/README.md), and similarly, successful or failed indications are also sent back to the life cycle controller. +The commands are issued from the [life cycle controller](../../../../ip/lc_ctrl/README.md), and similarly, successful or failed indications are also sent back to the life cycle controller. Similar to the functional interface, the life cycle controller allows only one update per power cycle, and after a requested transition reverts to an inert state until reboot. For more details on how the software programs the OTP, please refer to the [Programmer's Guide](programmers_guide.md)) further below. @@ -237,7 +237,7 @@ For more details on how the software programs the OTP, please refer to the [Prog The following is a high-level block diagram that illustrates everything that has been discussed. -![OTP Controller Block Diagram](../doc/otp_ctrl_blockdiag.svg) +![OTP Controller Block Diagram](otp_ctrl_blockdiag.svg) Each of the partitions P0-P7 has its [own controller FSM](#partition-implementations) that interacts with the OTP wrapper and the [scrambling datapath](#scrambling-datapath) to fulfill its tasks. The partitions expose the address ranges and access control information to the Direct Access Interface (DAI) in order to block accesses that go to locked address ranges. @@ -274,7 +274,7 @@ Since the life cycle partition is the only partition that needs live updates in- The life cycle state is hence encoded such that incremental updates to the state are always carried out at the granularity of a 16bit word. Further, the life cycle transition counter is encoded such that each stroke consumes a full 16bit word for the same reason. -See [life cycle controller documentation](../../lc_ctrl/README.md) for more details on the life cycle encoding. +See [life cycle controller documentation](../../../../ip/lc_ctrl/README.md) for more details on the life cycle encoding. ### Partition Controllers @@ -284,7 +284,7 @@ The corresponding controller FSMs are explained in more detail below. #### Unbuffered Partition -![Unbuffered Partition FSM](../doc/otp_ctrl_unbuf_part_fsm.svg) +![Unbuffered Partition FSM](otp_ctrl_unbuf_part_fsm.svg) As shown above, the unbuffered partition module has a relatively simple controller FSM that only reads out the digest value of the partition upon initialization, and then basically waits for TL-UL read transactions to its corresponding window in the CSR space. @@ -296,7 +296,7 @@ Note that unrecoverable [OTP errors](#generalized-open-source-interface), ECC fa #### Buffered Partition -![Buffered Partition FSM](../doc/otp_ctrl_buf_part_fsm.svg) +![Buffered Partition FSM](otp_ctrl_buf_part_fsm.svg) The controller FSM of the buffered partition module is more complex than the unbuffered counterpart, since it has to account for scrambling and digest calculation. @@ -318,11 +318,11 @@ In case of a mismatch, the buffered values are gated to their default, and an al Note that in case of unrecoverable OTP errors or ECC failures in the buffer registers, the partition controller FSM is moved into a terminal error state, which locks down all access through DAI and clamps the values that are broadcast in hardware to their defaults. External escalation via the `lc_escalate_en` signal will move the partition controller FSM into the terminal error state as well. -See [life cycle controller documentation](../../lc_ctrl/README.md) for more details. +See [life cycle controller documentation](../../../../ip/lc_ctrl/README.md) for more details. ### Direct Access Interface Control -![Direct Access Interface FSM](../doc/otp_ctrl_dai_fsm.svg) +![Direct Access Interface FSM](otp_ctrl_dai_fsm.svg) Upon reset release, the DAI controller first sends an initialization command to the OTP macro. Once the OTP macro becomes operational, an initialization request is sent to all partition controllers, which will read out and initialize the corresponding buffer registers. @@ -337,7 +337,7 @@ Also, the DAI consumes the read and write access information provided by the par ### Life Cycle Interface Control -![Life Cycle Interface FSM](../doc/otp_ctrl_lci_fsm.svg) +![Life Cycle Interface FSM](otp_ctrl_lci_fsm.svg) Upon reset release the LCI FSM waits until the OTP controller has initialized and the LCI gets enabled. Once it is in the idle state, life cycle state updates can be initiated via the life cycle interface as [described here](#state-transitions). @@ -346,7 +346,7 @@ In case of unrecoverable OTP errors, the FSM signals an error to the life cycle ### Key Derivation Interface -![Key Derivation Interface FSM](../doc/otp_ctrl_kdi_fsm.svg) +![Key Derivation Interface FSM](otp_ctrl_kdi_fsm.svg) Upon reset release the KDI FSM waits until the OTP controller has initialized and the KDI gets enabled. Once it is in the idle state, key derivation can be requested via the [flash](#interface-to-flash-scrambler) and [sram](#interface-to-sram-and-otbn-scramblers) interfaces. @@ -354,9 +354,9 @@ Based on which interface makes the request, the KDI controller will evaluate a v ### Scrambling Datapath -![OTP Digest Mechanism](../doc/otp_ctrl_digest_mechanism.svg) +![OTP Digest Mechanism](otp_ctrl_digest_mechanism.svg) -The scrambling datapath is built around an iterative implementation of the [PRESENT lightweight cipher](../../prim/doc/prim_present.md) that performs one round per cycle. +The scrambling datapath is built around an iterative implementation of the [PRESENT lightweight cipher](../../../../ip/prim/doc/prim_present.md) that performs one round per cycle. The datapath contains some additional multiplexing circuitry to enable the DAI, KDI and partition controllers to evaluate different functions with the same datapath. The algorithmic steps of these functions are explained in more detail below. @@ -412,7 +412,7 @@ This is behavior illustrated in the example below. ### Primitive Wrapper and FPGA Emulation -![OTP Wrapper Block Diagram](../doc/otp_ctrl_prim_otp.svg) +![OTP Wrapper Block Diagram](otp_ctrl_prim_otp.svg) The OTP IP is wrapped up in a primitive wrapper that exposes a TL-UL interface for testing purposes, and a generalized open-source interface for functional operation (described below). Any OTP redundancy mechanism like per-word ECC is assumed to be handled inside the wrapper, which means that the word width exposed as part of the generalized interface is the effective word width. diff --git a/hw/ip_templates/otp_ctrl/dv/README.md.tpl b/hw/ip_templates/otp_ctrl/dv/README.md.tpl new file mode 100644 index 00000000000000..86bb17a5d11df3 --- /dev/null +++ b/hw/ip_templates/otp_ctrl/dv/README.md.tpl @@ -0,0 +1,146 @@ +# OTP_CTRL DV document + +${"##"} Goals +* **DV** + * Verify all OTP_CTRL IP features by running dynamic simulations with a SV/UVM based testbench + * Develop and run all tests based on the [testplan](#testplan) below towards closing code and functional coverage on the IP and all of its sub-modules +* **FPV** + * Verify TileLink device protocol compliance with an SVA based testbench + +${"##"} Current status +* [Design & verification stage](../../../../README.md) + * [HW development stages](../../../../../doc/project_governance/development_stages.md) +* [Simulation results](https://reports.opentitan.org/hw/top_${topname}/ip_autogen/otp_ctrl/dv/latest/report.html) + +${"##"} Design features +For detailed information on OTP_CTRL design features, please see the [OTP_CTRL HW IP technical specification](../README.md). + +${"##"} Testbench architecture +OTP_CTRL testbench has been constructed based on the [CIP testbench architecture](../../../../dv/sv/cip_lib/README.md). + +${"###"} Block diagram +![Block diagram](./doc/tb.svg) + +${"###"} Top level testbench +Top level testbench is located at `hw/top_${topname}/ip_autogen/otp_ctrl/dv/tb.sv`. It instantiates the OTP_CTRL DUT module `hw/top_${topname}/ip_autogen/otp_ctrl/rtl/otp_ctrl.sv`. +In addition, it instantiates the following interfaces, connects them to the DUT and sets their handle into `uvm_config_db`: +* [Clock and reset interface](../../../../dv/sv/common_ifs/README.md) +* [TileLink host interface](../../../../dv/sv/tl_agent/README.md) +* OTP_CTRL IOs +* Interrupts ([`pins_if`](../../../../dv/sv/common_ifs/README.md)) +* Alerts ([`alert_esc_if`](../../../../dv/sv/alert_esc_agent/README.md)) + +${"###"} Common DV utility components +The following utilities provide generic helper tasks and functions to perform activities that are common across the project: +* [dv_utils_pkg](../../../../dv/sv/dv_utils/README.md) +* [csr_utils_pkg](../../../../dv/sv/csr_utils/README.md) + +${"###"} Global types & methods +All common types and methods defined at the package level can be found in +`otp_ctrl_env_pkg`. Some of them in use are: +```systemverilog + parameter uint SCRAMBLE_DATA_SIZE = 64; + parameter uint SCRAMBLE_KEY_SIZE = 128; + parameter uint NUM_ROUND = 31; + + typedef enum bit [2:0] { + OtpNoError, + OtpMacroError, + OtpMacroEccCorrError, + OtpMacroEccUncorrError, + OtpMacroWriteBlankError, + OtpAccessError, + OtpCheckFailError, + OtpFsmStateError + } otp_err_code_e; +``` + +${"###"} TL_agent +OTP_CTRL testbench instantiates (already handled in CIP base env) [tl_agent](../../../../dv/sv/tl_agent/README.md), which provides the ability to drive and independently monitor random traffic via TL host interface into OTP_CTRL device. + +${"###"} Alert_agents +OTP_CTRL testbench instantiates (already handled in CIP base env) two [alert_agents](../../../../dv/sv/alert_esc_agent/README.md): +fatal_check_alert and fatal_macro_alert. +The alert_agents provide the ability to drive and independently monitor alert handshakes via alert interfaces in OTP_CTRL device. + +${"###"} OTP_CTRL interface +OTP_CTRL design has specific inputs and outputs to communicate with other IPs including LC_CTRL, OTBN, SRAM, FLASH etc. +This interface is created to initialize, use simple task to drive, and use assertions to monitor these signals. + +${"###"} Memory backdoor interface +OTP_CTRL testbench binds design's non-volatile OTP memory with a [`mem_bkdr_util`](../../../../dv/sv/mem_bkdr_util/README.md), which supports read, write, and injection of ECC errors to design's OTP memory. + +${"###"} UVM RAL model +The OTP_CTRL RAL model is created with the [`ralgen`](../../../../dv/tools/ralgen/README.md) FuseSoC generator script automatically when the simulation is at the build stage. + +It can be created manually by invoking [`regtool`](../../../../../util/reggen/doc/setup_and_use.md). + +${"###"} Reference models +The OTP_CTRL's utilizes [PRESENT](../../../../ip/prim/doc/prim_present.md as the cipher to scramble and protect secrets. +Thus OTP_CTRL's scoreboard adopted PRESENT's C reference model, located under `hw/ip/prim/dv/prim_present/` folder, for encryption and decryption purpose. + +${"###"} Stimulus strategy +${"####"} Test sequences +All test sequences reside in `hw/top_${topname}/ip_autogen/otp_ctrl/dv/env/seq_lib`. +The `otp_ctrl_base_vseq` virtual sequence is extended from `cip_base_vseq` and serves as a starting point. +All test sequences are extended from `otp_ctrl_base_vseq`. +It provides commonly used handles, variables, functions and tasks that the test sequences can simple use / call. +Some of the most commonly used tasks / functions are as follows: +* dai_wr: This task triggers an OTP write sequence via the DAI interface. +* dai_rd: This task triggers an OTP read sequence via the DAI interface. +* trigger_checks: This task triggers a one-time OTP check and user can choose to trigger consistency check or integrity check. +* randomize_dai_addr: This task takes a DAI address as input and randomize its last two bits, because the last two bits should be ignored in design. + +${"####"} Functional coverage +To ensure high quality constrained random stimulus, it is necessary to develop a functional coverage model. +The following two files declared OTP_CTRL's covergroups: +- `dv/env/otp_ctrl_env_cov.sv` declares functional or CSR related covergroups. +The functional coverage is collected manually inside OTP_CTRL's scoreboard by invoking the `sample` function. +- `dv/cov/otp_ctrl_cov_if.sv` declares interface signal related covergroups. +The functional coverage is collected automatically when the sampled signal is active. + +${"###"} Self-checking strategy +${"####"} Scoreboard +The `otp_ctrl_scoreboard` is primarily used for end to end checking. +It creates the following analysis ports to retrieve the data monitored by corresponding interface agents: +* tl_a_chan_fifo: tl address channel +* tl_d_chan_fifo: tl data channel +* alert_fifos: alert handshakes +* sram_fifos: sram requests +* otbn_fifo: otbn request +* lc_prog_fifo: life cycle programming request +* lc_token_fifo: life cycle token request +* flash_addr_fifo: flash address request +* flash_data_fifo: flash data request +* edn_fifo: edn response to OTP_CTRL + +For all requests to OTP_CTRL as listed above, scoreboard has a corresponding task to process request, check OTP_CTRL's response value against encryption, and collect coverage. + +OTP_CTRL's scoreboard has an internal array `otp_a` that tracks OTP memory data. +Every successful OTP write operation will update this internal array, and every successful OTP read operation will check the readout value against this internal array. +Note that in design, secret partitions will go through a encryption before writing to the actually OTP memory, and will be decrypted upon a read request. +For the simplicity of this internal array, we will skip this procedure. +However, if scoreboard backdoor read any secret partitions, we will decrypt the data then write the decrypted data to the internal array. +For any operation that fails, the scoreboard will predict the status and err_code according to the failure type. +If the error can trigger alert, scoreboard will use `set_exp_alert` task to check if the alert is firing correctly. +If a HW digest operation is triggered by sequence, scoreboard will calculate digest value with partition data from its internal array and update the digest value. +According to design spec, scoreboard won't lock the partition and predict the digest value to digest registers until next power cycle. + +If a reset or lc_escalation_en is issued during an OTP_CTRL write operation, scoreboard cannot accurately predict how many bits have been programmed into OTP memory. +To avoid mismatches, scoreboard utilizes flags `dai_wr_ip` and `dai_digest_ip` to track otp write operations, and issue a backdoor read if the write operation is interrupted. + +${"####"} Assertions +* TLUL assertions: The `tb/otp_ctrl_bind.sv` binds the `tlul_assert` [assertions](../../../../ip/tlul/doc/TlulProtocolChecker.md) to the IP to ensure TileLink interface protocol compliance. +* Unknown checks on DUT outputs: The RTL has assertions to ensure all outputs are initialized to known values after coming out of reset. +* OTP_CTRL_IF assertions: This interface has assertions to ensure certain OTP_CTRL's outputs (such as: otp_broadcast_o, keymgr_key_o) are stable after OTP initialization. + +${"##"} Building and running tests +We are using our in-house developed [regression tool](../../../../../util/dvsim/README.md) for building and running our tests and regressions. +Please take a look at the link for detailed information on the usage, capabilities, features and known issues. +Here's how to run a smoke test: +```console +$ $REPO_TOP/util/dvsim/dvsim.py $REPO_TOP/hw/top_${topname}/ip_autogen/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson -i otp_ctrl_smoke +``` + +${"##"} Testplan +[Testplan](../data/otp_ctrl_testplan.hjson) diff --git a/hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cov.core.tpl b/hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cov.core.tpl new file mode 100644 index 00000000000000..b2d8d0aa610130 --- /dev/null +++ b/hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cov.core.tpl @@ -0,0 +1,28 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: ${instance_vlnv("lowrisc:dv:otp_ctrl_cov")} +description: "OTP_CTRL functional coverage and bind files" +virtual: + - lowrisc:ip_interfaces:otp_ctrl_cov + +filesets: + files_rtl: + depend: + - lowrisc:ip_interfaces:pwrmgr_pkg + - ${instance_vlnv("lowrisc:ip:otp_ctrl_top_specific_pkg")} + + files_dv: + depend: + - lowrisc:dv:dv_utils + files: + - otp_ctrl_cov_if.sv + - otp_ctrl_cov_bind.sv + file_type: systemVerilogSource + +targets: + default: + filesets: + - files_rtl + - files_dv diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_cov_bind.sv.tpl b/hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv.tpl similarity index 96% rename from hw/ip/otp_ctrl/data/otp_ctrl_cov_bind.sv.tpl rename to hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv.tpl index fa61630f926783..8f02c7c190a371 100644 --- a/hw/ip/otp_ctrl/data/otp_ctrl_cov_bind.sv.tpl +++ b/hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv.tpl @@ -4,7 +4,6 @@ // // Binds OTP_CTRL functional coverage interaface to the top level OTP_CTRL module. // -${gen_comment} <% from topgen.lib import Name @@ -68,13 +67,13 @@ module otp_ctrl_cov_bind; ); // Mubi internal coverage for buffered and unbuffered partitions. -% for part in otp_mmap.config["partitions"][:-1]: +% for part in otp_mmap["partitions"][:-1]: <% part_name = Name.from_snake_case(part["name"]) %>\ `PART_MUBI_COV(${part_name.as_snake_case()}, otp_ctrl_part_pkg::${part_name.as_camel_case()}Idx) % endfor // Mubi internal coverage for DAI interface access -% for part in otp_mmap.config["partitions"][:-1]: +% for part in otp_mmap["partitions"][:-1]: <% part_name = Name.from_snake_case(part["name"]) %>\ `DAI_MUBI_COV(${part_name.as_snake_case()}, otp_ctrl_part_pkg::${part_name.as_camel_case()}Idx) % endfor diff --git a/hw/ip/otp_ctrl/dv/cov/otp_ctrl_cov_fsm_unr_excl.el b/hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cov_fsm_unr_excl.el similarity index 100% rename from hw/ip/otp_ctrl/dv/cov/otp_ctrl_cov_fsm_unr_excl.el rename to hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cov_fsm_unr_excl.el diff --git a/hw/ip/otp_ctrl/dv/cov/otp_ctrl_cov_if.sv b/hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cov_if.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/cov/otp_ctrl_cov_if.sv rename to hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cov_if.sv diff --git a/hw/ip/otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el b/hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el similarity index 100% rename from hw/ip/otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el rename to hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el diff --git a/hw/ip/otp_ctrl/dv/cov/otp_ctrl_cover.cfg b/hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cover.cfg similarity index 100% rename from hw/ip/otp_ctrl/dv/cov/otp_ctrl_cover.cfg rename to hw/ip_templates/otp_ctrl/dv/cov/otp_ctrl_cover.cfg diff --git a/hw/ip/otp_ctrl/dv/doc/tb.svg b/hw/ip_templates/otp_ctrl/dv/doc/tb.svg similarity index 100% rename from hw/ip/otp_ctrl/dv/doc/tb.svg rename to hw/ip_templates/otp_ctrl/dv/doc/tb.svg diff --git a/hw/ip/otp_ctrl/dv/env/otp_ctrl_ast_inputs_cfg.sv b/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_ast_inputs_cfg.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/otp_ctrl_ast_inputs_cfg.sv rename to hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_ast_inputs_cfg.sv diff --git a/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_env.core.tpl b/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_env.core.tpl new file mode 100644 index 00000000000000..a03ec7bcc37e33 --- /dev/null +++ b/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_env.core.tpl @@ -0,0 +1,62 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: ${instance_vlnv("lowrisc:dv:otp_ctrl_env:0.1")} +description: "OTP_CTRL DV UVM environment" +virtual: + - lowrisc:ip_interfaces:otp_ctrl_env + +filesets: + files_dv: + depend: + - lowrisc:dv:ralgen + - lowrisc:dv:cip_lib + - lowrisc:dv:mem_bkdr_util + - lowrisc:dv:crypto_dpi_present + - lowrisc:dv:lc_ctrl_dv_utils + files: + - otp_ctrl_env_pkg.sv + - otp_ctrl_if.sv + - otp_ctrl_ast_inputs_cfg.sv: {is_include_file: true} + - otp_ctrl_env_cfg.sv: {is_include_file: true} + - otp_ctrl_env_cov.sv: {is_include_file: true} + - otp_ctrl_virtual_sequencer.sv: {is_include_file: true} + - otp_ctrl_scoreboard.sv: {is_include_file: true} + - otp_ctrl_env.sv: {is_include_file: true} + - seq_lib/otp_ctrl_vseq_list.sv: {is_include_file: true} + - seq_lib/otp_ctrl_callback_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_base_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_common_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_wake_up_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_smoke_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_partition_walk_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_low_freq_read_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_init_fail_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_dai_lock_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_dai_errs_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_macro_errs_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_background_chks_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_check_fail_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_parallel_base_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_regwen_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_parallel_key_req_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_parallel_lc_req_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_parallel_lc_esc_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_test_access_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_stress_all_vseq.sv: {is_include_file: true} + file_type: systemVerilogSource + +generate: + ral: + generator: ralgen + parameters: + name: otp_ctrl + ip_hjson: ../../data/otp_ctrl.hjson + +targets: + default: + filesets: + - files_dv + generate: + - ral diff --git a/hw/ip/otp_ctrl/dv/env/otp_ctrl_env.sv b/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_env.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/otp_ctrl_env.sv rename to hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_env.sv diff --git a/hw/ip/otp_ctrl/dv/env/otp_ctrl_env_cfg.sv b/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_env_cfg.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/otp_ctrl_env_cfg.sv rename to hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_env_cfg.sv diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_env_cov.sv.tpl b/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_env_cov.sv.tpl similarity index 97% rename from hw/ip/otp_ctrl/data/otp_ctrl_env_cov.sv.tpl rename to hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_env_cov.sv.tpl index f20736c346159b..42cfc86ef9be77 100644 --- a/hw/ip/otp_ctrl/data/otp_ctrl_env_cov.sv.tpl +++ b/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_env_cov.sv.tpl @@ -1,7 +1,6 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -${gen_comment} /** * Covergoups that are dependent on run-time parameters that may be available * only in build_phase can be defined here @@ -10,16 +9,16 @@ ${gen_comment} <% from topgen.lib import Name -parts_without_lc = [part for part in otp_mmap.config["partitions"] if +parts_without_lc = [part for part in otp_mmap["partitions"] if part["variant"] in ["Buffered", "Unbuffered"]] -unbuffered_parts = [part for part in otp_mmap.config["partitions"] if +unbuffered_parts = [part for part in otp_mmap["partitions"] if part["variant"] == "Unbuffered"] unbuffered_parts_with_digest = [part for part in unbuffered_parts if (part["sw_digest"] or part["hw_digest"])] -buffered_parts = [part for part in otp_mmap.config["partitions"] if +buffered_parts = [part for part in otp_mmap["partitions"] if part["variant"] == "Buffered"] buffered_nonsecret_parts_with_digest = [part for part in buffered_parts if @@ -30,7 +29,7 @@ buffered_secret_parts_with_digest = [part for part in buffered_parts if (part["sw_digest"] or part["hw_digest"]) and part["secret"]] ## Partitions + LCI + DAI -num_err_code = len(otp_mmap.config["partitions"]) + 2 +num_err_code = len(otp_mmap["partitions"]) + 2 %>\ class otp_ctrl_unbuf_err_code_cg_wrap; // Unbuffered partition can use TLUL interface to read out but cannot write, thus error_code does @@ -234,7 +233,7 @@ class otp_ctrl_env_cov extends cip_base_env_cov #(.CFG_T(otp_ctrl_env_cfg)); illegal_bins illegal_err = default; } partition: coverpoint part_idx { -% for part in otp_mmap.config["partitions"]: +% for part in otp_mmap["partitions"]: <% part_name = Name.from_snake_case(part["name"]) %>\ bins ${part["name"].lower()} = {${part_name.as_camel_case()}Idx}; % endfor @@ -334,7 +333,7 @@ class otp_ctrl_env_cov extends cip_base_env_cov #(.CFG_T(otp_ctrl_env_cfg)); function void collect_err_code_cov(int part_idx, bit [TL_DW-1:0] val, int access_part_idx = DaiIdx); case (part_idx) -% for part in otp_mmap.config["partitions"]: +% for part in otp_mmap["partitions"]: <% part_name = Name.from_snake_case(part["name"]) %>\ Otp${part_name.as_camel_case()}ErrIdx: begin % if part in unbuffered_parts: diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_env_pkg.sv.tpl b/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_env_pkg.sv.tpl similarity index 97% rename from hw/ip/otp_ctrl/data/otp_ctrl_env_pkg.sv.tpl rename to hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_env_pkg.sv.tpl index 08c86cb2019374..fc8d6f4d747d80 100644 --- a/hw/ip/otp_ctrl/data/otp_ctrl_env_pkg.sv.tpl +++ b/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_env_pkg.sv.tpl @@ -1,13 +1,12 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -${gen_comment} <% from topgen.lib import Name -parts_without_lc = [part for part in otp_mmap.config["partitions"] if +parts_without_lc = [part for part in otp_mmap["partitions"] if part["variant"] in ["Buffered", "Unbuffered"]] -otp_size_as_hex_text = f'{(2 ** otp_mmap.config["otp"]["byte_addr_width"]):x}' +otp_size_as_hex_text = f'{(2 ** otp_mmap["otp"]["byte_addr_width"]):x}' %>\ package otp_ctrl_env_pkg; // dep packages @@ -25,6 +24,7 @@ package otp_ctrl_env_pkg; import otp_ctrl_reg_pkg::*; import otp_ctrl_pkg::*; import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; import lc_ctrl_pkg::*; import lc_ctrl_state_pkg::*; import lc_ctrl_dv_utils_pkg::*; @@ -110,7 +110,7 @@ package otp_ctrl_env_pkg; } otp_intr_e; typedef enum bit [5:0] { -% for part in otp_mmap.config["partitions"]: +% for part in otp_mmap["partitions"]: <% part_name = Name.from_snake_case(part["name"]) part_name_camel = part_name.as_camel_case() diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_if.sv.tpl b/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_if.sv.tpl similarity index 98% rename from hw/ip/otp_ctrl/data/otp_ctrl_if.sv.tpl rename to hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_if.sv.tpl index 4fac06be38100e..f5c93f1afae9e7 100644 --- a/hw/ip/otp_ctrl/data/otp_ctrl_if.sv.tpl +++ b/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_if.sv.tpl @@ -1,15 +1,14 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -${gen_comment} <% from topgen.lib import Name -unbuf_parts_with_digest = [part for part in otp_mmap.config["partitions"] if +unbuf_parts_with_digest = [part for part in otp_mmap["partitions"] if part["variant"] == "Unbuffered" and (part["sw_digest"] or part["hw_digest"])] -parts_without_lc = [part for part in otp_mmap.config["partitions"] if +parts_without_lc = [part for part in otp_mmap["partitions"] if part["variant"] in ["Buffered", "Unbuffered"]] -buf_parts_without_lc = [part for part in otp_mmap.config["partitions"] if +buf_parts_without_lc = [part for part in otp_mmap["partitions"] if part["variant"] == "Buffered"] %>\ // This interface collect the broadcast output data from OTP, diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_scoreboard.sv.tpl b/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_scoreboard.sv.tpl similarity index 99% rename from hw/ip/otp_ctrl/data/otp_ctrl_scoreboard.sv.tpl rename to hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_scoreboard.sv.tpl index 787611dc40d972..7d0674119d539e 100644 --- a/hw/ip/otp_ctrl/data/otp_ctrl_scoreboard.sv.tpl +++ b/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_scoreboard.sv.tpl @@ -1,20 +1,19 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -${gen_comment} <% from topgen.lib import Name -read_locked_csr_parts = [part for part in otp_mmap.config["partitions"] if +read_locked_csr_parts = [part for part in otp_mmap["partitions"] if part["read_lock"] == "CSR"] -write_locked_digest_parts = [part for part in otp_mmap.config["partitions"] if +write_locked_digest_parts = [part for part in otp_mmap["partitions"] if part["write_lock"] == "Digest"] -buf_parts_without_lc = [part for part in otp_mmap.config["partitions"] if +buf_parts_without_lc = [part for part in otp_mmap["partitions"] if part["variant"] == "Buffered"] -secret_parts = [part for part in otp_mmap.config["partitions"] if +secret_parts = [part for part in otp_mmap["partitions"] if part["secret"]] ## Partitions + LCI + DAI -num_err_code = len(otp_mmap.config["partitions"]) + 2 +num_err_code = len(otp_mmap["partitions"]) + 2 %>\ class otp_ctrl_scoreboard #(type CFG_T = otp_ctrl_env_cfg) extends cip_base_scoreboard #( @@ -221,7 +220,7 @@ class otp_ctrl_scoreboard #(type CFG_T = otp_ctrl_env_cfg) // Otp_keymgr outputs creator and owner keys from secret partitions. // Depends on lc_seed_hw_rd_en_i, it will output the real keys or a constant exp_keymgr_data = '0; -% for part in otp_mmap.config["partitions"]: +% for part in otp_mmap["partitions"]: <% part_name = Name.from_snake_case(part["name"]) part_name_camel = part_name.as_camel_case() diff --git a/hw/ip/otp_ctrl/dv/env/otp_ctrl_virtual_sequencer.sv b/hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_virtual_sequencer.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/otp_ctrl_virtual_sequencer.sv rename to hw/ip_templates/otp_ctrl/dv/env/otp_ctrl_virtual_sequencer.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_background_chks_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_background_chks_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_background_chks_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_background_chks_vseq.sv diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_base_vseq.sv.tpl b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_base_vseq.sv.tpl similarity index 97% rename from hw/ip/otp_ctrl/data/otp_ctrl_base_vseq.sv.tpl rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_base_vseq.sv.tpl index 1728c95812e581..887eae65a99ead 100644 --- a/hw/ip/otp_ctrl/data/otp_ctrl_base_vseq.sv.tpl +++ b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_base_vseq.sv.tpl @@ -1,21 +1,19 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -${gen_comment} <% from topgen.lib import Name -unbuf_parts_with_digest = [part for part in otp_mmap.config["partitions"] if +unbuf_parts_with_digest = [part for part in otp_mmap["partitions"] if part["variant"] == "Unbuffered" and (part["sw_digest"] or part["hw_digest"])] -parts_with_digest = [part for part in otp_mmap.config["partitions"] if +parts_with_digest = [part for part in otp_mmap["partitions"] if (part["sw_digest"] or part["hw_digest"])] -read_locked_csr_parts = [part for part in otp_mmap.config["partitions"] if +read_locked_csr_parts = [part for part in otp_mmap["partitions"] if part["read_lock"] == "CSR"] -write_locked_digest_parts = [part for part in otp_mmap.config["partitions"] if +write_locked_digest_parts = [part for part in otp_mmap["partitions"] if part["write_lock"] == "Digest"] -secret_parts = [part for part in otp_mmap.config["partitions"] if - part["secret"]] +secret_parts = [part for part in otp_mmap["partitions"] if part["secret"]] %>\ class otp_ctrl_base_vseq extends cip_base_vseq #( .RAL_T (otp_ctrl_core_reg_block), @@ -179,7 +177,7 @@ class otp_ctrl_base_vseq extends cip_base_vseq #( if (is_secret(addr) || is_sw_digest(addr)) csr_wr(ral.direct_access_wdata[1], wdata1); do_otp_wr = 1; - csr_wr(ral.direct_access_cmd, int'(otp_ctrl_pkg::DaiWrite)); + csr_wr(ral.direct_access_cmd, int'(otp_ctrl_top_specific_pkg::DaiWrite)); `uvm_info(`gfn, $sformatf("DAI write, address %0h, data0 %0h data1 %0h, is_secret = %0b", addr, wdata0, wdata1, is_secret(addr)), UVM_DEBUG) @@ -207,7 +205,7 @@ class otp_ctrl_base_vseq extends cip_base_vseq #( csr_wr(ral.direct_access_address, addr); do_otp_rd = 1; - csr_wr(ral.direct_access_cmd, int'(otp_ctrl_pkg::DaiRead)); + csr_wr(ral.direct_access_cmd, int'(otp_ctrl_top_specific_pkg::DaiRead)); if (cfg.zero_delays && is_valid_dai_op && cfg.otp_ctrl_vif.lc_escalate_en_i == lc_ctrl_pkg::Off) begin @@ -238,7 +236,7 @@ class otp_ctrl_base_vseq extends cip_base_vseq #( virtual task cal_digest(int part_idx); bit [TL_DW-1:0] val; csr_wr(ral.direct_access_address, PART_BASE_ADDRS[part_idx]); - csr_wr(ral.direct_access_cmd, otp_ctrl_pkg::DaiDigest); + csr_wr(ral.direct_access_cmd, otp_ctrl_top_specific_pkg::DaiDigest); if (cfg.zero_delays && is_valid_dai_op && cfg.otp_ctrl_vif.lc_escalate_en_i == lc_ctrl_pkg::Off) begin diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_callback_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_callback_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_callback_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_callback_vseq.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_check_fail_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_check_fail_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_check_fail_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_check_fail_vseq.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_common_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_common_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_common_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_common_vseq.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_dai_lock_vseq.sv.tpl b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv.tpl similarity index 94% rename from hw/ip/otp_ctrl/data/otp_ctrl_dai_lock_vseq.sv.tpl rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv.tpl index 05e81f1ab20f9c..d6a159173fd3c8 100644 --- a/hw/ip/otp_ctrl/data/otp_ctrl_dai_lock_vseq.sv.tpl +++ b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv.tpl @@ -1,14 +1,13 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -${gen_comment} <% from topgen.lib import Name -parts_without_lc = [part for part in otp_mmap.config["partitions"] if +parts_without_lc = [part for part in otp_mmap["partitions"] if part["variant"] in ["Buffered", "Unbuffered"]] -parts_with_digest = [part for part in otp_mmap.config["partitions"] if +parts_with_digest = [part for part in otp_mmap["partitions"] if (part["sw_digest"] or part["hw_digest"])] %>\ // otp_ctrl_dai_lock_vseq is developed to read/write lock DAI interface by partitions, and request diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_low_freq_read_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_low_freq_read_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_low_freq_read_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_low_freq_read_vseq.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_macro_errs_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_macro_errs_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_macro_errs_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_macro_errs_vseq.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_base_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_base_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_base_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_base_vseq.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_key_req_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_key_req_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_key_req_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_key_req_vseq.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_lc_esc_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_lc_esc_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_lc_esc_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_lc_esc_vseq.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_lc_req_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_lc_req_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_lc_req_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_parallel_lc_req_vseq.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_partition_walk_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_partition_walk_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_partition_walk_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_partition_walk_vseq.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_regwen_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_regwen_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_regwen_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_regwen_vseq.sv diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_smoke_vseq.sv.tpl b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_smoke_vseq.sv.tpl similarity index 99% rename from hw/ip/otp_ctrl/data/otp_ctrl_smoke_vseq.sv.tpl rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_smoke_vseq.sv.tpl index 904ff45494ba6d..558e22a2f3f366 100644 --- a/hw/ip/otp_ctrl/data/otp_ctrl_smoke_vseq.sv.tpl +++ b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_smoke_vseq.sv.tpl @@ -1,7 +1,6 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -${gen_comment} <% from topgen.lib import Name %>\ @@ -32,7 +31,7 @@ class otp_ctrl_smoke_vseq extends otp_ctrl_base_vseq; constraint partition_index_c {part_idx inside {[0:LifeCycleIdx-1]};} constraint dai_wr_legal_addr_c { -% for part in otp_mmap.config["partitions"]: +% for part in otp_mmap["partitions"]: <% part_name = Name.from_snake_case(part["name"]) part_name_camel = part_name.as_camel_case() diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_stress_all_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_stress_all_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_stress_all_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_stress_all_vseq.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_test_access_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_test_access_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_test_access_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_test_access_vseq.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_vseq_list.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_vseq_list.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_vseq_list.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_vseq_list.sv diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_wake_up_vseq.sv b/hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_wake_up_vseq.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_wake_up_vseq.sv rename to hw/ip_templates/otp_ctrl/dv/env/seq_lib/otp_ctrl_wake_up_vseq.sv diff --git a/hw/ip_templates/otp_ctrl/dv/otp_ctrl_sim.core.tpl b/hw/ip_templates/otp_ctrl/dv/otp_ctrl_sim.core.tpl new file mode 100644 index 00000000000000..47b3c112bcef66 --- /dev/null +++ b/hw/ip_templates/otp_ctrl/dv/otp_ctrl_sim.core.tpl @@ -0,0 +1,37 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: ${instance_vlnv("lowrisc:dv:otp_ctrl_sim:0.1")} +description: "OTP_CTRL DV sim target" +virtual: + - lowrisc:ip_interfaces:otp_ctrl_sim + +filesets: + files_rtl: + depend: + - ${instance_vlnv("lowrisc:ip:otp_ctrl")} + + files_dv: + depend: + - lowrisc:dv:mem_bkdr_util + - ${instance_vlnv("lowrisc:dv:otp_ctrl_test")} + - ${instance_vlnv("lowrisc:dv:otp_ctrl_sva")} + - ${instance_vlnv("lowrisc:dv:otp_ctrl_cov")} + files: + - tb.sv + file_type: systemVerilogSource + +targets: + default: &default_target + toplevel: tb + filesets: + - files_rtl + - files_dv + + sim: + <<: *default_target + default_tool: vcs + + lint: + <<: *default_target diff --git a/hw/ip_templates/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson.tpl b/hw/ip_templates/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson.tpl new file mode 100644 index 00000000000000..4702439c5259a1 --- /dev/null +++ b/hw/ip_templates/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson.tpl @@ -0,0 +1,179 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +{ + // Name of the sim cfg - typically same as the name of the DUT. + name: otp_ctrl + + // Top level dut name (sv module). + dut: otp_ctrl + + // Top level testbench name (sv module). + tb: tb + + // Simulator used to sign off this block + tool: vcs + + // Fusesoc core file used for building the file list. + fusesoc_core: ${instance_vlnv("lowrisc:dv:otp_ctrl_sim:0.1")} + + // Testplan hjson file. + testplan: "{self_dir}/../data/otp_ctrl_testplan.hjson" + + // RAL spec - used to generate the RAL model. + ral_spec: "{self_dir}/../data/otp_ctrl.hjson" + + // Import additional common sim cfg files. + import_cfgs: [// Project wide common sim cfg file + "{proj_root}/hw/dv/tools/dvsim/common_sim_cfg.hjson", + // Config files to get the correct flags for crypto_dpi_prince + "{proj_root}/hw/ip/prim/dv/prim_prince/crypto_dpi_prince/crypto_dpi_prince_sim_opts.hjson", + // Common CIP test lists + "{proj_root}/hw/dv/tools/dvsim/tests/csr_tests.hjson", + "{proj_root}/hw/dv/tools/dvsim/tests/mem_tests.hjson", + "{proj_root}/hw/dv/tools/dvsim/tests/intr_test.hjson", + "{proj_root}/hw/dv/tools/dvsim/tests/alert_test.hjson", + "{proj_root}/hw/dv/tools/dvsim/tests/tl_access_tests.hjson", + "{proj_root}/hw/dv/tools/dvsim/tests/sec_cm_tests.hjson", + "{proj_root}/hw/dv/tools/dvsim/tests/stress_tests.hjson"] + + en_build_modes: ["{tool}_crypto_dpi_prince_build_opts"] + + build_modes: [ + // Sim mode that enables build randomization. See the `build_seed` mode + // defined in `hw/dv/tools/dvsim/common_modes.hjson` for more details. + { + name: build_seed + pre_build_cmds: ["cd {proj_root} && ./util/design/gen-otp-mmap.py --seed {seed} --topname ${topname}"] + is_sim_mode: 1 + } + ] + + // Add additional tops for simulation. + sim_tops: ["otp_ctrl_bind", "otp_ctrl_cov_bind", + "sec_cm_prim_sparse_fsm_flop_bind", + "sec_cm_prim_count_bind", + "sec_cm_prim_double_lfsr_bind", + "sec_cm_prim_onehot_check_bind"] + + // Default iterations for all tests - each test entry can override this. + reseed: 50 + + // Add OTP_CTRL specific exclusion files. + vcs_cov_excl_files: ["{self_dir}/cov/otp_ctrl_cov_unr_excl.el", + "{self_dir}/cov/otp_ctrl_cov_fsm_unr_excl.el"] + + overrides: [ + { + name: default_vcs_cov_cfg_file + value: "-cm_hier {dv_root}/tools/vcs/cover.cfg+{dv_root}/tools/vcs/common_cov_excl.cfg+{self_dir}/cov/otp_ctrl_cover.cfg" + } + ] + + // Default UVM test and seq class name. + uvm_test: otp_ctrl_base_test + uvm_test_seq: otp_ctrl_base_vseq + + run_opts: ["+cdc_instrumentation_enabled=1"] + + // List of test specifications. + tests: [ + { + name: otp_ctrl_wake_up + uvm_test_seq: otp_ctrl_wake_up_vseq + run_opts: ["+en_scb=0"] + reseed: 1 + } + + { + name: otp_ctrl_smoke + uvm_test_seq: otp_ctrl_smoke_vseq + } + + { + name: otp_ctrl_partition_walk + uvm_test_seq: otp_ctrl_partition_walk_vseq + reseed: 1 + } + + { + name: otp_ctrl_low_freq_read + uvm_test_seq: otp_ctrl_low_freq_read_vseq + reseed: 1 + } + + { + name: otp_ctrl_init_fail + uvm_test_seq: otp_ctrl_init_fail_vseq + reseed: 300 + } + { + name: otp_ctrl_background_chks + uvm_test_seq: otp_ctrl_background_chks_vseq + reseed: 10 + } + + { + name: otp_ctrl_parallel_lc_req + uvm_test_seq: otp_ctrl_parallel_lc_req_vseq + run_opts: ["+zero_delays=1"] + } + + { + name: otp_ctrl_parallel_lc_esc + uvm_test_seq: otp_ctrl_parallel_lc_esc_vseq + reseed: 200 + } + + { + name: otp_ctrl_dai_lock + uvm_test_seq: otp_ctrl_dai_lock_vseq + } + + { + name: otp_ctrl_dai_errs + uvm_test_seq: otp_ctrl_dai_errs_vseq + } + + { + name: otp_ctrl_check_fail + uvm_test_seq: otp_ctrl_check_fail_vseq + } + + { + name: otp_ctrl_macro_errs + uvm_test_seq: otp_ctrl_macro_errs_vseq + } + + { + name: otp_ctrl_parallel_key_req + uvm_test_seq: otp_ctrl_parallel_key_req_vseq + } + + { + name: otp_ctrl_regwen + uvm_test_seq: otp_ctrl_regwen_vseq + // This test is to check reg programming is gated when direct_access_regwen=0 + // Thus this test is timing sensitive + run_opts: ["+zero_delays=1"] + } + + { + name: otp_ctrl_test_access + uvm_test_seq: otp_ctrl_test_access_vseq + } + + { + name: "{name}_stress_all_with_rand_reset" + reseed: 100 + } + ] + + // List of regressions. + regressions: [ + { + name: smoke + tests: ["otp_ctrl_smoke"] + } + ] +} diff --git a/hw/ip/otp_ctrl/dv/sva/otp_ctrl_bind.sv b/hw/ip_templates/otp_ctrl/dv/sva/otp_ctrl_bind.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/sva/otp_ctrl_bind.sv rename to hw/ip_templates/otp_ctrl/dv/sva/otp_ctrl_bind.sv diff --git a/hw/ip_templates/otp_ctrl/dv/sva/otp_ctrl_sva.core.tpl b/hw/ip_templates/otp_ctrl/dv/sva/otp_ctrl_sva.core.tpl new file mode 100644 index 00000000000000..4480e00bd2d1df --- /dev/null +++ b/hw/ip_templates/otp_ctrl/dv/sva/otp_ctrl_sva.core.tpl @@ -0,0 +1,41 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: ${instance_vlnv("lowrisc:dv:otp_ctrl_sva:0.1")} +description: "OTP_CTRL assertion modules and bind file." +virtual: + - lowrisc:ip_interfaces:otp_ctrl_sva + +filesets: + files_dv: + depend: + - lowrisc:tlul:headers + - lowrisc:fpv:csr_assert_gen + files: + - otp_ctrl_bind.sv + file_type: systemVerilogSource + + files_formal: + depend: + - ${instance_vlnv("lowrisc:ip:otp_ctrl")} + +generate: + csr_assert_gen: + generator: csr_assert_gen + parameters: + spec: ../../data/otp_ctrl.hjson + +targets: + default: &default_target + filesets: + - files_dv + generate: + - csr_assert_gen + + formal: + <<: *default_target + filesets: + - files_formal + - files_dv + toplevel: otp_ctrl diff --git a/hw/ip/otp_ctrl/dv/tb.sv b/hw/ip_templates/otp_ctrl/dv/tb.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/tb.sv rename to hw/ip_templates/otp_ctrl/dv/tb.sv diff --git a/hw/ip/otp_ctrl/dv/tests/otp_ctrl_base_test.sv b/hw/ip_templates/otp_ctrl/dv/tests/otp_ctrl_base_test.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/tests/otp_ctrl_base_test.sv rename to hw/ip_templates/otp_ctrl/dv/tests/otp_ctrl_base_test.sv diff --git a/hw/ip_templates/otp_ctrl/dv/tests/otp_ctrl_test.core.tpl b/hw/ip_templates/otp_ctrl/dv/tests/otp_ctrl_test.core.tpl new file mode 100644 index 00000000000000..04c039268326ff --- /dev/null +++ b/hw/ip_templates/otp_ctrl/dv/tests/otp_ctrl_test.core.tpl @@ -0,0 +1,22 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: ${instance_vlnv("lowrisc:dv:otp_ctrl_test:0.1")} +description: "OTP_CTRL DV UVM test" +virtual: + - lowrisc:ip_interfaces:otp_ctrl_test + +filesets: + files_dv: + depend: + - ${instance_vlnv("lowrisc:dv:otp_ctrl_env")} + files: + - otp_ctrl_test_pkg.sv + - otp_ctrl_base_test.sv: {is_include_file: true} + file_type: systemVerilogSource + +targets: + default: + filesets: + - files_dv diff --git a/hw/ip/otp_ctrl/dv/tests/otp_ctrl_test_pkg.sv b/hw/ip_templates/otp_ctrl/dv/tests/otp_ctrl_test_pkg.sv similarity index 100% rename from hw/ip/otp_ctrl/dv/tests/otp_ctrl_test_pkg.sv rename to hw/ip_templates/otp_ctrl/dv/tests/otp_ctrl_test_pkg.sv diff --git a/hw/ip/otp_ctrl/lint/otp_ctrl.vbl b/hw/ip_templates/otp_ctrl/lint/otp_ctrl.vbl similarity index 100% rename from hw/ip/otp_ctrl/lint/otp_ctrl.vbl rename to hw/ip_templates/otp_ctrl/lint/otp_ctrl.vbl diff --git a/hw/ip/otp_ctrl/lint/otp_ctrl.vlt b/hw/ip_templates/otp_ctrl/lint/otp_ctrl.vlt similarity index 100% rename from hw/ip/otp_ctrl/lint/otp_ctrl.vlt rename to hw/ip_templates/otp_ctrl/lint/otp_ctrl.vlt diff --git a/hw/ip/otp_ctrl/lint/otp_ctrl.waiver b/hw/ip_templates/otp_ctrl/lint/otp_ctrl.waiver similarity index 100% rename from hw/ip/otp_ctrl/lint/otp_ctrl.waiver rename to hw/ip_templates/otp_ctrl/lint/otp_ctrl.waiver diff --git a/hw/ip_templates/otp_ctrl/lint/otp_ctrl_pkg.vbl b/hw/ip_templates/otp_ctrl/lint/otp_ctrl_pkg.vbl new file mode 100644 index 00000000000000..709cf2bb305769 --- /dev/null +++ b/hw/ip_templates/otp_ctrl/lint/otp_ctrl_pkg.vbl @@ -0,0 +1,5 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# verible waiver file for OTP controller generic package diff --git a/hw/ip/otp_ctrl/lint/otp_ctrl_pkg.vlt b/hw/ip_templates/otp_ctrl/lint/otp_ctrl_pkg.vlt similarity index 77% rename from hw/ip/otp_ctrl/lint/otp_ctrl_pkg.vlt rename to hw/ip_templates/otp_ctrl/lint/otp_ctrl_pkg.vlt index 0df101fc3539da..fefa0f75baad1d 100644 --- a/hw/ip/otp_ctrl/lint/otp_ctrl_pkg.vlt +++ b/hw/ip_templates/otp_ctrl/lint/otp_ctrl_pkg.vlt @@ -2,4 +2,4 @@ // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 // -// waiver file for OTP controller package +// waiver file for OTP controller generic package diff --git a/hw/ip_templates/otp_ctrl/lint/otp_ctrl_pkg.waiver b/hw/ip_templates/otp_ctrl/lint/otp_ctrl_pkg.waiver new file mode 100644 index 00000000000000..e0846938574c63 --- /dev/null +++ b/hw/ip_templates/otp_ctrl/lint/otp_ctrl_pkg.waiver @@ -0,0 +1,5 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# ascentlint waiver file for OTP controller generic package diff --git a/hw/ip_templates/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vbl b/hw/ip_templates/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vbl new file mode 100644 index 00000000000000..dadab89d18ef1e --- /dev/null +++ b/hw/ip_templates/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vbl @@ -0,0 +1,8 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# waiver file for OTP controller top-specific package + +# waive long line violations in generated code +waive --rule=line-length --location="otp_ctrl_part_pkg.sv" diff --git a/hw/ip_templates/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vlt b/hw/ip_templates/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vlt new file mode 100644 index 00000000000000..06603a4760cf74 --- /dev/null +++ b/hw/ip_templates/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vlt @@ -0,0 +1,5 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// waiver file for OTP controller top-specific package diff --git a/hw/ip/otp_ctrl/lint/otp_ctrl_pkg.waiver b/hw/ip_templates/otp_ctrl/lint/otp_ctrl_top_specific_pkg.waiver similarity index 86% rename from hw/ip/otp_ctrl/lint/otp_ctrl_pkg.waiver rename to hw/ip_templates/otp_ctrl/lint/otp_ctrl_top_specific_pkg.waiver index ad32fe048a4e3d..69259ce4b16170 100644 --- a/hw/ip/otp_ctrl/lint/otp_ctrl_pkg.waiver +++ b/hw/ip_templates/otp_ctrl/lint/otp_ctrl_top_specific_pkg.waiver @@ -2,7 +2,7 @@ # Licensed under the Apache License, Version 2.0, see LICENSE for details. # SPDX-License-Identifier: Apache-2.0 # -# waiver file for OTP controller package +# waiver file for OTP controller top-specific package # waive line length violations in generated code waive -rules LINE_LENGTH -location {otp_ctrl_part_pkg.sv} \ diff --git a/hw/ip_templates/otp_ctrl/otp_ctrl.core.tpl b/hw/ip_templates/otp_ctrl/otp_ctrl.core.tpl new file mode 100644 index 00000000000000..f759777866a5ab --- /dev/null +++ b/hw/ip_templates/otp_ctrl/otp_ctrl.core.tpl @@ -0,0 +1,106 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: ${instance_vlnv("lowrisc:ip:otp_ctrl:1.0")} +description: "OTP Controller" +virtual: + - lowrisc:ip_interfaces:otp_ctrl + +filesets: + files_rtl: + depend: + - lowrisc:ip:otp_ctrl_pkg + - ${instance_vlnv("lowrisc:ip:otp_ctrl_top_specific_pkg")} + - lowrisc:ip:tlul + - lowrisc:prim:all + - lowrisc:prim:ram_1p + - lowrisc:prim:otp + - lowrisc:prim:double_lfsr + - lowrisc:prim:count + - lowrisc:prim:lc_sender + - lowrisc:prim:lc_sync + - lowrisc:prim:buf + - lowrisc:prim:flop + - lowrisc:prim:secded + - lowrisc:prim:edn_req + - lowrisc:prim:sec_anchor + - lowrisc:ip_interfaces:pwrmgr_pkg + - lowrisc:ip:edn_pkg + - lowrisc:prim:sparse_fsm + - "fileset_partner ? (partner:systems:ast_pkg)" + - "!fileset_partner ? (lowrisc:systems:ast_pkg)" + files: + - rtl/otp_ctrl_core_reg_top.sv + - rtl/otp_ctrl_ecc_reg.sv + - rtl/otp_ctrl_scrmbl.sv + - rtl/otp_ctrl_lfsr_timer.sv + - rtl/otp_ctrl_part_unbuf.sv + - rtl/otp_ctrl_part_buf.sv + - rtl/otp_ctrl_dai.sv + - rtl/otp_ctrl_kdi.sv + - rtl/otp_ctrl_lci.sv + - rtl/otp_ctrl.sv + file_type: systemVerilogSource + + files_verilator_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl.vlt + file_type: vlt + + files_ascentlint_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl.waiver + file_type: waiver + + files_veriblelint_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl.vbl + file_type: veribleLintWaiver + +parameters: + SYNTHESIS: + datatype: bool + paramtype: vlogdefine + + +targets: + default: &default_target + filesets: + - tool_verilator ? (files_verilator_waiver) + - tool_ascentlint ? (files_ascentlint_waiver) + - tool_veriblelint ? (files_veriblelint_waiver) + - files_rtl + toplevel: otp_ctrl + + lint: + <<: *default_target + default_tool: verilator + parameters: + - SYNTHESIS=true + tools: + verilator: + mode: lint-only + verilator_options: + - "-Wall" + + syn: + <<: *default_target + # TODO: set default to DC once + # this option is available + # olofk/edalize#89 + default_tool: icarus + parameters: + - SYNTHESIS=true diff --git a/hw/ip/otp_ctrl/otp_ctrl_pkg.core b/hw/ip_templates/otp_ctrl/otp_ctrl_pkg.core similarity index 90% rename from hw/ip/otp_ctrl/otp_ctrl_pkg.core rename to hw/ip_templates/otp_ctrl/otp_ctrl_pkg.core index 12628f906ee2b0..6698fbdb8541fa 100644 --- a/hw/ip/otp_ctrl/otp_ctrl_pkg.core +++ b/hw/ip_templates/otp_ctrl/otp_ctrl_pkg.core @@ -2,19 +2,15 @@ CAPI=2: # Copyright lowRISC contributors (OpenTitan project). # Licensed under the Apache License, Version 2.0, see LICENSE for details. # SPDX-License-Identifier: Apache-2.0 -name: "lowrisc:ip:otp_ctrl_pkg:1.0" +name: lowrisc:ip:otp_ctrl_pkg:1.0 description: "OTP Controller Package" filesets: files_rtl: depend: - - lowrisc:tlul:headers - lowrisc:ip:lc_ctrl_pkg - - lowrisc:prim:mubi files: - - rtl/otp_ctrl_reg_pkg.sv - rtl/otp_ctrl_pkg.sv - - rtl/otp_ctrl_part_pkg.sv file_type: systemVerilogSource files_verilator_waiver: diff --git a/hw/ip/otp_ctrl/otp_ctrl_prim_reg_top.core b/hw/ip_templates/otp_ctrl/otp_ctrl_prim_reg_top.core similarity index 71% rename from hw/ip/otp_ctrl/otp_ctrl_prim_reg_top.core rename to hw/ip_templates/otp_ctrl/otp_ctrl_prim_reg_top.core index 4d54817c20935c..817b3e74434c60 100644 --- a/hw/ip/otp_ctrl/otp_ctrl_prim_reg_top.core +++ b/hw/ip_templates/otp_ctrl/otp_ctrl_prim_reg_top.core @@ -7,7 +7,10 @@ description: "Generic register top for the OTP wrapper" filesets: files_rtl: depend: - - lowrisc:ip:otp_ctrl_pkg + # otp_ctrl_prim_reg_top.sv should depend on generic items for now. + # This may change and will require reworking the flow to generate + # the otp prim. + - lowrisc:ip_interfaces:otp_ctrl_top_specific_pkg files: - rtl/otp_ctrl_prim_reg_top.sv file_type: systemVerilogSource diff --git a/hw/ip_templates/otp_ctrl/otp_ctrl_top_specific_pkg.core.tpl b/hw/ip_templates/otp_ctrl/otp_ctrl_top_specific_pkg.core.tpl new file mode 100644 index 00000000000000..d90773b598db1c --- /dev/null +++ b/hw/ip_templates/otp_ctrl/otp_ctrl_top_specific_pkg.core.tpl @@ -0,0 +1,75 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: ${instance_vlnv("lowrisc:ip:otp_ctrl_top_specific_pkg:1.0")} +description: "OTP Controller Top Specific Packages" +virtual: + - lowrisc:ip_interfaces:otp_ctrl_top_specific_pkg + +filesets: + files_rtl: + depend: + - lowrisc:tlul:headers + - lowrisc:ip:lc_ctrl_pkg + - lowrisc:ip:otp_ctrl_pkg + - lowrisc:prim:mubi + + files: + - rtl/otp_ctrl_reg_pkg.sv + - rtl/otp_ctrl_top_specific_pkg.sv + - rtl/otp_ctrl_part_pkg.sv + file_type: systemVerilogSource + + files_verilator_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_top_specific_pkg.vlt + file_type: vlt + + files_ascentlint_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_top_specific_pkg.waiver + file_type: waiver + + files_veriblelint_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_top_specific_pkg.vbl + file_type: veribleLintWaiver + +parameters: + SYNTHESIS: + datatype: bool + paramtype: vlogdefine + + +targets: + default: &default_target + filesets: + - tool_verilator ? (files_verilator_waiver) + - tool_ascentlint ? (files_ascentlint_waiver) + - tool_veriblelint ? (files_veriblelint_waiver) + - files_rtl + toplevel: lc_ctrl + + lint: + <<: *default_target + default_tool: verilator + parameters: + - SYNTHESIS=true + tools: + verilator: + mode: lint-only + verilator_options: + - "-Wall" diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl.sv b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl.sv similarity index 99% rename from hw/ip/otp_ctrl/rtl/otp_ctrl.sv rename to hw/ip_templates/otp_ctrl/rtl/otp_ctrl.sv index 5558e2ac0960d5..6d5647a989a41b 100644 --- a/hw/ip/otp_ctrl/rtl/otp_ctrl.sv +++ b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl.sv @@ -11,6 +11,7 @@ module otp_ctrl import otp_ctrl_pkg::*; import otp_ctrl_reg_pkg::*; import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; #( // Enable asynchronous transitions on alerts. parameter logic [NumAlerts-1:0] AlertAsyncOn = {NumAlerts{1'b1}}, @@ -893,7 +894,8 @@ end // Since this scheme does not have built-in preemtion, it must be ensured that the agents // eventually release their locks for this to be fair. // - // See also https://docs.opentitan.org/hw/ip/otp_ctrl/doc/index.html#block-diagram for details. + // See also https://docs.opentitan.org/hw/ip/otp_ctrl/index.html#otp-controller-overview for + // details. typedef struct packed { otp_scrmbl_cmd_e cmd; digest_mode_e mode; diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl_dai.sv b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_dai.sv similarity index 99% rename from hw/ip/otp_ctrl/rtl/otp_ctrl_dai.sv rename to hw/ip_templates/otp_ctrl/rtl/otp_ctrl_dai.sv index 820eaafed37573..95240cfa9d165b 100644 --- a/hw/ip/otp_ctrl/rtl/otp_ctrl_dai.sv +++ b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_dai.sv @@ -11,6 +11,7 @@ module otp_ctrl_dai import otp_ctrl_pkg::*; import otp_ctrl_reg_pkg::*; import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; ( input clk_i, input rst_ni, diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl_ecc_reg.sv b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_ecc_reg.sv similarity index 100% rename from hw/ip/otp_ctrl/rtl/otp_ctrl_ecc_reg.sv rename to hw/ip_templates/otp_ctrl/rtl/otp_ctrl_ecc_reg.sv diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl_kdi.sv b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_kdi.sv similarity index 99% rename from hw/ip/otp_ctrl/rtl/otp_ctrl_kdi.sv rename to hw/ip_templates/otp_ctrl/rtl/otp_ctrl_kdi.sv index 581c280e854988..3b5d6b454f950f 100644 --- a/hw/ip/otp_ctrl/rtl/otp_ctrl_kdi.sv +++ b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_kdi.sv @@ -11,6 +11,7 @@ module otp_ctrl_kdi import otp_ctrl_pkg::*; import otp_ctrl_reg_pkg::*; import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; #( parameter scrmbl_key_init_t RndCnstScrmblKeyInit = RndCnstScrmblKeyInitDefault ) ( diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl_lci.sv b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_lci.sv similarity index 99% rename from hw/ip/otp_ctrl/rtl/otp_ctrl_lci.sv rename to hw/ip_templates/otp_ctrl/rtl/otp_ctrl_lci.sv index 122675069d059d..b2f98d41be9d06 100644 --- a/hw/ip/otp_ctrl/rtl/otp_ctrl_lci.sv +++ b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_lci.sv @@ -11,6 +11,7 @@ module otp_ctrl_lci import otp_ctrl_pkg::*; import otp_ctrl_reg_pkg::*; import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; #( // Lifecycle partition information parameter part_info_t Info = PartInfoDefault diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv similarity index 99% rename from hw/ip/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv rename to hw/ip_templates/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv index 22b3ec383a89c8..775c8586cfcaf4 100644 --- a/hw/ip/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv +++ b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv @@ -31,6 +31,7 @@ module otp_ctrl_lfsr_timer import otp_ctrl_pkg::*; import otp_ctrl_reg_pkg::*; + import otp_ctrl_top_specific_pkg::*; #( // Compile time random constants, to be overriden by topgen. parameter lfsr_seed_t RndCnstLfsrSeed = RndCnstLfsrSeedDefault, diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl_part_buf.sv b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_part_buf.sv similarity index 99% rename from hw/ip/otp_ctrl/rtl/otp_ctrl_part_buf.sv rename to hw/ip_templates/otp_ctrl/rtl/otp_ctrl_part_buf.sv index 27f6fc67d482d7..c4da5a89302840 100644 --- a/hw/ip/otp_ctrl/rtl/otp_ctrl_part_buf.sv +++ b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_part_buf.sv @@ -11,6 +11,7 @@ module otp_ctrl_part_buf import otp_ctrl_pkg::*; import otp_ctrl_reg_pkg::*; import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; #( // Partition information. parameter part_info_t Info = PartInfoDefault, diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_part_pkg.sv.tpl b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_part_pkg.sv.tpl similarity index 87% rename from hw/ip/otp_ctrl/data/otp_ctrl_part_pkg.sv.tpl rename to hw/ip_templates/otp_ctrl/rtl/otp_ctrl_part_pkg.sv.tpl index 6c8e0154054e7c..ba091f04b30d6f 100644 --- a/hw/ip/otp_ctrl/data/otp_ctrl_part_pkg.sv.tpl +++ b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_part_pkg.sv.tpl @@ -4,7 +4,6 @@ // // Package partition metadata. // -${gen_comment} <% from topgen.lib import Name %>\ @@ -13,13 +12,14 @@ package otp_ctrl_part_pkg; import prim_util_pkg::vbits; import otp_ctrl_reg_pkg::*; import otp_ctrl_pkg::*; + import otp_ctrl_top_specific_pkg::*; //////////////////////////////////// // Scrambling Constants and Types // //////////////////////////////////// - parameter int NumScrmblKeys = ${len(otp_mmap.config["scrambling"]["keys"])}; - parameter int NumDigestSets = ${len(otp_mmap.config["scrambling"]["digests"])}; + parameter int NumScrmblKeys = ${len(otp_mmap["scrambling"]["keys"])}; + parameter int NumDigestSets = ${len(otp_mmap["scrambling"]["digests"])}; parameter int ScrmblKeySelWidth = vbits(NumScrmblKeys); parameter int DigestSetSelWidth = vbits(NumDigestSets); @@ -37,21 +37,21 @@ package otp_ctrl_part_pkg; typedef logic [NumDigestSets-1:0][ScrmblBlockWidth-1:0] digest_iv_array_t; typedef enum logic [ConstSelWidth-1:0] { -% for key in otp_mmap.config["scrambling"]["keys"]: +% for key in otp_mmap["scrambling"]["keys"]: ${key["name"]}${"" if loop.last else ","} % endfor } key_sel_e; typedef enum logic [ConstSelWidth-1:0] { -% for dig in otp_mmap.config["scrambling"]["digests"]: +% for dig in otp_mmap["scrambling"]["digests"]: ${dig["name"]}${"" if loop.last else ","} % endfor } digest_sel_e; // SEC_CM: SECRET.MEM.SCRAMBLE parameter key_array_t RndCnstKey = { -% for key in otp_mmap.config["scrambling"]["keys"][::-1]: - ${"{0:}'h{1:0X}".format(otp_mmap.config["scrambling"]["key_size"] * 8, key["value"])}${"" if loop.last else ","} +% for key in otp_mmap["scrambling"]["keys"][::-1]: + ${"{0:}'h{1:0X}".format(otp_mmap["scrambling"]["key_size"] * 8, key["value"])}${"" if loop.last else ","} % endfor }; @@ -59,14 +59,14 @@ package otp_ctrl_part_pkg; // Note: digest set 0 is used for computing the partition digests. Constants at // higher indices are used to compute the scrambling keys. parameter digest_const_array_t RndCnstDigestConst = { -% for dig in otp_mmap.config["scrambling"]["digests"][::-1]: - ${"{0:}'h{1:0X}".format(otp_mmap.config["scrambling"]["cnst_size"] * 8, dig["cnst_value"])}${"" if loop.last else ","} +% for dig in otp_mmap["scrambling"]["digests"][::-1]: + ${"{0:}'h{1:0X}".format(otp_mmap["scrambling"]["cnst_size"] * 8, dig["cnst_value"])}${"" if loop.last else ","} % endfor }; parameter digest_iv_array_t RndCnstDigestIV = { -% for dig in otp_mmap.config["scrambling"]["digests"][::-1]: - ${"{0:}'h{1:0X}".format(otp_mmap.config["scrambling"]["iv_size"] * 8, dig["iv_value"])}${"" if loop.last else ","} +% for dig in otp_mmap["scrambling"]["digests"][::-1]: + ${"{0:}'h{1:0X}".format(otp_mmap["scrambling"]["iv_size"] * 8, dig["iv_value"])}${"" if loop.last else ","} % endfor }; @@ -119,11 +119,11 @@ package otp_ctrl_part_pkg; //////////////////////// localparam part_info_t PartInfo [NumPart] = '{ -% for part in otp_mmap.config["partitions"]: +% for part in otp_mmap["partitions"]: // ${part["name"]} '{ variant: ${part["variant"]}, - offset: ${otp_mmap.config["otp"]["byte_addr_width"]}'d${part["offset"]}, + offset: ${otp_mmap["otp"]["byte_addr_width"]}'d${part["offset"]}, size: ${part["size"]}, key_sel: ${part["key_sel"] if part["key_sel"] != "NoKey" else "key_sel_e'('0)"}, secret: 1'b${"1" if part["secret"] else "0"}, @@ -139,7 +139,7 @@ package otp_ctrl_part_pkg; }; typedef enum { -% for part in otp_mmap.config["partitions"]: +% for part in otp_mmap["partitions"]: ${Name.from_snake_case(part["name"]).as_camel_case()}Idx, % endfor // These are not "real partitions", but in terms of implementation it is convenient to @@ -154,7 +154,7 @@ package otp_ctrl_part_pkg; parameter int NumAgents = int'(NumAgentsIdx); // Breakout types for easier access of individual items. -% for part in otp_mmap.config["partitions"]: +% for part in otp_mmap["partitions"]: % if part["bkout_type"]: typedef struct packed {<% offset = part['offset'] + part['size'] %> % for item in part["items"][::-1]: @@ -193,7 +193,7 @@ package otp_ctrl_part_pkg; typedef struct packed { // This reuses the same encoding as the life cycle signals for indicating valid status. lc_ctrl_pkg::lc_tx_t valid; -% for part in otp_mmap.config["partitions"][::-1]: +% for part in otp_mmap["partitions"][::-1]: % if part["bkout_type"]: otp_${part["name"].lower()}_data_t ${part["name"].lower()}_data; % endif @@ -204,13 +204,13 @@ package otp_ctrl_part_pkg; <% k = 0 num_bkout = 0 - for part in otp_mmap.config["partitions"]: + for part in otp_mmap["partitions"]: if part["bkout_type"]: num_bkout += 1 %>\ parameter otp_broadcast_t OTP_BROADCAST_DEFAULT = '{ valid: lc_ctrl_pkg::Off, -% for part in otp_mmap.config["partitions"][::-1]: +% for part in otp_mmap["partitions"][::-1]: % if part["bkout_type"]: ${part["name"].lower()}_data: OTP_${part["name"].upper()}_DATA_DEFAULT${"" if k == num_bkout-1 else ","} <% k+=1 %>\ @@ -218,16 +218,16 @@ package otp_ctrl_part_pkg; % endfor }; -<% offset = int(otp_mmap.config["partitions"][-1]["offset"]) + int(otp_mmap.config["partitions"][-1]["size"]) %> +<% offset = int(otp_mmap["partitions"][-1]["offset"]) + int(otp_mmap["partitions"][-1]["size"]) %> // OTP invalid partition default for buffered partitions. parameter logic [${offset * 8 - 1}:0] PartInvDefault = ${offset * 8}'({ - % for k, part in enumerate(otp_mmap.config["partitions"][::-1]): + % for k, part in enumerate(otp_mmap["partitions"][::-1]): ${int(part["size"])*8}'({ % for item in part["items"][::-1]: % if offset != item['offset'] + item['size']: ${"{}'h{:0X}".format((offset - item['size'] - item['offset']) * 8, 0)}, // unallocated space<% offset = item['offset'] + item['size'] %> % endif - ${"{}'h{:0X}".format(item["size"] * 8, item["inv_default"])}${("\n })," if k < len(otp_mmap.config["partitions"])-1 else "\n })});") if loop.last else ","}<% offset -= item['size'] %> + ${"{}'h{:0X}".format(item["size"] * 8, item["inv_default"])}${("\n })," if k < len(otp_mmap["partitions"])-1 else "\n })});") if loop.last else ","}<% offset -= item['size'] %> % endfor % endfor @@ -241,7 +241,7 @@ package otp_ctrl_part_pkg; logic unused_sigs; unused_sigs = ^part_digest; hw2reg = '0; -% for k, part in enumerate(otp_mmap.config["partitions"]): +% for k, part in enumerate(otp_mmap["partitions"]): <% part_name = Name.from_snake_case(part["name"]) part_name_camel = part_name.as_camel_case() @@ -262,7 +262,7 @@ package otp_ctrl_part_pkg; part_access_pre = {{32'(2*NumPart)}{prim_mubi_pkg::MuBi8False}}; // Note: these could be made a MuBi CSRs in the future. // The main thing that is missing right now is proper support for W0C. -% for k, part in enumerate(otp_mmap.config["partitions"]): +% for k, part in enumerate(otp_mmap["partitions"]): % if part["read_lock"] == "CSR": // ${part["name"]} if (!reg2hw.${part["name"].lower()}_read_lock) begin @@ -281,7 +281,7 @@ package otp_ctrl_part_pkg; logic valid, unused; unused = 1'b0; valid = 1'b1; -% for part in otp_mmap.config["partitions"]: +% for part in otp_mmap["partitions"]: // ${part["name"]} <% part_name = Name.from_snake_case(part["name"]) @@ -310,7 +310,7 @@ package otp_ctrl_part_pkg; // interface to the keymgr remains stable. The type contains // a superset of all options, so we have to initialize it to '0 here. otp_keymgr_key = '0; -% for part in otp_mmap.config["partitions"]: +% for part in otp_mmap["partitions"]: // ${part["name"]} <% part_name = Name.from_snake_case(part["name"]) diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv similarity index 99% rename from hw/ip/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv rename to hw/ip_templates/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv index 23cf3171b929ec..09eb0fe0d73fa5 100644 --- a/hw/ip/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv +++ b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv @@ -11,6 +11,7 @@ module otp_ctrl_part_unbuf import otp_ctrl_pkg::*; import otp_ctrl_reg_pkg::*; import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; #( // Partition information. parameter part_info_t Info = PartInfoDefault diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl_pkg.sv b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_pkg.sv similarity index 70% rename from hw/ip/otp_ctrl/rtl/otp_ctrl_pkg.sv rename to hw/ip_templates/otp_ctrl/rtl/otp_ctrl_pkg.sv index 7a460fdd588248..0450b103b07741 100644 --- a/hw/ip/otp_ctrl/rtl/otp_ctrl_pkg.sv +++ b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_pkg.sv @@ -2,12 +2,11 @@ // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 // +// This package can be imported by generic IPs: +// - It does not import otp_ctrl_reg_pkg, which is generated and top-specific. package otp_ctrl_pkg; - import prim_util_pkg::vbits; - import otp_ctrl_reg_pkg::*; - //////////////////////// // General Parameters // //////////////////////// @@ -18,65 +17,14 @@ package otp_ctrl_pkg; parameter int OtpTestStatusWidth = 32; parameter int OtpTestVectWidth = 8; - // Width of entropy input - parameter int EdnDataWidth = 64; - - parameter int NumPartWidth = vbits(NumPart); - - parameter int SwWindowAddrWidth = vbits(NumSwCfgWindowWords); - - // Background check timer LFSR width. - parameter int LfsrWidth = 40; - // The LFSR will be reseeded once LfsrUsageThreshold - // values have been drawn from it. - parameter int LfsrUsageThreshold = 16; - - // Redundantly encoded and complementary values are used to for signalling to the partition - // controller FSMs and the DAI whether a partition is locked or not. Any other value than - // "Mubi8Lo" is interpreted as "Locked" in those FSMs. - typedef struct packed { - prim_mubi_pkg::mubi8_t read_lock; - prim_mubi_pkg::mubi8_t write_lock; - } part_access_t; - - parameter int DaiCmdWidth = 3; - typedef enum logic [DaiCmdWidth-1:0] { - DaiRead = 3'b001, - DaiWrite = 3'b010, - DaiDigest = 3'b100 - } dai_cmd_e; - parameter int DeviceIdWidth = 256; typedef logic [DeviceIdWidth-1:0] otp_device_id_t; parameter int ManufStateWidth = 256; typedef logic [ManufStateWidth-1:0] otp_manuf_state_t; - ////////////////////////////////////// - // Typedefs for OTP Macro Interface // - ////////////////////////////////////// - // OTP-macro specific - parameter int OtpWidth = 16; - parameter int OtpAddrWidth = OtpByteAddrWidth - $clog2(OtpWidth/8); - parameter int OtpDepth = 2**OtpAddrWidth; - parameter int OtpSizeWidth = 2; // Allows to transfer up to 4 native OTP words at once. - parameter int OtpErrWidth = 3; parameter int OtpPwrSeqWidth = 2; - parameter int OtpIfWidth = 2**OtpSizeWidth*OtpWidth; - // Number of Byte address bits to cut off in order to get the native OTP word address. - parameter int OtpAddrShift = OtpByteAddrWidth - OtpAddrWidth; - - typedef enum logic [OtpErrWidth-1:0] { - NoError = 3'h0, - MacroError = 3'h1, - MacroEccCorrError = 3'h2, - MacroEccUncorrError = 3'h3, - MacroWriteBlankError = 3'h4, - AccessError = 3'h5, - CheckFailError = 3'h6, - FsmStateError = 3'h7 - } otp_err_e; ///////////////////////////////// // Typedefs for OTP Scrambling // @@ -85,18 +33,6 @@ package otp_ctrl_pkg; parameter int ScrmblKeyWidth = 128; parameter int ScrmblBlockWidth = 64; - parameter int NumPresentRounds = 31; - parameter int ScrmblBlockHalfWords = ScrmblBlockWidth / OtpWidth; - - typedef enum logic [2:0] { - Decrypt, - Encrypt, - LoadShadow, - Digest, - DigestInit, - DigestFinalize - } otp_scrmbl_cmd_e; - /////////////////////////////// // Typedefs for LC Interface // /////////////////////////////// @@ -154,17 +90,6 @@ package otp_ctrl_pkg; logic ack; } lc_otp_program_rsp_t; - // RAW unlock token hashing request. - typedef struct packed { - logic req; - lc_ctrl_state_pkg::lc_token_t token_input; - } lc_otp_token_req_t; - - typedef struct packed { - logic ack; - lc_ctrl_state_pkg::lc_token_t hashed_token; - } lc_otp_token_rsp_t; - typedef struct packed { logic [OtpTestCtrlWidth-1:0] ctrl; } lc_otp_vendor_test_req_t; @@ -292,7 +217,6 @@ package otp_ctrl_pkg; logic idle; } otp_pwr_state_t; - /////////////////// // AST Interface // /////////////////// @@ -305,23 +229,4 @@ package otp_ctrl_pkg; logic [OtpPwrSeqWidth-1:0] pwr_seq_h; } otp_ast_rsp_t; - /////////////////////////////////////////// - // Defaults for random netlist constants // - /////////////////////////////////////////// - - // These LFSR parameters have been generated with - // $ util/design/gen-lfsr-seed.py --width 40 --seed 4247488366 - typedef logic [LfsrWidth-1:0] lfsr_seed_t; - typedef logic [LfsrWidth-1:0][$clog2(LfsrWidth)-1:0] lfsr_perm_t; - localparam lfsr_seed_t RndCnstLfsrSeedDefault = 40'h453d28ea98; - localparam lfsr_perm_t RndCnstLfsrPermDefault = - 240'h4235171482c225f79289b32181a0163a760355d3447063d16661e44c12a5; - - typedef struct packed { - sram_key_t key; - sram_nonce_t nonce; - } scrmbl_key_init_t; - localparam scrmbl_key_init_t RndCnstScrmblKeyInitDefault = - 256'hcebeb96ffe0eced795f8b2cfe23c1e519e4fa08047a6bcfb811b04f0a479006e; - endpackage : otp_ctrl_pkg diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl_scrmbl.sv b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_scrmbl.sv similarity index 97% rename from hw/ip/otp_ctrl/rtl/otp_ctrl_scrmbl.sv rename to hw/ip_templates/otp_ctrl/rtl/otp_ctrl_scrmbl.sv index 9470ff8eb1512c..7c5296929b6f85 100644 --- a/hw/ip/otp_ctrl/rtl/otp_ctrl_scrmbl.sv +++ b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_scrmbl.sv @@ -60,18 +60,20 @@ // DigestFinalize: This command encrypts the digest state with the finalization constant selected // by sel_i in order to form the final digest. // -// References: - https://docs.opentitan.org/hw/ip/otp_ctrl/doc/index.html#design-details -// - https://docs.opentitan.org/hw/ip/prim/doc/prim_present/ -// - https://en.wikipedia.org/wiki/Merkle-Damgard_construction -// - https://en.wikipedia.org/wiki/One-way_compression_function#Davies%E2%80%93Meyer -// - https://en.wikipedia.org/wiki/PRESENT -// - http://www.lightweightcrypto.org/present/present_ches2007.pdf +// References: +// - https://docs.opentitan.org/hw/ip/otp_ctrl/doc/theory_of_operation.html#block-diagram +// - https://docs.opentitan.org/hw/ip/prim/doc/prim_present/ +// - https://en.wikipedia.org/wiki/Merkle-Damgard_construction +// - https://en.wikipedia.org/wiki/One-way_compression_function#Davies%E2%80%93Meyer +// - https://en.wikipedia.org/wiki/PRESENT +// - http://www.lightweightcrypto.org/present/present_ches2007.pdf // `include "prim_flop_macros.sv" module otp_ctrl_scrmbl import otp_ctrl_pkg::*; + import otp_ctrl_top_specific_pkg::*; import otp_ctrl_part_pkg::*; ( input clk_i, diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl_token_const.sv b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_token_const.sv similarity index 96% rename from hw/ip/otp_ctrl/rtl/otp_ctrl_token_const.sv rename to hw/ip_templates/otp_ctrl/rtl/otp_ctrl_token_const.sv index 35cdc23dd0c7a6..2443cd8a11d61a 100644 --- a/hw/ip/otp_ctrl/rtl/otp_ctrl_token_const.sv +++ b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_token_const.sv @@ -35,7 +35,7 @@ module otp_ctrl_token_const import otp_ctrl_pkg::*; #( assign data[j][1] = RndCnstDigestConst[LcRawDigest]; // Each hash takes four invocations, see diagram c) on - // https://docs.opentitan.org/hw/ip/otp_ctrl/doc/index.html#scrambling-datapath + // https://docs.opentitan.org/hw/ip/otp_ctrl/doc/theory_of_operation.html#scrambling-datapath for (genvar k = 0; k < 4; k++) begin : gen_invocations logic [ScrmblBlockWidth-1:0] next_state; diff --git a/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv new file mode 100644 index 00000000000000..e9c41a99d920a2 --- /dev/null +++ b/hw/ip_templates/otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv @@ -0,0 +1,108 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// + +// This package contains top specific and implementation items. +// +// It imports otp_ctrl_reg_pkg, which is generated from the top specific hjson file. +// The items that are generic and used by top-independent IPs are placed in otp_ctrl_pkg. +package otp_ctrl_top_specific_pkg; + + import prim_util_pkg::vbits; + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + + //////////////////////// + // General Parameters // + //////////////////////// + + // Width of entropy input + parameter int EdnDataWidth = 64; + + parameter int NumPartWidth = vbits(NumPart); + + parameter int SwWindowAddrWidth = vbits(NumSwCfgWindowWords); + + // Background check timer LFSR width. + parameter int LfsrWidth = 40; + // The LFSR will be reseeded once LfsrUsageThreshold + // values have been drawn from it. + parameter int LfsrUsageThreshold = 16; + + // Redundantly encoded and complementary values are used to for signalling to the partition + // controller FSMs and the DAI whether a partition is locked or not. Any other value than + // "Mubi8Lo" is interpreted as "Locked" in those FSMs. + typedef struct packed { + prim_mubi_pkg::mubi8_t read_lock; + prim_mubi_pkg::mubi8_t write_lock; + } part_access_t; + + parameter int DaiCmdWidth = 3; + typedef enum logic [DaiCmdWidth-1:0] { + DaiRead = 3'b001, + DaiWrite = 3'b010, + DaiDigest = 3'b100 + } dai_cmd_e; + + ////////////////////////////////////// + // Typedefs for OTP Macro Interface // + ////////////////////////////////////// + + // OTP-macro specific + parameter int OtpWidth = 16; + parameter int OtpAddrWidth = OtpByteAddrWidth - $clog2(OtpWidth/8); + parameter int OtpDepth = 2**OtpAddrWidth; + parameter int OtpSizeWidth = 2; // Allows to transfer up to 4 native OTP words at once. + parameter int OtpErrWidth = 3; + parameter int OtpIfWidth = 2**OtpSizeWidth*OtpWidth; + // Number of Byte address bits to cut off in order to get the native OTP word address. + parameter int OtpAddrShift = OtpByteAddrWidth - OtpAddrWidth; + + typedef enum logic [OtpErrWidth-1:0] { + NoError = 3'h0, + MacroError = 3'h1, + MacroEccCorrError = 3'h2, + MacroEccUncorrError = 3'h3, + MacroWriteBlankError = 3'h4, + AccessError = 3'h5, + CheckFailError = 3'h6, + FsmStateError = 3'h7 + } otp_err_e; + + ///////////////////////////////// + // Typedefs for OTP Scrambling // + ///////////////////////////////// + + parameter int NumPresentRounds = 31; + parameter int ScrmblBlockHalfWords = ScrmblBlockWidth / OtpWidth; + + typedef enum logic [2:0] { + Decrypt, + Encrypt, + LoadShadow, + Digest, + DigestInit, + DigestFinalize + } otp_scrmbl_cmd_e; + + /////////////////////////////////////////// + // Defaults for random netlist constants // + /////////////////////////////////////////// + + // These LFSR parameters have been generated with + // $ util/design/gen-lfsr-seed.py --width 40 --seed 4247488366 + typedef logic [LfsrWidth-1:0] lfsr_seed_t; + typedef logic [LfsrWidth-1:0][$clog2(LfsrWidth)-1:0] lfsr_perm_t; + localparam lfsr_seed_t RndCnstLfsrSeedDefault = 40'h453d28ea98; + localparam lfsr_perm_t RndCnstLfsrPermDefault = + 240'h4235171482c225f79289b32181a0163a760355d3447063d16661e44c12a5; + + typedef struct packed { + sram_key_t key; + sram_nonce_t nonce; + } scrmbl_key_init_t; + localparam scrmbl_key_init_t RndCnstScrmblKeyInitDefault = + 256'hcebeb96ffe0eced795f8b2cfe23c1e519e4fa08047a6bcfb811b04f0a479006e; + +endpackage : otp_ctrl_top_specific_pkg diff --git a/hw/ip/otp_ctrl/syn/constraints.sdc b/hw/ip_templates/otp_ctrl/syn/constraints.sdc similarity index 100% rename from hw/ip/otp_ctrl/syn/constraints.sdc rename to hw/ip_templates/otp_ctrl/syn/constraints.sdc diff --git a/hw/ip_templates/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson.tpl b/hw/ip_templates/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson.tpl new file mode 100644 index 00000000000000..92d07daba8e153 --- /dev/null +++ b/hw/ip_templates/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson.tpl @@ -0,0 +1,24 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +{ + // Top level dut name (sv module). + name: otp_ctrl + + // Fusesoc core file used for building the file list. + fusesoc_core: lowrisc:opentitan:top_${topname}_{name}:0.1 + + import_cfgs: [// Project wide common GTECH synthesis config file + "{proj_root}/hw/syn/tools/dvsim/common_gtech_syn_cfg.hjson"] + + overrides: [ + { + name: design_level + value: "top" + } + { // Deletes black-boxed hierarchies before writing out the unmapped netlist + name: post_elab_script + value: "{proj_root}/hw/top_${topname}/ip_autogen/{name}/syn/post_elab_gtech.tcl" + } + ] +} diff --git a/hw/ip_templates/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson.tpl b/hw/ip_templates/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson.tpl new file mode 100644 index 00000000000000..effaf5f8ed0633 --- /dev/null +++ b/hw/ip_templates/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson.tpl @@ -0,0 +1,26 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +{ + // Top level dut name (sv module). + name: otp_ctrl + + // Fusesoc core file used for building the file list. + fusesoc_core: lowrisc:opentitan:top_${topname}_{name}:0.1 + + import_cfgs: [// Project wide common synthesis config file + "{proj_root}/hw/syn/tools/dvsim/common_syn_cfg.hjson"] + + overrides: [ + { + name: design_level + value: "top" + } + ] + + // Timing constraints for this module + sdc_file: "{proj_root}/hw/top_${topname}/ip_autogen/{name}/syn/constraints.sdc" + + // This is not needed for this module + foundry_sdc_file: "" +} diff --git a/hw/ip/otp_ctrl/syn/post_elab_gtech.tcl b/hw/ip_templates/otp_ctrl/syn/post_elab_gtech.tcl similarity index 100% rename from hw/ip/otp_ctrl/syn/post_elab_gtech.tcl rename to hw/ip_templates/otp_ctrl/syn/post_elab_gtech.tcl diff --git a/hw/ip_templates/pwrmgr/doc/theory_of_operation.md b/hw/ip_templates/pwrmgr/doc/theory_of_operation.md index 5f197a894c9e36..c46bc74cd1697f 100644 --- a/hw/ip_templates/pwrmgr/doc/theory_of_operation.md +++ b/hw/ip_templates/pwrmgr/doc/theory_of_operation.md @@ -69,7 +69,7 @@ Instead the system goes into a terminal non-responsive state where a user or hos The fast clock domain FSM (referred to as fast FSM from here on) resets to `Low Power` state and waits for a power-up request from the slow FSM. Once received, the fast FSM releases the life cycle reset stage (see [reset controller](../../rstmgr/README.md) for more details). -This allows the [OTP](../../../../ip/otp_ctrl/README.md) to begin sensing. +This allows the [OTP](../../otp_ctrl/README.md) to begin sensing. Once OTP sensing completes, the life cycle controller is initialized. The initialization of the life cycle controller puts the device into its allowed operating state (see [life cycle controller](../../../../ip/lc_ctrl/README.md) for more details). diff --git a/hw/top_darjeeling/data/autogen/defs.bzl b/hw/top_darjeeling/data/autogen/defs.bzl index ede9b996d63d0d..5aef4e6d28c6d5 100644 --- a/hw/top_darjeeling/data/autogen/defs.bzl +++ b/hw/top_darjeeling/data/autogen/defs.bzl @@ -24,7 +24,7 @@ load("//hw/ip/kmac:defs.bzl", "KMAC") load("//hw/ip/lc_ctrl:defs.bzl", "LC_CTRL") load("//hw/ip/mbx:defs.bzl", "MBX") load("//hw/ip/otbn:defs.bzl", "OTBN") -load("//hw/ip/otp_ctrl:defs.bzl", "OTP_CTRL") +load("//hw/top_darjeeling/ip_autogen/otp_ctrl:defs.bzl", "OTP_CTRL") load("//hw/top_darjeeling/ip_autogen/pinmux:defs.bzl", "PINMUX") load("//hw/top_darjeeling/ip_autogen/pwrmgr:defs.bzl", "PWRMGR") load("//hw/ip/rom_ctrl:defs.bzl", "ROM_CTRL") diff --git a/hw/top_darjeeling/data/autogen/top_darjeeling.gen.hjson b/hw/top_darjeeling/data/autogen/top_darjeeling.gen.hjson index 23a1d4f398e68d..cd8155b0760ce1 100644 --- a/hw/top_darjeeling/data/autogen/top_darjeeling.gen.hjson +++ b/hw/top_darjeeling/data/autogen/top_darjeeling.gen.hjson @@ -1052,6 +1052,7 @@ hart: 0x30138000 } } + attr: ipgen clock_connections: { clk_i: clkmgr_aon_clocks.clk_io_div4_secure @@ -1077,31 +1078,31 @@ { name: RndCnstLfsrSeed desc: Compile-time random bits for initial LFSR seed - type: otp_ctrl_pkg::lfsr_seed_t + type: otp_ctrl_top_specific_pkg::lfsr_seed_t randcount: 40 randtype: data name_top: RndCnstOtpCtrlLfsrSeed - default: 0x508e576e43 + default: 0x543a0130b3 randwidth: 40 } { name: RndCnstLfsrPerm desc: Compile-time random permutation for LFSR output - type: otp_ctrl_pkg::lfsr_perm_t + type: otp_ctrl_top_specific_pkg::lfsr_perm_t randcount: 40 randtype: perm name_top: RndCnstOtpCtrlLfsrPerm - default: 0x7da5dd44319081848f24210e8de8a498830a0015672c74cd6e116559c654 + default: 0x5764388600460c7159a11cf29648d91b4de41120e0a31677e52e050969d randwidth: 240 } { name: RndCnstScrmblKeyInit desc: Compile-time random permutation for scrambling key/nonce register reset value - type: otp_ctrl_pkg::scrmbl_key_init_t + type: otp_ctrl_top_specific_pkg::scrmbl_key_init_t randcount: 256 randtype: data name_top: RndCnstOtpCtrlScrmblKeyInit - default: 0x986313d612fdcd62afd0c3a5fc772ceb91c16f5d6e17dff0661bfba6f4e0571e + default: 0xbe84f30380655ee26d62dee6ae95aafb7a44e26df9cbedbc3d512527fe67e8b2 randwidth: 256 } ] @@ -1539,7 +1540,7 @@ randcount: 128 randtype: data name_top: RndCnstLcCtrlLcKeymgrDivInvalid - default: 0xae0fc6e6a1a665f042709b54cb121f70 + default: 0xe07d55510f7df200e9385195de4f2fb2 randwidth: 128 } { @@ -1549,7 +1550,7 @@ randcount: 128 randtype: data name_top: RndCnstLcCtrlLcKeymgrDivTestUnlocked - default: 0x2a405ecdcf1feb0fa7a14877ec88637 + default: 0x8a71e82207a34abd439b462f7fdb8fc randwidth: 128 } { @@ -1559,7 +1560,7 @@ randcount: 128 randtype: data name_top: RndCnstLcCtrlLcKeymgrDivDev - default: 0x87af272efaa51f0de710c891d47ff720 + default: 0x20e2960fa1c637588c08c38374903b58 randwidth: 128 } { @@ -1569,7 +1570,7 @@ randcount: 128 randtype: data name_top: RndCnstLcCtrlLcKeymgrDivProduction - default: 0x9f95d614848e38361cd9b7eb23d532c0 + default: 0xbc3484df9b0e9c2f06f0694cedab6be2 randwidth: 128 } { @@ -1579,7 +1580,7 @@ randcount: 128 randtype: data name_top: RndCnstLcCtrlLcKeymgrDivRma - default: 0xab44e1e92ccfebadad9193a3ee4ecd8c + default: 0x563d7c0c31cb754a523eafe90d67c2c5 randwidth: 128 } { @@ -1589,7 +1590,7 @@ randcount: 1024 randtype: data name_top: RndCnstLcCtrlInvalidTokens - default: 0x32462b3e18549cb1e70d5a83de6673d2c1c0a7b55176264e55c329c8ae4725aeaf6728150127ac755d70063277642b5309a163990b966cd494444c3bcdf8087b7facd65e9654cd85bc66d10208c4fb51b97bbf4d12c378ccfd6a5a5bf3032db51fa1eb92f6ce10e90f9c5c06f733dc911a321dd4c0ac2406d467215dab3f2b5 + default: 0x5f3d8ce40af0adbbe93a5f9bab97f2a9139a0ffa956aeadb13baaa10d2336e399e5f1aeb58c2a1ba65d13a0fe39b01c95a626001cc969493d06cdb450c26a87f7bf58dda1149efdc5f023299dfb44d70a9f906859e02c05185213fcf029cb3e62ce6fddcba7f6c9d2519ea1aacc8e1922af7b82d7479cd41d20282eb0d61aa54 randwidth: 1024 } { @@ -2140,7 +2141,7 @@ randcount: 32 randtype: data name_top: RndCnstAlertHandlerLfsrSeed - default: 0x4a52e672 + default: 0xcc696d54 randwidth: 32 } { @@ -2150,7 +2151,7 @@ randcount: 32 randtype: perm name_top: RndCnstAlertHandlerLfsrPerm - default: 0x1626ca235b3b6c124bfe5e54ebf4a934783055f0 + default: 0x2e8c1e3532551d3f32ff7f6275c336de060412a2 randwidth: 160 } ] @@ -4069,7 +4070,7 @@ randcount: 128 randtype: data name_top: RndCnstSramCtrlRetAonSramKey - default: 0x55e0f8150242862eb20eea9a1f754471 + default: 0x224f25dab6226bf545b7fc56675745fe randwidth: 128 } { @@ -4079,7 +4080,7 @@ randcount: 128 randtype: data name_top: RndCnstSramCtrlRetAonSramNonce - default: 0x6551512a112ded678ce824a08e29c772 + default: 0xec587dcb2a9253769d73f614f297c5f7 randwidth: 128 } { @@ -4089,7 +4090,7 @@ randcount: 32 randtype: data name_top: RndCnstSramCtrlRetAonLfsrSeed - default: 0x795a358e + default: 0x24f5ad9 randwidth: 32 } { @@ -4099,7 +4100,7 @@ randcount: 32 randtype: perm name_top: RndCnstSramCtrlRetAonLfsrPerm - default: 0x44cfc8b58b71209029f2b7fa22de86de7d5d0c38 + default: 0xd1ef554482059a9c7f7c28fc87519d5cb2684c2f randwidth: 160 } { @@ -4843,7 +4844,7 @@ randcount: 64 randtype: data name_top: RndCnstAesClearingLfsrSeed - default: 0xd1095889fbd8a7b5 + default: 0xbee8c0a3f22571ac randwidth: 64 } { @@ -4853,7 +4854,7 @@ randcount: 64 randtype: perm name_top: RndCnstAesClearingLfsrPerm - default: 0x392fa90de595ff713b4289dd4e117922432f280449db552506281ac8698b62bbb13cdf02b73d2a71b8d7f7ac191f8b1f + default: 0x967dd7a7a90eda298c67e0d0e05d09079b9f013d42c2a6b180446f21877b6e170f49efedf07546372acab23a33f4b594 randwidth: 384 } { @@ -4863,7 +4864,7 @@ randcount: 64 randtype: perm name_top: RndCnstAesClearingSharePerm - default: 0x3e5ab51c865bf6d4ddc1eb8653f32333b7dfca24814fa1f11d61b095a971ec8101b4c0e8d14a79392e689e6a9c6eec3 + default: 0x62a78d67ccc2d1bbcc2b604953a9a011f916757d77469c5ca3f0d0c23b1ae7e2e5f67e3baed1d3c8e3c5ba11a2488540 randwidth: 384 } { @@ -4873,7 +4874,7 @@ randcount: 288 randtype: data name_top: RndCnstAesMaskingLfsrSeed - default: 0x82fa14a8f2b75c265c9cb3b2f1c44a7287ab4c0618822e92a084a048ffe9fb86140a79d3 + default: 0x812b74cc981f89b9b95175a8f02ea42f7883754a0eaa875bd05ee93e58e0e42986cc76ac randwidth: 288 } { @@ -4883,7 +4884,7 @@ randcount: 160 randtype: perm name_top: RndCnstAesMaskingLfsrPerm - default: 0x706668853241427a1d5665590d530c47458c23975c21863c917c5160370783248b9a7f93736277065a392f1b8e7e6f4e81118a1c79589d74042b0817710f252d486488192a2e49004610129e1a3192288257843f894c090a50695440616a1e966c362038144d159535346d02875d9c18754a268f0b015b1f781376299827220e9f7b8d90437d4f6b3d723a05639967169b30446e52942c335e033e3b5f4b5580 + default: 0x3e319d0b5818107800733c3b1274948679110a7a892c5e9a6039852787045c5f018c0d164942214f72335b385d829509255a75227c8e593707083a1d973d81268a1b622076998424301e0c23962f9b037f19454c9e5132914e356d4d71772d4b474a8d694363366a667e6f9c46569050531f2a8354801a1c9f931561136898923f677006287b29402b57440241052e6552640e0f5588176b7d6e8b6c8f483414 randwidth: 1280 } ] @@ -5158,7 +5159,7 @@ randcount: 288 randtype: data name_top: RndCnstKmacLfsrSeed - default: 0xe933d88e2e1cf65460f23fb780499e6fcce64ceafd282c0e33fd2c07986c2a511755f072 + default: 0x534d7ac4bd0ca8b03546a98188556c8cbf9b24d1d32c9bd86dfed26f1fe72ffbb4b9762d randwidth: 288 } { @@ -5168,7 +5169,7 @@ randcount: 800 randtype: perm name_top: RndCnstKmacLfsrPerm - default: 0x99e8e2963e89488b131d0c872324f63425f3cede85ef7490761b0ec0ad251fc8163069a65313c5ada542e455d553aeb37506368df89c1876c38c3186c6aa35aced838342b42d847188a8c8685c27895eb4905481af5de7e87210799f63a6ad1388d0561b49c2a1e99b8763592caa899bb52e072e92ba9a20abd6235d97aa5183c30f09cff5bcde2d5b6ae2e17b9347dd7c97164a698e620cb22e087c2318b8090190182aa8262847c4f77240a1453ab18bf07857400be5bb0d07c504fe02279ec0fc901a03ec0c339a18b2554d68dcb19079beec0c1c169bd569b62826ace962b26fcbb3e4b11019ebe0a119b149e27da4222ac45f4230123306a30ee28244bb52ca91b27f56c759b0625d131a005f6ff09569886c5932f6b70d2345cf83128c7c9426e6b049d9575eb1a8995d16a728e190adb5547b2e6617e6d750da619da7216bd1b65d6ad9348f1699a543925e3bdd8039a5180002beb529c9638b000c19271ee67b5c65c140b94d65e51b6021259cc0a71e5476c36e9acb0ab960504a791c6426c9d1b4a1ce8adc622b1d7c0a50195b377c6b4d824afe9391ac1b4dd07d4a6218718b56151de7868fc38f17c1429106e97f1cd398b08d0a4beafd13c217490c355a09048a6aafe0a1c804dbc939cc98f1021cc45da92af8a3a2cfe6e6706c6d4a9e3c80492540ca1571911d2252a83aea4344603c1160a089f3615c82ee88b0d370d2aa5a49161e14e4802baef60a8fbe6d912a60944cc0c68a1f6b44cecd07449179a8516c9825e403cfbc557ba75fb314c25d45266910a8571900c3b819d2f19c8db0a6580b7e9097a1993ca47860b358c48850370047e2d88474dd428b1e1d4dc88e59271f49d4088a44bbb00d99112654e72de2fb68fd3ea3a9e50db7c3552129ada65ba93c118c7512056f90a650930650b532f56b93b2ca4d008a883eb3b88898ca45499662757fbc2ca6d58d1b8107c83d41e7c43e94b6224a615a544363645c4f82660df8b1ebbb2f1369054a44058fe1faaa2b9ed49779e94c15f6dddd1d1ce8edf3b3f08a7c5ab40cf08b1ba943b06db86897395c5bb20b8988429683c20985a1a89712f492914564637bceb8102cc3a7a7a98f6f4cd9b92e7fa55485547fd3d706523d1e31e23323a1962903c4a0605d3208e88518ce662f2836155497245ce59207353cf51326915aee9d80cb90a99a87725ed1702304d6ae4e16d9c64faf2130e941a35dc47e7220cb4a5c04c26d58dd166101c14ee74099f61d13114436a470174abb5a5b94e9b1bef1560208f6f35e091aad19596391851e3827f00c591f0c1069e03e2b03a942768d4a0039b8e7723d1a87a85a81c5b31370bd2f4b44db765601cbd04673ab16885aa255d7e54d915e81436f1cc2d0ebf15ba0444c4638 + default: 0x3f69d7b0444f62fb85e1bbcce9a23d2529fc2ca657aa713b124da4521a14c7d29761753441488b0585d2c74c7f2c23c8cc1179ec9b7145489bb894f5c7a124a973566537dc7c3cddbb06b6b1c384714216df2afdf0211c39b9824695c3e8d7921a37a8b787506c10f8399465bd559956913590c4c3100c189c5a7a38e4aa87e2d56da015aa9037ad08b54d847ec2041c3e73a36802cb6b6cd7b4450c14c312c834af80a966150cfab0b2034197a16e926e8835a58b53f751203cafaadc852c6930088d780fb86cd3096eb4bd0c8a232a408304ce187c0753ada40a9c6012d4ce3b174a816acf118217da20809a60f579559442ed962fd8cb971659ca7c25cba9abc548c64de7610639ae68c0e46e0156aa35a923c084a23e909627143a076097f5f1a2032231abb6c800984b92c1a0cca0bc6a53f8eb45e431601536d710d47578d87816a1b5ae7b0c599846251d34c22df3b1d25971b9719b1a21591ade77dc7bba844e0a576a7744914c04e24169e8ba9257d8b2b0dd86a555e2a8142c2306f2a6e86a5149c680a0d13bc72403a57e910978d0ba98ac0301408059305b19a0976400f0710703327e222f02b8dfc740e42a634c509a5c7819702b38632b4f19e53c844016fdaa4389c13e0a238a9aac9308e593e5d017ec4279ff31e789b5e21b6b05de8ac0c0b09e52812be46631370e0146d74981baa64c6e4b682eccbda746d13905e87bfca45c83bbcd884c1d1980871b02e2aa7f2261087b10af5a11bdb59066636918205b22757a9bc9555237724cb3567c026729470e9c1f4c603c28557898cd5c26c0bc297f1952cd6bc344063df9bb5cc6e0283d2d5b1ad044c0ca589f9cd8d7642b86135822eaa890f7a698abc47858d007cefab93ea83000419db8e4f5a06983d4367a5bad5fe19a29b770c80cdec1a5e91dc85d2f51846475c128e1153fd037996ca694b03dcd63a24a2d9984d81ab4571a8b9646f43096466aba94056929a22468f9a6588c01c5fc9b26223359f846c7b60e6b5eee06606b4125f5770f12472288c88302daccbe61a3f6487d8aa4d9d4bdb292789c821705e3f1f3871c0362d19f4674752344076b511e67db3b64e830e2c39d48240af53eb84e81e7adee325b09a9be52a89bf1500d873495c1c1cc510d611289aac4ed7752142ce31c5ca3b855b4c5349299c5e19a0c742d23aa54aba8d97616d2969110102401aec6de644cb0aa15582f0d962621352567d4e5beddcb8a4e0fc7a6f5ba7967a791ef838a36b9de9500095d4173e35229603297d6a0b812267ac9d6459fabf84c3dd2b628f675508b9ae9b912e435a4683151dd51bf563bdd1f406bbb157c1e71181ceab625d6b19c8adb16f1b6109c69ef145342225899249c2be5afa042c14a066515ba4460 randwidth: 8000 } { @@ -5178,7 +5179,7 @@ randcount: 800 randtype: data name_top: RndCnstKmacBufferLfsrSeed - default: 0x46c49f15a16a2a1fd01f8e09d92dd13e0298799d91c54bfd2a51f45aadf344c262bb0abde9b240d763da6c4250ba5a44a7d4f4d7f99b4a6480da2d53e7ca15201a8f56530010117f58c9e9946c11180407feb4710f404e6e4face59b972d1fdcaf711c05 + default: 0xb7480d9d5001b7755de3ef2618e28371dd02b1db04469e5770d10b0069e332be0f35a726b339d953a1dd95a3f78e14389414adc19300e0934269b3a30b61ea4d9790de0fb09d2792d26bf9565775a2843ffb7d42993c0a1c3d4d2937f997a3b51aafebc6 randwidth: 800 } { @@ -5188,7 +5189,7 @@ randcount: 64 randtype: perm name_top: RndCnstKmacMsgPerm - default: 0x1abba9f8ce16cf92d4b5c3c26435ecd4dde0615485c5d6e89bb244d3d7f0abc0641d346786323688abc0eb2fe541e39a + default: 0x2a89790db63eff5a927c7d96c2f0b23ba17cad7ede662023b24ccce2e804a4b689b48510f749a719c4c1ddd3d5851c4d randwidth: 384 } ] @@ -5382,7 +5383,7 @@ randcount: 256 randtype: data name_top: RndCnstOtbnUrndPrngSeed - default: 0xe24be7e1c96737b726a4c6f04993b3e6aae70f22aaf748b57c320264e157e2c7 + default: 0x4569b54e4cc3468ba78749a7093fe9eb381742e695130a4fb5238ebb74b4f311 randwidth: 256 } { @@ -5420,7 +5421,7 @@ randcount: 128 randtype: data name_top: RndCnstOtbnOtbnKey - default: 0x4a28033261ee6cec10075c55a820f842 + default: 0x8274ea0b356b051535eac4e098ea131f randwidth: 128 } { @@ -5430,7 +5431,7 @@ randcount: 64 randtype: data name_top: RndCnstOtbnOtbnNonce - default: 0xd43ee1ab1b3d78f9 + default: 0x95c233e670a4321b randwidth: 64 } ] @@ -5664,7 +5665,7 @@ randcount: 64 randtype: data name_top: RndCnstKeymgrDpeLfsrSeed - default: 0x92d223469898eb2a + default: 0xad599d4df689612a randwidth: 64 } { @@ -5674,7 +5675,7 @@ randcount: 64 randtype: perm name_top: RndCnstKeymgrDpeLfsrPerm - default: 0xba87c33dd8319de0a103ececead9692398efc95d9b9383505fc4b550b9b459caa265a3b0311f582412ab4c5ec71371bf + default: 0x961d0532d78d8ace1ff82767391f0f515416ebf4a96b95fb20bc2e1c472064af75c406efad88d3df3903076ca8266a86 randwidth: 384 } { @@ -5684,7 +5685,7 @@ randcount: 32 randtype: perm name_top: RndCnstKeymgrDpeRandPerm - default: 0xbce268203cc807b4b242f394f262a5b535f3f5ab + default: 0x95ca1a671b6b87f5f2cfd226a0445e21bb53c12c randwidth: 160 } { @@ -5694,7 +5695,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrDpeRevisionSeed - default: 0x5e456f3b26d15f5584bb9879bd2497852c23d0fa289c3348ce364ed534970e1f + default: 0x46794cfca2b0c6cbfb61f1bd7ed0a8f926da506ed059049dc9a6e2c1826aa5bf randwidth: 256 } { @@ -5704,7 +5705,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrDpeSoftOutputSeed - default: 0xd45ec606be134fc08318bd29b22b08df647bdc30068e7513200fa828bfee3ecc + default: 0xdf596fb9a391c6722b8b5cad051f3edb286e554001528f8e9a5e55ebb82a171c randwidth: 256 } { @@ -5714,7 +5715,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrDpeHardOutputSeed - default: 0xf02417fa68d004534267f60b34a8f072e77ee4859b243f92ceb098be6e65c359 + default: 0xf1e411adfe91411b1d8ee0161ecd699002ef8026e474e85739ef0277414266b4 randwidth: 256 } { @@ -5724,7 +5725,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrDpeAesSeed - default: 0x480858c39611e658c97fb8d3e9cc0e7f8fa8725200c726c8f90ef5e2b74beee2 + default: 0xf811ebd01e7ea4acf08b65a5cf51ae528473dd4661d013fb441057830dd1b830 randwidth: 256 } { @@ -5734,7 +5735,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrDpeKmacSeed - default: 0x7cfa19ea576ca8a8225c8e17053e44b68dd7822a6859f2cf52a55fd181b082d0 + default: 0xbf365bbfc1deb8921d7a6896fd1e6013622d66142ee83239ba8e92c61de956a5 randwidth: 256 } { @@ -5744,7 +5745,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrDpeOtbnSeed - default: 0x3c583756f11aa5b37f9544c4aa90eb97e1c0f8086f627a281c5681d6566641ee + default: 0x2768b6d5f1650f1b01969710a957dbfb2e2443f4fe2d0a760f83c1313d3ef78f randwidth: 256 } { @@ -5754,7 +5755,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrDpeNoneSeed - default: 0xcccc88876ad113c50ad1be4d94dce8fca54128a3860ae1d3493fbef6b9ba5dfe + default: 0xea4d3d48549217248330d048f78e497e63710e9720ac0f46c68e74cab245be89 randwidth: 256 } ] @@ -5958,7 +5959,7 @@ randcount: 384 randtype: data name_top: RndCnstCsrngCsKeymgrDivNonProduction - default: 0x4463a4615ef46762a794a8c2597e69fb259b961914cd75bbfd36ba48e780d27cd582540c7a68ff254edfd852829feccc + default: 0x11081c180f03480661762d84bfb803130ea592f63f968d16d49748cbc79f398e2139541731f82f0e5e0f2ea961c615fc randwidth: 384 } { @@ -5968,7 +5969,7 @@ randcount: 384 randtype: data name_top: RndCnstCsrngCsKeymgrDivProduction - default: 0xabd4af9c9032da20bdaf59fe2ae209b8325d2c8c35fc960679b106a87e59e6aeb1b5302d33401070251696fbb4ba169a + default: 0x96398950eded5e999cbb16670229d5a24df595be24c50547533e86ef45dc765bc3fc1c4492b8cd264903ba93e2074570 randwidth: 384 } { @@ -6312,7 +6313,7 @@ randcount: 128 randtype: data name_top: RndCnstSramCtrlMainSramKey - default: 0x6cd82fad46a0a5c04009bb934c7ef7b8 + default: 0x4fc7f5a263f7641c180b43ff715aa5a2 randwidth: 128 } { @@ -6322,7 +6323,7 @@ randcount: 128 randtype: data name_top: RndCnstSramCtrlMainSramNonce - default: 0x3b6e40610b4309fcb9dc54d4276137f9 + default: 0xbd79305346ebe883e11eea60c36419a2 randwidth: 128 } { @@ -6332,7 +6333,7 @@ randcount: 32 randtype: data name_top: RndCnstSramCtrlMainLfsrSeed - default: 0x901d09a7 + default: 0x27f0cbb5 randwidth: 32 } { @@ -6342,7 +6343,7 @@ randcount: 32 randtype: perm name_top: RndCnstSramCtrlMainLfsrPerm - default: 0x1036ed2be8b646c824a692f95f1031b9df89d3ad + default: 0x8e38b367e30f41ab4c553c0ae96e8a621b74bc9e randwidth: 160 } { @@ -6584,7 +6585,7 @@ randcount: 128 randtype: data name_top: RndCnstSramCtrlMboxSramKey - default: 0xe927f6d9e4abac398d42c745eef646c1 + default: 0x7c5f7fb39afb3660baba1f1bac202d76 randwidth: 128 } { @@ -6594,7 +6595,7 @@ randcount: 128 randtype: data name_top: RndCnstSramCtrlMboxSramNonce - default: 0x464dca86dafd7c7c71e6058ddfd871c5 + default: 0xe627cbb30ad1bc1e181db312b8d5a947 randwidth: 128 } { @@ -6604,7 +6605,7 @@ randcount: 32 randtype: data name_top: RndCnstSramCtrlMboxLfsrSeed - default: 0x1cacbaf4 + default: 0x9c07c534 randwidth: 32 } { @@ -6614,7 +6615,7 @@ randcount: 32 randtype: perm name_top: RndCnstSramCtrlMboxLfsrPerm - default: 0x2c32ae9387fda299e9ccf52abbc8627b4d8d8028 + default: 0xeebf897277731400c88d2ed0175ec6de69578d2 randwidth: 160 } { @@ -6855,7 +6856,7 @@ randcount: 64 randtype: data name_top: RndCnstRomCtrl0ScrNonce - default: 0x6fd468a77efde3de + default: 0xfa5ba5cf4337a897 randwidth: 64 } { @@ -6865,7 +6866,7 @@ randcount: 128 randtype: data name_top: RndCnstRomCtrl0ScrKey - default: 0x5b4caf4776a247baba4c9908ed16bc54 + default: 0x7e31c7b94c2bcf03781ee0d8b1ba84cb randwidth: 128 } { @@ -7039,7 +7040,7 @@ randcount: 64 randtype: data name_top: RndCnstRomCtrl1ScrNonce - default: 0x15ec16d28c535513 + default: 0xb565e8f2f8be62fe randwidth: 64 } { @@ -7049,7 +7050,7 @@ randcount: 128 randtype: data name_top: RndCnstRomCtrl1ScrKey - default: 0x12fcedcf2832a66ceacf8ed4d5b61617 + default: 0x9d2d4204b2f54be875ae480277680cba randwidth: 128 } { @@ -8955,7 +8956,7 @@ randcount: 32 randtype: data name_top: RndCnstRvCoreIbexLfsrSeed - default: 0x7dd43b56 + default: 0x442baa3a randwidth: 32 } { @@ -8965,7 +8966,7 @@ randcount: 32 randtype: perm name_top: RndCnstRvCoreIbexLfsrPerm - default: 0x1440b451b8d7081796663dfdbee6d2832aa1a5df + default: 0x959de8f20436d6d531ffc06fd48ea0d0a8839679 randwidth: 160 } { @@ -8975,7 +8976,7 @@ randcount: 128 randtype: data name_top: RndCnstRvCoreIbexIbexKeyDefault - default: 0x74f35c79f397c4e4c7e22b7581848a90 + default: 0xa690c16dd41b3d1fa05f7ecc83245ccc randwidth: 128 } { @@ -8985,7 +8986,7 @@ randcount: 64 randtype: data name_top: RndCnstRvCoreIbexIbexNonceDefault - default: 0xa1254b2276bec9fc + default: 0x8577daafb97c26a7 randwidth: 64 } { @@ -12411,7 +12412,7 @@ { hart: 0x30130000 } - size_byte: 0x1000 + size_byte: 0x8000 } ] xbar: false diff --git a/hw/top_darjeeling/data/otp/otp_ctrl_img_creator_sw_cfg.hjson b/hw/top_darjeeling/data/otp/otp_ctrl_img_creator_sw_cfg.hjson new file mode 100644 index 00000000000000..a57cd0d6421fe8 --- /dev/null +++ b/hw/top_darjeeling/data/otp/otp_ctrl_img_creator_sw_cfg.hjson @@ -0,0 +1,154 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Use the gen-otp-img.py script to convert this configuration into +// a MEM file for preloading the OTP in FPGA synthesis or simulation. +// + +{ + // The partition and item names must correspond with the OTP memory map. + partitions: [ + { + name: "CREATOR_SW_CFG", + items: [ + { + name: "CREATOR_SW_CFG_DIGEST", + value: "0x0", + }, + { + name: "CREATOR_SW_CFG_SIGVERIFY_RSA_MOD_EXP_IBEX_EN", + // Use software mod_exp implementation for signature + // verification. See the definition of `hardened_bool_t` in + // sw/lib/sw/device/base/hardened.h. + value: "0x739", + }, + { + name: "CREATOR_SW_CFG_SIGVERIFY_RSA_KEY_EN", + // Mark the first three keys as valid and remaining as + // invalid since we currently have only three keys. See the + // definition of `hardened_byte_bool_t` in + // sw/lib/sw/device/base/hardened.h. + value: "0x4ba5a5a5a5a5a5a5", + }, + { + name: "CREATOR_SW_CFG_SIGVERIFY_SPX_EN", + // Disable SPX+ signature verification. See the definitions + // of `kSigverifySpxDisabledOtp` and + // `kSigverifySpxEnabledOtp` in + // sw/lib/sw/device/silicon_creator/sigverify/spx_verify.h. + value: "0x8d6c8c17", + }, + { + name: "CREATOR_SW_CFG_SIGVERIFY_SPX_KEY_EN", + // Mark the first three keys as valid and remaining as + // invalid since we currently have only three keys. See the + // definition of `hardened_byte_bool_t` in + // sw/lib/sw/device/base/hardened.h. + value: "0x4ba5a5a5a5a5a5a5", + }, + { + name: "CREATOR_SW_CFG_FLASH_DATA_DEFAULT_CFG", + // Default values for flash scramble / ecc / he_en. This OTP + // word contains byte-aligned, packed, 4-bit mubi values. + // See the flash_ctrl driver bitfield definitions in + // sw/device/silicon_creator/lib/drivers/flash_ctrl.h. + value: "0x0", + }, + { + name: "CREATOR_SW_CFG_RNG_EN", + // Enable use of entropy for countermeasures. See the + // definition of `hardened_bool_t` in + // sw/lib/sw/device/base/hardened.h. + value: "0x739", + }, + { + name: "CREATOR_SW_CFG_ROM_EXEC_EN", + // ROM execution is enabled if this item is set to a + // non-zero value. + value: "0xffffffff", + }, + { + name: "CREATOR_SW_CFG_CPUCTRL", + // Value to write to the `cpuctrl` CSR in `rom_init()`. + // Note: Only bits 5:0 are written to the `cpuctrl` CSR. + // See: https://ibex-core.readthedocs.io/en/latest/03_reference/cs_registers.html#cpu-control-register-cpuctrl + value: "0x1", + }, + { + name: "CREATOR_SW_CFG_JITTER_EN", + value: "0x9", + }, + { + name: "CREATOR_SW_CFG_MIN_SEC_VER_ROM_EXT", + // Value of the min_security_version_rom_ext field of the + // default boot data. + value: "0x0", + }, + { + name: "CREATOR_SW_CFG_MIN_SEC_VER_BL0", + // Value of the min_security_version_bl0 field of the + // default boot data. + value: "0x0", + }, + { + name: "CREATOR_SW_CFG_DEFAULT_BOOT_DATA_IN_PROD_EN", + // Enable the default boot data in PROD and PROD_END life + // cycle states. See the definition of `hardened_bool_t` in + // sw/lib/sw/device/base/hardened.h. + value: "0x739", + }, + { + name: "CREATOR_SW_CFG_RNG_REPCNT_THRESHOLDS", + value: "0xffffffff", + } + { + name: "CREATOR_SW_CFG_RNG_REPCNTS_THRESHOLDS", + value: "0xffffffff", + } + { + name: "CREATOR_SW_CFG_RNG_ADAPTP_HI_THRESHOLDS", + value: "0xffffffff", + } + { + name: "CREATOR_SW_CFG_RNG_ADAPTP_LO_THRESHOLDS", + value: "0x0", + } + { + name: "CREATOR_SW_CFG_RNG_BUCKET_THRESHOLDS", + value: "0xffffffff", + } + { + name: "CREATOR_SW_CFG_RNG_MARKOV_HI_THRESHOLDS", + value: "0xffffffff", + } + { + name: "CREATOR_SW_CFG_RNG_MARKOV_LO_THRESHOLDS", + value: "0x0", + } + { + name: "CREATOR_SW_CFG_RNG_EXTHT_HI_THRESHOLDS", + value: "0xffffffff", + } + { + name: "CREATOR_SW_CFG_RNG_EXTHT_LO_THRESHOLDS", + value: "0x0", + } + { + name: "CREATOR_SW_CFG_RNG_ALERT_THRESHOLD", + value: "0xfffd0002", + } + { + name: "CREATOR_SW_CFG_RNG_HEALTH_CONFIG_DIGEST", + value: "0x8264cf75", + } + { + // Using the default register reset value. Two consecutive + // failures trigger an alert. + name: "CREATOR_SW_CFG_RNG_ALERT_THRESHOLD", + value: "0xfffd0002", + } + ], + } + ] +} diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_img_dev.hjson b/hw/top_darjeeling/data/otp/otp_ctrl_img_dev.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_img_dev.hjson rename to hw/top_darjeeling/data/otp/otp_ctrl_img_dev.hjson diff --git a/hw/top_darjeeling/data/otp/otp_ctrl_img_hw_cfg.hjson b/hw/top_darjeeling/data/otp/otp_ctrl_img_hw_cfg.hjson new file mode 100644 index 00000000000000..d0e10bf85f1292 --- /dev/null +++ b/hw/top_darjeeling/data/otp/otp_ctrl_img_hw_cfg.hjson @@ -0,0 +1,25 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Use the gen-otp-img.py script to convert this configuration into +// a MEM file for preloading the OTP in FPGA synthesis or simulation. +// + +{ + // The partition and item names must correspond with the OTP memory map. + partitions: [ + { + name: "HW_CFG0", + // If set to true, this computes the HW digest value + // and locks the partition. + lock: "True", + items: [ + { + name: "DEVICE_ID", + value: "", + }, + ], + } + ] +} diff --git a/hw/top_darjeeling/data/otp/otp_ctrl_img_owner_sw_cfg.hjson b/hw/top_darjeeling/data/otp/otp_ctrl_img_owner_sw_cfg.hjson new file mode 100644 index 00000000000000..53ee59e8ded9fa --- /dev/null +++ b/hw/top_darjeeling/data/otp/otp_ctrl_img_owner_sw_cfg.hjson @@ -0,0 +1,245 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Use the gen-otp-img.py script to convert this configuration into +// a MEM file for preloading the OTP in FPGA synthesis or simulation. +// + +{ + // The partition and item names must correspond with the OTP memory map. + partitions: [ + { + name: "OWNER_SW_CFG", + items: [ + { + name: "OWNER_SW_CFG_DIGEST", + value: "0x0", + }, + { + name: "OWNER_SW_CFG_ROM_BOOTSTRAP_DIS", + // Enable bootstrap. See `hardened_bool_t` in + // sw/lib/sw/device/base/hardened.h. + value: "0x1d4", + }, + { + name: "OWNER_SW_CFG_ROM_KEYMGR_ROM_EXT_MEAS_EN", + // Set to 0x739 to use the ROM_EXT hash measurement as the + // key manager attestation binding value. + value: "0x0", + }, + { + name: "OWNER_SW_CFG_ROM_ALERT_CLASS_EN" + // Set the enables to kAlertEnableNone. + // See `alert_enable_t` + // in sw/device/silicon_creator/lib/drivers/alert.h + value: "0xa9a9a9a9", + }, + { + name: "OWNER_SW_CFG_ROM_ALERT_ESCALATION" + // Set the esclation policies to kAlertEscalateNone. + // See `alert_escalate_t` + // in sw/device/silicon_creator/lib/drivers/alert.h + value: "0xd1d1d1d1", + }, + { + name: "OWNER_SW_CFG_ROM_ALERT_CLASSIFICATION" + // Set the classifiactions to kAlertClassX. + // See `alert_class_t` + // in sw/device/silicon_creator/lib/drivers/alert.h + value: [ + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + ], + }, + { + name: "OWNER_SW_CFG_ROM_LOCAL_ALERT_CLASSIFICATION" + // Set the classifiactions to kAlertClassX. + // See `alert_class_t` + // in sw/device/silicon_creator/lib/drivers/alert.h + value: [ + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + "0x94949494", + ], + }, + { + name: "OWNER_SW_CFG_ROM_ALERT_ACCUM_THRESH" + // Set the alert accumulation thresholds to 0 per class. + value: [ + "0x00000000", + "0x00000000", + "0x00000000", + "0x00000000", + ], + }, + { + name: "OWNER_SW_CFG_ROM_ALERT_TIMEOUT_CYCLES" + // Set the alert timeout cycles to 0 per class. + value: [ + "0x00000000", + "0x00000000", + "0x00000000", + "0x00000000", + ], + }, + { + name: "OWNER_SW_CFG_ROM_ALERT_PHASE_CYCLES" + // Set the alert phase cycles to 0,10,10,0xFFFFFFFF for + // classes A and B, and to all zeros for classes C and D. + value: [ + "0x0", "0xa", "0xa", "0xFFFFFFFF", + "0x0", "0xa", "0xa", "0xFFFFFFFF", + "0x0", "0x0", "0x0", "0x0", + "0x0", "0x0", "0x0", "0x0", + ], + }, + // Note, these values need updating whenever the top-specific + // alert_handler parameterization changes. To update the + // digests values, perform the following steps: + // + // 1. Make sure the alert_handler register map in + // + // sw/host/opentitanlib/src/otp/alert_handler_regs.rs + // + // is up to date. See also + // + // https://github.com/lowRISC/opentitan/issues/19501 + // + // 2. Run opentitantool + // + // opentitantool --rcfile="" otp alert-digest \ + // hw/top_darjeeling/data/otp/otp_ctrl_img_owner_sw_cfg.hjson + // + // and enter the new values in this file. + { + name: "OWNER_SW_CFG_ROM_ALERT_DIGEST_DEV", + value: "0x8a12908b", + }, + { + name: "OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD", + value: "0xe4bab764", + }, + { + name: "OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD_END", + value: "0x10f153e1", + }, + { + name: "OWNER_SW_CFG_ROM_ALERT_DIGEST_RMA", + value: "0x4ec41fc0", + } + ], + } + ] +} diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_img_prod.hjson b/hw/top_darjeeling/data/otp/otp_ctrl_img_prod.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_img_prod.hjson rename to hw/top_darjeeling/data/otp/otp_ctrl_img_prod.hjson diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_img_raw.hjson b/hw/top_darjeeling/data/otp/otp_ctrl_img_raw.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_img_raw.hjson rename to hw/top_darjeeling/data/otp/otp_ctrl_img_raw.hjson diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_img_rma.hjson b/hw/top_darjeeling/data/otp/otp_ctrl_img_rma.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_img_rma.hjson rename to hw/top_darjeeling/data/otp/otp_ctrl_img_rma.hjson diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_img_test_locked0.hjson b/hw/top_darjeeling/data/otp/otp_ctrl_img_test_locked0.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_img_test_locked0.hjson rename to hw/top_darjeeling/data/otp/otp_ctrl_img_test_locked0.hjson diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_img_test_locked1.hjson b/hw/top_darjeeling/data/otp/otp_ctrl_img_test_locked1.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_img_test_locked1.hjson rename to hw/top_darjeeling/data/otp/otp_ctrl_img_test_locked1.hjson diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_img_test_unlocked0.hjson b/hw/top_darjeeling/data/otp/otp_ctrl_img_test_unlocked0.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_img_test_unlocked0.hjson rename to hw/top_darjeeling/data/otp/otp_ctrl_img_test_unlocked0.hjson diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_img_test_unlocked1.hjson b/hw/top_darjeeling/data/otp/otp_ctrl_img_test_unlocked1.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_img_test_unlocked1.hjson rename to hw/top_darjeeling/data/otp/otp_ctrl_img_test_unlocked1.hjson diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_img_test_unlocked2.hjson b/hw/top_darjeeling/data/otp/otp_ctrl_img_test_unlocked2.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_img_test_unlocked2.hjson rename to hw/top_darjeeling/data/otp/otp_ctrl_img_test_unlocked2.hjson diff --git a/hw/top_darjeeling/data/otp/otp_ctrl_mmap.hjson b/hw/top_darjeeling/data/otp/otp_ctrl_mmap.hjson new file mode 100644 index 00000000000000..08169609519fe4 --- /dev/null +++ b/hw/top_darjeeling/data/otp/otp_ctrl_mmap.hjson @@ -0,0 +1,1102 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Use the gen-otp-mmap.py script to update dependent files (like documentation +// tables the comportable hjson and metadata SV package): +// +// $ ./util/design/gen-otp-mmap.py +// +// Make sure to regenerate the CSRs after converting the memory map: +// +// $ cd ${PROJ_ROOT} +// $ make -C hw regs +// + +{ + // 256 bit seed to be used for generation of partition item default values. + // Can be overridden on the command line with the --seed switch. + seed: "36021179872380457113239299468132194022238108125576166239904535336103582949069" + + otp: { + width: "2", // bytes + depth: "8192" + } + + // Definition of scrambling and digest constants and keys. + scrambling: { + key_size: "16", + iv_size: "8", + cnst_size: "16", + keys: [ + { + name: "Secret0Key", + value: "", + } + { + name: "Secret1Key", + value: "", + } + { + name: "Secret2Key", + value: "", + } + { + name: "Secret3Key", + value: "", + } + ] + digests: [ + // This is the consistency digest used by all partitions. + { + name: "CnstyDigest", + iv_value: "", + cnst_value: "", + } + // The other digest configurations below are used for + // key derivation and token hashing. + { + name: "FlashDataKey", + iv_value: "", + cnst_value: "", + } + { + name: "FlashAddrKey", + iv_value: "", + cnst_value: "", + } + { + name: "SramDataKey", + iv_value: "", + cnst_value: "", + } + ] + } + + // The enumeration order below defines the address map of the OTP controller, + // if the offsets are not defined explicitly via the "offset" key. + // Note that the digest items are added automatically to the address map. + partitions: [ + { + name: "VENDOR_TEST", + variant: "Unbuffered", + absorb: false, + size: "64", // in bytes + secret: false, + sw_digest: true, + hw_digest: false, + write_lock: "Digest", + read_lock: "CSR", + key_sel: "NoKey", + integrity: false, // Do not use integrity (ECC) on this partition. + bkout_type: false, // Do not generate a breakout type for this partition. + items: [ + { + name: "SCRATCH", + size: "56" + } + ], + desc: '''Vendor test partition. + This is reserved for manufacturing smoke checks. The OTP wrapper + control logic inside prim_otp is allowed to read/write to this + region. ECC uncorrectable errors seen on the functional prim_otp + interface will not lead to an alert for this partition. + Instead, such errors will be reported as correctable ECC errors. + ''' + } + { + name: "CREATOR_SW_CFG", + variant: "Unbuffered", + absorb: false, + size: "320", // in bytes + secret: false, + sw_digest: true, + hw_digest: false, + write_lock: "Digest", + read_lock: "CSR", + key_sel: "NoKey", + integrity: true, // Use integrity (ECC) on this partition. + bkout_type: false, // Do not generate a breakout type for this partition. + items: [ + { + name: "CREATOR_SW_CFG_AST_CFG", + size: "124" + } + { + name: "CREATOR_SW_CFG_AST_INIT_EN", + size: "4" + } + { + name: "CREATOR_SW_CFG_OVERRIDES", + size: "32" + } + { + name: "CREATOR_SW_CFG_ROM_EXT_SKU", + size: "4" + } + { + name: "CREATOR_SW_CFG_SIGVERIFY_RSA_MOD_EXP_IBEX_EN", + size: "4" + } + { + name: "CREATOR_SW_CFG_SIGVERIFY_RSA_KEY_EN", + size: "8" + } + { + name: "CREATOR_SW_CFG_SIGVERIFY_SPX_EN", + size: "4" + } + { + name: "CREATOR_SW_CFG_SIGVERIFY_SPX_KEY_EN", + size: "8" + } + { + name: "CREATOR_SW_CFG_FLASH_DATA_DEFAULT_CFG", + size: "4" + } + { + name: "CREATOR_SW_CFG_FLASH_INFO_BOOT_DATA_CFG", + size: "4" + } + { + name: "CREATOR_SW_CFG_FLASH_HW_INFO_CFG_OVERRIDE", + size: "4" + } + { + name: "CREATOR_SW_CFG_RNG_EN", + size: "4" + } + { + name: "CREATOR_SW_CFG_JITTER_EN", + size: "4" + } + { + name: "CREATOR_SW_CFG_RET_RAM_RESET_MASK", + size: "4" + } + { + name: "CREATOR_SW_CFG_MANUF_STATE", + size: "4" + } + { + name: "CREATOR_SW_CFG_ROM_EXEC_EN", + size: "4" + } + { + name: "CREATOR_SW_CFG_CPUCTRL", + size: "4" + } + { + name: "CREATOR_SW_CFG_MIN_SEC_VER_ROM_EXT", + size: "4" + } + { + name: "CREATOR_SW_CFG_MIN_SEC_VER_BL0", + size: "4" + } + { + name: "CREATOR_SW_CFG_DEFAULT_BOOT_DATA_IN_PROD_EN", + size: "4" + } + { + name: "CREATOR_SW_CFG_RMA_SPIN_EN", + size: "4" + } + { + name: "CREATOR_SW_CFG_RMA_SPIN_CYCLES", + size: "4" + } + { + name: "CREATOR_SW_CFG_RNG_REPCNT_THRESHOLDS", + size: "4" + } + { + name: "CREATOR_SW_CFG_RNG_REPCNTS_THRESHOLDS", + size: "4" + } + { + name: "CREATOR_SW_CFG_RNG_ADAPTP_HI_THRESHOLDS", + size: "4" + } + { + name: "CREATOR_SW_CFG_RNG_ADAPTP_LO_THRESHOLDS", + size: "4" + } + { + name: "CREATOR_SW_CFG_RNG_BUCKET_THRESHOLDS", + size: "4" + } + { + name: "CREATOR_SW_CFG_RNG_MARKOV_HI_THRESHOLDS", + size: "4" + } + { + name: "CREATOR_SW_CFG_RNG_MARKOV_LO_THRESHOLDS", + size: "4" + } + { + name: "CREATOR_SW_CFG_RNG_EXTHT_HI_THRESHOLDS", + size: "4" + } + { + name: "CREATOR_SW_CFG_RNG_EXTHT_LO_THRESHOLDS", + size: "4" + } + { + name: "CREATOR_SW_CFG_RNG_ALERT_THRESHOLD", + size: "4" + } + { + name: "CREATOR_SW_CFG_RNG_HEALTH_CONFIG_DIGEST", + size: "4" + } + { + name: "CREATOR_SW_CFG_SRAM_KEY_RENEW_EN", + size: "4" + } + ], + desc: '''Software configuration partition. + This is for device-specific calibration data. For example, clock, + LDO, RNG. + ''' + } + { + name: "OWNER_SW_CFG", + variant: "Unbuffered", + absorb: false, + size: "632", // in bytes + secret: false, + sw_digest: true, + hw_digest: false, + write_lock: "Digest", + read_lock: "CSR", + key_sel: "NoKey", + integrity: true, + bkout_type: false, + items: [ + { + name: "OWNER_SW_CFG_ROM_ERROR_REPORTING", + size: "4" + } + { + name: "OWNER_SW_CFG_ROM_BOOTSTRAP_DIS", + size: "4" + } + { + name: "OWNER_SW_CFG_ROM_ALERT_CLASS_EN", + size: "4" + } + { + name: "OWNER_SW_CFG_ROM_ALERT_ESCALATION", + size: "4" + } + { + name: "OWNER_SW_CFG_ROM_ALERT_CLASSIFICATION", + size: "400" + } + { + name: "OWNER_SW_CFG_ROM_LOCAL_ALERT_CLASSIFICATION", + size: "64" + } + { + name: "OWNER_SW_CFG_ROM_ALERT_ACCUM_THRESH", + size: "16" + } + { + name: "OWNER_SW_CFG_ROM_ALERT_TIMEOUT_CYCLES", + size: "16" + } + { + name: "OWNER_SW_CFG_ROM_ALERT_PHASE_CYCLES", + size: "64" + } + { + name: "OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD", + size: "4", + } + { + name: "OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD_END", + size: "4", + } + { + name: "OWNER_SW_CFG_ROM_ALERT_DIGEST_DEV", + size: "4", + } + { + name: "OWNER_SW_CFG_ROM_ALERT_DIGEST_RMA", + size: "4", + } + { + name: "OWNER_SW_CFG_ROM_WATCHDOG_BITE_THRESHOLD_CYCLES", + size: "4" + } + { + name: "OWNER_SW_CFG_ROM_KEYMGR_ROM_EXT_MEAS_EN", + size: "4" + } + { + name: "OWNER_SW_CFG_MANUF_STATE", + size: "4" + } + { + name: "OWNER_SW_CFG_ROM_RSTMGR_INFO_EN", + size: "4" + } + ], + desc: '''Software configuration partition. + This contains data that changes software behavior in the ROM, for + example enabling defensive features in ROM or selecting failure + modes if verification fails. + ''' + } + { + name: "OWNERSHIP_SLOT_STATE", + variant: "Unbuffered", + absorb: false, + secret: false, + sw_digest: false, + hw_digest: false, + write_lock: "None", + read_lock: "CSR", + key_sel: "NoKey", + integrity: true, + bkout_type: false, + items: [ + { + name: "OWNERSHIP_SLOT_STATE_ROT_OWNER_AUTH", + size: "16" + desc: ''' Creator Ownership transfer state management + OWNERSHIP_ST_RAW: ownership not yet claimed (factory default) + OWNERSHIP_ST_LOCKED0: first ownership slot claimed + OWNERSHIP_ST_RELEASED0: first ownership slot released (assets have been cleared, ready to xfer) + OWNERSHIP_ST_LOCKED1: second ownership slot claimed + OWNERSHIP_ST_SCRAPPED: scrap state - terminal + ''' + } + { + name: "OWNERSHIP_SLOT_STATE_PLAT_INTEG_AUTH", + size: "16" + desc: ''' Creator Ownership transfer state management + OWNERSHIP_ST_RAW: ownership not yet claimed (factory default) + OWNERSHIP_ST_LOCKED0: first ownership slot claimed + OWNERSHIP_ST_RELEASED0: first ownership slot released (assets have been cleared, ready to xfer) + OWNERSHIP_ST_LOCKED1: second ownership slot claimed + OWNERSHIP_ST_SCRAPPED: scrap state - terminal + ''' + } + { + name: "OWNERSHIP_SLOT_STATE_PLAT_OWNER_AUTH", + size: "16" + desc: ''' Creator Ownership transfer state management + OWNERSHIP_ST_RAW: ownership not yet claimed (factory default) + OWNERSHIP_ST_LOCKED0: first ownership slot claimed + OWNERSHIP_ST_RELEASED0: first ownership slot released (assets have been cleared, ready to xfer) + OWNERSHIP_ST_LOCKED1: second ownership slot claimed + OWNERSHIP_ST_RELEASED1: second ownership slot released (assets have been cleared, ready to xfer) + OWNERSHIP_ST_LOCKED2: third ownership slot claimed + OWNERSHIP_ST_RELEASED2: third ownership slot released (assets have been cleared, ready to xfer) + OWNERSHIP_ST_LOCKED3: fourth ownership slot claimed + OWNERSHIP_ST_SCRAPPED: scrap state - terminal + ''' + } + ] + desc: '''SW managed asset ownership states partition. + + Multibit enable value for the tracking the asset ownership states. + Note that the states can be written multiple times in a device lifetime. + The values to be written are engineered in the same way as the LC_CTRL + state encoding words so that the ECC encoding remains valid even after + updating the values. + + The constants can be found in the lc_ctrl_state_pkg.sv package. + + The programming order has to adhere to: + + OWNERSHIP_ST_RAW (factory all-zero state) -> + OWNERSHIP_ST_LOCKED0 -> + OWNERSHIP_ST_RELEASED0 -> + ... + OWNERSHIP_ST_SCRAPPED + + Note that if there are less than 4 slots available the higher slot states + become logically equivalent to OWNERSHIP_SCRAPPED (firmware has to handle + this correctly). + ''' + } + { + name: "ROT_CREATOR_AUTH", + variant: "Unbuffered", + absorb: false, + secret: false, + sw_digest: true, + hw_digest: false, + write_lock: "Digest", + read_lock: "CSR", + key_sel: "NoKey", + integrity: true, + bkout_type: false, + items: [ + { + name: "ROT_CREATOR_AUTH_NON_RAW_MFW_CODESIGN_KEY", + size: "160" + desc: ''' Pub Key used to verify the manufacturing provisioning Software container + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + { + name: "ROT_CREATOR_AUTH_OWNERSHIP_STATE", + size: "4" + desc: ''' Creator Ownership transfer state management + UNLOCKED - ownership of ROT not yet claimed + LOCKED - ownership claimed and + RELEASED - Release ownership (assets have been cleared, ready to xfer) + ''' + } + { + name: "ROT_CREATOR_AUTH_ROM2_PATCH_SIGVERIFY_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROM2 OTP based patch + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + OTP based key (instead of ROM) facilitates easy key rotation + ''' + } + { + name: "ROT_CREATOR_AUTH_KEYMANIFEST_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT CREATOR KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + { + name: "ROT_CREATOR_AUTH_UNLOCK4XFER_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT CREATOR KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + { + name: "ROT_CREATOR_AUTH_IDENTITY_CERT", + size: "768" + desc: ''' ROT Device Identity Certificate endorsed by Silicon Creator stable off device PKI + ''' + } + ], + desc: '''Software managed creator partition. + ''' + } + { + name: "ROT_OWNER_AUTH_SLOT0", + variant: "Unbuffered", + absorb: false, + secret: false, + sw_digest: true, + hw_digest: false, + write_lock: "Digest", + read_lock: "CSR", + key_sel: "NoKey", + integrity: true, + bkout_type: false, + items: [ + { + name: "ROT_OWNER_AUTH_SLOT0_KEYMANIFEST_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + { + name: "ROT_OWNER_AUTH_SLOT0_UNLOCK4XFER_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + ], + desc: '''Software managed owner slot 0 partition. + ''' + } + { + name: "ROT_OWNER_AUTH_SLOT1", + variant: "Unbuffered", + absorb: false, + secret: false, + sw_digest: true, + hw_digest: false, + write_lock: "Digest", + read_lock: "CSR", + key_sel: "NoKey", + integrity: true, + bkout_type: false, + items: [ + { + name: "ROT_OWNER_AUTH_SLOT1_KEYMANIFEST_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + { + name: "ROT_OWNER_AUTH_SLOT1_UNLOCK4XFER_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + ], + desc: '''Software managed owner slot 1 partition. + ''' + } + { + name: "PLAT_INTEG_AUTH_SLOT0", + variant: "Unbuffered", + absorb: false, + secret: false, + sw_digest: true, + hw_digest: false, + write_lock: "Digest", + read_lock: "CSR", + key_sel: "NoKey", + integrity: true, + bkout_type: false, + items: [ + { + name: "PLAT_INTEG_AUTH_SLOT0_KEYMANIFEST_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + { + name: "PLAT_INTEG_AUTH_SLOT0_UNLOCK4XFER_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + ], + desc: '''Software managed platform integrator slot 0 partition. + ''' + } + { + name: "PLAT_INTEG_AUTH_SLOT1", + variant: "Unbuffered", + absorb: false, + secret: false, + sw_digest: true, + hw_digest: false, + write_lock: "Digest", + read_lock: "CSR", + key_sel: "NoKey", + integrity: true, + bkout_type: false, + items: [ + { + name: "PLAT_INTEG_AUTH_SLOT1_KEYMANIFEST_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + { + name: "PLAT_INTEG_AUTH_SLOT1_UNLOCK4XFER_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + ], + desc: '''Software managed platform integrator slot 1 partition. + ''' + } + { + name: "PLAT_OWNER_AUTH_SLOT0", + variant: "Unbuffered", + absorb: false, + secret: false, + sw_digest: true, + hw_digest: false, + write_lock: "Digest", + read_lock: "CSR", + key_sel: "NoKey", + integrity: true, + bkout_type: false, + items: [ + { + name: "PLAT_OWNER_AUTH_SLOT0_KEYMANIFEST_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + { + name: "PLAT_OWNER_AUTH_SLOT0_UNLOCK4XFER_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + ], + desc: '''Software managed platform owner slot 0 partition. + ''' + } + { + name: "PLAT_OWNER_AUTH_SLOT1", + variant: "Unbuffered", + absorb: false, + secret: false, + sw_digest: true, + hw_digest: false, + write_lock: "Digest", + read_lock: "CSR", + key_sel: "NoKey", + integrity: true, + bkout_type: false, + items: [ + { + name: "PLAT_OWNER_AUTH_SLOT1_KEYMANIFEST_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + { + name: "PLAT_OWNER_AUTH_SLOT1_UNLOCK4XFER_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + ], + desc: '''Software managed platform owner slot 1 partition. + ''' + } + { + name: "PLAT_OWNER_AUTH_SLOT2", + variant: "Unbuffered", + absorb: false, + secret: false, + sw_digest: true, + hw_digest: false, + write_lock: "Digest", + read_lock: "CSR", + key_sel: "NoKey", + integrity: true, + bkout_type: false, + items: [ + { + name: "PLAT_OWNER_AUTH_SLOT2_KEYMANIFEST_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + { + name: "PLAT_OWNER_AUTH_SLOT2_UNLOCK4XFER_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + ], + desc: '''Software managed platform owner slot 2 partition. + ''' + } + { + name: "PLAT_OWNER_AUTH_SLOT3", + variant: "Unbuffered", + absorb: false, + secret: false, + sw_digest: true, + hw_digest: false, + write_lock: "Digest", + read_lock: "CSR", + key_sel: "NoKey", + integrity: true, + bkout_type: false, + items: [ + { + name: "PLAT_OWNER_AUTH_SLOT3_KEYMANIFEST_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + { + name: "PLAT_OWNER_AUTH_SLOT3_UNLOCK4XFER_KEY", + size: "160" + desc: ''' Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + } + ], + desc: '''Software managed platform owner slot 3 partition. + ''' + } + { + name: "EXT_NVM", + variant: "Unbuffered", + absorb: false, + secret: false, + sw_digest: false, + hw_digest: false, + write_lock: "None", + read_lock: "CSR", + key_sel: "NoKey", + // This is a strike counter, hence we need to disable ECC integrity for this to work. + // Integrity is handled at a higher level by SW as described below. + integrity: false, + bkout_type: false, + items: [ + { + name: "EXT_NVM_ANTIREPLAY_FRESHNESS_CNT", + size: "1024" + desc: ''' Fully SW managed patition + 1 bit (double bit allocated for redundancy - i.e. 2 physical bits for every logical bit) per day + assuming 10 year product lifecycle, allocating 8192 bits of freshness protection. + Implemented as thermometer encoding. Redundant bit programming & logical OR-ing managed by + software. Cannot be ECC protected since different bits in a fuse row are programmed at different times. + ''' + } + ], + desc: '''Anti-replay protection Strike Counters partition. + ''' + } + { + name: "ROM_PATCH", + variant: "Unbuffered", + absorb: true, + secret: false, + sw_digest: true, + hw_digest: false, + write_lock: "Digest", + read_lock: "CSR", + key_sel: "NoKey", + integrity: true, + bkout_type: false, + items: [ + { + name: "ROM_PATCH_DATA", + size: "9192" + desc: ''' Fully SW managed patition + ROM patch section + signed patch(es) + Each patch Contains header, signature, patch match/redirect configuration and patch code + Please refer to the ROM boot & patching specification for Integrated OT for more details + ''' + } + ], + desc: '''ROM Patch Code section. May contain multiple signed ROM2 patches. + ''' + } + { + name: "HW_CFG0", + variant: "Buffered", + secret: false, + sw_digest: false, + hw_digest: true, + write_lock: "Digest", + read_lock: "None", + key_sel: "NoKey", + integrity: true, + bkout_type: true, + items: [ + { + name: "DEVICE_ID", + size: "32", + // Default value to be output in case partition has not + // initialized or is in error state. If not specified, + // a value of '0 will be used. + inv_default: "", + }, + { + name: "MANUF_STATE", + size: "32", + inv_default: "", + } + ], + desc: '''Hardware configuration 0 partition. + This contains + - DEVICE_ID: Unique device identifier. + - MANUF_STATE: Vector for capturing the manufacturing status. + ''' + } + { + name: "HW_CFG1", + variant: "Buffered", + secret: false, + sw_digest: false, + hw_digest: true, + write_lock: "Digest", + read_lock: "None", + key_sel: "NoKey", + integrity: true, + bkout_type: true, + items: [ + { + name: "SOC_DBG_STATE", + size: "4", + inv_default: "0x0" + desc: '''Multibit enable value for the SOC debug authorization. + Encoding: + SOC_DBG_RAW: this value is all-zeroes and will be the NOP state; + the LC controller will take precedence. + SOC_DBG_PRE_PROD: this is where the ROT will be in PROD state + but SOC will be in the pre-production unlock state + SOC_DBG_PROD: this is the state where the SOC moves to production, + and the challenge-response based authentication protocol is + required to unlock SOC debug features + ''' + }, + { + name: "EN_SRAM_IFETCH", + size: "1", + ismubi: true, + inv_default: false + }, + ], + desc: '''Hardware configuration 1 partition. + This contains + EN_SRAM_IFETCH: Enable / disable execute from SRAM CSR switch. + SOC_DBG_STATE: Multibit enable value for the SOC debug authorization. + Note SOC_DBG_STATE will be written twice in a device lifetime. The + values to be written are engineered in the same way as the LC_CTRL + state encoding words: the ECC encoding remains valid even after + writing the second value on top of the first. + + The constants can be found in the lc_ctrl_state_pkg.sv package. + + The programming order has to adhere to: + + SOC_DBG_RAW -> SOC_DBG_PRE_PROD -> SOC_DBG_PROD. + ''' + } + { + name: "SECRET0", + variant: "Buffered", + secret: true, + sw_digest: false, + hw_digest: true, + write_lock: "Digest", + read_lock: "Digest", + key_sel: "Secret0Key", + integrity: true, + bkout_type: false, + items: [ + { + name: "TEST_UNLOCK_TOKEN", + // This will generate a random default to be output in + // case partition has not initialized or is in error state. + // If not specified, a value of '0 will be used. + inv_default: "", + size: "16" + } + { + name: "TEST_EXIT_TOKEN", + inv_default: "", + size: "16" + } + ], + desc: '''Secret partition 0. + This contains TEST lifecycle unlock tokens. + ''' + } + { + name: "SECRET1", + variant: "Buffered", + secret: true, + sw_digest: false, + hw_digest: true, + write_lock: "Digest", + read_lock: "Digest", + key_sel: "Secret1Key", + integrity: true, + bkout_type: false, + items: [ + { + name: "FLASH_ADDR_KEY_SEED", + inv_default: "", + size: "32" + } + { + name: "FLASH_DATA_KEY_SEED", + inv_default: "", + size: "32" + } + { + name: "SRAM_DATA_KEY_SEED", + inv_default: "", + size: "16" + } + ], + desc: '''Secret partition 1. + This contains SRAM and flash scrambling keys. + ''' + } + { + name: "SECRET2", + variant: "Buffered", + secret: true, + sw_digest: false, + hw_digest: true, + write_lock: "Digest", + read_lock: "Digest", + key_sel: "Secret2Key", + integrity: true, + bkout_type: false, + items: [ + { + name: "RMA_TOKEN", + inv_default: "", + size: "16" + } + { + name: "CREATOR_ROOT_KEY_SHARE0", + inv_default: "", + size: "32", + iskeymgr_creator: true + } + { + name: "CREATOR_ROOT_KEY_SHARE1", + inv_default: "", + size: "32" + iskeymgr_creator: true + } + { + name: "CREATOR_SEED", + inv_default: "", + size: "32", + iskeymgr_creator: true + } + ], + desc: '''Secret partition 2. + This contains RMA unlock token, creator root key, and creator seed. + ''' + } + { + name: "SECRET3", + variant: "Buffered", + secret: true, + sw_digest: false, + hw_digest: true, + write_lock: "Digest", + read_lock: "Digest", + key_sel: "Secret3Key", + integrity: true, + bkout_type: false, + items: [ + { + name: "OWNER_SEED", + inv_default: "", + size: "32", + iskeymgr_owner: true + } + ], + desc: '''Secret partition 3. + This contains the owner seed. + ''' + } + { + name: "LIFE_CYCLE", + variant: "LifeCycle", + secret: false, + sw_digest: false, + hw_digest: false, + write_lock: "None", + read_lock: "None", + key_sel: "NoKey", + integrity: true, + bkout_type: false, + items: [ + // The life cycle transition count is specified + // first such that any programming attempt of the life cycle + // partition through the LCI will always write the transition + // counter words first when programming an updated state vector. + // This is an additional safeguard, to the sequencing in the + // life cycle controller to ensure that the counter is always written + // before any state update. I.e., the life cycle controller + // already splits the counter and state updates into two + // supsequent requests through the LCI, where the first request + // only contains the updated transition counter, and the second + // request the updated transition counter and state. + { + name: "LC_TRANSITION_CNT", + inv_default: "", + size: "48" + } + { + name: "LC_STATE", + inv_default: "", + size: "40" + } + ], + desc: '''Lifecycle partition. + This contains lifecycle transition count and state. This partition + cannot be locked since the life cycle state needs to advance to RMA + in-field. Note that while this partition is not marked secret, it + is not readable nor writeable via the DAI. Only the LC controller + can access this partition, and even via the LC controller it is not + possible to read the raw manufacturing life cycle state in encoded + form, since that encoding is considered a netlist secret. The LC + controller only exposes a decoded version of this state. + ''' + } + ] +} diff --git a/hw/top_darjeeling/data/top_darjeeling.hjson b/hw/top_darjeeling/data/top_darjeeling.hjson index 489db7b9595250..1bc337b5670cb0 100644 --- a/hw/top_darjeeling/data/top_darjeeling.hjson +++ b/hw/top_darjeeling/data/top_darjeeling.hjson @@ -321,6 +321,7 @@ core: {hart: "0x30130000"}, prim: {hart: "0x30138000"}, } + attr: "ipgen", }, { name: "lc_ctrl", type: "lc_ctrl", diff --git a/hw/top_darjeeling/dv/autogen/xbar_env_pkg__params.sv b/hw/top_darjeeling/dv/autogen/xbar_env_pkg__params.sv index 3a9c2ca599a6d2..4e071f693a6beb 100644 --- a/hw/top_darjeeling/dv/autogen/xbar_env_pkg__params.sv +++ b/hw/top_darjeeling/dv/autogen/xbar_env_pkg__params.sv @@ -137,7 +137,7 @@ tl_device_t xbar_devices[$] = '{ '{32'h30460000, 32'h304607ff} }}, '{"otp_ctrl__core", '{ - '{32'h30130000, 32'h30130fff} + '{32'h30130000, 32'h30137fff} }}, '{"otp_ctrl__prim", '{ '{32'h30138000, 32'h3013801f} diff --git a/hw/top_darjeeling/dv/autogen/xbar_tgl_excl.cfg b/hw/top_darjeeling/dv/autogen/xbar_tgl_excl.cfg index b2763b01cb5eb8..d85bc202aec457 100644 --- a/hw/top_darjeeling/dv/autogen/xbar_tgl_excl.cfg +++ b/hw/top_darjeeling/dv/autogen/xbar_tgl_excl.cfg @@ -184,7 +184,7 @@ -node tb.dut*.u_pinmux_aon tl_*i.a_address[21:19] -node tb.dut*.u_pinmux_aon tl_*i.a_address[27:23] -node tb.dut*.u_pinmux_aon tl_*i.a_address[31:30] --node tb.dut*.u_otp_ctrl core_tl_*i.a_address[15:12] +-node tb.dut*.u_otp_ctrl core_tl_*i.a_address[15:15] -node tb.dut*.u_otp_ctrl core_tl_*i.a_address[19:18] -node tb.dut*.u_otp_ctrl core_tl_*i.a_address[27:21] -node tb.dut*.u_otp_ctrl core_tl_*i.a_address[31:30] diff --git a/hw/top_darjeeling/dv/top_darjeeling_sim_cfgs.hjson b/hw/top_darjeeling/dv/top_darjeeling_sim_cfgs.hjson index 162520da8f3a9d..2c70d84b659849 100644 --- a/hw/top_darjeeling/dv/top_darjeeling_sim_cfgs.hjson +++ b/hw/top_darjeeling/dv/top_darjeeling_sim_cfgs.hjson @@ -36,7 +36,6 @@ "{proj_root}/hw/ip/lc_ctrl/dv/lc_ctrl_dmi_volatile_unlock_enabled_sim_cfg.hjson", "{proj_root}/hw/ip/mbx/dv/mbx_sim_cfg.hjson", "{proj_root}/hw/ip/otbn/dv/uvm/otbn_sim_cfg.hjson", - "{proj_root}/hw/ip/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson", "{proj_root}/hw/ip/prim/dv/prim_alert/prim_alert_sim_cfg.hjson", "{proj_root}/hw/ip/prim/dv/prim_esc/prim_esc_sim_cfg.hjson", "{proj_root}/hw/ip/prim/dv/prim_lfsr/prim_lfsr_sim_cfg.hjson", @@ -56,6 +55,7 @@ "{proj_root}/hw/top_darjeeling/ip_autogen/alert_handler/dv/alert_handler_sim_cfg.hjson", "{proj_root}/hw/top_darjeeling/ip_autogen/clkmgr/dv/clkmgr_sim_cfg.hjson", "{proj_root}/hw/top_darjeeling/ip_autogen/flash_ctrl/dv/flash_ctrl_sim_cfg.hjson", + "{proj_root}/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson", "{proj_root}/hw/top_darjeeling/ip_autogen/pwrmgr/dv/pwrmgr_sim_cfg.hjson", "{proj_root}/hw/top_darjeeling/ip_autogen/rstmgr/dv/rstmgr_cnsty_chk/rstmgr_cnsty_chk_sim_cfg.hjson", "{proj_root}/hw/top_darjeeling/ip_autogen/rstmgr/dv/rstmgr_sim_cfg.hjson", diff --git a/hw/top_darjeeling/ip/xbar_peri/data/autogen/xbar_peri.gen.hjson b/hw/top_darjeeling/ip/xbar_peri/data/autogen/xbar_peri.gen.hjson index b24bc7fa51bdd6..b2574927d6a008 100644 --- a/hw/top_darjeeling/ip/xbar_peri/data/autogen/xbar_peri.gen.hjson +++ b/hw/top_darjeeling/ip/xbar_peri/data/autogen/xbar_peri.gen.hjson @@ -295,7 +295,7 @@ { hart: 0x30130000 } - size_byte: 0x1000 + size_byte: 0x8000 } ] xbar: false diff --git a/hw/top_darjeeling/ip/xbar_peri/dv/autogen/xbar_cover.cfg b/hw/top_darjeeling/ip/xbar_peri/dv/autogen/xbar_cover.cfg index a71c3ee01b1a11..82de3e8f525788 100644 --- a/hw/top_darjeeling/ip/xbar_peri/dv/autogen/xbar_cover.cfg +++ b/hw/top_darjeeling/ip/xbar_peri/dv/autogen/xbar_cover.cfg @@ -54,7 +54,7 @@ -node tb.dut tl_pinmux_aon_o.a_address[21:19] -node tb.dut tl_pinmux_aon_o.a_address[27:23] -node tb.dut tl_pinmux_aon_o.a_address[31:30] --node tb.dut tl_otp_ctrl__core_o.a_address[15:12] +-node tb.dut tl_otp_ctrl__core_o.a_address[15:15] -node tb.dut tl_otp_ctrl__core_o.a_address[19:18] -node tb.dut tl_otp_ctrl__core_o.a_address[27:21] -node tb.dut tl_otp_ctrl__core_o.a_address[31:30] diff --git a/hw/top_darjeeling/ip/xbar_peri/dv/autogen/xbar_env_pkg__params.sv b/hw/top_darjeeling/ip/xbar_peri/dv/autogen/xbar_env_pkg__params.sv index 2f93c5bb4cb2e3..4d2a6e2b59fc47 100644 --- a/hw/top_darjeeling/ip/xbar_peri/dv/autogen/xbar_env_pkg__params.sv +++ b/hw/top_darjeeling/ip/xbar_peri/dv/autogen/xbar_env_pkg__params.sv @@ -38,7 +38,7 @@ tl_device_t xbar_devices[$] = '{ '{32'h30460000, 32'h304607ff} }}, '{"otp_ctrl__core", '{ - '{32'h30130000, 32'h30130fff} + '{32'h30130000, 32'h30137fff} }}, '{"otp_ctrl__prim", '{ '{32'h30138000, 32'h3013801f} diff --git a/hw/top_darjeeling/ip/xbar_peri/rtl/autogen/tl_peri_pkg.sv b/hw/top_darjeeling/ip/xbar_peri/rtl/autogen/tl_peri_pkg.sv index 1740d9ca1f50aa..13b832121ffa36 100644 --- a/hw/top_darjeeling/ip/xbar_peri/rtl/autogen/tl_peri_pkg.sv +++ b/hw/top_darjeeling/ip/xbar_peri/rtl/autogen/tl_peri_pkg.sv @@ -37,7 +37,7 @@ package tl_peri_pkg; localparam logic [31:0] ADDR_MASK_RSTMGR_AON = 32'h 0000007f; localparam logic [31:0] ADDR_MASK_CLKMGR_AON = 32'h 0000007f; localparam logic [31:0] ADDR_MASK_PINMUX_AON = 32'h 000007ff; - localparam logic [31:0] ADDR_MASK_OTP_CTRL__CORE = 32'h 00000fff; + localparam logic [31:0] ADDR_MASK_OTP_CTRL__CORE = 32'h 00007fff; localparam logic [31:0] ADDR_MASK_OTP_CTRL__PRIM = 32'h 0000001f; localparam logic [31:0] ADDR_MASK_LC_CTRL__REGS = 32'h 000000ff; localparam logic [31:0] ADDR_MASK_SENSOR_CTRL = 32'h 0000003f; diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/BUILD b/hw/top_darjeeling/ip_autogen/otp_ctrl/BUILD new file mode 100644 index 00000000000000..dda56de039e27f --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/BUILD @@ -0,0 +1,10 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 + +package(default_visibility = ["//visibility:public"]) + +filegroup( + name = "all_files", + srcs = glob(["**"]), +) diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/README.md b/hw/top_darjeeling/ip_autogen/otp_ctrl/README.md new file mode 100644 index 00000000000000..43ddcd76f4a908 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/README.md @@ -0,0 +1,63 @@ +# OTP Controller Technical Specification + +[`otp_ctrl`](https://reports.opentitan.org/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/latest/report.html): +![](https://dashboards.lowrisc.org/badges/dv/otp_ctrl/test.svg) +![](https://dashboards.lowrisc.org/badges/dv/otp_ctrl/passing.svg) +![](https://dashboards.lowrisc.org/badges/dv/otp_ctrl/functional.svg) +![](https://dashboards.lowrisc.org/badges/dv/otp_ctrl/code.svg) + +# Overview + +This document specifies the functionality of the one time programmable (OTP) memory controller. +The OTP controller is a module that is a peripheral on the chip interconnect bus, and thus follows the [Comportability Specification](../../../../doc/contributing/hw/comportability/README.md). + +The OTP is a module that provides a device with one-time-programming functionality. +The result of this programming is non-volatile, and unlike flash, cannot be reversed. +The OTP functionality is constructed through an open-source OTP controller and a proprietary OTP IP. + +The OTP controller provides: +- An open-source abstraction interface that software can use to interact with a proprietary OTP block underneath. +- An open-source abstraction interface that hardware components (for example [life cycle controller](../../../ip/lc_ctrl/README.md) and [key manager](../../../ip/keymgr/README.md)) can use to interact with a proprietary OTP block underneath. +- High level logical security protection, such as integrity checks and scrambling of sensitive content. +- Software isolation for when OTP contents are readable and programmable. + +The proprietary OTP IP provides: +- Reliable, non-volatile storage. +- Technology-specific redundancy or error correction mechanisms. +- Physical defensive features such as SCA and FI resistance. +- Visual and electrical probing resistance. + +Together, the OTP controller and IP provide secure one-time-programming functionality that is used throughout the life cycle (LC) of a device. + +## Features + +- Multiple logical partitions of the underlying OTP IP + - Each partition is lockable and integrity checked + - Integrity digests are stored alongside each logical bank +- Periodic / persistent checks of OTP values + - Periodic checks of shadowed content vs digests + - Periodic checks of OTP stored content and shadowed content + - Persistent checks for immediate errors +- Separate life cycle partition and interface to life cycle controller + - Supports life cycle functions, but cannot be integrity locked +- Lightweight scrambling of secret OTP partition using a global netlist constant +- Lightweight ephemeral key derivation function for RAM scrambling mechanisms +- Lightweight key derivation function for FLASH scrambling mechanism + +## OTP Controller Overview + +The functionality of OTP is split into an open-source and a closed-source part, with a clearly defined boundary in between, as illustrated in the simplified high-level block diagram below. + +![OTP Controller Overview](./doc/otp_ctrl_overview.svg) + +It is the task of the open-source controller to provide a common, non-technology specific interface to OTP users with a common register interface and a clearly defined I/O interface to hardware. +The open-source controller implements logical isolation and partitioning of OTP storage that enables users to separate different functions of the OTP into "partitions" with different properties. +Finally, the open-source controller provides a high level of security for specific partitions by provisioning integrity digests for each partition, and scrambling of partitions where required. + +The proprietary IP on the other hand translates a common access interface to the technology-specific OTP interface, both for functional and debug accesses (for example register accesses to the macro-internal control structure). + +This split implies that every proprietary OTP IP must implement a translation layer from a standardized OpenTitan interface to the module underneath. +It also implies that no matter how the OTP storage or word size may change underneath, the open-source controller must present a consistent and coherent software and hardware interface. +This standardized interface is defined further below, and the wrapper leverages the same [technology primitive mechanism](../../../ip/prim/README.md) that is employed in other parts of OpenTitan in order to wrap and abstract technology-specific macros (such as memories and clocking cells) that are potentially closed-source. + +In order to enable simulation and FPGA emulation of the OTP controller even without access to the proprietary OTP IP, a generalized and synthesizable model of the OTP IP is provided in the form of a [generic technology primitive](https://github.com/lowRISC/opentitan/blob/master/hw/ip/prim_generic/rtl/prim_generic_otp.sv). diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet b/hw/top_darjeeling/ip_autogen/otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet new file mode 100644 index 00000000000000..766a1a333d5840 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet @@ -0,0 +1,684 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +<% +from topgen.lib import Name + +parts = otp_mmap["partitions"] +digest_parts = [part for part in parts if + part["hw_digest"] or part["sw_digest"]] +read_locked_csr_parts = [part for part in parts if part["read_lock"] == "CSR"] +secret_parts = [part for part in parts if part["secret"]] +%>\ +#include "sw/device/lib/dif/dif_otp_ctrl.h" + +#include + +#include "sw/device/lib/base/bitfield.h" +#include "sw/device/lib/base/macros.h" +#include "sw/device/lib/dif/dif_base.h" + +#include "otp_ctrl_regs.h" // Generated. + +/** + * Checks if integrity/consistency-check-related operations are locked. + * + * This is a convenience function to avoid superfluous error-checking in all the + * functions that can be locked out by this register. + * + * @param check_config True to check the config regwen. False to check the + * trigger regwen. + */ +static bool checks_are_locked(const dif_otp_ctrl_t *otp, bool check_config) { + ptrdiff_t reg_offset = check_config + ? OTP_CTRL_CHECK_REGWEN_REG_OFFSET + : OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET; + size_t regwen_bit = + check_config ? OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT + : OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT; + uint32_t locked = mmio_region_read32(otp->base_addr, reg_offset); + return !bitfield_bit32_read(locked, regwen_bit); +} + +dif_result_t dif_otp_ctrl_configure(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_config_t config) { + if (otp == NULL) { + return kDifBadArg; + } + if (checks_are_locked(otp, /*check_config=*/true)) { + return kDifLocked; + } + + mmio_region_write32(otp->base_addr, OTP_CTRL_CHECK_TIMEOUT_REG_OFFSET, + config.check_timeout); + mmio_region_write32(otp->base_addr, + OTP_CTRL_INTEGRITY_CHECK_PERIOD_REG_OFFSET, + config.integrity_period_mask); + mmio_region_write32(otp->base_addr, + OTP_CTRL_CONSISTENCY_CHECK_PERIOD_REG_OFFSET, + config.consistency_period_mask); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_check_integrity(const dif_otp_ctrl_t *otp) { + if (otp == NULL) { + return kDifBadArg; + } + if (checks_are_locked(otp, /*check_config=*/false)) { + return kDifLocked; + } + + uint32_t reg = + bitfield_bit32_write(0, OTP_CTRL_CHECK_TRIGGER_INTEGRITY_BIT, true); + mmio_region_write32(otp->base_addr, OTP_CTRL_CHECK_TRIGGER_REG_OFFSET, reg); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_check_consistency(const dif_otp_ctrl_t *otp) { + if (otp == NULL) { + return kDifBadArg; + } + if (checks_are_locked(otp, /*check_config=*/false)) { + return kDifLocked; + } + + uint32_t reg = + bitfield_bit32_write(0, OTP_CTRL_CHECK_TRIGGER_CONSISTENCY_BIT, true); + mmio_region_write32(otp->base_addr, OTP_CTRL_CHECK_TRIGGER_REG_OFFSET, reg); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_lock(const dif_otp_ctrl_t *otp) { + if (otp == NULL) { + return kDifBadArg; + } + + uint32_t reg = bitfield_bit32_write( + 0, OTP_CTRL_DIRECT_ACCESS_REGWEN_DIRECT_ACCESS_REGWEN_BIT, false); + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_REGWEN_REG_OFFSET, + reg); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_is_locked(const dif_otp_ctrl_t *otp, + bool *is_locked) { + if (otp == NULL || is_locked == NULL) { + return kDifBadArg; + } + + uint32_t reg = mmio_region_read32(otp->base_addr, + OTP_CTRL_DIRECT_ACCESS_REGWEN_REG_OFFSET); + *is_locked = !bitfield_bit32_read( + reg, OTP_CTRL_DIRECT_ACCESS_REGWEN_DIRECT_ACCESS_REGWEN_BIT); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_lock_config(const dif_otp_ctrl_t *otp) { + if (otp == NULL) { + return kDifBadArg; + } + + uint32_t reg = + bitfield_bit32_write(0, OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT, false); + mmio_region_write32(otp->base_addr, OTP_CTRL_CHECK_REGWEN_REG_OFFSET, reg); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_config_is_locked(const dif_otp_ctrl_t *otp, + bool *is_locked) { + if (otp == NULL || is_locked == NULL) { + return kDifBadArg; + } + + *is_locked = checks_are_locked(otp, /*check_config=*/true); + return kDifOk; +} + +dif_result_t dif_otp_ctrl_lock_check_trigger(const dif_otp_ctrl_t *otp) { + if (otp == NULL) { + return kDifBadArg; + } + + uint32_t reg = bitfield_bit32_write( + 0, OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT, false); + mmio_region_write32(otp->base_addr, OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET, + reg); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_check_trigger_is_locked(const dif_otp_ctrl_t *otp, + bool *is_locked) { + if (otp == NULL || is_locked == NULL) { + return kDifBadArg; + } + + *is_locked = checks_are_locked(otp, /*check_config=*/false); + return kDifOk; +} + +static bool sw_read_lock_reg_offset(dif_otp_ctrl_partition_t partition, + ptrdiff_t *reg_offset, + bitfield_bit32_index_t *index) { + switch (partition) { +% for part in read_locked_csr_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_define = part_name.as_c_define() + index_line = f"*index = OTP_CTRL_{part_name_define}_READ_LOCK_{part_name_define}_READ_LOCK_BIT;" +%>\ + case kDifOtpCtrlPartition${part_name.as_camel_case()}: + *reg_offset = OTP_CTRL_${part_name_define}_READ_LOCK_REG_OFFSET; + % if len(index_line) > 80 - 6: + *index = + OTP_CTRL_${part_name_define}_READ_LOCK_${part_name_define}_READ_LOCK_BIT; + % else: + ${index_line} + % endif + break; +% endfor + default: + return false; + } + return true; +} + +dif_result_t dif_otp_ctrl_lock_reading(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition) { + if (otp == NULL) { + return kDifBadArg; + } + + ptrdiff_t offset; + bitfield_bit32_index_t index; + if (!sw_read_lock_reg_offset(partition, &offset, &index)) { + return kDifBadArg; + } + + uint32_t reg = bitfield_bit32_write(0, index, false); + mmio_region_write32(otp->base_addr, offset, reg); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_reading_is_locked(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + bool *is_locked) { + if (otp == NULL || is_locked == NULL) { + return kDifBadArg; + } + + ptrdiff_t offset; + bitfield_bit32_index_t index; + if (!sw_read_lock_reg_offset(partition, &offset, &index)) { + return kDifBadArg; + } + + uint32_t reg = mmio_region_read32(otp->base_addr, offset); + *is_locked = !bitfield_bit32_read(reg, index); + return kDifOk; +} + +dif_result_t dif_otp_ctrl_get_status(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_status_t *status) { + if (otp == NULL || status == NULL) { + return kDifBadArg; + } + + static const bitfield_bit32_index_t kIndices[] = { +% for part in parts: +<% + part_name = Name.from_snake_case(part["name"]) + lhs = f'[kDifOtpCtrlStatusCode{part_name.as_camel_case()}Error]' + rhs = f'OTP_CTRL_STATUS_{part_name.as_c_define()}_ERROR_BIT' + line = f'{lhs} = {rhs},' +%>\ + % if len(line) > 80 - 6: + ${lhs} = + ${rhs}, + % else: + ${line} + % endif +% endfor + [kDifOtpCtrlStatusCodeDaiError] = OTP_CTRL_STATUS_DAI_ERROR_BIT, + [kDifOtpCtrlStatusCodeLciError] = OTP_CTRL_STATUS_LCI_ERROR_BIT, + [kDifOtpCtrlStatusCodeTimeoutError] = OTP_CTRL_STATUS_TIMEOUT_ERROR_BIT, + [kDifOtpCtrlStatusCodeLfsrError] = OTP_CTRL_STATUS_LFSR_FSM_ERROR_BIT, + [kDifOtpCtrlStatusCodeScramblingError] = + OTP_CTRL_STATUS_SCRAMBLING_FSM_ERROR_BIT, + [kDifOtpCtrlStatusCodeKdfError] = OTP_CTRL_STATUS_KEY_DERIV_FSM_ERROR_BIT, + [kDifOtpCtrlStatusCodeBusIntegError] = + OTP_CTRL_STATUS_BUS_INTEG_ERROR_BIT, + [kDifOtpCtrlStatusCodeDaiIdle] = OTP_CTRL_STATUS_DAI_IDLE_BIT, + [kDifOtpCtrlStatusCodeCheckPending] = OTP_CTRL_STATUS_CHECK_PENDING_BIT, + }; + + status->codes = 0; + uint32_t status_code = + mmio_region_read32(otp->base_addr, OTP_CTRL_STATUS_REG_OFFSET); + for (int i = 0; i < ARRAYSIZE(kIndices); ++i) { + // If the error is not present at all, we clear its cause bit if relevant, + // and bail immediately. + if (!bitfield_bit32_read(status_code, kIndices[i])) { + if (i <= kDifOtpCtrlStatusCodeHasCauseLast) { + status->causes[i] = kDifOtpCtrlErrorOk; + } + continue; + } + + status->codes = + bitfield_bit32_write(status->codes, (bitfield_bit32_index_t)i, true); + + if (i <= kDifOtpCtrlStatusCodeHasCauseLast) { + bitfield_field32_t field; + field = (bitfield_field32_t){ + .mask = OTP_CTRL_ERR_CODE_0_ERR_CODE_0_MASK, + .index = OTP_CTRL_ERR_CODE_0_ERR_CODE_0_OFFSET, + }; + + ptrdiff_t address = + OTP_CTRL_ERR_CODE_0_REG_OFFSET + i * (ptrdiff_t)sizeof(uint32_t); + uint32_t error_code = mmio_region_read32(otp->base_addr, address); + + dif_otp_ctrl_error_t err; + switch (bitfield_field32_read(error_code, field)) { + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_NO_ERROR: + err = kDifOtpCtrlErrorOk; + break; + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_ERROR: + err = kDifOtpCtrlErrorMacroUnspecified; + break; + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_ECC_CORR_ERROR: + err = kDifOtpCtrlErrorMacroRecoverableRead; + break; + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_ECC_UNCORR_ERROR: + err = kDifOtpCtrlErrorMacroUnrecoverableRead; + break; + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_WRITE_BLANK_ERROR: + err = kDifOtpCtrlErrorMacroBlankCheckFailed; + break; + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_ACCESS_ERROR: + err = kDifOtpCtrlErrorLockedAccess; + break; + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_CHECK_FAIL_ERROR: + err = kDifOtpCtrlErrorBackgroundCheckFailed; + break; + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_FSM_STATE_ERROR: + err = kDifOtpCtrlErrorFsmBadState; + break; + default: + return kDifError; + } + status->causes[i] = err; + } + } + + return kDifOk; +} + +typedef struct partition_info { + /** + * The absolute OTP address at which this partition starts. + */ + uint32_t start_addr; + /** + * The length of this partition, in bytes, including the digest. + * + * If the partition has a digest, it is expected to be at address + * `start_addr + len - sizeof(uint64_t)`. + */ + uint32_t len; + /** + * The alignment mask for this partition. + * + * A valid address for this partition must be such that + * `addr & align_mask == 0`. + */ + uint32_t align_mask; + + /** + * Whether this is a software-managed partition with a software-managed + * digest. + */ + bool is_software; + + /** + * Whether this partition has a digest field. + */ + bool has_digest; + + /** + * Whether this partition is the lifecycle partition. + */ + bool is_lifecycle; +} partition_info_t; + +// This is generates too many lines with different formatting variants, so +// We opt to just disable formatting. +// clang-format off +static const partition_info_t kPartitions[] = { +% for part in parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() + part_name_define = part_name.as_c_define() + has_digest = part["hw_digest"] or part["sw_digest"] + is_lifecycle = part["variant"] == "LifeCycle" + is_software = part["variant"] == "Unbuffered" +%>\ + [kDifOtpCtrlPartition${part_name_camel}] = { + .start_addr = OTP_CTRL_PARAM_${part_name_define}_OFFSET, + .len = OTP_CTRL_PARAM_${part_name_define}_SIZE, + .align_mask = ${"0x7" if part in secret_parts else "0x3"}, + .is_software = ${"true" if is_software else "false"}, + .has_digest = ${"true" if has_digest else "false"}, + .is_lifecycle = ${"true" if is_lifecycle else "false"}}, +% endfor +}; +// clang-format on + +dif_result_t dif_otp_ctrl_relative_address(dif_otp_ctrl_partition_t partition, + uint32_t abs_address, + uint32_t *relative_address) { + *relative_address = 0; + + if (partition >= ARRAYSIZE(kPartitions)) { + return kDifBadArg; + } + + if ((abs_address & kPartitions[partition].align_mask) != 0) { + return kDifUnaligned; + } + + if (abs_address < kPartitions[partition].start_addr) { + return kDifOutOfRange; + } + + *relative_address = abs_address - kPartitions[partition].start_addr; + if (*relative_address >= kPartitions[partition].len) { + *relative_address = 0; + return kDifOutOfRange; + } + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_read_start(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address) { + if (otp == NULL || partition >= ARRAYSIZE(kPartitions)) { + return kDifBadArg; + } + + if ((address & kPartitions[partition].align_mask) != 0) { + return kDifUnaligned; + } + + if (address >= kPartitions[partition].len) { + return kDifOutOfRange; + } + + address += kPartitions[partition].start_addr; + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + address); + + uint32_t cmd = + bitfield_bit32_write(0, OTP_CTRL_DIRECT_ACCESS_CMD_RD_BIT, true); + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + cmd); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_read32_end(const dif_otp_ctrl_t *otp, + uint32_t *value) { + if (otp == NULL || value == NULL) { + return kDifBadArg; + } + + *value = mmio_region_read32(otp->base_addr, + OTP_CTRL_DIRECT_ACCESS_RDATA_0_REG_OFFSET); + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_read64_end(const dif_otp_ctrl_t *otp, + uint64_t *value) { + if (otp == NULL || value == NULL) { + return kDifBadArg; + } + + *value = mmio_region_read32(otp->base_addr, + OTP_CTRL_DIRECT_ACCESS_RDATA_1_REG_OFFSET); + *value <<= 32; + *value |= mmio_region_read32(otp->base_addr, + OTP_CTRL_DIRECT_ACCESS_RDATA_0_REG_OFFSET); + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_program32(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address, uint32_t value) { + if (otp == NULL || partition >= ARRAYSIZE(kPartitions)) { + return kDifBadArg; + } + + // Ensure that we are writing to a 32-bit-access partition by checking that + // the alignment mask is 0b11. + // + // Note furthermore that the LC partition is *not* writeable, so we eject + // here. + if (kPartitions[partition].align_mask != 0x3 || + kPartitions[partition].is_lifecycle) { + return kDifError; + } + + if ((address & kPartitions[partition].align_mask) != 0) { + return kDifUnaligned; + } + + // NOTE: The bounds check is tightened here, since we disallow writing the + // digest directly. If the partition does not have a digest, no tightening is + // needed. + size_t digest_size = kPartitions[partition].has_digest * sizeof(uint64_t); + if (address >= kPartitions[partition].len - digest_size) { + return kDifOutOfRange; + } + + address += kPartitions[partition].start_addr; + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + address); + + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET, + value); + + uint32_t cmd = + bitfield_bit32_write(0, OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT, true); + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + cmd); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_program64(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address, uint64_t value) { + if (otp == NULL || partition >= ARRAYSIZE(kPartitions)) { + return kDifBadArg; + } + + // Ensure that we are writing to a 64-bit-access partition by checking that + // the alignment mask is 0b111. + if (kPartitions[partition].align_mask != 0x7) { + return kDifError; + } + + if ((address & kPartitions[partition].align_mask) != 0) { + return kDifUnaligned; + } + + // NOTE: The bounds check is tightened here, since we disallow writing the + // digest directly. + size_t digest_size = sizeof(uint64_t); + if (address >= kPartitions[partition].len - digest_size) { + return kDifOutOfRange; + } + + address += kPartitions[partition].start_addr; + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + address); + + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET, + value & UINT32_MAX); + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_WDATA_1_REG_OFFSET, + value >> 32); + + uint32_t cmd = + bitfield_bit32_write(0, OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT, true); + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + cmd); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_digest(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint64_t digest) { + if (otp == NULL || partition >= ARRAYSIZE(kPartitions)) { + return kDifBadArg; + } + + // Not all partitions have a digest. + if (!kPartitions[partition].has_digest) { + return kDifError; + } + + // For software partitions, the digest must be nonzero; for all other + // partitions it must be zero. + bool is_sw = kPartitions[partition].is_software; + if (is_sw == (digest == 0)) { + return kDifBadArg; + } + + uint32_t address = kPartitions[partition].start_addr; + if (is_sw) { + address += kPartitions[partition].len - sizeof(digest); + } + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + address); + + if (digest != 0) { + mmio_region_write32(otp->base_addr, + OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET, + digest & 0xffffffff); + mmio_region_write32(otp->base_addr, + OTP_CTRL_DIRECT_ACCESS_WDATA_1_REG_OFFSET, + digest >> 32); + } + + bitfield_bit32_index_t cmd_bit = is_sw + ? OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT + : OTP_CTRL_DIRECT_ACCESS_CMD_DIGEST_BIT; + uint32_t cmd = bitfield_bit32_write(0, cmd_bit, true); + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + cmd); + + return kDifOk; +} + +static bool get_digest_regs(dif_otp_ctrl_partition_t partition, ptrdiff_t *reg0, + ptrdiff_t *reg1) { + switch (partition) { +% for part in digest_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_define = part_name.as_c_define() +%>\ + case kDifOtpCtrlPartition${part_name.as_camel_case()}: + *reg0 = OTP_CTRL_${part_name_define}_DIGEST_0_REG_OFFSET; + *reg1 = OTP_CTRL_${part_name_define}_DIGEST_1_REG_OFFSET; + break; +% endfor + default: + return false; + } + + return true; +} + +dif_result_t dif_otp_ctrl_is_digest_computed(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + bool *is_computed) { + if (otp == NULL || is_computed == NULL) { + return kDifBadArg; + } + + ptrdiff_t reg0, reg1; + if (!get_digest_regs(partition, ®0, ®1)) { + return kDifBadArg; + } + + uint64_t value = mmio_region_read32(otp->base_addr, reg1); + value <<= 32; + value |= mmio_region_read32(otp->base_addr, reg0); + + *is_computed = value != 0; + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_get_digest(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint64_t *digest) { + if (otp == NULL || digest == NULL) { + return kDifBadArg; + } + + ptrdiff_t reg0, reg1; + if (!get_digest_regs(partition, ®0, ®1)) { + return kDifBadArg; + } + + uint64_t value = mmio_region_read32(otp->base_addr, reg1); + value <<= 32; + value |= mmio_region_read32(otp->base_addr, reg0); + + if (value == 0) { + return kDifError; + } + *digest = value; + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_read_blocking(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address, uint32_t *buf, + size_t len) { + if (otp == NULL || partition >= ARRAYSIZE(kPartitions) || buf == NULL) { + return kDifBadArg; + } + + if (!kPartitions[partition].is_software) { + return kDifError; + } + + if ((address & kPartitions[partition].align_mask) != 0) { + return kDifUnaligned; + } + + if (address + len >= kPartitions[partition].len) { + return kDifOutOfRange; + } + + uint32_t reg_offset = OTP_CTRL_SW_CFG_WINDOW_REG_OFFSET + + kPartitions[partition].start_addr + address; + mmio_region_memcpy_from_mmio32(otp->base_addr, reg_offset, buf, + len * sizeof(uint32_t)); + return kDifOk; +} diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet b/hw/top_darjeeling/ip_autogen/otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet new file mode 100644 index 00000000000000..fbab47b27d488d --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet @@ -0,0 +1,614 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +<% +from topgen.lib import Name + +parts = otp_mmap["partitions"] +digest_parts = [part for part in parts if + part["hw_digest"] == "true" or part["sw_digest"] == "true"] +read_locked_csr_parts = [part for part in parts if part["read_lock"] == "CSR"] +secret_parts = [part for part in parts if part["secret"] == "true"] +%>\ +#ifndef OPENTITAN_SW_DEVICE_LIB_DIF_DIF_OTP_CTRL_H_ +#define OPENTITAN_SW_DEVICE_LIB_DIF_DIF_OTP_CTRL_H_ + +/** + * @file + * @brief + * OTP Controller Device Interface Functions + */ + +#include + +#include "sw/device/lib/base/macros.h" +#include "sw/device/lib/base/mmio.h" +#include "sw/device/lib/dif/dif_base.h" + +#include "sw/device/lib/dif/autogen/dif_otp_ctrl_autogen.h" + +// Header Extern Guard (so header can be used from C and C++) +#ifdef __cplusplus +extern "C" { +#endif // __cplusplus + +/** + * A partition within OTP memory. + */ +typedef enum dif_otp_ctrl_partition { +% for part in parts: +<% + part_name = Name.from_snake_case(part["name"]) + short_desc = part["desc"].split(".")[0].strip().replace("\n", " ") + long_desc_lines = part["desc"].split(".", 1)[1].strip().splitlines() + long_desc = "\n".join([" *" + (" " if line else "") + line for + line in long_desc_lines]) +%>\ + /** + * ${short_desc}. + * + % if long_desc: +${long_desc} + %endif + */ + kDifOtpCtrlPartition${part_name.as_camel_case()}, +% endfor +} dif_otp_ctrl_partition_t; + +/** + * Runtime configuration for OTP. + * + * This struct describes runtime information for one-time configuration of the + * hardware. + */ +typedef struct dif_otp_ctrl_config { + /** + * The timeout for an integrity or consistency check to succeed, in cycles. + * + * 100'000 is recommended as a minimum safe value. + */ + uint32_t check_timeout; + /** + * A mask for the pseudo-random integrity check period. + * + * The value of this mask limits the period of the integrity check; when the + * pseudo-random period is computed, this mask is applied to limit it. For + * example, a value of 0x3'ffff would correspond to a maximum period of about + * 2.8s at 24MHz. + * + * A value of zero disables the check. + */ + uint32_t integrity_period_mask; + /** + * A mask for the pseudo-random consistency check period. + * + * The value of this mask limits the period of the consistency check; when the + * pseudo-random period is computed, this mask is applied to limit it. For + * example, a value of 0x3ff'ffff would correspond to a maximum period of + * about 716s at 24MHz. + * + * A value of zero disables the check. + */ + uint32_t consistency_period_mask; +} dif_otp_ctrl_config_t; + +/** + * A hardware-level status code. + */ +typedef enum dif_otp_ctrl_status_code { + // NOTE: This enum's API *requires* that all "error"-like codes (that is, + // those which have associated cause registers) be a prefix of the enum + // values. + // + // Note furthermore that these enum variants are intended as bit indices, so + // their values should not be randomized. +% for part in parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() +%>\ + /** + * Indicates an error occurred in the `${part_name_camel}` partition. + */ + kDifOtpCtrlStatusCode${part_name_camel}Error${" = 0" if loop.first else ""}, +% endfor + /** + * Indicates an error occurred in the direct access interface. + */ + kDifOtpCtrlStatusCodeDaiError, + /** + * Indicates an error occurred in the lifecycle interface. + */ + kDifOtpCtrlStatusCodeLciError, + /** + * This is not a status code; rather, it represents the last error code which + * has a corresponding "cause" register. + * + * See `dif_otp_ctrl_status_t` for information on how to use this. + */ + kDifOtpCtrlStatusCodeHasCauseLast = kDifOtpCtrlStatusCodeLciError, + /** + * Indicates that an integrity or consistency check has timed out. + * + * This error is unrecoverable. + */ + kDifOtpCtrlStatusCodeTimeoutError, + /** + * Indicates that the LFSR that generates pseudo-random integrity and + * consistency checks is in a bad state. + * + * This error is unrecoverable. + */ + kDifOtpCtrlStatusCodeLfsrError, + /** + * Indicates that the scrambling hardware is in a bad state. + * + * This error is unrecoverable. + */ + kDifOtpCtrlStatusCodeScramblingError, + /** + * Indicates that the key derivation hardware is in a bad state. + * + * This error is unrecoverable. + */ + kDifOtpCtrlStatusCodeKdfError, + /** + * Indicates a bus integrity error. + * + * This error will raise an alert. + */ + kDifOtpCtrlStatusCodeBusIntegError, + /** + * Indicates that the direct access interface is idle. + */ + kDifOtpCtrlStatusCodeDaiIdle, + /** + * Indicates that an integrity or consistency check is currently pending. + */ + kDifOtpCtrlStatusCodeCheckPending, +} dif_otp_ctrl_status_code_t; + +/** + * A hardware-level error code, associated with a particular error defined in + * `dif_otp_ctrl_status_t`. + */ +typedef enum dif_otp_ctrl_error { + /** + * Indicates no error. + */ + kDifOtpCtrlErrorOk, + /** + * Indicates that an OTP macro command was invalid or did not + * complete successfully. + * + * This error indicates non-recoverable hardware malfunction. + */ + kDifOtpCtrlErrorMacroUnspecified, + /** + * Indicates a recoverable error during a read operation. + * + * A followup read should work as expected. + */ + kDifOtpCtrlErrorMacroRecoverableRead, + /** + * Indicates an unrecoverable error during a read operation. + * + * This error indicates non-recoverable hardware malfunction. + */ + kDifOtpCtrlErrorMacroUnrecoverableRead, + /** + * Indicates that the blank write check failed during a write operation. + */ + kDifOtpCtrlErrorMacroBlankCheckFailed, + /** + * Indicates a locked memory region was accessed. + */ + kDifOtpCtrlErrorLockedAccess, + /** + * Indicates a parity, integrity or consistency check failed in the buffer + * registers. + * + * This error indicates non-recoverable hardware malfunction. + */ + kDifOtpCtrlErrorBackgroundCheckFailed, + /** + * Indicates that the FSM of the controller is in a bad state or that the + * controller's FSM has been moved into its terminal state due to escalation + * via the alert subsystem. + * + * This error indicates that the device has been glitched by an attacker. + */ + kDifOtpCtrlErrorFsmBadState, +} dif_otp_ctrl_error_t; + +/** + * The overall status of the OTP controller. + * + * See `dif_otp_ctrl_get_status()`. + */ +typedef struct dif_otp_ctrl_status { + /** + * Currently active statuses, given as a bit vector. To check whether a + * particular status code was returned, write + * + * bool has_code = (status.codes >> kMyStatusCode) & 1; + * + * Note that it is possible to quickly check that the controller is idle and + * error-free by writing + * + * bool is_ok = status.codes == (1 << kDifOtpStatusCodeDaiIdle); + */ + uint32_t codes; + /** + * A list of root causes for each error status code. + * + * If the error status code `error` is present in `codes`, and + * `error <= kDifOtpCtrlStatusCodeHasCauseLast`, then `causes[error]` + * will contain its root cause. + */ + dif_otp_ctrl_error_t causes[kDifOtpCtrlStatusCodeHasCauseLast + 1]; +} dif_otp_ctrl_status_t; + +/** + * Configures OTP with runtime information. + * + * This function should need to be called at most once for the lifetime of + * `otp`. + * + * @param otp An OTP handle. + * @param config Runtime configuration parameters. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_configure(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_config_t config); + +/** + * Runs an integrity check on the OTP hardware. + * + * This function can be used to trigger an integrity check independent of the + * pseudo-random hardware-generated checks. + * + * @param otp An OTP handle. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_check_integrity(const dif_otp_ctrl_t *otp); + +/** + * Runs a consistency check on the OTP hardware. + * + * This function can be used to trigger a consistency check independent of the + * pseudo-random hardware-generated checks. + * + * @param otp An OTP handle. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_check_consistency(const dif_otp_ctrl_t *otp); + +/** + * Locks out access to the direct access interface registers. + * + * This function is idempotent: calling it while functionality is locked will + * have no effect and return `kDifOk`. + * + * @param otp An OTP handle. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_lock(const dif_otp_ctrl_t *otp); + +/** + * Checks whether access to the direct access interface is locked. + * + * Note that besides locking the DAI out until the next reset using the + * dif_otp_ctrl_dai_lock function, the DAI is also temporarily locked by the + * HW itself when it is busy processing a DAI command. In such a case, the + * kDifOtpCtrlStatusCodeDaiIdle status bit will be set to 0 as well. + * + * @param otp An OTP handle. + * @param[out] is_locked Out-param for the locked state. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_is_locked(const dif_otp_ctrl_t *otp, + bool *is_locked); + +/** + * Locks out `dif_otp_ctrl_configure()` function. + * + * This function is idempotent: calling it while functionality is locked will + * have no effect and return `kDifOk`. + * + * @param otp An OTP handle. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_lock_config(const dif_otp_ctrl_t *otp); + +/** + * Checks whether `dif_otp_ctrl_configure()` function is locked-out. + * + * @param otp An OTP handle. + * @param[out] is_locked Out-param for the locked state. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_config_is_locked(const dif_otp_ctrl_t *otp, + bool *is_locked); + +/** + * Locks out `dif_otp_ctrl_check_*()` functions. + * + * This function is idempotent: calling it while functionality is locked will + * have no effect and return `kDifOk`. + * + * @param otp An OTP handle. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_lock_check_trigger(const dif_otp_ctrl_t *otp); + +/** + * Checks whether the `dif_otp_ctrl_check_*()` functions are locked-out. + * + * @param otp An OTP handle. + * @param[out] is_locked Out-param for the locked state. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_check_trigger_is_locked(const dif_otp_ctrl_t *otp, + bool *is_locked); + +/** + * Locks out reads to a SW partition. + * + * This function should only be called on SW partitions; doing otherwise will + * return an error. + * + * Note that this is distinct from the write-locking performed by calling + * `dif_otp_ctrl_dai_digest()`. In particular, the effects of this function will + * not persist past a system reset. + * + * This function is idempotent: calling it while functionality is locked will + * have no effect and return `kDifOk`. + * + * @param otp An OTP handle. + * @param partition The SW partition to lock. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_lock_reading(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition); + +/** + * Checks whether reads to a SW partition are locked out. + * + * This function should only be called on SW partitions; doing otherwise will + * return an error. + * + * @param otp An OTP handle. + * @param partition the SW partition to check for locking. + * @param[out] is_locked Out-param for the locked state. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_reading_is_locked(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + bool *is_locked); + +/** + * Gets the current status of the OTP controller. + * + * @param otp An OTP handle. + * @param[out] status Out-param for the controller's status. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_get_status(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_status_t *status); + +/** + * Calculates a `relative_address` with respect to a `partition` start + * address. + * + * @param partition The partition to use to calculate the reference start + * address. + * @param abs_address Input address relative to the OTP memory start address. + * @param[out] relative_address The result relative address with respect to the + * `partition` start address. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_relative_address(dif_otp_ctrl_partition_t partition, + uint32_t abs_address, + uint32_t *relative_address); + +/** + * Schedules a read on the Direct Access Interface. + * + * Reads are performed relative to a partition; `address` should be given + * relative to the start of `partition`. An error is returned for out-of-bounds + * access. + * + * Furthermore, `address` must be well-aligned: it must be four-byte aligned for + * normal partitions and eight-byte-aligned for secret partitions. An error is + * returned for unaligned access. + * + * @param otp An OTP handle. + * @param partition The partition to read from. + * @param address A partition-relative address to read from. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_read_start(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address); + +/** + * Gets the result of a completed 32-bit read operation on the Direct Access + * Interface. + * + * Whether this function or its 64-bit variant should be called is dependent on + * the most recent partition read from. + * + * @param otp An OTP handle. + * @param[out] value Out-param for the read value. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_read32_end(const dif_otp_ctrl_t *otp, + uint32_t *value); + +/** + * Gets the result of a completed 64-bit read operation on the Direct Access + * Interface. + * + * Whether this function or its 32-bit variant should be called is dependent on + * the most recent partition read from. + * + * @param otp An OTP handle. + * @param[out] value Out-param for the read value. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_read64_end(const dif_otp_ctrl_t *otp, + uint64_t *value); + +/** + * Schedules a 32-bit write on the Direct Access Interface. + * + * Writes are performed relative to a partition; `address` should be given + * relative to the start of `partition`. An error is returned for out-of-bounds + * access. + * + * Furthermore, `address` must be four-byte-aligned, and `partition` must not be + * a secret partition. An error is returned if neither condition is met. + * + * Note that this function cannot be used to program the digest at the end of a + * `SW` partition; `dif_otp_ctrl_dai_digest()` must be used instead. + * + * @param otp An OTP handle. + * @param partition The partition to program. + * @param address A partition-relative address to program. + * @param value The value to program into the OTP. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_program32(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address, uint32_t value); + +/** + * Schedules a 64-bit write on the Direct Access Interface. + * + * Writes are performed relative to a partition; `address` should be given + * relative to the start of `partition`. An error is returned for out-of-bounds + * access. + * + * Furthermore, `address` must be eight-byte-aligned, and `partition` must be + * a secret partition. An error is returned if neither condition is met. + * + * @param otp An OTP handle. + * @param partition The partition to program. + * @param address A partition-relative address to program. + * @param value The value to program into the OTP. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_program64(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address, uint64_t value); + +/** + * Schedules a hardware digest operation on the Direct Access Interface. + * + * **This operation will also lock writes for the given partition.** + * + * If `partition` is a SW partition, `digest` must be non-zero; if it is a + * partition with a hardware-managed digest, `digest` *must* be zero (since the + * digest will be generated by the hardware). An error is returned if either + * precondition is not met. + * + * This function does not work with the lifecycle state partition, and will + * return an error in that case. + * + * @param otp An OTP handle. + * @param partition The partition to digest and lock. + * @param digest The digest to program (for SW partitions). + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_digest(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint64_t digest); + +/** + * Checks if the digest value for the given partition has been computed. Once a + * digest has been computed for a partition, the partition is write-locked + * (additionally, read-locked if the partition is secret). + * + * The lifecycle partition does not have a digest, and checking if this region + * has a computed digest will return an error. + * + * @param otp An OTP handle. + * @param partition The partition to check the digest of. + * @param[out] is_computed Indicates if the digest has been computed. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_is_digest_computed(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + bool *is_computed); + +/** + * Gets the buffered digest value for the given partition. + * + * Note that this value is only updated when the device is reset; if the digest + * has not been computed yet, or has been computed but not since device reset, + * this function will return an error. + * + * The lifecycle partition does not have a digest and will result in an error + * being returned. + * + * @param otp An OTP handle. + * @param partition The partition to get a digest for. + * @param[out] digest Out-param for the digest. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_get_digest(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint64_t *digest); + +/** + * Performs a memory-mapped read of the given partition, if it supports them. + * + * In particular, this function will read `len` words, starting at `address`, + * relative to the start of `partition`. + * + * The same caveats for `dif_otp_ctrl_dai_read_start()` apply to `address`; in + * addition, `address + len` must also be in-range and must not overflow. + * + * This function will block until the read completes, unlike Direct Access + * Interface functions. + * + * @param otp An OTP handle. + * @param partition The partition to read from. + * @param address A partition-relative address to read from. + * @param[out] buf A buffer of words to write read values to. + * @param len The number of words to read. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_read_blocking(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address, uint32_t *buf, + size_t len); + +#ifdef __cplusplus +} // extern "C" +#endif // __cplusplus + +#endif // OPENTITAN_SW_DEVICE_LIB_DIF_DIF_OTP_CTRL_H_ diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl.not_yet b/hw/top_darjeeling/ip_autogen/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl.not_yet new file mode 100644 index 00000000000000..2063fd5971ebc5 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl.not_yet @@ -0,0 +1,789 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +<% +from topgen.lib import Name + +parts = otp_mmap["partitions"] +digest_parts = [part for part in parts if + part["hw_digest"] or part["sw_digest"]] +hw_digest_parts = [part for part in parts if part["hw_digest"]] +sw_digest_parts = [part for part in parts if part["sw_digest"]] +read_locked_csr_parts = [part for part in parts if part["read_lock"] == "CSR"] +secret_parts = [part for part in parts if part["secret"]] +%>\ +#include "sw/device/lib/dif/dif_otp_ctrl.h" + +#include +#include +#include + +#include "gtest/gtest.h" +#include "sw/device/lib/base/mmio.h" +#include "sw/device/lib/base/mock_mmio.h" +#include "sw/device/lib/dif/dif_test_base.h" + +#include "otp_ctrl_regs.h" // Generated. + +namespace dif_otp_ctrl_unittest { +namespace { +using ::mock_mmio::LeInt; +using ::mock_mmio::MmioTest; +using ::mock_mmio::MockDevice; +using ::testing::Each; +using ::testing::ElementsAre; + +class OtpTest : public testing::Test, public MmioTest { + protected: + dif_otp_ctrl_t otp_ = {.base_addr = dev().region()}; +}; + +class DaiRegwenTest : public OtpTest {}; + +TEST_F(DaiRegwenTest, LockDai) { + EXPECT_WRITE32( + OTP_CTRL_DIRECT_ACCESS_REGWEN_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_REGWEN_DIRECT_ACCESS_REGWEN_BIT, false}}); + EXPECT_DIF_OK(dif_otp_ctrl_dai_lock(&otp_)); +} + +TEST_F(DaiRegwenTest, IsDaiLocked) { + bool flag; + + EXPECT_READ32( + OTP_CTRL_DIRECT_ACCESS_REGWEN_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_REGWEN_DIRECT_ACCESS_REGWEN_BIT, true}}); + EXPECT_DIF_OK(dif_otp_ctrl_dai_is_locked(&otp_, &flag)); + EXPECT_FALSE(flag); + + EXPECT_READ32( + OTP_CTRL_DIRECT_ACCESS_REGWEN_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_REGWEN_DIRECT_ACCESS_REGWEN_BIT, false}}); + EXPECT_DIF_OK(dif_otp_ctrl_dai_is_locked(&otp_, &flag)); + EXPECT_TRUE(flag); +} + +TEST_F(DaiRegwenTest, NullArgs) { + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_lock(nullptr)); + + bool flag; + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_is_locked(nullptr, &flag)); + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_is_locked(&otp_, nullptr)); +} + +class ConfigTest : public OtpTest {}; + +TEST_F(ConfigTest, Basic) { + dif_otp_ctrl_config_t config = { + .check_timeout = 100'000, + .integrity_period_mask = 0x3'ffff, + .consistency_period_mask = 0x3ff'ffff, + }; + + EXPECT_READ32(OTP_CTRL_CHECK_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT, true}}); + + EXPECT_WRITE32(OTP_CTRL_CHECK_TIMEOUT_REG_OFFSET, config.check_timeout); + EXPECT_WRITE32(OTP_CTRL_INTEGRITY_CHECK_PERIOD_REG_OFFSET, + config.integrity_period_mask); + EXPECT_WRITE32(OTP_CTRL_CONSISTENCY_CHECK_PERIOD_REG_OFFSET, + config.consistency_period_mask); + + EXPECT_DIF_OK(dif_otp_ctrl_configure(&otp_, config)); +} + +TEST_F(ConfigTest, Locked) { + EXPECT_READ32(OTP_CTRL_CHECK_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT, false}}); + + EXPECT_EQ(dif_otp_ctrl_configure(&otp_, {}), kDifLocked); +} + +TEST_F(ConfigTest, IsConfigLocked) { + bool flag; + + EXPECT_READ32(OTP_CTRL_CHECK_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT, true}}); + EXPECT_DIF_OK(dif_otp_ctrl_config_is_locked(&otp_, &flag)); + EXPECT_FALSE(flag); + + EXPECT_READ32(OTP_CTRL_CHECK_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT, false}}); + EXPECT_DIF_OK(dif_otp_ctrl_config_is_locked(&otp_, &flag)); + EXPECT_TRUE(flag); +} + +TEST_F(ConfigTest, LockConfig) { + EXPECT_WRITE32(OTP_CTRL_CHECK_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT, false}}); + EXPECT_DIF_OK(dif_otp_ctrl_lock_config(&otp_)); +} + +TEST_F(ConfigTest, NullArgs) { + EXPECT_DIF_BADARG(dif_otp_ctrl_configure(nullptr, {})); + + bool flag; + EXPECT_DIF_BADARG(dif_otp_ctrl_config_is_locked(nullptr, &flag)); + EXPECT_DIF_BADARG(dif_otp_ctrl_config_is_locked(&otp_, nullptr)); + + EXPECT_DIF_BADARG(dif_otp_ctrl_lock_config(nullptr)); +} + +class CheckTest : public OtpTest {}; + +TEST_F(CheckTest, Integrity) { + EXPECT_READ32( + OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT, true}}); + EXPECT_WRITE32(OTP_CTRL_CHECK_TRIGGER_REG_OFFSET, + {{OTP_CTRL_CHECK_TRIGGER_INTEGRITY_BIT, true}}); + + EXPECT_DIF_OK(dif_otp_ctrl_check_integrity(&otp_)); +} + +TEST_F(CheckTest, Consistency) { + EXPECT_READ32( + OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT, true}}); + EXPECT_WRITE32(OTP_CTRL_CHECK_TRIGGER_REG_OFFSET, + {{OTP_CTRL_CHECK_TRIGGER_CONSISTENCY_BIT, true}}); + + EXPECT_DIF_OK(dif_otp_ctrl_check_consistency(&otp_)); +} + +TEST_F(CheckTest, LockTrigger) { + EXPECT_WRITE32( + OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT, false}}); + EXPECT_DIF_OK(dif_otp_ctrl_lock_check_trigger(&otp_)); +} + +TEST_F(CheckTest, Locked) { + EXPECT_READ32( + OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT, false}}); + EXPECT_EQ(dif_otp_ctrl_check_integrity(&otp_), kDifLocked); + + EXPECT_READ32( + OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT, false}}); + EXPECT_EQ(dif_otp_ctrl_check_consistency(&otp_), kDifLocked); +} + +TEST_F(CheckTest, NullArgs) { + EXPECT_DIF_BADARG(dif_otp_ctrl_check_integrity(nullptr)); + EXPECT_DIF_BADARG(dif_otp_ctrl_check_consistency(nullptr)); +} + +class ReadLockTest : public OtpTest {}; + +// Too many formatting variants in template code, so disabling clang-format. +// clang-format off +TEST_F(ReadLockTest, IsLocked) { + bool flag; + +% for part in read_locked_csr_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() + part_name_define = part_name.as_c_define() +%>\ + EXPECT_READ32( + OTP_CTRL_${part_name_define}_READ_LOCK_REG_OFFSET, + {{OTP_CTRL_${part_name_define}_READ_LOCK_${part_name_define}_READ_LOCK_BIT, + true}}); + EXPECT_DIF_OK(dif_otp_ctrl_reading_is_locked( + &otp_, kDifOtpCtrlPartition${part_name_camel}, &flag)); + EXPECT_FALSE(flag); + + EXPECT_READ32( + OTP_CTRL_${part_name_define}_READ_LOCK_REG_OFFSET, + {{OTP_CTRL_${part_name_define}_READ_LOCK_${part_name_define}_READ_LOCK_BIT, + false}}); + EXPECT_DIF_OK(dif_otp_ctrl_reading_is_locked( + &otp_, kDifOtpCtrlPartition${part_name_camel}, &flag)); + EXPECT_TRUE(flag); + % if not loop.last: + + %endif +% endfor +} + +TEST_F(ReadLockTest, Lock) { +% for part in read_locked_csr_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() + part_name_define = part_name.as_c_define() +%>\ + EXPECT_WRITE32( + OTP_CTRL_${part_name_define}_READ_LOCK_REG_OFFSET, + {{OTP_CTRL_${part_name_define}_READ_LOCK_${part_name_define}_READ_LOCK_BIT, + false}}); + EXPECT_DIF_OK(dif_otp_ctrl_lock_reading( + &otp_, kDifOtpCtrlPartition${part_name_camel})); + % if not loop.last: + + %endif +% endfor +} + +TEST_F(ReadLockTest, NotLockablePartitions) { + bool flag; +% for part in [p for p in parts if p not in read_locked_csr_parts]: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() +%>\ + EXPECT_DIF_BADARG( + dif_otp_ctrl_lock_reading(&otp_, kDifOtpCtrlPartition${part_name_camel})); + EXPECT_DIF_BADARG(dif_otp_ctrl_reading_is_locked( + &otp_, kDifOtpCtrlPartition${part_name_camel}, &flag)); + % if not loop.last: + + %endif +% endfor +} +// clang-format on + +TEST_F(ReadLockTest, NullArgs) { + bool flag; +% for part in read_locked_csr_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() + lock_reading_line = f"dif_otp_ctrl_lock_reading(nullptr, kDifOtpCtrlPartition{part_name_camel}));" +%>\ + EXPECT_DIF_BADARG(dif_otp_ctrl_reading_is_locked( + nullptr, kDifOtpCtrlPartition${part_name_camel}, &flag)); + EXPECT_DIF_BADARG(dif_otp_ctrl_reading_is_locked( + &otp_, kDifOtpCtrlPartition${part_name_camel}, nullptr)); + % if len(lock_reading_line) > 80 - 6: + EXPECT_DIF_BADARG(dif_otp_ctrl_lock_reading( + nullptr, kDifOtpCtrlPartition${part_name_camel})); + % else: + EXPECT_DIF_BADARG( + ${lock_reading_line} + % endif + % if not loop.last: + + %endif +% endfor +} + +class StatusTest : public OtpTest {}; + +TEST_F(StatusTest, Idle) { + dif_otp_ctrl_status_t status; + + EXPECT_READ32(OTP_CTRL_STATUS_REG_OFFSET, + {{OTP_CTRL_STATUS_DAI_IDLE_BIT, true}}); + EXPECT_DIF_OK(dif_otp_ctrl_get_status(&otp_, &status)); + + EXPECT_EQ(status.codes, 1 << kDifOtpCtrlStatusCodeDaiIdle); + EXPECT_THAT(status.causes, Each(kDifOtpCtrlErrorOk)); +} + +TEST_F(StatusTest, Errors) { + dif_otp_ctrl_status_t status; + + EXPECT_READ32(OTP_CTRL_STATUS_REG_OFFSET, + { + {OTP_CTRL_STATUS_DAI_IDLE_BIT, true}, + {OTP_CTRL_STATUS_HW_CFG0_ERROR_BIT, true}, + {OTP_CTRL_STATUS_LCI_ERROR_BIT, true}, + }); + +<% + hw_cfg0_error_index = [i for i, p in enumerate(parts) + if p["name"] == "HW_CFG0"][0] + lci_error_index = len(parts) + 1 +%>\ + EXPECT_READ32(OTP_CTRL_ERR_CODE_${hw_cfg0_error_index}_REG_OFFSET, + {{OTP_CTRL_ERR_CODE_0_ERR_CODE_0_OFFSET, + OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_ECC_CORR_ERROR}}); + EXPECT_READ32(OTP_CTRL_ERR_CODE_${lci_error_index}_REG_OFFSET, + {{OTP_CTRL_ERR_CODE_0_ERR_CODE_0_OFFSET, + OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_ERROR}}); + + EXPECT_DIF_OK(dif_otp_ctrl_get_status(&otp_, &status)); + EXPECT_EQ(status.codes, (1 << kDifOtpCtrlStatusCodeDaiIdle) | + (1 << kDifOtpCtrlStatusCodeHwCfg0Error) | + (1 << kDifOtpCtrlStatusCodeLciError)); + EXPECT_EQ(status.causes[kDifOtpCtrlStatusCodeHwCfg0Error], + kDifOtpCtrlErrorMacroRecoverableRead); + EXPECT_EQ(status.causes[kDifOtpCtrlStatusCodeLciError], + kDifOtpCtrlErrorMacroUnspecified); +} + +TEST_F(StatusTest, NullArgs) { + dif_otp_ctrl_status_t status; + + EXPECT_DIF_BADARG(dif_otp_ctrl_get_status(nullptr, &status)); + EXPECT_DIF_BADARG(dif_otp_ctrl_get_status(&otp_, nullptr)); +} + +struct RelativeAddressParams { + std::string name; + dif_otp_ctrl_partition_t partition; + uint32_t abs_address; + dif_result_t expected_result; + uint32_t expected_relative_address; +}; + +class RelativeAddress + : public OtpTest, + public testing::WithParamInterface {}; + +TEST_P(RelativeAddress, RelativeAddress) { + uint32_t got_relative_address; + dif_result_t got_result = dif_otp_ctrl_relative_address( + GetParam().partition, GetParam().abs_address, &got_relative_address); + EXPECT_EQ(got_result, GetParam().expected_result); + EXPECT_EQ(got_relative_address, GetParam().expected_relative_address); +} + +INSTANTIATE_TEST_SUITE_P( + AllPartitions, RelativeAddress, + testing::Values( +% for part in parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() + part_name_define = part_name.as_c_define() + step = 8 if part["secret"] else 4 +%>\ + RelativeAddressParams{ + "${part_name_camel}Okay", + kDifOtpCtrlPartition${part_name_camel}, + OTP_CTRL_PARAM_${part_name_define}_OFFSET + ${step}, + kDifOk, + ${step}, + }, + RelativeAddressParams{ + "${part_name_camel}Unaligned", + kDifOtpCtrlPartition${part_name_camel}, + OTP_CTRL_PARAM_${part_name_define}_OFFSET + 1, + kDifUnaligned, + 0, + }, +<% + ## Exclude first partition to avoid a negative offset. +%>\ + % if not loop.first: + RelativeAddressParams{ + "${part_name_camel}OutOfRangeBeforeStart", + kDifOtpCtrlPartition${part_name_camel}, + OTP_CTRL_PARAM_${part_name_define}_OFFSET - ${step}, + kDifOutOfRange, + 0, + }, + % endif + RelativeAddressParams{ + "${part_name_camel}OutOfRangePastEnd", + kDifOtpCtrlPartition${part_name_camel}, + % if len(f"OTP_CTRL_PARAM_{part_name_define}_OFFSET + OTP_CTRL_PARAM_{part_name_define}_SIZE,") <= 80 - 12: + OTP_CTRL_PARAM_${part_name_define}_OFFSET + OTP_CTRL_PARAM_${part_name_define}_SIZE, + % else: + OTP_CTRL_PARAM_${part_name_define}_OFFSET + + OTP_CTRL_PARAM_${part_name_define}_SIZE, + % endif + kDifOutOfRange, + 0, + }${")," if loop.last else ","} +% endfor + [](const testing::TestParamInfo &info) { + return info.param.name; + }); + +class DaiReadTest : public OtpTest {}; + +TEST_F(DaiReadTest, Read32) { + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + OTP_CTRL_PARAM_MANUF_STATE_OFFSET); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_CMD_RD_BIT, true}}); + + EXPECT_DIF_OK(dif_otp_ctrl_dai_read_start(&otp_, kDifOtpCtrlPartitionHwCfg0, + /*address=*/0x20)); + + EXPECT_READ32(OTP_CTRL_DIRECT_ACCESS_RDATA_0_REG_OFFSET, 0x12345678); + + uint32_t val; + EXPECT_DIF_OK(dif_otp_ctrl_dai_read32_end(&otp_, &val)); + EXPECT_EQ(val, 0x12345678); +} + +TEST_F(DaiReadTest, Read64) { + uint64_t val; +% for part in secret_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() + part_name_define = part_name.as_c_define() +%>\ + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + OTP_CTRL_PARAM_${part_name_define}_OFFSET + 0x8); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_CMD_RD_BIT, true}}); + + EXPECT_DIF_OK(dif_otp_ctrl_dai_read_start(&otp_, kDifOtpCtrlPartition${part_name_camel}, + /*address=*/0x8)); + + EXPECT_READ32(OTP_CTRL_DIRECT_ACCESS_RDATA_1_REG_OFFSET, 0x12345678); + EXPECT_READ32(OTP_CTRL_DIRECT_ACCESS_RDATA_0_REG_OFFSET, 0x90abcdef); + + EXPECT_DIF_OK(dif_otp_ctrl_dai_read64_end(&otp_, &val)); + EXPECT_EQ(val, 0x1234567890abcdef); + % if not loop.last: + + % endif +% endfor +} + +TEST_F(DaiReadTest, Unaligned) { + EXPECT_EQ(dif_otp_ctrl_dai_read_start(&otp_, kDifOtpCtrlPartitionHwCfg0, + /*address=*/0b01), + kDifUnaligned); + EXPECT_EQ(dif_otp_ctrl_dai_read_start(&otp_, kDifOtpCtrlPartitionSecret2, + /*address=*/0b100), + kDifUnaligned); +} + +TEST_F(DaiReadTest, OutOfRange) { + EXPECT_EQ(dif_otp_ctrl_dai_read_start(&otp_, kDifOtpCtrlPartitionHwCfg0, + /*address=*/0x100), + kDifOutOfRange); +} + +TEST_F(DaiReadTest, NullArgs) { + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_read_start(nullptr, + kDifOtpCtrlPartitionHwCfg0, + /*address=*/0x0)); + + uint32_t val32; + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_read32_end(nullptr, &val32)); + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_read32_end(&otp_, nullptr)); + + uint64_t val64; + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_read64_end(nullptr, &val64)); + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_read64_end(&otp_, nullptr)); +} + +class DaiProgramTest : public OtpTest {}; + +TEST_F(DaiProgramTest, Program32) { + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + OTP_CTRL_PARAM_MANUF_STATE_OFFSET); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET, 0x12345678); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT, true}}); + + EXPECT_DIF_OK(dif_otp_ctrl_dai_program32(&otp_, kDifOtpCtrlPartitionHwCfg0, + /*address=*/0x20, + /*value=*/0x12345678)); +} + +TEST_F(DaiProgramTest, Program64) { + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + OTP_CTRL_PARAM_SECRET2_OFFSET + 0x8); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET, 0x90abcdef); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_WDATA_1_REG_OFFSET, 0x12345678); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT, true}}); + + EXPECT_DIF_OK(dif_otp_ctrl_dai_program64(&otp_, kDifOtpCtrlPartitionSecret2, + /*address=*/0x8, + /*value=*/0x1234567890abcdef)); +} + +TEST_F(DaiProgramTest, BadPartition) { + EXPECT_EQ(dif_otp_ctrl_dai_program32(&otp_, kDifOtpCtrlPartitionSecret1, + /*address=*/0x0, /*value=*/42), + kDifError); + EXPECT_EQ(dif_otp_ctrl_dai_program64(&otp_, kDifOtpCtrlPartitionHwCfg0, + /*address=*/0x0, /*value=*/42), + kDifError); + + // LC is never writeable. + EXPECT_EQ(dif_otp_ctrl_dai_program32(&otp_, kDifOtpCtrlPartitionLifeCycle, + /*address=*/0x0, /*value=*/42), + kDifError); +} + +TEST_F(DaiProgramTest, Unaligned) { + EXPECT_EQ(dif_otp_ctrl_dai_program32(&otp_, kDifOtpCtrlPartitionHwCfg0, + /*address=*/0b01, /*value=*/42), + kDifUnaligned); + EXPECT_EQ(dif_otp_ctrl_dai_program64(&otp_, kDifOtpCtrlPartitionSecret2, + /*address=*/0b100, /*value=*/42), + kDifUnaligned); +} + +TEST_F(DaiProgramTest, OutOfRange) { + // Check that we can't write a digest directly. + EXPECT_EQ(dif_otp_ctrl_dai_program32( + &otp_, kDifOtpCtrlPartitionCreatorSwCfg, + /*address=*/OTP_CTRL_PARAM_CREATOR_SW_CFG_DIGEST_OFFSET, + /*value=*/42), + kDifOutOfRange); + + // Same digest check for 64-bit. + EXPECT_EQ(dif_otp_ctrl_dai_program64( + &otp_, kDifOtpCtrlPartitionSecret2, + /*address=*/OTP_CTRL_PARAM_SECRET2_DIGEST_OFFSET, /*value=*/42), + kDifOutOfRange); +} + +TEST_F(DaiProgramTest, NullArgs) { + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_program32(nullptr, + kDifOtpCtrlPartitionHwCfg0, + /*address=*/0x0, /*value=*/42)); + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_program64(nullptr, + kDifOtpCtrlPartitionSecret0, + /*address=*/0x0, /*value=*/42)); +} + +class DaiDigestTest : public OtpTest {}; + +TEST_F(DaiDigestTest, DigestSw) { +% for part in sw_digest_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_define = part_name.as_c_define() + part_name_camel = part_name.as_camel_case() + dai_digest_line = ("EXPECT_DIF_OK(dif_otp_ctrl_dai_digest(&otp_, " + + f"kDifOtpCtrlPartition{part_name_camel},") +%>\ + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + OTP_CTRL_PARAM_${part_name.as_c_define()}_DIGEST_OFFSET); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET, 0x00abcdef); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_WDATA_1_REG_OFFSET, 0xabcdef00); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT, true}}); + + % if len(dai_digest_line) > 80 - 2: +<% dai_digest_line = (f"kDifOtpCtrlPartition{part_name_camel},") %>\ + % if len(dai_digest_line) > 80 - 40: + EXPECT_DIF_OK( + dif_otp_ctrl_dai_digest(&otp_, kDifOtpCtrlPartition${part_name_camel}, + /*digest=*/0xabcdef0000abcdef)); + % else: + EXPECT_DIF_OK(dif_otp_ctrl_dai_digest(&otp_, + kDifOtpCtrlPartition${part_name_camel}, + /*digest=*/0xabcdef0000abcdef)); + % endif + % else: + ${dai_digest_line} + /*digest=*/0xabcdef0000abcdef)); + % endif + % if not loop.last: + + % endif +% endfor +} + +TEST_F(DaiDigestTest, DigestHw) { +% for part in hw_digest_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_define = part_name.as_c_define() + part_name_camel = part_name.as_camel_case() + dai_digest_line = ("EXPECT_DIF_OK(dif_otp_ctrl_dai_digest(&otp_, " + + f"kDifOtpCtrlPartition{part_name_camel},") +%>\ + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + OTP_CTRL_PARAM_${part_name_define}_OFFSET); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_CMD_DIGEST_BIT, true}}); + + EXPECT_DIF_OK(dif_otp_ctrl_dai_digest(&otp_, kDifOtpCtrlPartition${part_name_camel}, + /*digest=*/0)); + % if not loop.last: + + % endif +% endfor +} + +TEST_F(DaiDigestTest, BadPartition) { + EXPECT_EQ(dif_otp_ctrl_dai_digest(&otp_, kDifOtpCtrlPartitionLifeCycle, + /*digest=*/0), + kDifError); +} + +TEST_F(DaiDigestTest, BadDigest) { + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_digest(&otp_, kDifOtpCtrlPartitionHwCfg0, + /*digest=*/0xabcdef0000abcdef)); + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_digest(&otp_, + kDifOtpCtrlPartitionCreatorSwCfg, + /*digest=*/0)); +} + +TEST_F(DaiDigestTest, NullArgs) { + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_digest(nullptr, + kDifOtpCtrlPartitionCreatorSwCfg, + /*digest=*/0xabcdef0000abcdef)); +} + +class IsDigestComputed : public OtpTest {}; + +TEST_F(IsDigestComputed, NullArgs) { + bool is_computed; + EXPECT_DIF_BADARG(dif_otp_ctrl_is_digest_computed( + nullptr, kDifOtpCtrlPartitionSecret2, &is_computed)); + EXPECT_DIF_BADARG(dif_otp_ctrl_is_digest_computed( + &otp_, kDifOtpCtrlPartitionSecret2, nullptr)); + EXPECT_DIF_BADARG(dif_otp_ctrl_is_digest_computed( + nullptr, kDifOtpCtrlPartitionSecret2, nullptr)); +} + +TEST_F(IsDigestComputed, BadPartition) { + bool is_computed; + EXPECT_DIF_BADARG(dif_otp_ctrl_is_digest_computed( + &otp_, kDifOtpCtrlPartitionLifeCycle, &is_computed)); +} + +TEST_F(IsDigestComputed, Success) { + bool is_computed; + + EXPECT_READ32(OTP_CTRL_SECRET2_DIGEST_1_REG_OFFSET, 0x98abcdef); + EXPECT_READ32(OTP_CTRL_SECRET2_DIGEST_0_REG_OFFSET, 0xabcdef01); + EXPECT_DIF_OK(dif_otp_ctrl_is_digest_computed( + &otp_, kDifOtpCtrlPartitionSecret2, &is_computed)); + EXPECT_TRUE(is_computed); + + EXPECT_READ32(OTP_CTRL_SECRET2_DIGEST_1_REG_OFFSET, 0); + EXPECT_READ32(OTP_CTRL_SECRET2_DIGEST_0_REG_OFFSET, 0); + EXPECT_DIF_OK(dif_otp_ctrl_is_digest_computed( + &otp_, kDifOtpCtrlPartitionSecret2, &is_computed)); + EXPECT_FALSE(is_computed); +} + +struct DigestParams { + dif_otp_ctrl_partition_t partition; + bool has_digest; + ptrdiff_t reg0, reg1; +}; + +class GetDigest : public OtpTest, + public testing::WithParamInterface {}; + +TEST_P(GetDigest, GetDigest) { + if (!GetParam().has_digest) { + uint64_t digest; + EXPECT_DIF_BADARG( + dif_otp_ctrl_get_digest(&otp_, GetParam().partition, &digest)); + return; + } + + EXPECT_READ32(GetParam().reg1, 0xabcdef99); + EXPECT_READ32(GetParam().reg0, 0x99abcdef); + + uint64_t digest; + EXPECT_DIF_OK(dif_otp_ctrl_get_digest(&otp_, GetParam().partition, &digest)); + EXPECT_EQ(digest, 0xabcdef9999abcdef); +} + +TEST_P(GetDigest, BadDigest) { + if (!GetParam().has_digest) { + return; + } + + EXPECT_READ32(GetParam().reg1, 0x0); + EXPECT_READ32(GetParam().reg0, 0x0); + + uint64_t digest; + EXPECT_EQ(dif_otp_ctrl_get_digest(&otp_, GetParam().partition, &digest), + kDifError); +} + +TEST_P(GetDigest, NullArgs) { + uint64_t digest; + EXPECT_DIF_BADARG( + dif_otp_ctrl_get_digest(nullptr, GetParam().partition, &digest)); + EXPECT_DIF_BADARG( + dif_otp_ctrl_get_digest(&otp_, GetParam().partition, nullptr)); +} + +// This depends on the maximum length of partition names, which will +// be changing, so turn formatting off. +// clang-format off +INSTANTIATE_TEST_SUITE_P( + AllDigests, GetDigest, + testing::Values( +% for part in parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() + part_name_define = part_name.as_c_define() +%>\ + % if part in digest_parts: + DigestParams{ + kDifOtpCtrlPartition${part_name_camel}, + true, + OTP_CTRL_${part_name_define}_DIGEST_0_REG_OFFSET, + OTP_CTRL_${part_name_define}_DIGEST_1_REG_OFFSET, + }${"" if loop.last else ","} + % else: + DigestParams{ + kDifOtpCtrlPartition${part_name_camel}, + false, + 0, + 0, + }${"));" if loop.last else ","} + % endif +% endfor +// clang-format on + +class BlockingIoTest : public OtpTest { + protected: + static constexpr size_t kWords = 4; +}; + +TEST_F(BlockingIoTest, Read) { + for (size_t i = 0; i < kWords; ++i) { + auto offset = + OTP_CTRL_PARAM_OWNER_SW_CFG_OFFSET + 0x10 + i * sizeof(uint32_t); + EXPECT_READ32(OTP_CTRL_SW_CFG_WINDOW_REG_OFFSET + offset, i + 1); + } + + std::vector buf(kWords); + EXPECT_DIF_OK(dif_otp_ctrl_read_blocking( + &otp_, kDifOtpCtrlPartitionOwnerSwCfg, 0x10, buf.data(), buf.size())); + EXPECT_THAT(buf, ElementsAre(1, 2, 3, 4)); +} + +TEST_F(BlockingIoTest, BadPartition) { + std::vector buf(kWords); + EXPECT_EQ(dif_otp_ctrl_read_blocking(&otp_, kDifOtpCtrlPartitionHwCfg0, 0x10, + buf.data(), buf.size()), + kDifError); +} + +TEST_F(BlockingIoTest, Unaligned) { + std::vector buf(kWords); + EXPECT_EQ(dif_otp_ctrl_read_blocking(&otp_, kDifOtpCtrlPartitionOwnerSwCfg, + 0x11, buf.data(), buf.size()), + kDifUnaligned); +} + +TEST_F(BlockingIoTest, OutOfRange) { + std::vector buf(0x2f0); + EXPECT_EQ(dif_otp_ctrl_read_blocking(&otp_, kDifOtpCtrlPartitionOwnerSwCfg, + 0x300, buf.data(), buf.size()), + kDifOutOfRange); + EXPECT_EQ(dif_otp_ctrl_read_blocking(&otp_, kDifOtpCtrlPartitionOwnerSwCfg, + 0x10, buf.data(), 0x330), + kDifOutOfRange); +} + +TEST_F(BlockingIoTest, NullArgs) { + std::vector buf(kWords); + EXPECT_DIF_BADARG(dif_otp_ctrl_read_blocking( + nullptr, kDifOtpCtrlPartitionOwnerSwCfg, 0x10, buf.data(), buf.size())); + EXPECT_DIF_BADARG(dif_otp_ctrl_read_blocking( + &otp_, kDifOtpCtrlPartitionOwnerSwCfg, 0x10, nullptr, buf.size())); +} + +} // namespace +} // namespace dif_otp_ctrl_unittest diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/data/otp_ctrl.hjson b/hw/top_darjeeling/ip_autogen/otp_ctrl/data/otp_ctrl.hjson new file mode 100644 index 00000000000000..fc5b51d69a5a1d --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/data/otp_ctrl.hjson @@ -0,0 +1,3927 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// HJSON with partition metadata. +// +{ + name: "otp_ctrl", + human_name: "One-Time Programmable Memory Controller", + one_line_desc: "Interfaces integrated one-time programmable memory, supports scrambling, integrity and secure wipe", + one_paragraph_desc: ''' + One-Time Programmable (OTP) Memory Controller provides an open source abstraction interface for software and other hardware components such as Life Cycle Controller and Key Manager to interact with an integrated, closed source, proprietary OTP memory. + On top of defensive features provided by the proprietary OTP memory to deter side-channel analysis (SCA), fault injection (FI) attacks, and visual and electrical probing, the open source OTP controller features high-level logical security protection such as integrity checks and scrambling, as well as software isolation for when OTP contents are readable and programmable. + It features multiple individually-lockable logical partitions, periodic / persistent checking of OTP values, and a separate partition and interface for Life Cycle Controller. + ''' + // Unique comportable IP identifier defined under KNOWN_CIP_IDS in the regtool. + cip_id: "16", + design_spec: "../doc", + dv_doc: "../doc/dv", + hw_checklist: "../doc/checklist", + sw_checklist: "/sw/device/lib/dif/dif_otp_ctrl", + revisions: [ + { + version: "0.1.0", + life_stage: "L1", + design_stage: "D2", + verification_stage: "V2", + dif_stage: "S1", + commit_id: "127b109e2fab9336e830158abe449a3922544ded", + notes: "", + } + { + version: "1.0.0", + life_stage: "L1", + design_stage: "D3", + verification_stage: "V2S", + dif_stage: "S2", + notes: "", + } + { + version: "2.0.0", + life_stage: "L1", + design_stage: "D3", + verification_stage: "V2S", + dif_stage: "S2", + notes: "", + } + ] + clocking: [ + {clock: "clk_i", reset: "rst_ni", primary: true}, + {clock: "clk_edn_i", reset: "rst_edn_ni"} + ] + scan: "true", // Enable `scanmode_i` port + scan_reset: "true", // Enable `scan_rst_ni` port + scan_en: "true", // Enable `scan_en_i` port + bus_interfaces: [ + { protocol: "tlul", direction: "device", name: "core" } + { protocol: "tlul", direction: "device", name: "prim", hier_path: "u_otp.gen_generic.u_impl_generic.u_reg_top" } + ], + + available_output_list: [ + { name: "test", + width: 8, + desc: "Test-related GPIOs. Only active in DFT-enabled life cycle states." + } + ], + + /////////////////////////// + // Interrupts and Alerts // + /////////////////////////// + + interrupt_list: [ + { name: "otp_operation_done", + desc: "A direct access command or digest calculation operation has completed." + } + { name: "otp_error", + desc: "An error has occurred in the OTP controller. Check the !!ERR_CODE register to get more information." + } + ], + + alert_list: [ + { name: "fatal_macro_error", + desc: "This alert triggers if hardware detects an uncorrectable error during an OTP transaction, for example an uncorrectable ECC error in the OTP array.", + } + { name: "fatal_check_error", + desc: "This alert triggers if any of the background checks fails. This includes the digest checks and concurrent ECC checks in the buffer registers.", + } + { name: "fatal_bus_integ_error", + desc: "This fatal alert is triggered when a fatal TL-UL bus integrity fault is detected." + } + { name: "fatal_prim_otp_alert", + desc: "Fatal alert triggered inside the OTP primitive, including fatal TL-UL bus integrity faults of the test interface." + } + { name: "recov_prim_otp_alert", + desc: "Recoverable alert triggered inside the OTP primitive." + } + ], + + //////////////// + // Parameters // + //////////////// + param_list: [ + // Init file + { name: "MemInitFile", + desc: "VMEM file to initialize the OTP macro.", + type: "", + default: '""', + expose: "true", + local: "false" + } + // Random netlist constants + { name: "RndCnstLfsrSeed", + desc: "Compile-time random bits for initial LFSR seed", + type: "otp_ctrl_top_specific_pkg::lfsr_seed_t" + randcount: "40", + randtype: "data", // randomize randcount databits + } + { name: "RndCnstLfsrPerm", + desc: "Compile-time random permutation for LFSR output", + type: "otp_ctrl_top_specific_pkg::lfsr_perm_t" + randcount: "40", + randtype: "perm", // random permutation for randcount elements + } + { name: "RndCnstScrmblKeyInit", + desc: "Compile-time random permutation for scrambling key/nonce register reset value", + type: "otp_ctrl_top_specific_pkg::scrmbl_key_init_t" + randcount: "256", + randtype: "data", // random permutation for randcount elements + } + // Normal parameters + { name: "NumSramKeyReqSlots", + desc: "Number of key slots", + type: "int", + default: "4", + local: "true" + }, + { name: "OtpByteAddrWidth", + desc: "Width of the OTP byte address.", + type: "int", + default: "14", + local: "true" + }, + { name: "NumErrorEntries", + desc: "Number of error register entries.", + type: "int", + default: "24", // partitions + DAI/LCI + local: "true" + }, + { name: "NumDaiWords", + desc: "Number of 32bit words in the DAI.", + type: "int", + default: "2", + local: "true" + }, + { name: "NumDigestWords", + desc: "Size of the digest fields in 32bit words.", + type: "int", + default: "2", + local: "true" + }, + { name: "NumSwCfgWindowWords", + desc: "Size of the TL-UL window in 32bit words. Note that the effective partition size is smaller than that.", + type: "int", + default: "4096", + local: "true" + } + + // Memory map Info + { name: "NumPart", + desc: "Number of partitions", + type: "int", + default: "22", + local: "true" + }, + { name: "NumPartUnbuf", + desc: "Number of unbuffered partitions", + type: "int", + default: "15", + local: "true" + }, + { name: "NumPartBuf", + desc: "Number of buffered partitions (including 1 lifecycle partition)", + type: "int", + default: "7", + local: "true" + }, + { name: "VendorTestOffset", + desc: "Offset of the VENDOR_TEST partition", + type: "int", + default: "0", + local: "true" + }, + { name: "VendorTestSize", + desc: "Size of the VENDOR_TEST partition", + type: "int", + default: "64", + local: "true" + }, + { name: "ScratchOffset", + desc: "Offset of SCRATCH", + type: "int", + default: "0", + local: "true" + }, + { name: "ScratchSize", + desc: "Size of SCRATCH", + type: "int", + default: "56", + local: "true" + }, + { name: "VendorTestDigestOffset", + desc: "Offset of VENDOR_TEST_DIGEST", + type: "int", + default: "56", + local: "true" + }, + { name: "VendorTestDigestSize", + desc: "Size of VENDOR_TEST_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "CreatorSwCfgOffset", + desc: "Offset of the CREATOR_SW_CFG partition", + type: "int", + default: "64", + local: "true" + }, + { name: "CreatorSwCfgSize", + desc: "Size of the CREATOR_SW_CFG partition", + type: "int", + default: "320", + local: "true" + }, + { name: "CreatorSwCfgAstCfgOffset", + desc: "Offset of CREATOR_SW_CFG_AST_CFG", + type: "int", + default: "64", + local: "true" + }, + { name: "CreatorSwCfgAstCfgSize", + desc: "Size of CREATOR_SW_CFG_AST_CFG", + type: "int", + default: "124", + local: "true" + }, + { name: "CreatorSwCfgAstInitEnOffset", + desc: "Offset of CREATOR_SW_CFG_AST_INIT_EN", + type: "int", + default: "188", + local: "true" + }, + { name: "CreatorSwCfgAstInitEnSize", + desc: "Size of CREATOR_SW_CFG_AST_INIT_EN", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgOverridesOffset", + desc: "Offset of CREATOR_SW_CFG_OVERRIDES", + type: "int", + default: "192", + local: "true" + }, + { name: "CreatorSwCfgOverridesSize", + desc: "Size of CREATOR_SW_CFG_OVERRIDES", + type: "int", + default: "32", + local: "true" + }, + { name: "CreatorSwCfgRomExtSkuOffset", + desc: "Offset of CREATOR_SW_CFG_ROM_EXT_SKU", + type: "int", + default: "224", + local: "true" + }, + { name: "CreatorSwCfgRomExtSkuSize", + desc: "Size of CREATOR_SW_CFG_ROM_EXT_SKU", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgSigverifyRsaModExpIbexEnOffset", + desc: "Offset of CREATOR_SW_CFG_SIGVERIFY_RSA_MOD_EXP_IBEX_EN", + type: "int", + default: "228", + local: "true" + }, + { name: "CreatorSwCfgSigverifyRsaModExpIbexEnSize", + desc: "Size of CREATOR_SW_CFG_SIGVERIFY_RSA_MOD_EXP_IBEX_EN", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgSigverifyRsaKeyEnOffset", + desc: "Offset of CREATOR_SW_CFG_SIGVERIFY_RSA_KEY_EN", + type: "int", + default: "232", + local: "true" + }, + { name: "CreatorSwCfgSigverifyRsaKeyEnSize", + desc: "Size of CREATOR_SW_CFG_SIGVERIFY_RSA_KEY_EN", + type: "int", + default: "8", + local: "true" + }, + { name: "CreatorSwCfgSigverifySpxEnOffset", + desc: "Offset of CREATOR_SW_CFG_SIGVERIFY_SPX_EN", + type: "int", + default: "240", + local: "true" + }, + { name: "CreatorSwCfgSigverifySpxEnSize", + desc: "Size of CREATOR_SW_CFG_SIGVERIFY_SPX_EN", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgSigverifySpxKeyEnOffset", + desc: "Offset of CREATOR_SW_CFG_SIGVERIFY_SPX_KEY_EN", + type: "int", + default: "244", + local: "true" + }, + { name: "CreatorSwCfgSigverifySpxKeyEnSize", + desc: "Size of CREATOR_SW_CFG_SIGVERIFY_SPX_KEY_EN", + type: "int", + default: "8", + local: "true" + }, + { name: "CreatorSwCfgFlashDataDefaultCfgOffset", + desc: "Offset of CREATOR_SW_CFG_FLASH_DATA_DEFAULT_CFG", + type: "int", + default: "252", + local: "true" + }, + { name: "CreatorSwCfgFlashDataDefaultCfgSize", + desc: "Size of CREATOR_SW_CFG_FLASH_DATA_DEFAULT_CFG", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgFlashInfoBootDataCfgOffset", + desc: "Offset of CREATOR_SW_CFG_FLASH_INFO_BOOT_DATA_CFG", + type: "int", + default: "256", + local: "true" + }, + { name: "CreatorSwCfgFlashInfoBootDataCfgSize", + desc: "Size of CREATOR_SW_CFG_FLASH_INFO_BOOT_DATA_CFG", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgFlashHwInfoCfgOverrideOffset", + desc: "Offset of CREATOR_SW_CFG_FLASH_HW_INFO_CFG_OVERRIDE", + type: "int", + default: "260", + local: "true" + }, + { name: "CreatorSwCfgFlashHwInfoCfgOverrideSize", + desc: "Size of CREATOR_SW_CFG_FLASH_HW_INFO_CFG_OVERRIDE", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRngEnOffset", + desc: "Offset of CREATOR_SW_CFG_RNG_EN", + type: "int", + default: "264", + local: "true" + }, + { name: "CreatorSwCfgRngEnSize", + desc: "Size of CREATOR_SW_CFG_RNG_EN", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgJitterEnOffset", + desc: "Offset of CREATOR_SW_CFG_JITTER_EN", + type: "int", + default: "268", + local: "true" + }, + { name: "CreatorSwCfgJitterEnSize", + desc: "Size of CREATOR_SW_CFG_JITTER_EN", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRetRamResetMaskOffset", + desc: "Offset of CREATOR_SW_CFG_RET_RAM_RESET_MASK", + type: "int", + default: "272", + local: "true" + }, + { name: "CreatorSwCfgRetRamResetMaskSize", + desc: "Size of CREATOR_SW_CFG_RET_RAM_RESET_MASK", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgManufStateOffset", + desc: "Offset of CREATOR_SW_CFG_MANUF_STATE", + type: "int", + default: "276", + local: "true" + }, + { name: "CreatorSwCfgManufStateSize", + desc: "Size of CREATOR_SW_CFG_MANUF_STATE", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRomExecEnOffset", + desc: "Offset of CREATOR_SW_CFG_ROM_EXEC_EN", + type: "int", + default: "280", + local: "true" + }, + { name: "CreatorSwCfgRomExecEnSize", + desc: "Size of CREATOR_SW_CFG_ROM_EXEC_EN", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgCpuctrlOffset", + desc: "Offset of CREATOR_SW_CFG_CPUCTRL", + type: "int", + default: "284", + local: "true" + }, + { name: "CreatorSwCfgCpuctrlSize", + desc: "Size of CREATOR_SW_CFG_CPUCTRL", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgMinSecVerRomExtOffset", + desc: "Offset of CREATOR_SW_CFG_MIN_SEC_VER_ROM_EXT", + type: "int", + default: "288", + local: "true" + }, + { name: "CreatorSwCfgMinSecVerRomExtSize", + desc: "Size of CREATOR_SW_CFG_MIN_SEC_VER_ROM_EXT", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgMinSecVerBl0Offset", + desc: "Offset of CREATOR_SW_CFG_MIN_SEC_VER_BL0", + type: "int", + default: "292", + local: "true" + }, + { name: "CreatorSwCfgMinSecVerBl0Size", + desc: "Size of CREATOR_SW_CFG_MIN_SEC_VER_BL0", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgDefaultBootDataInProdEnOffset", + desc: "Offset of CREATOR_SW_CFG_DEFAULT_BOOT_DATA_IN_PROD_EN", + type: "int", + default: "296", + local: "true" + }, + { name: "CreatorSwCfgDefaultBootDataInProdEnSize", + desc: "Size of CREATOR_SW_CFG_DEFAULT_BOOT_DATA_IN_PROD_EN", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRmaSpinEnOffset", + desc: "Offset of CREATOR_SW_CFG_RMA_SPIN_EN", + type: "int", + default: "300", + local: "true" + }, + { name: "CreatorSwCfgRmaSpinEnSize", + desc: "Size of CREATOR_SW_CFG_RMA_SPIN_EN", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRmaSpinCyclesOffset", + desc: "Offset of CREATOR_SW_CFG_RMA_SPIN_CYCLES", + type: "int", + default: "304", + local: "true" + }, + { name: "CreatorSwCfgRmaSpinCyclesSize", + desc: "Size of CREATOR_SW_CFG_RMA_SPIN_CYCLES", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRngRepcntThresholdsOffset", + desc: "Offset of CREATOR_SW_CFG_RNG_REPCNT_THRESHOLDS", + type: "int", + default: "308", + local: "true" + }, + { name: "CreatorSwCfgRngRepcntThresholdsSize", + desc: "Size of CREATOR_SW_CFG_RNG_REPCNT_THRESHOLDS", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRngRepcntsThresholdsOffset", + desc: "Offset of CREATOR_SW_CFG_RNG_REPCNTS_THRESHOLDS", + type: "int", + default: "312", + local: "true" + }, + { name: "CreatorSwCfgRngRepcntsThresholdsSize", + desc: "Size of CREATOR_SW_CFG_RNG_REPCNTS_THRESHOLDS", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRngAdaptpHiThresholdsOffset", + desc: "Offset of CREATOR_SW_CFG_RNG_ADAPTP_HI_THRESHOLDS", + type: "int", + default: "316", + local: "true" + }, + { name: "CreatorSwCfgRngAdaptpHiThresholdsSize", + desc: "Size of CREATOR_SW_CFG_RNG_ADAPTP_HI_THRESHOLDS", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRngAdaptpLoThresholdsOffset", + desc: "Offset of CREATOR_SW_CFG_RNG_ADAPTP_LO_THRESHOLDS", + type: "int", + default: "320", + local: "true" + }, + { name: "CreatorSwCfgRngAdaptpLoThresholdsSize", + desc: "Size of CREATOR_SW_CFG_RNG_ADAPTP_LO_THRESHOLDS", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRngBucketThresholdsOffset", + desc: "Offset of CREATOR_SW_CFG_RNG_BUCKET_THRESHOLDS", + type: "int", + default: "324", + local: "true" + }, + { name: "CreatorSwCfgRngBucketThresholdsSize", + desc: "Size of CREATOR_SW_CFG_RNG_BUCKET_THRESHOLDS", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRngMarkovHiThresholdsOffset", + desc: "Offset of CREATOR_SW_CFG_RNG_MARKOV_HI_THRESHOLDS", + type: "int", + default: "328", + local: "true" + }, + { name: "CreatorSwCfgRngMarkovHiThresholdsSize", + desc: "Size of CREATOR_SW_CFG_RNG_MARKOV_HI_THRESHOLDS", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRngMarkovLoThresholdsOffset", + desc: "Offset of CREATOR_SW_CFG_RNG_MARKOV_LO_THRESHOLDS", + type: "int", + default: "332", + local: "true" + }, + { name: "CreatorSwCfgRngMarkovLoThresholdsSize", + desc: "Size of CREATOR_SW_CFG_RNG_MARKOV_LO_THRESHOLDS", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRngExthtHiThresholdsOffset", + desc: "Offset of CREATOR_SW_CFG_RNG_EXTHT_HI_THRESHOLDS", + type: "int", + default: "336", + local: "true" + }, + { name: "CreatorSwCfgRngExthtHiThresholdsSize", + desc: "Size of CREATOR_SW_CFG_RNG_EXTHT_HI_THRESHOLDS", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRngExthtLoThresholdsOffset", + desc: "Offset of CREATOR_SW_CFG_RNG_EXTHT_LO_THRESHOLDS", + type: "int", + default: "340", + local: "true" + }, + { name: "CreatorSwCfgRngExthtLoThresholdsSize", + desc: "Size of CREATOR_SW_CFG_RNG_EXTHT_LO_THRESHOLDS", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRngAlertThresholdOffset", + desc: "Offset of CREATOR_SW_CFG_RNG_ALERT_THRESHOLD", + type: "int", + default: "344", + local: "true" + }, + { name: "CreatorSwCfgRngAlertThresholdSize", + desc: "Size of CREATOR_SW_CFG_RNG_ALERT_THRESHOLD", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgRngHealthConfigDigestOffset", + desc: "Offset of CREATOR_SW_CFG_RNG_HEALTH_CONFIG_DIGEST", + type: "int", + default: "348", + local: "true" + }, + { name: "CreatorSwCfgRngHealthConfigDigestSize", + desc: "Size of CREATOR_SW_CFG_RNG_HEALTH_CONFIG_DIGEST", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgSramKeyRenewEnOffset", + desc: "Offset of CREATOR_SW_CFG_SRAM_KEY_RENEW_EN", + type: "int", + default: "352", + local: "true" + }, + { name: "CreatorSwCfgSramKeyRenewEnSize", + desc: "Size of CREATOR_SW_CFG_SRAM_KEY_RENEW_EN", + type: "int", + default: "4", + local: "true" + }, + { name: "CreatorSwCfgDigestOffset", + desc: "Offset of CREATOR_SW_CFG_DIGEST", + type: "int", + default: "376", + local: "true" + }, + { name: "CreatorSwCfgDigestSize", + desc: "Size of CREATOR_SW_CFG_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "OwnerSwCfgOffset", + desc: "Offset of the OWNER_SW_CFG partition", + type: "int", + default: "384", + local: "true" + }, + { name: "OwnerSwCfgSize", + desc: "Size of the OWNER_SW_CFG partition", + type: "int", + default: "632", + local: "true" + }, + { name: "OwnerSwCfgRomErrorReportingOffset", + desc: "Offset of OWNER_SW_CFG_ROM_ERROR_REPORTING", + type: "int", + default: "384", + local: "true" + }, + { name: "OwnerSwCfgRomErrorReportingSize", + desc: "Size of OWNER_SW_CFG_ROM_ERROR_REPORTING", + type: "int", + default: "4", + local: "true" + }, + { name: "OwnerSwCfgRomBootstrapDisOffset", + desc: "Offset of OWNER_SW_CFG_ROM_BOOTSTRAP_DIS", + type: "int", + default: "388", + local: "true" + }, + { name: "OwnerSwCfgRomBootstrapDisSize", + desc: "Size of OWNER_SW_CFG_ROM_BOOTSTRAP_DIS", + type: "int", + default: "4", + local: "true" + }, + { name: "OwnerSwCfgRomAlertClassEnOffset", + desc: "Offset of OWNER_SW_CFG_ROM_ALERT_CLASS_EN", + type: "int", + default: "392", + local: "true" + }, + { name: "OwnerSwCfgRomAlertClassEnSize", + desc: "Size of OWNER_SW_CFG_ROM_ALERT_CLASS_EN", + type: "int", + default: "4", + local: "true" + }, + { name: "OwnerSwCfgRomAlertEscalationOffset", + desc: "Offset of OWNER_SW_CFG_ROM_ALERT_ESCALATION", + type: "int", + default: "396", + local: "true" + }, + { name: "OwnerSwCfgRomAlertEscalationSize", + desc: "Size of OWNER_SW_CFG_ROM_ALERT_ESCALATION", + type: "int", + default: "4", + local: "true" + }, + { name: "OwnerSwCfgRomAlertClassificationOffset", + desc: "Offset of OWNER_SW_CFG_ROM_ALERT_CLASSIFICATION", + type: "int", + default: "400", + local: "true" + }, + { name: "OwnerSwCfgRomAlertClassificationSize", + desc: "Size of OWNER_SW_CFG_ROM_ALERT_CLASSIFICATION", + type: "int", + default: "400", + local: "true" + }, + { name: "OwnerSwCfgRomLocalAlertClassificationOffset", + desc: "Offset of OWNER_SW_CFG_ROM_LOCAL_ALERT_CLASSIFICATION", + type: "int", + default: "800", + local: "true" + }, + { name: "OwnerSwCfgRomLocalAlertClassificationSize", + desc: "Size of OWNER_SW_CFG_ROM_LOCAL_ALERT_CLASSIFICATION", + type: "int", + default: "64", + local: "true" + }, + { name: "OwnerSwCfgRomAlertAccumThreshOffset", + desc: "Offset of OWNER_SW_CFG_ROM_ALERT_ACCUM_THRESH", + type: "int", + default: "864", + local: "true" + }, + { name: "OwnerSwCfgRomAlertAccumThreshSize", + desc: "Size of OWNER_SW_CFG_ROM_ALERT_ACCUM_THRESH", + type: "int", + default: "16", + local: "true" + }, + { name: "OwnerSwCfgRomAlertTimeoutCyclesOffset", + desc: "Offset of OWNER_SW_CFG_ROM_ALERT_TIMEOUT_CYCLES", + type: "int", + default: "880", + local: "true" + }, + { name: "OwnerSwCfgRomAlertTimeoutCyclesSize", + desc: "Size of OWNER_SW_CFG_ROM_ALERT_TIMEOUT_CYCLES", + type: "int", + default: "16", + local: "true" + }, + { name: "OwnerSwCfgRomAlertPhaseCyclesOffset", + desc: "Offset of OWNER_SW_CFG_ROM_ALERT_PHASE_CYCLES", + type: "int", + default: "896", + local: "true" + }, + { name: "OwnerSwCfgRomAlertPhaseCyclesSize", + desc: "Size of OWNER_SW_CFG_ROM_ALERT_PHASE_CYCLES", + type: "int", + default: "64", + local: "true" + }, + { name: "OwnerSwCfgRomAlertDigestProdOffset", + desc: "Offset of OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD", + type: "int", + default: "960", + local: "true" + }, + { name: "OwnerSwCfgRomAlertDigestProdSize", + desc: "Size of OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD", + type: "int", + default: "4", + local: "true" + }, + { name: "OwnerSwCfgRomAlertDigestProdEndOffset", + desc: "Offset of OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD_END", + type: "int", + default: "964", + local: "true" + }, + { name: "OwnerSwCfgRomAlertDigestProdEndSize", + desc: "Size of OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD_END", + type: "int", + default: "4", + local: "true" + }, + { name: "OwnerSwCfgRomAlertDigestDevOffset", + desc: "Offset of OWNER_SW_CFG_ROM_ALERT_DIGEST_DEV", + type: "int", + default: "968", + local: "true" + }, + { name: "OwnerSwCfgRomAlertDigestDevSize", + desc: "Size of OWNER_SW_CFG_ROM_ALERT_DIGEST_DEV", + type: "int", + default: "4", + local: "true" + }, + { name: "OwnerSwCfgRomAlertDigestRmaOffset", + desc: "Offset of OWNER_SW_CFG_ROM_ALERT_DIGEST_RMA", + type: "int", + default: "972", + local: "true" + }, + { name: "OwnerSwCfgRomAlertDigestRmaSize", + desc: "Size of OWNER_SW_CFG_ROM_ALERT_DIGEST_RMA", + type: "int", + default: "4", + local: "true" + }, + { name: "OwnerSwCfgRomWatchdogBiteThresholdCyclesOffset", + desc: "Offset of OWNER_SW_CFG_ROM_WATCHDOG_BITE_THRESHOLD_CYCLES", + type: "int", + default: "976", + local: "true" + }, + { name: "OwnerSwCfgRomWatchdogBiteThresholdCyclesSize", + desc: "Size of OWNER_SW_CFG_ROM_WATCHDOG_BITE_THRESHOLD_CYCLES", + type: "int", + default: "4", + local: "true" + }, + { name: "OwnerSwCfgRomKeymgrRomExtMeasEnOffset", + desc: "Offset of OWNER_SW_CFG_ROM_KEYMGR_ROM_EXT_MEAS_EN", + type: "int", + default: "980", + local: "true" + }, + { name: "OwnerSwCfgRomKeymgrRomExtMeasEnSize", + desc: "Size of OWNER_SW_CFG_ROM_KEYMGR_ROM_EXT_MEAS_EN", + type: "int", + default: "4", + local: "true" + }, + { name: "OwnerSwCfgManufStateOffset", + desc: "Offset of OWNER_SW_CFG_MANUF_STATE", + type: "int", + default: "984", + local: "true" + }, + { name: "OwnerSwCfgManufStateSize", + desc: "Size of OWNER_SW_CFG_MANUF_STATE", + type: "int", + default: "4", + local: "true" + }, + { name: "OwnerSwCfgRomRstmgrInfoEnOffset", + desc: "Offset of OWNER_SW_CFG_ROM_RSTMGR_INFO_EN", + type: "int", + default: "988", + local: "true" + }, + { name: "OwnerSwCfgRomRstmgrInfoEnSize", + desc: "Size of OWNER_SW_CFG_ROM_RSTMGR_INFO_EN", + type: "int", + default: "4", + local: "true" + }, + { name: "OwnerSwCfgDigestOffset", + desc: "Offset of OWNER_SW_CFG_DIGEST", + type: "int", + default: "1008", + local: "true" + }, + { name: "OwnerSwCfgDigestSize", + desc: "Size of OWNER_SW_CFG_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "OwnershipSlotStateOffset", + desc: "Offset of the OWNERSHIP_SLOT_STATE partition", + type: "int", + default: "1016", + local: "true" + }, + { name: "OwnershipSlotStateSize", + desc: "Size of the OWNERSHIP_SLOT_STATE partition", + type: "int", + default: "48", + local: "true" + }, + { name: "OwnershipSlotStateRotOwnerAuthOffset", + desc: "Offset of OWNERSHIP_SLOT_STATE_ROT_OWNER_AUTH", + type: "int", + default: "1016", + local: "true" + }, + { name: "OwnershipSlotStateRotOwnerAuthSize", + desc: "Size of OWNERSHIP_SLOT_STATE_ROT_OWNER_AUTH", + type: "int", + default: "16", + local: "true" + }, + { name: "OwnershipSlotStatePlatIntegAuthOffset", + desc: "Offset of OWNERSHIP_SLOT_STATE_PLAT_INTEG_AUTH", + type: "int", + default: "1032", + local: "true" + }, + { name: "OwnershipSlotStatePlatIntegAuthSize", + desc: "Size of OWNERSHIP_SLOT_STATE_PLAT_INTEG_AUTH", + type: "int", + default: "16", + local: "true" + }, + { name: "OwnershipSlotStatePlatOwnerAuthOffset", + desc: "Offset of OWNERSHIP_SLOT_STATE_PLAT_OWNER_AUTH", + type: "int", + default: "1048", + local: "true" + }, + { name: "OwnershipSlotStatePlatOwnerAuthSize", + desc: "Size of OWNERSHIP_SLOT_STATE_PLAT_OWNER_AUTH", + type: "int", + default: "16", + local: "true" + }, + { name: "RotCreatorAuthOffset", + desc: "Offset of the ROT_CREATOR_AUTH partition", + type: "int", + default: "1064", + local: "true" + }, + { name: "RotCreatorAuthSize", + desc: "Size of the ROT_CREATOR_AUTH partition", + type: "int", + default: "1424", + local: "true" + }, + { name: "RotCreatorAuthNonRawMfwCodesignKeyOffset", + desc: "Offset of ROT_CREATOR_AUTH_NON_RAW_MFW_CODESIGN_KEY", + type: "int", + default: "1064", + local: "true" + }, + { name: "RotCreatorAuthNonRawMfwCodesignKeySize", + desc: "Size of ROT_CREATOR_AUTH_NON_RAW_MFW_CODESIGN_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "RotCreatorAuthOwnershipStateOffset", + desc: "Offset of ROT_CREATOR_AUTH_OWNERSHIP_STATE", + type: "int", + default: "1224", + local: "true" + }, + { name: "RotCreatorAuthOwnershipStateSize", + desc: "Size of ROT_CREATOR_AUTH_OWNERSHIP_STATE", + type: "int", + default: "4", + local: "true" + }, + { name: "RotCreatorAuthRom2PatchSigverifyKeyOffset", + desc: "Offset of ROT_CREATOR_AUTH_ROM2_PATCH_SIGVERIFY_KEY", + type: "int", + default: "1228", + local: "true" + }, + { name: "RotCreatorAuthRom2PatchSigverifyKeySize", + desc: "Size of ROT_CREATOR_AUTH_ROM2_PATCH_SIGVERIFY_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "RotCreatorAuthKeymanifestKeyOffset", + desc: "Offset of ROT_CREATOR_AUTH_KEYMANIFEST_KEY", + type: "int", + default: "1388", + local: "true" + }, + { name: "RotCreatorAuthKeymanifestKeySize", + desc: "Size of ROT_CREATOR_AUTH_KEYMANIFEST_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "RotCreatorAuthUnlock4xferKeyOffset", + desc: "Offset of ROT_CREATOR_AUTH_UNLOCK4XFER_KEY", + type: "int", + default: "1548", + local: "true" + }, + { name: "RotCreatorAuthUnlock4xferKeySize", + desc: "Size of ROT_CREATOR_AUTH_UNLOCK4XFER_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "RotCreatorAuthIdentityCertOffset", + desc: "Offset of ROT_CREATOR_AUTH_IDENTITY_CERT", + type: "int", + default: "1708", + local: "true" + }, + { name: "RotCreatorAuthIdentityCertSize", + desc: "Size of ROT_CREATOR_AUTH_IDENTITY_CERT", + type: "int", + default: "768", + local: "true" + }, + { name: "RotCreatorAuthDigestOffset", + desc: "Offset of ROT_CREATOR_AUTH_DIGEST", + type: "int", + default: "2480", + local: "true" + }, + { name: "RotCreatorAuthDigestSize", + desc: "Size of ROT_CREATOR_AUTH_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "RotOwnerAuthSlot0Offset", + desc: "Offset of the ROT_OWNER_AUTH_SLOT0 partition", + type: "int", + default: "2488", + local: "true" + }, + { name: "RotOwnerAuthSlot0Size", + desc: "Size of the ROT_OWNER_AUTH_SLOT0 partition", + type: "int", + default: "328", + local: "true" + }, + { name: "RotOwnerAuthSlot0KeymanifestKeyOffset", + desc: "Offset of ROT_OWNER_AUTH_SLOT0_KEYMANIFEST_KEY", + type: "int", + default: "2488", + local: "true" + }, + { name: "RotOwnerAuthSlot0KeymanifestKeySize", + desc: "Size of ROT_OWNER_AUTH_SLOT0_KEYMANIFEST_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "RotOwnerAuthSlot0Unlock4xferKeyOffset", + desc: "Offset of ROT_OWNER_AUTH_SLOT0_UNLOCK4XFER_KEY", + type: "int", + default: "2648", + local: "true" + }, + { name: "RotOwnerAuthSlot0Unlock4xferKeySize", + desc: "Size of ROT_OWNER_AUTH_SLOT0_UNLOCK4XFER_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "RotOwnerAuthSlot0DigestOffset", + desc: "Offset of ROT_OWNER_AUTH_SLOT0_DIGEST", + type: "int", + default: "2808", + local: "true" + }, + { name: "RotOwnerAuthSlot0DigestSize", + desc: "Size of ROT_OWNER_AUTH_SLOT0_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "RotOwnerAuthSlot1Offset", + desc: "Offset of the ROT_OWNER_AUTH_SLOT1 partition", + type: "int", + default: "2816", + local: "true" + }, + { name: "RotOwnerAuthSlot1Size", + desc: "Size of the ROT_OWNER_AUTH_SLOT1 partition", + type: "int", + default: "328", + local: "true" + }, + { name: "RotOwnerAuthSlot1KeymanifestKeyOffset", + desc: "Offset of ROT_OWNER_AUTH_SLOT1_KEYMANIFEST_KEY", + type: "int", + default: "2816", + local: "true" + }, + { name: "RotOwnerAuthSlot1KeymanifestKeySize", + desc: "Size of ROT_OWNER_AUTH_SLOT1_KEYMANIFEST_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "RotOwnerAuthSlot1Unlock4xferKeyOffset", + desc: "Offset of ROT_OWNER_AUTH_SLOT1_UNLOCK4XFER_KEY", + type: "int", + default: "2976", + local: "true" + }, + { name: "RotOwnerAuthSlot1Unlock4xferKeySize", + desc: "Size of ROT_OWNER_AUTH_SLOT1_UNLOCK4XFER_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "RotOwnerAuthSlot1DigestOffset", + desc: "Offset of ROT_OWNER_AUTH_SLOT1_DIGEST", + type: "int", + default: "3136", + local: "true" + }, + { name: "RotOwnerAuthSlot1DigestSize", + desc: "Size of ROT_OWNER_AUTH_SLOT1_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "PlatIntegAuthSlot0Offset", + desc: "Offset of the PLAT_INTEG_AUTH_SLOT0 partition", + type: "int", + default: "3144", + local: "true" + }, + { name: "PlatIntegAuthSlot0Size", + desc: "Size of the PLAT_INTEG_AUTH_SLOT0 partition", + type: "int", + default: "328", + local: "true" + }, + { name: "PlatIntegAuthSlot0KeymanifestKeyOffset", + desc: "Offset of PLAT_INTEG_AUTH_SLOT0_KEYMANIFEST_KEY", + type: "int", + default: "3144", + local: "true" + }, + { name: "PlatIntegAuthSlot0KeymanifestKeySize", + desc: "Size of PLAT_INTEG_AUTH_SLOT0_KEYMANIFEST_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "PlatIntegAuthSlot0Unlock4xferKeyOffset", + desc: "Offset of PLAT_INTEG_AUTH_SLOT0_UNLOCK4XFER_KEY", + type: "int", + default: "3304", + local: "true" + }, + { name: "PlatIntegAuthSlot0Unlock4xferKeySize", + desc: "Size of PLAT_INTEG_AUTH_SLOT0_UNLOCK4XFER_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "PlatIntegAuthSlot0DigestOffset", + desc: "Offset of PLAT_INTEG_AUTH_SLOT0_DIGEST", + type: "int", + default: "3464", + local: "true" + }, + { name: "PlatIntegAuthSlot0DigestSize", + desc: "Size of PLAT_INTEG_AUTH_SLOT0_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "PlatIntegAuthSlot1Offset", + desc: "Offset of the PLAT_INTEG_AUTH_SLOT1 partition", + type: "int", + default: "3472", + local: "true" + }, + { name: "PlatIntegAuthSlot1Size", + desc: "Size of the PLAT_INTEG_AUTH_SLOT1 partition", + type: "int", + default: "328", + local: "true" + }, + { name: "PlatIntegAuthSlot1KeymanifestKeyOffset", + desc: "Offset of PLAT_INTEG_AUTH_SLOT1_KEYMANIFEST_KEY", + type: "int", + default: "3472", + local: "true" + }, + { name: "PlatIntegAuthSlot1KeymanifestKeySize", + desc: "Size of PLAT_INTEG_AUTH_SLOT1_KEYMANIFEST_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "PlatIntegAuthSlot1Unlock4xferKeyOffset", + desc: "Offset of PLAT_INTEG_AUTH_SLOT1_UNLOCK4XFER_KEY", + type: "int", + default: "3632", + local: "true" + }, + { name: "PlatIntegAuthSlot1Unlock4xferKeySize", + desc: "Size of PLAT_INTEG_AUTH_SLOT1_UNLOCK4XFER_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "PlatIntegAuthSlot1DigestOffset", + desc: "Offset of PLAT_INTEG_AUTH_SLOT1_DIGEST", + type: "int", + default: "3792", + local: "true" + }, + { name: "PlatIntegAuthSlot1DigestSize", + desc: "Size of PLAT_INTEG_AUTH_SLOT1_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "PlatOwnerAuthSlot0Offset", + desc: "Offset of the PLAT_OWNER_AUTH_SLOT0 partition", + type: "int", + default: "3800", + local: "true" + }, + { name: "PlatOwnerAuthSlot0Size", + desc: "Size of the PLAT_OWNER_AUTH_SLOT0 partition", + type: "int", + default: "328", + local: "true" + }, + { name: "PlatOwnerAuthSlot0KeymanifestKeyOffset", + desc: "Offset of PLAT_OWNER_AUTH_SLOT0_KEYMANIFEST_KEY", + type: "int", + default: "3800", + local: "true" + }, + { name: "PlatOwnerAuthSlot0KeymanifestKeySize", + desc: "Size of PLAT_OWNER_AUTH_SLOT0_KEYMANIFEST_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "PlatOwnerAuthSlot0Unlock4xferKeyOffset", + desc: "Offset of PLAT_OWNER_AUTH_SLOT0_UNLOCK4XFER_KEY", + type: "int", + default: "3960", + local: "true" + }, + { name: "PlatOwnerAuthSlot0Unlock4xferKeySize", + desc: "Size of PLAT_OWNER_AUTH_SLOT0_UNLOCK4XFER_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "PlatOwnerAuthSlot0DigestOffset", + desc: "Offset of PLAT_OWNER_AUTH_SLOT0_DIGEST", + type: "int", + default: "4120", + local: "true" + }, + { name: "PlatOwnerAuthSlot0DigestSize", + desc: "Size of PLAT_OWNER_AUTH_SLOT0_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "PlatOwnerAuthSlot1Offset", + desc: "Offset of the PLAT_OWNER_AUTH_SLOT1 partition", + type: "int", + default: "4128", + local: "true" + }, + { name: "PlatOwnerAuthSlot1Size", + desc: "Size of the PLAT_OWNER_AUTH_SLOT1 partition", + type: "int", + default: "328", + local: "true" + }, + { name: "PlatOwnerAuthSlot1KeymanifestKeyOffset", + desc: "Offset of PLAT_OWNER_AUTH_SLOT1_KEYMANIFEST_KEY", + type: "int", + default: "4128", + local: "true" + }, + { name: "PlatOwnerAuthSlot1KeymanifestKeySize", + desc: "Size of PLAT_OWNER_AUTH_SLOT1_KEYMANIFEST_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "PlatOwnerAuthSlot1Unlock4xferKeyOffset", + desc: "Offset of PLAT_OWNER_AUTH_SLOT1_UNLOCK4XFER_KEY", + type: "int", + default: "4288", + local: "true" + }, + { name: "PlatOwnerAuthSlot1Unlock4xferKeySize", + desc: "Size of PLAT_OWNER_AUTH_SLOT1_UNLOCK4XFER_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "PlatOwnerAuthSlot1DigestOffset", + desc: "Offset of PLAT_OWNER_AUTH_SLOT1_DIGEST", + type: "int", + default: "4448", + local: "true" + }, + { name: "PlatOwnerAuthSlot1DigestSize", + desc: "Size of PLAT_OWNER_AUTH_SLOT1_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "PlatOwnerAuthSlot2Offset", + desc: "Offset of the PLAT_OWNER_AUTH_SLOT2 partition", + type: "int", + default: "4456", + local: "true" + }, + { name: "PlatOwnerAuthSlot2Size", + desc: "Size of the PLAT_OWNER_AUTH_SLOT2 partition", + type: "int", + default: "328", + local: "true" + }, + { name: "PlatOwnerAuthSlot2KeymanifestKeyOffset", + desc: "Offset of PLAT_OWNER_AUTH_SLOT2_KEYMANIFEST_KEY", + type: "int", + default: "4456", + local: "true" + }, + { name: "PlatOwnerAuthSlot2KeymanifestKeySize", + desc: "Size of PLAT_OWNER_AUTH_SLOT2_KEYMANIFEST_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "PlatOwnerAuthSlot2Unlock4xferKeyOffset", + desc: "Offset of PLAT_OWNER_AUTH_SLOT2_UNLOCK4XFER_KEY", + type: "int", + default: "4616", + local: "true" + }, + { name: "PlatOwnerAuthSlot2Unlock4xferKeySize", + desc: "Size of PLAT_OWNER_AUTH_SLOT2_UNLOCK4XFER_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "PlatOwnerAuthSlot2DigestOffset", + desc: "Offset of PLAT_OWNER_AUTH_SLOT2_DIGEST", + type: "int", + default: "4776", + local: "true" + }, + { name: "PlatOwnerAuthSlot2DigestSize", + desc: "Size of PLAT_OWNER_AUTH_SLOT2_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "PlatOwnerAuthSlot3Offset", + desc: "Offset of the PLAT_OWNER_AUTH_SLOT3 partition", + type: "int", + default: "4784", + local: "true" + }, + { name: "PlatOwnerAuthSlot3Size", + desc: "Size of the PLAT_OWNER_AUTH_SLOT3 partition", + type: "int", + default: "328", + local: "true" + }, + { name: "PlatOwnerAuthSlot3KeymanifestKeyOffset", + desc: "Offset of PLAT_OWNER_AUTH_SLOT3_KEYMANIFEST_KEY", + type: "int", + default: "4784", + local: "true" + }, + { name: "PlatOwnerAuthSlot3KeymanifestKeySize", + desc: "Size of PLAT_OWNER_AUTH_SLOT3_KEYMANIFEST_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "PlatOwnerAuthSlot3Unlock4xferKeyOffset", + desc: "Offset of PLAT_OWNER_AUTH_SLOT3_UNLOCK4XFER_KEY", + type: "int", + default: "4944", + local: "true" + }, + { name: "PlatOwnerAuthSlot3Unlock4xferKeySize", + desc: "Size of PLAT_OWNER_AUTH_SLOT3_UNLOCK4XFER_KEY", + type: "int", + default: "160", + local: "true" + }, + { name: "PlatOwnerAuthSlot3DigestOffset", + desc: "Offset of PLAT_OWNER_AUTH_SLOT3_DIGEST", + type: "int", + default: "5104", + local: "true" + }, + { name: "PlatOwnerAuthSlot3DigestSize", + desc: "Size of PLAT_OWNER_AUTH_SLOT3_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "ExtNvmOffset", + desc: "Offset of the EXT_NVM partition", + type: "int", + default: "5112", + local: "true" + }, + { name: "ExtNvmSize", + desc: "Size of the EXT_NVM partition", + type: "int", + default: "1024", + local: "true" + }, + { name: "ExtNvmAntireplayFreshnessCntOffset", + desc: "Offset of EXT_NVM_ANTIREPLAY_FRESHNESS_CNT", + type: "int", + default: "5112", + local: "true" + }, + { name: "ExtNvmAntireplayFreshnessCntSize", + desc: "Size of EXT_NVM_ANTIREPLAY_FRESHNESS_CNT", + type: "int", + default: "1024", + local: "true" + }, + { name: "RomPatchOffset", + desc: "Offset of the ROM_PATCH partition", + type: "int", + default: "6136", + local: "true" + }, + { name: "RomPatchSize", + desc: "Size of the ROM_PATCH partition", + type: "int", + default: "9784", + local: "true" + }, + { name: "RomPatchDataOffset", + desc: "Offset of ROM_PATCH_DATA", + type: "int", + default: "6136", + local: "true" + }, + { name: "RomPatchDataSize", + desc: "Size of ROM_PATCH_DATA", + type: "int", + default: "9192", + local: "true" + }, + { name: "RomPatchDigestOffset", + desc: "Offset of ROM_PATCH_DIGEST", + type: "int", + default: "15912", + local: "true" + }, + { name: "RomPatchDigestSize", + desc: "Size of ROM_PATCH_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "HwCfg0Offset", + desc: "Offset of the HW_CFG0 partition", + type: "int", + default: "15920", + local: "true" + }, + { name: "HwCfg0Size", + desc: "Size of the HW_CFG0 partition", + type: "int", + default: "72", + local: "true" + }, + { name: "DeviceIdOffset", + desc: "Offset of DEVICE_ID", + type: "int", + default: "15920", + local: "true" + }, + { name: "DeviceIdSize", + desc: "Size of DEVICE_ID", + type: "int", + default: "32", + local: "true" + }, + { name: "ManufStateOffset", + desc: "Offset of MANUF_STATE", + type: "int", + default: "15952", + local: "true" + }, + { name: "ManufStateSize", + desc: "Size of MANUF_STATE", + type: "int", + default: "32", + local: "true" + }, + { name: "HwCfg0DigestOffset", + desc: "Offset of HW_CFG0_DIGEST", + type: "int", + default: "15984", + local: "true" + }, + { name: "HwCfg0DigestSize", + desc: "Size of HW_CFG0_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "HwCfg1Offset", + desc: "Offset of the HW_CFG1 partition", + type: "int", + default: "15992", + local: "true" + }, + { name: "HwCfg1Size", + desc: "Size of the HW_CFG1 partition", + type: "int", + default: "16", + local: "true" + }, + { name: "SocDbgStateOffset", + desc: "Offset of SOC_DBG_STATE", + type: "int", + default: "15992", + local: "true" + }, + { name: "SocDbgStateSize", + desc: "Size of SOC_DBG_STATE", + type: "int", + default: "4", + local: "true" + }, + { name: "EnSramIfetchOffset", + desc: "Offset of EN_SRAM_IFETCH", + type: "int", + default: "15996", + local: "true" + }, + { name: "EnSramIfetchSize", + desc: "Size of EN_SRAM_IFETCH", + type: "int", + default: "1", + local: "true" + }, + { name: "HwCfg1DigestOffset", + desc: "Offset of HW_CFG1_DIGEST", + type: "int", + default: "16000", + local: "true" + }, + { name: "HwCfg1DigestSize", + desc: "Size of HW_CFG1_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "Secret0Offset", + desc: "Offset of the SECRET0 partition", + type: "int", + default: "16008", + local: "true" + }, + { name: "Secret0Size", + desc: "Size of the SECRET0 partition", + type: "int", + default: "40", + local: "true" + }, + { name: "TestUnlockTokenOffset", + desc: "Offset of TEST_UNLOCK_TOKEN", + type: "int", + default: "16008", + local: "true" + }, + { name: "TestUnlockTokenSize", + desc: "Size of TEST_UNLOCK_TOKEN", + type: "int", + default: "16", + local: "true" + }, + { name: "TestExitTokenOffset", + desc: "Offset of TEST_EXIT_TOKEN", + type: "int", + default: "16024", + local: "true" + }, + { name: "TestExitTokenSize", + desc: "Size of TEST_EXIT_TOKEN", + type: "int", + default: "16", + local: "true" + }, + { name: "Secret0DigestOffset", + desc: "Offset of SECRET0_DIGEST", + type: "int", + default: "16040", + local: "true" + }, + { name: "Secret0DigestSize", + desc: "Size of SECRET0_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "Secret1Offset", + desc: "Offset of the SECRET1 partition", + type: "int", + default: "16048", + local: "true" + }, + { name: "Secret1Size", + desc: "Size of the SECRET1 partition", + type: "int", + default: "88", + local: "true" + }, + { name: "FlashAddrKeySeedOffset", + desc: "Offset of FLASH_ADDR_KEY_SEED", + type: "int", + default: "16048", + local: "true" + }, + { name: "FlashAddrKeySeedSize", + desc: "Size of FLASH_ADDR_KEY_SEED", + type: "int", + default: "32", + local: "true" + }, + { name: "FlashDataKeySeedOffset", + desc: "Offset of FLASH_DATA_KEY_SEED", + type: "int", + default: "16080", + local: "true" + }, + { name: "FlashDataKeySeedSize", + desc: "Size of FLASH_DATA_KEY_SEED", + type: "int", + default: "32", + local: "true" + }, + { name: "SramDataKeySeedOffset", + desc: "Offset of SRAM_DATA_KEY_SEED", + type: "int", + default: "16112", + local: "true" + }, + { name: "SramDataKeySeedSize", + desc: "Size of SRAM_DATA_KEY_SEED", + type: "int", + default: "16", + local: "true" + }, + { name: "Secret1DigestOffset", + desc: "Offset of SECRET1_DIGEST", + type: "int", + default: "16128", + local: "true" + }, + { name: "Secret1DigestSize", + desc: "Size of SECRET1_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "Secret2Offset", + desc: "Offset of the SECRET2 partition", + type: "int", + default: "16136", + local: "true" + }, + { name: "Secret2Size", + desc: "Size of the SECRET2 partition", + type: "int", + default: "120", + local: "true" + }, + { name: "RmaTokenOffset", + desc: "Offset of RMA_TOKEN", + type: "int", + default: "16136", + local: "true" + }, + { name: "RmaTokenSize", + desc: "Size of RMA_TOKEN", + type: "int", + default: "16", + local: "true" + }, + { name: "CreatorRootKeyShare0Offset", + desc: "Offset of CREATOR_ROOT_KEY_SHARE0", + type: "int", + default: "16152", + local: "true" + }, + { name: "CreatorRootKeyShare0Size", + desc: "Size of CREATOR_ROOT_KEY_SHARE0", + type: "int", + default: "32", + local: "true" + }, + { name: "CreatorRootKeyShare1Offset", + desc: "Offset of CREATOR_ROOT_KEY_SHARE1", + type: "int", + default: "16184", + local: "true" + }, + { name: "CreatorRootKeyShare1Size", + desc: "Size of CREATOR_ROOT_KEY_SHARE1", + type: "int", + default: "32", + local: "true" + }, + { name: "CreatorSeedOffset", + desc: "Offset of CREATOR_SEED", + type: "int", + default: "16216", + local: "true" + }, + { name: "CreatorSeedSize", + desc: "Size of CREATOR_SEED", + type: "int", + default: "32", + local: "true" + }, + { name: "Secret2DigestOffset", + desc: "Offset of SECRET2_DIGEST", + type: "int", + default: "16248", + local: "true" + }, + { name: "Secret2DigestSize", + desc: "Size of SECRET2_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "Secret3Offset", + desc: "Offset of the SECRET3 partition", + type: "int", + default: "16256", + local: "true" + }, + { name: "Secret3Size", + desc: "Size of the SECRET3 partition", + type: "int", + default: "40", + local: "true" + }, + { name: "OwnerSeedOffset", + desc: "Offset of OWNER_SEED", + type: "int", + default: "16256", + local: "true" + }, + { name: "OwnerSeedSize", + desc: "Size of OWNER_SEED", + type: "int", + default: "32", + local: "true" + }, + { name: "Secret3DigestOffset", + desc: "Offset of SECRET3_DIGEST", + type: "int", + default: "16288", + local: "true" + }, + { name: "Secret3DigestSize", + desc: "Size of SECRET3_DIGEST", + type: "int", + default: "8", + local: "true" + }, + { name: "LifeCycleOffset", + desc: "Offset of the LIFE_CYCLE partition", + type: "int", + default: "16296", + local: "true" + }, + { name: "LifeCycleSize", + desc: "Size of the LIFE_CYCLE partition", + type: "int", + default: "88", + local: "true" + }, + { name: "LcTransitionCntOffset", + desc: "Offset of LC_TRANSITION_CNT", + type: "int", + default: "16296", + local: "true" + }, + { name: "LcTransitionCntSize", + desc: "Size of LC_TRANSITION_CNT", + type: "int", + default: "48", + local: "true" + }, + { name: "LcStateOffset", + desc: "Offset of LC_STATE", + type: "int", + default: "16344", + local: "true" + }, + { name: "LcStateSize", + desc: "Size of LC_STATE", + type: "int", + default: "40", + local: "true" + }, + ] + + ///////////////////////////// + // Intermodule Connections // + ///////////////////////////// + + inter_signal_list: [ + // OTP dedicated power connection from AST + { struct: "" + type: "io" + name: "otp_ext_voltage_h" + act: "none" + default: "'0" + package: "", + } + // Power sequencing signals to AST + { struct: "otp_ast_req" + type: "uni" + name: "otp_ast_pwr_seq" + act: "req" + default: "'0" + package: "otp_ctrl_pkg" + desc: "Power sequencing signals to AST (VDD domain)." + } + // Power sequencing signals from AST + { struct: "otp_ast_rsp" + type: "uni" + name: "otp_ast_pwr_seq_h" + act: "rcv" + default: "'0" + package: "otp_ctrl_pkg" + desc: "Power sequencing signals coming from AST (VCC domain)." + } + // EDN interface + { struct: "edn" + type: "req_rsp" + name: "edn" + act: "req" + package: "edn_pkg" + desc: "Entropy request to the entropy distribution network for LFSR reseeding and ephemeral key derivation." + } + // Power manager init command + { struct: "pwr_otp" + type: "req_rsp" + name: "pwr_otp" + act: "rsp" + default: "'0" + package: "pwrmgr_pkg" + desc: "Initialization request/acknowledge from/to power manager." + } + // Macro-specific test signals to/from LC TAP + { struct: "lc_otp_vendor_test" + type: "req_rsp" + name: "lc_otp_vendor_test" + act: "rsp" + default: "'0" + package: "otp_ctrl_pkg" + desc: "Vendor test control signals from/to the life cycle TAP." + } + // LC transition command + { struct: "lc_otp_program" + type: "req_rsp" + name: "lc_otp_program" + act: "rsp" + default: "'0" + package: "otp_ctrl_pkg" + desc: "Life cycle state transition interface." + } + // Broadcast to LC + { struct: "otp_lc_data" + type: "uni" + name: "otp_lc_data" + act: "req" + default: "'0" + package: "otp_ctrl_pkg" + desc: ''' + Life cycle state output holding the current life cycle state, + the value of the transition counter and the tokens needed for life cycle transitions. + ''' + } + // Broadcast from LC + { struct: "lc_tx" + type: "uni" + name: "lc_escalate_en" + act: "rcv" + default: "lc_ctrl_pkg::Off" + package: "lc_ctrl_pkg" + desc: ''' + Life cycle escalation enable coming from life cycle controller. + This signal moves all FSMs within OTP into the error state. + ''' + } + { struct: "lc_tx" + type: "uni" + name: "lc_creator_seed_sw_rw_en" + act: "rcv" + default: "lc_ctrl_pkg::Off" + package: "lc_ctrl_pkg" + desc: ''' + Provision enable qualifier coming from life cycle controller. + This signal enables SW read / write access to the RMA_TOKEN and CREATOR_ROOT_KEY_SHARE0 and CREATOR_ROOT_KEY_SHARE1. + ''' + } + { struct: "lc_tx" + type: "uni" + name: "lc_owner_seed_sw_rw_en" + act: "rcv" + default: "lc_ctrl_pkg::Off" + package: "lc_ctrl_pkg" + desc: ''' + Provision enable qualifier coming from life cycle controller. + This signal enables SW read / write access to the OWNER_SEED. + ''' + } + { struct: "lc_tx" + type: "uni" + name: "lc_seed_hw_rd_en" + act: "rcv" + default: "lc_ctrl_pkg::Off" + package: "lc_ctrl_pkg" + desc: ''' + Seed read enable coming from life cycle controller. + This signal enables HW read access to the CREATOR_ROOT_KEY_SHARE0 and CREATOR_ROOT_KEY_SHARE1. + ''' + } + { struct: "lc_tx" + type: "uni" + name: "lc_dft_en" + act: "rcv" + default: "lc_ctrl_pkg::Off" + package: "lc_ctrl_pkg" + desc: ''' + Test enable qualifier coming from life cycle controller. + This signals enables the TL-UL access port to the proprietary OTP IP. + ''' + } + { struct: "lc_tx" + type: "uni" + name: "lc_check_byp_en" + act: "rcv" + default: "lc_ctrl_pkg::Off" + package: "lc_ctrl_pkg" + desc: ''' + Life cycle partition check bypass signal. + This signal causes the life cycle partition to bypass consistency checks during life cycle state transitions in order to prevent spurious consistency check failures. + ''' + } + // Broadcast to Key Manager + { struct: "otp_keymgr_key" + type: "uni" + name: "otp_keymgr_key" + act: "req" + default: "'0" + package: "otp_ctrl_pkg" + desc: "Key output to the key manager holding CREATOR_ROOT_KEY_SHARE0 and CREATOR_ROOT_KEY_SHARE1." + } + // Broadcast to Flash Controller + { struct: "flash_otp_key" + type: "req_rsp" + name: "flash_otp_key" + act: "rsp" + default: "'0" + package: "otp_ctrl_pkg" + desc: "Key derivation interface for FLASH scrambling." + } + // Key request from SRAM scramblers + { struct: "sram_otp_key" + // TODO: would be nice if this could accept parameters. + // Split this out into an issue. + width: "4" + type: "req_rsp" + name: "sram_otp_key" + act: "rsp" + default: "'0" + package: "otp_ctrl_pkg" + desc: "Array with key derivation interfaces for SRAM scrambling devices." + } + // Key request from OTBN RAM Scrambler + { struct: "otbn_otp_key" + type: "req_rsp" + name: "otbn_otp_key" + act: "rsp" + default: "'0" + package: "otp_ctrl_pkg" + desc: "Key derivation interface for OTBN scrambling devices." + } + // Hardware config partition + { struct: "otp_broadcast" + type: "uni" + name: "otp_broadcast" + act: "req" + default: "'0" + package: "otp_ctrl_part_pkg" + desc: "Output of the HW partitions with breakout data types." + } + // AST observability control + { struct: "ast_obs_ctrl", + type: "uni", + name: "obs_ctrl", + act: "rcv", + package: "ast_pkg" + desc: "AST observability control signals." + } + // prim otp observe bus + { struct: "logic", + type: "uni", + name: "otp_obs", + act: "req", + width: "8", + package: "" + desc: "AST observability bus." + } + ] // inter_signal_list + + ///////////////////// + // Countermeasures // + ///////////////////// + + countermeasures: [ + { name: "BUS.INTEGRITY", + desc: "End-to-end bus integrity scheme." + } + { name: "SECRET.MEM.SCRAMBLE", + desc: "Secret partitions are scrambled with a full-round PRESENT cipher." + } + { name: "PART.MEM.DIGEST", + desc: "Integrity of buffered partitions is ensured via a 64bit digest." + } + { name: "DAI.FSM.SPARSE", + desc: "The direct access interface FSM is sparsely encoded." + } + { name: "KDI.FSM.SPARSE", + desc: "The key derivation interface FSM is sparsely encoded." + } + { name: "LCI.FSM.SPARSE", + desc: "The life cycle interface FSM is sparsely encoded." + } + { name: "PART.FSM.SPARSE", + desc: "The partition FSMs are sparsely encoded." + } + { name: "SCRMBL.FSM.SPARSE", + desc: "The scramble datapath FSM is sparsely encoded." + } + { name: "TIMER.FSM.SPARSE", + desc: "The background check timer FSM is sparsely encoded." + } + { name: "DAI.CTR.REDUN", + desc: "The direct access interface address counter employs a cross-counter implementation." + } + { name: "KDI_SEED.CTR.REDUN", + desc: "The key derivation interface counter employs a cross-counter implementation." + } + { name: "KDI_ENTROPY.CTR.REDUN", + desc: "The key derivation entropy counter employs a cross-counter implementation." + } + { name: "LCI.CTR.REDUN", + desc: "The life cycle interface address counter employs a cross-counter implementation." + } + { name: "PART.CTR.REDUN", + desc: "The address counter of buffered partitions employs a cross-counter implementation." + } + { name: "SCRMBL.CTR.REDUN", + desc: "The srambling datapath counter employs a cross-counter implementation." + } + { name: "TIMER_INTEG.CTR.REDUN", + desc: "The background integrity check timer employs a duplicated counter implementation." + } + { name: "TIMER_CNSTY.CTR.REDUN", + desc: "The background consistency check timer employs a duplicated counter implementation." + } + { name: "TIMER.LFSR.REDUN", + desc: "The background check LFSR is duplicated." + } + { name: "DAI.FSM.LOCAL_ESC", + desc: "The direct access interface FSM is moved into an invalid state upon local escalation." + } + { name: "LCI.FSM.LOCAL_ESC", + desc: "The life cycle interface FSM is moved into an invalid state upon local escalation." + } + { name: "KDI.FSM.LOCAL_ESC", + desc: "The key derivation interface FSM is moved into an invalid state upon local escalation." + } + { name: "PART.FSM.LOCAL_ESC", + desc: "The partition FSMs are moved into an invalid state upon local escalation." + } + { name: "SCRMBL.FSM.LOCAL_ESC", + desc: "The scramble datapath FSM is moved into an invalid state upon local escalation." + } + { name: "TIMER.FSM.LOCAL_ESC", + desc: "The background check timer FSM is moved into an invalid state upon local escalation." + } + { name: "DAI.FSM.GLOBAL_ESC", + desc: "The direct access interface FSM is moved into an invalid state upon global escalation via life cycle." + } + { name: "LCI.FSM.GLOBAL_ESC", + desc: "The life cycle interface FSM is moved into an invalid state upon global escalation via life cycle." + } + { name: "KDI.FSM.GLOBAL_ESC", + desc: "The key derivation interface FSM is moved into an invalid state upon global escalation via life cycle." + } + { name: "PART.FSM.GLOBAL_ESC", + desc: "The partition FSMs are moved into an invalid state upon global escalation via life cycle." + } + { name: "SCRMBL.FSM.GLOBAL_ESC", + desc: "The scramble datapath FSM is moved into an invalid state upon global escalation via life cycle." + } + { name: "TIMER.FSM.GLOBAL_ESC", + desc: "The background check timer FSM is moved into an invalid state upon global escalation via life cycle." + } + { name: "PART.DATA_REG.INTEGRITY", + desc: "All partition buffer registers are protected with ECC on 64bit blocks." + } + { name: "PART.DATA_REG.BKGN_CHK", + desc: "The digest of buffered partitions is recomputed and checked at pseudorandom intervals in the background." + } + { name: "PART.MEM.REGREN" + desc: "Unbuffered ('software') partitions can be read-locked via a CSR until the next system reset." + } + { name: "PART.MEM.SW_UNREADABLE" + desc: "Secret buffered partitions become unreadable to software once they are locked via the digest." + } + { name: "PART.MEM.SW_UNWRITABLE" + desc: "All partitions become unwritable by software once they are locked via the digest." + } + { name: "LC_PART.MEM.SW_NOACCESS" + desc: "The life cycle partition is not directly readable nor writable via software." + } + { name: "ACCESS.CTRL.MUBI", + desc: "The access control signals going from the partitions to the DAI are MUBI encoded." + } + { name: "TOKEN_VALID.CTRL.MUBI", + desc: "The token valid signals going to the life cycle controller are MUBI encoded." + } + { name: "LC_CTRL.INTERSIG.MUBI", + desc: "The life cycle control signals are multibit encoded." + } + { name: "TEST.BUS.LC_GATED", + desc: "Prevent access to test signals and the OTP backdoor interface in non-test lifecycle states." + } + { name: "TEST_TL_LC_GATE.FSM.SPARSE", + desc: "The control FSM inside the TL-UL gating primitive is sparsely encoded." + } + { name: "DIRECT_ACCESS.CONFIG.REGWEN", + desc: "The direct access CSRs are REGWEN protected." + } + { name: "CHECK_TRIGGER.CONFIG.REGWEN", + desc: "The check trigger CSR is REGWEN protected." + } + { name: "CHECK.CONFIG.REGWEN", + desc: "The check CSR is REGWEN protected." + } + { name: "MACRO.MEM.INTEGRITY", + desc: ''' + The OTP macro employs a vendor-specific integrity scheme at the granularity of the native 16bit OTP words. + The scheme is able to at least detect single bit errors. + ''' + } + { name: "MACRO.MEM.CM", + desc: "The OTP macro may contain additional vendor-specific countermeasures." + } + ] + + features: [ + { + name: "OTP_CTRL.PARTITION.VENDOR_TEST" + desc: '''Vendor test partition is used for OTP programming smoke check during manufacturing flow. + In this partition, ECC uncorrectable errors will not lead to fatal errors and alerts. + Instead the error will be reported as correctable ECC error. + ''' + } + { + name: "OTP_CTRL.PARTITION.CREATOR_SW_CFG" + desc: '''During calibration stage, various parameters (clock, voltage, and timing sources) are calibrated and recorded to CREATOR_SW_CFG partition. + ''' + } + { + name: "OTP_CTRL.PARTITION.OWNER_SW_CFG" + desc: "Define attriutes for rom code execution" + } + { + name: "OTP_CTRL.INIT" + desc: '''When power is up, OTP controller reads devices status. + After all reads complete, the controller performs integrity check on the HW_CFG* and SECRET partitions. + Once all integrity checks are complete, the controller marks outputs as valid. + ''' + } + { + name: "OTP_CTRL.ENTROPY_READ" + desc: '''Firmware can read entropy from ENTROPY_SRC block by configuring following field of HW_CFG* partition. + - EN_CSRNG_SW_APP_READ + ''' + } + { + name: "OTP_CTRL.KEY_DERIVATION" + desc: "OTP controller participate key derivation process by providing scramble key seed to SRAM_CTRL and FLASH_CTRL." + } + { + name: "OTP_CTRL.PROGRAM" + desc: '''All other partitions except life cycle partition are programmed through DAI interface. + And once non-zero digest is programmed to these partition, no further write access is allowed. + Life cycle partition is programmed by LC_CTRL. + ''' + } + { + name: "OTP_CTRL.PARTITION.SECRET0" + desc: "Test unlock tokens, Test exit token" + } + { + name: "OTP_CTRL.PARTITION.SECRET1" + desc: "SRAM and FLASH scrambling key" + } + { + name: "OTP_CTRL.PARTITION.SECRET2" + desc: "RMA unlock token and creator root key" + } + { + name: "OTP_CTRL.PARTITION.LIFE_CYCLE" + desc: '''LC state, LC transition count. + This feature is owned by the LC_CTRL and cannot be tested well through the OTP_CTRL CSR interface. + ''' + } + { + name: "OTP_CTRL.PARTITIONS_FEATURE.READ_LOCK" + desc: '''Following partitions can be read lockable by CSR. + - VENDOR_TEST + - CREATOR_SW_CFG + - OWNER_SW_CFG + Following partitions can be read lockable by writing digest. + - SECRET0 + - SECRET1 + - RECRET2 + All read attempt to these partitions after read is locked will trigger AccessError (recoverable). + ''' + } + { + name: "OTP_CTRL.PARTITIONS_FEATURE.WRITE_LOCK" + desc: "All partitions except LIFE_CYCLE can be write lockable by writing digest." + } + { + name: "OTP_CTRL.ERROR_HANDLING.RECOVERABLE" + desc: "Recoverable error is created when unauthorized access atempt are detected via dai interface." + } + { + name: "OTP_CTRL.ERROR_HANDLING.FATAL" + desc: "Unrecoverable errors are created for uncorrectable ecc error, otp macro malfunction and unauthorized access via lc_ctrl." + } + { + name: "OTP_CTRL.BACKGROUND_CHECK.CHECK_TIMEOUT" + desc: "Timeout value for the integrity and consistency checks." + } + { + name: "OTP_CTRL.BACKGROUND_CHECK.INTEGRITY_CHECK_PERIOD" + desc: "The interval which the digest of the partition is recomputed to check integrity of locked partition." + } + { + name: "OTP_CTRL.BACKGROUND_CHECK.CONSISTENCY_CHECK_PERIOD" + desc: "Re-read period of the buffer registers to ensure data is matched with the associated OTP partition." + } + ] + + /////////////// + // Registers // + /////////////// + + regwidth: "32", + registers: { + core: [ + //////////////////////// + // Ctrl / Status CSRs // + //////////////////////// + + { name: "STATUS", + desc: "OTP status register.", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + resval: 0, + tags: [ // OTP internal HW can modify status register + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "0" + name: "VENDOR_TEST_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "1" + name: "CREATOR_SW_CFG_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "2" + name: "OWNER_SW_CFG_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "3" + name: "OWNERSHIP_SLOT_STATE_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "4" + name: "ROT_CREATOR_AUTH_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "5" + name: "ROT_OWNER_AUTH_SLOT0_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "6" + name: "ROT_OWNER_AUTH_SLOT1_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "7" + name: "PLAT_INTEG_AUTH_SLOT0_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "8" + name: "PLAT_INTEG_AUTH_SLOT1_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "9" + name: "PLAT_OWNER_AUTH_SLOT0_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "10" + name: "PLAT_OWNER_AUTH_SLOT1_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "11" + name: "PLAT_OWNER_AUTH_SLOT2_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "12" + name: "PLAT_OWNER_AUTH_SLOT3_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "13" + name: "EXT_NVM_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "14" + name: "ROM_PATCH_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "15" + name: "HW_CFG0_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "16" + name: "HW_CFG1_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "17" + name: "SECRET0_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "18" + name: "SECRET1_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "19" + name: "SECRET2_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "20" + name: "SECRET3_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "21" + name: "LIFE_CYCLE_ERROR" + desc: ''' + Set to 1 if an error occurred in this partition. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "22" + name: "DAI_ERROR" + desc: ''' + Set to 1 if an error occurred in the DAI. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "23" + name: "LCI_ERROR" + desc: ''' + Set to 1 if an error occurred in the LCI. + If set to 1, SW should check the !!ERR_CODE register at the corresponding index. + ''' + } + { bits: "24" + name: "TIMEOUT_ERROR" + desc: ''' + Set to 1 if an integrity or consistency check times out. + This raises an fatal_check_error alert and is an unrecoverable error condition. + ''' + } + { bits: "25" + name: "LFSR_FSM_ERROR" + desc: ''' + Set to 1 if the LFSR timer FSM has reached an invalid state. + This raises an fatal_check_error alert and is an unrecoverable error condition. + ''' + } + { bits: "26" + name: "SCRAMBLING_FSM_ERROR" + desc: ''' + Set to 1 if the scrambling datapath FSM has reached an invalid state. + This raises an fatal_check_error alert and is an unrecoverable error condition. + ''' + } + { bits: "27" + name: "KEY_DERIV_FSM_ERROR" + desc: ''' + Set to 1 if the key derivation FSM has reached an invalid state. + This raises an fatal_check_error alert and is an unrecoverable error condition. + ''' + } + { bits: "28" + name: "BUS_INTEG_ERROR" + desc: ''' + This bit is set to 1 if a fatal bus integrity fault is detected. + This error triggers a fatal_bus_integ_error alert. + ''' + } + { bits: "29" + name: "DAI_IDLE" + desc: "Set to 1 if the DAI is idle and ready to accept commands." + } + { bits: "30" + name: "CHECK_PENDING" + desc: "Set to 1 if an integrity or consistency check triggered by the LFSR timer or via !!CHECK_TRIGGER is pending." + } + ] + } + { multireg: { + name: "ERR_CODE", + desc: ''' + This register holds information about error conditions that occurred in the agents + interacting with the OTP macro via the internal bus. The error codes should be checked + if the partitions, DAI or LCI flag an error in the !!STATUS register, or when an + !!INTR_STATE.otp_error has been triggered. Note that all errors trigger an otp_error + interrupt, and in addition some errors may trigger either an fatal_macro_error or an + fatal_check_error alert. + ''', + count: "NumErrorEntries", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "AGENT", + compact: "false", + resval: 0, + tags: [ // OTP internal HW can modify the error code registers + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { + bits: "2:0" + enum: [ + { value: "0", + name: "NO_ERROR", + desc: ''' + No error condition has occurred. + ''' + }, + { value: "1", + name: "MACRO_ERROR", + desc: ''' + Returned if the OTP macro command was invalid or did not complete successfully + due to a macro malfunction. + This error should never occur during normal operation and is not recoverable. + This error triggers an fatal_macro_error alert. + ''' + }, + { value: "2", + name: "MACRO_ECC_CORR_ERROR", + desc: ''' + A correctable ECC error has occured during an OTP read operation. + The corresponding controller automatically recovers from this error when + issuing a new command. + ''' + }, + { value: "3", + name: "MACRO_ECC_UNCORR_ERROR", + desc: ''' + An uncorrectable ECC error has occurred during an OTP read operation. + This error should never occur during normal operation and is not recoverable. + If this error is present this may be a sign that the device is malfunctioning. + This error triggers an fatal_macro_error alert. + ''' + }, + { value: "4", + name: "MACRO_WRITE_BLANK_ERROR", + desc: ''' + This error is returned if a programming operation attempted to clear a bit that has previously been programmed to 1. + The corresponding controller automatically recovers from this error when issuing a new command. + + Note however that the affected OTP word may be left in an inconsistent state if this error occurs. + This can cause several issues when the word is accessed again (either as part of a regular read operation, as part of the readout at boot, or as part of a background check). + + It is important that SW ensures that each word is only written once, since this can render the device useless. + ''' + }, + { value: "5", + name: "ACCESS_ERROR", + desc: ''' + This error indicates that a locked memory region has been accessed. + The corresponding controller automatically recovers from this error when issuing a new command. + ''' + }, + { value: "6", + name: "CHECK_FAIL_ERROR", + desc: ''' + An ECC, integrity or consistency mismatch has been detected in the buffer registers. + This error should never occur during normal operation and is not recoverable. + This error triggers an fatal_check_error alert. + ''' + }, + { value: "7", + name: "FSM_STATE_ERROR", + desc: ''' + The FSM of the corresponding controller has reached an invalid state, or the FSM has + been moved into a terminal error state due to an escalation action via lc_escalate_en_i. + This error should never occur during normal operation and is not recoverable. + If this error is present, this is a sign that the device has fallen victim to + an invasive attack. This error triggers an fatal_check_error alert. + ''' + }, + ] + } + ] + } + } + { name: "DIRECT_ACCESS_REGWEN", + desc: ''' + Register write enable for all direct access interface registers. + ''', + swaccess: "rw0c", + hwaccess: "hrw", + hwext: "true", + hwqe: "true", + tags: [ // OTP internal HW will set this enable register to 0 when OTP is not under IDLE + // state, so could not auto-predict its value + "excl:CsrNonInitTests:CsrExclCheck"], + fields: [ + { + bits: "0", + desc: ''' + This bit controls whether the DAI registers can be written. + Write 0 to it in order to clear the bit. + + Note that the hardware also modulates this bit and sets it to 0 temporarily + during an OTP operation such that the corresponding address and data registers + cannot be modified while an operation is pending. The !!DAI_IDLE status bit + will also be set to 0 in such a case. + ''' + resval: 1, + }, + ] + }, + { name: "DIRECT_ACCESS_CMD", + desc: "Command register for direct accesses.", + swaccess: "r0w1c", + hwaccess: "hro", + hwqe: "true", + hwext: "true", + resval: 0, + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // Write to DIRECT_ACCESS_CMD randomly might cause OTP_ERRORs and illegal sequences + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + name: "RD", + desc: ''' + Initiates a readout sequence that reads the location specified + by !!DIRECT_ACCESS_ADDRESS. The command places the data read into + !!DIRECT_ACCESS_RDATA_0 and !!DIRECT_ACCESS_RDATA_1 (for 64bit partitions). + ''' + } + { bits: "1", + name: "WR", + desc: ''' + Initiates a programming sequence that writes the data in !!DIRECT_ACCESS_WDATA_0 + and !!DIRECT_ACCESS_WDATA_1 (for 64bit partitions) to the location specified by + !!DIRECT_ACCESS_ADDRESS. + ''' + } + { bits: "2", + name: "DIGEST", + desc: ''' + Initiates the digest calculation and locking sequence for the partition specified by + !!DIRECT_ACCESS_ADDRESS. + ''' + } + ] + } + { name: "DIRECT_ACCESS_ADDRESS", + desc: "Address register for direct accesses.", + swaccess: "rw", + hwaccess: "hro", + hwqe: "false", + resval: 0, + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The enable register "DIRECT_ACCESS_REGWEN" is HW controlled, + // so not able to predict this register value automatically + "excl:CsrNonInitTests:CsrExclCheck"], + fields: [ + { bits: "OtpByteAddrWidth-1:0", + desc: ''' + This is the address for the OTP word to be read or written through + the direct access interface. Note that the address is aligned to the access size + internally, hence bits 1:0 are ignored for 32bit accesses, and bits 2:0 are ignored + for 64bit accesses. + + For the digest calculation command, set this register to the partition base offset. + ''' + } + ] + } + { multireg: { + name: "DIRECT_ACCESS_WDATA", + desc: '''Write data for direct accesses. + Hardware automatically determines the access granule (32bit or 64bit) based on which + partition is being written to. + ''', + count: "NumDaiWords", // 2 x 32bit = 64bit + swaccess: "rw", + hwaccess: "hro", + hwqe: "false", + regwen: "DIRECT_ACCESS_REGWEN", + cname: "WORD", + resval: 0, + tags: [ // The value of this register is written from "DIRECT_ACCESS_RDATA", + // so could not predict this register value automatically + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "DIRECT_ACCESS_RDATA", + desc: '''Read data for direct accesses. + Hardware automatically determines the access granule (32bit or 64bit) based on which + partition is read from. + ''', + count: "NumDaiWords", // 2 x 32bit = 64bit + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + fields: [ + { bits: "31:0" + } + ] + } + }, + + ////////////////////////////////////// + // Integrity and Consistency Checks // + ////////////////////////////////////// + { name: "CHECK_TRIGGER_REGWEN", + desc: ''' + Register write enable for !!CHECK_TRIGGER. + ''', + swaccess: "rw0c", + hwaccess: "none", + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, the !!CHECK_TRIGGER register cannot be written anymore. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "CHECK_TRIGGER", + desc: "Command register for direct accesses.", + swaccess: "r0w1c", + hwaccess: "hro", + hwqe: "true", + hwext: "true", + resval: 0, + regwen: "CHECK_TRIGGER_REGWEN", + fields: [ + { bits: "0", + name: "INTEGRITY", + desc: ''' + Writing 1 to this bit triggers an integrity check. SW should monitor !!STATUS.CHECK_PENDING + and wait until the check has been completed. If there are any errors, those will be flagged + in the !!STATUS and !!ERR_CODE registers, and via the interrupts and alerts. + ''' + } + { bits: "1", + name: "CONSISTENCY", + desc: ''' + Writing 1 to this bit triggers a consistency check. SW should monitor !!STATUS.CHECK_PENDING + and wait until the check has been completed. If there are any errors, those will be flagged + in the !!STATUS and !!ERR_CODE registers, and via interrupts and alerts. + ''' + } + ] + }, + { name: "CHECK_REGWEN", + desc: ''' + Register write enable for !!INTEGRITY_CHECK_PERIOD and !!CONSISTENCY_CHECK_PERIOD. + ''', + swaccess: "rw0c", + hwaccess: "none", + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, !!INTEGRITY_CHECK_PERIOD and !!CONSISTENCY_CHECK_PERIOD registers cannot be written anymore. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "CHECK_TIMEOUT", + desc: ''' + Timeout value for the integrity and consistency checks. + ''', + swaccess: "rw", + hwaccess: "hro", + regwen: "CHECK_REGWEN", + tags: [ // Do not write to this automatically, as it may trigger fatal alert, and cause + // escalation. + "excl:CsrAllTests:CsrExclWrite"], + fields: [ + { bits: "31:0", + desc: ''' + Timeout value in cycles for the for the integrity and consistency checks. If an integrity or consistency + check does not complete within the timeout window, an error will be flagged in the !!STATUS register, + an otp_error interrupt will be raised, and an fatal_check_error alert will be sent out. The timeout should + be set to a large value to stay on the safe side. The maximum check time can be upper bounded by the + number of cycles it takes to readout, scramble and digest the entire OTP array. Since this amounts to + roughly 25k cycles, it is recommended to set this value to at least 100'000 cycles in order to stay on the + safe side. A value of zero disables the timeout mechanism (default). + ''' + resval: 0, + }, + ] + }, + { name: "INTEGRITY_CHECK_PERIOD", + desc: ''' + This value specifies the maximum period that can be generated pseudo-randomly. + Only applies to the HW_CFG* and SECRET* partitions once they are locked. + ''' + swaccess: "rw", + hwaccess: "hro", + regwen: "CHECK_REGWEN", + fields: [ + { bits: "31:0", + desc: ''' + The pseudo-random period is generated using a 40bit LFSR internally, and this register defines + the bit mask to be applied to the LFSR output in order to limit its range. The value of this + register is left shifted by 8bits and the lower bits are set to 8'hFF in order to form the 40bit mask. + A recommended value is 0x3_FFFF, corresponding to a maximum period of ~2.8s at 24MHz. + A value of zero disables the timer (default). Note that a one-off check can always be triggered via + !!CHECK_TRIGGER.INTEGRITY. + ''' + resval: "0" + } + ] + } + { name: "CONSISTENCY_CHECK_PERIOD", + desc: ''' + This value specifies the maximum period that can be generated pseudo-randomly. + This applies to the LIFE_CYCLE partition and the HW_CFG* and SECRET* partitions once they are locked. + ''' + swaccess: "rw", + hwaccess: "hro", + regwen: "CHECK_REGWEN", + fields: [ + { bits: "31:0", + desc: ''' + The pseudo-random period is generated using a 40bit LFSR internally, and this register defines + the bit mask to be applied to the LFSR output in order to limit its range. The value of this + register is left shifted by 8bits and the lower bits are set to 8'hFF in order to form the 40bit mask. + A recommended value is 0x3FF_FFFF, corresponding to a maximum period of ~716s at 24MHz. + A value of zero disables the timer (default). Note that a one-off check can always be triggered via + !!CHECK_TRIGGER.CONSISTENCY. + ''' + resval: "0" + } + ] + } + + //////////////////////////////////// + // Dynamic Locks of SW Parititons // + //////////////////////////////////// + { name: "VENDOR_TEST_READ_LOCK", + desc: ''' + Runtime read lock for the VENDOR_TEST partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the VENDOR_TEST partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "CREATOR_SW_CFG_READ_LOCK", + desc: ''' + Runtime read lock for the CREATOR_SW_CFG partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the CREATOR_SW_CFG partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "OWNER_SW_CFG_READ_LOCK", + desc: ''' + Runtime read lock for the OWNER_SW_CFG partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the OWNER_SW_CFG partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "OWNERSHIP_SLOT_STATE_READ_LOCK", + desc: ''' + Runtime read lock for the OWNERSHIP_SLOT_STATE partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the OWNERSHIP_SLOT_STATE partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "ROT_CREATOR_AUTH_READ_LOCK", + desc: ''' + Runtime read lock for the ROT_CREATOR_AUTH partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the ROT_CREATOR_AUTH partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "ROT_OWNER_AUTH_SLOT0_READ_LOCK", + desc: ''' + Runtime read lock for the ROT_OWNER_AUTH_SLOT0 partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the ROT_OWNER_AUTH_SLOT0 partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "ROT_OWNER_AUTH_SLOT1_READ_LOCK", + desc: ''' + Runtime read lock for the ROT_OWNER_AUTH_SLOT1 partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the ROT_OWNER_AUTH_SLOT1 partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "PLAT_INTEG_AUTH_SLOT0_READ_LOCK", + desc: ''' + Runtime read lock for the PLAT_INTEG_AUTH_SLOT0 partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the PLAT_INTEG_AUTH_SLOT0 partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "PLAT_INTEG_AUTH_SLOT1_READ_LOCK", + desc: ''' + Runtime read lock for the PLAT_INTEG_AUTH_SLOT1 partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the PLAT_INTEG_AUTH_SLOT1 partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "PLAT_OWNER_AUTH_SLOT0_READ_LOCK", + desc: ''' + Runtime read lock for the PLAT_OWNER_AUTH_SLOT0 partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the PLAT_OWNER_AUTH_SLOT0 partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "PLAT_OWNER_AUTH_SLOT1_READ_LOCK", + desc: ''' + Runtime read lock for the PLAT_OWNER_AUTH_SLOT1 partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the PLAT_OWNER_AUTH_SLOT1 partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "PLAT_OWNER_AUTH_SLOT2_READ_LOCK", + desc: ''' + Runtime read lock for the PLAT_OWNER_AUTH_SLOT2 partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the PLAT_OWNER_AUTH_SLOT2 partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "PLAT_OWNER_AUTH_SLOT3_READ_LOCK", + desc: ''' + Runtime read lock for the PLAT_OWNER_AUTH_SLOT3 partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the PLAT_OWNER_AUTH_SLOT3 partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "EXT_NVM_READ_LOCK", + desc: ''' + Runtime read lock for the EXT_NVM partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the EXT_NVM partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + { name: "ROM_PATCH_READ_LOCK", + desc: ''' + Runtime read lock for the ROM_PATCH partition. + ''', + swaccess: "rw0c", + hwaccess: "hro", + regwen: "DIRECT_ACCESS_REGWEN", + tags: [ // The value of this register can affect the read access of the this + // partition's memory window. Excluding this register from writing can ensure + // memories have read and write access. + "excl:CsrNonInitTests:CsrExclWrite"], + fields: [ + { bits: "0", + desc: ''' + When cleared to 0, read access to the ROM_PATCH partition is locked. + Write 0 to clear this bit. + ''' + resval: 1, + }, + ] + }, + + /////////////////////// + // Integrity Digests // + /////////////////////// + { multireg: { + name: "VENDOR_TEST_DIGEST", + desc: ''' + Integrity digest for the VENDOR_TEST partition. + The integrity digest is 0 by default. Software must write this + digest value via the direct access interface in order to lock the partition. + After a reset, write access to the VENDOR_TEST partition is locked and + the digest becomes visible in this CSR. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "CREATOR_SW_CFG_DIGEST", + desc: ''' + Integrity digest for the CREATOR_SW_CFG partition. + The integrity digest is 0 by default. Software must write this + digest value via the direct access interface in order to lock the partition. + After a reset, write access to the CREATOR_SW_CFG partition is locked and + the digest becomes visible in this CSR. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "OWNER_SW_CFG_DIGEST", + desc: ''' + Integrity digest for the OWNER_SW_CFG partition. + The integrity digest is 0 by default. Software must write this + digest value via the direct access interface in order to lock the partition. + After a reset, write access to the OWNER_SW_CFG partition is locked and + the digest becomes visible in this CSR. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "ROT_CREATOR_AUTH_DIGEST", + desc: ''' + Integrity digest for the ROT_CREATOR_AUTH partition. + The integrity digest is 0 by default. Software must write this + digest value via the direct access interface in order to lock the partition. + After a reset, write access to the ROT_CREATOR_AUTH partition is locked and + the digest becomes visible in this CSR. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "ROT_OWNER_AUTH_SLOT0_DIGEST", + desc: ''' + Integrity digest for the ROT_OWNER_AUTH_SLOT0 partition. + The integrity digest is 0 by default. Software must write this + digest value via the direct access interface in order to lock the partition. + After a reset, write access to the ROT_OWNER_AUTH_SLOT0 partition is locked and + the digest becomes visible in this CSR. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "ROT_OWNER_AUTH_SLOT1_DIGEST", + desc: ''' + Integrity digest for the ROT_OWNER_AUTH_SLOT1 partition. + The integrity digest is 0 by default. Software must write this + digest value via the direct access interface in order to lock the partition. + After a reset, write access to the ROT_OWNER_AUTH_SLOT1 partition is locked and + the digest becomes visible in this CSR. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "PLAT_INTEG_AUTH_SLOT0_DIGEST", + desc: ''' + Integrity digest for the PLAT_INTEG_AUTH_SLOT0 partition. + The integrity digest is 0 by default. Software must write this + digest value via the direct access interface in order to lock the partition. + After a reset, write access to the PLAT_INTEG_AUTH_SLOT0 partition is locked and + the digest becomes visible in this CSR. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "PLAT_INTEG_AUTH_SLOT1_DIGEST", + desc: ''' + Integrity digest for the PLAT_INTEG_AUTH_SLOT1 partition. + The integrity digest is 0 by default. Software must write this + digest value via the direct access interface in order to lock the partition. + After a reset, write access to the PLAT_INTEG_AUTH_SLOT1 partition is locked and + the digest becomes visible in this CSR. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "PLAT_OWNER_AUTH_SLOT0_DIGEST", + desc: ''' + Integrity digest for the PLAT_OWNER_AUTH_SLOT0 partition. + The integrity digest is 0 by default. Software must write this + digest value via the direct access interface in order to lock the partition. + After a reset, write access to the PLAT_OWNER_AUTH_SLOT0 partition is locked and + the digest becomes visible in this CSR. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "PLAT_OWNER_AUTH_SLOT1_DIGEST", + desc: ''' + Integrity digest for the PLAT_OWNER_AUTH_SLOT1 partition. + The integrity digest is 0 by default. Software must write this + digest value via the direct access interface in order to lock the partition. + After a reset, write access to the PLAT_OWNER_AUTH_SLOT1 partition is locked and + the digest becomes visible in this CSR. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "PLAT_OWNER_AUTH_SLOT2_DIGEST", + desc: ''' + Integrity digest for the PLAT_OWNER_AUTH_SLOT2 partition. + The integrity digest is 0 by default. Software must write this + digest value via the direct access interface in order to lock the partition. + After a reset, write access to the PLAT_OWNER_AUTH_SLOT2 partition is locked and + the digest becomes visible in this CSR. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "PLAT_OWNER_AUTH_SLOT3_DIGEST", + desc: ''' + Integrity digest for the PLAT_OWNER_AUTH_SLOT3 partition. + The integrity digest is 0 by default. Software must write this + digest value via the direct access interface in order to lock the partition. + After a reset, write access to the PLAT_OWNER_AUTH_SLOT3 partition is locked and + the digest becomes visible in this CSR. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "ROM_PATCH_DIGEST", + desc: ''' + Integrity digest for the ROM_PATCH partition. + The integrity digest is 0 by default. Software must write this + digest value via the direct access interface in order to lock the partition. + After a reset, write access to the ROM_PATCH partition is locked and + the digest becomes visible in this CSR. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "HW_CFG0_DIGEST", + desc: ''' + Integrity digest for the HW_CFG0 partition. + The integrity digest is 0 by default. The digest calculation can be triggered via the !!DIRECT_ACCESS_CMD. + After a reset, the digest then becomes visible in this CSR, and the corresponding partition becomes write-locked. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "HW_CFG1_DIGEST", + desc: ''' + Integrity digest for the HW_CFG1 partition. + The integrity digest is 0 by default. The digest calculation can be triggered via the !!DIRECT_ACCESS_CMD. + After a reset, the digest then becomes visible in this CSR, and the corresponding partition becomes write-locked. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "SECRET0_DIGEST", + desc: ''' + Integrity digest for the SECRET0 partition. + The integrity digest is 0 by default. The digest calculation can be triggered via the !!DIRECT_ACCESS_CMD. + After a reset, the digest then becomes visible in this CSR, and the corresponding partition becomes write-locked. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "SECRET1_DIGEST", + desc: ''' + Integrity digest for the SECRET1 partition. + The integrity digest is 0 by default. The digest calculation can be triggered via the !!DIRECT_ACCESS_CMD. + After a reset, the digest then becomes visible in this CSR, and the corresponding partition becomes write-locked. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "SECRET2_DIGEST", + desc: ''' + Integrity digest for the SECRET2 partition. + The integrity digest is 0 by default. The digest calculation can be triggered via the !!DIRECT_ACCESS_CMD. + After a reset, the digest then becomes visible in this CSR, and the corresponding partition becomes write-locked. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + { multireg: { + name: "SECRET3_DIGEST", + desc: ''' + Integrity digest for the SECRET3 partition. + The integrity digest is 0 by default. The digest calculation can be triggered via the !!DIRECT_ACCESS_CMD. + After a reset, the digest then becomes visible in this CSR, and the corresponding partition becomes write-locked. + ''', + count: "NumDigestWords", + swaccess: "ro", + hwaccess: "hwo", + hwext: "true", + cname: "WORD", + resval: 0, + tags: [ // OTP internal HW will update status so can not auto-predict its value. + "excl:CsrAllTests:CsrExclCheck"], + fields: [ + { bits: "31:0" + } + ] + } + }, + + //////////////////////////////// + // Software Config Partitions // + //////////////////////////////// + { skipto: "0x4000" } + + { window: { + name: "SW_CFG_WINDOW" + items: "NumSwCfgWindowWords" + swaccess: "ro", + desc: ''' + Any read to this window directly maps to the corresponding offset in the creator and owner software + config partitions, and triggers an OTP readout of the bytes requested. Note that the transaction + will block until OTP readout has completed. + ''' + } + } + ], + + // OTP wrapper-specific registers + prim: [ + { name: "CSR0", + desc: "" + swaccess: "rw", + hwaccess: "hro", + hwext: "false", + hwqe: "false", + fields: [ + { bits: "0", + name: "field0", + desc: "", + resval: "0x0", + } + { bits: "1", + name: "field1", + desc: "", + resval: "0x0", + } + { bits: "2", + name: "field2", + desc: "", + resval: "0x0", + } + { bits: "13:4", + name: "field3", + desc: "" + resval: "0x0", + } + { bits: "26:16", + name: "field4", + desc: "" + resval: "0x0", + } + ] + }, + { name: "CSR1", + desc: "" + swaccess: "rw", + hwaccess: "hro", + hwext: "false", + hwqe: "false", + fields: [ + { bits: "6:0", + name: "field0", + desc: "" + resval: "0x0", + } + { bits: "7:7", + name: "field1", + desc: "", + resval: "0x0", + } + { bits: "14:8", + name: "field2", + desc: "" + resval: "0x0", + } + { bits: "15:15", + name: "field3", + desc: "", + resval: "0x0", + } + { bits: "31:16", + name: "field4", + desc: "", + resval: "0x0", + } + ] + }, + { name: "CSR2", + desc: "" + swaccess: "rw", + hwaccess: "hro", + hwext: "false", + hwqe: "false", + fields: [ + { bits: "0", + name: "field0", + desc: "", + resval: "0x0", + } + ] + }, + { name: "CSR3", + desc: "" + swaccess: "rw", + hwaccess: "hrw", + hwext: "false", + hwqe: "false", + fields: [ + { bits: "2:0", + name: "field0", + desc: "" + swaccess: "rw1c", + resval: "0x0", + } + { bits: "13:4", + name: "field1", + desc: "", + swaccess: "rw1c", + resval: "0x0", + } + { bits: "16", + name: "field2", + desc: "", + swaccess: "rw1c", + resval: "0x0", + } + { bits: "17", + name: "field3", + desc: "", + swaccess: "ro", + resval: "0x0", + } + { bits: "18", + name: "field4", + desc: "", + swaccess: "ro", + resval: "0x0", + } + { bits: "19", + name: "field5", + desc: "", + swaccess: "ro", + resval: "0x0", + } + { bits: "20", + name: "field6", + desc: "", + swaccess: "ro", + resval: "0x0", + } + { bits: "21", + name: "field7", + desc: "", + swaccess: "ro", + resval: "0x0", + } + { bits: "22", + name: "field8", + desc: "", + swaccess: "ro", + resval: "0x0", + } + ] + }, + { name: "CSR4", + desc: "" + swaccess: "rw", + hwaccess: "hro", + hwext: "false", + hwqe: "false", + fields: [ + { bits: "9:0", + name: "field0", + desc: "" + resval: "0x0", + } + { bits: "12", + name: "field1", + desc: "" + resval: "0x0", + } + { bits: "13", + name: "field2", + desc: "" + resval: "0x0", + } + { bits: "14", + name: "field3", + desc: "" + resval: "0x0", + } + ] + }, + { name: "CSR5", + desc: "" + swaccess: "rw", + hwaccess: "hrw", + hwext: "false", + hwqe: "false", + fields: [ + { bits: "5:0", + name: "field0", + desc: "" + swaccess: "rw", + resval: "0x0", + } + { bits: "7:6", + name: "field1", + desc: "" + swaccess: "rw", + resval: "0x0", + } + { bits: "8", + name: "field2", + desc: "", + swaccess: "ro", + resval: "0x0", + } + { bits: "11:9", + name: "field3", + desc: "" + swaccess: "ro", + resval: "0x0", + } + { bits: "12", + name: "field4", + desc: "" + swaccess: "ro", + resval: "0x0", + } + { bits: "13", + name: "field5", + desc: "" + swaccess: "ro", + resval: "0x0", + } + { bits: "31:16", + name: "field6", + desc: "" + swaccess: "rw", + resval: "0x0", + } + ] + }, + { name: "CSR6", + desc: "" + swaccess: "rw", + hwaccess: "hro", + hwext: "false", + hwqe: "false", + fields: [ + { bits: "9:0", + name: "field0", + desc: "" + resval: "0x0", + } + { bits: "11", + name: "field1", + desc: "", + swaccess: "rw", + resval: "0x0", + } + { bits: "12", + name: "field2", + desc: "", + swaccess: "rw", + resval: "0x0", + } + { bits: "31:16", + name: "field3", + desc: "" + resval: "0x0", + } + ] + }, + { name: "CSR7", + desc: "", + swaccess: "ro", + hwaccess: "hrw", + hwext: "false", + hwqe: "false", + fields: [ + { bits: "5:0", + name: "field0", + desc: "" + swaccess: "ro", + resval: "0x0", + } + { bits: "10:8", + name: "field1", + desc: "", + swaccess: "ro", + resval: "0x0", + } + { bits: "14", + name: "field2", + desc: "", + swaccess: "ro", + resval: "0x0", + } + { bits: "15", + name: "field3", + desc: "", + swaccess: "ro", + resval: "0x0", + } + ] + }, + ] + } +} diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/data/otp_ctrl_sec_cm_testplan.hjson b/hw/top_darjeeling/ip_autogen/otp_ctrl/data/otp_ctrl_sec_cm_testplan.hjson new file mode 100644 index 00000000000000..650a36067a49ca --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/data/otp_ctrl_sec_cm_testplan.hjson @@ -0,0 +1,303 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// Security countermeasures testplan extracted from the IP Hjson using reggen. +// +// This testplan is auto-generated only the first time it is created. This is +// because this testplan needs to be hand-editable. It is possible that these +// testpoints can go out of date if the spec is updated with new +// countermeasures. When `reggen` is invoked when this testplan already exists, +// It checks if the list of testpoints is up-to-date and enforces the user to +// make further manual updates. +// +// These countermeasures and their descriptions can be found here: +// .../otp_ctrl/data/otp_ctrl.hjson +// +// It is possible that the testing of some of these countermeasures may already +// be covered as a testpoint in a different testplan. This duplication is ok - +// the test would have likely already been developed. We simply map those tests +// to the testpoints below using the `tests` key. +// +// Please ensure that this testplan is imported in: +// .../otp_ctrl/data/otp_ctrl_testplan.hjson +{ + testpoints: [ + { + name: sec_cm_bus_integrity + desc: "Verify the countermeasure(s) BUS.INTEGRITY." + stage: V2S + tests: ["otp_ctrl_tl_intg_err"] + } + { + name: sec_cm_secret_mem_scramble + desc: "Verify the countermeasure(s) SECRET.MEM.SCRAMBLE." + stage: V2S + tests: ["otp_ctrl_smoke"] + } + { + name: sec_cm_part_mem_digest + desc: "Verify the countermeasure(s) PART.MEM.DIGEST." + stage: V2S + tests: ["otp_ctrl_smoke"] + } + { + name: sec_cm_dai_fsm_sparse + desc: "Verify the countermeasure(s) DAI.FSM.SPARSE." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_kdi_fsm_sparse + desc: "Verify the countermeasure(s) KDI.FSM.SPARSE." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_lci_fsm_sparse + desc: "Verify the countermeasure(s) LCI.FSM.SPARSE." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_part_fsm_sparse + desc: "Verify the countermeasure(s) PART.FSM.SPARSE." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_scrmbl_fsm_sparse + desc: "Verify the countermeasure(s) SCRMBL.FSM.SPARSE." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_timer_fsm_sparse + desc: "Verify the countermeasure(s) TIMER.FSM.SPARSE." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_dai_ctr_redun + desc: "Verify the countermeasure(s) DAI.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_kdi_seed_ctr_redun + desc: "Verify the countermeasure(s) KDI_SEED.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_kdi_entropy_ctr_redun + desc: "Verify the countermeasure(s) KDI_ENTROPY.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_lci_ctr_redun + desc: "Verify the countermeasure(s) LCI.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_part_ctr_redun + desc: "Verify the countermeasure(s) PART.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_scrmbl_ctr_redun + desc: "Verify the countermeasure(s) SCRMBL.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_timer_integ_ctr_redun + desc: "Verify the countermeasure(s) TIMER_INTEG.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_timer_cnsty_ctr_redun + desc: "Verify the countermeasure(s) TIMER_CNSTY.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_timer_lfsr_redun + desc: "Verify the countermeasure(s) TIMER.LFSR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_dai_fsm_local_esc + desc: "Verify the countermeasure(s) DAI.FSM.LOCAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc", "otp_ctrl_sec_cm"] + } + { + name: sec_cm_lci_fsm_local_esc + desc: "Verify the countermeasure(s) LCI.FSM.LOCAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc"] + } + { + name: sec_cm_kdi_fsm_local_esc + desc: "Verify the countermeasure(s) KDI.FSM.LOCAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc"] + } + { + name: sec_cm_part_fsm_local_esc + desc: "Verify the countermeasure(s) PART.FSM.LOCAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc", "otp_ctrl_macro_errs"] + } + { + name: sec_cm_scrmbl_fsm_local_esc + desc: "Verify the countermeasure(s) SCRMBL.FSM.LOCAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc"] + } + { + name: sec_cm_timer_fsm_local_esc + desc: "Verify the countermeasure(s) TIMER.FSM.LOCAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc", "otp_ctrl_sec_cm"] + } + { + name: sec_cm_dai_fsm_global_esc + desc: "Verify the countermeasure(s) DAI.FSM.GLOBAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc", "otp_ctrl_sec_cm"] + } + { + name: sec_cm_lci_fsm_global_esc + desc: "Verify the countermeasure(s) LCI.FSM.GLOBAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc"] + } + { + name: sec_cm_kdi_fsm_global_esc + desc: "Verify the countermeasure(s) KDI.FSM.GLOBAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc"] + } + { + name: sec_cm_part_fsm_global_esc + desc: "Verify the countermeasure(s) PART.FSM.GLOBAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc", "otp_ctrl_macro_errs"] + } + { + name: sec_cm_scrmbl_fsm_global_esc + desc: "Verify the countermeasure(s) SCRMBL.FSM.GLOBAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc"] + } + { + name: sec_cm_timer_fsm_global_esc + desc: "Verify the countermeasure(s) TIMER.FSM.GLOBAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc", "otp_ctrl_sec_cm"] + } + { + name: sec_cm_part_data_reg_integrity + desc: "Verify the countermeasure(s) PART.DATA_REG.INTEGRITY." + stage: V2S + tests: ["otp_ctrl_init_fail"] + } + { + name: sec_cm_part_data_reg_bkgn_chk + desc: "Verify the countermeasure(s) PART.DATA_REG.BKGN_CHK." + stage: V2S + tests: ["otp_ctrl_check_fail"] + } + { + name: sec_cm_part_mem_regren + desc: "Verify the countermeasure(s) PART.MEM.REGREN." + stage: V2S + tests: ["otp_ctrl_dai_lock"] + } + { + name: sec_cm_part_mem_sw_unreadable + desc: "Verify the countermeasure(s) PART.MEM.SW_UNREADABLE." + stage: V2S + tests: ["otp_ctrl_dai_lock"] + } + { + name: sec_cm_part_mem_sw_unwritable + desc: "Verify the countermeasure(s) PART.MEM.SW_UNWRITABLE." + stage: V2S + tests: ["otp_ctrl_dai_lock"] + } + { + name: sec_cm_lc_part_mem_sw_noaccess + desc: "Verify the countermeasure(s) LC_PART.MEM.SW_NOACCESS." + stage: V2S + tests: ["otp_ctrl_dai_lock"] + } + { + name: sec_cm_access_ctrl_mubi + desc: "Verify the countermeasure(s) ACCESS.CTRL.MUBI." + stage: V2S + tests: ["otp_ctrl_dai_lock"] + } + { + name: sec_cm_token_valid_ctrl_mubi + desc: "Verify the countermeasure(s) TOKEN_VALID.CTRL.MUBI." + stage: V2S + tests: ["otp_ctrl_smoke"] + } + { + name: sec_cm_lc_ctrl_intersig_mubi + desc: "Verify the countermeasure(s) LC_CTRL.INTERSIG.MUBI." + stage: V2S + tests: ["otp_ctrl_dai_lock"] + } + { + name: sec_cm_test_bus_lc_gated + desc: "Verify the countermeasure(s) TEST.BUS.LC_GATED." + stage: V2S + tests: ["otp_ctrl_smoke"] + } + { + name: sec_cm_test_tl_lc_gate_fsm_sparse + desc: "Verify the countermeasure(s) TEST_TL_LC_GATE.FSM.SPARSE." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_direct_access_config_regwen + desc: "Verify the countermeasure(s) DIRECT_ACCESS.CONFIG.REGWEN." + stage: V2S + tests: ["otp_ctrl_regwen"] + } + { + name: sec_cm_check_trigger_config_regwen + desc: "Verify the countermeasure(s) CHECK_TRIGGER.CONFIG.REGWEN." + stage: V2S + tests: ["otp_ctrl_smoke"] + } + { + name: sec_cm_check_config_regwen + desc: "Verify the countermeasure(s) CHECK.CONFIG.REGWEN." + stage: V2S + tests: ["otp_ctrl_smoke"] + } + { + name: sec_cm_macro_mem_integrity + desc: "Verify the countermeasure(s) MACRO.MEM.INTEGRITY." + stage: V2S + tests: ["otp_ctrl_macro_errs"] + } + { + name: sec_cm_macro_mem_cm + desc: "Verify the countermeasure(s) MACRO.MEM.CM." + stage: V2S + tests: ["N/A"] + } + ] +} diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/data/otp_ctrl_testplan.hjson b/hw/top_darjeeling/ip_autogen/otp_ctrl/data/otp_ctrl_testplan.hjson new file mode 100644 index 00000000000000..95b6081b7da897 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/data/otp_ctrl_testplan.hjson @@ -0,0 +1,404 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +{ + name: "otp_ctrl" + import_testplans: ["hw/dv/tools/dvsim/testplans/csr_testplan.hjson", + "hw/dv/tools/dvsim/testplans/mem_testplan.hjson", + "hw/dv/tools/dvsim/testplans/intr_test_testplan.hjson", + "hw/dv/tools/dvsim/testplans/alert_test_testplan.hjson", + "hw/dv/tools/dvsim/testplans/tl_device_access_types_testplan.hjson", + "hw/dv/tools/dvsim/testplans/sec_cm_count_testplan.hjson", + "hw/dv/tools/dvsim/testplans/sec_cm_fsm_testplan.hjson", + "hw/dv/tools/dvsim/testplans/stress_all_with_reset_testplan.hjson", + "otp_ctrl_sec_cm_testplan.hjson"] + testpoints: [ + { + name: wake_up + desc: ''' + Wake_up test walks through otp_ctrl's power-on initialization, read, program, and + digest functionalities. + + - drive pwrmgr's request pin to trigger OTP initialization after reset, check status + after OTP initialization + - write all-ones to a random address within OTP partition 0, wait until this operation + completes + - read out the random selected write address, check if the readout value is all-ones + - trigger a digest calculation for a Software partition, check if the OtpError + interrupt is set + - trigger a digest calculation for a non-software partition, expect operation completes + without the OtpError interrupt + - read out secrets through the hardware interfaces + ''' + stage: V1 + tests: ["otp_ctrl_wake_up"] + } + { + name: smoke + desc: ''' + OTP_CTRL smoke test provisions and locks partitions. + + - drive pwrmgr's request pin to trigger OTP initialization after reset, check status + after OTP initialization + - randomly read out keys pertaining to `key_manager`, `flash`, `sram`, `otbn` + - randomly issue LC program request + - write random values to random addresses within each OTP partition + - read out the random selected write addresses, check if the readout values are expected + - during read and write operations, check if direct_access_regwen is correctly set by HW + - perform a system-level reset and check corresponding CSRs are set correctly + - lock all partitions except life_cycle by triggering digest calculations + - read back and verify the digest + - perform a system-level reset to verify the corresponding CSRs exposing the digests + have been populated + + **Checks**: + - Assertion checks to ensure vendor specific I/Os: `otp_vendor_test_status_o`, + `otp_vendor_test_ctrl_i`, `cio_test_o`, and `cio_test_en_o` are connected currently + with `lc_dft_en_i` On and Off. + ''' + stage: V1 + tests: ["otp_ctrl_smoke"] + } + { + name: dai_access_partition_walk + desc: ''' + Similar to UVM's memory walk test, this test ensures every address in each partition + can be accessed successfully via DAI and TLUL interfacs according to its access policy. + ''' + stage: V2 + tests: ["otp_ctrl_partition_walk"] + } + { + name: init_fail + desc: ''' + Based on OTP_CTRL smoke test, this test creates OTP_CTRL's initialization failure: + - write and read OTP memory via DAI interface + - randomly issue DAI digest command to lock HW partitions + - keep writing to OTP memory via DAI interface without asserting reset + - if digests are not locked, backdoor inject ECC correctable or uncorrectable errors + - issue reset and power initialization + - if the injected errors are all correctable errors, disable the `lc_bypass_chk_en` + after LC program request to create an LC partition check failure + + If fatal error is triggered, this test will check: + - OTP initialization failure triggers fatal alert + - `status`, `intr_state`, `err_code` CSRs reflect correct fatal error + + If OTP initialization finished without any fatal error, this test will check: + - OTP initialization finishes with power init output goes to 1 + - `status`, `intr_state`, `err_code` CSRs reflect ECC correctable error + ''' + stage: V2 + tests: ["otp_ctrl_init_fail"] + } + { + name: partition_check + desc: ''' + Randomly program the partition check related CSRs including: + - `check_timeout` + - `integrity_check_period` + - `consistency_check_period` + - `check_trigger` + + Create a failure scenario by randomly picking one of these three methods: + - inject ECC errors into the OTP macro via backdoor + - set the `check_timeout` CSR with a very small value + - write to a random OTP partition after digest is issued but before reset is asserted + + **Checks**: + - the corresponding alerts are triggered + - the error_code register is set correctly + Note that due to limited simulation time, for background checks, this test only write + random value that is less than 20 to the check period. + ''' + stage: V2 + tests: ["otp_ctrl_check_fail", "otp_ctrl_background_chks"] + } + { + name: regwen_during_otp_init + desc: ''' + The `direct_access_regwen` is a RO register which controls the write-enable of other + reigsters. It is not verified by the common CSR tests. HW sets it to 0 when the DAI + interface is busy. + + Stimulus and checks: + - randomly read `direct_access_regwen` and verify that it returns 0 during OTP + initialization + - verify that the writes to the registers controlled by it do not go through during OTP + initialization + ''' + stage: V2 + tests: ["otp_ctrl_regwen"] + } + { + name: partition_lock + desc: ''' + This test will cover two methods of locking read and write: digest calculation and CSR + write. After locking the partitions, issue read or program sequences and check if the + operations are locked correctly, and check if the `AccessError` is set. + ''' + stage: V2 + tests: ["otp_ctrl_dai_lock"] + } + { + name: interface_key_check + desc: ''' + OTP_CTRL will generate keys to `flash`, `sram`, and `otbn` upon their requests. + Based on the DAI access sequence, this test will run key requests sequence in + parallel, and check if correct keys are generated. + ''' + stage: V2 + tests: ["otp_ctrl_parallel_key_req"] + } + { + name: lc_interactions + desc: ''' + Verify the procotols between OTP_CTRL and LC_CTRL. Based on the DAI access sequence, + run the following sequences in parallel: + + - request a LC state transition via the programming interface + - enable the `lc_escalation_en` signal + + **Checks**: + - if the LC program request has `AccessError`, check the LC program response sets + the `error` bit to 1 + - if `lc_escalation_en` is enabled, verify that alert is triggered and OTP_CTRL entered + terminal state + ''' + stage: V2 + tests: ["otp_ctrl_parallel_lc_req", "otp_ctrl_parallel_lc_esc"] + } + { name: otp_dai_errors + desc: ''' + Based on the otp_dai_lock test, this test will randomly run the following OTP errors: + - DAI interface writes non-blank OTP address + - DAI interface accesses LC partition + - DAI interface writes HW digests + - DAI interface writes non-empty memory + + **Checks**: + - `err_code` and `status` CSRs + - `otp_error` interrupt + ''' + stage: V2 + tests: ["otp_ctrl_dai_errs"] + } + { name: otp_macro_errors + desc: ''' + Randomly run the following OTP errors: + - MacroError + - MacroEccCorrError + - MacroEccUncorrError + + **Checks**: + - `err_code` and `status` CSRs + - `otp_error` interrupt + - if the error is unrecoverable, verify that alert is triggered and OTP_CTRL entered + terminal state + ''' + stage: V2 + tests: ["otp_ctrl_macro_errs"] + } + { + name: test_access + desc: ''' + This test checks if the test access to OTP macro is connected correctly. + + **Stimulus and Checks**: + - Write and check read results from the prim_tl_i/o. + - Ensure no error or alert occurs from DUT. + ''' + stage: V2 + tests: ["otp_ctrl_test_access"] + } + { + name: stress_all + desc: ''' + - combine above sequences in one test to run sequentially, except csr sequence + - randomly add reset between each sequence + ''' + stage: V2 + tests: ["{name}_stress_all"] + } + { + name: sec_cm_additional_check + desc: ''' + Verify the outcome of injecting faults to security countermeasures. + + Stimulus: + As mentioned in `prim_count_check`, `prim_fsm_check` and `prim_double_lfsr_check`. + + Checks: + - Check the value of status register according to where the fault is injected. + - Check OTP_CTRL is locked after the fatal fault injection by trying to access OTP_CTRL + via dai, kdi, and lci interfaces. + ''' + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: otp_ctrl_low_freq_read + desc: ''' + This test checks if OTP's read operation can operate successfully in a low clock + frequency before the clock is calibrated. + + **Stimulus and Checks**: + - Configure OTP_CTRL's clock to 6MHz low frequency. + - Backdoor write OTP memory. + - Use DAI access to read each memory address and compare if the value is correct. + - If DAI address is in a SW partition, read and check again via TLUL interface. + ''' + stage: V3 + tests: ["otp_ctrl_low_freq_read"] + } + ] + + covergroups: [ + { + name: power_on_cg + desc: '''Covers the following conditions when OTP_CTRL finishes power-on initialization: + - whether `lc_escalation_en` is On + - whether any partition (except life cycle partition) is locked + ''' + } + { + name: flash_req_cg + desc: '''Covers whether secret1 partition is locked during `flash` data or address + request.''' + } + { + name: sram_req_cg + desc: '''Covers whether secret1 partition is locked during all `srams` key request.''' + } + { + name: otbn_req_cg + desc: '''Covers whether secret1 partition is locked during `otbn` key request.''' + } + { + name: lc_prog_cg + desc: '''Covers whether the error bit is set during LC program request.''' + } + { + name: keymgr_o_cg + desc: '''Covers the following conditions when scoreboard checks `keymgr_o` value: + - whether secret2 partition is locked + - whether `lc_seed_hw_rd_en_i` is On + ''' + } + { + name: req_dai_access_after_alert_cg + desc: '''Covers if sequence issued various DAI requests after any fatal alert is + triggered.''' + } + { + name: issue_checks_after_alert_cg + desc: '''Covers if sequence issued various OTP_CTRL's background checks after any fatal alert + is triggered.''' + } + { + name: csr_rd_after_alert_cg + desc: '''Covers if the following CSRs are being read and the value is checked in scoreboard + after any fatal alert is triggered: + - unbuffered partitions' digest CSRs + - HW partition's digest CSRs + - secrets partitions' digest CSRs + - direct_access read data CSRs + - status CSR + - error_code CSR + ''' + } + { + name: dai_err_code_cg + desc: '''Covers all applicable error codes in DAI, and cross each error code with all + 7 partitions.''' + } + { + name: lci_err_code_cg + desc: '''Covers all applicable error codes in LCI.''' + } + { + name: unbuf_err_code_cg + desc: '''This is an array of covergroups to cover all applicable error codes in three + unbuffered partitions.''' + } + { + name: buf_err_code_cg + desc: '''This is an array of covergroups to cover all applicable error codes in five + buffered partitions.''' + } + { + name: unbuf_access_lock_cg_wrap_cg + desc: '''This is an array of covergroups to cover lock conditions below in three + unbuffered partitions: + - the partition is write-locked + - the partition is read-locked + - the current operation type + Then cross the three coverpoints.''' + } + { + name: dai_access_secret2_cg + desc: '''Covers whether `lc_creator_seed_sw_rw_en` is On during any DAI accesses.''' + } + { + name: status_csr_cg + desc: '''Covers the value of every bit in `status` CSR.''' + } + // The following covergroups are implemented in `otp_ctrl_cov_if.sv`. + { + name: lc_esc_en_condition_cg + desc: '''Covers the following conditions when `lc_escalation_en` is On: + - whether any key requests is in progress + - whether LC program reqeust is in progress + - whether DAI interface is busy + ''' + } + { + name: flash_data_req_condition_cg + desc: '''Covers the following conditions when `lc_escalation_en` is On: + - whether any key requests is in progress + - whether DAI interface is busy + - whether lc_esc_en is On + ''' + } + { + name: flash_addr_req_condition_cg + desc: '''Covers the following conditions when `lc_escalation_en` is On: + - whether any key requests is in progress + - whether DAI interface is busy + - whether lc_esc_en is On + ''' + } + { + name: sram_0_req_condition_cg + desc: '''Covers the following conditions when `lc_escalation_en` is On: + - whether any key requests is in progress + - whether DAI interface is busy + - whether lc_esc_en is On + ''' + } + { + name: sram_1_req_condition_cg + desc: '''Covers the following conditions when `lc_escalation_en` is On: + - whether any key requests is in progress + - whether DAI interface is busy + - whether lc_esc_en is On + ''' + } + { + name: otbn_req_condition_cg + desc: '''Covers the following conditions when `lc_escalation_en` is On: + - whether any key requests is in progress + - whether DAI interface is busy + - whether lc_esc_en is On + ''' + } + { + name: lc_prog_req_condition_cg + desc: '''Covers the following conditions when `lc_escalation_en` is On: + - whether any key requests is in progress + - whether DAI interface is busy + - whether lc_esc_en is On + ''' + } + ] +} diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/data/top_darjeeling_otp_ctrl.ipconfig.hjson b/hw/top_darjeeling/ip_autogen/otp_ctrl/data/top_darjeeling_otp_ctrl.ipconfig.hjson new file mode 100644 index 00000000000000..ddd0e65a6c7505 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/data/top_darjeeling_otp_ctrl.ipconfig.hjson @@ -0,0 +1,1992 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +{ + instance_name: top_darjeeling_otp_ctrl + param_values: + { + otp_mmap: + { + seed: 36021179872380457113239299468132194022238108125576166239904535336103582949069 + otp: + { + width: 2 + depth: 8192 + size: 16384 + addr_width: 13 + byte_addr_width: 14 + } + scrambling: + { + key_size: 16 + iv_size: 8 + cnst_size: 16 + keys: + [ + { + name: Secret0Key + value: 79261096542507317550041524418378318810 + } + { + name: Secret1Key + value: 318985785134653313354004803705056942690 + } + { + name: Secret2Key + value: 177669530988656006400383198120896388291 + } + { + name: Secret3Key + value: 253454544458117277620971692296657999195 + } + ] + digests: + [ + { + name: CnstyDigest + iv_value: 5573918854495152458 + cnst_value: 298124614738804023684088847145512094502 + } + { + name: FlashDataKey + iv_value: 13142649492737956675 + cnst_value: 15226184123672563655096903138034867314 + } + { + name: FlashAddrKey + iv_value: 10432573278431539239 + cnst_value: 331706166944643955321785563334435677807 + } + { + name: SramDataKey + iv_value: 13755002077647639451 + cnst_value: 132555613982997757926354237115728852983 + } + ] + } + partitions: + [ + { + name: VENDOR_TEST + variant: Unbuffered + absorb: false + size: 64 + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: false + bkout_type: false + items: + [ + { + name: SCRATCH + size: 56 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 0 + } + { + name: VENDOR_TEST_DIGEST + size: 8 + offset: 56 + ismubi: false + isdigest: true + inv_default: 12267607958787918204 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Vendor test partition. + This is reserved for manufacturing smoke checks. The OTP wrapper + control logic inside prim_otp is allowed to read/write to this + region. ECC uncorrectable errors seen on the functional prim_otp + interface will not lead to an alert for this partition. + Instead, such errors will be reported as correctable ECC errors. + ''' + iskeymgr_creator: false + iskeymgr_owner: false + offset: 0 + } + { + name: CREATOR_SW_CFG + variant: Unbuffered + absorb: false + size: 320 + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: CREATOR_SW_CFG_AST_CFG + size: 124 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 64 + } + { + name: CREATOR_SW_CFG_AST_INIT_EN + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 188 + } + { + name: CREATOR_SW_CFG_OVERRIDES + size: 32 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 192 + } + { + name: CREATOR_SW_CFG_ROM_EXT_SKU + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 224 + } + { + name: CREATOR_SW_CFG_SIGVERIFY_RSA_MOD_EXP_IBEX_EN + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 228 + } + { + name: CREATOR_SW_CFG_SIGVERIFY_RSA_KEY_EN + size: 8 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 232 + } + { + name: CREATOR_SW_CFG_SIGVERIFY_SPX_EN + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 240 + } + { + name: CREATOR_SW_CFG_SIGVERIFY_SPX_KEY_EN + size: 8 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 244 + } + { + name: CREATOR_SW_CFG_FLASH_DATA_DEFAULT_CFG + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 252 + } + { + name: CREATOR_SW_CFG_FLASH_INFO_BOOT_DATA_CFG + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 256 + } + { + name: CREATOR_SW_CFG_FLASH_HW_INFO_CFG_OVERRIDE + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 260 + } + { + name: CREATOR_SW_CFG_RNG_EN + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 264 + } + { + name: CREATOR_SW_CFG_JITTER_EN + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 268 + } + { + name: CREATOR_SW_CFG_RET_RAM_RESET_MASK + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 272 + } + { + name: CREATOR_SW_CFG_MANUF_STATE + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 276 + } + { + name: CREATOR_SW_CFG_ROM_EXEC_EN + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 280 + } + { + name: CREATOR_SW_CFG_CPUCTRL + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 284 + } + { + name: CREATOR_SW_CFG_MIN_SEC_VER_ROM_EXT + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 288 + } + { + name: CREATOR_SW_CFG_MIN_SEC_VER_BL0 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 292 + } + { + name: CREATOR_SW_CFG_DEFAULT_BOOT_DATA_IN_PROD_EN + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 296 + } + { + name: CREATOR_SW_CFG_RMA_SPIN_EN + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 300 + } + { + name: CREATOR_SW_CFG_RMA_SPIN_CYCLES + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 304 + } + { + name: CREATOR_SW_CFG_RNG_REPCNT_THRESHOLDS + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 308 + } + { + name: CREATOR_SW_CFG_RNG_REPCNTS_THRESHOLDS + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 312 + } + { + name: CREATOR_SW_CFG_RNG_ADAPTP_HI_THRESHOLDS + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 316 + } + { + name: CREATOR_SW_CFG_RNG_ADAPTP_LO_THRESHOLDS + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 320 + } + { + name: CREATOR_SW_CFG_RNG_BUCKET_THRESHOLDS + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 324 + } + { + name: CREATOR_SW_CFG_RNG_MARKOV_HI_THRESHOLDS + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 328 + } + { + name: CREATOR_SW_CFG_RNG_MARKOV_LO_THRESHOLDS + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 332 + } + { + name: CREATOR_SW_CFG_RNG_EXTHT_HI_THRESHOLDS + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 336 + } + { + name: CREATOR_SW_CFG_RNG_EXTHT_LO_THRESHOLDS + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 340 + } + { + name: CREATOR_SW_CFG_RNG_ALERT_THRESHOLD + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 344 + } + { + name: CREATOR_SW_CFG_RNG_HEALTH_CONFIG_DIGEST + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 348 + } + { + name: CREATOR_SW_CFG_SRAM_KEY_RENEW_EN + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 352 + } + { + name: CREATOR_SW_CFG_DIGEST + size: 8 + offset: 376 + ismubi: false + isdigest: true + inv_default: 7474818370217309020 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Software configuration partition. + This is for device-specific calibration data. For example, clock, + LDO, RNG. + ''' + iskeymgr_creator: false + iskeymgr_owner: false + offset: 64 + } + { + name: OWNER_SW_CFG + variant: Unbuffered + absorb: false + size: 632 + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: OWNER_SW_CFG_ROM_ERROR_REPORTING + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 384 + } + { + name: OWNER_SW_CFG_ROM_BOOTSTRAP_DIS + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 388 + } + { + name: OWNER_SW_CFG_ROM_ALERT_CLASS_EN + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 392 + } + { + name: OWNER_SW_CFG_ROM_ALERT_ESCALATION + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 396 + } + { + name: OWNER_SW_CFG_ROM_ALERT_CLASSIFICATION + size: 400 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 400 + } + { + name: OWNER_SW_CFG_ROM_LOCAL_ALERT_CLASSIFICATION + size: 64 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 800 + } + { + name: OWNER_SW_CFG_ROM_ALERT_ACCUM_THRESH + size: 16 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 864 + } + { + name: OWNER_SW_CFG_ROM_ALERT_TIMEOUT_CYCLES + size: 16 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 880 + } + { + name: OWNER_SW_CFG_ROM_ALERT_PHASE_CYCLES + size: 64 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 896 + } + { + name: OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 960 + } + { + name: OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD_END + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 964 + } + { + name: OWNER_SW_CFG_ROM_ALERT_DIGEST_DEV + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 968 + } + { + name: OWNER_SW_CFG_ROM_ALERT_DIGEST_RMA + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 972 + } + { + name: OWNER_SW_CFG_ROM_WATCHDOG_BITE_THRESHOLD_CYCLES + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 976 + } + { + name: OWNER_SW_CFG_ROM_KEYMGR_ROM_EXT_MEAS_EN + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 980 + } + { + name: OWNER_SW_CFG_MANUF_STATE + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 984 + } + { + name: OWNER_SW_CFG_ROM_RSTMGR_INFO_EN + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 988 + } + { + name: OWNER_SW_CFG_DIGEST + size: 8 + offset: 1008 + ismubi: false + isdigest: true + inv_default: 503838644190341903 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Software configuration partition. + This contains data that changes software behavior in the ROM, for + example enabling defensive features in ROM or selecting failure + modes if verification fails. + ''' + iskeymgr_creator: false + iskeymgr_owner: false + offset: 384 + } + { + name: OWNERSHIP_SLOT_STATE + variant: Unbuffered + absorb: false + secret: false + sw_digest: false + hw_digest: false + write_lock: None + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: OWNERSHIP_SLOT_STATE_ROT_OWNER_AUTH + size: 16 + desc: + ''' + Creator Ownership transfer state management + OWNERSHIP_ST_RAW: ownership not yet claimed (factory default) + OWNERSHIP_ST_LOCKED0: first ownership slot claimed + OWNERSHIP_ST_RELEASED0: first ownership slot released (assets have been cleared, ready to xfer) + OWNERSHIP_ST_LOCKED1: second ownership slot claimed + OWNERSHIP_ST_SCRAPPED: scrap state - terminal + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1016 + } + { + name: OWNERSHIP_SLOT_STATE_PLAT_INTEG_AUTH + size: 16 + desc: + ''' + Creator Ownership transfer state management + OWNERSHIP_ST_RAW: ownership not yet claimed (factory default) + OWNERSHIP_ST_LOCKED0: first ownership slot claimed + OWNERSHIP_ST_RELEASED0: first ownership slot released (assets have been cleared, ready to xfer) + OWNERSHIP_ST_LOCKED1: second ownership slot claimed + OWNERSHIP_ST_SCRAPPED: scrap state - terminal + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1032 + } + { + name: OWNERSHIP_SLOT_STATE_PLAT_OWNER_AUTH + size: 16 + desc: + ''' + Creator Ownership transfer state management + OWNERSHIP_ST_RAW: ownership not yet claimed (factory default) + OWNERSHIP_ST_LOCKED0: first ownership slot claimed + OWNERSHIP_ST_RELEASED0: first ownership slot released (assets have been cleared, ready to xfer) + OWNERSHIP_ST_LOCKED1: second ownership slot claimed + OWNERSHIP_ST_RELEASED1: second ownership slot released (assets have been cleared, ready to xfer) + OWNERSHIP_ST_LOCKED2: third ownership slot claimed + OWNERSHIP_ST_RELEASED2: third ownership slot released (assets have been cleared, ready to xfer) + OWNERSHIP_ST_LOCKED3: fourth ownership slot claimed + OWNERSHIP_ST_SCRAPPED: scrap state - terminal + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1048 + } + ] + desc: + ''' + SW managed asset ownership states partition. + + Multibit enable value for the tracking the asset ownership states. + Note that the states can be written multiple times in a device lifetime. + The values to be written are engineered in the same way as the LC_CTRL + state encoding words so that the ECC encoding remains valid even after + updating the values. + + The constants can be found in the lc_ctrl_state_pkg.sv package. + + The programming order has to adhere to: + + OWNERSHIP_ST_RAW (factory all-zero state) -> + OWNERSHIP_ST_LOCKED0 -> + OWNERSHIP_ST_RELEASED0 -> + ... + OWNERSHIP_ST_SCRAPPED + + Note that if there are less than 4 slots available the higher slot states + become logically equivalent to OWNERSHIP_SCRAPPED (firmware has to handle + this correctly). + ''' + iskeymgr_creator: false + iskeymgr_owner: false + size: 48 + offset: 1016 + } + { + name: ROT_CREATOR_AUTH + variant: Unbuffered + absorb: false + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: ROT_CREATOR_AUTH_NON_RAW_MFW_CODESIGN_KEY + size: 160 + desc: + ''' + Pub Key used to verify the manufacturing provisioning Software container + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1064 + } + { + name: ROT_CREATOR_AUTH_OWNERSHIP_STATE + size: 4 + desc: + ''' + Creator Ownership transfer state management + UNLOCKED - ownership of ROT not yet claimed + LOCKED - ownership claimed and + RELEASED - Release ownership (assets have been cleared, ready to xfer) + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1224 + } + { + name: ROT_CREATOR_AUTH_ROM2_PATCH_SIGVERIFY_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROM2 OTP based patch + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + OTP based key (instead of ROM) facilitates easy key rotation + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1228 + } + { + name: ROT_CREATOR_AUTH_KEYMANIFEST_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT CREATOR KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1388 + } + { + name: ROT_CREATOR_AUTH_UNLOCK4XFER_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT CREATOR KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1548 + } + { + name: ROT_CREATOR_AUTH_IDENTITY_CERT + size: 768 + desc: ROT Device Identity Certificate endorsed by Silicon Creator stable off device PKI + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1708 + } + { + name: ROT_CREATOR_AUTH_DIGEST + size: 8 + offset: 2480 + ismubi: false + isdigest: true + inv_default: 4720744928697992353 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: Software managed creator partition. + iskeymgr_creator: false + iskeymgr_owner: false + size: 1424 + offset: 1064 + } + { + name: ROT_OWNER_AUTH_SLOT0 + variant: Unbuffered + absorb: false + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: ROT_OWNER_AUTH_SLOT0_KEYMANIFEST_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 2488 + } + { + name: ROT_OWNER_AUTH_SLOT0_UNLOCK4XFER_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 2648 + } + { + name: ROT_OWNER_AUTH_SLOT0_DIGEST + size: 8 + offset: 2808 + ismubi: false + isdigest: true + inv_default: 4321159426593998838 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: Software managed owner slot 0 partition. + iskeymgr_creator: false + iskeymgr_owner: false + size: 328 + offset: 2488 + } + { + name: ROT_OWNER_AUTH_SLOT1 + variant: Unbuffered + absorb: false + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: ROT_OWNER_AUTH_SLOT1_KEYMANIFEST_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 2816 + } + { + name: ROT_OWNER_AUTH_SLOT1_UNLOCK4XFER_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 2976 + } + { + name: ROT_OWNER_AUTH_SLOT1_DIGEST + size: 8 + offset: 3136 + ismubi: false + isdigest: true + inv_default: 3300532900599057395 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: Software managed owner slot 1 partition. + iskeymgr_creator: false + iskeymgr_owner: false + size: 328 + offset: 2816 + } + { + name: PLAT_INTEG_AUTH_SLOT0 + variant: Unbuffered + absorb: false + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: PLAT_INTEG_AUTH_SLOT0_KEYMANIFEST_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 3144 + } + { + name: PLAT_INTEG_AUTH_SLOT0_UNLOCK4XFER_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 3304 + } + { + name: PLAT_INTEG_AUTH_SLOT0_DIGEST + size: 8 + offset: 3464 + ismubi: false + isdigest: true + inv_default: 6970134785508705831 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: Software managed platform integrator slot 0 partition. + iskeymgr_creator: false + iskeymgr_owner: false + size: 328 + offset: 3144 + } + { + name: PLAT_INTEG_AUTH_SLOT1 + variant: Unbuffered + absorb: false + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: PLAT_INTEG_AUTH_SLOT1_KEYMANIFEST_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 3472 + } + { + name: PLAT_INTEG_AUTH_SLOT1_UNLOCK4XFER_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 3632 + } + { + name: PLAT_INTEG_AUTH_SLOT1_DIGEST + size: 8 + offset: 3792 + ismubi: false + isdigest: true + inv_default: 13321159437274557062 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: Software managed platform integrator slot 1 partition. + iskeymgr_creator: false + iskeymgr_owner: false + size: 328 + offset: 3472 + } + { + name: PLAT_OWNER_AUTH_SLOT0 + variant: Unbuffered + absorb: false + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: PLAT_OWNER_AUTH_SLOT0_KEYMANIFEST_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 3800 + } + { + name: PLAT_OWNER_AUTH_SLOT0_UNLOCK4XFER_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 3960 + } + { + name: PLAT_OWNER_AUTH_SLOT0_DIGEST + size: 8 + offset: 4120 + ismubi: false + isdigest: true + inv_default: 13771798441966885772 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: Software managed platform owner slot 0 partition. + iskeymgr_creator: false + iskeymgr_owner: false + size: 328 + offset: 3800 + } + { + name: PLAT_OWNER_AUTH_SLOT1 + variant: Unbuffered + absorb: false + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: PLAT_OWNER_AUTH_SLOT1_KEYMANIFEST_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 4128 + } + { + name: PLAT_OWNER_AUTH_SLOT1_UNLOCK4XFER_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 4288 + } + { + name: PLAT_OWNER_AUTH_SLOT1_DIGEST + size: 8 + offset: 4448 + ismubi: false + isdigest: true + inv_default: 7227229721152720746 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: Software managed platform owner slot 1 partition. + iskeymgr_creator: false + iskeymgr_owner: false + size: 328 + offset: 4128 + } + { + name: PLAT_OWNER_AUTH_SLOT2 + variant: Unbuffered + absorb: false + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: PLAT_OWNER_AUTH_SLOT2_KEYMANIFEST_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 4456 + } + { + name: PLAT_OWNER_AUTH_SLOT2_UNLOCK4XFER_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 4616 + } + { + name: PLAT_OWNER_AUTH_SLOT2_DIGEST + size: 8 + offset: 4776 + ismubi: false + isdigest: true + inv_default: 388365934790665074 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: Software managed platform owner slot 2 partition. + iskeymgr_creator: false + iskeymgr_owner: false + size: 328 + offset: 4456 + } + { + name: PLAT_OWNER_AUTH_SLOT3 + variant: Unbuffered + absorb: false + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: PLAT_OWNER_AUTH_SLOT3_KEYMANIFEST_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + KM is used to provisioning additional keys in (external) flash + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + It is expected to be provisioned at Chip Probe stage when LC = TEST + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 4784 + } + { + name: PLAT_OWNER_AUTH_SLOT3_UNLOCK4XFER_KEY + size: 160 + desc: + ''' + Pub Key used to verify the ROT OWNER KEY MANIFEST. + Used to authenticate the Unlock Ownership payload. Key Type: ECC NIST-P384 Curve. + 160B = Custom Cert --> {PubKey:48, Signature:48, Device ID:32, metadata:16} + Can be made part bound i.e. unique per part key cert + i.e. provisioning protected under proper lifecycle logistics / secure supplychain management + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 4944 + } + { + name: PLAT_OWNER_AUTH_SLOT3_DIGEST + size: 8 + offset: 5104 + ismubi: false + isdigest: true + inv_default: 5138760742690312726 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: Software managed platform owner slot 3 partition. + iskeymgr_creator: false + iskeymgr_owner: false + size: 328 + offset: 4784 + } + { + name: EXT_NVM + variant: Unbuffered + absorb: false + secret: false + sw_digest: false + hw_digest: false + write_lock: None + read_lock: CSR + key_sel: NoKey + integrity: false + bkout_type: false + items: + [ + { + name: EXT_NVM_ANTIREPLAY_FRESHNESS_CNT + size: 1024 + desc: + ''' + Fully SW managed patition + 1 bit (double bit allocated for redundancy - i.e. 2 physical bits for every logical bit) per day + assuming 10 year product lifecycle, allocating 8192 bits of freshness protection. + Implemented as thermometer encoding. Redundant bit programming & logical OR-ing managed by + software. Cannot be ECC protected since different bits in a fuse row are programmed at different times. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 5112 + } + ] + desc: Anti-replay protection Strike Counters partition. + iskeymgr_creator: false + iskeymgr_owner: false + size: 1024 + offset: 5112 + } + { + name: ROM_PATCH + variant: Unbuffered + absorb: true + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: ROM_PATCH_DATA + size: 9192 + desc: + ''' + Fully SW managed patition + ROM patch section + signed patch(es) + Each patch Contains header, signature, patch match/redirect configuration and patch code + Please refer to the ROM boot & patching specification for Integrated OT for more details + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 6136 + } + { + name: ROM_PATCH_DIGEST + size: 8 + offset: 15912 + ismubi: false + isdigest: true + inv_default: 5553021194950512605 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: ROM Patch Code section. May contain multiple signed ROM2 patches. + iskeymgr_creator: false + iskeymgr_owner: false + size: 9784 + offset: 6136 + } + { + name: HW_CFG0 + variant: Buffered + secret: false + sw_digest: false + hw_digest: true + write_lock: Digest + read_lock: None + key_sel: NoKey + integrity: true + bkout_type: true + items: + [ + { + name: DEVICE_ID + size: 32 + inv_default: 81256773759085553590251771618097271052792280600555849662047277760132860422574 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 15920 + } + { + name: MANUF_STATE + size: 32 + inv_default: 28979123207589896436497502269821436538902023475862784930259344116068944661506 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 15952 + } + { + name: HW_CFG0_DIGEST + size: 8 + offset: 15984 + ismubi: false + isdigest: true + inv_default: 13231792541330219804 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Hardware configuration 0 partition. + This contains + - DEVICE_ID: Unique device identifier. + - MANUF_STATE: Vector for capturing the manufacturing status. + ''' + absorb: false + iskeymgr_creator: false + iskeymgr_owner: false + size: 72 + offset: 15920 + } + { + name: HW_CFG1 + variant: Buffered + secret: false + sw_digest: false + hw_digest: true + write_lock: Digest + read_lock: None + key_sel: NoKey + integrity: true + bkout_type: true + items: + [ + { + name: SOC_DBG_STATE + size: 4 + inv_default: 0 + desc: + ''' + Multibit enable value for the SOC debug authorization. + Encoding: + SOC_DBG_RAW: this value is all-zeroes and will be the NOP state; + the LC controller will take precedence. + SOC_DBG_PRE_PROD: this is where the ROT will be in PROD state + but SOC will be in the pre-production unlock state + SOC_DBG_PROD: this is the state where the SOC moves to production, + and the challenge-response based authentication protocol is + required to unlock SOC debug features + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 15992 + } + { + name: EN_SRAM_IFETCH + size: 1 + ismubi: true + inv_default: 105 + isdigest: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 15996 + } + { + name: HW_CFG1_DIGEST + size: 8 + offset: 16000 + ismubi: false + isdigest: true + inv_default: 13264097316726039890 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Hardware configuration 1 partition. + This contains + EN_SRAM_IFETCH: Enable / disable execute from SRAM CSR switch. + SOC_DBG_STATE: Multibit enable value for the SOC debug authorization. + Note SOC_DBG_STATE will be written twice in a device lifetime. The + values to be written are engineered in the same way as the LC_CTRL + state encoding words: the ECC encoding remains valid even after + writing the second value on top of the first. + + The constants can be found in the lc_ctrl_state_pkg.sv package. + + The programming order has to adhere to: + + SOC_DBG_RAW -> SOC_DBG_PRE_PROD -> SOC_DBG_PROD. + ''' + absorb: false + iskeymgr_creator: false + iskeymgr_owner: false + size: 16 + offset: 15992 + } + { + name: SECRET0 + variant: Buffered + secret: true + sw_digest: false + hw_digest: true + write_lock: Digest + read_lock: Digest + key_sel: Secret0Key + integrity: true + bkout_type: false + items: + [ + { + name: TEST_UNLOCK_TOKEN + inv_default: 150353733052794565317978902093985221175 + size: 16 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 16008 + } + { + name: TEST_EXIT_TOKEN + inv_default: 292140191890949749780621917839409679837 + size: 16 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 16024 + } + { + name: SECRET0_DIGEST + size: 8 + offset: 16040 + ismubi: false + isdigest: true + inv_default: 364196237501841799 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Secret partition 0. + This contains TEST lifecycle unlock tokens. + ''' + absorb: false + iskeymgr_creator: false + iskeymgr_owner: false + size: 40 + offset: 16008 + } + { + name: SECRET1 + variant: Buffered + secret: true + sw_digest: false + hw_digest: true + write_lock: Digest + read_lock: Digest + key_sel: Secret1Key + integrity: true + bkout_type: false + items: + [ + { + name: FLASH_ADDR_KEY_SEED + inv_default: 99384293387760300854449600575324305105411001866428193504478753183549638555178 + size: 32 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 16048 + } + { + name: FLASH_DATA_KEY_SEED + inv_default: 38814441500057672017335312405494979779257105989744047190036253646062333315222 + size: 32 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 16080 + } + { + name: SRAM_DATA_KEY_SEED + inv_default: 197791420282250691661260177080212266122 + size: 16 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 16112 + } + { + name: SECRET1_DIGEST + size: 8 + offset: 16128 + ismubi: false + isdigest: true + inv_default: 8239982531674452350 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Secret partition 1. + This contains SRAM and flash scrambling keys. + ''' + absorb: false + iskeymgr_creator: false + iskeymgr_owner: false + size: 88 + offset: 16048 + } + { + name: SECRET2 + variant: Buffered + secret: true + sw_digest: false + hw_digest: true + write_lock: Digest + read_lock: Digest + key_sel: Secret2Key + integrity: true + bkout_type: false + items: + [ + { + name: RMA_TOKEN + inv_default: 277449187385066184202615139448003396941 + size: 16 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 16136 + } + { + name: CREATOR_ROOT_KEY_SHARE0 + inv_default: 42674330470926980475213495781321682493466151735469203603345579470514264688031 + size: 32 + iskeymgr_creator: true + isdigest: false + ismubi: false + iskeymgr_owner: false + offset: 16152 + } + { + name: CREATOR_ROOT_KEY_SHARE1 + inv_default: 1120155637374664855185290527946769138206986700190628506580202980715210922324 + size: 32 + iskeymgr_creator: true + isdigest: false + ismubi: false + iskeymgr_owner: false + offset: 16184 + } + { + name: CREATOR_SEED + inv_default: 486631087464711132701499347019667055309060048237121089079485173130930646974 + size: 32 + iskeymgr_creator: true + isdigest: false + ismubi: false + iskeymgr_owner: false + offset: 16216 + } + { + name: SECRET2_DIGEST + size: 8 + offset: 16248 + ismubi: false + isdigest: true + inv_default: 4468065780390625656 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Secret partition 2. + This contains RMA unlock token, creator root key, and creator seed. + ''' + absorb: false + iskeymgr_creator: true + iskeymgr_owner: false + size: 120 + offset: 16136 + } + { + name: SECRET3 + variant: Buffered + secret: true + sw_digest: false + hw_digest: true + write_lock: Digest + read_lock: Digest + key_sel: Secret3Key + integrity: true + bkout_type: false + items: + [ + { + name: OWNER_SEED + inv_default: 66811761207540413034971297187170994293973443726025940164646970994927536258183 + size: 32 + iskeymgr_owner: true + isdigest: false + ismubi: false + iskeymgr_creator: false + offset: 16256 + } + { + name: SECRET3_DIGEST + size: 8 + offset: 16288 + ismubi: false + isdigest: true + inv_default: 15941764794715087011 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Secret partition 3. + This contains the owner seed. + ''' + absorb: false + iskeymgr_creator: false + iskeymgr_owner: true + size: 40 + offset: 16256 + } + { + name: LIFE_CYCLE + variant: LifeCycle + secret: false + sw_digest: false + hw_digest: false + write_lock: None + read_lock: None + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: LC_TRANSITION_CNT + inv_default: 25283854627019714730581267052739223508902103880390507329789576351568549647905897976711414103002515926044760045076405 + size: 48 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 16296 + } + { + name: LC_STATE + inv_default: 2073276575949915156582572495874492333682185188009011646190809543994273356493467859637226911344936 + size: 40 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 16344 + } + ] + desc: + ''' + Lifecycle partition. + This contains lifecycle transition count and state. This partition + cannot be locked since the life cycle state needs to advance to RMA + in-field. Note that while this partition is not marked secret, it + is not readable nor writeable via the DAI. Only the LC controller + can access this partition, and even via the LC controller it is not + possible to read the raw manufacturing life cycle state in encoded + form, since that encoding is considered a netlist secret. The LC + controller only exposes a decoded version of this state. + ''' + absorb: false + iskeymgr_creator: false + iskeymgr_owner: false + size: 88 + offset: 16296 + } + ] + } + topname: darjeeling + } +} diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/defs.bzl b/hw/top_darjeeling/ip_autogen/otp_ctrl/defs.bzl new file mode 100644 index 00000000000000..810a1a1878c5cf --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/defs.bzl @@ -0,0 +1,9 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +load("//rules/opentitan:hw.bzl", "opentitan_ip") + +OTP_CTRL = opentitan_ip( + name = "otp_ctrl", + hjson = "//hw/top_darjeeling/ip_autogen/otp_ctrl:data/otp_ctrl.hjson", +) diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/checklist.md b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/checklist.md new file mode 100644 index 00000000000000..117db7dd8d60bd --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/checklist.md @@ -0,0 +1,271 @@ +# OTP_CTRL Checklist + + +This checklist is for [Hardware Stage](../../../../../doc/project_governance/development_stages.md) transitions for the [OTP_CTRL peripheral.](../README.md) +All checklist items refer to the content in the [Checklist.](../../../../../doc/project_governance/checklist/README.md) + +## Design Checklist + +### D1 + +Type | Item | Resolution | Note/Collaterals +--------------|--------------------------------|-------------|------------------ +Documentation | [SPEC_COMPLETE][] | Done | [OTP_CTRL Design Spec](../README.md) +Documentation | [CSR_DEFINED][] | Done | +RTL | [CLKRST_CONNECTED][] | Done | +RTL | [IP_TOP][] | Done | +RTL | [IP_INSTANTIABLE][] | Done | +RTL | [PHYSICAL_MACROS_DEFINED_80][] | Done | +RTL | [FUNC_IMPLEMENTED][] | Done | +RTL | [ASSERT_KNOWN_ADDED][] | Done | +Code Quality | [LINT_SETUP][] | Done | + +[SPEC_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#spec_complete +[CSR_DEFINED]: ../../../../../doc/project_governance/checklist/README.md#csr_defined +[CLKRST_CONNECTED]: ../../../../../doc/project_governance/checklist/README.md#clkrst_connected +[IP_TOP]: ../../../../../doc/project_governance/checklist/README.md#ip_top +[IP_INSTANTIABLE]: ../../../../../doc/project_governance/checklist/README.md#ip_instantiable +[PHYSICAL_MACROS_DEFINED_80]: ../../../../../doc/project_governance/checklist/README.md#mem_instanced_80 +[FUNC_IMPLEMENTED]: ../../../../../doc/project_governance/checklist/README.md#func_implemented +[ASSERT_KNOWN_ADDED]: ../../../../../doc/project_governance/checklist/README.md#assert_known_added +[LINT_SETUP]: ../../../../../doc/project_governance/checklist/README.md#lint_setup + +### D2 + +Type | Item | Resolution | Note/Collaterals +--------------|---------------------------|-------------|------------------ +Documentation | [NEW_FEATURES][] | Done | +Documentation | [BLOCK_DIAGRAM][] | Done | +Documentation | [DOC_INTERFACE][] | Done | +Documentation | [DOC_INTEGRATION_GUIDE][] | Waived | This checklist item has been added retrospectively. +Documentation | [MISSING_FUNC][] | Done | +Documentation | [FEATURE_FROZEN][] | Done | +RTL | [FEATURE_COMPLETE][] | Done | +RTL | [PORT_FROZEN][] | Done | +RTL | [ARCHITECTURE_FROZEN][] | Done | +RTL | [REVIEW_TODO][] | Done | +RTL | [STYLE_X][] | Done | +RTL | [CDC_SYNCMACRO][] | N/A | +Code Quality | [LINT_PASS][] | Done | +Code Quality | [CDC_SETUP][] | Waived | No block-level flow available - waived to top-level signoff. +Code Quality | [RDC_SETUP][] | Waived | No block-level flow available - waived to top-level signoff. +Code Quality | [AREA_CHECK][] | Done | +Code Quality | [TIMING_CHECK][] | Done | +Security | [SEC_CM_DOCUMENTED][] | Done | + +[NEW_FEATURES]: ../../../../../doc/project_governance/checklist/README.md#new_features +[BLOCK_DIAGRAM]: ../../../../../doc/project_governance/checklist/README.md#block_diagram +[DOC_INTERFACE]: ../../../../../doc/project_governance/checklist/README.md#doc_interface +[DOC_INTEGRATION_GUIDE]: ../../../../../doc/project_governance/checklist/README.md#doc_integration_guide +[MISSING_FUNC]: ../../../../../doc/project_governance/checklist/README.md#missing_func +[FEATURE_FROZEN]: ../../../../../doc/project_governance/checklist/README.md#feature_frozen +[FEATURE_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#feature_complete +[PORT_FROZEN]: ../../../../../doc/project_governance/checklist/README.md#port_frozen +[ARCHITECTURE_FROZEN]: ../../../../../doc/project_governance/checklist/README.md#architecture_frozen +[REVIEW_TODO]: ../../../../../doc/project_governance/checklist/README.md#review_todo +[STYLE_X]: ../../../../../doc/project_governance/checklist/README.md#style_x +[CDC_SYNCMACRO]: ../../../../../doc/project_governance/checklist/README.md#cdc_syncmacro +[LINT_PASS]: ../../../../../doc/project_governance/checklist/README.md#lint_pass +[CDC_SETUP]: ../../../../../doc/project_governance/checklist/README.md#cdc_setup +[RDC_SETUP]: ../../../../../doc/project_governance/checklist/README.md#rdc_setup +[AREA_CHECK]: ../../../../../doc/project_governance/checklist/README.md#area_check +[TIMING_CHECK]: ../../../../../doc/project_governance/checklist/README.md#timing_check +[SEC_CM_DOCUMENTED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_documented + +### D2S + + Type | Item | Resolution | Note/Collaterals +--------------|------------------------------|-------------|------------------ +Security | [SEC_CM_ASSETS_LISTED][] | Done | +Security | [SEC_CM_IMPLEMENTED][] | Done | +Security | [SEC_CM_RND_CNST][] | Done | +Security | [SEC_CM_NON_RESET_FLOPS][] | Done | +Security | [SEC_CM_SHADOW_REGS][] | Done | +Security | [SEC_CM_RTL_REVIEWED][] | Done | +Security | [SEC_CM_COUNCIL_REVIEWED][] | Done | + +[SEC_CM_ASSETS_LISTED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_assets_listed +[SEC_CM_IMPLEMENTED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_implemented +[SEC_CM_RND_CNST]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_rnd_cnst +[SEC_CM_NON_RESET_FLOPS]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_non_reset_flops +[SEC_CM_SHADOW_REGS]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_shadow_regs +[SEC_CM_RTL_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_rtl_reviewed +[SEC_CM_COUNCIL_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_council_reviewed + +### D3 + + Type | Item | Resolution | Note/Collaterals +--------------|-------------------------|-------------|------------------ +Documentation | [NEW_FEATURES_D3][] | Done | +RTL | [TODO_COMPLETE][] | Done | +Code Quality | [LINT_COMPLETE][] | Done | Waiver files approved by TC on 2024-08-08. +Code Quality | [CDC_COMPLETE][] | Waived | No block-level flow available - waived to top-level signoff. +Code Quality | [RDC_COMPLETE][] | Waived | No block-level flow available - waived to top-level signoff. +Review | [REVIEW_RTL][] | Done | +Review | [REVIEW_DELETED_FF][] | Waived | No block-level flow available - waived to top-level signoff. +Review | [REVIEW_SW_CHANGE][] | Done | +Review | [REVIEW_SW_ERRATA][] | Done | +Review | Reviewer(s) | Done | adk@ vogelpi@ +Review | Signoff date | Done | 2024-08-08 + +[NEW_FEATURES_D3]: ../../../../../doc/project_governance/checklist/README.md#new_features_d3 +[TODO_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#todo_complete +[LINT_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#lint_complete +[CDC_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#cdc_complete +[RDC_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#rdc_complete +[REVIEW_RTL]: ../../../../../doc/project_governance/checklist/README.md#review_rtl +[REVIEW_DELETED_FF]: ../../../../../doc/project_governance/checklist/README.md#review_deleted_ff +[REVIEW_SW_CHANGE]: ../../../../../doc/project_governance/checklist/README.md#review_sw_change +[REVIEW_SW_ERRATA]: ../../../../../doc/project_governance/checklist/README.md#review_sw_errata + +## Verification Checklist + +### V1 + + Type | Item | Resolution | Note/Collaterals +--------------|---------------------------------------|-------------|------------------ +Documentation | [DV_DOC_DRAFT_COMPLETED][] | Done | [OTP_CTRL DV document](../dv/README.md) +Documentation | [TESTPLAN_COMPLETED][] | Done | [OTP_CTRL Testplan](../dv/README.md#testplan) +Testbench | [TB_TOP_CREATED][] | Done | +Testbench | [PRELIMINARY_ASSERTION_CHECKS_ADDED][]| Done | +Testbench | [SIM_TB_ENV_CREATED][] | Done | +Testbench | [SIM_RAL_MODEL_GEN_AUTOMATED][] | Done | +Testbench | [CSR_CHECK_GEN_AUTOMATED][] | Done | +Testbench | [TB_GEN_AUTOMATED][] | Done | +Tests | [SIM_SMOKE_TEST_PASSING][] | Done | +Tests | [SIM_CSR_MEM_TEST_SUITE_PASSING][] | Done | +Tests | [FPV_MAIN_ASSERTIONS_PROVEN][] | N/A | +Tool Setup | [SIM_ALT_TOOL_SETUP][] | Done | +Regression | [SIM_SMOKE_REGRESSION_SETUP][] | Done | +Regression | [SIM_NIGHTLY_REGRESSION_SETUP][] | Done | +Regression | [FPV_REGRESSION_SETUP][] | N/A | +Coverage | [SIM_COVERAGE_MODEL_ADDED][] | Done | +Code Quality | [TB_LINT_SETUP][] | Done | +Integration | [PRE_VERIFIED_SUB_MODULES_V1][] | N/A | Exception for IP modules +Review | [DESIGN_SPEC_REVIEWED][] | Done | +Review | [TESTPLAN_REVIEWED][] | Done | +Review | [STD_TEST_CATEGORIES_PLANNED][] | Done | Exception (Security, Power, Debug) +Review | [V2_CHECKLIST_SCOPED][] | Done | + +[DV_DOC_DRAFT_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#dv_doc_draft_completed +[TESTPLAN_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#testplan_completed +[TB_TOP_CREATED]: ../../../../../doc/project_governance/checklist/README.md#tb_top_created +[PRELIMINARY_ASSERTION_CHECKS_ADDED]: ../../../../../doc/project_governance/checklist/README.md#preliminary_assertion_checks_added +[SIM_TB_ENV_CREATED]: ../../../../../doc/project_governance/checklist/README.md#sim_tb_env_created +[SIM_RAL_MODEL_GEN_AUTOMATED]: ../../../../../doc/project_governance/checklist/README.md#sim_ral_model_gen_automated +[CSR_CHECK_GEN_AUTOMATED]: ../../../../../doc/project_governance/checklist/README.md#csr_check_gen_automated +[TB_GEN_AUTOMATED]: ../../../../../doc/project_governance/checklist/README.md#tb_gen_automated +[SIM_SMOKE_TEST_PASSING]: ../../../../../doc/project_governance/checklist/README.md#sim_smoke_test_passing +[SIM_CSR_MEM_TEST_SUITE_PASSING]: ../../../../../doc/project_governance/checklist/README.md#sim_csr_mem_test_suite_passing +[FPV_MAIN_ASSERTIONS_PROVEN]: ../../../../../doc/project_governance/checklist/README.md#fpv_main_assertions_proven +[SIM_ALT_TOOL_SETUP]: ../../../../../doc/project_governance/checklist/README.md#sim_alt_tool_setup +[SIM_SMOKE_REGRESSION_SETUP]: ../../../../../doc/project_governance/checklist/README.md#sim_smoke_regression_setup +[SIM_NIGHTLY_REGRESSION_SETUP]: ../../../../../doc/project_governance/checklist/README.md#sim_nightly_regression_setup +[FPV_REGRESSION_SETUP]: ../../../../../doc/project_governance/checklist/README.md#fpv_regression_setup +[SIM_COVERAGE_MODEL_ADDED]: ../../../../../doc/project_governance/checklist/README.md#sim_coverage_model_added +[PRE_VERIFIED_SUB_MODULES_V1]: ../../../../../doc/project_governance/checklist/README.md#pre_verified_sub_modules_v1 +[TB_LINT_SETUP]: ../../../../../doc/project_governance/checklist/README.md#tb_lint_setup +[DESIGN_SPEC_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#design_spec_reviewed +[TESTPLAN_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#testplan_reviewed +[STD_TEST_CATEGORIES_PLANNED]: ../../../../../doc/project_governance/checklist/README.md#std_test_categories_planned +[V2_CHECKLIST_SCOPED]: ../../../../../doc/project_governance/checklist/README.md#v2_checklist_scoped + +### V2 + + Type | Item | Resolution | Note/Collaterals +--------------|-----------------------------------------|-------------|------------------ +Documentation | [DESIGN_DELTAS_CAPTURED_V2][] | Done | +Documentation | [DV_DOC_COMPLETED][] | Done | +Testbench | [FUNCTIONAL_COVERAGE_IMPLEMENTED][] | Done | +Testbench | [ALL_INTERFACES_EXERCISED][] | Done | `prim_tl_o/i` has a simple prim_tl_agent support, but need to be replaced with auto-generated tl_agent once reggen tool is optimized. +Testbench | [ALL_ASSERTION_CHECKS_ADDED][] | Done | +Testbench | [SIM_TB_ENV_COMPLETED][] | Done | +Tests | [SIM_ALL_TESTS_PASSING][] | Done | +Tests | [FPV_ALL_ASSERTIONS_WRITTEN][] | N/A | +Tests | [FPV_ALL_ASSUMPTIONS_REVIEWED][] | N/A | +Tests | [SIM_FW_SIMULATED][] | N/A | +Regression | [SIM_NIGHTLY_REGRESSION_V2][] | Done | +Coverage | [SIM_CODE_COVERAGE_V2][] | Done | +Coverage | [SIM_FUNCTIONAL_COVERAGE_V2][] | Done | +Coverage | [FPV_CODE_COVERAGE_V2][] | N/A | +Coverage | [FPV_COI_COVERAGE_V2][] | N/A | +Integration | [PRE_VERIFIED_SUB_MODULES_V2][] | Done | Waived `prim_alert_sender` and `prim_lfsr` +Issues | [NO_HIGH_PRIORITY_ISSUES_PENDING][] | Done | +Issues | [ALL_LOW_PRIORITY_ISSUES_ROOT_CAUSED][] | Done | +Review | [DV_DOC_TESTPLAN_REVIEWED][] | Done | Reviewed on 05/24/2021 +Review | [V3_CHECKLIST_SCOPED][] | Done | + +[DESIGN_DELTAS_CAPTURED_V2]: ../../../../../doc/project_governance/checklist/README.md#design_deltas_captured_v2 +[DV_DOC_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#dv_doc_completed +[FUNCTIONAL_COVERAGE_IMPLEMENTED]: ../../../../../doc/project_governance/checklist/README.md#functional_coverage_implemented +[ALL_INTERFACES_EXERCISED]: ../../../../../doc/project_governance/checklist/README.md#all_interfaces_exercised +[ALL_ASSERTION_CHECKS_ADDED]: ../../../../../doc/project_governance/checklist/README.md#all_assertion_checks_added +[SIM_TB_ENV_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#sim_tb_env_completed +[SIM_ALL_TESTS_PASSING]: ../../../../../doc/project_governance/checklist/README.md#sim_all_tests_passing +[FPV_ALL_ASSERTIONS_WRITTEN]: ../../../../../doc/project_governance/checklist/README.md#fpv_all_assertions_written +[FPV_ALL_ASSUMPTIONS_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#fpv_all_assumptions_reviewed +[SIM_FW_SIMULATED]: ../../../../../doc/project_governance/checklist/README.md#sim_fw_simulated +[SIM_NIGHTLY_REGRESSION_V2]: ../../../../../doc/project_governance/checklist/README.md#sim_nightly_regression_v2 +[SIM_CODE_COVERAGE_V2]: ../../../../../doc/project_governance/checklist/README.md#sim_code_coverage_v2 +[SIM_FUNCTIONAL_COVERAGE_V2]: ../../../../../doc/project_governance/checklist/README.md#sim_functional_coverage_v2 +[FPV_CODE_COVERAGE_V2]: ../../../../../doc/project_governance/checklist/README.md#fpv_code_coverage_v2 +[FPV_COI_COVERAGE_V2]: ../../../../../doc/project_governance/checklist/README.md#fpv_coi_coverage_v2 +[PRE_VERIFIED_SUB_MODULES_V2]: ../../../../../doc/project_governance/checklist/README.md#pre_verified_sub_modules_v2 +[NO_HIGH_PRIORITY_ISSUES_PENDING]: ../../../../../doc/project_governance/checklist/README.md#no_high_priority_issues_pending +[ALL_LOW_PRIORITY_ISSUES_ROOT_CAUSED]:../../../../../doc/project_governance/checklist/README.md#all_low_priority_issues_root_caused +[DV_DOC_TESTPLAN_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#dv_doc_testplan_reviewed +[V3_CHECKLIST_SCOPED]: ../../../../../doc/project_governance/checklist/README.md#v3_checklist_scoped + +### V2S + + Type | Item | Resolution | Note/Collaterals +--------------|-----------------------------------------|-------------|------------------ +Documentation | [SEC_CM_TESTPLAN_COMPLETED][] | Done | +Tests | [FPV_SEC_CM_VERIFIED][] | Done | +Tests | [SIM_SEC_CM_VERIFIED][] | Done | +Coverage | [SIM_COVERAGE_REVIEWED][] | Done | +Review | [SEC_CM_DV_REVIEWED][] | Done | + +[SEC_CM_TESTPLAN_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_testplan_completed +[FPV_SEC_CM_VERIFIED]: ../../../../../doc/project_governance/checklist/README.md#fpv_sec_cm_verified +[SIM_SEC_CM_VERIFIED]: ../../../../../doc/project_governance/checklist/README.md#sim_sec_cm_verified +[SIM_COVERAGE_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#sim_coverage_reviewed +[SEC_CM_DV_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_dv_reviewed + +### V3 + + Type | Item | Resolution | Note/Collaterals +--------------|-----------------------------------|-------------|------------------ +Documentation | [DESIGN_DELTAS_CAPTURED_V3][] | Not Started | +Tests | [X_PROP_ANALYSIS_COMPLETED][] | Not Started | +Tests | [FPV_ASSERTIONS_PROVEN_AT_V3][] | Not Started | +Regression | [SIM_NIGHTLY_REGRESSION_AT_V3][] | Not Started | +Coverage | [SIM_CODE_COVERAGE_AT_100][] | Not Started | +Coverage | [SIM_FUNCTIONAL_COVERAGE_AT_100][]| Not Started | +Coverage | [FPV_CODE_COVERAGE_AT_100][] | Not Started | +Coverage | [FPV_COI_COVERAGE_AT_100][] | Not Started | +Code Quality | [ALL_TODOS_RESOLVED][] | Not Started | +Code Quality | [NO_TOOL_WARNINGS_THROWN][] | Not Started | +Code Quality | [TB_LINT_COMPLETE][] | Not Started | +Integration | [PRE_VERIFIED_SUB_MODULES_V3][] | Not Started | +Issues | [NO_ISSUES_PENDING][] | Not Started | +Review | Reviewer(s) | Not Started | +Review | Signoff date | Not Started | + +[DESIGN_DELTAS_CAPTURED_V3]: ../../../../../doc/project_governance/checklist/README.md#design_deltas_captured_v3 +[X_PROP_ANALYSIS_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#x_prop_analysis_completed +[FPV_ASSERTIONS_PROVEN_AT_V3]: ../../../../../doc/project_governance/checklist/README.md#fpv_assertions_proven_at_v3 +[SIM_NIGHTLY_REGRESSION_AT_V3]: ../../../../../doc/project_governance/checklist/README.md#sim_nightly_regression_at_v3 +[SIM_CODE_COVERAGE_AT_100]: ../../../../../doc/project_governance/checklist/README.md#sim_code_coverage_at_100 +[SIM_FUNCTIONAL_COVERAGE_AT_100]:../../../../../doc/project_governance/checklist/README.md#sim_functional_coverage_at_100 +[FPV_CODE_COVERAGE_AT_100]: ../../../../../doc/project_governance/checklist/README.md#fpv_code_coverage_at_100 +[FPV_COI_COVERAGE_AT_100]: ../../../../../doc/project_governance/checklist/README.md#fpv_coi_coverage_at_100 +[ALL_TODOS_RESOLVED]: ../../../../../doc/project_governance/checklist/README.md#all_todos_resolved +[NO_TOOL_WARNINGS_THROWN]: ../../../../../doc/project_governance/checklist/README.md#no_tool_warnings_thrown +[TB_LINT_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#tb_lint_complete +[PRE_VERIFIED_SUB_MODULES_V3]: ../../../../../doc/project_governance/checklist/README.md#pre_verified_sub_modules_v3 +[NO_ISSUES_PENDING]: ../../../../../doc/project_governance/checklist/README.md#no_issues_pending diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/interfaces.md b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/interfaces.md new file mode 100644 index 00000000000000..5460fbb792b19c --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/interfaces.md @@ -0,0 +1,301 @@ +# Hardware Interfaces + +## Parameters + +The following table lists the instantiation parameters of OTP. +Note that parameters prefixed with `RndCnst` are random netlist constants that need to be regenerated via topgen before the tapeout (typically by the silicon creator). + +Parameter | Default (Max) | Top Earlgrey | Description +----------------------------|---------------|--------------|--------------- +`AlertAsyncOn` | 2'b11 | 2'b11 | +`RndCnstLfsrSeed` | (see RTL) | (see RTL) | Seed to be used for the internal 40bit partition check timer LFSR. This needs to be replaced by the silicon creator before the tapeout. +`RndCnstLfsrPerm` | (see RTL) | (see RTL) | Permutation to be used for the internal 40bit partition check timer LFSR. This needs to be replaced by the silicon creator before the tapeout. +`RndCnstKey` | (see RTL) | (see RTL) | Random scrambling keys for secret partitions, to be used in the [scrambling datapath](#scrambling-datapath). +`RndCnstDigestConst` | (see RTL) | (see RTL) | Random digest finalization constants, to be used in the [scrambling datapath](#scrambling-datapath). +`RndCnstDigestIV` | (see RTL) | (see RTL) | Random digest initialization vectors, to be used in the [scrambling datapath](#scrambling-datapath). +`RndCnstRawUnlockToken` | (see RTL) | (see RTL) | Global RAW unlock token to be used for the first life cycle transition. See also [conditional life cycle transitions](../../../../ip/lc_ctrl/README.md#conditional-transitions). + + +Referring to the [Comportable guideline for peripheral device functionality](https://opentitan.org/book/doc/contributing/hw/comportability), the module **`otp_ctrl`** has the following hardware interfaces defined +- Primary Clock: **`clk_i`** +- Other Clocks: **`clk_edn_i`** +- Bus Device Interfaces (TL-UL): **`core_tl`**, **`prim_tl`** +- Bus Host Interfaces (TL-UL): *none* + +## Peripheral Pins for Chip IO + +| Pin name | Direction | Description | +|:-----------|:------------|:------------------------------------------------------------------| +| test[7:0] | output | Test-related GPIOs. Only active in DFT-enabled life cycle states. | + +## [Inter-Module Signals](https://opentitan.org/book/doc/contributing/hw/comportability/index.html#inter-signal-handling) + +| Port Name | Package::Struct | Type | Act | Width | Description | +|:-------------------------|:---------------------------------|:--------|:------|--------:|:----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| otp_ext_voltage_h | | io | none | 1 | | +| otp_ast_pwr_seq | otp_ctrl_pkg::otp_ast_req | uni | req | 1 | Power sequencing signals to AST (VDD domain). | +| otp_ast_pwr_seq_h | otp_ctrl_pkg::otp_ast_rsp | uni | rcv | 1 | Power sequencing signals coming from AST (VCC domain). | +| edn | edn_pkg::edn | req_rsp | req | 1 | Entropy request to the entropy distribution network for LFSR reseeding and ephemeral key derivation. | +| pwr_otp | pwrmgr_pkg::pwr_otp | req_rsp | rsp | 1 | Initialization request/acknowledge from/to power manager. | +| lc_otp_vendor_test | otp_ctrl_pkg::lc_otp_vendor_test | req_rsp | rsp | 1 | Vendor test control signals from/to the life cycle TAP. | +| lc_otp_program | otp_ctrl_pkg::lc_otp_program | req_rsp | rsp | 1 | Life cycle state transition interface. | +| otp_lc_data | otp_ctrl_pkg::otp_lc_data | uni | req | 1 | Life cycle state output holding the current life cycle state, the value of the transition counter and the tokens needed for life cycle transitions. | +| lc_escalate_en | lc_ctrl_pkg::lc_tx | uni | rcv | 1 | Life cycle escalation enable coming from life cycle controller. This signal moves all FSMs within OTP into the error state. | +| lc_creator_seed_sw_rw_en | lc_ctrl_pkg::lc_tx | uni | rcv | 1 | Provision enable qualifier coming from life cycle controller. This signal enables SW read / write access to the RMA_TOKEN and CREATOR_ROOT_KEY_SHARE0 and CREATOR_ROOT_KEY_SHARE1. | +| lc_owner_seed_sw_rw_en | lc_ctrl_pkg::lc_tx | uni | rcv | 1 | Provision enable qualifier coming from life cycle controller. This signal enables SW read / write access to the OWNER_SEED. | +| lc_seed_hw_rd_en | lc_ctrl_pkg::lc_tx | uni | rcv | 1 | Seed read enable coming from life cycle controller. This signal enables HW read access to the CREATOR_ROOT_KEY_SHARE0 and CREATOR_ROOT_KEY_SHARE1. | +| lc_dft_en | lc_ctrl_pkg::lc_tx | uni | rcv | 1 | Test enable qualifier coming from life cycle controller. This signals enables the TL-UL access port to the proprietary OTP IP. | +| lc_check_byp_en | lc_ctrl_pkg::lc_tx | uni | rcv | 1 | Life cycle partition check bypass signal. This signal causes the life cycle partition to bypass consistency checks during life cycle state transitions in order to prevent spurious consistency check failures. | +| otp_keymgr_key | otp_ctrl_pkg::otp_keymgr_key | uni | req | 1 | Key output to the key manager holding CREATOR_ROOT_KEY_SHARE0 and CREATOR_ROOT_KEY_SHARE1. | +| flash_otp_key | otp_ctrl_pkg::flash_otp_key | req_rsp | rsp | 1 | Key derivation interface for FLASH scrambling. | +| sram_otp_key | otp_ctrl_pkg::sram_otp_key | req_rsp | rsp | 4 | Array with key derivation interfaces for SRAM scrambling devices. | +| otbn_otp_key | otp_ctrl_pkg::otbn_otp_key | req_rsp | rsp | 1 | Key derivation interface for OTBN scrambling devices. | +| otp_broadcast | otp_ctrl_part_pkg::otp_broadcast | uni | req | 1 | Output of the HW partitions with breakout data types. | +| obs_ctrl | ast_pkg::ast_obs_ctrl | uni | rcv | 1 | AST observability control signals. | +| otp_obs | logic | uni | req | 8 | AST observability bus. | +| core_tl | tlul_pkg::tl | req_rsp | rsp | 1 | | +| prim_tl | tlul_pkg::tl | req_rsp | rsp | 1 | | + +## Interrupts + +| Interrupt Name | Type | Description | +|:-------------------|:-------|:-----------------------------------------------------------------------------------------------------------------------------| +| otp_operation_done | Event | A direct access command or digest calculation operation has completed. | +| otp_error | Event | An error has occurred in the OTP controller. Check the [`ERR_CODE`](registers.md#err_code) register to get more information. | + +## Security Alerts + +| Alert Name | Description | +|:----------------------|:---------------------------------------------------------------------------------------------------------------------------------------------------| +| fatal_macro_error | This alert triggers if hardware detects an uncorrectable error during an OTP transaction, for example an uncorrectable ECC error in the OTP array. | +| fatal_check_error | This alert triggers if any of the background checks fails. This includes the digest checks and concurrent ECC checks in the buffer registers. | +| fatal_bus_integ_error | This fatal alert is triggered when a fatal TL-UL bus integrity fault is detected. | +| fatal_prim_otp_alert | Fatal alert triggered inside the OTP primitive, including fatal TL-UL bus integrity faults of the test interface. | +| recov_prim_otp_alert | Recoverable alert triggered inside the OTP primitive. | + +## Security Countermeasures + +| Countermeasure ID | Description | +|:-------------------------------------|:--------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| OTP_CTRL.BUS.INTEGRITY | End-to-end bus integrity scheme. | +| OTP_CTRL.SECRET.MEM.SCRAMBLE | Secret partitions are scrambled with a full-round PRESENT cipher. | +| OTP_CTRL.PART.MEM.DIGEST | Integrity of buffered partitions is ensured via a 64bit digest. | +| OTP_CTRL.DAI.FSM.SPARSE | The direct access interface FSM is sparsely encoded. | +| OTP_CTRL.KDI.FSM.SPARSE | The key derivation interface FSM is sparsely encoded. | +| OTP_CTRL.LCI.FSM.SPARSE | The life cycle interface FSM is sparsely encoded. | +| OTP_CTRL.PART.FSM.SPARSE | The partition FSMs are sparsely encoded. | +| OTP_CTRL.SCRMBL.FSM.SPARSE | The scramble datapath FSM is sparsely encoded. | +| OTP_CTRL.TIMER.FSM.SPARSE | The background check timer FSM is sparsely encoded. | +| OTP_CTRL.DAI.CTR.REDUN | The direct access interface address counter employs a cross-counter implementation. | +| OTP_CTRL.KDI_SEED.CTR.REDUN | The key derivation interface counter employs a cross-counter implementation. | +| OTP_CTRL.KDI_ENTROPY.CTR.REDUN | The key derivation entropy counter employs a cross-counter implementation. | +| OTP_CTRL.LCI.CTR.REDUN | The life cycle interface address counter employs a cross-counter implementation. | +| OTP_CTRL.PART.CTR.REDUN | The address counter of buffered partitions employs a cross-counter implementation. | +| OTP_CTRL.SCRMBL.CTR.REDUN | The srambling datapath counter employs a cross-counter implementation. | +| OTP_CTRL.TIMER_INTEG.CTR.REDUN | The background integrity check timer employs a duplicated counter implementation. | +| OTP_CTRL.TIMER_CNSTY.CTR.REDUN | The background consistency check timer employs a duplicated counter implementation. | +| OTP_CTRL.TIMER.LFSR.REDUN | The background check LFSR is duplicated. | +| OTP_CTRL.DAI.FSM.LOCAL_ESC | The direct access interface FSM is moved into an invalid state upon local escalation. | +| OTP_CTRL.LCI.FSM.LOCAL_ESC | The life cycle interface FSM is moved into an invalid state upon local escalation. | +| OTP_CTRL.KDI.FSM.LOCAL_ESC | The key derivation interface FSM is moved into an invalid state upon local escalation. | +| OTP_CTRL.PART.FSM.LOCAL_ESC | The partition FSMs are moved into an invalid state upon local escalation. | +| OTP_CTRL.SCRMBL.FSM.LOCAL_ESC | The scramble datapath FSM is moved into an invalid state upon local escalation. | +| OTP_CTRL.TIMER.FSM.LOCAL_ESC | The background check timer FSM is moved into an invalid state upon local escalation. | +| OTP_CTRL.DAI.FSM.GLOBAL_ESC | The direct access interface FSM is moved into an invalid state upon global escalation via life cycle. | +| OTP_CTRL.LCI.FSM.GLOBAL_ESC | The life cycle interface FSM is moved into an invalid state upon global escalation via life cycle. | +| OTP_CTRL.KDI.FSM.GLOBAL_ESC | The key derivation interface FSM is moved into an invalid state upon global escalation via life cycle. | +| OTP_CTRL.PART.FSM.GLOBAL_ESC | The partition FSMs are moved into an invalid state upon global escalation via life cycle. | +| OTP_CTRL.SCRMBL.FSM.GLOBAL_ESC | The scramble datapath FSM is moved into an invalid state upon global escalation via life cycle. | +| OTP_CTRL.TIMER.FSM.GLOBAL_ESC | The background check timer FSM is moved into an invalid state upon global escalation via life cycle. | +| OTP_CTRL.PART.DATA_REG.INTEGRITY | All partition buffer registers are protected with ECC on 64bit blocks. | +| OTP_CTRL.PART.DATA_REG.BKGN_CHK | The digest of buffered partitions is recomputed and checked at pseudorandom intervals in the background. | +| OTP_CTRL.PART.MEM.REGREN | Unbuffered ('software') partitions can be read-locked via a CSR until the next system reset. | +| OTP_CTRL.PART.MEM.SW_UNREADABLE | Secret buffered partitions become unreadable to software once they are locked via the digest. | +| OTP_CTRL.PART.MEM.SW_UNWRITABLE | All partitions become unwritable by software once they are locked via the digest. | +| OTP_CTRL.LC_PART.MEM.SW_NOACCESS | The life cycle partition is not directly readable nor writable via software. | +| OTP_CTRL.ACCESS.CTRL.MUBI | The access control signals going from the partitions to the DAI are MUBI encoded. | +| OTP_CTRL.TOKEN_VALID.CTRL.MUBI | The token valid signals going to the life cycle controller are MUBI encoded. | +| OTP_CTRL.LC_CTRL.INTERSIG.MUBI | The life cycle control signals are multibit encoded. | +| OTP_CTRL.TEST.BUS.LC_GATED | Prevent access to test signals and the OTP backdoor interface in non-test lifecycle states. | +| OTP_CTRL.TEST_TL_LC_GATE.FSM.SPARSE | The control FSM inside the TL-UL gating primitive is sparsely encoded. | +| OTP_CTRL.DIRECT_ACCESS.CONFIG.REGWEN | The direct access CSRs are REGWEN protected. | +| OTP_CTRL.CHECK_TRIGGER.CONFIG.REGWEN | The check trigger CSR is REGWEN protected. | +| OTP_CTRL.CHECK.CONFIG.REGWEN | The check CSR is REGWEN protected. | +| OTP_CTRL.MACRO.MEM.INTEGRITY | The OTP macro employs a vendor-specific integrity scheme at the granularity of the native 16bit OTP words. The scheme is able to at least detect single bit errors. | +| OTP_CTRL.MACRO.MEM.CM | The OTP macro may contain additional vendor-specific countermeasures. | + + + + +The OTP controller contains various interfaces that connect to other comportable IPs within OpenTitan, and these are briefly explained further below. + +### EDN Interface + +The entropy request interface that talks to EDN in order to fetch fresh entropy for ephemeral SRAM scrambling key derivation and the LFSR counters for background checks. +It is comprised of the `otp_edn_o` and `otp_edn_i` signals and follows a req / ack protocol. + +See also [EDN documentation](../../../../ip/edn/README.md). + +### Power Manager Interface + +The power manager interface is comprised of three signals overall: an initialization request (`pwr_otp_i.otp_init`), an initialization done response (`pwr_otp_o.otp_done`) and an idle indicator (`pwr_otp_o.otp_idle`). + +The power manager asserts `pwr_otp_i.otp_init` in order to signal to the OTP controller that it can start initialization, and the OTP controller signals completion of the initialization sequence by asserting `pwr_otp_o.otp_done` (the signal will remain high until reset). + +The idle indication signal `pwr_otp_o.otp_idle` indicates whether there is an ongoing write operation in the Direct Access Interface (DAI) or Life Cycle Interface (LCI), and the power manager uses that indication to determine whether a power down request needs to be aborted. + +Since the power manager may run in a different clock domain, the `pwr_otp_i.otp_init` signal is synchronized within the OTP controller. +The power manager is responsible for synchronizing the `pwr_otp_o.otp_done` and `pwr_otp_o.otp_idle` signals. + +See also [power manager documentation](../../pwrmgr/README.md). + +### Life Cycle Interfaces + +The interface to the life cycle controller can be split into three functional sub-interfaces (vendor test, state output, state transitions), and these are explained in more detail below. +Note that the OTP and life cycle controllers are supposed to be in the same clock domain, hence no additional signal synchronization is required. +See also [life cycle controller documentation](../../../../ip/lc_ctrl/README.md) for more details. + +#### Vendor Test Signals + +The `lc_otp_vendor_test_i` and `lc_otp_vendor_test_o` signals are connected to a 32bit control and a 32bit status register in the life cycle TAP, respectively, and are directly routed to the `prim_otp` wrapper. +These control and status signals may be used by the silicon creator to exercise the OTP programming smoke checks on the VENDOR_TEST partition. +The signals are gated with the life cycle state inside the life cycle controller such that they do not have any effect in production life cycle states. + +#### State, Counter and Token Output + +After initialization, the life cycle partition contents, as well as the tokens and personalization status is output to the life cycle controller via the `otp_lc_data_o` struct. +The life cycle controller uses this information to determine the life cycle state, and steer the appropriate qualifier signals. +Some of these qualifier signals (`lc_dft_en_i`, `lc_creator_seed_sw_rw_en_i`, `lc_seed_hw_rd_en_i` and `lc_escalate_en_i`) are fed back to the OTP controller in order to ungate testing logic to the OTP macro; enable SW write access to the `SECRET2` partition; enable hardware read access to the root key in the `SECRET2` partition; or to push the OTP controller into escalation state. + +A possible sequence for the signals described is illustrated below. +```wavejson +{signal: [ + {name: 'clk_i', wave: 'p.................'}, + {name: 'otp_lc_data_o.valid', wave: '0.|...|.1.|...|...'}, + {name: 'otp_lc_data_o.state', wave: '03|...|...|...|...'}, + {name: 'otp_lc_data_o.count', wave: '03|...|...|...|...'}, + {}, + {name: 'otp_lc_data_o.test_unlock_token', wave: '0.|...|.3.|...|...'}, + {name: 'otp_lc_data_o.test_exit_token', wave: '0.|...|.3.|...|...'}, + {name: 'otp_lc_data_o.test_tokens_valid', wave: '0.|...|.3.|...|...'}, + {}, + {name: 'otp_lc_data_o.rma_token', wave: '0.|.3.|...|...|...'}, + {name: 'otp_lc_data_o.rma_token_valid', wave: '0.|.3.|...|...|...'}, + {}, + {name: 'otp_lc_data_o.secrets_valid', wave: '0.|.3.|...|...|...'}, + {}, + {name: 'lc_creator_seed_sw_rw_en_i', wave: '0.|...|...|.4.|...'}, + {name: 'lc_seed_hw_rd_en_i', wave: '0.|...|...|.4.|...'}, + {name: 'lc_dft_en_i', wave: '0.|...|...|.4.|...'}, + {}, + {name: 'lc_escalate_en_i', wave: '0.|...|...|...|.5.'}, +]} +``` + +Note that the `otp_lc_data_o.valid` signal is only asserted after the `LIFE_CYCLE`, `SECRET0` and `SECRET2` partitions have successfully initialized, since the life cycle collateral contains information from all three partitions. +The `otp_lc_data_o.test_tokens_valid` and `otp_lc_data_o.rma_token_valid` signals are multibit valid signals indicating whether the corresponding tokens are valid. +The ``otp_lc_data_o.secrets_valid`` signal is a multibit valid signal that is set to `lc_ctrl_pkg::On` iff the `SECRET2` partition containing the root keys has been locked with a digest. + + +#### State Transitions + +In order to perform life cycle state transitions, the life cycle controller can present the new value of the life cycle state and counter via the programming interface as shown below: + +```wavejson +{signal: [ + {name: 'clk_i', wave: 'p.......'}, + {name: 'lc_otp_program_i.req', wave: '01.|..0.'}, + {name: 'lc_otp_program_i.state', wave: '03.|..0.'}, + {name: 'lc_otp_program_i.count', wave: '03.|..0.'}, + {name: 'lc_otp_program_o.ack', wave: '0..|.10.'}, + {name: 'lc_otp_program_o.err', wave: '0..|.40.'}, +]} +``` + +The request must remain asserted until the life cycle controller has responded. +An error is fatal and indicates that the OTP programming operation has failed. + +Note that the new state must not clear any bits that have already been programmed to OTP - i.e., the new state must be incrementally programmable on top of the previous state. +There are hence some implications on the life cycle encoding due to the ECC employed, see [life cycle state encoding](../../../../ip/lc_ctrl/README.md#life-cycle-manufacturing-state-encodings) for details. + +Note that the behavior of the `lc_otp_program_i.otp_test_ctrl` signal is vendor-specific, and hence the signal is set to `x` in the timing diagram above. +The purpose of this signal is to control vendor-specific test mechanisms, and its value will only be forwarded to the OTP macro in RAW, TEST_* and RMA states. +In all other life cycle states this signal will be clamped to zero. + +### Interface to Key Manager + +The interface to the key manager is a simple struct that outputs the CREATOR_ROOT_KEY_SHARE0 and CREATOR_ROOT_KEY_SHARE1 keys via `otp_keymgr_key_o` if these secrets have been provisioned and locked (via CREATOR_KEY_LOCK). +Otherwise, this signal is tied to a random netlist constant. + +Since the key manager may run in a different clock domain, key manager is responsible for synchronizing the `otp_keymgr_key_o` signals. + +### Interface to Flash Scrambler + +The interface to the FLASH scrambling device is a simple req/ack interface that provides the flash controller with the two 128bit keys for data and address scrambling. + +The keys can be requested as illustrated below: + +```wavejson +{signal: [ + {name: 'clk_i', wave: 'p...........'}, + {name: 'flash_otp_key_i.data_req', wave: '01.|..0.|...'}, + {name: 'flash_otp_key_i.addr_req', wave: '01.|....|..0'}, + {name: 'flash_otp_key_o.data_ack', wave: '0..|.10.|...'}, + {name: 'flash_otp_key_o.addr_ack', wave: '0..|....|.10'}, + {name: 'flash_otp_key_o.key', wave: '0..|.30.|.40'}, + {name: 'flash_otp_key_o.seed_valid', wave: '0..|.10.|.10'}, +]} +``` + +The keys are derived from the FLASH_DATA_KEY_SEED and FLASH_ADDR_KEY_SEED values stored in the `SECRET1` partition using the [scrambling primitive](#scrambling-datapath). +If the key seeds have not yet been provisioned, the keys are derived from all-zero constants, and the `flash_otp_key_o.seed_valid` signal will be set to 0 in the response. +The resulting scrambling key is still ephemeral (i.e., it is derived using entropy from CSRNG) and okay to be used. + +Note that the req/ack protocol runs on the OTP clock. +It is the task of the scrambling device to synchronize the handshake protocol by instantiating the `prim_sync_reqack.sv` primitive as shown below. + +![OTP Key Req Ack](../doc/otp_ctrl_key_req_ack.svg) + +Note that the key and nonce output signals on the OTP controller side are guaranteed to remain stable for at least 62 OTP clock cycles after the `ack` signal is pulsed high, because the derivation of a 64bit half-key takes at least two passes through the 31-cycle PRESENT primitive. +Hence, if the scrambling device clock is faster or in the same order of magnitude as the OTP clock, the data can be directly sampled upon assertion of `src_ack_o`. +If the scrambling device runs on a significantly slower clock than OTP, an additional register (as indicated with dashed grey lines in the figure) has to be added. + +### Interfaces to SRAM and OTBN Scramblers + +The interfaces to the SRAM and OTBN scrambling devices follow a req / ack protocol, where the scrambling device first requests a new ephemeral key by asserting the request channel (`sram_otp_key_i[*]`, `otbn_otp_key_i`). +The OTP controller then fetches entropy from EDN and derives an ephemeral key using the SRAM_DATA_KEY_SEED and the [PRESENT scrambling data path](#scrambling-datapath). +Finally, the OTP controller returns a fresh ephemeral key via the response channels (`sram_otp_key_o[*]`, `otbn_otp_key_o`), which complete the req / ack handshake. +The wave diagram below illustrates this process for the OTBN scrambling device. + +```wavejson +{signal: [ + {name: 'clk_i', wave: 'p.......'}, + {name: 'otbn_otp_key_i.req', wave: '01.|..0.'}, + {name: 'otbn_otp_key_o.ack', wave: '0..|.10.'}, + {name: 'otbn_otp_key_o.nonce', wave: '0..|.30.'}, + {name: 'otbn_otp_key_o.key', wave: '0..|.30.'}, + {name: 'otbn_otp_key_o.seed_valid', wave: '0..|.10.'}, +]} +``` + +If the key seeds have not yet been provisioned, the keys are derived from all-zero constants, and the `*.seed_valid` signal will be set to 0 in the response. +The resulting scrambling key is still ephemeral (i.e., it is derived using entropy from CSRNG) and okay to be used. +It should be noted that this mechanism requires the EDN and entropy distribution network to be operational, and a key derivation request will block if they are not. + +Note that the req/ack protocol runs on the OTP clock. +It is the task of the scrambling device to perform the synchronization as described in the previous subsection on the [flash scrambler interface](#interface-to-flash-scrambler). + +### Hardware Config Bits + +The bits of the HW_CFG* partitions are output via the `otp_ctrl_otp_broadcast_o` struct. +IPs that consume collateral stored in this partition shall connect to this struct via the topgen feature, and break out the appropriate bits by either accessing the correct index or using the struct fields. +These fields are autogenerated from the memory map items allocated to the HW_CFG* partitions, and the autogenerated struct type can be found in the `otp_ctrl_part_pkg.sv` package. +Note that it is the task of the receiving IP to synchronize these bits accordingly to the local clock. +For convenience, a valid bit is also available in that struct. +The valid bit indicates that the HW_CFG* partitions have initialized. + +## Parameter and Memory Map Changes after D3/V3 + +Note that all instantiation parameters can be changed without affecting D3/V3 status of the module. +Similarly, it is permissible to change the contents (partition size, adding and removing items) of the `CREATOR_SW_CFG`, `OWNER_SW_CFG` and `HW_CFG*` partitions without affecting D3 status. +Note however that partition size changes may affect V3 coverage metrics, hence if the size any of the above three partitions is changed, V3 needs to be re-assessed. diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_behavioral_model.svg b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_behavioral_model.svg new file mode 100644 index 00000000000000..a5dab4a4bfc212 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_behavioral_model.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_blockdiag.svg b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_blockdiag.svg new file mode 100644 index 00000000000000..f2169883ce4b2b --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_blockdiag.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg new file mode 100644 index 00000000000000..d06037468e3c21 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_dai_fsm.svg b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_dai_fsm.svg new file mode 100644 index 00000000000000..f84a80cc0f0f0b --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_dai_fsm.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_digest_mechanism.svg b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_digest_mechanism.svg new file mode 100644 index 00000000000000..dac2323afce333 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_digest_mechanism.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_digests.md b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_digests.md new file mode 100644 index 00000000000000..ddb4029750b9d6 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_digests.md @@ -0,0 +1,17 @@ + + +| Digest Name | Affected Partition | Calculated by HW | +|:---------------------------------------------------------------------------:|:-------------------------:|:------------------:| +| [VENDOR_TEST_DIGEST](#Reg_vendor_test_digest_0) | VENDOR_TEST | no | +| [CREATOR_SW_CFG_DIGEST](#Reg_creator_sw_cfg_digest_0) | CREATOR_SW_CFG | no | +| [OWNER_SW_CFG_DIGEST](#Reg_owner_sw_cfg_digest_0) | OWNER_SW_CFG | no | +| [ROT_CREATOR_AUTH_CODESIGN_DIGEST](#Reg_rot_creator_auth_codesign_digest_0) | ROT_CREATOR_AUTH_CODESIGN | no | +| [ROT_CREATOR_AUTH_STATE_DIGEST](#Reg_rot_creator_auth_state_digest_0) | ROT_CREATOR_AUTH_STATE | no | +| [HW_CFG0_DIGEST](#Reg_hw_cfg0_digest_0) | HW_CFG0 | yes | +| [HW_CFG1_DIGEST](#Reg_hw_cfg1_digest_0) | HW_CFG1 | yes | +| [SECRET0_DIGEST](#Reg_secret0_digest_0) | SECRET0 | yes | +| [SECRET1_DIGEST](#Reg_secret1_digest_0) | SECRET1 | yes | +| [SECRET2_DIGEST](#Reg_secret2_digest_0) | SECRET2 | yes | diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_field_descriptions.md b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_field_descriptions.md new file mode 100644 index 00000000000000..700506ae7c1b75 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_field_descriptions.md @@ -0,0 +1,101 @@ + + +| Partition | Item | Size [B] | Description | +|:-------------------------:|:-----------------------------------------------:|:----------:|:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| CREATOR_SW_CFG | CREATOR_SW_CFG_AST_CFG | 156 | AST configuration data. These values get blindly copied to the AST CSRs during ROM execution. | +| | CREATOR_SW_CFG_AST_INIT_EN | 4 | Controls whether or not the CREATOR_SW_CFG_AST_CFG values get copied to the AST CSRs during ROM execution. A value of kMultiBitBool4True enables copying; all other values disable. | +| | CREATOR_SW_CFG_ROM_EXT_SKU | 4 | SKU identifier metadata. Unused by SiliconCreator software. | +| | CREATOR_SW_CFG_SIGVERIFY_SPX_EN | 4 | Controls whether or not SPHINCS+ signature verification will be executed when the ROM attempts to boot the ROM_EXT. A value of kSigverifySpxDisabledOtp disables SPHINCS+ signature verification, while all other values enable it. Note, SPHINCS+ signature verification is always disabled in TEST_UNLOCKED* LC states. | +| | CREATOR_SW_CFG_FLASH_DATA_DEFAULT_CFG | 4 | The default scrambling, ECC, and high endurance configuration settings for flash data pages set during ROM execution. | +| | CREATOR_SW_CFG_FLASH_INFO_BOOT_DATA_CFG | 4 | The scrambling, ECC, and high endurance configuration settings for the boot data flash info pages set during ROM execution. | +| | CREATOR_SW_CFG_FLASH_HW_INFO_CFG_OVERRIDE | 4 | Scrambling and ECC configuration overrides (set by the ROM) for the Creator and OwnerSeed flash info pages (pages 1 and 2 in partition 0) that are hardwired to the keygmr. By default, scrambling and ECC on these flash info pages must be enabled in order to successfully crank the keygmr (in the ROM_EXT). However, values of kMultiBitBool4True for each subfield in this field will disable this requirement. See the HW_INFO_CFG_OVERRIDE flash_ctrl CSR for more details. | +| | CREATOR_SW_CFG_RNG_EN | 4 | Whether or not to enable use of hardware generated entropy (from the entropy complex via EDN) in the `rnd_uint32` function. A value of kHardenedBoolTrue enables the use of hardware generated entropy, while all other values disable. | +| | CREATOR_SW_CFG_JITTER_EN | 4 | Whether or not to enable clock jitter. A value of kMultiBitBool4False disables, while all other values enable. | +| | CREATOR_SW_CFG_RET_RAM_RESET_MASK | 4 | Reset reason mask used to initialize (by overwriting with random data) retention SRAM during ROM execution. A value of 0 only initializes retention SRAM on power-on-resets.See rstmgr RESET_INFO CSR documentation for more details. | +| | CREATOR_SW_CFG_MANUF_STATE | 4 | Manufacturing state binding field. For use by SiliconCreators or SiliconOwners to bind ROM_EXT images to a specific device or set of devices. | +| | CREATOR_SW_CFG_ROM_EXEC_EN | 4 | Whether or not to enable execution of ROM. A value of 0 disables, while all other values enable. This enables provisioning flows to attach JTAG connections and halt the CPU before the device has been fully provisioned. All SKUs should set this field to a non-zero value. Provisioning flows shall take care to program this field at the appropriate time during. | +| | CREATOR_SW_CFG_CPUCTRL | 4 | Value to write to the Ibex CPUCTRL CSR during ROM execution. This field controls settings such as ICACHE enablement. See Ibex documentation for more information. | +| | CREATOR_SW_CFG_MIN_SEC_VER_ROM_EXT | 4 | Value of the min_security_version_rom_ext field of the default boot data. | +| | CREATOR_SW_CFG_MIN_SEC_VER_BL0 | 4 | Value of the min_security_version_bl0 field of the default boot data. | +| | CREATOR_SW_CFG_DEFAULT_BOOT_DATA_IN_PROD_EN | 4 | Whether or not to enable the default boot data in PROD and PROD_END life cycle states. A value of kHardenedBoolTrue enables, all other values disable. If left disabled, provisioning flows are required to setup boot data pages prior to enabling ROM execution. | +| | CREATOR_SW_CFG_RMA_SPIN_EN | 4 | Whether or not to enable a busy-wait delay loop in the ROM, when a specific SW strapping configuration is applied during boot, to provide time to trigger an RMA lifecycle transition over JTAG. A value of kHardenedBoolTrue enables, all other values disable. | +| | CREATOR_SW_CFG_RMA_SPIN_CYCLES | 4 | The number of Ibex clock cycles to spin for when waiting for an RMA transition. Used in combination with the CREATOR_SW_CFG_RMA_SPIN_EN field. | +| | CREATOR_SW_CFG_RNG_REPCNT_THRESHOLDS | 4 | The repetition count health test thresholds to enable entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_REPCNTS_THRESHOLDS | 4 | The repetition count symbol health test thresholds to enable entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_ADAPTP_HI_THRESHOLDS | 4 | The adaptive proportion health test high thresholds to enable entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_ADAPTP_LO_THRESHOLDS | 4 | The adaptive proportion health test low thresholds to enable entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_BUCKET_THRESHOLDS | 4 | The bucket health test thresholds to enable entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_MARKOV_HI_THRESHOLDS | 4 | The Markov health test high thresholds to enable entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_MARKOV_LO_THRESHOLDS | 4 | The Markov health test low thresholds to enable entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_EXTHT_HI_THRESHOLDS | 4 | The external health test high thresholds to enable the entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_EXTHT_LO_THRESHOLDS | 4 | The external health test low thresholds to enable the entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_ALERT_THRESHOLD | 4 | The alert threshold to configure the entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_HEALTH_CONFIG_DIGEST | 4 | A CRC32 digest of all entropy_src health test threshold configuration fields above. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_SRAM_KEY_RENEW_EN | 4 | Whether or not the ROM should request SRAM to be rescrambled with a new key on every boot. kHardenedBoolFalse disables, while all other values enable. | +| | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_EN | 4 | Enablement of the ROM_EXT immutable code section. A value of kHardenedBoolTrue enables the feature. All other values disable it. | +| | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_START_OFFSET | 4 | Relative offset from the start of the ROM_EXT slot to find the immutable code section at. | +| | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_LENGTH | 4 | Length (in bytes) of the immutable code section. | +| | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_SHA256_HASH | 32 | SHA256 hash of the immutable ROM_EXT section. | +| | CREATOR_SW_CFG_RESERVED | 32 | Unused bits in the CREATOR_SW_CFG OTP partition. These can be claimed by software as needed. | +| OWNER_SW_CFG | OWNER_SW_CFG_ROM_ERROR_REPORTING | 4 | The shutdown error reporting verbosity used by the ROM. Should be configured to one of several `shutdown_error_redact_t` values. See `sw/device/silicon_creator/lib/shutdown.h` for more details. | +| | OWNER_SW_CFG_ROM_BOOTSTRAP_DIS | 4 | Whether or not to disable ROM bootstrap mechanism. A value of kHardenedBoolTrue disable bootstrap mechanism in the ROM, while all other values enable it. Note, the provisioning flow should take care when to program this field if it is used by a SKU, as there is no way to get firmware into flash in a PROD LC state if a valid ROM_EXT does not already exist in flash once this value is configured to true. | +| | OWNER_SW_CFG_ROM_ALERT_CLASS_EN | 4 | A four byte packed field, where each byte controls whether or not the ROM enables each alert class (A through D) of the alert_handler. The byte-sized subfields are arranged from D to A, MSB to LSB. Each byte should be set to an `alert_enable_t` value accordingly. See the alert_handler documentation for more details. | +| | OWNER_SW_CFG_ROM_ALERT_ESCALATION | 4 | A four byte packed field, where each byte controls the escalation configuration for each alert class (A through D) of the alert_handler configured by the ROM. The byte-sized subfields are arranged from D to A, MSB to LSB. Each byte should be set to an `alert_escalate_t` value accordingly. See the alert_handler documentation for more details. | +| | OWNER_SW_CFG_ROM_ALERT_CLASSIFICATION | 320 | The alert classifications (A through D) for each alert source of the alert_handler to be configured by the ROM. The field consists of a contiguous 320-byte block, or 80 32-bit words. The four bytes in each word encode the configuration of a single alert source across four lifecycle states, in order from LSB to MSB: PROD, PROD_END, DEV, and RMA. Each byte should be set to an `alert_class_t` value accordingly. The order of the 80 32-bit words, from LSB to MSB can be found in the EARLGREY_ALERTS list in `rules/const.bzl`. See the alert_handler documentation for more details. | +| | OWNER_SW_CFG_ROM_LOCAL_ALERT_CLASSIFICATION | 64 | Same as the `OWNER_SW_CFG_ROM_ALERT_CLASSIFICATION` field, except these configuration correspond to the local alert sources found in the `EARLGREY_LOC_ALERTS list in `rules/const.bzl`. | +| | OWNER_SW_CFG_ROM_ALERT_ACCUM_THRESH | 16 | The alert accumulation threshold values for each alert class (A through D) of the alert_handler to be configured by the ROM. This field consists of four 32-bit words encoding the accumulation thresholds for each alert class A through D arranged LSW to MSW. See the alert_handler documentation for more details. | +| | OWNER_SW_CFG_ROM_ALERT_TIMEOUT_CYCLES | 16 | Same as the `OWNER_SW_CFG_ROM_ALERT_ACCUM_THRESH` field, except each value corresponds to the interrupt timeout configuration of an alert class. | +| | OWNER_SW_CFG_ROM_ALERT_PHASE_CYCLES | 64 | The alert escalation phase durations, measured in clock cycles, the ROM will configure the four alert phases for each alert class of the alert_handler. This field consists of a contiguous 64-byte block, or an array of four 128-bit fields. Each 128-bit subfield encodes four 32-bit words that contain the alert phase cycle count configurations for alert escalation phases 0 to 3, from LSW to MSW. Each 128-bit subfield is contains all cofigurations for a single alert class, arranged from class A to D, from LS to MS. For example, the cycle durations of each escalation phase in this field should be configured as such, from LSB to MSB: ......... . See the alert_handler documentation for more details. | +| | OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD | 4 | The expected CRC32 digest over all of the alert_handler configurations set up by the ROM for a device operating in the PROD LC state. The ROM reads this field and checks it against a digest it computes over the alert_handler configuration it programmed. This field is expected to be automatically computed by the `otp_alert_digest()` Bazel rule. See the `alert_config_crc32()` function in the SiliconCreator alert_handler driver for more details on what configurations are included in this digest. | +| | OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD_END | 4 | Same as the `OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD` field, except the expected digest is for chips operating in the PROD_END LC state. | +| | OWNER_SW_CFG_ROM_ALERT_DIGEST_DEV | 4 | Same as the `OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD` field, except the expected digest is for chips operating in the DEV LC state. | +| | OWNER_SW_CFG_ROM_ALERT_DIGEST_RMA | 4 | Same as the `OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD` field, except the expected digest is for chips operating in the RMA LC state. | +| | OWNER_SW_CFG_ROM_WATCHDOG_BITE_THRESHOLD_CYCLES | 4 | Watchdog timer bite threshold (in cycles) configured by the ROM. | +| | OWNER_SW_CFG_ROM_KEYMGR_OTP_MEAS_EN | 4 | Whether or not to configure the attestation SW binding CSRs of the keymgr with the value in ROM_EXT manifest or the measurement of the OTP CreatorSwCfg, OwnerSwCfg, and secure boot key integrity digest. A value of kHardenedBoolTrue uses the ROM computed OTP measurements, while all other values trigger the use of the binding values included in the ROM_EXT manifest. | +| | OWNER_SW_CFG_MANUF_STATE | 4 | Manufacturing state binding field. For use by SiliconCreators or SiliconOwners to bind BL0 images to a specific device or set of devices. | +| | OWNER_SW_CFG_ROM_RSTMGR_INFO_EN | 4 | A two byte packed word that indicates the expected rstmgr alert and CPU info dump enable states, configured in the rstmgr's ALERT_INFO_CTRL and CPU_INFO_CTRL CSRs respectively. The expected format of this fields is {0,0,kHardenedBool*,kHardenedBool*}, read MSB to LSB, where the left most kHardenedBool* entry indicates the expected enablement state of the ALERT_INFO_CTRL, and the right most indicates the enablement state of the CPU_INFO_CTRL. Since the ROM expects both to be disabled upon handing over execution control to the ROM_EXT, this entire OTP field **should be left unprovisioned, or set to all 0**. | +| | OWNER_SW_CFG_ROM_EXT_BOOTSTRAP_EN | 4 | Unused. Set to 0. | +| | OWNER_SW_CFG_ROM_SENSOR_CTRL_ALERT_CFG | 12 | Alert configuration values for the sensor_ctrl block that will be configured by the ROM. This field is 12 bytes long, where each byte contains two 4-bit packed subfields, encoding two four bit boolean values (kMultiBitBool4False or kMultiBitBool4True), as follows (read MSB to LSB): {fatality, enablement}. For example, the byte 0x69 would encode the alert is: 1) recoverable, and 2) disabled. Currently, there are only 11 alerts in sensor_ctrl to configure, thus only the least significant bytes in this field are used. | +| | OWNER_SW_CFG_ROM_SRAM_READBACK_EN | 4 | This field encodes the enablements of the readback security features for the main and retention SRAMs to be configured by the ROM. This field is four bytes, but the LSB contains two 4-bit packed kMultiBitBool4* values that indicate the enablement of the feature for the retention SRAM and main SRAM, from MSB to LSB respectively. See the READBACK CSR of the sram_ctrl for more details on this feature. | +| | OWNER_SW_CFG_ROM_PRESERVE_RESET_REASON_EN | 4 | Whether or not the ROM should preserve the reset reasons CSR state in the rstmgr, or clear it. A value of kHardenedBoolTrue preserves the CSR, while other values trigger the ROM to clear the CSR after copying the reason to the retention SRAM. | +| | OWNER_SW_CFG_ROM_RESET_REASON_CHECK_VALUE | 4 | Two packed 16-bit values that indicate whether the ROM should perform a validation check on the the reset reasons during boot. The validation check is a hardening mechanism that checks the reset reasons for consistency at two different points in time. Both packed values should be the same. Values of kHardenedBoolFalse will instruct the ROM to skip the reset reasons validation check, while all other values will instruct the ROM to perform the check. | +| | OWNER_SW_CFG_ROM_BANNER_EN | 4 | Whether or not the ROM should print a banner message to the console UART during boot. A value of kHardenedBoolFalse disables the message printing, while all other values enable it. | +| | OWNER_SW_CFG_ROM_FLASH_ECC_EXC_HANDLER_EN | 4 | Whether or not the ROM should use the flash ECC exception handler during execution. A value of kHardenedBoolTrue allows the ROM to use an the exception handler that recovers gracefully and continues the boot process if a flash ECC error is encountered during verification of a specific ROM_EXT slot. This enables the ROM to attempt booting the next ROM_EXT slot if the first slot attempted has been corrupted. All other values trigger default ROM exception handling, which is all exceptions trigger a chip shutdown and reset. | +| | OWNER_SW_CFG_RESERVED | 128 | Unused bits in the OWNER_SW_CFG OTP partition. These can be claimed by software as needed. | +| ROT_CREATOR_AUTH_CODESIGN | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE0 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY0 | 64 | | +| | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE1 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY1 | 64 | | +| | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE2 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY2 | 64 | | +| | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE3 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY3 | 64 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE0 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY0 | 32 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG0 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE1 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY1 | 32 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG1 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE2 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY2 | 32 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG2 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE3 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY3 | 32 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG3 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_BLOCK_SHA2_256_HASH | 32 | | +| ROT_CREATOR_AUTH_STATE | ROT_CREATOR_AUTH_STATE_ECDSA_KEY0 | 4 | | +| | ROT_CREATOR_AUTH_STATE_ECDSA_KEY1 | 4 | | +| | ROT_CREATOR_AUTH_STATE_ECDSA_KEY2 | 4 | | +| | ROT_CREATOR_AUTH_STATE_ECDSA_KEY3 | 4 | | +| | ROT_CREATOR_AUTH_STATE_SPX_KEY0 | 4 | | +| | ROT_CREATOR_AUTH_STATE_SPX_KEY1 | 4 | | +| | ROT_CREATOR_AUTH_STATE_SPX_KEY2 | 4 | | +| | ROT_CREATOR_AUTH_STATE_SPX_KEY3 | 4 | | +| HW_CFG0 | DEVICE_ID | 32 | Unique device identifier that is always exposed through the lifecycle JTAG tap. See OpenTitan documentation on Device Identifiers. | +| | MANUF_STATE | 32 | Field to capture manufacturing status. Currently unused. Set to 0. | +| HW_CFG1 | EN_SRAM_IFETCH | 1 | Enablement of execute from SRAM switch in the sram_ctrl (see EXEC CSR). A kMultiBitBool8True value enables, while all other values disable. | +| | EN_CSRNG_SW_APP_READ | 1 | Enablement of CSRNG software application interface. A kMultiBitBool8True value enables, while all other values disable. Enablement is required to extract output from CSRNG via software. | +| | DIS_RV_DM_LATE_DEBUG | 1 | Disablement of RV_DM late debug feature (see rv_dm documentation). A kMultiBitBool8True disables the late debug feature and renders the rv_dm fully ungated in DEV lifecycle states. All other values gate rv_dm reachability based on the value of LATE_DEBUG_ENABLE CSR in the rv_dm block. | diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_kdi_fsm.svg b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_kdi_fsm.svg new file mode 100644 index 00000000000000..0f91a212c440d3 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_kdi_fsm.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_key_req_ack.svg b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_key_req_ack.svg new file mode 100644 index 00000000000000..c5cd58b5c1578b --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_key_req_ack.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_lci_fsm.svg b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_lci_fsm.svg new file mode 100644 index 00000000000000..63459e87af287b --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_lci_fsm.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_mmap.md b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_mmap.md new file mode 100644 index 00000000000000..ee030099dbeeea --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_mmap.md @@ -0,0 +1,122 @@ + + +| Index | Partition | Size [B] | Access Granule | Item | Byte Address | Size [B] | +|:-------:|:-------------------------:|:----------:|:----------------:|:---------------------------------------------------------------------------:|:--------------:|:----------:| +| 0 | VENDOR_TEST | 64 | 32bit | SCRATCH | 0x000 | 56 | +| | | | 64bit | [VENDOR_TEST_DIGEST](#Reg_vendor_test_digest_0) | 0x038 | 8 | +| 1 | CREATOR_SW_CFG | 368 | 32bit | CREATOR_SW_CFG_AST_CFG | 0x040 | 156 | +| | | | 32bit | CREATOR_SW_CFG_AST_INIT_EN | 0x0DC | 4 | +| | | | 32bit | CREATOR_SW_CFG_ROM_EXT_SKU | 0x0E0 | 4 | +| | | | 32bit | CREATOR_SW_CFG_SIGVERIFY_SPX_EN | 0x0E4 | 4 | +| | | | 32bit | CREATOR_SW_CFG_FLASH_DATA_DEFAULT_CFG | 0x0E8 | 4 | +| | | | 32bit | CREATOR_SW_CFG_FLASH_INFO_BOOT_DATA_CFG | 0x0EC | 4 | +| | | | 32bit | CREATOR_SW_CFG_FLASH_HW_INFO_CFG_OVERRIDE | 0x0F0 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_EN | 0x0F4 | 4 | +| | | | 32bit | CREATOR_SW_CFG_JITTER_EN | 0x0F8 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RET_RAM_RESET_MASK | 0x0FC | 4 | +| | | | 32bit | CREATOR_SW_CFG_MANUF_STATE | 0x100 | 4 | +| | | | 32bit | CREATOR_SW_CFG_ROM_EXEC_EN | 0x104 | 4 | +| | | | 32bit | CREATOR_SW_CFG_CPUCTRL | 0x108 | 4 | +| | | | 32bit | CREATOR_SW_CFG_MIN_SEC_VER_ROM_EXT | 0x10C | 4 | +| | | | 32bit | CREATOR_SW_CFG_MIN_SEC_VER_BL0 | 0x110 | 4 | +| | | | 32bit | CREATOR_SW_CFG_DEFAULT_BOOT_DATA_IN_PROD_EN | 0x114 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RMA_SPIN_EN | 0x118 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RMA_SPIN_CYCLES | 0x11C | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_REPCNT_THRESHOLDS | 0x120 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_REPCNTS_THRESHOLDS | 0x124 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_ADAPTP_HI_THRESHOLDS | 0x128 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_ADAPTP_LO_THRESHOLDS | 0x12C | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_BUCKET_THRESHOLDS | 0x130 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_MARKOV_HI_THRESHOLDS | 0x134 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_MARKOV_LO_THRESHOLDS | 0x138 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_EXTHT_HI_THRESHOLDS | 0x13C | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_EXTHT_LO_THRESHOLDS | 0x140 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_ALERT_THRESHOLD | 0x144 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_HEALTH_CONFIG_DIGEST | 0x148 | 4 | +| | | | 32bit | CREATOR_SW_CFG_SRAM_KEY_RENEW_EN | 0x14C | 4 | +| | | | 32bit | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_EN | 0x150 | 4 | +| | | | 32bit | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_START_OFFSET | 0x154 | 4 | +| | | | 32bit | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_LENGTH | 0x158 | 4 | +| | | | 32bit | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_SHA256_HASH | 0x15C | 32 | +| | | | 32bit | CREATOR_SW_CFG_RESERVED | 0x17C | 32 | +| | | | 64bit | [CREATOR_SW_CFG_DIGEST](#Reg_creator_sw_cfg_digest_0) | 0x1A8 | 8 | +| 2 | OWNER_SW_CFG | 712 | 32bit | OWNER_SW_CFG_ROM_ERROR_REPORTING | 0x1B0 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_BOOTSTRAP_DIS | 0x1B4 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_CLASS_EN | 0x1B8 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_ESCALATION | 0x1BC | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_CLASSIFICATION | 0x1C0 | 320 | +| | | | 32bit | OWNER_SW_CFG_ROM_LOCAL_ALERT_CLASSIFICATION | 0x300 | 64 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_ACCUM_THRESH | 0x340 | 16 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_TIMEOUT_CYCLES | 0x350 | 16 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_PHASE_CYCLES | 0x360 | 64 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD | 0x3A0 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD_END | 0x3A4 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_DIGEST_DEV | 0x3A8 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_DIGEST_RMA | 0x3AC | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_WATCHDOG_BITE_THRESHOLD_CYCLES | 0x3B0 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_KEYMGR_OTP_MEAS_EN | 0x3B4 | 4 | +| | | | 32bit | OWNER_SW_CFG_MANUF_STATE | 0x3B8 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_RSTMGR_INFO_EN | 0x3BC | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_EXT_BOOTSTRAP_EN | 0x3C0 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_SENSOR_CTRL_ALERT_CFG | 0x3C4 | 12 | +| | | | 32bit | OWNER_SW_CFG_ROM_SRAM_READBACK_EN | 0x3D0 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_PRESERVE_RESET_REASON_EN | 0x3D4 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_RESET_REASON_CHECK_VALUE | 0x3D8 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_BANNER_EN | 0x3DC | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_FLASH_ECC_EXC_HANDLER_EN | 0x3E0 | 4 | +| | | | 32bit | OWNER_SW_CFG_RESERVED | 0x3E4 | 128 | +| | | | 64bit | [OWNER_SW_CFG_DIGEST](#Reg_owner_sw_cfg_digest_0) | 0x470 | 8 | +| 3 | ROT_CREATOR_AUTH_CODESIGN | 472 | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE0 | 0x478 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY0 | 0x47C | 64 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE1 | 0x4BC | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY1 | 0x4C0 | 64 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE2 | 0x500 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY2 | 0x504 | 64 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE3 | 0x544 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY3 | 0x548 | 64 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE0 | 0x588 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY0 | 0x58C | 32 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG0 | 0x5AC | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE1 | 0x5B0 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY1 | 0x5B4 | 32 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG1 | 0x5D4 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE2 | 0x5D8 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY2 | 0x5DC | 32 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG2 | 0x5FC | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE3 | 0x600 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY3 | 0x604 | 32 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG3 | 0x624 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_BLOCK_SHA2_256_HASH | 0x628 | 32 | +| | | | 64bit | [ROT_CREATOR_AUTH_CODESIGN_DIGEST](#Reg_rot_creator_auth_codesign_digest_0) | 0x648 | 8 | +| 4 | ROT_CREATOR_AUTH_STATE | 40 | 32bit | ROT_CREATOR_AUTH_STATE_ECDSA_KEY0 | 0x650 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_STATE_ECDSA_KEY1 | 0x654 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_STATE_ECDSA_KEY2 | 0x658 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_STATE_ECDSA_KEY3 | 0x65C | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_STATE_SPX_KEY0 | 0x660 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_STATE_SPX_KEY1 | 0x664 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_STATE_SPX_KEY2 | 0x668 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_STATE_SPX_KEY3 | 0x66C | 4 | +| | | | 64bit | [ROT_CREATOR_AUTH_STATE_DIGEST](#Reg_rot_creator_auth_state_digest_0) | 0x670 | 8 | +| 5 | HW_CFG0 | 72 | 32bit | DEVICE_ID | 0x678 | 32 | +| | | | 32bit | MANUF_STATE | 0x698 | 32 | +| | | | 64bit | [HW_CFG0_DIGEST](#Reg_hw_cfg0_digest_0) | 0x6B8 | 8 | +| 6 | HW_CFG1 | 16 | 32bit | EN_SRAM_IFETCH | 0x6C0 | 1 | +| | | | 32bit | EN_CSRNG_SW_APP_READ | 0x6C1 | 1 | +| | | | 32bit | DIS_RV_DM_LATE_DEBUG | 0x6C2 | 1 | +| | | | 64bit | [HW_CFG1_DIGEST](#Reg_hw_cfg1_digest_0) | 0x6C8 | 8 | +| 7 | SECRET0 | 40 | 64bit | TEST_UNLOCK_TOKEN | 0x6D0 | 16 | +| | | | 64bit | TEST_EXIT_TOKEN | 0x6E0 | 16 | +| | | | 64bit | [SECRET0_DIGEST](#Reg_secret0_digest_0) | 0x6F0 | 8 | +| 8 | SECRET1 | 88 | 64bit | FLASH_ADDR_KEY_SEED | 0x6F8 | 32 | +| | | | 64bit | FLASH_DATA_KEY_SEED | 0x718 | 32 | +| | | | 64bit | SRAM_DATA_KEY_SEED | 0x738 | 16 | +| | | | 64bit | [SECRET1_DIGEST](#Reg_secret1_digest_0) | 0x748 | 8 | +| 9 | SECRET2 | 88 | 64bit | RMA_TOKEN | 0x750 | 16 | +| | | | 64bit | CREATOR_ROOT_KEY_SHARE0 | 0x760 | 32 | +| | | | 64bit | CREATOR_ROOT_KEY_SHARE1 | 0x780 | 32 | +| | | | 64bit | [SECRET2_DIGEST](#Reg_secret2_digest_0) | 0x7A0 | 8 | +| 10 | LIFE_CYCLE | 88 | 32bit | LC_TRANSITION_CNT | 0x7A8 | 48 | +| | | | 32bit | LC_STATE | 0x7D8 | 40 | diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_overview.svg b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_overview.svg new file mode 100644 index 00000000000000..ef2adaf1f99f82 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_overview.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_partitions.md b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_partitions.md new file mode 100644 index 00000000000000..e19aad14717b80 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_partitions.md @@ -0,0 +1,57 @@ + + +| Partition | Secret | Buffered | Integrity | WR Lockable | RD Lockable | Description | +|:-------------------------:|:--------:|:----------:|:-----------:|:-------------:|:-------------:|:--------------------------------------------------------------------| +| VENDOR_TEST | no | no | no | yes (Digest) | yes (CSR) | Vendor test partition. | +| | | | | | | This is reserved for manufacturing smoke checks. The OTP wrapper | +| | | | | | | control logic inside prim_otp is allowed to read/write to this | +| | | | | | | region. ECC uncorrectable errors seen on the functional prim_otp | +| | | | | | | interface will not lead to an alert for this partition. | +| | | | | | | Instead, such errors will be reported as correctable ECC errors. | +| CREATOR_SW_CFG | no | no | yes | yes (Digest) | yes (CSR) | Software configuration partition. | +| | | | | | | This is for device-specific calibration data, e.g, clock, LDO, RNG, | +| | | | | | | and configuration settings set by the ROM. | +| OWNER_SW_CFG | no | no | yes | yes (Digest) | yes (CSR) | Software configuration partition. | +| | | | | | | This contains data that changes software behavior in the ROM, for | +| | | | | | | example enabling defensive features in ROM or selecting failure | +| | | | | | | modes if verification fails. | +| ROT_CREATOR_AUTH_CODESIGN | no | no | yes | yes (Digest) | yes (CSR) | This OTP partition is used to store four P-256 keys | +| | | | | | | and four Sphincs+ keys. The partition requires 464 | +| | | | | | | bytes of software visible storage. The partition is | +| | | | | | | locked at manufacturing time to protect against | +| | | | | | | malicious write attempts. | +| ROT_CREATOR_AUTH_STATE | no | no | yes | yes (Digest) | yes (CSR) | This OTP partition is used to capture the state of | +| | | | | | | each key slot. Each key can be in one of the | +| | | | | | | following states: BLANK, ENABLED, DISABLED. The | +| | | | | | | encoded values are such that transitions between | +| | | | | | | BLANK -> ENABLED -> DISABLED are possible without | +| | | | | | | causing ECC errors (this is a mechanism similar to | +| | | | | | | how we manage life cycle state transitions). The | +| | | | | | | partition is left unlocked to allow STATE updates in | +| | | | | | | the field. The ROM_EXT is required to lock access to | +| | | | | | | the OTP Direct Access Interface to prevent DoS | +| | | | | | | attacks from malicious code executing on Silicon | +| | | | | | | Owner partitions. DAI write locking is available in | +| | | | | | | EarlGrey. | +| HW_CFG0 | no | yes | yes | yes (Digest) | no | Hardware configuration 0 partition. | +| | | | | | | This contains a device identifier and manufacturing state. | +| HW_CFG1 | no | yes | yes | yes (Digest) | no | Hardware configuration 1 partition. | +| | | | | | | This contains several hardware feature switches. | +| SECRET0 | yes | yes | yes | yes (Digest) | yes (Digest) | Secret partition 0. | +| | | | | | | This contains TEST lifecycle unlock tokens. | +| SECRET1 | yes | yes | yes | yes (Digest) | yes (Digest) | Secret partition 1. | +| | | | | | | This contains SRAM and flash scrambling keys. | +| SECRET2 | yes | yes | yes | yes (Digest) | yes (Digest) | Secret partition 2. | +| | | | | | | This contains RMA unlock token, creator root key, and creator seed. | +| LIFE_CYCLE | no | yes | yes | no | no | Lifecycle partition. | +| | | | | | | This contains lifecycle transition count and state. This partition | +| | | | | | | cannot be locked since the life cycle state needs to advance to RMA | +| | | | | | | in-field. Note that while this partition is not marked secret, it | +| | | | | | | is not readable nor writeable via the DAI. Only the LC controller | +| | | | | | | can access this partition, and even via the LC controller it is not | +| | | | | | | possible to read the raw manufacturing life cycle state in encoded | +| | | | | | | form, since that encoding is considered a netlist secret. The LC | +| | | | | | | controller only exposes a decoded version of this state. | diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_prim_otp.svg b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_prim_otp.svg new file mode 100644 index 00000000000000..df6b31f078ed95 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_prim_otp.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg new file mode 100644 index 00000000000000..cd1a7ca1d70528 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/programmers_guide.md b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/programmers_guide.md new file mode 100644 index 00000000000000..a3d00735f7668c --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/programmers_guide.md @@ -0,0 +1,261 @@ +# Programmer's Guide + +During provisioning and manufacturing, SW interacts with the OTP controller mostly through the Direct Access Interface (DAI), which is described below. +Afterwards during production, SW is expected to perform only read accesses via the exposed CSRs and CSR windows, since all write access to the partitions has been locked down. + +The following sections provide some general guidance, followed by an explanation of the DAI and a detailed OTP memory map. +Typical programming sequences are explained at the end of the Programmer's guide. + +## General Guidance + +### Initialization + +The OTP controller initializes automatically upon power-up and is fully operational by the time the processor boots. +The only initialization steps that SW should perform are: + +1. Check that the OTP controller has successfully initialized by reading [`STATUS`](registers.md#status). I.e., make sure that none of the ERROR bits are set, and that the DAI is idle ([`STATUS.DAI_IDLE`](registers.md#status)). +2. Set up the periodic background checks: + - Choose whether to enable periodic [background checks](#partition-checks) by programming nonzero mask values to [`INTEGRITY_CHECK_PERIOD`](registers.md#integrity_check_period) and [`CONSISTENCY_CHECK_PERIOD`](registers.md#consistency_check_period). + - Choose whether such checks shall be subject to a timeout by programming a nonzero timeout cycle count to [`CHECK_TIMEOUT`](registers.md#check_timeout). + - It is recommended to lock down the background check registers via [`CHECK_REGWEN`](registers.md#check_regwen), once the background checks have been set up. + +If needed, one-off integrity and consistency checks can be triggered via [`CHECK_TRIGGER`](registers.md#check_trigger). +If this functionality is not needed, it is recommended to lock down the trigger register via [`CHECK_TRIGGER_REGWEN`](registers.md#check_trigger_regwen). + +Later on during the boot process, SW may also choose to block read access to the SW managed partitions via the associated partition lock registers, e.g. [`CREATOR_SW_CFG_READ_LOCK`](registers.md#creator_sw_cfg_read_lock) or [`OWNER_SW_CFG_READ_LOCK`](registers.md#owner_sw_cfg_read_lock). + +### Reset Considerations + +It is important to note that values in OTP **can be corrupted** if a reset occurs during a programming operation. +This should be of minor concern for SW, however, since all partitions except for the LIFE_CYCLE partition are being provisioned in secure and controlled environments, and not in the field. +The LIFE_CYCLE partition is the only partition that is modified in the field - but that partition is entirely owned by the life cycle controller and not by SW. + +### Programming Already Programmed Regions + +OTP words cannot be programmed twice, and doing so may damage the memory array. +Hence the OTP controller performs a blank check and returns an error if a write operation is issued to an already programmed location. + +### Potential Side-Effects on Flash via Life Cycle + +It should be noted that the locked status of the partition holding the creator root key (i.e., the value of the [`SECRET2_DIGEST_0`](registers.md#secret2_digest)) determines the ID_STATUS of the device, which in turn determines SW accessibility of creator seed material in flash and OTP. +That means that creator-seed-related collateral needs to be provisioned to Flash **before** the OTP digest lockdown mechanism is triggered, since otherwise accessibility to the corresponding flash region is lost. +See the [life cycle controller documentation](../../../../ip/lc_ctrl/README.md#id-state-of-the-device) for more details. + +## Direct Access Interface + +OTP has to be programmed via the Direct Access Interface, which is comprised of the following CSRs: + +CSR Name | Description +-------------------------------------|------------------------------------ +[`DIRECT_ACCESS_WDATA_0`](registers.md#direct_access_wdata) | Low 32bit word to be written. +[`DIRECT_ACCESS_WDATA_1`](registers.md#direct_access_wdata) | High 32bit word to be written. +[`DIRECT_ACCESS_RDATA_0`](registers.md#direct_access_rdata) | Low 32bit word that has been read. +[`DIRECT_ACCESS_RDATA_1`](registers.md#direct_access_rdata) | High 32bit word that has been read. +[`DIRECT_ACCESS_ADDRESS`](registers.md#direct_access_address) | byte address for the access. +[`DIRECT_ACCESS_CMD`](registers.md#direct_access_cmd) | Command register to trigger a read or a write access. +[`DIRECT_ACCESS_REGWEN`](registers.md#direct_access_regwen) | Write protection register for DAI. + +See further below for a detailed [Memory Map](#direct-access-memory-map) of the address space accessible via the DAI. + +### Readout Sequence + +A typical readout sequence looks as follows: + +1. Check whether the DAI is idle by reading the [`STATUS`](registers.md#status) register. +2. Write the byte address for the access to [`DIRECT_ACCESS_ADDRESS`](registers.md#direct_access_address). +Note that the address is aligned with the granule, meaning that either 2 or 3 LSBs of the address are ignored, depending on whether the access granule is 32 or 64bit. +3. Trigger a read command by writing 0x1 to [`DIRECT_ACCESS_CMD`](registers.md#direct_access_cmd). +4. Poll the [`STATUS`](registers.md#status) until the DAI state goes back to idle. +Alternatively, the `otp_operation_done` interrupt can be enabled up to notify the processor once an access has completed. +5. If the status register flags a DAI error, additional handling is required (see [Section on Error handling](#error-handling)). +6. If the region accessed has a 32bit access granule, the 32bit chunk of read data can be read from [`DIRECT_ACCESS_RDATA_0`](registers.md#direct_access_rdata). +If the region accessed has a 64bit access granule, the 64bit chunk of read data can be read from the [`DIRECT_ACCESS_RDATA_0`](registers.md#direct_access_rdata) and [`DIRECT_ACCESS_RDATA_1`](registers.md#direct_access_rdata) registers. +7. Go back to 1. and repeat until all data has been read. + +The hardware will set [`DIRECT_ACCESS_REGWEN`](registers.md#direct_access_regwen) to 0x0 while an operation is pending in order to temporarily lock write access to the CSRs registers. + +### Programming Sequence + +A typical programming sequence looks as follows: + +1. Check whether the DAI is idle by reading the [`STATUS`](registers.md#status) register. +2. If the region to be accessed has a 32bit access granule, place a 32bit chunk of data into [`DIRECT_ACCESS_WDATA_0`](registers.md#direct_access_wdata). +If the region to be accessed has a 64bit access granule, both the [`DIRECT_ACCESS_WDATA_0`](registers.md#direct_access_wdata) and [`DIRECT_ACCESS_WDATA_1`](registers.md#direct_access_wdata) registers have to be used. +3. Write the byte address for the access to [`DIRECT_ACCESS_ADDRESS`](registers.md#direct_access_address). +Note that the address is aligned with the granule, meaning that either 2 or 3 LSBs of the address are ignored, depending on whether the access granule is 32 or 64bit. +4. Trigger a write command by writing 0x2 to [`DIRECT_ACCESS_CMD`](registers.md#direct_access_cmd). +5. Poll the [`STATUS`](registers.md#status) until the DAI state goes back to idle. +Alternatively, the `otp_operation_done` interrupt can be enabled up to notify the processor once an access has completed. +6. If the status register flags a DAI error, additional handling is required (see [Section on Error handling](#error-handling)). +7. Go back to 1. and repeat until all data has been written. + +The hardware will set [`DIRECT_ACCESS_REGWEN`](registers.md#direct_access_regwen) to 0x0 while an operation is pending in order to temporarily lock write access to the CSRs registers. + +Note that SW is responsible for keeping track of already programmed OTP word locations during the provisioning phase. +**It is imperative that SW does not write the same word location twice**, since this can lead to ECC inconsistencies, thereby potentially rendering the device useless. + +### Digest Calculation Sequence + +The hardware digest computation for the hardware and secret partitions can be triggered as follows: + +1. Check whether the DAI is idle by reading the [`STATUS`](registers.md#status) register. +3. Write the partition base address to [`DIRECT_ACCESS_ADDRESS`](registers.md#direct_access_address). +4. Trigger a digest calculation command by writing 0x4 to [`DIRECT_ACCESS_CMD`](registers.md#direct_access_cmd). +5. Poll the [`STATUS`](registers.md#status) until the DAI state goes back to idle. +Alternatively, the `otp_operation_done` interrupt can be enabled up to notify the processor once an access has completed. +6. If the status register flags a DAI error, additional handling is required (see [Section on Error handling](#error-handling)). + +The hardware will set [`DIRECT_ACCESS_REGWEN`](registers.md#direct_access_regwen) to 0x0 while an operation is pending in order to temporarily lock write access to the CSRs registers. + +It should also be noted that the effect of locking a partition via the digest only takes effect **after** the next system reset. +To prevent integrity check failures SW must therefore ensure that no more programming operations are issued to the affected partition after initiating the digest calculation sequence. + +### Software Integrity Handling + +As opposed to buffered partitions, the digest and integrity handling of unbuffered partitions is entirely up to software. +The only hardware-assisted feature in unbuffered partitions is the digest lock, which locks write access to an unbuffered partition once a nonzero value has been programmed to the 64bit digest location. + +In a similar vein, it should be noted that the system-wide bus-integrity metadata does not travel alongside the data end-to-end in the OTP controller (i.e., the bus-integrity metadata bits are not stored into the OTP memory array). +This means that data written to and read from the OTP macro is not protected by the bus integrity feature at all stages. +In case of buffered partitions this does not pose a concern since data integrity in these partitions is checked via the hardware assisted digest mechanism. +In case of unbuffered partitions however, the data integrity checking is entirely up to software. +I.e., if data is read from an unbuffered partition (either through the DAI or CSR windows), software should perform an integrity check on that data. + +## Error Handling + +The agents that can access the OTP macro (DAI, LCI, buffered/unbuffered partitions) expose detailed error codes that can be used to root cause any failure. +The error codes are defined in the table below, and the corresponding `otp_err_e` enum type can be found in the `otp_ctrl_pkg`. +The table also lists which error codes are supported by which agent. + +Errors that are not "recoverable" are severe errors that move the corresponding partition or DAI/LCI FSM into a terminal error state, where no more commands can be accepted (a system reset is required to restore functionality in that case). +Errors that are "recoverable" are less severe and do not cause the FSM to jump into a terminal error state. + +Note that error codes that originate in the physical OTP macro are prefixed with `Macro*`. + +Error Code | Enum Name | Recoverable | DAI | LCI | Unbuf | Buf | Description +-----------|------------------------|-------------|-----|-----|-------|-------|------------- +0x0 | `NoError` | - | x | x | x | x | No error has occurred. +0x1 | `MacroError` | no | x | x | x | x | Returned if the OTP macro command did not complete successfully due to a macro malfunction. +0x2 | `MacroEccCorrError` | yes | x | - | x | x | A correctable ECC error has occurred during a read operation in the OTP macro. +0x3 | `MacroEccUncorrError` | no | x | - | x* | x | An uncorrectable ECC error has occurred during a read operation in the OTP macro. Note (*): This error is collapsed into `MacroEccCorrError` if the partition is a vendor test partition. It then becomes a recoverable error. +0x4 | `MacroWriteBlankError` | yes / no* | x | x | - | - | This error is returned if a write operation attempted to clear an already programmed bit location. Note (*): This error is recoverable if encountered in the DAI, but unrecoverable if encountered in the LCI. +0x5 | `AccessError` | yes | x | - | x | - | An access error has occurred (e.g. write to write-locked region, or read to a read-locked region). +0x6 | `CheckFailError` | no | - | - | x | x | An unrecoverable ECC, integrity or consistency error has been detected. +0x7 | `FsmStateError` | no | x | x | x | x | The FSM has been glitched into an invalid state, or escalation has been triggered and the FSM has been moved into a terminal error state. + +All non-zero error codes listed above trigger an `otp_error` interrupt. +In addition, all unrecoverable OTP `Macro*` errors (codes 0x1, 0x3) trigger a `fatal_macro_error` alert, while all remaining unrecoverable errors trigger a `fatal_check_error` alert. + +If software receives an `otp_error` interrupt, but all error codes read back as 0x0 (`NoError`), this should be treated as a fatal error condition, and the system should be shut down as soon as possible. + +Note that the `MacroWriteBlankError` will only be generated if the write attempt over already written data fails within the OTP macro after applying any means supported within it to enable a write on existing data, e.g., a bit-reversal option. +Also note that while this error is marked as a recoverable error, the affected OTP word may be in an inconsistent state after this error has been returned. +This can cause several issues when the word is accessed again (either as part of a regular read operation, as part of the readout at boot, or as part of a background check). +It is important that SW ensures that each word is only written once, since this can render the device useless. + +## Direct Access Memory Map + +The table below provides a detailed overview of the items stored in the OTP partitions. +Some of the items that are buffered in registers is readable via memory mapped CSRs, and these CSRs are linked in the table below. +Items that are not linked can only be accessed via the direct programming interface (if the partition is not locked via the corresponding digest). +It should be noted that CREATOR_SW_CFG and OWNER_SW_CFG are accessible through a memory mapped window, and content of these partitions is not buffered. +Hence, a read access to those windows will take in the order of 10-20 cycles until the read returns. + +Sizes below are specified in multiples of 32bit words. + +{{#include otp_ctrl_mmap.md}} + +Note that since the content in the SECRET* partitions are scrambled using a 64bit PRESENT cipher, read and write access through the DAI needs to occur at a 64bit granularity. +Also, all digests (no matter whether they are SW or HW digests) have an access granule of 64bit. + +The table below lists digests locations, and the corresponding locked partitions. + +{{#include otp_ctrl_digests.md}} + +Write access to the affected partition will be locked if the digest has a nonzero value. + +For the software partition digests, it is entirely up to software to decide on the digest algorithm to be used. +Hardware will determine the lock condition only based on whether a non-zero value is present at that location or not. + +For the hardware partitions, hardware calculates this digest and uses it for [background verification](#partition-checks). +Digest calculation can be triggered via the DAI. + +Finally, it should be noted that the RMA_TOKEN and CREATOR_ROOT_KEY_SHARE0 / CREATOR_ROOT_KEY_SHARE1 items can only be programmed when the device is in the DEV, PROD, PROD_END and RMA stages. +Please consult the [life cycle controller documentation](../../../../ip/lc_ctrl/README.md) documentation for more information. + +## OTP Field Descriptions + +The table below describes what each field in the OTP partitions is used for. + +{{#include otp_ctrl_field_descriptions.md}} + +## Examples + +### Provisioning Items + +The following represents a typical provisioning sequence for items in all partitions (except for the LIFE_CYCLE partition, which is not software-programmable): + +1. [Program](#programming-sequence) the item in 32bit or 64bit chunks via the DAI. +2. [Read back](#readout-sequence) and verify the item via the DAI. +3. If the item is exposed via CSRs or a CSR window, perform a full-system reset and verify whether those fields are correctly populated. + +Note that any unrecoverable errors during the programming steps, or mismatches during the readback and verification steps indicate that the device might be malfunctioning (possibly due to fabrication defects) and hence the device may have to be scrapped. +This is however rare and should not happen after fabrication testing. + +### Locking Partitions + +Once a partition has been fully populated, write access to that partition has to be permanently locked. +For the HW_CFG* and SECRET* partitions, this can be achieved as follows: + +1. [Trigger](#digest-calculation-sequence) a digest calculation via the DAI. +2. [Read back](#readout-sequence) and verify the digest location via the DAI. +3. Perform a full-system reset and verify that the corresponding CSRs exposing the 64bit digest have been populated ([`HW_CFG_DIGEST_0`](registers.md#hw_cfg_digest), [`SECRET0_DIGEST_0`](registers.md#secret0_digest), [`SECRET1_DIGEST_0`](registers.md#secret1_digest) or [`SECRET2_DIGEST_0`](registers.md#secret2_digest)). + +It should be noted that locking only takes effect after a system reset since the affected partitions first have to re-sense the digest values. +Hence, it is critical that SW ensures that no more data is written to the partition to be locked after triggering the hardware digest calculation. +Otherwise, the device will likely be rendered inoperable as this can lead to permanent digest mismatch errors after system reboot. + +For the [`CREATOR_SW_CFG`](registers.md#creator_sw_cfg) and [`OWNER_SW_CFG`](registers.md#owner_sw_cfg) partitions, the process is similar, but computation and programming of the digest is entirely up to software: + +1. Compute a 64bit digest over the relevant parts of the partition, and [program](#programming-sequence) that value to [`CREATOR_SW_CFG_DIGEST_0`](registers.md#creator_sw_cfg_digest) or [`OWNER_SW_CFG_DIGEST_0`](registers.md#owner_sw_cfg_digest) via the DAI. Note that digest accesses through the DAI have an access granule of 64bit. +2. [Read back](#readout-sequence) and verify the digest location via the DAI. +3. Perform a full-system reset and verify that the corresponding digest CSRs [`CREATOR_SW_CFG_DIGEST_0`](registers.md#creator_sw_cfg_digest) or [`OWNER_SW_CFG_DIGEST_0`](registers.md#owner_sw_cfg_digest) have been populated with the correct 64bit value. + +Note that any unrecoverable errors during the programming steps, or mismatches during the read-back and verification steps indicate that the device might be malfunctioning (possibly due to fabrication defects) and hence the device may have to be scrapped. +This is however rare and should not happen after fabrication testing. + +## Device Interface Functions (DIFs) + +- [Device Interface Functions](../../../../../sw/device/lib/dif/dif_otp_ctrl.h) + +# Additional Notes + +## OTP IP Assumptions + +It is assumed the OTP IP employed in production has reasonable physical defense characteristics. +Specifically which defensive features will likely be use case dependent, but at a minimum they should have the properties below. +Note some properties are worded with "SHALL" and others with "SHOULD". +"SHALL" refers to features that must be present, while "SHOULD" refers to features that are ideal, but optional. + +- The contents shall not be observable via optical microscopy (for example anti-fuse technology). +- The IP lifetime shall not be limited by the amount of read cycles performed. +- If the IP contains field programmability (internal charge pumps and LDOs), there shall be mechanisms in place to selectively disable this function based on device context. +- If the IP contains redundant columns, rows, pages or banks for yield improvement, it shall provide a mechanism to lock down arbitrary manipulation of page / bank swapping during run-time. +- The IP shall be clear on what bits must be manipulated by the user, what bits are automatically manipulated by hardware (for example ECC or redundancy) and what areas the user can influence. +- The IP shall be compatible, through the use of a proprietary wrapper or shim, with an open-source friendly IO interface. +- The IP should functionally support the programming of already programmed bits without information leakage. +- The IP should offer SCA resistance: + - For example, the content may be stored differentially. + - For example, the sensing exhibits similar power signatures no matter if the stored bit is 0 or 1. +- The IP interface shall be memory-like if beyond a certain size. +- When a particular location is read, a fixed width output is returned; similar when a particular location is programmed, a fixed width input is supplied. +- The IP does not output all stored bits in parallel. +- The contents should be electrically hidden. For example, it should be difficult for an attacker to energize the fuse array and observe how the charge leaks. +- The IP should route critical nets at lower metal levels to avoid probing. +- The IP should contain native detectors for fault injection attacks. +- The IP should contain mechanisms to guard against interrupted programming - either through malicious intent or unexpected power loss and glitched address lines. +- The IP should contain mechanisms for error corrections (single bit errors). + - For example ECC or redundant bits voting / or-ing. + - As error correction mechanisms are technology dependent, that information should not be exposed to the open-source controller, instead the controller should simply receive information on whether a read / program was successful. +- The IP should have self-test functionality to assess the health of the storage and analog structures. +- The IP may contain native PUF-like functionality. diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/registers.md b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/registers.md new file mode 100644 index 00000000000000..dbdc4f146b87d9 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/registers.md @@ -0,0 +1,1525 @@ +# Registers + + +## Summary of the **`core`** interface's registers + +| Name | Offset | Length | Description | +|:-------------------------------------------------------------------------------|:---------|---------:|:----------------------------------------------------------------------------------------------------| +| otp_ctrl.[`INTR_STATE`](#intr_state) | 0x0 | 4 | Interrupt State Register | +| otp_ctrl.[`INTR_ENABLE`](#intr_enable) | 0x4 | 4 | Interrupt Enable Register | +| otp_ctrl.[`INTR_TEST`](#intr_test) | 0x8 | 4 | Interrupt Test Register | +| otp_ctrl.[`ALERT_TEST`](#alert_test) | 0xc | 4 | Alert Test Register | +| otp_ctrl.[`STATUS`](#status) | 0x10 | 4 | OTP status register. | +| otp_ctrl.[`ERR_CODE_0`](#err_code) | 0x14 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_1`](#err_code) | 0x18 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_2`](#err_code) | 0x1c | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_3`](#err_code) | 0x20 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_4`](#err_code) | 0x24 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_5`](#err_code) | 0x28 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_6`](#err_code) | 0x2c | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_7`](#err_code) | 0x30 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_8`](#err_code) | 0x34 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_9`](#err_code) | 0x38 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_10`](#err_code) | 0x3c | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_11`](#err_code) | 0x40 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_12`](#err_code) | 0x44 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_13`](#err_code) | 0x48 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_14`](#err_code) | 0x4c | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_15`](#err_code) | 0x50 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_16`](#err_code) | 0x54 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_17`](#err_code) | 0x58 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_18`](#err_code) | 0x5c | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_19`](#err_code) | 0x60 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_20`](#err_code) | 0x64 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_21`](#err_code) | 0x68 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_22`](#err_code) | 0x6c | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`ERR_CODE_23`](#err_code) | 0x70 | 4 | This register holds information about error conditions that occurred in the agents | +| otp_ctrl.[`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) | 0x74 | 4 | Register write enable for all direct access interface registers. | +| otp_ctrl.[`DIRECT_ACCESS_CMD`](#direct_access_cmd) | 0x78 | 4 | Command register for direct accesses. | +| otp_ctrl.[`DIRECT_ACCESS_ADDRESS`](#direct_access_address) | 0x7c | 4 | Address register for direct accesses. | +| otp_ctrl.[`DIRECT_ACCESS_WDATA_0`](#direct_access_wdata) | 0x80 | 4 | Write data for direct accesses. | +| otp_ctrl.[`DIRECT_ACCESS_WDATA_1`](#direct_access_wdata) | 0x84 | 4 | Write data for direct accesses. | +| otp_ctrl.[`DIRECT_ACCESS_RDATA_0`](#direct_access_rdata) | 0x88 | 4 | Read data for direct accesses. | +| otp_ctrl.[`DIRECT_ACCESS_RDATA_1`](#direct_access_rdata) | 0x8c | 4 | Read data for direct accesses. | +| otp_ctrl.[`CHECK_TRIGGER_REGWEN`](#check_trigger_regwen) | 0x90 | 4 | Register write enable for !!CHECK_TRIGGER. | +| otp_ctrl.[`CHECK_TRIGGER`](#check_trigger) | 0x94 | 4 | Command register for direct accesses. | +| otp_ctrl.[`CHECK_REGWEN`](#check_regwen) | 0x98 | 4 | Register write enable for !!INTEGRITY_CHECK_PERIOD and !!CONSISTENCY_CHECK_PERIOD. | +| otp_ctrl.[`CHECK_TIMEOUT`](#check_timeout) | 0x9c | 4 | Timeout value for the integrity and consistency checks. | +| otp_ctrl.[`INTEGRITY_CHECK_PERIOD`](#integrity_check_period) | 0xa0 | 4 | This value specifies the maximum period that can be generated pseudo-randomly. | +| otp_ctrl.[`CONSISTENCY_CHECK_PERIOD`](#consistency_check_period) | 0xa4 | 4 | This value specifies the maximum period that can be generated pseudo-randomly. | +| otp_ctrl.[`VENDOR_TEST_READ_LOCK`](#vendor_test_read_lock) | 0xa8 | 4 | Runtime read lock for the VENDOR_TEST partition. | +| otp_ctrl.[`CREATOR_SW_CFG_READ_LOCK`](#creator_sw_cfg_read_lock) | 0xac | 4 | Runtime read lock for the CREATOR_SW_CFG partition. | +| otp_ctrl.[`OWNER_SW_CFG_READ_LOCK`](#owner_sw_cfg_read_lock) | 0xb0 | 4 | Runtime read lock for the OWNER_SW_CFG partition. | +| otp_ctrl.[`OWNERSHIP_SLOT_STATE_READ_LOCK`](#ownership_slot_state_read_lock) | 0xb4 | 4 | Runtime read lock for the OWNERSHIP_SLOT_STATE partition. | +| otp_ctrl.[`ROT_CREATOR_AUTH_READ_LOCK`](#rot_creator_auth_read_lock) | 0xb8 | 4 | Runtime read lock for the ROT_CREATOR_AUTH partition. | +| otp_ctrl.[`ROT_OWNER_AUTH_SLOT0_READ_LOCK`](#rot_owner_auth_slot0_read_lock) | 0xbc | 4 | Runtime read lock for the ROT_OWNER_AUTH_SLOT0 partition. | +| otp_ctrl.[`ROT_OWNER_AUTH_SLOT1_READ_LOCK`](#rot_owner_auth_slot1_read_lock) | 0xc0 | 4 | Runtime read lock for the ROT_OWNER_AUTH_SLOT1 partition. | +| otp_ctrl.[`PLAT_INTEG_AUTH_SLOT0_READ_LOCK`](#plat_integ_auth_slot0_read_lock) | 0xc4 | 4 | Runtime read lock for the PLAT_INTEG_AUTH_SLOT0 partition. | +| otp_ctrl.[`PLAT_INTEG_AUTH_SLOT1_READ_LOCK`](#plat_integ_auth_slot1_read_lock) | 0xc8 | 4 | Runtime read lock for the PLAT_INTEG_AUTH_SLOT1 partition. | +| otp_ctrl.[`PLAT_OWNER_AUTH_SLOT0_READ_LOCK`](#plat_owner_auth_slot0_read_lock) | 0xcc | 4 | Runtime read lock for the PLAT_OWNER_AUTH_SLOT0 partition. | +| otp_ctrl.[`PLAT_OWNER_AUTH_SLOT1_READ_LOCK`](#plat_owner_auth_slot1_read_lock) | 0xd0 | 4 | Runtime read lock for the PLAT_OWNER_AUTH_SLOT1 partition. | +| otp_ctrl.[`PLAT_OWNER_AUTH_SLOT2_READ_LOCK`](#plat_owner_auth_slot2_read_lock) | 0xd4 | 4 | Runtime read lock for the PLAT_OWNER_AUTH_SLOT2 partition. | +| otp_ctrl.[`PLAT_OWNER_AUTH_SLOT3_READ_LOCK`](#plat_owner_auth_slot3_read_lock) | 0xd8 | 4 | Runtime read lock for the PLAT_OWNER_AUTH_SLOT3 partition. | +| otp_ctrl.[`EXT_NVM_READ_LOCK`](#ext_nvm_read_lock) | 0xdc | 4 | Runtime read lock for the EXT_NVM partition. | +| otp_ctrl.[`ROM_PATCH_READ_LOCK`](#rom_patch_read_lock) | 0xe0 | 4 | Runtime read lock for the ROM_PATCH partition. | +| otp_ctrl.[`VENDOR_TEST_DIGEST_0`](#vendor_test_digest) | 0xe4 | 4 | Integrity digest for the VENDOR_TEST partition. | +| otp_ctrl.[`VENDOR_TEST_DIGEST_1`](#vendor_test_digest) | 0xe8 | 4 | Integrity digest for the VENDOR_TEST partition. | +| otp_ctrl.[`CREATOR_SW_CFG_DIGEST_0`](#creator_sw_cfg_digest) | 0xec | 4 | Integrity digest for the CREATOR_SW_CFG partition. | +| otp_ctrl.[`CREATOR_SW_CFG_DIGEST_1`](#creator_sw_cfg_digest) | 0xf0 | 4 | Integrity digest for the CREATOR_SW_CFG partition. | +| otp_ctrl.[`OWNER_SW_CFG_DIGEST_0`](#owner_sw_cfg_digest) | 0xf4 | 4 | Integrity digest for the OWNER_SW_CFG partition. | +| otp_ctrl.[`OWNER_SW_CFG_DIGEST_1`](#owner_sw_cfg_digest) | 0xf8 | 4 | Integrity digest for the OWNER_SW_CFG partition. | +| otp_ctrl.[`ROT_CREATOR_AUTH_DIGEST_0`](#rot_creator_auth_digest) | 0xfc | 4 | Integrity digest for the ROT_CREATOR_AUTH partition. | +| otp_ctrl.[`ROT_CREATOR_AUTH_DIGEST_1`](#rot_creator_auth_digest) | 0x100 | 4 | Integrity digest for the ROT_CREATOR_AUTH partition. | +| otp_ctrl.[`ROT_OWNER_AUTH_SLOT0_DIGEST_0`](#rot_owner_auth_slot0_digest) | 0x104 | 4 | Integrity digest for the ROT_OWNER_AUTH_SLOT0 partition. | +| otp_ctrl.[`ROT_OWNER_AUTH_SLOT0_DIGEST_1`](#rot_owner_auth_slot0_digest) | 0x108 | 4 | Integrity digest for the ROT_OWNER_AUTH_SLOT0 partition. | +| otp_ctrl.[`ROT_OWNER_AUTH_SLOT1_DIGEST_0`](#rot_owner_auth_slot1_digest) | 0x10c | 4 | Integrity digest for the ROT_OWNER_AUTH_SLOT1 partition. | +| otp_ctrl.[`ROT_OWNER_AUTH_SLOT1_DIGEST_1`](#rot_owner_auth_slot1_digest) | 0x110 | 4 | Integrity digest for the ROT_OWNER_AUTH_SLOT1 partition. | +| otp_ctrl.[`PLAT_INTEG_AUTH_SLOT0_DIGEST_0`](#plat_integ_auth_slot0_digest) | 0x114 | 4 | Integrity digest for the PLAT_INTEG_AUTH_SLOT0 partition. | +| otp_ctrl.[`PLAT_INTEG_AUTH_SLOT0_DIGEST_1`](#plat_integ_auth_slot0_digest) | 0x118 | 4 | Integrity digest for the PLAT_INTEG_AUTH_SLOT0 partition. | +| otp_ctrl.[`PLAT_INTEG_AUTH_SLOT1_DIGEST_0`](#plat_integ_auth_slot1_digest) | 0x11c | 4 | Integrity digest for the PLAT_INTEG_AUTH_SLOT1 partition. | +| otp_ctrl.[`PLAT_INTEG_AUTH_SLOT1_DIGEST_1`](#plat_integ_auth_slot1_digest) | 0x120 | 4 | Integrity digest for the PLAT_INTEG_AUTH_SLOT1 partition. | +| otp_ctrl.[`PLAT_OWNER_AUTH_SLOT0_DIGEST_0`](#plat_owner_auth_slot0_digest) | 0x124 | 4 | Integrity digest for the PLAT_OWNER_AUTH_SLOT0 partition. | +| otp_ctrl.[`PLAT_OWNER_AUTH_SLOT0_DIGEST_1`](#plat_owner_auth_slot0_digest) | 0x128 | 4 | Integrity digest for the PLAT_OWNER_AUTH_SLOT0 partition. | +| otp_ctrl.[`PLAT_OWNER_AUTH_SLOT1_DIGEST_0`](#plat_owner_auth_slot1_digest) | 0x12c | 4 | Integrity digest for the PLAT_OWNER_AUTH_SLOT1 partition. | +| otp_ctrl.[`PLAT_OWNER_AUTH_SLOT1_DIGEST_1`](#plat_owner_auth_slot1_digest) | 0x130 | 4 | Integrity digest for the PLAT_OWNER_AUTH_SLOT1 partition. | +| otp_ctrl.[`PLAT_OWNER_AUTH_SLOT2_DIGEST_0`](#plat_owner_auth_slot2_digest) | 0x134 | 4 | Integrity digest for the PLAT_OWNER_AUTH_SLOT2 partition. | +| otp_ctrl.[`PLAT_OWNER_AUTH_SLOT2_DIGEST_1`](#plat_owner_auth_slot2_digest) | 0x138 | 4 | Integrity digest for the PLAT_OWNER_AUTH_SLOT2 partition. | +| otp_ctrl.[`PLAT_OWNER_AUTH_SLOT3_DIGEST_0`](#plat_owner_auth_slot3_digest) | 0x13c | 4 | Integrity digest for the PLAT_OWNER_AUTH_SLOT3 partition. | +| otp_ctrl.[`PLAT_OWNER_AUTH_SLOT3_DIGEST_1`](#plat_owner_auth_slot3_digest) | 0x140 | 4 | Integrity digest for the PLAT_OWNER_AUTH_SLOT3 partition. | +| otp_ctrl.[`ROM_PATCH_DIGEST_0`](#rom_patch_digest) | 0x144 | 4 | Integrity digest for the ROM_PATCH partition. | +| otp_ctrl.[`ROM_PATCH_DIGEST_1`](#rom_patch_digest) | 0x148 | 4 | Integrity digest for the ROM_PATCH partition. | +| otp_ctrl.[`HW_CFG0_DIGEST_0`](#hw_cfg0_digest) | 0x14c | 4 | Integrity digest for the HW_CFG0 partition. | +| otp_ctrl.[`HW_CFG0_DIGEST_1`](#hw_cfg0_digest) | 0x150 | 4 | Integrity digest for the HW_CFG0 partition. | +| otp_ctrl.[`HW_CFG1_DIGEST_0`](#hw_cfg1_digest) | 0x154 | 4 | Integrity digest for the HW_CFG1 partition. | +| otp_ctrl.[`HW_CFG1_DIGEST_1`](#hw_cfg1_digest) | 0x158 | 4 | Integrity digest for the HW_CFG1 partition. | +| otp_ctrl.[`SECRET0_DIGEST_0`](#secret0_digest) | 0x15c | 4 | Integrity digest for the SECRET0 partition. | +| otp_ctrl.[`SECRET0_DIGEST_1`](#secret0_digest) | 0x160 | 4 | Integrity digest for the SECRET0 partition. | +| otp_ctrl.[`SECRET1_DIGEST_0`](#secret1_digest) | 0x164 | 4 | Integrity digest for the SECRET1 partition. | +| otp_ctrl.[`SECRET1_DIGEST_1`](#secret1_digest) | 0x168 | 4 | Integrity digest for the SECRET1 partition. | +| otp_ctrl.[`SECRET2_DIGEST_0`](#secret2_digest) | 0x16c | 4 | Integrity digest for the SECRET2 partition. | +| otp_ctrl.[`SECRET2_DIGEST_1`](#secret2_digest) | 0x170 | 4 | Integrity digest for the SECRET2 partition. | +| otp_ctrl.[`SECRET3_DIGEST_0`](#secret3_digest) | 0x174 | 4 | Integrity digest for the SECRET3 partition. | +| otp_ctrl.[`SECRET3_DIGEST_1`](#secret3_digest) | 0x178 | 4 | Integrity digest for the SECRET3 partition. | +| otp_ctrl.[`SW_CFG_WINDOW`](#sw_cfg_window) | 0x4000 | 16384 | Any read to this window directly maps to the corresponding offset in the creator and owner software | + +## INTR_STATE +Interrupt State Register +- Offset: `0x0` +- Reset default: `0x0` +- Reset mask: `0x3` + +### Fields + +```wavejson +{"reg": [{"name": "otp_operation_done", "bits": 1, "attr": ["rw1c"], "rotate": -90}, {"name": "otp_error", "bits": 1, "attr": ["rw1c"], "rotate": -90}, {"bits": 30}], "config": {"lanes": 1, "fontsize": 10, "vspace": 200}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------------------|:-----------------------------------------------------------------------------------------------------------------| +| 31:2 | | | | Reserved | +| 1 | rw1c | 0x0 | otp_error | An error has occurred in the OTP controller. Check the [`ERR_CODE`](#err_code) register to get more information. | +| 0 | rw1c | 0x0 | otp_operation_done | A direct access command or digest calculation operation has completed. | + +## INTR_ENABLE +Interrupt Enable Register +- Offset: `0x4` +- Reset default: `0x0` +- Reset mask: `0x3` + +### Fields + +```wavejson +{"reg": [{"name": "otp_operation_done", "bits": 1, "attr": ["rw"], "rotate": -90}, {"name": "otp_error", "bits": 1, "attr": ["rw"], "rotate": -90}, {"bits": 30}], "config": {"lanes": 1, "fontsize": 10, "vspace": 200}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------------------|:-----------------------------------------------------------------------------| +| 31:2 | | | | Reserved | +| 1 | rw | 0x0 | otp_error | Enable interrupt when [`INTR_STATE.otp_error`](#intr_state) is set. | +| 0 | rw | 0x0 | otp_operation_done | Enable interrupt when [`INTR_STATE.otp_operation_done`](#intr_state) is set. | + +## INTR_TEST +Interrupt Test Register +- Offset: `0x8` +- Reset default: `0x0` +- Reset mask: `0x3` + +### Fields + +```wavejson +{"reg": [{"name": "otp_operation_done", "bits": 1, "attr": ["wo"], "rotate": -90}, {"name": "otp_error", "bits": 1, "attr": ["wo"], "rotate": -90}, {"bits": 30}], "config": {"lanes": 1, "fontsize": 10, "vspace": 200}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------------------|:----------------------------------------------------------------------| +| 31:2 | | | | Reserved | +| 1 | wo | 0x0 | otp_error | Write 1 to force [`INTR_STATE.otp_error`](#intr_state) to 1. | +| 0 | wo | 0x0 | otp_operation_done | Write 1 to force [`INTR_STATE.otp_operation_done`](#intr_state) to 1. | + +## ALERT_TEST +Alert Test Register +- Offset: `0xc` +- Reset default: `0x0` +- Reset mask: `0x1f` + +### Fields + +```wavejson +{"reg": [{"name": "fatal_macro_error", "bits": 1, "attr": ["wo"], "rotate": -90}, {"name": "fatal_check_error", "bits": 1, "attr": ["wo"], "rotate": -90}, {"name": "fatal_bus_integ_error", "bits": 1, "attr": ["wo"], "rotate": -90}, {"name": "fatal_prim_otp_alert", "bits": 1, "attr": ["wo"], "rotate": -90}, {"name": "recov_prim_otp_alert", "bits": 1, "attr": ["wo"], "rotate": -90}, {"bits": 27}], "config": {"lanes": 1, "fontsize": 10, "vspace": 230}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:----------------------|:-------------------------------------------------| +| 31:5 | | | | Reserved | +| 4 | wo | 0x0 | recov_prim_otp_alert | Write 1 to trigger one alert event of this kind. | +| 3 | wo | 0x0 | fatal_prim_otp_alert | Write 1 to trigger one alert event of this kind. | +| 2 | wo | 0x0 | fatal_bus_integ_error | Write 1 to trigger one alert event of this kind. | +| 1 | wo | 0x0 | fatal_check_error | Write 1 to trigger one alert event of this kind. | +| 0 | wo | 0x0 | fatal_macro_error | Write 1 to trigger one alert event of this kind. | + +## STATUS +OTP status register. +- Offset: `0x10` +- Reset default: `0x0` +- Reset mask: `0x7fffffff` + +### Fields + +```wavejson +{"reg": [{"name": "VENDOR_TEST_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "CREATOR_SW_CFG_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "OWNER_SW_CFG_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "OWNERSHIP_SLOT_STATE_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "ROT_CREATOR_AUTH_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "ROT_OWNER_AUTH_SLOT0_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "ROT_OWNER_AUTH_SLOT1_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "PLAT_INTEG_AUTH_SLOT0_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "PLAT_INTEG_AUTH_SLOT1_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "PLAT_OWNER_AUTH_SLOT0_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "PLAT_OWNER_AUTH_SLOT1_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "PLAT_OWNER_AUTH_SLOT2_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "PLAT_OWNER_AUTH_SLOT3_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "EXT_NVM_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "ROM_PATCH_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "HW_CFG0_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "HW_CFG1_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "SECRET0_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "SECRET1_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "SECRET2_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "SECRET3_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "LIFE_CYCLE_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "DAI_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "LCI_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "TIMEOUT_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "LFSR_FSM_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "SCRAMBLING_FSM_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "KEY_DERIV_FSM_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "BUS_INTEG_ERROR", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "DAI_IDLE", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "CHECK_PENDING", "bits": 1, "attr": ["ro"], "rotate": -90}, {"bits": 1}], "config": {"lanes": 1, "fontsize": 10, "vspace": 290}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:----------------------------|:------------------------------------------------------------------------------------------------------------------------------------------------------| +| 31 | | | | Reserved | +| 30 | ro | 0x0 | CHECK_PENDING | Set to 1 if an integrity or consistency check triggered by the LFSR timer or via [`CHECK_TRIGGER`](#check_trigger) is pending. | +| 29 | ro | 0x0 | DAI_IDLE | Set to 1 if the DAI is idle and ready to accept commands. | +| 28 | ro | 0x0 | BUS_INTEG_ERROR | This bit is set to 1 if a fatal bus integrity fault is detected. This error triggers a fatal_bus_integ_error alert. | +| 27 | ro | 0x0 | KEY_DERIV_FSM_ERROR | Set to 1 if the key derivation FSM has reached an invalid state. This raises an fatal_check_error alert and is an unrecoverable error condition. | +| 26 | ro | 0x0 | SCRAMBLING_FSM_ERROR | Set to 1 if the scrambling datapath FSM has reached an invalid state. This raises an fatal_check_error alert and is an unrecoverable error condition. | +| 25 | ro | 0x0 | LFSR_FSM_ERROR | Set to 1 if the LFSR timer FSM has reached an invalid state. This raises an fatal_check_error alert and is an unrecoverable error condition. | +| 24 | ro | 0x0 | TIMEOUT_ERROR | Set to 1 if an integrity or consistency check times out. This raises an fatal_check_error alert and is an unrecoverable error condition. | +| 23 | ro | 0x0 | LCI_ERROR | Set to 1 if an error occurred in the LCI. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 22 | ro | 0x0 | DAI_ERROR | Set to 1 if an error occurred in the DAI. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 21 | ro | 0x0 | LIFE_CYCLE_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 20 | ro | 0x0 | SECRET3_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 19 | ro | 0x0 | SECRET2_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 18 | ro | 0x0 | SECRET1_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 17 | ro | 0x0 | SECRET0_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 16 | ro | 0x0 | HW_CFG1_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 15 | ro | 0x0 | HW_CFG0_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 14 | ro | 0x0 | ROM_PATCH_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 13 | ro | 0x0 | EXT_NVM_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 12 | ro | 0x0 | PLAT_OWNER_AUTH_SLOT3_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 11 | ro | 0x0 | PLAT_OWNER_AUTH_SLOT2_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 10 | ro | 0x0 | PLAT_OWNER_AUTH_SLOT1_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 9 | ro | 0x0 | PLAT_OWNER_AUTH_SLOT0_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 8 | ro | 0x0 | PLAT_INTEG_AUTH_SLOT1_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 7 | ro | 0x0 | PLAT_INTEG_AUTH_SLOT0_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 6 | ro | 0x0 | ROT_OWNER_AUTH_SLOT1_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 5 | ro | 0x0 | ROT_OWNER_AUTH_SLOT0_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 4 | ro | 0x0 | ROT_CREATOR_AUTH_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 3 | ro | 0x0 | OWNERSHIP_SLOT_STATE_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 2 | ro | 0x0 | OWNER_SW_CFG_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 1 | ro | 0x0 | CREATOR_SW_CFG_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | +| 0 | ro | 0x0 | VENDOR_TEST_ERROR | Set to 1 if an error occurred in this partition. If set to 1, SW should check the [`ERR_CODE`](#err_code) register at the corresponding index. | + +## ERR_CODE +This register holds information about error conditions that occurred in the agents +interacting with the OTP macro via the internal bus. The error codes should be checked +if the partitions, DAI or LCI flag an error in the [`STATUS`](#status) register, or when an +[`INTR_STATE.otp_error`](#intr_state) has been triggered. Note that all errors trigger an otp_error +interrupt, and in addition some errors may trigger either an fatal_macro_error or an +fatal_check_error alert. +- Reset default: `0x0` +- Reset mask: `0x7` + +### Instances + +| Name | Offset | +|:------------|:---------| +| ERR_CODE_0 | 0x14 | +| ERR_CODE_1 | 0x18 | +| ERR_CODE_2 | 0x1c | +| ERR_CODE_3 | 0x20 | +| ERR_CODE_4 | 0x24 | +| ERR_CODE_5 | 0x28 | +| ERR_CODE_6 | 0x2c | +| ERR_CODE_7 | 0x30 | +| ERR_CODE_8 | 0x34 | +| ERR_CODE_9 | 0x38 | +| ERR_CODE_10 | 0x3c | +| ERR_CODE_11 | 0x40 | +| ERR_CODE_12 | 0x44 | +| ERR_CODE_13 | 0x48 | +| ERR_CODE_14 | 0x4c | +| ERR_CODE_15 | 0x50 | +| ERR_CODE_16 | 0x54 | +| ERR_CODE_17 | 0x58 | +| ERR_CODE_18 | 0x5c | +| ERR_CODE_19 | 0x60 | +| ERR_CODE_20 | 0x64 | +| ERR_CODE_21 | 0x68 | +| ERR_CODE_22 | 0x6c | +| ERR_CODE_23 | 0x70 | + + +### Fields + +```wavejson +{"reg": [{"name": "ERR_CODE", "bits": 3, "attr": ["ro"], "rotate": -90}, {"bits": 29}], "config": {"lanes": 1, "fontsize": 10, "vspace": 100}} +``` + +| Bits | Type | Reset | Name | +|:------:|:------:|:-------:|:--------------------------------| +| 31:3 | | | Reserved | +| 2:0 | ro | 0x0 | [ERR_CODE](#err_code--err_code) | + +### ERR_CODE . ERR_CODE + +| Value | Name | Description | +|:--------|:------------------------|:---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| 0x0 | NO_ERROR | No error condition has occurred. | +| 0x1 | MACRO_ERROR | Returned if the OTP macro command was invalid or did not complete successfully due to a macro malfunction. This error should never occur during normal operation and is not recoverable. This error triggers an fatal_macro_error alert. | +| 0x2 | MACRO_ECC_CORR_ERROR | A correctable ECC error has occured during an OTP read operation. The corresponding controller automatically recovers from this error when issuing a new command. | +| 0x3 | MACRO_ECC_UNCORR_ERROR | An uncorrectable ECC error has occurred during an OTP read operation. This error should never occur during normal operation and is not recoverable. If this error is present this may be a sign that the device is malfunctioning. This error triggers an fatal_macro_error alert. | +| 0x4 | MACRO_WRITE_BLANK_ERROR | This error is returned if a programming operation attempted to clear a bit that has previously been programmed to 1. The corresponding controller automatically recovers from this error when issuing a new command. Note however that the affected OTP word may be left in an inconsistent state if this error occurs. This can cause several issues when the word is accessed again (either as part of a regular read operation, as part of the readout at boot, or as part of a background check). It is important that SW ensures that each word is only written once, since this can render the device useless. | +| 0x5 | ACCESS_ERROR | This error indicates that a locked memory region has been accessed. The corresponding controller automatically recovers from this error when issuing a new command. | +| 0x6 | CHECK_FAIL_ERROR | An ECC, integrity or consistency mismatch has been detected in the buffer registers. This error should never occur during normal operation and is not recoverable. This error triggers an fatal_check_error alert. | +| 0x7 | FSM_STATE_ERROR | The FSM of the corresponding controller has reached an invalid state, or the FSM has been moved into a terminal error state due to an escalation action via lc_escalate_en_i. This error should never occur during normal operation and is not recoverable. If this error is present, this is a sign that the device has fallen victim to an invasive attack. This error triggers an fatal_check_error alert. | + + +## DIRECT_ACCESS_REGWEN +Register write enable for all direct access interface registers. +- Offset: `0x74` +- Reset default: `0x1` +- Reset mask: `0x1` + +### Fields + +```wavejson +{"reg": [{"name": "DIRECT_ACCESS_REGWEN", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 220}} +``` + +| Bits | Type | Reset | Name | +|:------:|:------:|:-------:|:--------------------------------------------------------------------| +| 31:1 | | | Reserved | +| 0 | rw0c | 0x1 | [DIRECT_ACCESS_REGWEN](#direct_access_regwen--direct_access_regwen) | + +### DIRECT_ACCESS_REGWEN . DIRECT_ACCESS_REGWEN +This bit controls whether the DAI registers can be written. +Write 0 to it in order to clear the bit. + +Note that the hardware also modulates this bit and sets it to 0 temporarily +during an OTP operation such that the corresponding address and data registers +cannot be modified while an operation is pending. The [`DAI_IDLE`](#dai_idle) status bit +will also be set to 0 in such a case. + +## DIRECT_ACCESS_CMD +Command register for direct accesses. +- Offset: `0x78` +- Reset default: `0x0` +- Reset mask: `0x7` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "RD", "bits": 1, "attr": ["r0w1c"], "rotate": -90}, {"name": "WR", "bits": 1, "attr": ["r0w1c"], "rotate": -90}, {"name": "DIGEST", "bits": 1, "attr": ["r0w1c"], "rotate": -90}, {"bits": 29}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------|:-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| 31:3 | | | | Reserved | +| 2 | r0w1c | 0x0 | DIGEST | Initiates the digest calculation and locking sequence for the partition specified by [`DIRECT_ACCESS_ADDRESS.`](#direct_access_address) | +| 1 | r0w1c | 0x0 | WR | Initiates a programming sequence that writes the data in [`DIRECT_ACCESS_WDATA_0`](#direct_access_wdata_0) and [`DIRECT_ACCESS_WDATA_1`](#direct_access_wdata_1) (for 64bit partitions) to the location specified by [`DIRECT_ACCESS_ADDRESS.`](#direct_access_address) | +| 0 | r0w1c | 0x0 | RD | Initiates a readout sequence that reads the location specified by [`DIRECT_ACCESS_ADDRESS.`](#direct_access_address) The command places the data read into [`DIRECT_ACCESS_RDATA_0`](#direct_access_rdata_0) and [`DIRECT_ACCESS_RDATA_1`](#direct_access_rdata_1) (for 64bit partitions). | + +## DIRECT_ACCESS_ADDRESS +Address register for direct accesses. +- Offset: `0x7c` +- Reset default: `0x0` +- Reset mask: `0x3fff` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "DIRECT_ACCESS_ADDRESS", "bits": 14, "attr": ["rw"], "rotate": 0}, {"bits": 18}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | +|:------:|:------:|:-------:|:-----------------------------------------------------------------------| +| 31:14 | | | Reserved | +| 13:0 | rw | 0x0 | [DIRECT_ACCESS_ADDRESS](#direct_access_address--direct_access_address) | + +### DIRECT_ACCESS_ADDRESS . DIRECT_ACCESS_ADDRESS +This is the address for the OTP word to be read or written through +the direct access interface. Note that the address is aligned to the access size +internally, hence bits 1:0 are ignored for 32bit accesses, and bits 2:0 are ignored +for 64bit accesses. + +For the digest calculation command, set this register to the partition base offset. + +## DIRECT_ACCESS_WDATA +Write data for direct accesses. +Hardware automatically determines the access granule (32bit or 64bit) based on which +partition is being written to. +- Reset default: `0x0` +- Reset mask: `0xffffffff` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Instances + +| Name | Offset | +|:----------------------|:---------| +| DIRECT_ACCESS_WDATA_0 | 0x80 | +| DIRECT_ACCESS_WDATA_1 | 0x84 | + + +### Fields + +```wavejson +{"reg": [{"name": "DIRECT_ACCESS_WDATA", "bits": 32, "attr": ["rw"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:--------------------|:--------------| +| 31:0 | rw | 0x0 | DIRECT_ACCESS_WDATA | | + +## DIRECT_ACCESS_RDATA +Read data for direct accesses. +Hardware automatically determines the access granule (32bit or 64bit) based on which +partition is read from. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:----------------------|:---------| +| DIRECT_ACCESS_RDATA_0 | 0x88 | +| DIRECT_ACCESS_RDATA_1 | 0x8c | + + +### Fields + +```wavejson +{"reg": [{"name": "DIRECT_ACCESS_RDATA", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:--------------------|:--------------| +| 31:0 | ro | 0x0 | DIRECT_ACCESS_RDATA | | + +## CHECK_TRIGGER_REGWEN +Register write enable for [`CHECK_TRIGGER.`](#check_trigger) +- Offset: `0x90` +- Reset default: `0x1` +- Reset mask: `0x1` + +### Fields + +```wavejson +{"reg": [{"name": "CHECK_TRIGGER_REGWEN", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 220}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:---------------------|:------------------------------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | CHECK_TRIGGER_REGWEN | When cleared to 0, the [`CHECK_TRIGGER`](#check_trigger) register cannot be written anymore. Write 0 to clear this bit. | + +## CHECK_TRIGGER +Command register for direct accesses. +- Offset: `0x94` +- Reset default: `0x0` +- Reset mask: `0x3` +- Register enable: [`CHECK_TRIGGER_REGWEN`](#check_trigger_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "INTEGRITY", "bits": 1, "attr": ["r0w1c"], "rotate": -90}, {"name": "CONSISTENCY", "bits": 1, "attr": ["r0w1c"], "rotate": -90}, {"bits": 30}], "config": {"lanes": 1, "fontsize": 10, "vspace": 130}} +``` + +| Bits | Type | Reset | Name | +|:------:|:------:|:-------:|:-------------------------------------------| +| 31:2 | | | Reserved | +| 1 | r0w1c | 0x0 | [CONSISTENCY](#check_trigger--consistency) | +| 0 | r0w1c | 0x0 | [INTEGRITY](#check_trigger--integrity) | + +### CHECK_TRIGGER . CONSISTENCY +Writing 1 to this bit triggers a consistency check. SW should monitor [`STATUS.CHECK_PENDING`](#status) +and wait until the check has been completed. If there are any errors, those will be flagged +in the [`STATUS`](#status) and [`ERR_CODE`](#err_code) registers, and via interrupts and alerts. + +### CHECK_TRIGGER . INTEGRITY +Writing 1 to this bit triggers an integrity check. SW should monitor [`STATUS.CHECK_PENDING`](#status) +and wait until the check has been completed. If there are any errors, those will be flagged +in the [`STATUS`](#status) and [`ERR_CODE`](#err_code) registers, and via the interrupts and alerts. + +## CHECK_REGWEN +Register write enable for [`INTEGRITY_CHECK_PERIOD`](#integrity_check_period) and [`CONSISTENCY_CHECK_PERIOD.`](#consistency_check_period) +- Offset: `0x98` +- Reset default: `0x1` +- Reset mask: `0x1` + +### Fields + +```wavejson +{"reg": [{"name": "CHECK_REGWEN", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 140}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------------|:---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | CHECK_REGWEN | When cleared to 0, [`INTEGRITY_CHECK_PERIOD`](#integrity_check_period) and [`CONSISTENCY_CHECK_PERIOD`](#consistency_check_period) registers cannot be written anymore. Write 0 to clear this bit. | + +## CHECK_TIMEOUT +Timeout value for the integrity and consistency checks. +- Offset: `0x9c` +- Reset default: `0x0` +- Reset mask: `0xffffffff` +- Register enable: [`CHECK_REGWEN`](#check_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "CHECK_TIMEOUT", "bits": 32, "attr": ["rw"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | +|:------:|:------:|:-------:|:-----------------------------------------------| +| 31:0 | rw | 0x0 | [CHECK_TIMEOUT](#check_timeout--check_timeout) | + +### CHECK_TIMEOUT . CHECK_TIMEOUT +Timeout value in cycles for the for the integrity and consistency checks. If an integrity or consistency +check does not complete within the timeout window, an error will be flagged in the [`STATUS`](#status) register, +an otp_error interrupt will be raised, and an fatal_check_error alert will be sent out. The timeout should +be set to a large value to stay on the safe side. The maximum check time can be upper bounded by the +number of cycles it takes to readout, scramble and digest the entire OTP array. Since this amounts to +roughly 25k cycles, it is recommended to set this value to at least 100'000 cycles in order to stay on the +safe side. A value of zero disables the timeout mechanism (default). + +## INTEGRITY_CHECK_PERIOD +This value specifies the maximum period that can be generated pseudo-randomly. +Only applies to the HW_CFG* and SECRET* partitions once they are locked. +- Offset: `0xa0` +- Reset default: `0x0` +- Reset mask: `0xffffffff` +- Register enable: [`CHECK_REGWEN`](#check_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "INTEGRITY_CHECK_PERIOD", "bits": 32, "attr": ["rw"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | +|:------:|:------:|:-------:|:--------------------------------------------------------------------------| +| 31:0 | rw | 0x0 | [INTEGRITY_CHECK_PERIOD](#integrity_check_period--integrity_check_period) | + +### INTEGRITY_CHECK_PERIOD . INTEGRITY_CHECK_PERIOD +The pseudo-random period is generated using a 40bit LFSR internally, and this register defines +the bit mask to be applied to the LFSR output in order to limit its range. The value of this +register is left shifted by 8bits and the lower bits are set to 8'hFF in order to form the 40bit mask. +A recommended value is 0x3_FFFF, corresponding to a maximum period of ~2.8s at 24MHz. +A value of zero disables the timer (default). Note that a one-off check can always be triggered via +[`CHECK_TRIGGER.INTEGRITY.`](#check_trigger) + +## CONSISTENCY_CHECK_PERIOD +This value specifies the maximum period that can be generated pseudo-randomly. +This applies to the LIFE_CYCLE partition and the HW_CFG* and SECRET* partitions once they are locked. +- Offset: `0xa4` +- Reset default: `0x0` +- Reset mask: `0xffffffff` +- Register enable: [`CHECK_REGWEN`](#check_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "CONSISTENCY_CHECK_PERIOD", "bits": 32, "attr": ["rw"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | +|:------:|:------:|:-------:|:--------------------------------------------------------------------------------| +| 31:0 | rw | 0x0 | [CONSISTENCY_CHECK_PERIOD](#consistency_check_period--consistency_check_period) | + +### CONSISTENCY_CHECK_PERIOD . CONSISTENCY_CHECK_PERIOD +The pseudo-random period is generated using a 40bit LFSR internally, and this register defines +the bit mask to be applied to the LFSR output in order to limit its range. The value of this +register is left shifted by 8bits and the lower bits are set to 8'hFF in order to form the 40bit mask. +A recommended value is 0x3FF_FFFF, corresponding to a maximum period of ~716s at 24MHz. +A value of zero disables the timer (default). Note that a one-off check can always be triggered via +[`CHECK_TRIGGER.CONSISTENCY.`](#check_trigger) + +## VENDOR_TEST_READ_LOCK +Runtime read lock for the VENDOR_TEST partition. +- Offset: `0xa8` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "VENDOR_TEST_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 230}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:----------------------|:--------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | VENDOR_TEST_READ_LOCK | When cleared to 0, read access to the VENDOR_TEST partition is locked. Write 0 to clear this bit. | + +## CREATOR_SW_CFG_READ_LOCK +Runtime read lock for the CREATOR_SW_CFG partition. +- Offset: `0xac` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "CREATOR_SW_CFG_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 260}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------------------------|:-----------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | CREATOR_SW_CFG_READ_LOCK | When cleared to 0, read access to the CREATOR_SW_CFG partition is locked. Write 0 to clear this bit. | + +## OWNER_SW_CFG_READ_LOCK +Runtime read lock for the OWNER_SW_CFG partition. +- Offset: `0xb0` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "OWNER_SW_CFG_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 240}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-----------------------|:---------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | OWNER_SW_CFG_READ_LOCK | When cleared to 0, read access to the OWNER_SW_CFG partition is locked. Write 0 to clear this bit. | + +## OWNERSHIP_SLOT_STATE_READ_LOCK +Runtime read lock for the OWNERSHIP_SLOT_STATE partition. +- Offset: `0xb4` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "OWNERSHIP_SLOT_STATE_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 320}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------------------------------|:-----------------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | OWNERSHIP_SLOT_STATE_READ_LOCK | When cleared to 0, read access to the OWNERSHIP_SLOT_STATE partition is locked. Write 0 to clear this bit. | + +## ROT_CREATOR_AUTH_READ_LOCK +Runtime read lock for the ROT_CREATOR_AUTH partition. +- Offset: `0xb8` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "ROT_CREATOR_AUTH_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 280}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:---------------------------|:-------------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | ROT_CREATOR_AUTH_READ_LOCK | When cleared to 0, read access to the ROT_CREATOR_AUTH partition is locked. Write 0 to clear this bit. | + +## ROT_OWNER_AUTH_SLOT0_READ_LOCK +Runtime read lock for the ROT_OWNER_AUTH_SLOT0 partition. +- Offset: `0xbc` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "ROT_OWNER_AUTH_SLOT0_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 320}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------------------------------|:-----------------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | ROT_OWNER_AUTH_SLOT0_READ_LOCK | When cleared to 0, read access to the ROT_OWNER_AUTH_SLOT0 partition is locked. Write 0 to clear this bit. | + +## ROT_OWNER_AUTH_SLOT1_READ_LOCK +Runtime read lock for the ROT_OWNER_AUTH_SLOT1 partition. +- Offset: `0xc0` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "ROT_OWNER_AUTH_SLOT1_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 320}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------------------------------|:-----------------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | ROT_OWNER_AUTH_SLOT1_READ_LOCK | When cleared to 0, read access to the ROT_OWNER_AUTH_SLOT1 partition is locked. Write 0 to clear this bit. | + +## PLAT_INTEG_AUTH_SLOT0_READ_LOCK +Runtime read lock for the PLAT_INTEG_AUTH_SLOT0 partition. +- Offset: `0xc4` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "PLAT_INTEG_AUTH_SLOT0_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 330}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:--------------------------------|:------------------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | PLAT_INTEG_AUTH_SLOT0_READ_LOCK | When cleared to 0, read access to the PLAT_INTEG_AUTH_SLOT0 partition is locked. Write 0 to clear this bit. | + +## PLAT_INTEG_AUTH_SLOT1_READ_LOCK +Runtime read lock for the PLAT_INTEG_AUTH_SLOT1 partition. +- Offset: `0xc8` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "PLAT_INTEG_AUTH_SLOT1_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 330}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:--------------------------------|:------------------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | PLAT_INTEG_AUTH_SLOT1_READ_LOCK | When cleared to 0, read access to the PLAT_INTEG_AUTH_SLOT1 partition is locked. Write 0 to clear this bit. | + +## PLAT_OWNER_AUTH_SLOT0_READ_LOCK +Runtime read lock for the PLAT_OWNER_AUTH_SLOT0 partition. +- Offset: `0xcc` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "PLAT_OWNER_AUTH_SLOT0_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 330}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:--------------------------------|:------------------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | PLAT_OWNER_AUTH_SLOT0_READ_LOCK | When cleared to 0, read access to the PLAT_OWNER_AUTH_SLOT0 partition is locked. Write 0 to clear this bit. | + +## PLAT_OWNER_AUTH_SLOT1_READ_LOCK +Runtime read lock for the PLAT_OWNER_AUTH_SLOT1 partition. +- Offset: `0xd0` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "PLAT_OWNER_AUTH_SLOT1_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 330}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:--------------------------------|:------------------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | PLAT_OWNER_AUTH_SLOT1_READ_LOCK | When cleared to 0, read access to the PLAT_OWNER_AUTH_SLOT1 partition is locked. Write 0 to clear this bit. | + +## PLAT_OWNER_AUTH_SLOT2_READ_LOCK +Runtime read lock for the PLAT_OWNER_AUTH_SLOT2 partition. +- Offset: `0xd4` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "PLAT_OWNER_AUTH_SLOT2_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 330}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:--------------------------------|:------------------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | PLAT_OWNER_AUTH_SLOT2_READ_LOCK | When cleared to 0, read access to the PLAT_OWNER_AUTH_SLOT2 partition is locked. Write 0 to clear this bit. | + +## PLAT_OWNER_AUTH_SLOT3_READ_LOCK +Runtime read lock for the PLAT_OWNER_AUTH_SLOT3 partition. +- Offset: `0xd8` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "PLAT_OWNER_AUTH_SLOT3_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 330}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:--------------------------------|:------------------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | PLAT_OWNER_AUTH_SLOT3_READ_LOCK | When cleared to 0, read access to the PLAT_OWNER_AUTH_SLOT3 partition is locked. Write 0 to clear this bit. | + +## EXT_NVM_READ_LOCK +Runtime read lock for the EXT_NVM partition. +- Offset: `0xdc` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "EXT_NVM_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 190}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:------------------|:----------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | EXT_NVM_READ_LOCK | When cleared to 0, read access to the EXT_NVM partition is locked. Write 0 to clear this bit. | + +## ROM_PATCH_READ_LOCK +Runtime read lock for the ROM_PATCH partition. +- Offset: `0xe0` +- Reset default: `0x1` +- Reset mask: `0x1` +- Register enable: [`DIRECT_ACCESS_REGWEN`](#direct_access_regwen) + +### Fields + +```wavejson +{"reg": [{"name": "ROM_PATCH_READ_LOCK", "bits": 1, "attr": ["rw0c"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 210}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:--------------------|:------------------------------------------------------------------------------------------------| +| 31:1 | | | | Reserved | +| 0 | rw0c | 0x1 | ROM_PATCH_READ_LOCK | When cleared to 0, read access to the ROM_PATCH partition is locked. Write 0 to clear this bit. | + +## VENDOR_TEST_DIGEST +Integrity digest for the VENDOR_TEST partition. +The integrity digest is 0 by default. Software must write this +digest value via the direct access interface in order to lock the partition. +After a reset, write access to the VENDOR_TEST partition is locked and +the digest becomes visible in this CSR. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:---------------------|:---------| +| VENDOR_TEST_DIGEST_0 | 0xe4 | +| VENDOR_TEST_DIGEST_1 | 0xe8 | + + +### Fields + +```wavejson +{"reg": [{"name": "VENDOR_TEST_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------------------|:--------------| +| 31:0 | ro | 0x0 | VENDOR_TEST_DIGEST | | + +## CREATOR_SW_CFG_DIGEST +Integrity digest for the CREATOR_SW_CFG partition. +The integrity digest is 0 by default. Software must write this +digest value via the direct access interface in order to lock the partition. +After a reset, write access to the CREATOR_SW_CFG partition is locked and +the digest becomes visible in this CSR. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:------------------------|:---------| +| CREATOR_SW_CFG_DIGEST_0 | 0xec | +| CREATOR_SW_CFG_DIGEST_1 | 0xf0 | + + +### Fields + +```wavejson +{"reg": [{"name": "CREATOR_SW_CFG_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:----------------------|:--------------| +| 31:0 | ro | 0x0 | CREATOR_SW_CFG_DIGEST | | + +## OWNER_SW_CFG_DIGEST +Integrity digest for the OWNER_SW_CFG partition. +The integrity digest is 0 by default. Software must write this +digest value via the direct access interface in order to lock the partition. +After a reset, write access to the OWNER_SW_CFG partition is locked and +the digest becomes visible in this CSR. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:----------------------|:---------| +| OWNER_SW_CFG_DIGEST_0 | 0xf4 | +| OWNER_SW_CFG_DIGEST_1 | 0xf8 | + + +### Fields + +```wavejson +{"reg": [{"name": "OWNER_SW_CFG_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:--------------------|:--------------| +| 31:0 | ro | 0x0 | OWNER_SW_CFG_DIGEST | | + +## ROT_CREATOR_AUTH_DIGEST +Integrity digest for the ROT_CREATOR_AUTH partition. +The integrity digest is 0 by default. Software must write this +digest value via the direct access interface in order to lock the partition. +After a reset, write access to the ROT_CREATOR_AUTH partition is locked and +the digest becomes visible in this CSR. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:--------------------------|:---------| +| ROT_CREATOR_AUTH_DIGEST_0 | 0xfc | +| ROT_CREATOR_AUTH_DIGEST_1 | 0x100 | + + +### Fields + +```wavejson +{"reg": [{"name": "ROT_CREATOR_AUTH_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:------------------------|:--------------| +| 31:0 | ro | 0x0 | ROT_CREATOR_AUTH_DIGEST | | + +## ROT_OWNER_AUTH_SLOT0_DIGEST +Integrity digest for the ROT_OWNER_AUTH_SLOT0 partition. +The integrity digest is 0 by default. Software must write this +digest value via the direct access interface in order to lock the partition. +After a reset, write access to the ROT_OWNER_AUTH_SLOT0 partition is locked and +the digest becomes visible in this CSR. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:------------------------------|:---------| +| ROT_OWNER_AUTH_SLOT0_DIGEST_0 | 0x104 | +| ROT_OWNER_AUTH_SLOT0_DIGEST_1 | 0x108 | + + +### Fields + +```wavejson +{"reg": [{"name": "ROT_OWNER_AUTH_SLOT0_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:----------------------------|:--------------| +| 31:0 | ro | 0x0 | ROT_OWNER_AUTH_SLOT0_DIGEST | | + +## ROT_OWNER_AUTH_SLOT1_DIGEST +Integrity digest for the ROT_OWNER_AUTH_SLOT1 partition. +The integrity digest is 0 by default. Software must write this +digest value via the direct access interface in order to lock the partition. +After a reset, write access to the ROT_OWNER_AUTH_SLOT1 partition is locked and +the digest becomes visible in this CSR. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:------------------------------|:---------| +| ROT_OWNER_AUTH_SLOT1_DIGEST_0 | 0x10c | +| ROT_OWNER_AUTH_SLOT1_DIGEST_1 | 0x110 | + + +### Fields + +```wavejson +{"reg": [{"name": "ROT_OWNER_AUTH_SLOT1_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:----------------------------|:--------------| +| 31:0 | ro | 0x0 | ROT_OWNER_AUTH_SLOT1_DIGEST | | + +## PLAT_INTEG_AUTH_SLOT0_DIGEST +Integrity digest for the PLAT_INTEG_AUTH_SLOT0 partition. +The integrity digest is 0 by default. Software must write this +digest value via the direct access interface in order to lock the partition. +After a reset, write access to the PLAT_INTEG_AUTH_SLOT0 partition is locked and +the digest becomes visible in this CSR. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:-------------------------------|:---------| +| PLAT_INTEG_AUTH_SLOT0_DIGEST_0 | 0x114 | +| PLAT_INTEG_AUTH_SLOT0_DIGEST_1 | 0x118 | + + +### Fields + +```wavejson +{"reg": [{"name": "PLAT_INTEG_AUTH_SLOT0_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-----------------------------|:--------------| +| 31:0 | ro | 0x0 | PLAT_INTEG_AUTH_SLOT0_DIGEST | | + +## PLAT_INTEG_AUTH_SLOT1_DIGEST +Integrity digest for the PLAT_INTEG_AUTH_SLOT1 partition. +The integrity digest is 0 by default. Software must write this +digest value via the direct access interface in order to lock the partition. +After a reset, write access to the PLAT_INTEG_AUTH_SLOT1 partition is locked and +the digest becomes visible in this CSR. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:-------------------------------|:---------| +| PLAT_INTEG_AUTH_SLOT1_DIGEST_0 | 0x11c | +| PLAT_INTEG_AUTH_SLOT1_DIGEST_1 | 0x120 | + + +### Fields + +```wavejson +{"reg": [{"name": "PLAT_INTEG_AUTH_SLOT1_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-----------------------------|:--------------| +| 31:0 | ro | 0x0 | PLAT_INTEG_AUTH_SLOT1_DIGEST | | + +## PLAT_OWNER_AUTH_SLOT0_DIGEST +Integrity digest for the PLAT_OWNER_AUTH_SLOT0 partition. +The integrity digest is 0 by default. Software must write this +digest value via the direct access interface in order to lock the partition. +After a reset, write access to the PLAT_OWNER_AUTH_SLOT0 partition is locked and +the digest becomes visible in this CSR. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:-------------------------------|:---------| +| PLAT_OWNER_AUTH_SLOT0_DIGEST_0 | 0x124 | +| PLAT_OWNER_AUTH_SLOT0_DIGEST_1 | 0x128 | + + +### Fields + +```wavejson +{"reg": [{"name": "PLAT_OWNER_AUTH_SLOT0_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-----------------------------|:--------------| +| 31:0 | ro | 0x0 | PLAT_OWNER_AUTH_SLOT0_DIGEST | | + +## PLAT_OWNER_AUTH_SLOT1_DIGEST +Integrity digest for the PLAT_OWNER_AUTH_SLOT1 partition. +The integrity digest is 0 by default. Software must write this +digest value via the direct access interface in order to lock the partition. +After a reset, write access to the PLAT_OWNER_AUTH_SLOT1 partition is locked and +the digest becomes visible in this CSR. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:-------------------------------|:---------| +| PLAT_OWNER_AUTH_SLOT1_DIGEST_0 | 0x12c | +| PLAT_OWNER_AUTH_SLOT1_DIGEST_1 | 0x130 | + + +### Fields + +```wavejson +{"reg": [{"name": "PLAT_OWNER_AUTH_SLOT1_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-----------------------------|:--------------| +| 31:0 | ro | 0x0 | PLAT_OWNER_AUTH_SLOT1_DIGEST | | + +## PLAT_OWNER_AUTH_SLOT2_DIGEST +Integrity digest for the PLAT_OWNER_AUTH_SLOT2 partition. +The integrity digest is 0 by default. Software must write this +digest value via the direct access interface in order to lock the partition. +After a reset, write access to the PLAT_OWNER_AUTH_SLOT2 partition is locked and +the digest becomes visible in this CSR. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:-------------------------------|:---------| +| PLAT_OWNER_AUTH_SLOT2_DIGEST_0 | 0x134 | +| PLAT_OWNER_AUTH_SLOT2_DIGEST_1 | 0x138 | + + +### Fields + +```wavejson +{"reg": [{"name": "PLAT_OWNER_AUTH_SLOT2_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-----------------------------|:--------------| +| 31:0 | ro | 0x0 | PLAT_OWNER_AUTH_SLOT2_DIGEST | | + +## PLAT_OWNER_AUTH_SLOT3_DIGEST +Integrity digest for the PLAT_OWNER_AUTH_SLOT3 partition. +The integrity digest is 0 by default. Software must write this +digest value via the direct access interface in order to lock the partition. +After a reset, write access to the PLAT_OWNER_AUTH_SLOT3 partition is locked and +the digest becomes visible in this CSR. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:-------------------------------|:---------| +| PLAT_OWNER_AUTH_SLOT3_DIGEST_0 | 0x13c | +| PLAT_OWNER_AUTH_SLOT3_DIGEST_1 | 0x140 | + + +### Fields + +```wavejson +{"reg": [{"name": "PLAT_OWNER_AUTH_SLOT3_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-----------------------------|:--------------| +| 31:0 | ro | 0x0 | PLAT_OWNER_AUTH_SLOT3_DIGEST | | + +## ROM_PATCH_DIGEST +Integrity digest for the ROM_PATCH partition. +The integrity digest is 0 by default. Software must write this +digest value via the direct access interface in order to lock the partition. +After a reset, write access to the ROM_PATCH partition is locked and +the digest becomes visible in this CSR. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:-------------------|:---------| +| ROM_PATCH_DIGEST_0 | 0x144 | +| ROM_PATCH_DIGEST_1 | 0x148 | + + +### Fields + +```wavejson +{"reg": [{"name": "ROM_PATCH_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-----------------|:--------------| +| 31:0 | ro | 0x0 | ROM_PATCH_DIGEST | | + +## HW_CFG0_DIGEST +Integrity digest for the HW_CFG0 partition. +The integrity digest is 0 by default. The digest calculation can be triggered via the [`DIRECT_ACCESS_CMD.`](#direct_access_cmd) +After a reset, the digest then becomes visible in this CSR, and the corresponding partition becomes write-locked. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:-----------------|:---------| +| HW_CFG0_DIGEST_0 | 0x14c | +| HW_CFG0_DIGEST_1 | 0x150 | + + +### Fields + +```wavejson +{"reg": [{"name": "HW_CFG0_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:---------------|:--------------| +| 31:0 | ro | 0x0 | HW_CFG0_DIGEST | | + +## HW_CFG1_DIGEST +Integrity digest for the HW_CFG1 partition. +The integrity digest is 0 by default. The digest calculation can be triggered via the [`DIRECT_ACCESS_CMD.`](#direct_access_cmd) +After a reset, the digest then becomes visible in this CSR, and the corresponding partition becomes write-locked. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:-----------------|:---------| +| HW_CFG1_DIGEST_0 | 0x154 | +| HW_CFG1_DIGEST_1 | 0x158 | + + +### Fields + +```wavejson +{"reg": [{"name": "HW_CFG1_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:---------------|:--------------| +| 31:0 | ro | 0x0 | HW_CFG1_DIGEST | | + +## SECRET0_DIGEST +Integrity digest for the SECRET0 partition. +The integrity digest is 0 by default. The digest calculation can be triggered via the [`DIRECT_ACCESS_CMD.`](#direct_access_cmd) +After a reset, the digest then becomes visible in this CSR, and the corresponding partition becomes write-locked. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:-----------------|:---------| +| SECRET0_DIGEST_0 | 0x15c | +| SECRET0_DIGEST_1 | 0x160 | + + +### Fields + +```wavejson +{"reg": [{"name": "SECRET0_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:---------------|:--------------| +| 31:0 | ro | 0x0 | SECRET0_DIGEST | | + +## SECRET1_DIGEST +Integrity digest for the SECRET1 partition. +The integrity digest is 0 by default. The digest calculation can be triggered via the [`DIRECT_ACCESS_CMD.`](#direct_access_cmd) +After a reset, the digest then becomes visible in this CSR, and the corresponding partition becomes write-locked. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:-----------------|:---------| +| SECRET1_DIGEST_0 | 0x164 | +| SECRET1_DIGEST_1 | 0x168 | + + +### Fields + +```wavejson +{"reg": [{"name": "SECRET1_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:---------------|:--------------| +| 31:0 | ro | 0x0 | SECRET1_DIGEST | | + +## SECRET2_DIGEST +Integrity digest for the SECRET2 partition. +The integrity digest is 0 by default. The digest calculation can be triggered via the [`DIRECT_ACCESS_CMD.`](#direct_access_cmd) +After a reset, the digest then becomes visible in this CSR, and the corresponding partition becomes write-locked. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:-----------------|:---------| +| SECRET2_DIGEST_0 | 0x16c | +| SECRET2_DIGEST_1 | 0x170 | + + +### Fields + +```wavejson +{"reg": [{"name": "SECRET2_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:---------------|:--------------| +| 31:0 | ro | 0x0 | SECRET2_DIGEST | | + +## SECRET3_DIGEST +Integrity digest for the SECRET3 partition. +The integrity digest is 0 by default. The digest calculation can be triggered via the [`DIRECT_ACCESS_CMD.`](#direct_access_cmd) +After a reset, the digest then becomes visible in this CSR, and the corresponding partition becomes write-locked. +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Instances + +| Name | Offset | +|:-----------------|:---------| +| SECRET3_DIGEST_0 | 0x174 | +| SECRET3_DIGEST_1 | 0x178 | + + +### Fields + +```wavejson +{"reg": [{"name": "SECRET3_DIGEST", "bits": 32, "attr": ["ro"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:---------------|:--------------| +| 31:0 | ro | 0x0 | SECRET3_DIGEST | | + +## SW_CFG_WINDOW +Any read to this window directly maps to the corresponding offset in the creator and owner software +config partitions, and triggers an OTP readout of the bytes requested. Note that the transaction +will block until OTP readout has completed. + +- Word Aligned Offset Range: `0x4000`to`0x7ffc` +- Size (words): `4096` +- Access: `ro` +- Byte writes are *not* supported. + +## Summary of the **`prim`** interface's registers + +| Name | Offset | Length | Description | +|:-------------------------|:---------|---------:|:--------------| +| otp_ctrl.[`CSR0`](#csr0) | 0x0 | 4 | | +| otp_ctrl.[`CSR1`](#csr1) | 0x4 | 4 | | +| otp_ctrl.[`CSR2`](#csr2) | 0x8 | 4 | | +| otp_ctrl.[`CSR3`](#csr3) | 0xc | 4 | | +| otp_ctrl.[`CSR4`](#csr4) | 0x10 | 4 | | +| otp_ctrl.[`CSR5`](#csr5) | 0x14 | 4 | | +| otp_ctrl.[`CSR6`](#csr6) | 0x18 | 4 | | +| otp_ctrl.[`CSR7`](#csr7) | 0x1c | 4 | | + +## CSR0 + +- Offset: `0x0` +- Reset default: `0x0` +- Reset mask: `0x7ff3ff7` + +### Fields + +```wavejson +{"reg": [{"name": "field0", "bits": 1, "attr": ["rw"], "rotate": -90}, {"name": "field1", "bits": 1, "attr": ["rw"], "rotate": -90}, {"name": "field2", "bits": 1, "attr": ["rw"], "rotate": -90}, {"bits": 1}, {"name": "field3", "bits": 10, "attr": ["rw"], "rotate": 0}, {"bits": 2}, {"name": "field4", "bits": 11, "attr": ["rw"], "rotate": 0}, {"bits": 5}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------|:--------------| +| 31:27 | | | | Reserved | +| 26:16 | rw | 0x0 | field4 | | +| 15:14 | | | | Reserved | +| 13:4 | rw | 0x0 | field3 | | +| 3 | | | | Reserved | +| 2 | rw | 0x0 | field2 | | +| 1 | rw | 0x0 | field1 | | +| 0 | rw | 0x0 | field0 | | + +## CSR1 + +- Offset: `0x4` +- Reset default: `0x0` +- Reset mask: `0xffffffff` + +### Fields + +```wavejson +{"reg": [{"name": "field0", "bits": 7, "attr": ["rw"], "rotate": 0}, {"name": "field1", "bits": 1, "attr": ["rw"], "rotate": -90}, {"name": "field2", "bits": 7, "attr": ["rw"], "rotate": 0}, {"name": "field3", "bits": 1, "attr": ["rw"], "rotate": -90}, {"name": "field4", "bits": 16, "attr": ["rw"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------|:--------------| +| 31:16 | rw | 0x0 | field4 | | +| 15 | rw | 0x0 | field3 | | +| 14:8 | rw | 0x0 | field2 | | +| 7 | rw | 0x0 | field1 | | +| 6:0 | rw | 0x0 | field0 | | + +## CSR2 + +- Offset: `0x8` +- Reset default: `0x0` +- Reset mask: `0x1` + +### Fields + +```wavejson +{"reg": [{"name": "field0", "bits": 1, "attr": ["rw"], "rotate": -90}, {"bits": 31}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------|:--------------| +| 31:1 | | | | Reserved | +| 0 | rw | 0x0 | field0 | | + +## CSR3 + +- Offset: `0xc` +- Reset default: `0x0` +- Reset mask: `0x7f3ff7` + +### Fields + +```wavejson +{"reg": [{"name": "field0", "bits": 3, "attr": ["rw1c"], "rotate": -90}, {"bits": 1}, {"name": "field1", "bits": 10, "attr": ["rw1c"], "rotate": 0}, {"bits": 2}, {"name": "field2", "bits": 1, "attr": ["rw1c"], "rotate": -90}, {"name": "field3", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "field4", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "field5", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "field6", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "field7", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "field8", "bits": 1, "attr": ["ro"], "rotate": -90}, {"bits": 9}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------|:--------------| +| 31:23 | | | | Reserved | +| 22 | ro | 0x0 | field8 | | +| 21 | ro | 0x0 | field7 | | +| 20 | ro | 0x0 | field6 | | +| 19 | ro | 0x0 | field5 | | +| 18 | ro | 0x0 | field4 | | +| 17 | ro | 0x0 | field3 | | +| 16 | rw1c | 0x0 | field2 | | +| 15:14 | | | | Reserved | +| 13:4 | rw1c | 0x0 | field1 | | +| 3 | | | | Reserved | +| 2:0 | rw1c | 0x0 | field0 | | + +## CSR4 + +- Offset: `0x10` +- Reset default: `0x0` +- Reset mask: `0x73ff` + +### Fields + +```wavejson +{"reg": [{"name": "field0", "bits": 10, "attr": ["rw"], "rotate": 0}, {"bits": 2}, {"name": "field1", "bits": 1, "attr": ["rw"], "rotate": -90}, {"name": "field2", "bits": 1, "attr": ["rw"], "rotate": -90}, {"name": "field3", "bits": 1, "attr": ["rw"], "rotate": -90}, {"bits": 17}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------|:--------------| +| 31:15 | | | | Reserved | +| 14 | rw | 0x0 | field3 | | +| 13 | rw | 0x0 | field2 | | +| 12 | rw | 0x0 | field1 | | +| 11:10 | | | | Reserved | +| 9:0 | rw | 0x0 | field0 | | + +## CSR5 + +- Offset: `0x14` +- Reset default: `0x0` +- Reset mask: `0xffff3fff` + +### Fields + +```wavejson +{"reg": [{"name": "field0", "bits": 6, "attr": ["rw"], "rotate": 0}, {"name": "field1", "bits": 2, "attr": ["rw"], "rotate": -90}, {"name": "field2", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "field3", "bits": 3, "attr": ["ro"], "rotate": -90}, {"name": "field4", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "field5", "bits": 1, "attr": ["ro"], "rotate": -90}, {"bits": 2}, {"name": "field6", "bits": 16, "attr": ["rw"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------|:--------------| +| 31:16 | rw | 0x0 | field6 | | +| 15:14 | | | | Reserved | +| 13 | ro | 0x0 | field5 | | +| 12 | ro | 0x0 | field4 | | +| 11:9 | ro | 0x0 | field3 | | +| 8 | ro | 0x0 | field2 | | +| 7:6 | rw | 0x0 | field1 | | +| 5:0 | rw | 0x0 | field0 | | + +## CSR6 + +- Offset: `0x18` +- Reset default: `0x0` +- Reset mask: `0xffff1bff` + +### Fields + +```wavejson +{"reg": [{"name": "field0", "bits": 10, "attr": ["rw"], "rotate": 0}, {"bits": 1}, {"name": "field1", "bits": 1, "attr": ["rw"], "rotate": -90}, {"name": "field2", "bits": 1, "attr": ["rw"], "rotate": -90}, {"bits": 3}, {"name": "field3", "bits": 16, "attr": ["rw"], "rotate": 0}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------|:--------------| +| 31:16 | rw | 0x0 | field3 | | +| 15:13 | | | | Reserved | +| 12 | rw | 0x0 | field2 | | +| 11 | rw | 0x0 | field1 | | +| 10 | | | | Reserved | +| 9:0 | rw | 0x0 | field0 | | + +## CSR7 + +- Offset: `0x1c` +- Reset default: `0x0` +- Reset mask: `0xc73f` + +### Fields + +```wavejson +{"reg": [{"name": "field0", "bits": 6, "attr": ["ro"], "rotate": 0}, {"bits": 2}, {"name": "field1", "bits": 3, "attr": ["ro"], "rotate": -90}, {"bits": 3}, {"name": "field2", "bits": 1, "attr": ["ro"], "rotate": -90}, {"name": "field3", "bits": 1, "attr": ["ro"], "rotate": -90}, {"bits": 16}], "config": {"lanes": 1, "fontsize": 10, "vspace": 80}} +``` + +| Bits | Type | Reset | Name | Description | +|:------:|:------:|:-------:|:-------|:--------------| +| 31:16 | | | | Reserved | +| 15 | ro | 0x0 | field3 | | +| 14 | ro | 0x0 | field2 | | +| 13:11 | | | | Reserved | +| 10:8 | ro | 0x0 | field1 | | +| 7:6 | | | | Reserved | +| 5:0 | ro | 0x0 | field0 | | + + + diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/theory_of_operation.md b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/theory_of_operation.md new file mode 100644 index 00000000000000..e3c5de4040a111 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/doc/theory_of_operation.md @@ -0,0 +1,514 @@ +# Theory of Operation + +Conceptually speaking, the OTP functionality is at a high level split into "front-end" and "back-end". +The "front-end" contains the logical partitions that feed the hardware and software consumer interfaces of the system. +The "back-end" represents the programming interface used by hardware and software components to stage the upcoming values. +The diagram below illustrates this behavioral model. + +![OTP Controller Block Diagram](otp_ctrl_behavioral_model.svg) + +Note that the front-end contains both buffered and unbuffered partitions. +Buffered partitions are sensed once per power cycle and their contents are stored in registers, whereas unbuffered partitions are read on-demand. +The former are typically partitions that contain data like hardware configuration bits, key material and the life cycle state that need to be always available to the hardware, whereas the latter are large partitions that are accessed infrequently, such as the software configurations. +Values that are programmed into a buffered partition via the programming interface (coupled with read verification) are merely "staged", and do not take effect until the next power cycle. + +The sections below describe the operation of various pieces of the OTP controller and how it supports the described functionality. + +## Logical Partitions + +The OTP is logically separated into partitions that represent different functions. +This means the isolation is virtual and maintained by the OTP controller instead of the underlying OTP IP. + +Within each logical partition, there are specific enforceable properties + +- Confidentiality via secret partitions + - This controls whether a particular partition contains secret data. + - If secret, a partition is not readable by software once locked, and is scrambled in storage. +- Read lockability + - This controls whether a particular partition disables software readability for later stage software. + - Some partitions can be locked statically (by computing and storing an associated digest in OTP), others can be read locked at runtime via CSRs. +- Write lockability + - This controls whether a partition is locked and prevented from future updates. + - A locked partition is stored alongside a digest to be used later for integrity verification. +- Integrity Verification + - Once a partition is write-locked by calculating and writing a non-zero [digest](#locking-a-partition) to it, it can undergo periodic verification (time-scale configurable by software). +This verification takes two forms, partition integrity checks, and storage consistency checks. + +Since the OTP is memory-like in nature (it only outputs a certain number of bits per address location), some of the logical partitions are buffered in registers for instantaneous and parallel access by hardware. +This is a critical point, since after power-up, these particular OTP contents are stored in flip flops and sourced to the system. +I.e., buffered partitions are **NOT** directly sourced from the OTP macro itself. +Thus the security of both volatile (OTP controller) and non-volatile (OTP IP) storage becomes important. + +### Partition Listing and Description + +The OTP controller for OpenTitan contains the seven logical partitions shown below. + +{{#include otp_ctrl_partitions.md}} + +Generally speaking, the production life cycle of a device is split into 5 stages "Manufacturing" -> "Calibration and Testing" -> "Provisioning" -> "Mission" -> "RMA". +OTP values are usually programmed during "Calibration and Testing", "Provisioning" and "RMA" stages, as explained below. +A detailed listing of all the items and the corresponding memory map can be found in the [Programmer's Guide](programmers_guide.md)) further below. + +### Calibration and Test + +During this stage, the device is tested for functionality and calibrated to ensure uniformity. +The calibration can focus on a number of things, but usually is centered around adjusting clock, voltage and timing sources to remove process variation. +These calibration values are programmed into the CREATOR_SW_CFG partition, as they are non-secret values meant to be read out by software and programmed into respective peripherals. + +Early on during this stage, the various tokens are also programmed into the secret partitions and harvested by the silicon creator. + +### Provisioning + +During this stage, the device is provisioned with the final firmware and a "unique" seed or identity. +The secret partitions are populated with root secrets and keys that are critical to establishing the device identity. + +As part of injecting the final firmware, the stock-keeping-unit-specific hardware and software configurations are also programmed. + +### Life Cycle Partition + +The life cycle partition is active throughout all stages and hence it is the **ONLY** partition that cannot be locked. +After the device finishes provisioning and goes into production, it must retain the ability to transition back to RMA in case of unexpected failures. + +In order to support this transition, the [life cycle state](../../../../ip/lc_ctrl/README.md) and counters must always be update-able. + +## Locking a Partition + +Write access to a partition can be permanently locked when software determines it will no longer make any updates to that partition. +To lock, an integrity constant is calculated and programmed alongside the other data of that partition. +The size of that integrity constant depends on the partition size granule, and is either 32bit or 64bit (see also [Direct Access Memory Map](#direct-access-memory-map)). + +Once the "integrity digest" is non-zero, no further updates are allowed. +If the partition is secret, software is in addition no longer able to read its contents (see [Secret Partition description](#secret-vs-nonsecret-partitions)). + +Note however, in all partitions, the digest itself is **ALWAYS** readable. +This gives software an opportunity to confirm that the locking operation has proceeded correctly, and if not, scrap the part immediately. + +Calculation of the integrity digest depends on whether the partition requires periodic background verification. + +### Vendor Test Partition + +The vendor test partition is intended to be used for OTP programming smoke checks during the manufacturing flow. +The silicon creator may implement these checks inside the proprietary version of the `prim_otp` wrapper. +This partition behaves like any other SW partition, with the exception that ECC uncorrectable errors will not lead to fatal errors / alerts as they do in all other partitions. +This is due to the nature of the OTP programming smoke checks, which may leave certain OTP words in a state inconsistent with the ECC polynomial employed upon OTP readout. + +### Software Configuration Partitions + +The software configuration partitions are used as non-volatile storage for flags, configuration and calibration data. +As such, the contents of this partition are usually consumed once as part of code execution, or moved to another storage compartment somewhere in the design. +For example, the clock calibration values and the LDO calibration values are programmed to the analog sensor top (AST) at startup. + +As such, it is not necessary to check periodically at the OTP source. +Instead, software can simply check as part of secure boot and take other measures when these values are programmed into peripherals. + +For this partition it is thus the responsibility of software to calculate the integrity digest and program it into the OTP. +It is also reasonable to shadow (parts of) this partition in main memory, and there is not an immediate impact from OTP contents to hardware. + +### Hardware Configuration and Secret Partitions + +The hardware and secret partitions directly affect downstream hardware. +The contents must go through periodic integrity checks and therefore the stored digest is calculated by hardware when software provides the intent to lock (as opposed to the software partitions where the digest has to be calculated by software). + +### Life Cycle Partition + +The life cycle partition cannot be locked and will therefore not contain a stored digest. +Note however that only the life cycle controller has access to this partition, i.e., the Direct Access Interface (DAI) cannot read nor write from/to the life cycle partition. + +## Secret vs Non-Secret Partitions + +Non-secret OTP partitions hold data that can be public; or data that has no impact on security. +For example, the current value of lock bits or clock calibration values. +These values are stored in OTP as plaintext. + +Secret partitions contain data that are critical to security, for example FLASH scrambling keys, device root secret and unlock tokens. +These values are stored scrambled in OTP, and are descrambled upon read. +The currently employed cipher is PRESENT, as it lends itself well to iterative decomposition, and it is a proven lightweight block cipher (see also [PRESENT Scrambling Primitive](../../../../ip/prim/doc/prim_present.md). +The usage of a block cipher however implies that the secret partitions can only be written in 64bit chunks. + +Further, the contents of a particular secret partition are not readable by software once locked (other than the digest which must be always readable); while non-secret partitions are always readable unless read accessibility is explicitly removed by software. + +Unfortunately, secret partitions must utilize a global netlist key for the scrambling operation, as there is no other non-volatile storage to store a unique key. + + +## Partition Checks + +### Integrity + +Once the appropriate partitions have been locked, the hardware integrity checker employs two integrity checks to verify the content of the volatile buffer registers: + +1. All buffered partitions have additional ECC protection (8bit ECC for each 64bit block) that is concurrently monitored. +2. The digest of the partition is recomputed at semi-random intervals and compared to the digest stored alongside the partition. + +The purpose of this check is NOT to check between the storage flops and the OTP, but whether the buffer register contents remain consistent with the calculated digest. +This verification is primarily concerned with whether the storage flops have experienced fault attacks. +This check applies to only the HW_CFG* and SECRET* partitions. +If a failure is encountered, the OTP controller will send out a `fatal_check_error` alert and reset all of its hardware outputs to their defaults. + +### Storage Consistency + +This verification ensures the value stored in the buffer registers remain consistent with those in the OTP. +This process re-reads the OTP at semi-random intervals and confirms the value read is the same as the value stored. +Note, given there are integrity checks in parallel, it is not necessary for some partitions to check ALL read contents for consistency. +If there is an integrity digest, only the digest needs to be read; otherwise, all values must be read. + + +This check applies to LIFE_CYCLE, HW_CFG* and SECRET* partitions. +If a failure is encountered, the OTP controller will send out a `fatal_check_error` alert and reset all of its hardware outputs to their defaults. + +Note that checks applied to life cycle could cause a failure if life cycle is updated, because life cycle is the only partition that may contain live updates. +The controller hence detects this condition based on the `lc_check_byp_en_i` signal coming from the life cycle controller, and pauses background checks on this partition in order to prevent false positives. + +### Secret Partition Integrity Checks + +Since the secret partitions are stored scrambled, this also implies the integrity digest is calculated over the scrambled form. +In order to balance the amount of buffer registers needed, only the decrypted form of the secret partitions is held in buffer registers. +Hardware calculates the digest by re-scrambling the data before passing it through the digest. + + +## Power-up and Sense + +The OTP controller partition storage must output a specified safe default (it is not always 0 like a blank OTP) upon reset release. +This default output must remain until the OTP controller completes all checks. + +The OTP controller reads from the OTP IP. +If the reads pass OTP IP internal checks (for example ECC or redundancy), the partition storage is updated; however the output is still held at the default state via an output mux. +After all read is complete, the OTP controller performs integrity checks on the HW_CFG* and SECRET* partitions. +If a partition fails the integrity checks at this point it would signal an initialization error in the status CSR and abort further initialization. + +After all integrity checks are complete, the OTP controller releases the output gating and marks outputs as valid. +However, any partition marked with "error" continues to hold its output in the default state. + +Once the above steps are complete, the partition storage in buffered registers is not updated again (except for updates to the life cycle partition through the life cycle interface). +I.e., values programmed to OTP via the programming interface will not be visible in buffered registers until after the next power cycle. + +At this point, outputs of the partition storage are NOT expected to change unless a periodic check suddenly fails. +When this failure occurs, all outputs are reverted to their default state, and an alert is immediately triggered to the alert handler. +For timing purposes, OTP outputs can be treated as semi-static, as this error event should be rare and exceptional. + + +## Partition Defaults + +Partition defaults are context specific. +For example, a hardware configuration item that locks down specific access should default to "no access". +This ensures that a glitch attack on the OTP cannot easily revert the design to an insecure state. + +This hence suggests that when an OTP is all 0's and all 1's, it should, whenever possible, reflect an invalid or inert state in the encoding space of the affected item. +This also implies the reset state of consuming agents (for example key manager and life cycle), should default to invalid / inert state as well. + + +## Program and Read Ports + +As shown previously, the OTP is split into a front and back end. +The back-end interface is primarily used to update OTP contents, and read back for debug and verification purposes. +Despite being a separate functional access port from the logical partitions, the program and read ports are subjected to the same access controls. + +When a partition is write-locked, programming accesses are disallowed. +If the partition is secret, read accesses by the back-end interface are also disallowed (except for the digest which must always be readable). +Software can also disable any read accesses to the software configuration partitions via CSR settings to prevent later stage software from reading any content. + +The exception to the above is the life cycle partition. +The life cycle controller interface also acts as a "back-end" interface that always has programming access to ensure life cycle state can be advanced. + +Note, the program and read ports can conflict with ongoing background storage checks, and the OTP controller arbitrates between these two sides. +An in-progress operation will always be completed. +Afterwards, or when two requests arrive at the same time, the priority is life cycle > programming interface > on-demand read accesses via CSR windows > background checks. + + +## Programming the OTP + +The OTP controller has two programming paths: + +1. a functional programming path through software (the program port), +2. Life cycle programming path through hardware. + +The functional interface is used to update all partitions except for life cycle. +As mentioned previously, any updates made during the current power cycle are **NOT** reflected in the buffered partitions until the next reboot. + +The life cycle interface is used to update the life cycle state and transition counter only. +The commands are issued from the [life cycle controller](../../../../ip/lc_ctrl/README.md), and similarly, successful or failed indications are also sent back to the life cycle controller. +Similar to the functional interface, the life cycle controller allows only one update per power cycle, and after a requested transition reverts to an inert state until reboot. + +For more details on how the software programs the OTP, please refer to the [Programmer's Guide](programmers_guide.md)) further below. + + +## Design Details + +### Block Diagram + +The following is a high-level block diagram that illustrates everything that has been discussed. + +![OTP Controller Block Diagram](otp_ctrl_blockdiag.svg) + +Each of the partitions P0-P7 has its [own controller FSM](#partition-implementations) that interacts with the OTP wrapper and the [scrambling datapath](#scrambling-datapath) to fulfill its tasks. +The partitions expose the address ranges and access control information to the Direct Access Interface (DAI) in order to block accesses that go to locked address ranges. +Further, the only two blocks that have (conditional) write access to the OTP are the DAI and the Life Cycle Interface (LCI) blocks. +The partitions can only issue read transactions to the OTP macro. +Note that the access ranges of the DAI and the LCI are mutually exclusive. +I.e., the DAI cannot read from nor write to the life cycle partition. +The LCI cannot read the OTP, but is allowed to write to the life cycle partition. + +The CSR node on the left side of this diagram connects to the DAI, the OTP partitions (P0-P7) and the OTP wrapper through a gated TL-UL interface. +All connections from the partitions to the CSR node are read-only, and typically only carry a subset of the information available. +E.g., the secret partitions only expose their digest value via the CSRs. + +The Key Derivation Interface (KDI) on the bottom right side interacts with the scrambling datapath, the EDN and the partition holding the scrambling root keys in order to derive static and ephemeral scrambling keys for FLASH and SRAM scrambling. + +The test access gate shown at the top of the block diagram is governed by the life cycle qualification signal `dft_en_i`, which is only enabled during the TEST_UNLOCKED* life cycle states. +Otherwise, test access via this TL-UL window is locked down. + +In addition to the blocks mentioned so far, the OTP controller also contains an LFSR timer that creates pseudo-randomly distributed partition check requests, and provides pseudo random data at high bandwidth in the event of a secure erase request due to chip-wide alert escalation. +For security reasons, the LFSR is periodically reseeded with entropy coming from EDN. + +### Data Allocation and Packing +#### Software View + +The effective word width of an OTP IP typically depends on a couple of factors, including the redundancy scheme employed. +For this the design at hand, it is assumed that this native OTP word-width is 16bit. +For software convenience, however, these details are abstracted and the open-source OTP controller exposes the OTP storage as a linear address space of 32bit words, which is aligned with the machine word size of the Ibex processor. +Since the OTP IP employs a redundancy mechanism similar to ECC, this implies however that write operations take place at a granularity of 32bit blocks for non-secret and 64bit blocks for secret partitions (due to the scrambling). +Hence, software is responsible to appropriately pack and program items, since each 32bit location can only be programmed once. + +#### Life Cycle View + +Since the life cycle partition is the only partition that needs live updates in-field, proper care must be taken to properly encode data in this partition such that incremental updates are possible. +The life cycle state is hence encoded such that incremental updates to the state are always carried out at the granularity of a 16bit word. +Further, the life cycle transition counter is encoded such that each stroke consumes a full 16bit word for the same reason. + +See [life cycle controller documentation](../../../../ip/lc_ctrl/README.md) for more details on the life cycle encoding. + +### Partition Controllers + +In RTL, we distinguish between buffered and unbuffered partition modules. +These are parameterized, such that we can assemble the array of OTP partitions with these two modules only. +The corresponding controller FSMs are explained in more detail below. + +#### Unbuffered Partition + +![Unbuffered Partition FSM](otp_ctrl_unbuf_part_fsm.svg) + +As shown above, the unbuffered partition module has a relatively simple controller FSM that only reads out the digest value of the partition upon initialization, and then basically waits for TL-UL read transactions to its corresponding window in the CSR space. + +Write access through the DAI will be locked in case the digest is set to a non-zero value. +Also, read access through the DAI and the CSR window can be locked at runtime via a CSR. +Read transactions through the CSR window will error out if they are out of bounds, or if read access is locked. + +Note that unrecoverable [OTP errors](#generalized-open-source-interface), ECC failures in the digest register or external escalation via `lc_escalate_en` will move the partition controller into a terminal error state. + +#### Buffered Partition + +![Buffered Partition FSM](otp_ctrl_buf_part_fsm.svg) + +The controller FSM of the buffered partition module is more complex than the unbuffered counterpart, since it has to account for scrambling and digest calculation. + +Upon initialization, the controller reads out the whole partition and descrambles it on the fly if needed. + +Then, right after the initial readout, the partition controller jumps into the first integrity check, which behaves somewhat differently, depending on whether the partition is digest protected (or not) and/or scrambled (or not). +If the partition is not digest protected, or if the digest has not yet been computed, the check completes right away, and the buffered values are released for hardware broadcast. +Otherwise, the partition contents in the buffer registers are re-scrambled if needed, and a digest is computed on the fly. +If the computed digest matches with the one that has been read out before, the buffered registers are released for hardware broadcast. +Otherwise, the buffered values are gated to their default, and an alert is triggered through the error handling logic. + +After initialization, the integrity check (as described above) and the consistency check can be triggered by the LFSR timer mechanism on a periodic basis. + +The consistency check behaves differently, depending on whether the partition is digest protected or not. +If it is, the consistency check will read out the digest stored in OTP and compare it with the value stored in the buffer register. +Otherwise, if no digest is available, the controller will read out the whole partition from OTP, and compare it to the contents stored in the buffer registers. +In case of a mismatch, the buffered values are gated to their default, and an alert is triggered through the error handling logic. + +Note that in case of unrecoverable OTP errors or ECC failures in the buffer registers, the partition controller FSM is moved into a terminal error state, which locks down all access through DAI and clamps the values that are broadcast in hardware to their defaults. + +External escalation via the `lc_escalate_en` signal will move the partition controller FSM into the terminal error state as well. +See [life cycle controller documentation](../../../../ip/lc_ctrl/README.md) for more details. + +### Direct Access Interface Control + +![Direct Access Interface FSM](otp_ctrl_dai_fsm.svg) + +Upon reset release, the DAI controller first sends an initialization command to the OTP macro. +Once the OTP macro becomes operational, an initialization request is sent to all partition controllers, which will read out and initialize the corresponding buffer registers. +The DAI then becomes operational once all partitions have initialized, and supports read, write and digest calculation commands (see [here](#direct-access-interface) for more information about how to interact with the DAI through the CSRs). + +Read and write commands transfer either 32bit or 64bit of data from the OTP to the corresponding CSR and vice versa. The access size is determined automatically, depending on whether the partition is scrambled or not. Also, (de)scrambling is performed transparently, depending on whether the partition is scrambled or not. + +Digest calculation commands read out the complete contents of a particular partition, compute a digest and write that digest value to the predefined location at the end of the partition. + +Note that any unrecoverable OTP error will move the DAI into a terminal error state, where all access through the DAI will be locked. +Also, the DAI consumes the read and write access information provided by the partition controller, and if a certain read or write access is not permitted, a recoverable error will be flagged in the status / error CSRs. + +### Life Cycle Interface Control + +![Life Cycle Interface FSM](otp_ctrl_lci_fsm.svg) + +Upon reset release the LCI FSM waits until the OTP controller has initialized and the LCI gets enabled. +Once it is in the idle state, life cycle state updates can be initiated via the life cycle interface as [described here](#state-transitions). +The LCI controller takes the life cycle state to be programmed and writes all 16bit words to OTP. +In case of unrecoverable OTP errors, the FSM signals an error to the life cycle controller and moves into a terminal error state. + +### Key Derivation Interface + +![Key Derivation Interface FSM](otp_ctrl_kdi_fsm.svg) + +Upon reset release the KDI FSM waits until the OTP controller has initialized and the KDI gets enabled. +Once it is in the idle state, key derivation can be requested via the [flash](#interface-to-flash-scrambler) and [sram](#interface-to-sram-and-otbn-scramblers) interfaces. +Based on which interface makes the request, the KDI controller will evaluate a variant of the PRESENT digest mechanism as described in more detail below. + +### Scrambling Datapath + +![OTP Digest Mechanism](otp_ctrl_digest_mechanism.svg) + +The scrambling datapath is built around an iterative implementation of the [PRESENT lightweight cipher](../../../../ip/prim/doc/prim_present.md) that performs one round per cycle. +The datapath contains some additional multiplexing circuitry to enable the DAI, KDI and partition controllers to evaluate different functions with the same datapath. +The algorithmic steps of these functions are explained in more detail below. + +#### Scrambling + +As illustrated in subfigure a) in the diagram above, the standard 128bit-key PRESENT configuration with 31 rounds is used for scrambling operations. +The key used for scrambling is a global netlist constant chosen by the silicon creator, and all secret partitions are encrypted using the their own distinct netlist constant. +Note that the amount of data that is being scrambled is small (160byte = 20 x 64bit blocks) and the scrambled data remains constant. +Hence, no additional masking or diversification scheme is applied since only a very limited amount of information can be gathered by observing the scrambling operation via side-channels. + +#### Digest Calculation + +The integrity digests used in the [partition checks](#partition-checks) are computed using a custom [Merkle-Damgard](https://en.wikipedia.org/wiki/Merkle%E2%80%93Damg%C3%A5rd_construction) scheme, where the employed one-way compression function F is constructed by using PRESENT in a [Davies-Meyer arrangement](https://en.wikipedia.org/wiki/One-way_compression_function#Davies%E2%80%93Meyer). +This is illustrated in subfigure b). + +At the beginning of the digest calculation the 64bit state is initialized with an initialization vector (IV). +Then, the data to be digested is split into 128bit chunks, each of which is used as a 128bit key input for updating the 64bit state with the compression function F. +Chunks that are not aligned with 128bit are padded with zero, and the finalization operation consists of another 31-round encryption pass with a finalization constant. +Note that both the IV as well as the finalization constant are global netlist constants chosen by the silicon creator. + +#### Scrambling Key Derivation + +The key derivation functions for ephemeral SRAM and static FLASH scrambling keys employ a similar construction as the digest calculation function. +In particular, the keys are derived by repeatedly reducing a (partially random) block of data into a 64bit block, as illustrated in subfigures c) and d). + +For ephemeral SRAM scrambling keys, the data block is composed of the 128bit SRAM_DATA_KEY_SEED stored in OTP, as well as 128bit of fresh entropy fetched from the EDN. +This process is repeated twice in order to produce a 128bit key. + +For static FLASH scrambling keys, the data block is composed of a 128bit part of either the FLASH_DATA_KEY_SEED or the FLASH_ADDR_KEY_SEED stored in OTP. +These key seeds are 256bit in size, allowing to use a unique chunk of 128bit of key seed data to derive a 64bit halve of a particular scrambling key. + +Note that the IV and finalization constants are distinct for SRAM and FLASH data and FLASH address scrambling keys. +These constants are chosen by the silicon creator prior to the tapeout. + +### Access Arbitration + +Access to the OTP wrapper and the scrambling datapath are both round-robin arbitrated, where the former arbitration occurs at cycle level (i.e., individual OTP memory accesses), and the latter occurs at the level of complete transactions (i.e., full digest or encryption). +Arbitration at transaction level is implemented similarly to cycle-based arbitration, with the difference that the grant signals remain asserted until the requestor deasserts the request (thereby releasing the arbiter, which acts as a mutex in this case). +This is behavior illustrated in the example below. + +```wavejson +{signal: [ + {name: 'clk_i', wave: 'p............'}, + {name: 'part_scrmbl_mtx_req[0]', wave: '01....0.1....'}, + {name: 'part_scrmbl_mtx_req[1]', wave: '0.1......0...'}, + {name: 'part_scrmbl_mtx_req[2]', wave: '0.1........0.'}, + {}, + {name: 'part_scrmbl_mtx_gnt[0]', wave: '01....0....1.'}, + {name: 'part_scrmbl_mtx_gnt[1]', wave: '0.....1..0...'}, + {name: 'part_scrmbl_mtx_gnt[2]', wave: '0........1.0.'}, +]} +``` + +### Primitive Wrapper and FPGA Emulation + +![OTP Wrapper Block Diagram](otp_ctrl_prim_otp.svg) + +The OTP IP is wrapped up in a primitive wrapper that exposes a TL-UL interface for testing purposes, and a generalized open-source interface for functional operation (described below). +Any OTP redundancy mechanism like per-word ECC is assumed to be handled inside the wrapper, which means that the word width exposed as part of the generalized interface is the effective word width. + +Note that the register space exposed via the TL-UL test interface, as well as DFT and power-related signals are dependent on the underlying proprietary OTP IP. +They are therefore not further described in this document. + +#### Generalized Open-source Interface + +The generalized open-source interface uses a couple of parameters (defaults set for Earlgrey configuration). + +Parameter | Default | Top Earlgrey | Description +---------------|---------|---------------|--------------- +`Width` | 16 | 16 | Native OTP word width. +`Depth` | 1024 | 1024 | Depth of OTP macro. +`CmdWidth` | 7 | 7 | Width of the OTP command. +`ErrWidth` | 3 | 3 | Width of error code output signal. +`PwrSeqWidth` | 2 | 2 | Width of power sequencing signals to/from AST. +`SizeWidth` | 2 | 2 | Width of the size field. +`IfWidth` | 2^`SizeWidth` * `Width` | 2^`SizeWidth` * `Width` | Data interface width. + +The generalized open-source interface is a simple command interface with a ready / valid handshake that makes it possible to introduce back pressure if the OTP macro is not able to accept a command due to an ongoing operation. + +In order to facilitate the scrambling and digest operations, the data width has been sized such that data blocks up to the PRESENT block size (64bit) can be transferred across the generalized interface. The actual size of a transfer is determined via the size_i field. Transfer sizes are specified in multiples of the native OTP block size, as listed below. + +Value of `size_i` | #Native OTP Words | Bit Slice +------------------|-------------------|------------ +2'b00 | 1 | `{word0} = data[15:0]` +2'b01 | 2 | `{word1, word0} = data[31:0]` +2'b10 | 3 | `{word2, word1, word0} = data[47:0]` +2'b11 | 4 | `{word3, word2, word1, word0} = data[63:0]` + +Responses are returned in-order via an unidirectional response interface (i.e., without back pressure capability). +Downstream logic must be able to sink the response in any case. +The response optionally carries read data, depending on whether the operation that took place was a read or not. +Also, an error signal returns a non-zero error code in case an error occurred while carrying out the OTP command. + +The signals pertaining to the generalized open-source interface are listed below. + +Signal | Direction | Type | Description +------------------------|------------------|-----------------------------|--------------- +`fatal_alert_o` | `output` | `logic` | Fatal alert output from the primitive. This is connected to a separate alert channel in the instantiating IP. The instantiating IP latches the alert indication and continuously outputs alert events until reset. +`recov_alert_o` | `output` | `logic` | Recoverable alert output from the primitive. This is connected to a separate alert channel in the instantiating IP. Should only be pulsed high for each alert occurrence. The instantiating IP then sends out a single alert event for each pulse. +`ready_o` | `output` | `logic` | Ready signal for the command handshake. +`valid_i` | `input` | `logic` | Valid signal for the command handshake. +`size_i` | `input` | `logic [SizeWidth-1:0]` | Number of native OTP words to transfer, minus one: `2'b00 = 1 native word` ... `2'b11 = 4 native words`. +`cmd_i` | `input` | `logic [CmdWidth-1:0]` | OTP command: `7'b1000101 = read`, `7'b0110111 = write`, `7'b1111001 = read raw`, `7'b1100010 = write raw`, `7'b0101100 = initialize` +`addr_i` | `input` | `logic [$clog2(Depth)-1:0]` | OTP word address. +`wdata_i` | `input` | `logic [IfWidth-1:0]` | Write data for write commands. +`valid_o` | `output` | `logic` | Valid signal for command response. +`rdata_o` | `output` | `logic [IfWidth-1:0]` | Read data from read commands. +`err_o` | `output` | `logic [ErrWidth-1:0]` | Error code. + +The `write raw` and `read raw` command instructs the `prim_otp` wrapper to store / read the data in raw format without generating nor checking integrity information. +That means that the wrapper must return the raw, uncorrected data and no integrity errors. + +The `prim_otp` wrapper implements the `Macro*` error codes (0x0 - 0x4) defined in [OTP error handling](#error-handling). + +The timing diagram below illustrates the timing of a command. +Note that both read and write commands return a response, and each command is independent of the previously issued commands. +The latency from accepting a command to returning a response depends on the underlying OTP IP and is typically larger than 10 cycles. +The returned values depend on the command type and whether an error occurred or not. + +```wavejson +{ + signal: [ + { name: 'clk_i', wave: 'p.............' }, + { name: 'ready_o', wave: '0..10|.10.|...' , node: '...a...c'}, + { name: 'valid_i', wave: '01..0|1.0.|...' }, + { name: 'size_i', wave: '03..0|3.0.|...' }, + { name: 'cmd_i', wave: '04..0|4.0.|...' }, + { name: 'wdata_i', wave: '05..0|5.0.|...' }, + { name: 'valid_o', wave: '0....|..10|.10' , node: '........b...d'}, + { name: 'rdata_o', wave: '0....|..50|.50' }, + { name: 'err_o', wave: '0....|..40|.40' }, + ], + edge: [ + 'a~>b', + 'c~>d', + ], + head: { + text: 'Timing of an OTP command.', + }, + foot: { + text: "Cmd's are accepted in cycles 3/7, and the corresponding responses return in cycles 8/12.", + tick: 0, + } +} +``` + +Note that the open source OTP controller allows up to two outstanding OTP commands, meaning that it is permissible to acknowledge an incoming command and start working on it while the results of the last command are still in the process of being output (e.g., due to an output register stage). + +#### Generic Simulation and FPGA Emulation Model + +For open-source simulation and FPGA emulation, a synthesizable and generic OTP wrapper module is provided (`prim_generic_otp`). +This is automatically selected in the OpenTitan build flow via the technology primitive mechanism if no proprietary OTP IP is available for a specific technology. +The OTP storage in `prim_generic_otp` is emulated using a standard RAM primitive `prim_generic_ram_1p`. +While this storage element is volatile, the primitive is constructed such that the contents are not wiped upon a system-wide reset. +I.e., only a power-cycle wipes the RAM primitive, thereby enabling limited emulation of the OTP function and life cycle transitions also on an FPGA device. diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/README.md b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/README.md new file mode 100644 index 00000000000000..ce51dd4e94cc4e --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/README.md @@ -0,0 +1,146 @@ +# OTP_CTRL DV document + +## Goals +* **DV** + * Verify all OTP_CTRL IP features by running dynamic simulations with a SV/UVM based testbench + * Develop and run all tests based on the [testplan](#testplan) below towards closing code and functional coverage on the IP and all of its sub-modules +* **FPV** + * Verify TileLink device protocol compliance with an SVA based testbench + +## Current status +* [Design & verification stage](../../../../README.md) + * [HW development stages](../../../../../doc/project_governance/development_stages.md) +* [Simulation results](https://reports.opentitan.org/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/latest/report.html) + +## Design features +For detailed information on OTP_CTRL design features, please see the [OTP_CTRL HW IP technical specification](../README.md). + +## Testbench architecture +OTP_CTRL testbench has been constructed based on the [CIP testbench architecture](../../../../dv/sv/cip_lib/README.md). + +### Block diagram +![Block diagram](./doc/tb.svg) + +### Top level testbench +Top level testbench is located at `hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tb.sv`. It instantiates the OTP_CTRL DUT module `hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl.sv`. +In addition, it instantiates the following interfaces, connects them to the DUT and sets their handle into `uvm_config_db`: +* [Clock and reset interface](../../../../dv/sv/common_ifs/README.md) +* [TileLink host interface](../../../../dv/sv/tl_agent/README.md) +* OTP_CTRL IOs +* Interrupts ([`pins_if`](../../../../dv/sv/common_ifs/README.md)) +* Alerts ([`alert_esc_if`](../../../../dv/sv/alert_esc_agent/README.md)) + +### Common DV utility components +The following utilities provide generic helper tasks and functions to perform activities that are common across the project: +* [dv_utils_pkg](../../../../dv/sv/dv_utils/README.md) +* [csr_utils_pkg](../../../../dv/sv/csr_utils/README.md) + +### Global types & methods +All common types and methods defined at the package level can be found in +`otp_ctrl_env_pkg`. Some of them in use are: +```systemverilog + parameter uint SCRAMBLE_DATA_SIZE = 64; + parameter uint SCRAMBLE_KEY_SIZE = 128; + parameter uint NUM_ROUND = 31; + + typedef enum bit [2:0] { + OtpNoError, + OtpMacroError, + OtpMacroEccCorrError, + OtpMacroEccUncorrError, + OtpMacroWriteBlankError, + OtpAccessError, + OtpCheckFailError, + OtpFsmStateError + } otp_err_code_e; +``` + +### TL_agent +OTP_CTRL testbench instantiates (already handled in CIP base env) [tl_agent](../../../../dv/sv/tl_agent/README.md), which provides the ability to drive and independently monitor random traffic via TL host interface into OTP_CTRL device. + +### Alert_agents +OTP_CTRL testbench instantiates (already handled in CIP base env) two [alert_agents](../../../../dv/sv/alert_esc_agent/README.md): +fatal_check_alert and fatal_macro_alert. +The alert_agents provide the ability to drive and independently monitor alert handshakes via alert interfaces in OTP_CTRL device. + +### OTP_CTRL interface +OTP_CTRL design has specific inputs and outputs to communicate with other IPs including LC_CTRL, OTBN, SRAM, FLASH etc. +This interface is created to initialize, use simple task to drive, and use assertions to monitor these signals. + +### Memory backdoor interface +OTP_CTRL testbench binds design's non-volatile OTP memory with a [`mem_bkdr_util`](../../../../dv/sv/mem_bkdr_util/README.md), which supports read, write, and injection of ECC errors to design's OTP memory. + +### UVM RAL model +The OTP_CTRL RAL model is created with the [`ralgen`](../../../../dv/tools/ralgen/README.md) FuseSoC generator script automatically when the simulation is at the build stage. + +It can be created manually by invoking [`regtool`](../../../../../util/reggen/doc/setup_and_use.md). + +### Reference models +The OTP_CTRL's utilizes [PRESENT](../../../../ip/prim/doc/prim_present.md as the cipher to scramble and protect secrets. +Thus OTP_CTRL's scoreboard adopted PRESENT's C reference model, located under `hw/ip/prim/dv/prim_present/` folder, for encryption and decryption purpose. + +### Stimulus strategy +#### Test sequences +All test sequences reside in `hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib`. +The `otp_ctrl_base_vseq` virtual sequence is extended from `cip_base_vseq` and serves as a starting point. +All test sequences are extended from `otp_ctrl_base_vseq`. +It provides commonly used handles, variables, functions and tasks that the test sequences can simple use / call. +Some of the most commonly used tasks / functions are as follows: +* dai_wr: This task triggers an OTP write sequence via the DAI interface. +* dai_rd: This task triggers an OTP read sequence via the DAI interface. +* trigger_checks: This task triggers a one-time OTP check and user can choose to trigger consistency check or integrity check. +* randomize_dai_addr: This task takes a DAI address as input and randomize its last two bits, because the last two bits should be ignored in design. + +#### Functional coverage +To ensure high quality constrained random stimulus, it is necessary to develop a functional coverage model. +The following two files declared OTP_CTRL's covergroups: +- `dv/env/otp_ctrl_env_cov.sv` declares functional or CSR related covergroups. +The functional coverage is collected manually inside OTP_CTRL's scoreboard by invoking the `sample` function. +- `dv/cov/otp_ctrl_cov_if.sv` declares interface signal related covergroups. +The functional coverage is collected automatically when the sampled signal is active. + +### Self-checking strategy +#### Scoreboard +The `otp_ctrl_scoreboard` is primarily used for end to end checking. +It creates the following analysis ports to retrieve the data monitored by corresponding interface agents: +* tl_a_chan_fifo: tl address channel +* tl_d_chan_fifo: tl data channel +* alert_fifos: alert handshakes +* sram_fifos: sram requests +* otbn_fifo: otbn request +* lc_prog_fifo: life cycle programming request +* lc_token_fifo: life cycle token request +* flash_addr_fifo: flash address request +* flash_data_fifo: flash data request +* edn_fifo: edn response to OTP_CTRL + +For all requests to OTP_CTRL as listed above, scoreboard has a corresponding task to process request, check OTP_CTRL's response value against encryption, and collect coverage. + +OTP_CTRL's scoreboard has an internal array `otp_a` that tracks OTP memory data. +Every successful OTP write operation will update this internal array, and every successful OTP read operation will check the readout value against this internal array. +Note that in design, secret partitions will go through a encryption before writing to the actually OTP memory, and will be decrypted upon a read request. +For the simplicity of this internal array, we will skip this procedure. +However, if scoreboard backdoor read any secret partitions, we will decrypt the data then write the decrypted data to the internal array. +For any operation that fails, the scoreboard will predict the status and err_code according to the failure type. +If the error can trigger alert, scoreboard will use `set_exp_alert` task to check if the alert is firing correctly. +If a HW digest operation is triggered by sequence, scoreboard will calculate digest value with partition data from its internal array and update the digest value. +According to design spec, scoreboard won't lock the partition and predict the digest value to digest registers until next power cycle. + +If a reset or lc_escalation_en is issued during an OTP_CTRL write operation, scoreboard cannot accurately predict how many bits have been programmed into OTP memory. +To avoid mismatches, scoreboard utilizes flags `dai_wr_ip` and `dai_digest_ip` to track otp write operations, and issue a backdoor read if the write operation is interrupted. + +#### Assertions +* TLUL assertions: The `tb/otp_ctrl_bind.sv` binds the `tlul_assert` [assertions](../../../../ip/tlul/doc/TlulProtocolChecker.md) to the IP to ensure TileLink interface protocol compliance. +* Unknown checks on DUT outputs: The RTL has assertions to ensure all outputs are initialized to known values after coming out of reset. +* OTP_CTRL_IF assertions: This interface has assertions to ensure certain OTP_CTRL's outputs (such as: otp_broadcast_o, keymgr_key_o) are stable after OTP initialization. + +## Building and running tests +We are using our in-house developed [regression tool](../../../../../util/dvsim/README.md) for building and running our tests and regressions. +Please take a look at the link for detailed information on the usage, capabilities, features and known issues. +Here's how to run a smoke test: +```console +$ $REPO_TOP/util/dvsim/dvsim.py $REPO_TOP/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson -i otp_ctrl_smoke +``` + +## Testplan +[Testplan](../data/otp_ctrl_testplan.hjson) diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov.core b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov.core new file mode 100644 index 00000000000000..9ad10c4464d9ae --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov.core @@ -0,0 +1,28 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: lowrisc:opentitan:top_darjeeling_otp_ctrl_cov +description: "OTP_CTRL functional coverage and bind files" +virtual: + - lowrisc:ip_interfaces:otp_ctrl_cov + +filesets: + files_rtl: + depend: + - lowrisc:ip_interfaces:pwrmgr_pkg + - lowrisc:opentitan:top_darjeeling_otp_ctrl_top_specific_pkg + + files_dv: + depend: + - lowrisc:dv:dv_utils + files: + - otp_ctrl_cov_if.sv + - otp_ctrl_cov_bind.sv + file_type: systemVerilogSource + +targets: + default: + filesets: + - files_rtl + - files_dv diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv new file mode 100644 index 00000000000000..68f38f1bb31d9e --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv @@ -0,0 +1,112 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Binds OTP_CTRL functional coverage interaface to the top level OTP_CTRL module. +// +`define PART_MUBI_COV(__part_name, __index) \ + bind otp_ctrl cip_mubi_cov_if #(.Width(8)) ``__part_name``_read_lock_mubi_cov_if ( \ + .rst_ni (rst_ni), \ + .mubi (part_access[``__index``].read_lock) \ + ); \ + bind otp_ctrl cip_mubi_cov_if #(.Width(8)) ``__part_name``_write_lock_mubi_cov_if ( \ + .rst_ni (rst_ni), \ + .mubi (part_access[``__index``].write_lock) \ + ); + +`define DAI_MUBI_COV(__part_name, __index) \ + bind otp_ctrl cip_mubi_cov_if #(.Width(8)) dai_``__part_name``_read_lock_mubi_cov_if ( \ + .rst_ni (rst_ni), \ + .mubi (part_access_dai[``__index``].read_lock) \ + ); \ + bind otp_ctrl cip_mubi_cov_if #(.Width(8)) dai_``__part_name``_write_lock_mubi_cov_if ( \ + .rst_ni (rst_ni), \ + .mubi (part_access_dai[``__index``].write_lock) \ + ); + +module otp_ctrl_cov_bind; + import otp_ctrl_part_pkg::*; + + bind otp_ctrl otp_ctrl_cov_if u_otp_ctrl_cov_if ( + .pwr_otp_o (pwr_otp_o), + .lc_otp_program_i (lc_otp_program_i), + .lc_escalate_en_i (lc_escalate_en_i), + .flash_otp_key_i (flash_otp_key_i), + .sram_otp_key_i (sram_otp_key_i), + .otbn_otp_key_i (otbn_otp_key_i) + ); + + bind otp_ctrl cip_lc_tx_cov_if u_lc_creator_seed_sw_rw_en_cov_if ( + .rst_ni (rst_ni), + .val (lc_creator_seed_sw_rw_en_i) + ); + + bind otp_ctrl cip_lc_tx_cov_if u_lc_seed_hw_rd_en_cov_if ( + .rst_ni (rst_ni), + .val (lc_seed_hw_rd_en_i) + ); + + bind otp_ctrl cip_lc_tx_cov_if u_lc_dft_en_cov_if ( + .rst_ni (rst_ni), + .val (lc_dft_en_i) + ); + + bind otp_ctrl cip_lc_tx_cov_if u_lc_escalate_en_cov_if ( + .rst_ni (rst_ni), + .val (lc_escalate_en_i) + ); + + bind otp_ctrl cip_lc_tx_cov_if u_lc_check_byp_en_cov_if ( + .rst_ni (rst_ni), + .val (lc_check_byp_en_i) + ); + + // Mubi internal coverage for buffered and unbuffered partitions. + `PART_MUBI_COV(vendor_test, otp_ctrl_part_pkg::VendorTestIdx) + `PART_MUBI_COV(creator_sw_cfg, otp_ctrl_part_pkg::CreatorSwCfgIdx) + `PART_MUBI_COV(owner_sw_cfg, otp_ctrl_part_pkg::OwnerSwCfgIdx) + `PART_MUBI_COV(ownership_slot_state, otp_ctrl_part_pkg::OwnershipSlotStateIdx) + `PART_MUBI_COV(rot_creator_auth, otp_ctrl_part_pkg::RotCreatorAuthIdx) + `PART_MUBI_COV(rot_owner_auth_slot0, otp_ctrl_part_pkg::RotOwnerAuthSlot0Idx) + `PART_MUBI_COV(rot_owner_auth_slot1, otp_ctrl_part_pkg::RotOwnerAuthSlot1Idx) + `PART_MUBI_COV(plat_integ_auth_slot0, otp_ctrl_part_pkg::PlatIntegAuthSlot0Idx) + `PART_MUBI_COV(plat_integ_auth_slot1, otp_ctrl_part_pkg::PlatIntegAuthSlot1Idx) + `PART_MUBI_COV(plat_owner_auth_slot0, otp_ctrl_part_pkg::PlatOwnerAuthSlot0Idx) + `PART_MUBI_COV(plat_owner_auth_slot1, otp_ctrl_part_pkg::PlatOwnerAuthSlot1Idx) + `PART_MUBI_COV(plat_owner_auth_slot2, otp_ctrl_part_pkg::PlatOwnerAuthSlot2Idx) + `PART_MUBI_COV(plat_owner_auth_slot3, otp_ctrl_part_pkg::PlatOwnerAuthSlot3Idx) + `PART_MUBI_COV(ext_nvm, otp_ctrl_part_pkg::ExtNvmIdx) + `PART_MUBI_COV(rom_patch, otp_ctrl_part_pkg::RomPatchIdx) + `PART_MUBI_COV(hw_cfg0, otp_ctrl_part_pkg::HwCfg0Idx) + `PART_MUBI_COV(hw_cfg1, otp_ctrl_part_pkg::HwCfg1Idx) + `PART_MUBI_COV(secret0, otp_ctrl_part_pkg::Secret0Idx) + `PART_MUBI_COV(secret1, otp_ctrl_part_pkg::Secret1Idx) + `PART_MUBI_COV(secret2, otp_ctrl_part_pkg::Secret2Idx) + `PART_MUBI_COV(secret3, otp_ctrl_part_pkg::Secret3Idx) + + // Mubi internal coverage for DAI interface access + `DAI_MUBI_COV(vendor_test, otp_ctrl_part_pkg::VendorTestIdx) + `DAI_MUBI_COV(creator_sw_cfg, otp_ctrl_part_pkg::CreatorSwCfgIdx) + `DAI_MUBI_COV(owner_sw_cfg, otp_ctrl_part_pkg::OwnerSwCfgIdx) + `DAI_MUBI_COV(ownership_slot_state, otp_ctrl_part_pkg::OwnershipSlotStateIdx) + `DAI_MUBI_COV(rot_creator_auth, otp_ctrl_part_pkg::RotCreatorAuthIdx) + `DAI_MUBI_COV(rot_owner_auth_slot0, otp_ctrl_part_pkg::RotOwnerAuthSlot0Idx) + `DAI_MUBI_COV(rot_owner_auth_slot1, otp_ctrl_part_pkg::RotOwnerAuthSlot1Idx) + `DAI_MUBI_COV(plat_integ_auth_slot0, otp_ctrl_part_pkg::PlatIntegAuthSlot0Idx) + `DAI_MUBI_COV(plat_integ_auth_slot1, otp_ctrl_part_pkg::PlatIntegAuthSlot1Idx) + `DAI_MUBI_COV(plat_owner_auth_slot0, otp_ctrl_part_pkg::PlatOwnerAuthSlot0Idx) + `DAI_MUBI_COV(plat_owner_auth_slot1, otp_ctrl_part_pkg::PlatOwnerAuthSlot1Idx) + `DAI_MUBI_COV(plat_owner_auth_slot2, otp_ctrl_part_pkg::PlatOwnerAuthSlot2Idx) + `DAI_MUBI_COV(plat_owner_auth_slot3, otp_ctrl_part_pkg::PlatOwnerAuthSlot3Idx) + `DAI_MUBI_COV(ext_nvm, otp_ctrl_part_pkg::ExtNvmIdx) + `DAI_MUBI_COV(rom_patch, otp_ctrl_part_pkg::RomPatchIdx) + `DAI_MUBI_COV(hw_cfg0, otp_ctrl_part_pkg::HwCfg0Idx) + `DAI_MUBI_COV(hw_cfg1, otp_ctrl_part_pkg::HwCfg1Idx) + `DAI_MUBI_COV(secret0, otp_ctrl_part_pkg::Secret0Idx) + `DAI_MUBI_COV(secret1, otp_ctrl_part_pkg::Secret1Idx) + `DAI_MUBI_COV(secret2, otp_ctrl_part_pkg::Secret2Idx) + `DAI_MUBI_COV(secret3, otp_ctrl_part_pkg::Secret3Idx) + +`undef PART_MUBI_COV +`undef DAI_MUBI_COV +endmodule diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_fsm_unr_excl.el b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_fsm_unr_excl.el new file mode 100644 index 00000000000000..1c3412b6954185 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_fsm_unr_excl.el @@ -0,0 +1,199 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// The current UNR flow marks some transition to reset states as unreachable. +// This file manually removed the reset transition states. +//================================================== +// This file contains the Excluded objects +// Generated By User: chencindy +// Format Version: 2 +// Date: Fri Jan 6 11:14:02 2023 +// ExclMode: default +//================================================== +CHECKSUM: "2063559012 2582220255" +INSTANCE: tb.dut.gen_partitions[3].gen_buffered.u_part_buf +Fsm state_q "441128463" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +Transition IntegDigSt->IntegDigFinSt "1890->1765" +Fsm state_q "441128463" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +State InitDescrWaitSt "2472" +Fsm state_q "441128463" +State IntegScrSt "3418" +Fsm state_q "441128463" +State IntegScrWaitSt "2207" +Fsm state_q "441128463" +State InitDescrSt "3204" +Fsm error_q "2410907799" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2410907799" +Transition CheckFailError->FsmStateError "6->7" +CHECKSUM: "2063559012 2582220255" +INSTANCE: tb.dut.gen_partitions[4].gen_buffered.u_part_buf +Fsm state_q "441128463" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +Transition IntegDigClrSt->IntegDigSt "2625->1890" +Fsm state_q "441128463" +Transition InitWaitSt->InitSt "945->3367" +Fsm state_q "441128463" +State IntegDigPadSt "855" +Fsm error_q "2410907799" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2410907799" +Transition CheckFailError->FsmStateError "6->7" +CHECKSUM: "2063559012 2582220255" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf +Fsm state_q "441128463" +Transition InitWaitSt->InitSt "945->3367" +Fsm state_q "441128463" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +Transition IntegDigClrSt->IntegDigSt "2625->1890" +Fsm state_q "441128463" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +State IntegDigPadSt "855" +Fsm error_q "2410907799" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2410907799" +Transition CheckFailError->FsmStateError "6->7" +CHECKSUM: "2063559012 2582220255" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf +Fsm state_q "441128463" +Transition InitWaitSt->InitSt "945->3367" +Fsm state_q "441128463" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +Transition IntegDigClrSt->IntegDigSt "2625->1890" +Fsm state_q "441128463" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +State IntegDigPadSt "855" +Fsm error_q "2410907799" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2410907799" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +Transition CheckFailError->FsmStateError "6->7" +CHECKSUM: "2063559012 2582220255" +INSTANCE: tb.dut.gen_partitions[7].gen_lifecycle.u_part_buf +Fsm state_q "441128463" +Transition IdleSt->IntegDigClrSt "1357->2625" +Fsm state_q "441128463" +State InitDescrWaitSt "2472" +Fsm state_q "441128463" +State IntegDigFinSt "1765" +Fsm state_q "441128463" +State IntegDigPadSt "855" +Fsm state_q "441128463" +State IntegDigSt "1890" +Fsm state_q "441128463" +State IntegDigWaitSt "2290" +Fsm state_q "441128463" +State IntegScrSt "3418" +Fsm state_q "441128463" +State IntegScrWaitSt "2207" +Fsm state_q "441128463" +State InitDescrSt "3204" +Fsm error_q "2410907799" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2410907799" +Transition CheckFailError->FsmStateError "6->7" +CHECKSUM: "2940612991 27342893" +INSTANCE: tb.dut.gen_partitions[0].gen_unbuffered.u_part_unbuf +Fsm error_q "2210720134" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "2210720134" +Transition CheckFailError->AccessError "6->5" +Fsm error_q "2210720134" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2210720134" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2210720134" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2210720134" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2210720134" +Transition MacroEccCorrError->AccessError "2->5" +Fsm error_q "2210720134" +Transition AccessError->CheckFailError "5->6" +CHECKSUM: "2940612991 27342893" +INSTANCE: tb.dut.gen_partitions[1].gen_unbuffered.u_part_unbuf +Fsm error_q "2210720134" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "2210720134" +Transition CheckFailError->AccessError "6->5" +Fsm error_q "2210720134" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2210720134" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2210720134" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2210720134" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2210720134" +Transition MacroEccCorrError->AccessError "2->5" +Fsm error_q "2210720134" +Transition AccessError->CheckFailError "5->6" +CHECKSUM: "2940612991 27342893" +INSTANCE: tb.dut.gen_partitions[2].gen_unbuffered.u_part_unbuf +Fsm error_q "2210720134" +Transition CheckFailError->AccessError "6->5" +Fsm error_q "2210720134" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2210720134" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2210720134" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2210720134" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2210720134" +Transition MacroEccCorrError->AccessError "2->5" +Fsm error_q "2210720134" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "2210720134" +Transition AccessError->CheckFailError "5->6" +CHECKSUM: "4205406832 4258846959" +INSTANCE: tb.dut.u_otp_ctrl_dai +Fsm error_q "1085514286" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "1085514286" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "1085514286" +Transition AccessError->MacroEccCorrError "5->2" +CHECKSUM: "761735614 2379312231" +INSTANCE: tb.dut.u_otp_ctrl_kdi +Fsm state_q "2979668442" +Transition DigWaitSt->FinishSt "913->760" +Fsm state_q "2979668442" +Transition DigWaitSt->DigLoadSt "913->183" diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_if.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_if.sv new file mode 100644 index 00000000000000..106178b88066d2 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_if.sv @@ -0,0 +1,113 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Implements functional coverage for OTP_CTRL + +interface otp_ctrl_cov_if + import otp_ctrl_reg_pkg::*; + ( + input pwrmgr_pkg::pwr_otp_rsp_t pwr_otp_o, + input otp_ctrl_pkg::lc_otp_program_req_t lc_otp_program_i, + input bit [3:0] lc_escalate_en_i, + input otp_ctrl_pkg::flash_otp_key_req_t flash_otp_key_i, + input otp_ctrl_pkg::sram_otp_key_req_t [NumSramKeyReqSlots-1:0] sram_otp_key_i, + input otp_ctrl_pkg::otbn_otp_key_req_t otbn_otp_key_i + ); + + import uvm_pkg::*; + import dv_utils_pkg::*; + `include "dv_fcov_macros.svh" + + covergroup lc_esc_en_condition_cg @(lc_escalate_en_i == lc_ctrl_pkg::On); + lc_esc_during_flash_data_req: coverpoint flash_otp_key_i.data_req; + lc_esc_during_flash_addr_req: coverpoint flash_otp_key_i.addr_req; + lc_esc_during_sram_0_req: coverpoint sram_otp_key_i[0].req; + lc_esc_during_sram_1_req: coverpoint sram_otp_key_i[1].req; + lc_esc_during_otbn_req: coverpoint otbn_otp_key_i.req; + lc_esc_during_otp_idle: coverpoint pwr_otp_o.otp_idle; + lc_esc_during_lc_otp_prog_req: coverpoint lc_otp_program_i.req; + endgroup + + covergroup flash_data_req_condition_cg @(flash_otp_key_i.data_req); + flash_data_req_during_lc_esc: coverpoint lc_escalate_en_i { + bins lc_esc_on = {lc_ctrl_pkg::On}; + bins lc_esc_off = {[0 : lc_ctrl_pkg::On-1], [lc_ctrl_pkg::On+1 : '1]}; + } + flash_data_req_during_flash_addr_req: coverpoint flash_otp_key_i.addr_req; + flash_data_req_during_sram_0_req: coverpoint sram_otp_key_i[0].req; + flash_data_req_during_sram_1_req: coverpoint sram_otp_key_i[1].req; + flash_data_req_during_otbn_req: coverpoint otbn_otp_key_i.req; + flash_data_req_during_otp_idle: coverpoint pwr_otp_o.otp_idle; + endgroup + + covergroup flash_addr_req_condition_cg @(flash_otp_key_i.addr_req); + flash_addr_req_during_lc_esc: coverpoint lc_escalate_en_i { + bins lc_esc_on = {lc_ctrl_pkg::On}; + bins lc_esc_off = {[0 : lc_ctrl_pkg::On-1], [lc_ctrl_pkg::On+1 : '1]}; + } + flash_addr_req_during_flash_data_req: coverpoint flash_otp_key_i.data_req; + flash_addr_req_during_sram_0_req: coverpoint sram_otp_key_i[0].req; + flash_addr_req_during_sram_1_req: coverpoint sram_otp_key_i[1].req; + flash_addr_req_during_otbn_req: coverpoint otbn_otp_key_i.req; + flash_addr_req_during_otp_idle: coverpoint pwr_otp_o.otp_idle; + endgroup + + covergroup sram_0_req_condition_cg @(sram_otp_key_i[0].req); + sram_0_req_during_lc_esc: coverpoint lc_escalate_en_i { + bins lc_esc_on = {lc_ctrl_pkg::On}; + bins lc_esc_off = {[0 : lc_ctrl_pkg::On-1], [lc_ctrl_pkg::On+1 : '1]}; + } + sram_0_req_during_flash_addr_req: coverpoint flash_otp_key_i.addr_req; + sram_0_req_during_flash_data_req: coverpoint flash_otp_key_i.data_req; + sram_0_req_during_sram_1_req: coverpoint sram_otp_key_i[1].req; + sram_0_req_during_otbn_req: coverpoint otbn_otp_key_i.req; + sram_0_req_during_otp_idle: coverpoint pwr_otp_o.otp_idle; + endgroup + + covergroup sram_1_req_condition_cg @(sram_otp_key_i[1].req); + sram_1_req_during_lc_esc: coverpoint lc_escalate_en_i { + bins lc_esc_on = {lc_ctrl_pkg::On}; + bins lc_esc_off = {[0 : lc_ctrl_pkg::On-1], [lc_ctrl_pkg::On+1 : '1]}; + } + sram_1_req_during_flash_addr_req: coverpoint flash_otp_key_i.addr_req; + sram_1_req_during_flash_data_req: coverpoint flash_otp_key_i.data_req; + sram_1_req_during_sram_0_req: coverpoint sram_otp_key_i[0].req; + sram_1_req_during_otbn_req: coverpoint otbn_otp_key_i.req; + sram_1_req_during_otp_idle: coverpoint pwr_otp_o.otp_idle; + endgroup + + covergroup otbn_req_condition_cg @(otbn_otp_key_i.req); + otbn_req_during_lc_esc: coverpoint lc_escalate_en_i { + bins lc_esc_on = {lc_ctrl_pkg::On}; + bins lc_esc_off = {[0 : lc_ctrl_pkg::On-1], [lc_ctrl_pkg::On+1 : '1]}; + } + otbn_req_during_flash_addr_req: coverpoint flash_otp_key_i.addr_req; + otbn_req_during_flash_data_req: coverpoint flash_otp_key_i.data_req; + otbn_req_during_sram_0_req: coverpoint sram_otp_key_i[0].req; + otbn_req_during_sram_1_req: coverpoint sram_otp_key_i[1].req; + otbn_req_during_otp_idle: coverpoint pwr_otp_o.otp_idle; + endgroup + + covergroup lc_prog_req_condition_cg @(lc_otp_program_i.req); + lc_prog_req_during_lc_esc: coverpoint lc_escalate_en_i { + bins lc_esc_on = {lc_ctrl_pkg::On}; + bins lc_esc_off = {[0 : lc_ctrl_pkg::On-1], [lc_ctrl_pkg::On+1 : '1]}; + } + lc_prog_req_during_flash_addr_req: coverpoint flash_otp_key_i.addr_req; + lc_prog_req_during_flash_data_req: coverpoint flash_otp_key_i.data_req; + lc_prog_req_during_sram_0_req: coverpoint sram_otp_key_i[0].req; + lc_prog_req_during_sram_1_req: coverpoint sram_otp_key_i[1].req; + lc_prog_req_during_otbn_req: coverpoint otbn_otp_key_i.req; + lc_prog_req_during_otp_idle: coverpoint pwr_otp_o.otp_idle; + endgroup + + `DV_FCOV_INSTANTIATE_CG(lc_esc_en_condition_cg) + `DV_FCOV_INSTANTIATE_CG(flash_data_req_condition_cg) + `DV_FCOV_INSTANTIATE_CG(flash_addr_req_condition_cg) + `DV_FCOV_INSTANTIATE_CG(sram_0_req_condition_cg) + `DV_FCOV_INSTANTIATE_CG(sram_1_req_condition_cg) + `DV_FCOV_INSTANTIATE_CG(otbn_req_condition_cg) + `DV_FCOV_INSTANTIATE_CG(lc_prog_req_condition_cg) + +endinterface diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el new file mode 100644 index 00000000000000..99dfe4f72ff2c9 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el @@ -0,0 +1,6133 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Generated UNR file from Synopsys UNR tool with security modules being black-boxed. +//================================================== +// This file contains the Excluded objects +// Generated By User: miguelosorio +// Format Version: 2 +// Date: Fri Aug 30 23:42:43 2024 +// ExclMode: default +//================================================== +CHECKSUM: "2868806991 3096942133" +INSTANCE: tb.dut.gen_alert_tx[4].u_prim_alert_sender +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 alert_ack_o "logic alert_ack_o" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 alert_ack_o "logic alert_ack_o" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 alert_state_o "logic alert_state_o" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 alert_state_o "logic alert_state_o" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[0].gen_unbuffered.u_part_unbuf.gen_ecc_reg.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[1].gen_unbuffered.u_part_unbuf.gen_ecc_reg.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[2].gen_unbuffered.u_part_unbuf.gen_ecc_reg.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[3].gen_unbuffered.u_part_unbuf.gen_ecc_reg.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[4].gen_unbuffered.u_part_unbuf.gen_ecc_reg.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[4].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[1].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[2].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[3].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[5].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[6].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[7].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[8].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[1].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[1].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[2].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[3].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[4].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[1].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[2].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[3].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[4].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[5].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[6].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[7].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[8].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[9].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[10].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[1].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[2].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[3].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[4].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[5].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[6].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[7].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[8].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[9].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[10].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[1].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[2].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[3].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[4].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[5].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[6].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[7].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[8].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[9].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[10].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "3215070453 3446030929" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_reqfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Block 19 "4019242409" "wptr_wrap_cnt_q <= (wptr_wrap_cnt_q + {{(WrapPtrW - 1) {1'b0}}, 1'b1});" +ANNOTATION: "VC_COV_UNR" +Block 28 "1113085816" "rptr_wrap_cnt_q <= (rptr_wrap_cnt_q + {{(WrapPtrW - 1) {1'b0}}, 1'b1});" +CHECKSUM: "3215070453 3446030929" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_sramreqfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Block 19 "4019242409" "wptr_wrap_cnt_q <= (wptr_wrap_cnt_q + {{(WrapPtrW - 1) {1'b0}}, 1'b1});" +ANNOTATION: "VC_COV_UNR" +Block 28 "1113085816" "rptr_wrap_cnt_q <= (rptr_wrap_cnt_q + {{(WrapPtrW - 1) {1'b0}}, 1'b1});" +CHECKSUM: "3215070453 3446030929" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_rspfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Block 19 "4019242409" "wptr_wrap_cnt_q <= (wptr_wrap_cnt_q + {{(WrapPtrW - 1) {1'b0}}, 1'b1});" +ANNOTATION: "VC_COV_UNR" +Block 28 "1113085816" "rptr_wrap_cnt_q <= (rptr_wrap_cnt_q + {{(WrapPtrW - 1) {1'b0}}, 1'b1});" +CHECKSUM: "3171246264 1537087436" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic +ANNOTATION: "VC_COV_UNR" +Block 24 "3494210324" ";" +CHECKSUM: "3665351474 3190968676" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Block 22 "662519215" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 23 "617613755" "state_d = InitDescrWaitSt;" +ANNOTATION: "VC_COV_UNR" +Block 25 "2942482414" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 26 "3475289755" "state_d = InitSt;" +ANNOTATION: "VC_COV_UNR" +Block 68 "746922678" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 69 "369318088" "state_d = IntegScrWaitSt;" +ANNOTATION: "VC_COV_UNR" +Block 71 "1545788457" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 72 "2540127236" "state_d = IntegDigSt;" +ANNOTATION: "VC_COV_UNR" +Block 78 "4176855572" "state_d = IntegDigPadSt;" +ANNOTATION: "VC_COV_UNR" +Block 86 "1288945484" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 87 "984470440" "state_d = IntegDigFinSt;" +ANNOTATION: "VC_COV_UNR" +Block 105 "1192338528" "state_d = ErrorSt;" +ANNOTATION: "VC_COV_UNR" +Block 106 "2515805717" "error_d = CheckFailError;" +CHECKSUM: "3665351474 328877407" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Block 22 "662519215" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 23 "617613755" "state_d = InitDescrWaitSt;" +ANNOTATION: "VC_COV_UNR" +Block 25 "2942482414" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 26 "3475289755" "state_d = InitSt;" +ANNOTATION: "VC_COV_UNR" +Block 68 "746922678" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 69 "369318088" "state_d = IntegScrWaitSt;" +ANNOTATION: "VC_COV_UNR" +Block 71 "1545788457" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 72 "2540127236" "state_d = IntegDigSt;" +ANNOTATION: "VC_COV_UNR" +Block 77 "3141955456" "scrmbl_cmd_o = Digest;" +ANNOTATION: "VC_COV_UNR" +Block 105 "1192338528" "state_d = ErrorSt;" +ANNOTATION: "VC_COV_UNR" +Block 106 "2515805717" "error_d = CheckFailError;" +CHECKSUM: "3665351474 4034042967" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Block 78 "4176855572" "state_d = IntegDigPadSt;" +ANNOTATION: "VC_COV_UNR" +Block 86 "1288945484" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 87 "984470440" "state_d = IntegDigFinSt;" +ANNOTATION: "VC_COV_UNR" +Block 105 "1192338528" "state_d = ErrorSt;" +ANNOTATION: "VC_COV_UNR" +Block 106 "2515805717" "error_d = CheckFailError;" +CHECKSUM: "3665351474 2385925532" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Block 78 "4176855572" "state_d = IntegDigPadSt;" +ANNOTATION: "VC_COV_UNR" +Block 86 "1288945484" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 87 "984470440" "state_d = IntegDigFinSt;" +ANNOTATION: "VC_COV_UNR" +Block 105 "1192338528" "state_d = ErrorSt;" +ANNOTATION: "VC_COV_UNR" +Block 106 "2515805717" "error_d = CheckFailError;" +CHECKSUM: "3665351474 2380508828" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Block 78 "4176855572" "state_d = IntegDigPadSt;" +ANNOTATION: "VC_COV_UNR" +Block 86 "1288945484" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 87 "984470440" "state_d = IntegDigFinSt;" +ANNOTATION: "VC_COV_UNR" +Block 105 "1192338528" "state_d = ErrorSt;" +ANNOTATION: "VC_COV_UNR" +Block 106 "2515805717" "error_d = CheckFailError;" +CHECKSUM: "3665351474 3973220603" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf +ANNOTATION: "VC_COV_UNR" +Block 22 "662519215" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 25 "2942482414" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 68 "746922678" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 71 "1545788457" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 74 "736744825" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 86 "1288945484" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 89 "967932616" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 92 "3253244488" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 105 "1192338528" "state_d = ErrorSt;" +ANNOTATION: "VC_COV_UNR" +Block 106 "2515805717" "error_d = CheckFailError;" +CHECKSUM: "3882079776 1965827338" +INSTANCE: tb.dut.u_otp_ctrl_scrmbl +ANNOTATION: "VC_COV_UNR" +Block 28 "3494210324" ";" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field3 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field4 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field5 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field6 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field7 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field8 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field2 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field3 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field4 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field5 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field0 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field1 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field2 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field3 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "3162909804 919553166" +INSTANCE: tb.dut.gen_partitions[0].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Block 17 "814628507" "error_d = MacroEccCorrError;" +ANNOTATION: "VC_COV_UNR" +Block 32 "3415759745" "error_d = MacroEccCorrError;" +CHECKSUM: "3162909804 2229302740" +INSTANCE: tb.dut.gen_partitions[4].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Block 44 "1192338528" "state_d = ErrorSt;" +ANNOTATION: "VC_COV_UNR" +Block 45 "2515805717" "error_d = CheckFailError;" +CHECKSUM: "662936270 450589591" +INSTANCE: tb.dut.u_tlul_adapter_sram +ANNOTATION: "VC_COV_UNR" +Block 22 "3478134645" "d_valid = 1'b1;" +CHECKSUM: "3436844037 1915699783" +INSTANCE: tb.dut +ANNOTATION: "VC_COV_UNR" +Block 15 "1636116421" "tlul_oob_err_d = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 32 "3434355351" "part_access_pre[k] = {2 {MuBi8True}};" +CHECKSUM: "1158524476 3747170265" +INSTANCE: tb.dut.u_otp_ctrl_kdi +ANNOTATION: "VC_COV_UNR" +Block 88 "3038555774" "state_d = DigLoadSt;" +CHECKSUM: "1611327958 113940473" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top +ANNOTATION: "VC_COV_UNR" +Condition 20 "3585319611" "(reg_we && ((!addrmiss))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 22 "1026062099" "(addrmiss | wr_err | intg_err) 1 -1" (4 "100") +CHECKSUM: "3171246264 2882888745" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic +ANNOTATION: "VC_COV_UNR" +Condition 1 "2104830463" "(cmd_i == Init) 1 -1" (1 "0") +CHECKSUM: "3162909804 3458814989" +INSTANCE: tb.dut.gen_partitions[1].gen_unbuffered.u_part_unbuf +Fsm state_q "4141872371" +ANNOTATION: "VC_COV_UNR" +Transition ResetSt->IdleSt "694->745" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->CheckFailError "5->6" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->AccessError "6->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition MacroEccCorrError->AccessError "2->5" +CHECKSUM: "3162909804 3458814989" +INSTANCE: tb.dut.gen_partitions[2].gen_unbuffered.u_part_unbuf +Fsm state_q "4141872371" +ANNOTATION: "VC_COV_UNR" +Transition ResetSt->IdleSt "694->745" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->CheckFailError "5->6" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->AccessError "6->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition MacroEccCorrError->AccessError "2->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition MacroEccCorrError->CheckFailError "2->6" +CHECKSUM: "3162909804 3458814989" +INSTANCE: tb.dut.gen_partitions[3].gen_unbuffered.u_part_unbuf +Fsm state_q "4141872371" +ANNOTATION: "VC_COV_UNR" +Transition ResetSt->IdleSt "694->745" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->CheckFailError "5->6" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->AccessError "6->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition MacroEccCorrError->AccessError "2->5" +CHECKSUM: "3162909804 3458814989" +INSTANCE: tb.dut.gen_partitions[4].gen_unbuffered.u_part_unbuf +Fsm state_q "4141872371" +ANNOTATION: "VC_COV_UNR" +Transition ResetSt->IdleSt "694->745" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +State CheckFailError "6" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition MacroEccCorrError->AccessError "2->5" +CHECKSUM: "3436844037 2267239089" +INSTANCE: tb.dut +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_ast_pwr_seq_o.pwr_seq [0] "logic otp_ast_pwr_seq_o.pwr_seq[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_ast_pwr_seq_o.pwr_seq [0] "logic otp_ast_pwr_seq_o.pwr_seq[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_ast_pwr_seq_o.pwr_seq [1] "logic otp_ast_pwr_seq_o.pwr_seq[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_ast_pwr_seq_o.pwr_seq [1] "logic otp_ast_pwr_seq_o.pwr_seq[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [0] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [0] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [1] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [1] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [2] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [2] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [3] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [3] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [4] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [4] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [5] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [5] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [6] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [6] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [7] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [7] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [8] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [8] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [9] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [9] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [10] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [10] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [11] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [11] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [12] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [12] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [13] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [13] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [14] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [14] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [15] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [15] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [16] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [16] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [17] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [17] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [18] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [18] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [19] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [19] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [20] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [20] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [21] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [21] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [22] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [22] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [23] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [23] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [24] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [24] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [25] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [25] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [26] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [26] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [27] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [27] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [28] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [28] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [29] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [29] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [30] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [30] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [31] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [31] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed_valid "logic otp_keymgr_key_o.owner_seed_valid" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed_valid "logic otp_keymgr_key_o.owner_seed_valid" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [0] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [0] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [1] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [1] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [2] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [2] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [3] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [3] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [4] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [4] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [5] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [5] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [6] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [6] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [7] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [7] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [8] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [8] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [9] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [9] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [10] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [10] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [11] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [11] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [12] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [12] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [13] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [13] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [14] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [14] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [15] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [15] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [16] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [16] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [17] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [17] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [18] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [18] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [19] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [19] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [20] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [20] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [21] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [21] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [22] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [22] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [23] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [23] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [24] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [24] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [25] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [25] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [26] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [26] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [27] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [27] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [28] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [28] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [29] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [29] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [30] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [30] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [31] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [31] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [32] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [32] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [33] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [33] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [34] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [34] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [35] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [35] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [36] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [36] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [37] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [37] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [38] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [38] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [39] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [39] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [40] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [40] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [41] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [41] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [42] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [42] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [43] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [43] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [44] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [44] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [45] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [45] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [46] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [46] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [47] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [47] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [48] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [48] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [49] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [49] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [50] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [50] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [51] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [51] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [52] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [52] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [53] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [53] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [54] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [54] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [55] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [55] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [56] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [56] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [57] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [57] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [58] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [58] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [59] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [59] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [60] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [60] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [61] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [61] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [62] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [62] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [63] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [63] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [64] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [64] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [65] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [65] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [66] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [66] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [67] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [67] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [68] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [68] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [69] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [69] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [70] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [70] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [71] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [71] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [72] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [72] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [73] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [73] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [74] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [74] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [75] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [75] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [76] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [76] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [77] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [77] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [78] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [78] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [79] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [79] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [80] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [80] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [81] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [81] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [82] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [82] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [83] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [83] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [84] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [84] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [85] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [85] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [86] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [86] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [87] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [87] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [88] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [88] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [89] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [89] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [90] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [90] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [91] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [91] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [92] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [92] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [93] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [93] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [94] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [94] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [95] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [95] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [96] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [96] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [97] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [97] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [98] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [98] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [99] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [99] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [100] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [100] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [101] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [101] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [102] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [102] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [103] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [103] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [104] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [104] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [105] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [105] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [106] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [106] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [107] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [107] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [108] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [108] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [109] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [109] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [110] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [110] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [111] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [111] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [112] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [112] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [113] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [113] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [114] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [114] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [115] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [115] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [116] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [116] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [117] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [117] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [118] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [118] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [119] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [119] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [120] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [120] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [121] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [121] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [122] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [122] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [123] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [123] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [124] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [124] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [125] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [125] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [126] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [126] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [127] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [127] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [128] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [128] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [129] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [129] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [130] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [130] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [131] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [131] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [132] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [132] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [133] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [133] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [134] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [134] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [135] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [135] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [136] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [136] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [137] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [137] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [138] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [138] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [139] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [139] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [140] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [140] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [141] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [141] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [142] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [142] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [143] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [143] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [144] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [144] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [145] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [145] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [146] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [146] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [147] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [147] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [148] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [148] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [149] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [149] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [150] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [150] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [151] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [151] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [152] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [152] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [153] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [153] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [154] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [154] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [155] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [155] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [156] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [156] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [157] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [157] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [158] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [158] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [159] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [159] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [160] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [160] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [161] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [161] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [162] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [162] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [163] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [163] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [164] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [164] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [165] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [165] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [166] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [166] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [167] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [167] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [168] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [168] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [169] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [169] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [170] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [170] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [171] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [171] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [172] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [172] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [173] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [173] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [174] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [174] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [175] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [175] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [176] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [176] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [177] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [177] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [178] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [178] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [179] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [179] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [180] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [180] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [181] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [181] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [182] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [182] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [183] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [183] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [184] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [184] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [185] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [185] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [186] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [186] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [187] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [187] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [188] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [188] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [189] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [189] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [190] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [190] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [191] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [191] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [192] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [192] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [193] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [193] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [194] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [194] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [195] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [195] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [196] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [196] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [197] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [197] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [198] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [198] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [199] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [199] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [200] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [200] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [201] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [201] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [202] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [202] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [203] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [203] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [204] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [204] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [205] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [205] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [206] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [206] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [207] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [207] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [208] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [208] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [209] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [209] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [210] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [210] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [211] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [211] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [212] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [212] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [213] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [213] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [214] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [214] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [215] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [215] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [216] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [216] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [217] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [217] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [218] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [218] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [219] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [219] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [220] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [220] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [221] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [221] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [222] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [222] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [223] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [223] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [224] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [224] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [225] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [225] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [226] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [226] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [227] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [227] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [228] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [228] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [229] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [229] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [230] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [230] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [231] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [231] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [232] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [232] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [233] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [233] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [234] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [234] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [235] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [235] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [236] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [236] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [237] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [237] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [238] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [238] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [239] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [239] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [240] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [240] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [241] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [241] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [242] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [242] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [243] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [243] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [244] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [244] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [245] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [245] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [246] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [246] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [247] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [247] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [248] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [248] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [249] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [249] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [250] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [250] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [251] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [251] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [252] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [252] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [253] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [253] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [254] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [254] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [255] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [255] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed_valid "logic otp_keymgr_key_o.creator_seed_valid" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed_valid "logic otp_keymgr_key_o.creator_seed_valid" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [0] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [0] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [1] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [1] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [2] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [2] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [3] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [3] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [4] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [4] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [5] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [5] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [6] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [6] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [7] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [7] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [8] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [8] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [9] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [9] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [10] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [10] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [11] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [11] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [12] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [12] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [13] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [13] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [14] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [14] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [15] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [15] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [16] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [16] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [17] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [17] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [18] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [18] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [19] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [19] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [20] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [20] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [21] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [21] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [22] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [22] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [23] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [23] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [24] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [24] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [25] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [25] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [26] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [26] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [27] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [27] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [28] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [28] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [29] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [29] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [30] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [30] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [31] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [31] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [32] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [32] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [33] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [33] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [34] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [34] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [35] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [35] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [36] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [36] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [37] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [37] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [38] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [38] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [39] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [39] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [40] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [40] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [41] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [41] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [42] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [42] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [43] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [43] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [44] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [44] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [45] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [45] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [46] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [46] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [47] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [47] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [48] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [48] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [49] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [49] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [50] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [50] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [51] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [51] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [52] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [52] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [53] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [53] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [54] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [54] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [55] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [55] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [56] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [56] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [57] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [57] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [58] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [58] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [59] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [59] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [60] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [60] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [61] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [61] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [62] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [62] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [63] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [63] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [64] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [64] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [65] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [65] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [66] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [66] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [67] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [67] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [68] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [68] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [69] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [69] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [70] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [70] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [71] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [71] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [72] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [72] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [73] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [73] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [74] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [74] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [75] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [75] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [76] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [76] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [77] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [77] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [78] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [78] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [79] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [79] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [80] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [80] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [81] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [81] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [82] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [82] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [83] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [83] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [84] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [84] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [85] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [85] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [86] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [86] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [87] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [87] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [88] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [88] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [89] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [89] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [90] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [90] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [91] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [91] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [92] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [92] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [93] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [93] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [94] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [94] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [95] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [95] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [96] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [96] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [97] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [97] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [98] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [98] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [99] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [99] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [100] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [100] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [101] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [101] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [102] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [102] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [103] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [103] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [104] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [104] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [105] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [105] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [106] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [106] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [107] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [107] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [108] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [108] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [109] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [109] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [110] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [110] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [111] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [111] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [112] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [112] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [113] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [113] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [114] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [114] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [115] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [115] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [116] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [116] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [117] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [117] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [118] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [118] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [119] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [119] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [120] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [120] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [121] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [121] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [122] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [122] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [123] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [123] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [124] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [124] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [125] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [125] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [126] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [126] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [127] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [127] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [128] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [128] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [129] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [129] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [130] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [130] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [131] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [131] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [132] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [132] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [133] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [133] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [134] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [134] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [135] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [135] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [136] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [136] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [137] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [137] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [138] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [138] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [139] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [139] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [140] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [140] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [141] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [141] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [142] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [142] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [143] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [143] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [144] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [144] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [145] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [145] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [146] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [146] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [147] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [147] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [148] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [148] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [149] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [149] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [150] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [150] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [151] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [151] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [152] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [152] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [153] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [153] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [154] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [154] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [155] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [155] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [156] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [156] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [157] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [157] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [158] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [158] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [159] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [159] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [160] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [160] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [161] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [161] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [162] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [162] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [163] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [163] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [164] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [164] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [165] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [165] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [166] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [166] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [167] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [167] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [168] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [168] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [169] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [169] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [170] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [170] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [171] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [171] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [172] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [172] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [173] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [173] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [174] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [174] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [175] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [175] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [176] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [176] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [177] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [177] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [178] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [178] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [179] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [179] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [180] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [180] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [181] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [181] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [182] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [182] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [183] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [183] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [184] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [184] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [185] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [185] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [186] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [186] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [187] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [187] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [188] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [188] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [189] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [189] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [190] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [190] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [191] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [191] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [192] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [192] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [193] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [193] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [194] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [194] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [195] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [195] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [196] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [196] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [197] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [197] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [198] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [198] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [199] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [199] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [200] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [200] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [201] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [201] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [202] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [202] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [203] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [203] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [204] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [204] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [205] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [205] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [206] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [206] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [207] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [207] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [208] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [208] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [209] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [209] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [210] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [210] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [211] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [211] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [212] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [212] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [213] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [213] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [214] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [214] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [215] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [215] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [216] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [216] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [217] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [217] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [218] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [218] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [219] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [219] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [220] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [220] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [221] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [221] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [222] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [222] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [223] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [223] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [224] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [224] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [225] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [225] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [226] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [226] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [227] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [227] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [228] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [228] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [229] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [229] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [230] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [230] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [231] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [231] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [232] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [232] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [233] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [233] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [234] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [234] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [235] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [235] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [236] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [236] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [237] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [237] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [238] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [238] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [239] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [239] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [240] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [240] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [241] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [241] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [242] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [242] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [243] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [243] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [244] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [244] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [245] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [245] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [246] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [246] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [247] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [247] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [248] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [248] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [249] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [249] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [250] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [250] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [251] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [251] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [252] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [252] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [253] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [253] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [254] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [254] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [255] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [255] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [0] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [0] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [1] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [1] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [2] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [2] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [3] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [3] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [4] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [4] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [5] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [5] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [6] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [6] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [7] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [7] "logic cio_test_o[7:0]" +CHECKSUM: "3665351474 2582220255" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State InitDescrSt "3204" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State InitDescrWaitSt "2472" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigPadSt "855" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegScrSt "3418" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegScrWaitSt "2207" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +CHECKSUM: "3665351474 2582220255" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State InitDescrSt "3204" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State InitDescrWaitSt "2472" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegScrSt "3418" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegScrWaitSt "2207" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigSt->IntegDigFinSt "1890->1765" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +CHECKSUM: "3665351474 2582220255" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigPadSt "855" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition InitWaitSt->InitSt "945->3367" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IntegDigSt "2625->1890" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +CHECKSUM: "3665351474 2582220255" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigPadSt "855" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition InitWaitSt->InitSt "945->3367" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IntegDigSt "2625->1890" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +CHECKSUM: "3665351474 2582220255" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigPadSt "855" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition InitWaitSt->InitSt "945->3367" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IntegDigSt "2625->1890" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +CHECKSUM: "3665351474 2582220255" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State InitDescrSt "3204" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State InitDescrWaitSt "2472" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigFinSt "1765" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigPadSt "855" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigSt "1890" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigWaitSt "2290" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegScrSt "3418" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegScrWaitSt "2207" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IdleSt->IntegDigClrSt "1357->2625" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +CHECKSUM: "3162909804 3458814989" +INSTANCE: tb.dut.gen_partitions[0].gen_unbuffered.u_part_unbuf +Fsm state_q "4141872371" +ANNOTATION: "VC_COV_UNR" +Transition ResetSt->IdleSt "694->745" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +State MacroEccCorrError "2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->AccessError "6->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +CHECKSUM: "903559179 4258846959" +INSTANCE: tb.dut.u_otp_ctrl_dai +Fsm error_q "1085514286" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "1085514286" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "1085514286" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +CHECKSUM: "1158524476 2379312231" +INSTANCE: tb.dut.u_otp_ctrl_kdi +Fsm state_q "2979668442" +ANNOTATION: "VC_COV_UNR" +Transition DigWaitSt->DigLoadSt "913->183" +CHECKSUM: "165375753 2072593586" +INSTANCE: tb.dut.u_otp_ctrl_lfsr_timer +ANNOTATION: "VC_COV_UNR" +Condition 15 "1557524712" "(edn_req_o & edn_ack_i) 1 -1" (1 "01") +CHECKSUM: "3436844037 2360113432" +INSTANCE: tb.dut +ANNOTATION: "VC_COV_UNR" +Condition 1 "83486244" "(tlul_part_sel_oh != '0) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 2 "1218027842" "(((|part_tlul_gnt)) | tlul_oob_err_q) 1 -1" (2 "01") +ANNOTATION: "VC_COV_UNR" +Condition 3 "3291433605" "(((|part_tlul_rvalid)) | tlul_oob_err_q) 1 -1" (2 "01") +ANNOTATION: "VC_COV_UNR" +Condition 4 "3779635843" "(fatal_bus_integ_error_q | ((|intg_error))) 1 -1" (3 "10") +ANNOTATION: "VC_COV_UNR" +Condition 5 "1876397257" "(part_error[k] == MacroError) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 10 "2334459673" "(otp_rvalid & otp_fifo_valid) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 13 "1319238213" "(reg2hw.direct_access_cmd.digest.qe | reg2hw.direct_access_cmd.wr.qe | reg2hw.direct_access_cmd.rd.qe) 1 -1" (2 "001") +ANNOTATION: "VC_COV_UNR" +Condition 13 "1319238213" "(reg2hw.direct_access_cmd.digest.qe | reg2hw.direct_access_cmd.wr.qe | reg2hw.direct_access_cmd.rd.qe) 1 -1" (3 "010") +ANNOTATION: "VC_COV_UNR" +Condition 13 "1319238213" "(reg2hw.direct_access_cmd.digest.qe | reg2hw.direct_access_cmd.wr.qe | reg2hw.direct_access_cmd.rd.qe) 1 -1" (4 "100") +ANNOTATION: "vcs_gen_start:k=10:vcs_gen_end:VC_COV_UNR" +Condition 42 "1479216946" "(({tlul_addr, 2'b0} >= 11'b11110101000) & ({1'b0, {tlul_addr, 2'b0}} < gen_part_sel[10].PartEnd)) 1 -1" (2 "10") +CHECKSUM: "1746381268 1271541636" +INSTANCE: tb.dut.u_reg_core.u_socket +ANNOTATION: "VC_COV_UNR" +Condition 3 "118253128" "(tl_t_o.a_valid & tl_t_i.a_ready) 1 -1" (1 "01") +CHECKSUM: "74367784 3785313510" +INSTANCE: tb.dut.u_reg_core.u_reg_if +ANNOTATION: "VC_COV_UNR" +Condition 18 "3340270436" "(addr_align_err | malformed_meta_err | tl_err | instr_error | intg_error) 1 -1" (5 "01000") +CHECKSUM: "74367784 3785313510" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_reg_if +ANNOTATION: "VC_COV_UNR" +Condition 18 "3340270436" "(addr_align_err | malformed_meta_err | tl_err | instr_error | intg_error) 1 -1" (5 "01000") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field3 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field4 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field5 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field6 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field7 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field8 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field2 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field4 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field5 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field2 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field3 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3858770513" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field3 +ANNOTATION: "VC_COV_UNR" +Condition 1 "1301967206" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3858770513" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field1 +ANNOTATION: "VC_COV_UNR" +Condition 1 "1301967206" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3201188367" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field0 +ANNOTATION: "VC_COV_UNR" +Condition 1 "1807203824" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "2099741489 1445279304" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field0.wr_en_data_arb +ANNOTATION: "VC_COV_UNR" +Condition 1 "505266581" "(we | de) 1 -1" (2 "01") +CHECKSUM: "2099741489 1445279304" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field1.wr_en_data_arb +ANNOTATION: "VC_COV_UNR" +Condition 1 "505266581" "(we | de) 1 -1" (2 "01") +CHECKSUM: "2099741489 3636044484" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field2.wr_en_data_arb +ANNOTATION: "VC_COV_UNR" +Condition 1 "505266581" "(we | de) 1 -1" (2 "01") +ANNOTATION: "VC_COV_UNR" +Condition 2 "2306794614" "((de ? d : q) & (we ? ((~wd)) : '1)) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 2 "2306794614" "((de ? d : q) & (we ? ((~wd)) : '1)) 1 -1" (3 "11") +ANNOTATION: "VC_COV_UNR" +Condition 3 "2289961458" "(de ? d : q) 1 -1" (2 "1") +CHECKSUM: "2099741489 1283100255" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field6.wr_en_data_arb +ANNOTATION: "VC_COV_UNR" +Condition 1 "505266581" "(we | de) 1 -1" (2 "01") +CHECKSUM: "2099741489 1077956591" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field0.wr_en_data_arb +ANNOTATION: "VC_COV_UNR" +Condition 1 "505266581" "(we | de) 1 -1" (2 "01") +CHECKSUM: "2099741489 4164822555" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field1.wr_en_data_arb +ANNOTATION: "VC_COV_UNR" +Condition 1 "505266581" "(we | de) 1 -1" (2 "01") +CHECKSUM: "662936270 2081652359" +INSTANCE: tb.dut.u_tlul_adapter_sram +ANNOTATION: "VC_COV_UNR" +Condition 2 "3455933711" "(reqfifo_rdata.op == OpRead) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 8 "3066913128" "(intg_error | rsp_fifo_error | sramreqfifo_error | reqfifo_error | intg_error_q) 1 -1" (2 "00001") +ANNOTATION: "VC_COV_UNR" +Condition 17 "3469950311" "(wr_attr_error | wr_vld_error | rd_vld_error | instr_error | tlul_error | intg_error) 1 -1" (7 "100000") +ANNOTATION: "VC_COV_UNR" +Condition 20 "709191362" "(req_o & gnt_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 21 "3623514242" "(d_valid & reqfifo_rvalid & rspfifo_rvalid & (reqfifo_rdata.op == OpRead)) 1 -1" (1 "0111") +ANNOTATION: "VC_COV_UNR" +Condition 21 "3623514242" "(d_valid & reqfifo_rvalid & rspfifo_rvalid & (reqfifo_rdata.op == OpRead)) 1 -1" (2 "1011") +ANNOTATION: "VC_COV_UNR" +Condition 21 "3623514242" "(d_valid & reqfifo_rvalid & rspfifo_rvalid & (reqfifo_rdata.op == OpRead)) 1 -1" (4 "1110") +ANNOTATION: "VC_COV_UNR" +Condition 25 "2807788926" "((vld_rd_rsp && reqfifo_rdata.error) ? error_blanking_integ : (vld_rd_rsp ? rspfifo_rdata.data_intg : prim_secded_pkg::SecdedInv3932ZeroEcc)) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 26 "561780173" "(vld_rd_rsp && reqfifo_rdata.error) 1 -1" (3 "11") +ANNOTATION: "VC_COV_UNR" +Condition 34 "201396280" "(d_valid && d_error) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 35 "3680494467" "((gnt_i | missed_err_gnt_q) & reqfifo_wready & sramreqfifo_wready & sramreqaddrfifo_wready) 1 -1" (3 "1101") +ANNOTATION: "VC_COV_UNR" +Condition 37 "2164803938" "(tl_i_int.a_valid & reqfifo_wready & ((~error_internal))) 1 -1" (1 "011") +ANNOTATION: "VC_COV_UNR" +Condition 42 "2041272341" "(sram_ack & ((~we_o))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 43 "721931741" "(rvalid_i & reqfifo_rvalid) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:i=0:vcs_gen_end:VC_COV_UNR" +Condition 48 "3548937587" "(((|wmask_intg)) & ((|wdata_intg))) 1 -1" (1 "01") +CHECKSUM: "2974379282 2951929728" +INSTANCE: tb.dut.u_part_sel_idx +ANNOTATION: "vcs_gen_start:level=0,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 1 "2750612666" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[0].gen_level[0].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[0].gen_level[0].C1]) 1 -1" (1 "00") +CHECKSUM: "2974379282 2951929728" +INSTANCE: tb.dut.u_otp_ctrl_dai.u_part_sel_idx +ANNOTATION: "vcs_gen_start:level=0,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 1 "2750612666" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[0].gen_level[0].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[0].gen_level[0].C1]) 1 -1" (1 "00") +CHECKSUM: "2032872600 2493710885" +INSTANCE: tb.dut.u_edn_arb +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 10 "635964333" "(req_i[0] & gen_normal_case.prio_mask_q[0]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 11 "2124571033" "(req_i[0] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[1].gen_level[0].Pa] & ready_i) 1 -1" (1 "011") +CHECKSUM: "2032872600 3068670743" +INSTANCE: tb.dut.u_otp_arb +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 118 "3667925887" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 119 "2125455247" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 119 "2125455247" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 120 "2371949082" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[6].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[6].C1]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 121 "4111290463" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 124 "2230658126" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[6].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel))) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 125 "1884579875" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[6].Pa] & gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 137 "1731868698" "(req_i[0] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[0].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 142 "1060032545" "(req_i[1] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[1].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 147 "1299459822" "(req_i[2] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[2].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 152 "358663893" "(req_i[3] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[3].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 157 "2252512415" "(req_i[4] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[4].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 162 "3730207908" "(req_i[5] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[5].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 167 "2886503019" "(req_i[6] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[6].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=7:vcs_gen_end:VC_COV_UNR" +Condition 172 "4095238736" "(req_i[7] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[7].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=8:vcs_gen_end:VC_COV_UNR" +Condition 177 "296455364" "(req_i[8] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[8].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=9:vcs_gen_end:VC_COV_UNR" +Condition 182 "1237279999" "(req_i[9] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[9].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=10:vcs_gen_end:VC_COV_UNR" +Condition 187 "565898016" "(req_i[10] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[10].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=11:vcs_gen_end:VC_COV_UNR" +Condition 192 "2218826282" "(req_i[11] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[11].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=12:vcs_gen_end:VC_COV_UNR" +Condition 197 "4217824807" "(req_i[12] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[12].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 201 "1343278836" "(req_i[13] & gen_normal_case.prio_mask_q[13]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 201 "1343278836" "(req_i[13] & gen_normal_case.prio_mask_q[13]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 202 "1587320621" "(req_i[13] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 202 "1587320621" "(req_i[13] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ready_i) 1 -1" (2 "101") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 202 "1587320621" "(req_i[13] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 202 "1587320621" "(req_i[13] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ready_i) 1 -1" (4 "111") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 204 "3359315662" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ((~ready_i)))) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 204 "3359315662" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 205 "87484194" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ((~ready_i))) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 205 "87484194" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ((~ready_i))) 1 -1" (3 "11") +CHECKSUM: "2032872600 835765284" +INSTANCE: tb.dut.u_scrmbl_mtx +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 10 "1993460578" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[1].gen_level[0].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C1])) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 10 "1993460578" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[1].gen_level[0].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 11 "1567077924" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 12 "3158492049" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[1].gen_level[0].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[1].gen_level[0].C1]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 13 "96040801" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 14 "1193252123" "(gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel ? gen_normal_case.idx_tree[gen_normal_case.gen_tree[1].gen_level[0].C1] : gen_normal_case.idx_tree[gen_normal_case.gen_tree[1].gen_level[0].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 15 "445016888" "(gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel ? gen_normal_case.data_tree[gen_normal_case.gen_tree[1].gen_level[0].C1] : gen_normal_case.data_tree[gen_normal_case.gen_tree[1].gen_level[0].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 16 "1314162940" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[1].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel))) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 16 "1314162940" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[1].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 17 "2175331913" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[1].gen_level[0].Pa] & gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 18 "2134997267" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[1].gen_level[0].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[1].gen_level[0].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 28 "2070917326" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[2].gen_level[0].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C1])) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 28 "2070917326" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[2].gen_level[0].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 29 "3264610632" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 29 "3264610632" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C1]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 30 "1496330476" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[2].gen_level[0].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[2].gen_level[0].C1]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 30 "1496330476" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[2].gen_level[0].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[2].gen_level[0].C1]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 31 "475390859" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 31 "475390859" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C0]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 32 "3851361741" "(gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel ? gen_normal_case.idx_tree[gen_normal_case.gen_tree[2].gen_level[0].C1] : gen_normal_case.idx_tree[gen_normal_case.gen_tree[2].gen_level[0].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 33 "834897871" "(gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel ? gen_normal_case.data_tree[gen_normal_case.gen_tree[2].gen_level[0].C1] : gen_normal_case.data_tree[gen_normal_case.gen_tree[2].gen_level[0].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 34 "827710109" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[2].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel))) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 34 "827710109" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[2].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel))) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 34 "827710109" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[2].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 35 "1296065718" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[2].gen_level[0].Pa] & gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 35 "1296065718" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[2].gen_level[0].Pa] & gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 36 "2587069038" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[2].gen_level[0].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[2].gen_level[0].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=2,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 45 "1389557192" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[2].gen_level[1].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[2].gen_level[1].C0]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 64 "210767427" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[0].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C1])) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 64 "210767427" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[0].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 65 "433071849" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 65 "433071849" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C1]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 66 "676485814" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[0].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[0].C1]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 66 "676485814" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[0].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[0].C1]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 67 "2859603461" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 67 "2859603461" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C0]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 68 "499725081" "(gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel ? gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[0].C1] : gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[0].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 69 "2136469291" "(gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel ? gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[0].C1] : gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[0].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 70 "3457930055" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel))) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 70 "3457930055" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel))) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 70 "3457930055" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 71 "1202012" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[0].Pa] & gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 71 "1202012" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[0].Pa] & gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 72 "3948011572" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[3].gen_level[0].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[0].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 73 "2058222911" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[1].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C1])) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 73 "2058222911" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[1].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 74 "3215638766" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 74 "3215638766" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C1]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 75 "3769836816" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[1].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[1].C1]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 75 "3769836816" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[1].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[1].C1]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 76 "2034837615" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 76 "2034837615" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C0]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 77 "2239186227" "(gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel ? gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[1].C1] : gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[1].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 78 "2004432906" "(gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel ? gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[1].C1] : gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[1].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 79 "1357880086" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel))) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 79 "1357880086" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel))) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 79 "1357880086" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 80 "3927989974" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] & gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 80 "3927989974" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] & gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 81 "598777746" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 81 "598777746" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].C0]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 82 "3219580251" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[2].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C1])) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 82 "3219580251" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[2].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 83 "26276459" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 84 "1842145640" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[2].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[2].C1]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 85 "147158886" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 86 "2038251222" "(gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel ? gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[2].C1] : gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[2].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 87 "1520563742" "(gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel ? gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[2].C1] : gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[2].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 88 "2599996672" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[2].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel))) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 88 "2599996672" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[2].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 89 "193340754" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[2].Pa] & gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 90 "2932823018" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[3].gen_level[2].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[2].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 90 "2932823018" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[3].gen_level[2].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[2].C0]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 109 "535495451" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[5].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C1])) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 109 "535495451" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[5].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 110 "3231051018" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 111 "2023010" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[5].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[5].C1]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 112 "2223050582" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 113 "1661998521" "(gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel ? gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[5].C1] : gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[5].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 114 "328876404" "(gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel ? gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[5].C1] : gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[5].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 115 "1323463256" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[5].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel))) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 115 "1323463256" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[5].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 116 "2448561063" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[5].Pa] & gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 117 "3273483488" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[3].gen_level[5].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[5].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 118 "3667925887" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1])) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 118 "3667925887" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 119 "2125455247" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 120 "2371949082" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[6].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[6].C1]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 121 "4111290463" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 122 "2669309532" "(gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel ? gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[6].C1] : gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[6].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 123 "1044549472" "(gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel ? gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[6].C1] : gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[6].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 124 "2230658126" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[6].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel))) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 124 "2230658126" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[6].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 125 "1884579875" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[6].Pa] & gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 126 "1315088536" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[3].gen_level[6].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[6].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 136 "2430171309" "(req_i[0] & gen_normal_case.prio_mask_q[0]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 136 "2430171309" "(req_i[0] & gen_normal_case.prio_mask_q[0]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 136 "2430171309" "(req_i[0] & gen_normal_case.prio_mask_q[0]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 137 "1731868698" "(req_i[0] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[0].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 139 "4029129651" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[0].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[0].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 140 "2870412309" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[0].Pa] & ((~ready_i))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 141 "2424159075" "(req_i[1] & gen_normal_case.prio_mask_q[1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 141 "2424159075" "(req_i[1] & gen_normal_case.prio_mask_q[1]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 141 "2424159075" "(req_i[1] & gen_normal_case.prio_mask_q[1]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 142 "1060032545" "(req_i[1] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[1].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 144 "2044991706" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[1].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[1].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 144 "2044991706" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[1].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[1].Pa] & ((~ready_i)))) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 145 "3094700199" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[1].Pa] & ((~ready_i))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 146 "2442198833" "(req_i[2] & gen_normal_case.prio_mask_q[2]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 146 "2442198833" "(req_i[2] & gen_normal_case.prio_mask_q[2]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 146 "2442198833" "(req_i[2] & gen_normal_case.prio_mask_q[2]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 147 "1299459822" "(req_i[2] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[2].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 149 "303405805" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[2].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[2].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 149 "303405805" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[2].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[2].Pa] & ((~ready_i)))) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 150 "3088640414" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[2].Pa] & ((~ready_i))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 151 "2436119807" "(req_i[3] & gen_normal_case.prio_mask_q[3]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 151 "2436119807" "(req_i[3] & gen_normal_case.prio_mask_q[3]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 151 "2436119807" "(req_i[3] & gen_normal_case.prio_mask_q[3]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 152 "358663893" "(req_i[3] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[3].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 154 "2614501764" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[3].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[3].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 154 "2614501764" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[3].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[3].Pa] & ((~ready_i)))) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 155 "2876935468" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[3].Pa] & ((~ready_i))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 156 "1150101804" "(req_i[4] & gen_normal_case.prio_mask_q[4]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 156 "1150101804" "(req_i[4] & gen_normal_case.prio_mask_q[4]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 156 "1150101804" "(req_i[4] & gen_normal_case.prio_mask_q[4]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 157 "2252512415" "(req_i[4] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[4].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 159 "2077403342" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[4].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[4].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 159 "2077403342" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[4].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[4].Pa] & ((~ready_i)))) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 160 "362208751" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[4].Pa] & ((~ready_i))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 164 "4061205415" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[5].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[5].Pa] & ((~ready_i)))) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=10:vcs_gen_end:VC_COV_UNR" +Condition 186 "1702269588" "(req_i[10] & gen_normal_case.prio_mask_q[10]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=10:vcs_gen_end:VC_COV_UNR" +Condition 186 "1702269588" "(req_i[10] & gen_normal_case.prio_mask_q[10]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=10:vcs_gen_end:VC_COV_UNR" +Condition 187 "565898016" "(req_i[10] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[10].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=10:vcs_gen_end:VC_COV_UNR" +Condition 189 "2771416427" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[10].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[10].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=10:vcs_gen_end:VC_COV_UNR" +Condition 190 "2876058957" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[10].Pa] & ((~ready_i))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=12:vcs_gen_end:VC_COV_UNR" +Condition 196 "3119114945" "(req_i[12] & gen_normal_case.prio_mask_q[12]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=12:vcs_gen_end:VC_COV_UNR" +Condition 196 "3119114945" "(req_i[12] & gen_normal_case.prio_mask_q[12]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=12:vcs_gen_end:VC_COV_UNR" +Condition 197 "4217824807" "(req_i[12] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[12].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=12:vcs_gen_end:VC_COV_UNR" +Condition 199 "2102103111" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[12].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[12].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=12:vcs_gen_end:VC_COV_UNR" +Condition 200 "3779810229" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[12].Pa] & ((~ready_i))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 204 "3359315662" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ((~ready_i)))) 1 -1" (1 "00") +CHECKSUM: "2032872600 3109464092" +INSTANCE: tb.dut.u_otp_ctrl_kdi.u_req_arb +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 65 "2951635521" "(req_i[0] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[0].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 70 "4160391802" "(req_i[1] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 75 "2241885365" "(req_i[2] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[2].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=3,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 80 "3719601294" "(req_i[3] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[3].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=3,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 85 "1318504132" "(req_i[4] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[4].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 90 "377630463" "(req_i[5] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[5].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 95 "1691930672" "(req_i[6] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[6].Pa] & ready_i) 1 -1" (1 "011") +CHECKSUM: "3655552781 2309055648" +INSTANCE: tb.dut.u_prim_edn_req.u_prim_packer_fifo +ANNOTATION: "VC_COV_UNR" +Condition 12 "2853235687" "((depth_q == FullDepth) && ((!clr_q))) 1 -1" (2 "10") +CHECKSUM: "4224194069 639524789" +INSTANCE: tb.dut.u_tlul_lc_gate +ANNOTATION: "VC_COV_UNR" +Condition 1 "1380914983" "(a_ack && ((!d_ack))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 2 "2824798557" "(d_ack && ((!a_ack))) 1 -1" (2 "10") +CHECKSUM: "903559179 1044941866" +INSTANCE: tb.dut.u_otp_ctrl_dai +ANNOTATION: "VC_COV_UNR" +Condition 6 "3686409107" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 8 "4163455672" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 9 "1539831361" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:k=10:vcs_gen_end:VC_COV_UNR" +Condition 33 "138415549" "((dai_addr_i >= 11'b11110101000) & ({1'b0, dai_addr_i} < gen_part_sel[10].PartEndInt[otp_ctrl_reg_pkg::OtpByteAddrWidth:0])) 1 -1" (2 "10") +CHECKSUM: "1158524476 2909360515" +INSTANCE: tb.dut.u_otp_ctrl_kdi +ANNOTATION: "VC_COV_UNR" +Condition 2 "1539831361" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +CHECKSUM: "3162909804 4223786199" +INSTANCE: tb.dut.gen_partitions[3].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Condition 14 "2502713177" "(({tlul_addr_q, 2'b0} >= 11'b10001111000) && ({1'b0, tlul_addr_q, 2'b0} < PartEnd)) 1 -1" (2 "10") +CHECKSUM: "3162909804 475089886" +INSTANCE: tb.dut.gen_partitions[0].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Condition 1 "1099175909" "(otp_err != NoError) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 2 "2883220586" "(otp_err != NoError) 1 -1" (2 "1") +CHECKSUM: "3162909804 1722272287" +INSTANCE: tb.dut.gen_partitions[1].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Condition 14 "478960819" "(({tlul_addr_q, 2'b0} >= 11'b00001000000) && ({1'b0, tlul_addr_q, 2'b0} < PartEnd)) 1 -1" (2 "10") +CHECKSUM: "3162909804 1775057944" +INSTANCE: tb.dut.gen_partitions[2].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Condition 14 "673734063" "(({tlul_addr_q, 2'b0} >= 11'b00110110000) && ({1'b0, tlul_addr_q, 2'b0} < PartEnd)) 1 -1" (2 "10") +CHECKSUM: "3162909804 4078376581" +INSTANCE: tb.dut.gen_partitions[4].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Condition 4 "3809681040" "(state_q != ErrorSt) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 4 "3809681040" "(state_q != ErrorSt) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 14 "677249997" "(({tlul_addr_q, 2'b0} >= 11'b11001010000) && ({1'b0, tlul_addr_q, 2'b0} < PartEnd)) 1 -1" (2 "10") +CHECKSUM: "3665351474 720525075" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (3 "11") +ANNOTATION: "VC_COV_UNR" +Condition 10 "4291765346" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (2 "1") +CHECKSUM: "3665351474 384791011" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 9 "1580211052" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (2 "1") +CHECKSUM: "3665351474 599055118" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (3 "11") +ANNOTATION: "VC_COV_UNR" +Condition 10 "4291765346" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (2 "1") +CHECKSUM: "3665351474 698462587" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 9 "1580211052" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (2 "1") +CHECKSUM: "3665351474 2302263073" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 18 "624370688" "((base_sel == DigOffset) ? DigestOffset : 11'b11110101000) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 19 "4038180897" "(base_sel == DigOffset) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 20 "705391888" "((data_sel == ScrmblData) ? scrmbl_data_i : otp_rdata_i) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 21 "1499701630" "(data_sel == ScrmblData) 1 -1" (2 "1") +CHECKSUM: "3665351474 691864715" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 9 "1580211052" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (2 "1") +CHECKSUM: "7115036 2825631531" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_sramreqfifo +ANNOTATION: "VC_COV_UNR" +Condition 2 "1709501387" "(((~gen_normal_fifo.empty)) & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "101") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (1 "011") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +CHECKSUM: "7115036 2432857915" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_rspfifo +ANNOTATION: "VC_COV_UNR" +Condition 2 "1709501387" "(((~gen_normal_fifo.empty)) & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "101") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (1 "011") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +ANNOTATION: "VC_COV_UNR" +Condition 6 "4208363759" "(gen_normal_fifo.fifo_empty && wvalid_i) 1 -1" (1 "01") +CHECKSUM: "7115036 3923796707" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_reqfifo +ANNOTATION: "VC_COV_UNR" +Condition 2 "1709501387" "(((~gen_normal_fifo.empty)) & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "101") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (1 "011") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +CHECKSUM: "7115036 2279662283" +INSTANCE: tb.dut.u_otp_rsp_fifo +ANNOTATION: "VC_COV_UNR" +Condition 1 "2400173860" "(((~full_o)) & ((~gen_normal_fifo.under_rst))) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 2 "1709501387" "(((~gen_normal_fifo.empty)) & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "101") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (1 "011") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +CHECKSUM: "3215070453 33318353" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_reqfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Condition 1 "2532211833" "(incr_wptr_i & (wptr_o == 1'((Depth - 1)))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "2597027294" "(incr_rptr_i & (rptr_o == 1'((Depth - 1)))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 8 "1599734576" "((wptr_wrap_msb == rptr_wrap_msb) ? ((1'(wptr_o) - 1'(rptr_o))) : (((1'(Depth) - 1'(rptr_o)) + 1'(wptr_o)))) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 9 "446195871" "(wptr_wrap_msb == rptr_wrap_msb) 1 -1" (1 "0") +CHECKSUM: "3215070453 33318353" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_sramreqfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Condition 1 "2532211833" "(incr_wptr_i & (wptr_o == 1'((Depth - 1)))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "2597027294" "(incr_rptr_i & (rptr_o == 1'((Depth - 1)))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 8 "1599734576" "((wptr_wrap_msb == rptr_wrap_msb) ? ((1'(wptr_o) - 1'(rptr_o))) : (((1'(Depth) - 1'(rptr_o)) + 1'(wptr_o)))) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 9 "446195871" "(wptr_wrap_msb == rptr_wrap_msb) 1 -1" (1 "0") +CHECKSUM: "3215070453 33318353" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_rspfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Condition 1 "2532211833" "(incr_wptr_i & (wptr_o == 1'((Depth - 1)))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "2597027294" "(incr_rptr_i & (rptr_o == 1'((Depth - 1)))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 8 "1599734576" "((wptr_wrap_msb == rptr_wrap_msb) ? ((1'(wptr_o) - 1'(rptr_o))) : (((1'(Depth) - 1'(rptr_o)) + 1'(wptr_o)))) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 9 "446195871" "(wptr_wrap_msb == rptr_wrap_msb) 1 -1" (1 "0") +CHECKSUM: "3215070453 563859410" +INSTANCE: tb.dut.u_otp_rsp_fifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Condition 5 "3619927060" "(wptr_wrap_cnt_q == (rptr_wrap_cnt_q ^ {1'b1, {(WrapPtrW - 1) {1'b0}}})) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 7 "4040012966" "(full_o ? (2'(Depth)) : ((wptr_wrap_msb == rptr_wrap_msb) ? ((2'(wptr_o) - 2'(rptr_o))) : (((2'(Depth) - 2'(rptr_o)) + 2'(wptr_o))))) 1 -1" (2 "1") +CHECKSUM: "3171246264 3336016746" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic +ANNOTATION: "VC_COV_UNR" +Branch 3 "1554177250" "state_q" (1) "state_q ResetSt ,1,0,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1554177250" "state_q" (8) "state_q IdleSt ,-,-,1,default,-,-,-,-,-,-,-,-" +CHECKSUM: "3436844037 1401458059" +INSTANCE: tb.dut +ANNOTATION: "VC_COV_UNR" +Branch 4 "1264131593" "tlul_req" (1) "tlul_req 1,0" +CHECKSUM: "1746381268 3161287359" +INSTANCE: tb.dut.u_reg_core.u_socket +ANNOTATION: "VC_COV_UNR" +Branch 4 "3202860295" "(!rst_ni)" (2) "(!rst_ni) 0,1,0,-" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field3 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field4 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field5 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field6 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field7 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field8 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field2 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field3 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field4 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field5 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field0 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field1 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field2 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field3 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "662936270 1924774061" +INSTANCE: tb.dut.u_tlul_adapter_sram +ANNOTATION: "VC_COV_UNR" +Branch 2 "1058271942" "(vld_rd_rsp && reqfifo_rdata.error)" (0) "(vld_rd_rsp && reqfifo_rdata.error) 1,-" +ANNOTATION: "VC_COV_UNR" +Branch 6 "744749108" "reqfifo_rvalid" (2) "reqfifo_rvalid 1,0,0" +CHECKSUM: "2032872600 3832429488" +INSTANCE: tb.dut.u_scrmbl_mtx +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Branch 2 "1747167515" "gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel" (1) "gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Branch 3 "1747167515" "gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel" (1) "gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Branch 6 "3123337246" "gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel" (1) "gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Branch 7 "3123337246" "gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel" (1) "gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Branch 14 "840384514" "gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Branch 15 "840384514" "gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Branch 16 "2813219087" "gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Branch 17 "2813219087" "gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Branch 18 "2452890474" "gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Branch 19 "2452890474" "gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Branch 24 "3811150440" "gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Branch 25 "3811150440" "gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Branch 26 "3602118669" "gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Branch 27 "3602118669" "gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel 0" +CHECKSUM: "4224194069 3219254590" +INSTANCE: tb.dut.u_tlul_lc_gate +ANNOTATION: "VC_COV_UNR" +Branch 2 "1850090820" "state_q" (8) "state_q StFlush ,-,-,-,0,0,-,-" +CHECKSUM: "3882079776 3692779052" +INSTANCE: tb.dut.u_otp_ctrl_scrmbl +ANNOTATION: "VC_COV_UNR" +Branch 5 "2137472258" "state_q" (8) "state_q IdleSt ,1,default,-,-,-,-,-" +CHECKSUM: "903559179 3978479804" +INSTANCE: tb.dut.u_otp_ctrl_dai +ANNOTATION: "VC_COV_UNR" +Branch 2 "2060689171" "state_q" (1) "state_q ResetSt ,1,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +CHECKSUM: "1158524476 2213598664" +INSTANCE: tb.dut.u_otp_ctrl_kdi +ANNOTATION: "VC_COV_UNR" +Branch 4 "853326673" "state_q" (20) "state_q DigWaitSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,1,-,-,-" +CHECKSUM: "3162909804 3977884699" +INSTANCE: tb.dut.gen_partitions[0].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Branch 5 "490981166" "state_q" (4) "state_q InitSt ,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "490981166" "state_q" (5) "state_q InitWaitSt ,-,-,-,1,1,1,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "490981166" "state_q" (14) "state_q ReadWaitSt ,-,-,-,-,-,-,-,-,-,1,1,1,-,-,-" +CHECKSUM: "3162909804 1847756134" +INSTANCE: tb.dut.gen_partitions[4].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (0) "ecc_err 1,1" +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (1) "ecc_err 1,0" +CHECKSUM: "3665351474 3964686460" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (20) "state_q CnstyReadSt ,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (35) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (37) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (43) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,0,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (47) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,-,0,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (49) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (50) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (0) "ecc_err 1,1" +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (1) "ecc_err 1,0" +CHECKSUM: "3665351474 3964686460" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (20) "state_q CnstyReadSt ,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (35) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (37) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (43) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,0,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (47) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,-,0,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (49) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (50) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (0) "ecc_err 1,1" +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (1) "ecc_err 1,0" +CHECKSUM: "3665351474 3964686460" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (20) "state_q CnstyReadSt ,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (35) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (37) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (43) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,0,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (47) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,-,0,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (49) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (50) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (0) "ecc_err 1,1" +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (1) "ecc_err 1,0" +CHECKSUM: "3665351474 2334161493" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (11) "state_q InitDescrSt ,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (12) "state_q InitDescrSt ,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (13) "state_q InitDescrWaitSt ,-,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (14) "state_q InitDescrWaitSt ,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (20) "state_q CnstyReadSt ,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (33) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (37) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (38) "state_q IntegScrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (39) "state_q IntegScrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (40) "state_q IntegScrWaitSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (41) "state_q IntegScrWaitSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (43) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,0,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (49) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (50) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "375157548" "ecc_err" (0) "ecc_err 1,1" +ANNOTATION: "VC_COV_UNR" +Branch 5 "375157548" "ecc_err" (1) "ecc_err 1,0" +CHECKSUM: "3665351474 2334161493" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (11) "state_q InitDescrSt ,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (12) "state_q InitDescrSt ,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (13) "state_q InitDescrWaitSt ,-,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (14) "state_q InitDescrWaitSt ,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (20) "state_q CnstyReadSt ,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (33) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (37) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (38) "state_q IntegScrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (39) "state_q IntegScrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (40) "state_q IntegScrWaitSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (41) "state_q IntegScrWaitSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (42) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,1,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (45) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,0,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "375157548" "ecc_err" (0) "ecc_err 1,1" +ANNOTATION: "VC_COV_UNR" +Branch 5 "375157548" "ecc_err" (1) "ecc_err 1,0" +CHECKSUM: "3665351474 2810977924" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf +ANNOTATION: "VC_COV_UNR" +Branch 0 "2541341865" "(base_sel == DigOffset)" (0) "(base_sel == DigOffset) 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "341865418" "(data_sel == ScrmblData)" (0) "(data_sel == ScrmblData) 1" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (12) "state_q InitDescrSt ,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (14) "state_q InitDescrWaitSt ,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (33) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (35) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (37) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (39) "state_q IntegScrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (41) "state_q IntegScrWaitSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (45) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,0,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (47) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,-,0,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (48) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (50) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (52) "state_q IntegDigFinSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (56) "state_q IntegDigWaitSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "375157548" "ecc_err" (0) "ecc_err 1,1" +ANNOTATION: "VC_COV_UNR" +Branch 4 "375157548" "ecc_err" (1) "ecc_err 1,0" +CHECKSUM: "3818998033 3877782530" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg +ANNOTATION: "VC_COV_UNR" +Branch 1 "2154824802" "(32'(addr_i) < Depth)" (2) "(32'(addr_i) < Depth) 0,-" +CHECKSUM: "3215070453 1827096802" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_reqfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Branch 0 "721764659" "full_o" (2) "full_o 0,0" +ANNOTATION: "VC_COV_UNR" +Branch 1 "2417346495" "(!rst_ni)" (3) "(!rst_ni) 0,0,0,1" +ANNOTATION: "VC_COV_UNR" +Branch 2 "456961687" "(!rst_ni)" (3) "(!rst_ni) 0,0,0,1" +CHECKSUM: "3215070453 1827096802" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_sramreqfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Branch 0 "721764659" "full_o" (2) "full_o 0,0" +ANNOTATION: "VC_COV_UNR" +Branch 1 "2417346495" "(!rst_ni)" (3) "(!rst_ni) 0,0,0,1" +ANNOTATION: "VC_COV_UNR" +Branch 2 "456961687" "(!rst_ni)" (3) "(!rst_ni) 0,0,0,1" +CHECKSUM: "3215070453 1827096802" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_rspfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Branch 0 "721764659" "full_o" (2) "full_o 0,0" +ANNOTATION: "VC_COV_UNR" +Branch 1 "2417346495" "(!rst_ni)" (3) "(!rst_ni) 0,0,0,1" +ANNOTATION: "VC_COV_UNR" +Branch 2 "456961687" "(!rst_ni)" (3) "(!rst_ni) 0,0,0,1" +CHECKSUM: "3215070453 1827096802" +INSTANCE: tb.dut.u_otp_rsp_fifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Branch 0 "721764659" "full_o" (0) "full_o 1,-" diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cover.cfg b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cover.cfg new file mode 100644 index 00000000000000..d69a1932762301 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cover.cfg @@ -0,0 +1,14 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + + +// The modules below are preverified in FPV testbench. +// There are many conditional coverage and hard to them all. +-moduletree prim_secded_inv_72_64_dec +-moduletree prim_secded_inv_72_64_enc + +begin tgl + +module prim_secded_inv_72_64_dec + +module prim_secded_inv_72_64_enc +end diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/doc/tb.svg b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/doc/tb.svg new file mode 100644 index 00000000000000..a910d806170370 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/doc/tb.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_ast_inputs_cfg.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_ast_inputs_cfg.sv new file mode 100644 index 00000000000000..4b723d7905ff2c --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_ast_inputs_cfg.sv @@ -0,0 +1,35 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// +// Configuration values for DUT input signals +// +// +// This class randomizes values for DUT signal inputs +// and sets constraints on these values. +// +// This class will be instantiated inside otp_ctrl_env_cfg object, and will connect +// to it's otp_ctrl_vif signals and drive them each reset event +// +// The constraints can be hardened and softened as needed in +// closed-source environment. +// In order to override these constraints, please inherit this class +// and set a type override in the closed source environment + +class otp_ctrl_ast_inputs_cfg extends uvm_object; + `uvm_object_utils(otp_ctrl_ast_inputs_cfg); + `uvm_object_new + + // Group: Variables + rand otp_ast_rsp_t otp_ast_pwr_seq_h; + rand logic [otp_ctrl_pkg::OtpTestCtrlWidth-1:0] otp_vendor_test_ctrl; + rand prim_mubi_pkg::mubi4_t scanmode; + rand logic scan_en, scan_rst_n; + + // Group: Constraints + constraint dut_values_c { + otp_vendor_test_ctrl == 32'h0; + } + +endclass: otp_ctrl_ast_inputs_cfg diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.core b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.core new file mode 100644 index 00000000000000..c1325aa9f2c068 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.core @@ -0,0 +1,62 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: lowrisc:opentitan:top_darjeeling_otp_ctrl_env:0.1 +description: "OTP_CTRL DV UVM environment" +virtual: + - lowrisc:ip_interfaces:otp_ctrl_env + +filesets: + files_dv: + depend: + - lowrisc:dv:ralgen + - lowrisc:dv:cip_lib + - lowrisc:dv:mem_bkdr_util + - lowrisc:dv:crypto_dpi_present + - lowrisc:dv:lc_ctrl_dv_utils + files: + - otp_ctrl_env_pkg.sv + - otp_ctrl_if.sv + - otp_ctrl_ast_inputs_cfg.sv: {is_include_file: true} + - otp_ctrl_env_cfg.sv: {is_include_file: true} + - otp_ctrl_env_cov.sv: {is_include_file: true} + - otp_ctrl_virtual_sequencer.sv: {is_include_file: true} + - otp_ctrl_scoreboard.sv: {is_include_file: true} + - otp_ctrl_env.sv: {is_include_file: true} + - seq_lib/otp_ctrl_vseq_list.sv: {is_include_file: true} + - seq_lib/otp_ctrl_callback_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_base_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_common_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_wake_up_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_smoke_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_partition_walk_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_low_freq_read_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_init_fail_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_dai_lock_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_dai_errs_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_macro_errs_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_background_chks_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_check_fail_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_parallel_base_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_regwen_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_parallel_key_req_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_parallel_lc_req_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_parallel_lc_esc_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_test_access_vseq.sv: {is_include_file: true} + - seq_lib/otp_ctrl_stress_all_vseq.sv: {is_include_file: true} + file_type: systemVerilogSource + +generate: + ral: + generator: ralgen + parameters: + name: otp_ctrl + ip_hjson: ../../data/otp_ctrl.hjson + +targets: + default: + filesets: + - files_dv + generate: + - ral diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.sv new file mode 100644 index 00000000000000..454d14df149ee4 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.sv @@ -0,0 +1,106 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +class otp_ctrl_env #( + type CFG_T = otp_ctrl_env_cfg, + type COV_T = otp_ctrl_env_cov, + type VIRTUAL_SEQUENCER_T = otp_ctrl_virtual_sequencer, + type SCOREBOARD_T = otp_ctrl_scoreboard + ) + extends cip_base_env #( + .CFG_T (CFG_T), + .COV_T (COV_T), + .VIRTUAL_SEQUENCER_T(VIRTUAL_SEQUENCER_T), + .SCOREBOARD_T (SCOREBOARD_T) + ); + `uvm_component_param_utils(otp_ctrl_env #(CFG_T, COV_T, VIRTUAL_SEQUENCER_T, SCOREBOARD_T)) + + `uvm_component_new + + push_pull_agent#(.DeviceDataWidth(SRAM_DATA_SIZE)) m_sram_pull_agent[NumSramKeyReqSlots]; + push_pull_agent#(.DeviceDataWidth(OTBN_DATA_SIZE)) m_otbn_pull_agent; + push_pull_agent#(.DeviceDataWidth(FLASH_DATA_SIZE)) m_flash_addr_pull_agent; + push_pull_agent#(.DeviceDataWidth(FLASH_DATA_SIZE)) m_flash_data_pull_agent; + push_pull_agent#(.DeviceDataWidth(1), .HostDataWidth(LC_PROG_DATA_SIZE)) m_lc_prog_pull_agent; + + function void build_phase(uvm_phase phase); + super.build_phase(phase); + + // build sram-otp pull agent + for (int i = 0; i < NumSramKeyReqSlots; i++) begin + string sram_agent_name = $sformatf("m_sram_pull_agent[%0d]", i); + m_sram_pull_agent[i] = push_pull_agent#(.DeviceDataWidth(SRAM_DATA_SIZE))::type_id::create( + sram_agent_name, this); + uvm_config_db#(push_pull_agent_cfg#(.DeviceDataWidth(SRAM_DATA_SIZE)))::set(this, + $sformatf("%0s*", sram_agent_name), "cfg", cfg.m_sram_pull_agent_cfg[i]); + end + + // build otbn-otp pull agent + m_otbn_pull_agent = push_pull_agent#(.DeviceDataWidth(OTBN_DATA_SIZE))::type_id::create( + "m_otbn_pull_agent", this); + uvm_config_db#(push_pull_agent_cfg#(.DeviceDataWidth(OTBN_DATA_SIZE)))::set( + this, "m_otbn_pull_agent", "cfg", cfg.m_otbn_pull_agent_cfg); + + // build flash-otp pull agent + m_flash_addr_pull_agent = push_pull_agent#(.DeviceDataWidth(FLASH_DATA_SIZE))::type_id::create( + "m_flash_addr_pull_agent", this); + uvm_config_db#(push_pull_agent_cfg#(.DeviceDataWidth(FLASH_DATA_SIZE)))::set( + this, "m_flash_addr_pull_agent", "cfg", cfg.m_flash_addr_pull_agent_cfg); + m_flash_data_pull_agent = push_pull_agent#(.DeviceDataWidth(FLASH_DATA_SIZE))::type_id::create( + "m_flash_data_pull_agent", this); + uvm_config_db#(push_pull_agent_cfg#(.DeviceDataWidth(FLASH_DATA_SIZE)))::set( + this, "m_flash_data_pull_agent", "cfg", cfg.m_flash_data_pull_agent_cfg); + + // build lc-otp program pull agent + m_lc_prog_pull_agent = push_pull_agent#(.HostDataWidth(LC_PROG_DATA_SIZE), .DeviceDataWidth(1)) + ::type_id::create("m_lc_prog_pull_agent", this); + uvm_config_db#(push_pull_agent_cfg#(.HostDataWidth(LC_PROG_DATA_SIZE), .DeviceDataWidth(1))):: + set(this, "m_lc_prog_pull_agent", "cfg", cfg.m_lc_prog_pull_agent_cfg); + + // config mem virtual interface + if (!uvm_config_db#(mem_bkdr_util)::get(this, "", "mem_bkdr_util", cfg.mem_bkdr_util_h)) begin + `uvm_fatal(`gfn, "failed to get mem_bkdr_util from uvm_config_db") + end + + // config otp_ctrl output data virtual interface + if (!uvm_config_db#(otp_ctrl_vif)::get(this, "", "otp_ctrl_vif", cfg.otp_ctrl_vif)) begin + `uvm_fatal(`gfn, "failed to get otp_ctrl_vif from uvm_config_db") + end + + // Check if `NumPart` constant is assigned to the correct value. + `DV_CHECK(NumPart == (LifeCycleIdx + 1)) + + endfunction + + function void connect_phase(uvm_phase phase); + super.connect_phase(phase); + + // connect SRAM sequencer and analysis ports + for (int i = 0; i < NumSramKeyReqSlots; i++) begin + virtual_sequencer.sram_pull_sequencer_h[i] = m_sram_pull_agent[i].sequencer; + if (cfg.en_scb) begin + m_sram_pull_agent[i].monitor.analysis_port.connect( + scoreboard.sram_fifos[i].analysis_export); + end + end + + virtual_sequencer.otbn_pull_sequencer_h = m_otbn_pull_agent.sequencer; + virtual_sequencer.flash_addr_pull_sequencer_h = m_flash_addr_pull_agent.sequencer; + virtual_sequencer.flash_data_pull_sequencer_h = m_flash_data_pull_agent.sequencer; + virtual_sequencer.lc_prog_pull_sequencer_h = m_lc_prog_pull_agent.sequencer; + + if (cfg.en_scb) begin + m_otbn_pull_agent.monitor.analysis_port.connect(scoreboard.otbn_fifo.analysis_export); + m_flash_addr_pull_agent.monitor.analysis_port.connect( + scoreboard.flash_addr_fifo.analysis_export); + m_flash_data_pull_agent.monitor.analysis_port.connect( + scoreboard.flash_data_fifo.analysis_export); + m_lc_prog_pull_agent.monitor.analysis_port.connect(scoreboard.lc_prog_fifo.analysis_export); + end + + // connect the DUT cfg instance to the handle in the otp_ctrl_vif + this.cfg.otp_ctrl_vif.dut_cfg = this.cfg.dut_cfg; + endfunction + +endclass diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cfg.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cfg.sv new file mode 100644 index 00000000000000..52c67e8ff2d209 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cfg.sv @@ -0,0 +1,106 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +`define OTP_CLK_CONSTRAINT(FREQ_) \ + FREQ_ dist { \ + 6 :/ 2, \ + [24:25] :/ 2, \ + [26:47] :/ 1, \ + [48:50] :/ 2, \ + [51:95] :/ 1, \ + 96 :/ 1, \ + [97:99] :/ 1, \ + 100 :/ 1 \ + }; + +class otp_ctrl_env_cfg extends cip_base_env_cfg #(.RAL_T(otp_ctrl_core_reg_block)); + + // ext component cfgs + rand push_pull_agent_cfg#(.DeviceDataWidth(SRAM_DATA_SIZE)) + m_sram_pull_agent_cfg[NumSramKeyReqSlots]; + rand push_pull_agent_cfg#(.DeviceDataWidth(OTBN_DATA_SIZE)) m_otbn_pull_agent_cfg; + rand push_pull_agent_cfg#(.DeviceDataWidth(FLASH_DATA_SIZE)) m_flash_data_pull_agent_cfg; + rand push_pull_agent_cfg#(.DeviceDataWidth(FLASH_DATA_SIZE)) m_flash_addr_pull_agent_cfg; + rand push_pull_agent_cfg#(.DeviceDataWidth(1), .HostDataWidth(LC_PROG_DATA_SIZE)) + m_lc_prog_pull_agent_cfg; + + // Memory backdoor util instance for OTP. + mem_bkdr_util mem_bkdr_util_h; + + // ext interfaces + otp_ctrl_vif otp_ctrl_vif; + virtual clk_rst_if clk_rst_vif_otp_ctrl_prim_reg_block; + + bit backdoor_clear_mem; + + // Check ECC errors + otp_ecc_err_e ecc_chk_err [NumPart] = '{default:OtpNoEccErr}; + + // values for otp_ctrl_if signals connected to DUT + rand otp_ctrl_ast_inputs_cfg dut_cfg; + + // Introduce this flag to avoid close source conflict. + bit create_prim_tl_agent = 1; + + `uvm_object_utils_begin(otp_ctrl_env_cfg) + `uvm_object_utils_end + + `uvm_object_new + + constraint clk_freq_mhz_c { + `OTP_CLK_CONSTRAINT(clk_freq_mhz) + foreach (clk_freqs_mhz[i]) { + `OTP_CLK_CONSTRAINT(clk_freqs_mhz[i]) + } + } + + virtual function void initialize(bit [31:0] csr_base_addr = '1); + string prim_ral_name = "otp_ctrl_prim_reg_block"; + ral_model_names.push_back(prim_ral_name); + clk_freqs_mhz[prim_ral_name] = clk_freq_mhz; + + list_of_alerts = otp_ctrl_env_pkg::LIST_OF_ALERTS; + num_edn = 1; + tl_intg_alert_name = "fatal_bus_integ_error"; + sec_cm_alert_name = "fatal_check_error"; + + super.initialize(csr_base_addr); + + // create push_pull agent config obj + for (int i = 0; i < NumSramKeyReqSlots; i++) begin + string cfg_name = $sformatf("sram_pull_agent_cfg[%0d]", i); + m_sram_pull_agent_cfg[i] = push_pull_agent_cfg#(.DeviceDataWidth(SRAM_DATA_SIZE))::type_id + ::create(cfg_name); + m_sram_pull_agent_cfg[i].agent_type = PullAgent; + end + + m_otbn_pull_agent_cfg = push_pull_agent_cfg#(.DeviceDataWidth(OTBN_DATA_SIZE))::type_id + ::create("m_otbn_pull_agent_cfg"); + m_otbn_pull_agent_cfg.agent_type = PullAgent; + + m_flash_data_pull_agent_cfg = push_pull_agent_cfg#(.DeviceDataWidth(FLASH_DATA_SIZE))::type_id + ::create("m_flash_data_pull_agent_cfg"); + m_flash_data_pull_agent_cfg.agent_type = PullAgent; + m_flash_addr_pull_agent_cfg = push_pull_agent_cfg#(.DeviceDataWidth(FLASH_DATA_SIZE))::type_id + ::create("m_flash_addr_pull_agent_cfg"); + m_flash_addr_pull_agent_cfg.agent_type = PullAgent; + + m_lc_prog_pull_agent_cfg = push_pull_agent_cfg#(.HostDataWidth(LC_PROG_DATA_SIZE), + .DeviceDataWidth(1))::type_id::create("m_lc_prog_pull_agent_cfg"); + m_lc_prog_pull_agent_cfg.agent_type = PullAgent; + + // set num_interrupts & num_alerts + num_interrupts = ral.intr_state.get_n_used_bits(); + + // only support 1 outstanding TL items in tlul_adapter + m_tl_agent_cfg.max_outstanding_req = 1; + m_tl_agent_cfgs["otp_ctrl_prim_reg_block"].max_outstanding_req = 1; + + // create the inputs cfg instance + dut_cfg = otp_ctrl_ast_inputs_cfg::type_id::create("dut_cfg"); + endfunction + +endclass + +`undef OTP_CLK_CONSTRAINT diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cov.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cov.sv new file mode 100644 index 00000000000000..902c6629c8ca27 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cov.sv @@ -0,0 +1,470 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +/** + * Covergoups that are dependent on run-time parameters that may be available + * only in build_phase can be defined here + * Covergroups may also be wrapped inside helper classes if needed. + */ +class otp_ctrl_unbuf_err_code_cg_wrap; + // Unbuffered partition can use TLUL interface to read out but cannot write, thus error_code does + // not have write_blank_err. + covergroup unbuf_err_code_cg(string name) with function sample(bit [TL_DW-1:0] val); + option.per_instance = 1; + option.name = name; + err_code_vals: coverpoint val { + bins no_err = {OtpNoError}; + bins macro_err = {OtpMacroError}; + bins ecc_corr_err = {OtpMacroEccCorrError}; + bins ecc_uncorr_err = {OtpMacroEccUncorrError}; + bins access_err = {OtpAccessError}; + bins check_fail = {OtpCheckFailError}; + bins fsm_err = {OtpFsmStateError}; + illegal_bins illegal_err = default; + } + endgroup + + function new(string name); + unbuf_err_code_cg = new(name); + endfunction +endclass + +class otp_ctrl_buf_err_code_cg_wrap; + // Buffered partition must use DAI interface to access partition, so it does not have access_err + // and write_blank err. + covergroup buf_err_code_cg(string name) with function sample(bit [TL_DW-1:0] val); + option.per_instance = 1; + option.name = name; + err_code_vals: coverpoint val { + bins no_err = {OtpNoError}; + bins macro_err = {OtpMacroError}; + bins ecc_corr_err = {OtpMacroEccCorrError}; + bins ecc_uncorr_err = {OtpMacroEccUncorrError}; + bins check_fail = {OtpCheckFailError}; + bins fsm_err = {OtpFsmStateError}; + illegal_bins illegal_err = default; + } + endgroup + + function new(string name); + buf_err_code_cg = new(name); + endfunction +endclass + +class otp_ctrl_csr_rd_after_alert_cg_wrap; + // This covergroup samples CSRs being checked (via CSR read) after fatal alert is issued. + covergroup csr_rd_after_alert_cg(otp_ctrl_core_reg_block ral) with function sample(bit[TL_DW-1:0] + csr_offset); + read_csr_after_alert_issued: coverpoint csr_offset { + bins unbuffered_digests = { + ral.vendor_test_digest[0].get_offset(), + ral.vendor_test_digest[1].get_offset(), + ral.creator_sw_cfg_digest[0].get_offset(), + ral.creator_sw_cfg_digest[1].get_offset(), + ral.owner_sw_cfg_digest[0].get_offset(), + ral.owner_sw_cfg_digest[1].get_offset(), + ral.rot_creator_auth_digest[0].get_offset(), + ral.rot_creator_auth_digest[1].get_offset(), + ral.rot_owner_auth_slot0_digest[0].get_offset(), + ral.rot_owner_auth_slot0_digest[1].get_offset(), + ral.rot_owner_auth_slot1_digest[0].get_offset(), + ral.rot_owner_auth_slot1_digest[1].get_offset(), + ral.plat_integ_auth_slot0_digest[0].get_offset(), + ral.plat_integ_auth_slot0_digest[1].get_offset(), + ral.plat_integ_auth_slot1_digest[0].get_offset(), + ral.plat_integ_auth_slot1_digest[1].get_offset(), + ral.plat_owner_auth_slot0_digest[0].get_offset(), + ral.plat_owner_auth_slot0_digest[1].get_offset(), + ral.plat_owner_auth_slot1_digest[0].get_offset(), + ral.plat_owner_auth_slot1_digest[1].get_offset(), + ral.plat_owner_auth_slot2_digest[0].get_offset(), + ral.plat_owner_auth_slot2_digest[1].get_offset(), + ral.plat_owner_auth_slot3_digest[0].get_offset(), + ral.plat_owner_auth_slot3_digest[1].get_offset(), + ral.rom_patch_digest[0].get_offset(), + ral.rom_patch_digest[1].get_offset() + }; + bins hw_digests = { + ral.hw_cfg0_digest[0].get_offset(), + ral.hw_cfg0_digest[1].get_offset(), + ral.hw_cfg1_digest[0].get_offset(), + ral.hw_cfg1_digest[1].get_offset() + }; + bins secret_digests = { + ral.secret0_digest[0].get_offset(), + ral.secret0_digest[1].get_offset(), + ral.secret1_digest[0].get_offset(), + ral.secret1_digest[1].get_offset(), + ral.secret2_digest[0].get_offset(), + ral.secret2_digest[1].get_offset(), + ral.secret3_digest[0].get_offset(), + ral.secret3_digest[1].get_offset() + }; + bins direct_access_rdata = { + ral.direct_access_rdata[0].get_offset(), + ral.direct_access_rdata[1].get_offset() + }; + bins status = { + ral.status.get_offset() + }; + bins error_code = { + ral.err_code[0].get_offset(), + ral.err_code[1].get_offset(), + ral.err_code[2].get_offset(), + ral.err_code[3].get_offset(), + ral.err_code[4].get_offset(), + ral.err_code[5].get_offset(), + ral.err_code[6].get_offset(), + ral.err_code[7].get_offset(), + ral.err_code[8].get_offset(), + ral.err_code[9].get_offset(), + ral.err_code[10].get_offset(), + ral.err_code[11].get_offset(), + ral.err_code[12].get_offset(), + ral.err_code[13].get_offset(), + ral.err_code[14].get_offset(), + ral.err_code[15].get_offset(), + ral.err_code[16].get_offset(), + ral.err_code[17].get_offset(), + ral.err_code[18].get_offset(), + ral.err_code[19].get_offset(), + ral.err_code[20].get_offset(), + ral.err_code[21].get_offset(), + ral.err_code[22].get_offset(), + ral.err_code[23].get_offset() + }; + } + endgroup + + function new(otp_ctrl_core_reg_block ral); + csr_rd_after_alert_cg = new(ral); + endfunction + + function void sample(bit[TL_DW-1:0] csr_offset); + csr_rd_after_alert_cg.sample(csr_offset); + endfunction +endclass + +class otp_ctrl_unbuf_access_lock_cg_wrap; + covergroup unbuf_access_lock_cg(string name) with function sample(bit read_lock, bit write_lock, + bit is_write); + option.per_instance = 1; + option.name = name; + read_access_locked: coverpoint read_lock; + write_access_locked: coverpoint write_lock; + operation_type: coverpoint is_write { + bins write_op = {1}; + bins read_op = {0}; + } + unbuf_part_access_cross: cross read_access_locked, write_access_locked, operation_type; + endgroup + + function new(string name); + unbuf_access_lock_cg = new(name); + endfunction + + function void sample(bit read_lock, bit write_lock, bit is_write); + unbuf_access_lock_cg.sample(read_lock, write_lock, is_write); + endfunction +endclass + +class otp_ctrl_env_cov extends cip_base_env_cov #(.CFG_T(otp_ctrl_env_cfg)); + `uvm_component_utils(otp_ctrl_env_cov) + + // the base class provides the following handles for use: + // otp_ctrl_env_cfg: cfg + + otp_ctrl_unbuf_err_code_cg_wrap unbuf_err_code_cg_wrap[NumPartUnbuf]; + otp_ctrl_buf_err_code_cg_wrap buf_err_code_cg_wrap[NumPartBuf]; + otp_ctrl_csr_rd_after_alert_cg_wrap csr_rd_after_alert_cg_wrap; + otp_ctrl_unbuf_access_lock_cg_wrap unbuf_access_lock_cg_wrap[NumPartUnbuf]; + + bit_toggle_cg_wrap lc_prog_cg; + bit_toggle_cg_wrap otbn_req_cg; + bit_toggle_cg_wrap status_csr_cg[OtpStatusFieldSize]; + + // covergroups + // This covergroup collects different conditions when outputs (hwcfg_o, keymgr_key_o) are checked + // in scb: + // - If lc_esc_en is On + // - If each partition is locked (expect LC) + covergroup power_on_cg with function sample (bit lc_esc_en, bit[NumPart-2:0] parts_locked); + lc_esc: coverpoint lc_esc_en; + vendor_test_lock: coverpoint parts_locked[0]; + creator_sw_cfg_lock: coverpoint parts_locked[1]; + owner_sw_cfg_lock: coverpoint parts_locked[2]; + ownership_slot_state_lock: coverpoint parts_locked[3]; + rot_creator_auth_lock: coverpoint parts_locked[4]; + rot_owner_auth_slot0_lock: coverpoint parts_locked[5]; + rot_owner_auth_slot1_lock: coverpoint parts_locked[6]; + plat_integ_auth_slot0_lock: coverpoint parts_locked[7]; + plat_integ_auth_slot1_lock: coverpoint parts_locked[8]; + plat_owner_auth_slot0_lock: coverpoint parts_locked[9]; + plat_owner_auth_slot1_lock: coverpoint parts_locked[10]; + plat_owner_auth_slot2_lock: coverpoint parts_locked[11]; + plat_owner_auth_slot3_lock: coverpoint parts_locked[12]; + ext_nvm_lock: coverpoint parts_locked[13]; + rom_patch_lock: coverpoint parts_locked[14]; + hw_cfg0_lock: coverpoint parts_locked[15]; + hw_cfg1_lock: coverpoint parts_locked[16]; + secret0_lock: coverpoint parts_locked[17]; + secret1_lock: coverpoint parts_locked[18]; + secret2_lock: coverpoint parts_locked[19]; + secret3_lock: coverpoint parts_locked[20]; + endgroup + + // This covergroup is sampled only if flash request passed scb check. + covergroup flash_req_cg with function sample (int index, bit locked); + flash_index: coverpoint index { + bins flash_data_key = {FlashDataKey}; + bins flash_addr_key = {FlashAddrKey}; + illegal_bins il = default; + } + secret1_lock: coverpoint locked; + flash_req_lock_cross: cross flash_index, secret1_lock; + endgroup + + // This covergroup is sampled only if sram request passed scb check. + covergroup sram_req_cg with function sample (int index, bit locked); + sram_index: coverpoint index { + bins sram_key[NumSramKeyReqSlots] = {[0:(NumSramKeyReqSlots-1)]}; + illegal_bins il = default; + } + secret1_lock: coverpoint locked; + sram_req_lock_cross: cross sram_index, secret1_lock; + endgroup + + // This covergroup is sampled only if keymgr output passed scb check. + covergroup keymgr_o_cg with function sample (bit lc_seed_hw_rd_en, bit locked); + keymgr_rd_en: coverpoint lc_seed_hw_rd_en; + // TODO: probably should add all partitions with keymgr material here. + secret2_lock: coverpoint locked; + keymgr_output_conditions: cross keymgr_rd_en, secret2_lock; + endgroup + + // This covergroup samples dai request being issued after fatal alert is issued. + covergroup req_dai_access_after_alert_cg with function sample(bit [TL_DW-1:0] val); + req_dai_access_after_alert_issued: coverpoint val { + bins dai_write = {DaiWrite}; + bins dai_read = {DaiRead}; + bins dai_digest = {DaiDigest}; + } + endgroup + + // This covergroup samples background check being issued after fatal alert is issued. + covergroup issue_checks_after_alert_cg with function sample(bit [TL_DW-1:0] val); + issue_checks_after_alert_issued: coverpoint val { + bins integrity_check = {1}; + bins consistency_check = {2}; + } + endgroup + + // This covergroup collects DAI err_code value. + // DAI access does not have checks, thus no check_fail error. + covergroup dai_err_code_cg with function sample(bit [TL_DW-1:0] val, int part_idx); + err_code_vals: coverpoint val { + bins no_err = {OtpNoError}; + bins macro_err = {OtpMacroError}; + bins ecc_corr_err = {OtpMacroEccCorrError}; + bins ecc_uncorr_err = {OtpMacroEccUncorrError}; + bins write_blank_err = {OtpMacroWriteBlankError}; + bins access_err = {OtpAccessError}; + bins fsm_err = {OtpFsmStateError}; + illegal_bins illegal_err = default; + } + partition: coverpoint part_idx { + bins vendor_test = {VendorTestIdx}; + bins creator_sw_cfg = {CreatorSwCfgIdx}; + bins owner_sw_cfg = {OwnerSwCfgIdx}; + bins ownership_slot_state = {OwnershipSlotStateIdx}; + bins rot_creator_auth = {RotCreatorAuthIdx}; + bins rot_owner_auth_slot0 = {RotOwnerAuthSlot0Idx}; + bins rot_owner_auth_slot1 = {RotOwnerAuthSlot1Idx}; + bins plat_integ_auth_slot0 = {PlatIntegAuthSlot0Idx}; + bins plat_integ_auth_slot1 = {PlatIntegAuthSlot1Idx}; + bins plat_owner_auth_slot0 = {PlatOwnerAuthSlot0Idx}; + bins plat_owner_auth_slot1 = {PlatOwnerAuthSlot1Idx}; + bins plat_owner_auth_slot2 = {PlatOwnerAuthSlot2Idx}; + bins plat_owner_auth_slot3 = {PlatOwnerAuthSlot3Idx}; + bins ext_nvm = {ExtNvmIdx}; + bins rom_patch = {RomPatchIdx}; + bins hw_cfg0 = {HwCfg0Idx}; + bins hw_cfg1 = {HwCfg1Idx}; + bins secret0 = {Secret0Idx}; + bins secret1 = {Secret1Idx}; + bins secret2 = {Secret2Idx}; + bins secret3 = {Secret3Idx}; + bins life_cycle = {LifeCycleIdx}; + bins illegal_idx = default; + } + // LC partition has a separate LCI err_code to collect macro related errors. + dai_err_code_for_all_partitions: cross err_code_vals, partition { + // Illegal bin - vendor_test partition does not have EccUncorrectable error. + illegal_bins vendor_test_ecc_uncorrectable_err = + binsof (partition.vendor_test) && binsof (err_code_vals.ecc_uncorr_err); + ignore_bins life_cycle_ignore = binsof (partition.life_cycle) && + binsof(err_code_vals) intersect {[OtpMacroError:OtpMacroWriteBlankError]}; + } + endgroup + + // This covergroup collects LCI err_code value. + // LCI access does not have digest, thus no access_err. Check_fail, ecc_errors are covered in lc + // buffered partition instead of LCI here. + covergroup lci_err_code_cg with function sample(bit [TL_DW-1:0] val); + err_code_vals: coverpoint val { + bins no_err = {OtpNoError}; + bins macro_err = {OtpMacroError}; + bins write_blank_err = {OtpMacroWriteBlankError}; + bins fsm_err = {OtpFsmStateError}; + illegal_bins illegal_err = default; + } + endgroup + + covergroup dai_access_secret2_cg with function sample(bit lc_rw_en, dai_cmd_e dai_cmd); + lc_creator_seed_sw_rw_en: coverpoint lc_rw_en; + dai_access_cmd: coverpoint dai_cmd { + bins dai_rd = {DaiRead}; + bins dai_wr = {DaiWrite}; + bins dai_digest = {DaiDigest}; + } + dai_access_secret2: cross lc_creator_seed_sw_rw_en, dai_access_cmd; + endgroup + + function new(string name, uvm_component parent); + super.new(name, parent); + // Create coverage from local covergroups. + power_on_cg = new(); + flash_req_cg = new(); + sram_req_cg = new(); + keymgr_o_cg = new(); + req_dai_access_after_alert_cg = new(); + issue_checks_after_alert_cg = new(); + dai_err_code_cg = new(); + lci_err_code_cg = new(); + dai_access_secret2_cg = new(); + endfunction : new + + virtual function void build_phase(uvm_phase phase); + super.build_phase(phase); + // Create instances from bit_toggle_cg_wrapper. + lc_prog_cg = new("lc_prog_cg", "", 0); + otbn_req_cg = new("otbn_req_cg", "", 0); + foreach (status_csr_cg[i]) begin + otp_status_e index = otp_status_e'(i); + status_csr_cg[i]= new(index.name, "status_csr_cg", 0); + end + + // Create instances from external wrapper classes. + csr_rd_after_alert_cg_wrap = new(cfg.ral); + foreach (unbuf_err_code_cg_wrap[i]) begin + otp_status_e index = otp_status_e'(i); + unbuf_err_code_cg_wrap[i] = new($sformatf("unbuf_err_code_cg_wrap[%0s]", index.name)); + end + foreach (buf_err_code_cg_wrap[i]) begin + otp_status_e index = otp_status_e'(i + 2); + buf_err_code_cg_wrap[i] = new($sformatf("buf_err_code_cg_wrap[%0s]", index.name)); + end + foreach (unbuf_access_lock_cg_wrap[i]) begin + part_idx_e index = part_idx_e'(i); + unbuf_access_lock_cg_wrap[i] = new($sformatf("buf_err_code_cg_wrap[%0s]", index.name)); + end + endfunction + + function void collect_status_cov(bit [TL_DW-1:0] val); + foreach (status_csr_cg[i]) begin + status_csr_cg[i].sample(val[i]); + end + endfunction + + // Collect coverage for err_code when it is a compact multi-reg. For DAI error it uses the given + // access_part_idx as the target of the DAI access. + function void collect_compact_err_code_cov(bit [TL_DW-1:0] val, int access_part_idx = DaiIdx); + dv_base_reg_field err_code_flds[$]; + cfg.ral.err_code[0].get_dv_base_reg_fields(err_code_flds); + foreach (err_code_flds[part]) begin + collect_err_code_cov(part, get_field_val(err_code_flds[part], val), access_part_idx); + end + endfunction + + // Collect coverage for a given partition error_code. For DAI error it uses the given + // access_part_idx as the target of the DAI access. + function void collect_err_code_cov(int part_idx, bit [TL_DW-1:0] val, + int access_part_idx = DaiIdx); + case (part_idx) + OtpVendorTestErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpCreatorSwCfgErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpOwnerSwCfgErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpOwnershipSlotStateErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpRotCreatorAuthErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpRotOwnerAuthSlot0ErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpRotOwnerAuthSlot1ErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpPlatIntegAuthSlot0ErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpPlatIntegAuthSlot1ErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpPlatOwnerAuthSlot0ErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpPlatOwnerAuthSlot1ErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpPlatOwnerAuthSlot2ErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpPlatOwnerAuthSlot3ErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpExtNvmErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpRomPatchErrIdx: begin + unbuf_err_code_cg_wrap[part_idx].unbuf_err_code_cg.sample(val); + end + OtpHwCfg0ErrIdx: begin + buf_err_code_cg_wrap[part_idx - NumPartUnbuf].buf_err_code_cg.sample(val); + end + OtpHwCfg1ErrIdx: begin + buf_err_code_cg_wrap[part_idx - NumPartUnbuf].buf_err_code_cg.sample(val); + end + OtpSecret0ErrIdx: begin + buf_err_code_cg_wrap[part_idx - NumPartUnbuf].buf_err_code_cg.sample(val); + end + OtpSecret1ErrIdx: begin + buf_err_code_cg_wrap[part_idx - NumPartUnbuf].buf_err_code_cg.sample(val); + end + OtpSecret2ErrIdx: begin + buf_err_code_cg_wrap[part_idx - NumPartUnbuf].buf_err_code_cg.sample(val); + end + OtpSecret3ErrIdx: begin + buf_err_code_cg_wrap[part_idx - NumPartUnbuf].buf_err_code_cg.sample(val); + end + OtpLifeCycleErrIdx: begin + end + OtpDaiErrIdx: begin + dai_err_code_cg.sample(val, access_part_idx); + end + OtpLciErrIdx: begin + lci_err_code_cg.sample(val); + end + default: begin + `uvm_fatal(`gfn, $sformatf("invalid err_code index %0d", part_idx)) + end + endcase + endfunction +endclass diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_pkg.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_pkg.sv new file mode 100644 index 00000000000000..9adb1e5e9c0136 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_pkg.sv @@ -0,0 +1,297 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +package otp_ctrl_env_pkg; + // dep packages + import uvm_pkg::*; + import top_pkg::*; + import dv_utils_pkg::*; + import dv_lib_pkg::*; + import dv_base_reg_pkg::*; + import tl_agent_pkg::*; + import cip_base_pkg::*; + import csr_utils_pkg::*; + import push_pull_agent_pkg::*; + import otp_ctrl_core_ral_pkg::*; + import otp_ctrl_prim_ral_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_pkg::*; + import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; + import lc_ctrl_pkg::*; + import lc_ctrl_state_pkg::*; + import lc_ctrl_dv_utils_pkg::*; + import mem_bkdr_util_pkg::*; + import otp_scrambler_pkg::*; + import sec_cm_pkg::*; + + // macro includes + `include "uvm_macros.svh" + `include "dv_macros.svh" + + // parameters + parameter string LIST_OF_ALERTS[] = {"fatal_macro_error", + "fatal_check_error", + "fatal_bus_integ_error", + "fatal_prim_otp_alert", + "recov_prim_otp_alert"}; + parameter uint NUM_ALERTS = 5; + parameter uint NUM_EDN = 1; + + parameter uint DIGEST_SIZE = 8; + parameter uint SW_WINDOW_BASE_ADDR = 'h4000; + parameter uint SW_WINDOW_SIZE = NumSwCfgWindowWords * 4; + + parameter uint TL_SIZE = (TL_DW / 8); + // LC has its own storage in scb + // we can use the LC offset here because it will always be the last partition. + parameter uint OTP_ARRAY_SIZE = LcTransitionCntOffset / TL_SIZE; + + parameter int OTP_ADDR_WIDTH = OtpByteAddrWidth-2; + + parameter uint NUM_PRIM_REG = 8; + + // sram rsp data has 1 bit for seed_valid, the rest are for key and nonce + parameter uint SRAM_DATA_SIZE = 1 + SramKeyWidth + SramNonceWidth; + // otbn rsp data has 1 bit for seed_valid, the rest are for key and nonce + parameter uint OTBN_DATA_SIZE = 1 + OtbnKeyWidth + OtbnNonceWidth; + // flash rsp data has 1 bit for seed_valid, the rest are for key + parameter uint FLASH_DATA_SIZE = 1 + FlashKeyWidth; + // lc program data has lc_state data and lc_cnt data + parameter uint LC_PROG_DATA_SIZE = LcStateWidth + LcCountWidth; + + parameter uint NUM_SRAM_EDN_REQ = 12; + parameter uint NUM_OTBN_EDN_REQ = 10; + + parameter uint CHK_TIMEOUT_CYC = 40; + + // When fatal alert triggered, all partitions and the DAI & LCI go to error state and status will + // be set to 1. + parameter bit [NumErrorEntries-1:0] FATAL_EXP_STATUS = '1; + + // lc does not have dai access + parameter int PART_BASE_ADDRS [NumPart-1] = { + VendorTestOffset, + CreatorSwCfgOffset, + OwnerSwCfgOffset, + OwnershipSlotStateOffset, + RotCreatorAuthOffset, + RotOwnerAuthSlot0Offset, + RotOwnerAuthSlot1Offset, + PlatIntegAuthSlot0Offset, + PlatIntegAuthSlot1Offset, + PlatOwnerAuthSlot0Offset, + PlatOwnerAuthSlot1Offset, + PlatOwnerAuthSlot2Offset, + PlatOwnerAuthSlot3Offset, + ExtNvmOffset, + RomPatchOffset, + HwCfg0Offset, + HwCfg1Offset, + Secret0Offset, + Secret1Offset, + Secret2Offset, + Secret3Offset + }; + + // lc does not have digest + parameter int PART_OTP_DIGEST_ADDRS [NumPart-1] = { + VendorTestDigestOffset >> 2, + CreatorSwCfgDigestOffset >> 2, + OwnerSwCfgDigestOffset >> 2, + -1, // This partition does not have a digest. + RotCreatorAuthDigestOffset >> 2, + RotOwnerAuthSlot0DigestOffset >> 2, + RotOwnerAuthSlot1DigestOffset >> 2, + PlatIntegAuthSlot0DigestOffset >> 2, + PlatIntegAuthSlot1DigestOffset >> 2, + PlatOwnerAuthSlot0DigestOffset >> 2, + PlatOwnerAuthSlot1DigestOffset >> 2, + PlatOwnerAuthSlot2DigestOffset >> 2, + PlatOwnerAuthSlot3DigestOffset >> 2, + -1, // This partition does not have a digest. + RomPatchDigestOffset >> 2, + HwCfg0DigestOffset >> 2, + HwCfg1DigestOffset >> 2, + Secret0DigestOffset >> 2, + Secret1DigestOffset >> 2, + Secret2DigestOffset >> 2, + Secret3DigestOffset >> 2 + }; + + // types + typedef enum bit [1:0] { + OtpOperationDone, + OtpErr, + NumOtpCtrlIntr + } otp_intr_e; + + typedef enum bit [5:0] { + OtpVendorTestErrIdx, + OtpCreatorSwCfgErrIdx, + OtpOwnerSwCfgErrIdx, + OtpOwnershipSlotStateErrIdx, + OtpRotCreatorAuthErrIdx, + OtpRotOwnerAuthSlot0ErrIdx, + OtpRotOwnerAuthSlot1ErrIdx, + OtpPlatIntegAuthSlot0ErrIdx, + OtpPlatIntegAuthSlot1ErrIdx, + OtpPlatOwnerAuthSlot0ErrIdx, + OtpPlatOwnerAuthSlot1ErrIdx, + OtpPlatOwnerAuthSlot2ErrIdx, + OtpPlatOwnerAuthSlot3ErrIdx, + OtpExtNvmErrIdx, + OtpRomPatchErrIdx, + OtpHwCfg0ErrIdx, + OtpHwCfg1ErrIdx, + OtpSecret0ErrIdx, + OtpSecret1ErrIdx, + OtpSecret2ErrIdx, + OtpSecret3ErrIdx, + OtpLifeCycleErrIdx, + OtpDaiErrIdx, + OtpLciErrIdx, + OtpTimeoutErrIdx, + OtpLfsrFsmErrIdx, + OtpScramblingFsmErrIdx, + OtpDerivKeyFsmErrIdx, + OtpBusIntegErrorIdx, + OtpDaiIdleIdx, + OtpCheckPendingIdx, + OtpStatusFieldSize + } otp_status_e; + + typedef enum bit [2:0] { + OtpNoError, + OtpMacroError, + OtpMacroEccCorrError, + OtpMacroEccUncorrError, + OtpMacroWriteBlankError, + OtpAccessError, + OtpCheckFailError, + OtpFsmStateError + } otp_err_code_e; + + typedef enum bit [1:0] { + OtpNoEccErr, + OtpEccCorrErr, + OtpEccUncorrErr + } otp_ecc_err_e; + + typedef enum bit [1:0] { + OtpNoAlert, + OtpCheckAlert, + OtpMacroAlert + } otp_alert_e; + + typedef struct packed { + bit read_lock; + bit write_lock; + } otp_part_access_lock_t; + + // OTP conditions when driving specific port. + typedef enum bit [2:0] { + DuringOTPInit, + DuringOTPDaiBusy, + DuringOTPDaiDigest, + DuringOTPRead, + DriveRandomly + } port_drive_condition_e; + + typedef virtual otp_ctrl_if otp_ctrl_vif; + + parameter otp_err_code_e OTP_TERMINAL_ERRS[4] = {OtpMacroEccUncorrError, + OtpCheckFailError, + OtpFsmStateError, + OtpMacroError}; + + // functions + function automatic int get_part_index(bit [TL_DW-1:0] addr); + int index; + for (index = 0; index < NumPart; index++) begin + if (PartInfo[index].offset > addr) begin + index--; + break; + end + end + if (index == NumPart) index--; + return index; + endfunction + + function automatic bit is_secret(bit [TL_DW-1:0] addr); + int part_index = get_part_index(addr); + return PartInfo[part_index].secret; + endfunction + + function automatic bit part_has_digest(int part_idx); + return PartInfo[part_idx].hw_digest || PartInfo[part_idx].sw_digest; + endfunction + + function automatic bit part_has_hw_digest(int part_idx); + return PartInfo[part_idx].hw_digest; + endfunction + + function automatic bit is_sw_digest(bit [TL_DW-1:0] addr); + int part_idx = get_part_index(addr); + if (PartInfo[part_idx].sw_digest) begin + // If the partition contains a digest, it will be located in the last 64bit of the partition. + return {addr[TL_DW-1:3], 3'b0} == ((PartInfo[part_idx].offset + PartInfo[part_idx].size) - 8); + end else begin + return 0; + end + endfunction + + function automatic bit is_digest(bit [TL_DW-1:0] addr); + int part_idx = get_part_index(addr); + if (PartInfo[part_idx].sw_digest || PartInfo[part_idx].hw_digest) begin + // If the partition contains a digest, it will be located in the last 64bit of the partition. + return {addr[TL_DW-1:3], 3'b0} == ((PartInfo[part_idx].offset + PartInfo[part_idx].size) - 8); + end else begin + return 0; + end + endfunction + + function automatic bit is_sw_part(bit [TL_DW-1:0] addr); + int part_idx = get_part_index(addr); + return is_sw_part_idx(part_idx); + endfunction + + function automatic bit is_sw_part_idx(int part_idx); + return (PartInfo[part_idx].variant == Unbuffered); + endfunction + + function automatic bit is_hw_part(bit [TL_DW-1:0] addr); + int part_idx = get_part_index(addr); + return is_hw_part_idx(part_idx); + endfunction + + function automatic bit is_hw_part_idx(int part_idx); + return (PartInfo[part_idx].variant == Buffered); + endfunction + + // Returns true if this partition supports ECC. Otherwise, no ECC errors are reported, and + // the single bit errors are not corrected. + function automatic bit part_has_integrity(int part_idx); + return PartInfo[part_idx].integrity; + endfunction + + // Resolve an offset within the software window as an offset within the whole otp_ctrl block. + function automatic bit [TL_AW-1:0] get_sw_window_offset(bit [TL_AW-1:0] dai_addr); + return dai_addr + SW_WINDOW_BASE_ADDR; + endfunction + + function automatic bit [TL_DW-1:0] normalize_dai_addr(bit [TL_DW-1:0] dai_addr); + normalize_dai_addr = (is_secret(dai_addr) || is_digest(dai_addr)) ? dai_addr >> 3 << 3 : + dai_addr >> 2 << 2; + endfunction + + // package sources + `include "otp_ctrl_ast_inputs_cfg.sv" + `include "otp_ctrl_env_cfg.sv" + `include "otp_ctrl_env_cov.sv" + `include "otp_ctrl_virtual_sequencer.sv" + `include "otp_ctrl_scoreboard.sv" + `include "otp_ctrl_env.sv" + `include "otp_ctrl_vseq_list.sv" + +endpackage diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_if.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_if.sv new file mode 100644 index 00000000000000..4baa5978ec5029 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_if.sv @@ -0,0 +1,459 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// This interface collect the broadcast output data from OTP, +// and drive input requests coming into OTP. +`define ECC_REG_PATH gen_ecc_reg.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec + +// This only supports buffered partitions. +`define BUF_PART_OTP_CMD_PATH(i) \ + tb.dut.gen_partitions[``i``].gen_buffered.u_part_buf.otp_cmd_o + +`define LC_PART_OTP_CMD_PATH \ + tb.dut.gen_partitions[LifeCycleIdx].gen_lifecycle.u_part_buf.otp_cmd_o + +`define FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(i) \ + if (forced_part_access_sel[``i``].read_lock) begin \ + force tb.dut.part_access[``i``].read_lock = get_rand_mubi8_val(.t_weight(0), .f_weight(0)); \ + force tb.dut.part_access_dai[``i``].read_lock = get_rand_mubi8_val(.t_weight(0), .f_weight(0)); \ + end \ + if (forced_part_access_sel[``i``].write_lock) begin \ + force tb.dut.part_access[``i``].write_lock = get_rand_mubi8_val(.t_weight(0), .f_weight(0)); \ + force tb.dut.part_access_dai[``i``].write_lock = get_rand_mubi8_val(.t_weight(0), .f_weight(0)); \ + end + +`ifndef PRIM_GENERIC_OTP_PATH + `define PRIM_GENERIC_OTP_PATH\ + tb.dut.u_otp +`endif + +`ifndef PRIM_GENERIC_OTP_CMD_I_PATH + `define PRIM_GENERIC_OTP_CMD_I_PATH \ + `PRIM_GENERIC_OTP_PATH.gen_generic.u_impl_generic.cmd_i +`endif + +interface otp_ctrl_if(input clk_i, input rst_ni); + import uvm_pkg::*; + import otp_ctrl_env_pkg::*; + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_part_pkg::*; + import cip_base_pkg::*; + + // Output from DUT + otp_broadcast_t otp_broadcast_o; + otp_keymgr_key_t keymgr_key_o; + otp_lc_data_t lc_data_o; + logic pwr_otp_done_o, pwr_otp_idle_o; + + // Inputs to DUT + logic pwr_otp_init_i, scan_en_i, scan_rst_ni, ext_voltage_h_io; + lc_ctrl_pkg::lc_tx_t lc_dft_en_i, lc_escalate_en_i, lc_check_byp_en_i, + lc_creator_seed_sw_rw_en_i, lc_owner_seed_sw_rw_en_i, + lc_seed_hw_rd_en_i; + prim_mubi_pkg::mubi4_t scanmode_i; + otp_ast_rsp_t otp_ast_pwr_seq_h_i; + ast_pkg::ast_obs_ctrl_t obs_ctrl_i; + + // Unused in prim_generic_otp memory. + logic [OtpTestCtrlWidth-1:0] otp_vendor_test_ctrl_i; + logic [OtpTestStatusWidth-1:0] otp_vendor_test_status_o; + logic [OtpTestVectWidth-1:0] cio_test_o; + logic [OtpTestVectWidth-1:0] cio_test_en_o; + + // Connect with lc_prog push_pull interface. + logic lc_prog_req, lc_prog_err; + logic lc_prog_err_dly1, lc_prog_no_sta_check; + + // Connect push_pull interfaces ack signals for assertion checks. + logic otbn_ack, lc_prog_ack; + logic [1:0] flash_acks; + logic [NumSramKeyReqSlots-1:0] sram_acks; + + // Variables for internal interface logic. + // `lc_escalate_en` is async, take two clock cycles to synchronize. + lc_ctrl_pkg::lc_tx_t lc_esc_dly1, lc_esc_dly2; + + // Variable for scoreboard. + // For `lc_escalate_en`, any value that is not `Off` is a `On`. + bit lc_esc_on; + + // Probe design signal for alert request. + logic alert_reqs; + + // Usually the `lc_check_byp_en` will be automatically set to `On` when LC program request is + // issued, and stays `On` until reset is issued. + // Set this variable to 0 after a LC program request might cause otp checks to fail. + bit lc_check_byp_en = 1; + + // Internal veriable to track which sw partitions have ECC reg error. + bit [NumPartUnbuf-1:0] force_sw_parts_ecc_reg; + + // DUT configuration object + otp_ctrl_ast_inputs_cfg dut_cfg; + + // for DV macros ID + string msg_id = "otp_ctrl_if"; + + // Lc_err could trigger during LC program, so check intr and status after lc_req is finished. + // Lc_err takes one clock cycle to propogate to intr signal. So avoid intr check if it happens + // during the transition. + always_ff @(posedge clk_i or negedge rst_ni) begin + if (!rst_ni) begin + lc_prog_err_dly1 <= 0; + lc_esc_dly1 <= lc_ctrl_pkg::Off; + lc_esc_dly2 <= lc_ctrl_pkg::Off; + lc_check_byp_en_i <= get_rand_lc_tx_val(); + lc_esc_on <= 0; + end else begin + lc_prog_err_dly1 <= lc_prog_err; + lc_esc_dly1 <= lc_escalate_en_i; + lc_esc_dly2 <= lc_esc_dly1; + if (lc_prog_req) begin + lc_check_byp_en_i <= lc_check_byp_en ? lc_ctrl_pkg::On : lc_ctrl_pkg::Off; + end + if (lc_esc_dly2 != lc_ctrl_pkg::Off && !lc_esc_on) begin + lc_esc_on <= 1; + end + end + end + + assign lc_prog_no_sta_check = lc_prog_err | lc_prog_err_dly1 | lc_prog_req | lc_esc_on; + + function automatic void drive_pwr_otp_init(logic val); + pwr_otp_init_i = val; + endfunction + + function automatic void drive_ext_voltage_h_io(logic val); + ext_voltage_h_io = val; + endfunction + + function automatic void drive_lc_creator_seed_sw_rw_en(lc_ctrl_pkg::lc_tx_t val); + lc_creator_seed_sw_rw_en_i = val; + endfunction + + function automatic void drive_lc_owner_seed_sw_rw_en(lc_ctrl_pkg::lc_tx_t val); + lc_owner_seed_sw_rw_en_i = val; + endfunction + + function automatic void drive_lc_dft_en(lc_ctrl_pkg::lc_tx_t val); + lc_dft_en_i = val; + endfunction + + function automatic void drive_lc_escalate_en(lc_ctrl_pkg::lc_tx_t val); + lc_escalate_en_i = val; + endfunction + + function automatic void drive_lc_seed_hw_rd_en(lc_ctrl_pkg::lc_tx_t val); + lc_seed_hw_rd_en_i = val; + endfunction + + function automatic bit under_error_states(); + return lc_esc_on | alert_reqs; + endfunction + + // SW partitions do not have any internal checks. + // Here we force internal ECC check to fail. + task automatic force_sw_check_fail( + bit[NumPartUnbuf-1:0] fail_idx = $urandom_range(1, (1'b1 << NumPartUnbuf) - 1)); + @(posedge clk_i); + if (fail_idx[VendorTestIdx]) begin + force tb.dut.gen_partitions[VendorTestIdx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0] = 1; + force_sw_parts_ecc_reg[VendorTestIdx] = 1; + end + if (fail_idx[CreatorSwCfgIdx]) begin + force tb.dut.gen_partitions[CreatorSwCfgIdx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0] = 1; + force_sw_parts_ecc_reg[CreatorSwCfgIdx] = 1; + end + if (fail_idx[OwnerSwCfgIdx]) begin + force tb.dut.gen_partitions[OwnerSwCfgIdx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0] = 1; + force_sw_parts_ecc_reg[OwnerSwCfgIdx] = 1; + end + if (fail_idx[RotCreatorAuthIdx]) begin + force tb.dut.gen_partitions[RotCreatorAuthIdx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0] = 1; + force_sw_parts_ecc_reg[RotCreatorAuthIdx] = 1; + end + if (fail_idx[RotOwnerAuthSlot0Idx]) begin + force tb.dut.gen_partitions[RotOwnerAuthSlot0Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0] = 1; + force_sw_parts_ecc_reg[RotOwnerAuthSlot0Idx] = 1; + end + if (fail_idx[RotOwnerAuthSlot1Idx]) begin + force tb.dut.gen_partitions[RotOwnerAuthSlot1Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0] = 1; + force_sw_parts_ecc_reg[RotOwnerAuthSlot1Idx] = 1; + end + if (fail_idx[PlatIntegAuthSlot0Idx]) begin + force tb.dut.gen_partitions[PlatIntegAuthSlot0Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0] = 1; + force_sw_parts_ecc_reg[PlatIntegAuthSlot0Idx] = 1; + end + if (fail_idx[PlatIntegAuthSlot1Idx]) begin + force tb.dut.gen_partitions[PlatIntegAuthSlot1Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0] = 1; + force_sw_parts_ecc_reg[PlatIntegAuthSlot1Idx] = 1; + end + if (fail_idx[PlatOwnerAuthSlot0Idx]) begin + force tb.dut.gen_partitions[PlatOwnerAuthSlot0Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0] = 1; + force_sw_parts_ecc_reg[PlatOwnerAuthSlot0Idx] = 1; + end + if (fail_idx[PlatOwnerAuthSlot1Idx]) begin + force tb.dut.gen_partitions[PlatOwnerAuthSlot1Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0] = 1; + force_sw_parts_ecc_reg[PlatOwnerAuthSlot1Idx] = 1; + end + if (fail_idx[PlatOwnerAuthSlot2Idx]) begin + force tb.dut.gen_partitions[PlatOwnerAuthSlot2Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0] = 1; + force_sw_parts_ecc_reg[PlatOwnerAuthSlot2Idx] = 1; + end + if (fail_idx[PlatOwnerAuthSlot3Idx]) begin + force tb.dut.gen_partitions[PlatOwnerAuthSlot3Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0] = 1; + force_sw_parts_ecc_reg[PlatOwnerAuthSlot3Idx] = 1; + end + if (fail_idx[RomPatchIdx]) begin + force tb.dut.gen_partitions[RomPatchIdx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0] = 1; + force_sw_parts_ecc_reg[RomPatchIdx] = 1; + end + endtask + + task automatic release_sw_check_fail(); + @(posedge clk_i); + if (force_sw_parts_ecc_reg[VendorTestIdx]) begin + release tb.dut.gen_partitions[VendorTestIdx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0]; + force_sw_parts_ecc_reg[VendorTestIdx] = 0; + end + if (force_sw_parts_ecc_reg[CreatorSwCfgIdx]) begin + release tb.dut.gen_partitions[CreatorSwCfgIdx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0]; + force_sw_parts_ecc_reg[CreatorSwCfgIdx] = 0; + end + if (force_sw_parts_ecc_reg[OwnerSwCfgIdx]) begin + release tb.dut.gen_partitions[OwnerSwCfgIdx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0]; + force_sw_parts_ecc_reg[OwnerSwCfgIdx] = 0; + end + if (force_sw_parts_ecc_reg[RotCreatorAuthIdx]) begin + release tb.dut.gen_partitions[RotCreatorAuthIdx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0]; + force_sw_parts_ecc_reg[RotCreatorAuthIdx] = 0; + end + if (force_sw_parts_ecc_reg[RotOwnerAuthSlot0Idx]) begin + release tb.dut.gen_partitions[RotOwnerAuthSlot0Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0]; + force_sw_parts_ecc_reg[RotOwnerAuthSlot0Idx] = 0; + end + if (force_sw_parts_ecc_reg[RotOwnerAuthSlot1Idx]) begin + release tb.dut.gen_partitions[RotOwnerAuthSlot1Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0]; + force_sw_parts_ecc_reg[RotOwnerAuthSlot1Idx] = 0; + end + if (force_sw_parts_ecc_reg[PlatIntegAuthSlot0Idx]) begin + release tb.dut.gen_partitions[PlatIntegAuthSlot0Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0]; + force_sw_parts_ecc_reg[PlatIntegAuthSlot0Idx] = 0; + end + if (force_sw_parts_ecc_reg[PlatIntegAuthSlot1Idx]) begin + release tb.dut.gen_partitions[PlatIntegAuthSlot1Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0]; + force_sw_parts_ecc_reg[PlatIntegAuthSlot1Idx] = 0; + end + if (force_sw_parts_ecc_reg[PlatOwnerAuthSlot0Idx]) begin + release tb.dut.gen_partitions[PlatOwnerAuthSlot0Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0]; + force_sw_parts_ecc_reg[PlatOwnerAuthSlot0Idx] = 0; + end + if (force_sw_parts_ecc_reg[PlatOwnerAuthSlot1Idx]) begin + release tb.dut.gen_partitions[PlatOwnerAuthSlot1Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0]; + force_sw_parts_ecc_reg[PlatOwnerAuthSlot1Idx] = 0; + end + if (force_sw_parts_ecc_reg[PlatOwnerAuthSlot2Idx]) begin + release tb.dut.gen_partitions[PlatOwnerAuthSlot2Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0]; + force_sw_parts_ecc_reg[PlatOwnerAuthSlot2Idx] = 0; + end + if (force_sw_parts_ecc_reg[PlatOwnerAuthSlot3Idx]) begin + release tb.dut.gen_partitions[PlatOwnerAuthSlot3Idx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0]; + force_sw_parts_ecc_reg[PlatOwnerAuthSlot3Idx] = 0; + end + if (force_sw_parts_ecc_reg[RomPatchIdx]) begin + release tb.dut.gen_partitions[RomPatchIdx].gen_unbuffered. + u_part_unbuf.`ECC_REG_PATH.data_i[0]; + force_sw_parts_ecc_reg[RomPatchIdx] = 0; + end + endtask + + // Force prim_generic_otp input cmd_i to a invalid value. + task automatic force_invalid_otp_cmd_i(); + @(posedge clk_i); + force `PRIM_GENERIC_OTP_CMD_I_PATH = prim_otp_pkg::cmd_e'(2'b10); + endtask + + task automatic release_invalid_otp_cmd_i(); + @(posedge clk_i); + release `PRIM_GENERIC_OTP_CMD_I_PATH; + endtask + + // Force part_buf partitions output otp_cmd_o to a invalid value. + task automatic force_invalid_part_cmd_o(int part_idx); + @(posedge clk_i); + case (part_idx) + HwCfg0Idx: force `BUF_PART_OTP_CMD_PATH(HwCfg0Idx) = prim_otp_pkg::cmd_e'(2'b10); + HwCfg1Idx: force `BUF_PART_OTP_CMD_PATH(HwCfg1Idx) = prim_otp_pkg::cmd_e'(2'b10); + Secret0Idx: force `BUF_PART_OTP_CMD_PATH(Secret0Idx) = prim_otp_pkg::cmd_e'(2'b10); + Secret1Idx: force `BUF_PART_OTP_CMD_PATH(Secret1Idx) = prim_otp_pkg::cmd_e'(2'b10); + Secret2Idx: force `BUF_PART_OTP_CMD_PATH(Secret2Idx) = prim_otp_pkg::cmd_e'(2'b10); + Secret3Idx: force `BUF_PART_OTP_CMD_PATH(Secret3Idx) = prim_otp_pkg::cmd_e'(2'b10); + LifeCycleIdx: force `LC_PART_OTP_CMD_PATH = prim_otp_pkg::cmd_e'(2'b10); + default: begin + `uvm_fatal("otp_ctrl_if", + $sformatf("force invalid otp_cmd_o only supports buffered partitions: %0d", part_idx)) + end + endcase + endtask + + task automatic release_invalid_part_cmd_o(int part_idx); + @(posedge clk_i); + case (part_idx) + HwCfg0Idx: release `BUF_PART_OTP_CMD_PATH(HwCfg0Idx); + HwCfg1Idx: release `BUF_PART_OTP_CMD_PATH(HwCfg1Idx); + Secret0Idx: release `BUF_PART_OTP_CMD_PATH(Secret0Idx); + Secret1Idx: release `BUF_PART_OTP_CMD_PATH(Secret1Idx); + Secret2Idx: release `BUF_PART_OTP_CMD_PATH(Secret2Idx); + Secret3Idx: release `BUF_PART_OTP_CMD_PATH(Secret3Idx); + LifeCycleIdx: release `LC_PART_OTP_CMD_PATH; + default: begin + `uvm_fatal("otp_ctrl_if", + $sformatf("release invalid otp_cmd_o only supports buffered partitions: %0d", + part_idx)) + end + endcase + endtask + + // This task forces otp_ctrl's internal mubi signals to values that are not mubi::true or mubi:: + // false. Then scb will check if design treats these values as locking the partition access. + task automatic force_part_access_mubi(otp_part_access_lock_t forced_part_access_sel[NumPart-1]); + @(posedge clk_i); + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(VendorTestIdx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(CreatorSwCfgIdx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(OwnerSwCfgIdx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(OwnershipSlotStateIdx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(RotCreatorAuthIdx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(RotOwnerAuthSlot0Idx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(RotOwnerAuthSlot1Idx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(PlatIntegAuthSlot0Idx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(PlatIntegAuthSlot1Idx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(PlatOwnerAuthSlot0Idx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(PlatOwnerAuthSlot1Idx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(PlatOwnerAuthSlot2Idx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(PlatOwnerAuthSlot3Idx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(ExtNvmIdx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(RomPatchIdx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(HwCfg0Idx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(HwCfg1Idx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(Secret0Idx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(Secret1Idx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(Secret2Idx) + `FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL(Secret3Idx) + endtask + + task automatic release_part_access_mubi(); + @(posedge clk_i); + release tb.dut.part_access; + release tb.dut.part_access_dai; + endtask + + // Connectivity assertions for test related I/Os. + `ASSERT(LcOtpTestStatusO_A, otp_vendor_test_status_o == `PRIM_GENERIC_OTP_PATH.test_status_o) + `ASSERT(LcOtpTestCtrlI_A, otp_vendor_test_ctrl_i == `PRIM_GENERIC_OTP_PATH.test_ctrl_i) + + `ASSERT(CioTestOWithDftOn_A, lc_dft_en_i == lc_ctrl_pkg::On |-> + ##[2:3] cio_test_o == `PRIM_GENERIC_OTP_PATH.test_vect_o) + `ASSERT(CioTestOWithDftOff_A, lc_dft_en_i != lc_ctrl_pkg::On |-> ##[2:3] cio_test_o == 0) + `ASSERT(CioTestEnOWithDftOn_A, lc_dft_en_i == lc_ctrl_pkg::On |-> ##[2:3] cio_test_en_o == '1) + `ASSERT(CioTestEnOWithDftOff_A, lc_dft_en_i != lc_ctrl_pkg::On |-> ##[2:3] cio_test_en_o == 0) + + + `define OTP_ASSERT_WO_LC_ESC(NAME, SEQ) \ + `ASSERT(NAME, SEQ, clk_i, !rst_ni || lc_esc_on || alert_reqs) + + // If pwr_otp_idle is set only if pwr_otp init is done + `OTP_ASSERT_WO_LC_ESC(OtpPwrDoneWhenIdle_A, pwr_otp_idle_o |-> pwr_otp_done_o) + + // otp_broadcast_o is valid only when otp init is done + `OTP_ASSERT_WO_LC_ESC(OtpHwCfgValidOn_A, pwr_otp_done_o |-> + otp_broadcast_o.valid == lc_ctrl_pkg::On) + // If otp_broadcast is Off, then hw partition is not finished calculation, + // then otp init is not done + `OTP_ASSERT_WO_LC_ESC(OtpHwCfgValidOff_A, otp_broadcast_o.valid == lc_ctrl_pkg::Off |-> + pwr_otp_done_o == 0) + // Once OTP init is done, otp_broadcast_o output value stays stable until next power cycle + `OTP_ASSERT_WO_LC_ESC(OtpHwCfgStable_A, otp_broadcast_o.valid == lc_ctrl_pkg::On |=> + $stable(otp_broadcast_o)) + + // Otp_keymgr valid is related to part_digest, should not be changed after otp_pwr_init + `OTP_ASSERT_WO_LC_ESC(OtpKeymgrValidStable0_A, pwr_otp_done_o |-> + $stable(keymgr_key_o.creator_root_key_share0_valid)) + `OTP_ASSERT_WO_LC_ESC(OtpKeymgrValidStable1_A, pwr_otp_done_o |-> + $stable(keymgr_key_o.creator_root_key_share1_valid)) + `OTP_ASSERT_WO_LC_ESC(OtpKeymgrValidStable2_A, pwr_otp_done_o |-> + $stable(keymgr_key_o.creator_seed_valid)) + `OTP_ASSERT_WO_LC_ESC(OtpKeymgrValidStable3_A, pwr_otp_done_o |-> + $stable(keymgr_key_o.owner_seed_valid)) + + // During lc_prog_req, either otp_idle will be reset or lc_error is set + `OTP_ASSERT_WO_LC_ESC(LcProgReq_A, $rose(lc_prog_req) |=> + (pwr_otp_idle_o == 0 || $rose(lc_prog_err)) within lc_prog_req[*1:$]) + + // During fatal alert, check if otp outputs revert back to default value. + // Wait three clock cycles until error propogates to each FSM states and regs. + `define OTP_FATAL_ERR_ASSERT(NAME, SEQ) \ + `ASSERT(FatalErr``NAME``, alert_reqs |-> ##3 SEQ) + + `OTP_FATAL_ERR_ASSERT(LcDataValid_A, lc_data_o.valid == 0 && lc_data_o.error == 1) + `OTP_FATAL_ERR_ASSERT(LcDataState_A, lc_data_o.state == + PartInvDefault[LcStateOffset*8+:LcStateSize*8]) + `OTP_FATAL_ERR_ASSERT(LcDataCount_A, lc_data_o.count == + PartInvDefault[LcTransitionCntOffset*8+:LcTransitionCntSize*8]) + `OTP_FATAL_ERR_ASSERT(LcDataTestUnlockToken_A, lc_data_o.test_unlock_token == + PartInvDefault[TestUnlockTokenOffset*8+:TestUnlockTokenSize*8]) + `OTP_FATAL_ERR_ASSERT(LcDataTestExitToken_A, lc_data_o.test_exit_token == + PartInvDefault[TestExitTokenOffset*8+:TestExitTokenSize*8]) + `OTP_FATAL_ERR_ASSERT(LcDataRmaToken_A, lc_data_o.rma_token == + PartInvDefault[RmaTokenOffset*8+:RmaTokenSize*8]) + + `OTP_FATAL_ERR_ASSERT(KeymgrKeyData_A, keymgr_key_o.creator_root_key_share0 == + PartInvDefault[CreatorRootKeyShare0Offset*8+:CreatorRootKeyShare0Size*8] && + keymgr_key_o.creator_root_key_share1 == + PartInvDefault[CreatorRootKeyShare1Offset*8+:CreatorRootKeyShare1Size*8]) + + `OTP_FATAL_ERR_ASSERT(HwCfgOValid_A, otp_broadcast_o.valid == lc_ctrl_pkg::Off) + `OTP_FATAL_ERR_ASSERT(HwCfg0OData_A, otp_broadcast_o.hw_cfg0_data == + PartInvDefault[HwCfg0Offset*8+:HwCfg0Size*8]) + `OTP_FATAL_ERR_ASSERT(HwCfg1OData_A, otp_broadcast_o.hw_cfg1_data == + PartInvDefault[HwCfg1Offset*8+:HwCfg1Size*8]) + + `OTP_FATAL_ERR_ASSERT(LcProgAck_A, lc_prog_ack == 0) + `OTP_FATAL_ERR_ASSERT(FlashAcks_A, flash_acks == 0) + `OTP_FATAL_ERR_ASSERT(SramAcks_A, sram_acks == 0) + `OTP_FATAL_ERR_ASSERT(OtbnAck_A, otbn_ack == 0) + + `undef OTP_ASSERT_WO_LC_ESC + `undef OTP_FATAL_ERR_ASSERT + `undef ECC_REG_PATH + `undef BUF_PART_OTP_CMD_PATH + `undef LC_PART_OTP_CMD_PATH + `undef PRIM_GENERIC_OTP_PATH + `undef PRIM_GENERIC_OTP_CMD_I_PATH + `undef FORCE_OTP_PART_LOCK_WITH_RAND_NON_MUBI_VAL +endinterface diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_scoreboard.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_scoreboard.sv new file mode 100644 index 00000000000000..05fabdb37a710e --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_scoreboard.sv @@ -0,0 +1,2108 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +class otp_ctrl_scoreboard #(type CFG_T = otp_ctrl_env_cfg) + extends cip_base_scoreboard #( + .CFG_T(CFG_T), + .RAL_T(otp_ctrl_core_reg_block), + .COV_T(otp_ctrl_env_cov) + ); + `uvm_component_param_utils(otp_ctrl_scoreboard #(CFG_T)) + + // local variables + bit [TL_DW-1:0] otp_a [OTP_ARRAY_SIZE]; + + // lc_state and lc_cnt that stored in OTP + bit [LC_PROG_DATA_SIZE-1:0] otp_lc_data; + bit [EDN_BUS_WIDTH-1:0] edn_data_q[$]; + + // This flag is used when reset is issued during otp dai write access. + bit dai_wr_ip; + int dai_digest_ip = LifeCycleIdx; // Default to LC as it does not have digest. + bit ignore_digest_chk = 0; + + // This bit is used for DAI interface to mark if the read access is valid. + bit dai_read_valid; + + // This captures the regwen state as configured by the SW side (i.e. without HW modulation + // with the idle signal overlaid). + bit direct_access_regwen_state = 1; + + // ICEBOX(#17798): currently scb will skip checking the readout value if the ECC error is + // uncorrectable. Because if the error is uncorrectable, current scb does not track all the + // backdoor injected values. + // This issue proposes to track the otp_memory_array in mem_bkdr_if and once backdoor inject any + // value, mem_bkdr_if will update its otp_memory_array. + bit check_dai_rd_data = 1; + + // Status related variables + bit under_chk, under_dai_access; + bit [TL_DW-1:0] exp_status, status_mask; + + otp_alert_e exp_alert = OtpNoAlert; + + // TLM agent fifos + uvm_tlm_analysis_fifo #(push_pull_item#(.DeviceDataWidth(SRAM_DATA_SIZE))) + sram_fifos[NumSramKeyReqSlots]; + uvm_tlm_analysis_fifo #(push_pull_item#(.DeviceDataWidth(OTBN_DATA_SIZE))) otbn_fifo; + uvm_tlm_analysis_fifo #(push_pull_item#(.DeviceDataWidth(FLASH_DATA_SIZE))) flash_addr_fifo; + uvm_tlm_analysis_fifo #(push_pull_item#(.DeviceDataWidth(FLASH_DATA_SIZE))) flash_data_fifo; + uvm_tlm_analysis_fifo #(push_pull_item#(.DeviceDataWidth(1), .HostDataWidth(LC_PROG_DATA_SIZE))) + lc_prog_fifo; + + // local queues to hold incoming packets pending comparison + + `uvm_component_new + + function void build_phase(uvm_phase phase); + super.build_phase(phase); + for (int i = 0; i < NumSramKeyReqSlots; i++) begin + sram_fifos[i] = new($sformatf("sram_fifos[%0d]", i), this); + end + otbn_fifo = new("otbn_fifo", this); + flash_addr_fifo = new("flash_addr_fifo", this); + flash_data_fifo = new("flash_data_fifo", this); + lc_prog_fifo = new("lc_prog_fifo", this); + endfunction + + function void connect_phase(uvm_phase phase); + super.connect_phase(phase); + endfunction + + task run_phase(uvm_phase phase); + super.run_phase(phase); + fork + process_wipe_mem(); + process_otp_power_up(); + process_lc_esc(); + process_lc_prog_req(); + process_edn_req(); + check_otbn_rsp(); + check_flash_rsps(); + check_sram_rsps(); + recover_lc_prog_req(); + join_none + endtask + + // Once sequence uses backdoor method to clear memory, this task resets internal otp_a and + // resets `cfg.backdoor_clear_mem` to 0. + virtual task process_wipe_mem(); + forever begin + @(posedge cfg.backdoor_clear_mem) begin + bit [SCRAMBLE_DATA_SIZE-1:0] data; + otp_a = '{default:0}; + otp_lc_data = '{default:0}; + // secret partitions have been scrambled before writing to OTP. + // here calculate the pre-srambled raw data when clearing internal OTP to all 0s. + data = descramble_data(0, Secret0Idx); + for (int i = Secret0Offset / TL_SIZE; + i <= Secret0DigestOffset / TL_SIZE - 1; + i++) begin + otp_a[i] = ((i - Secret0Offset / TL_SIZE) % 2) ? + data[SCRAMBLE_DATA_SIZE-1:TL_DW] : data[TL_DW-1:0]; + end + // secret partitions have been scrambled before writing to OTP. + // here calculate the pre-srambled raw data when clearing internal OTP to all 0s. + data = descramble_data(0, Secret1Idx); + for (int i = Secret1Offset / TL_SIZE; + i <= Secret1DigestOffset / TL_SIZE - 1; + i++) begin + otp_a[i] = ((i - Secret1Offset / TL_SIZE) % 2) ? + data[SCRAMBLE_DATA_SIZE-1:TL_DW] : data[TL_DW-1:0]; + end + // secret partitions have been scrambled before writing to OTP. + // here calculate the pre-srambled raw data when clearing internal OTP to all 0s. + data = descramble_data(0, Secret2Idx); + for (int i = Secret2Offset / TL_SIZE; + i <= Secret2DigestOffset / TL_SIZE - 1; + i++) begin + otp_a[i] = ((i - Secret2Offset / TL_SIZE) % 2) ? + data[SCRAMBLE_DATA_SIZE-1:TL_DW] : data[TL_DW-1:0]; + end + // secret partitions have been scrambled before writing to OTP. + // here calculate the pre-srambled raw data when clearing internal OTP to all 0s. + data = descramble_data(0, Secret3Idx); + for (int i = Secret3Offset / TL_SIZE; + i <= Secret3DigestOffset / TL_SIZE - 1; + i++) begin + otp_a[i] = ((i - Secret3Offset / TL_SIZE) % 2) ? + data[SCRAMBLE_DATA_SIZE-1:TL_DW] : data[TL_DW-1:0]; + end + `uvm_info(`gfn, "clear internal memory and digest", UVM_HIGH) + cfg.backdoor_clear_mem = 0; + dai_wr_ip = 0; + dai_digest_ip = LifeCycleIdx; + end + end + endtask + + // This task process the following logic in during otp_power_up: + // 1. After reset deasserted, otp access is locked until pwr_otp_done_o is set + // 2. After reset deasserted, if power otp_init request is on, and if testbench uses backdoor to + // clear OTP memory to all zeros, clear all digests and re-calculate secret partitions + virtual task process_otp_power_up(); + forever begin + wait (cfg.en_scb); + @(posedge cfg.otp_ctrl_vif.pwr_otp_done_o || cfg.under_reset || + cfg.otp_ctrl_vif.alert_reqs) begin + if (!cfg.under_reset && !cfg.otp_ctrl_vif.alert_reqs && cfg.en_scb) begin + otp_ctrl_part_pkg::otp_hw_cfg0_data_t exp_hw_cfg0_data; + otp_ctrl_part_pkg::otp_hw_cfg1_data_t exp_hw_cfg1_data; + otp_ctrl_pkg::otp_keymgr_key_t exp_keymgr_data; + otp_ctrl_pkg::otp_lc_data_t exp_lc_data; + bit [otp_ctrl_pkg::KeyMgrKeyWidth-1:0] exp_keymgr_key0, exp_keymgr_key1; + + if (PartInfo[dai_digest_ip].sw_digest || PartInfo[dai_digest_ip].hw_digest) begin + bit [TL_DW-1:0] otp_addr = PART_OTP_DIGEST_ADDRS[dai_digest_ip]; + otp_a[otp_addr] = cfg.mem_bkdr_util_h.read32(otp_addr << 2); + otp_a[otp_addr+1] = cfg.mem_bkdr_util_h.read32((otp_addr << 2) + 4); + dai_digest_ip = LifeCycleIdx; + end + predict_digest_csrs(); + + if (cfg.otp_ctrl_vif.under_error_states() == 0) begin + // Dai access is unlocked because the power init is done + void'(ral.direct_access_regwen.predict(direct_access_regwen_state)); + + // Dai idle is set because the otp init is done + exp_status[OtpDaiIdleIdx] = 1; + end + + // Hwcfg_o gets data from OTP HW cfg partition + exp_hw_cfg0_data = cfg.otp_ctrl_vif.under_error_states() ? + otp_ctrl_part_pkg::PartInvDefault[HwCfg0Offset*8 +: HwCfg0Size*8] : + otp_hw_cfg0_data_t'({<<32 {otp_a[HwCfg0Offset/4 +: HwCfg0Size/4]}}); + `DV_CHECK_EQ(cfg.otp_ctrl_vif.otp_broadcast_o.valid, lc_ctrl_pkg::On) + `DV_CHECK_EQ(cfg.otp_ctrl_vif.otp_broadcast_o.hw_cfg0_data, exp_hw_cfg0_data) + + // Hwcfg_o gets data from OTP HW cfg partition + exp_hw_cfg1_data = cfg.otp_ctrl_vif.under_error_states() ? + otp_ctrl_part_pkg::PartInvDefault[HwCfg1Offset*8 +: HwCfg1Size*8] : + otp_hw_cfg1_data_t'({<<32 {otp_a[HwCfg1Offset/4 +: HwCfg1Size/4]}}); + `DV_CHECK_EQ(cfg.otp_ctrl_vif.otp_broadcast_o.valid, lc_ctrl_pkg::On) + `DV_CHECK_EQ(cfg.otp_ctrl_vif.otp_broadcast_o.hw_cfg1_data, exp_hw_cfg1_data) + + if (!cfg.otp_ctrl_vif.under_error_states()) begin + // ---------------------- Check lc_data_o output ----------------------------------- + // Because initialization was succesful, the valid should be set and error should be + // reset. + exp_lc_data.valid = 1; + exp_lc_data.error = 0; + + // Secrets and tokens valid signals are depend on whether secret partitions are + // locked. + exp_lc_data.secrets_valid = get_otp_digest_val(Secret2Idx) ? On : Off; + exp_lc_data.test_tokens_valid = get_otp_digest_val(Secret0Idx) ? On : Off; + exp_lc_data.rma_token_valid = get_otp_digest_val(Secret2Idx) ? On : Off; + + // LC output is depend on LC partitions value. + exp_lc_data.count = otp_lc_data[0 +: LcCountWidth]; + exp_lc_data.state = otp_lc_data[LcCountWidth +: LcStateWidth]; + + // Token values are depend on secret partitions value. + exp_lc_data.test_unlock_token = + {<<32 {otp_a[TestUnlockTokenOffset/4 +: TestUnlockTokenSize/4]}}; + exp_lc_data.test_exit_token = + {<<32 {otp_a[TestExitTokenOffset/4 +: TestExitTokenSize/4]}}; + exp_lc_data.rma_token = {<<32 {otp_a[RmaTokenOffset/4 +: RmaTokenSize/4]}}; + + // Check otp_lc_data_t struct by item is easier to debug. + `DV_CHECK_EQ(cfg.otp_ctrl_vif.lc_data_o.valid, exp_lc_data.valid) + `DV_CHECK_EQ(cfg.otp_ctrl_vif.lc_data_o.error, exp_lc_data.error) + `DV_CHECK_EQ(cfg.otp_ctrl_vif.lc_data_o.state, exp_lc_data.state) + `DV_CHECK_EQ(cfg.otp_ctrl_vif.lc_data_o.count, exp_lc_data.count) + `DV_CHECK_EQ(cfg.otp_ctrl_vif.lc_data_o.secrets_valid, exp_lc_data.secrets_valid) + `DV_CHECK_EQ(cfg.otp_ctrl_vif.lc_data_o.test_tokens_valid, + exp_lc_data.test_tokens_valid) + `DV_CHECK_EQ(cfg.otp_ctrl_vif.lc_data_o.test_unlock_token, + exp_lc_data.test_unlock_token) + `DV_CHECK_EQ(cfg.otp_ctrl_vif.lc_data_o.test_exit_token, exp_lc_data.test_exit_token) + `DV_CHECK_EQ(cfg.otp_ctrl_vif.lc_data_o.rma_token_valid, exp_lc_data.rma_token_valid) + `DV_CHECK_EQ(cfg.otp_ctrl_vif.lc_data_o.rma_token, exp_lc_data.rma_token) + + // Check otp_lc_data_t all together in case there is any missed item. + `DV_CHECK_EQ(cfg.otp_ctrl_vif.lc_data_o, exp_lc_data) + + // ---------------------- Check keymgr_key_o output --------------------------------- + // Otp_keymgr outputs creator and owner keys from secret partitions. + // Depends on lc_seed_hw_rd_en_i, it will output the real keys or a constant + exp_keymgr_data = '0; + exp_keymgr_data.creator_root_key_share0_valid = get_otp_digest_val(Secret2Idx) != 0; + if (cfg.otp_ctrl_vif.lc_seed_hw_rd_en_i == lc_ctrl_pkg::On) begin + exp_keymgr_data.creator_root_key_share0 = + {<<32 {otp_a[CreatorRootKeyShare0Offset/4 +: CreatorRootKeyShare0Size/4]}}; + end else begin + exp_keymgr_data.creator_root_key_share0 = + PartInvDefault[CreatorRootKeyShare0Offset*8 +: CreatorRootKeyShare0Size*8]; + end + // Check otp_keymgr_key_t struct by item is easier to debug. + `DV_CHECK_EQ(cfg.otp_ctrl_vif.keymgr_key_o.creator_root_key_share0_valid, + exp_keymgr_data.creator_root_key_share0_valid) + exp_keymgr_data.creator_root_key_share1_valid = get_otp_digest_val(Secret2Idx) != 0; + if (cfg.otp_ctrl_vif.lc_seed_hw_rd_en_i == lc_ctrl_pkg::On) begin + exp_keymgr_data.creator_root_key_share1 = + {<<32 {otp_a[CreatorRootKeyShare1Offset/4 +: CreatorRootKeyShare1Size/4]}}; + end else begin + exp_keymgr_data.creator_root_key_share1 = + PartInvDefault[CreatorRootKeyShare1Offset*8 +: CreatorRootKeyShare1Size*8]; + end + // Check otp_keymgr_key_t struct by item is easier to debug. + `DV_CHECK_EQ(cfg.otp_ctrl_vif.keymgr_key_o.creator_root_key_share1_valid, + exp_keymgr_data.creator_root_key_share1_valid) + exp_keymgr_data.creator_seed_valid = get_otp_digest_val(Secret2Idx) != 0; + if (cfg.otp_ctrl_vif.lc_seed_hw_rd_en_i == lc_ctrl_pkg::On) begin + exp_keymgr_data.creator_seed = + {<<32 {otp_a[CreatorSeedOffset/4 +: CreatorSeedSize/4]}}; + end else begin + exp_keymgr_data.creator_seed = + PartInvDefault[CreatorSeedOffset*8 +: CreatorSeedSize*8]; + end + // Check otp_keymgr_key_t struct by item is easier to debug. + `DV_CHECK_EQ(cfg.otp_ctrl_vif.keymgr_key_o.creator_seed_valid, + exp_keymgr_data.creator_seed_valid) + exp_keymgr_data.owner_seed_valid = get_otp_digest_val(Secret3Idx) != 0; + if (cfg.otp_ctrl_vif.lc_seed_hw_rd_en_i == lc_ctrl_pkg::On) begin + exp_keymgr_data.owner_seed = + {<<32 {otp_a[OwnerSeedOffset/4 +: OwnerSeedSize/4]}}; + end else begin + exp_keymgr_data.owner_seed = + PartInvDefault[OwnerSeedOffset*8 +: OwnerSeedSize*8]; + end + // Check otp_keymgr_key_t struct by item is easier to debug. + `DV_CHECK_EQ(cfg.otp_ctrl_vif.keymgr_key_o.owner_seed_valid, + exp_keymgr_data.owner_seed_valid) + + // Check otp_keymgr_key_t struct all together in case there is any missed item. + `DV_CHECK_EQ(cfg.otp_ctrl_vif.keymgr_key_o, exp_keymgr_data) + + if (cfg.en_cov) begin + cov.keymgr_o_cg.sample(cfg.otp_ctrl_vif.lc_seed_hw_rd_en_i == lc_ctrl_pkg::On, + exp_keymgr_data.creator_root_key_share0_valid); + end + end + end else if (cfg.otp_ctrl_vif.alert_reqs) begin + // Ignore digest CSR check when otp_ctrl initialization is interrupted by fatal errors. + // SCB cannot predict how many partitions already finished initialization and updated + // the digest value to CSRs. + ignore_digest_chk = 1; + end + if (cfg.en_cov) begin + bit [NumPart-2:0] parts_locked; + foreach (parts_locked[i]) parts_locked[i] = (get_otp_digest_val(i) != 0); + cov.power_on_cg.sample(cfg.otp_ctrl_vif.lc_esc_on, parts_locked); + end + end + end + endtask + + // This task monitors internal escalation triggered by two methods: + // 1. Externally lc_escalation_en is set to lc_ctrl_pkg::On. + // 2. Internal fatal alert triggered and all partitions are driven to error states. + virtual task process_lc_esc(); + forever begin + wait(cfg.otp_ctrl_vif.alert_reqs == 1 && cfg.en_scb); + + if (cfg.otp_ctrl_vif.lc_esc_on == 0) `DV_CHECK_NE(exp_alert, OtpNoAlert) + + if (exp_alert != OtpCheckAlert) set_exp_alert("fatal_check_error", 1, 5); + + // If the lc_escalation is triggered by internal fatal alert, wait 2 negedge until status is + // updated internally + if (cfg.otp_ctrl_vif.lc_esc_on == 0) begin + cfg.clk_rst_vif.wait_n_clks(2); + exp_status[OtpCheckPendingIdx] = 0; + exp_status[OtpDaiIdleIdx] = 0; + end else begin + exp_status = '0; + // Only lc_esc_on will set these bits to 1. + exp_status[OtpDerivKeyFsmErrIdx:OtpLfsrFsmErrIdx] = '1; + end + + // Update status bits. + foreach (FATAL_EXP_STATUS[i]) begin + if (FATAL_EXP_STATUS[i]) begin + predict_err(.status_err_idx(otp_status_e'(i)), .err_code(OtpFsmStateError), + .update_esc_err(1)); + end + end + + // Update digest values and direct_access_regwen. + predict_rdata(1, 0, 0); + void'(ral.direct_access_regwen.predict(.value(0), .kind(UVM_PREDICT_READ))); + + // DAI access is locked until reset, so no need to backdoor read otp write value until reset. + + wait(cfg.otp_ctrl_vif.alert_reqs == 0); + end + endtask + + // This task monitors if lc_program req is interrupted by reset. + // If it happens, scb cannot predict how many bits have been written to OTP_CTRL. + // So here we will backdoor read back OTP lc partitions bits. + virtual task recover_lc_prog_req(); + forever begin + wait(cfg.otp_ctrl_vif.lc_prog_req == 1); + wait(cfg.otp_ctrl_vif.lc_prog_req == 0); + // Wait one 1ps to avoid race condition. + #1ps; + if (cfg.otp_ctrl_vif.rst_ni == 0) begin + for (int i = 0; i < LC_PROG_DATA_SIZE/32; i++) begin + otp_lc_data[i*32+:32] = cfg.mem_bkdr_util_h.read32(LifeCycleOffset + i * 4); + end + end + end + endtask + + virtual task process_lc_prog_req(); + forever begin + push_pull_item#(.DeviceDataWidth(1), .HostDataWidth(LC_PROG_DATA_SIZE)) rcv_item; + bit exp_err_bit; + bit [15:0] rcv_words [LC_PROG_DATA_SIZE/16]; + + lc_prog_fifo.get(rcv_item); + + // LCI is updated by OTP word. + rcv_words = {<< 16{rcv_item.h_data}}; + foreach (rcv_words[i]) begin + bit [15:0] curr_word = otp_lc_data[i*16 +: 16]; + if ((curr_word & rcv_words[i]) == curr_word) otp_lc_data[i*16 +: 16] = rcv_words[i]; + else exp_err_bit = 1; + end + + if (exp_err_bit) predict_err(OtpLciErrIdx, OtpMacroWriteBlankError); + else predict_no_err(OtpLciErrIdx); + + // LC program request data is valid means no OTP macro error. + `DV_CHECK_EQ(rcv_item.d_data, exp_err_bit) + + if (cfg.en_cov) cov.lc_prog_cg.sample(exp_err_bit); + end + endtask + + virtual task process_edn_req(); + forever begin + push_pull_item#(.DeviceDataWidth(EDN_DATA_WIDTH)) edn_item; + edn_fifos[0].get(edn_item); + edn_data_q.push_back(edn_item.d_data[EDN_BUS_WIDTH-1:0]); + end + endtask + + virtual task check_otbn_rsp(); + forever begin + push_pull_item#(.DeviceDataWidth(OTBN_DATA_SIZE)) rcv_item; + bit [SCRAMBLE_KEY_SIZE-1:0] edn_key2, edn_key1; + bit [SCRAMBLE_KEY_SIZE-1:0] sram_key; + bit [SCRAMBLE_DATA_SIZE-1:0] exp_key_lower, exp_key_higher; + bit [OtbnKeyWidth-1:0] key, exp_key; + bit [OtbnNonceWidth-1:0] nonce, exp_nonce; + bit seed_valid; + bit part_locked; + + otbn_fifo.get(rcv_item); + seed_valid = rcv_item.d_data[0]; + nonce = rcv_item.d_data[1+:OtbnNonceWidth]; + key = rcv_item.d_data[OtbnNonceWidth+1+:OtbnKeyWidth]; + part_locked = {`gmv(ral.secret1_digest[0]), `gmv(ral.secret1_digest[1])} != '0; + + // seed is valid as long as secret1 is locked + `DV_CHECK_EQ(seed_valid, part_locked, "otbn seed_valid mismatch") + + // If edn_data_q matches the OTBN requested size, check OTBN outputs + if (edn_data_q.size() == NUM_OTBN_EDN_REQ) begin + {exp_nonce, edn_key2, edn_key1} = {<<32{edn_data_q}}; + + // check nonce value + `DV_CHECK_EQ(nonce, exp_nonce, "otbn nonce mismatch") + + // calculate key + sram_key = get_key_from_otp(part_locked, SramDataKeySeedOffset / 4); + exp_key_lower = present_encode_with_final_const( + .data(RndCnstDigestIV[SramDataKey]), + .key(sram_key), + .final_const(RndCnstDigestConst[SramDataKey]), + .second_key(edn_key1), + .num_round(2)); + + exp_key_higher = present_encode_with_final_const( + .data(RndCnstDigestIV[SramDataKey]), + .key(sram_key), + .final_const(RndCnstDigestConst[SramDataKey]), + .second_key(edn_key2), + .num_round(2)); + exp_key = {exp_key_higher, exp_key_lower}; + `DV_CHECK_EQ(key, exp_key, "otbn key mismatch") + + if (cfg.en_cov) cov.otbn_req_cg.sample(part_locked); + + // If during OTBN key request, the LFSR timer expired and trigger an EDN request to acquire + // two EDN keys, then ignore the OTBN output checking, because scb did not know which EDN + // keys are used for LFSR. + // Thus any edn_data_q size equal to (16+2*N) is exempted from checking. + end else if ((edn_data_q.size() - NUM_OTBN_EDN_REQ) % 2 != 0) begin + `uvm_error(`gfn, $sformatf("Unexpected edn_data_q size (%0d) during OTBN request", + edn_data_q.size())) + end + edn_data_q.delete(); + end + endtask + + virtual task check_flash_rsps(); + for (int i = FlashDataKey; i <= FlashAddrKey; i++) begin + automatic digest_sel_e sel_flash = digest_sel_e'(i); + fork + forever begin + push_pull_item#(.DeviceDataWidth(FLASH_DATA_SIZE)) rcv_item; + bit [SCRAMBLE_KEY_SIZE-1:0] flash_key; + bit [SCRAMBLE_DATA_SIZE-1:0] exp_key_lower, exp_key_higher; + bit [FlashKeyWidth-1:0] key, exp_key; + bit seed_valid, part_locked; + int flash_key_index; + + if (sel_flash == FlashAddrKey) begin + flash_addr_fifo.get(rcv_item); + flash_key_index = FlashAddrKeySeedOffset / 4; + end else begin + flash_data_fifo.get(rcv_item); + flash_key_index = FlashDataKeySeedOffset / 4; + end + seed_valid = rcv_item.d_data[0]; + key = rcv_item.d_data[1+:FlashKeyWidth]; + part_locked = {`gmv(ral.secret1_digest[0]), `gmv(ral.secret1_digest[1])} != '0; + `DV_CHECK_EQ(seed_valid, part_locked, + $sformatf("flash %0s seed_valid mismatch", sel_flash.name())) + + // calculate key + flash_key = get_key_from_otp(part_locked, flash_key_index); + exp_key_lower = present_encode_with_final_const( + .data(RndCnstDigestIV[sel_flash]), + .key(flash_key), + .final_const(RndCnstDigestConst[sel_flash])); + + flash_key = get_key_from_otp(part_locked, flash_key_index + 4); + exp_key_higher = present_encode_with_final_const( + .data(RndCnstDigestIV[sel_flash]), + .key(flash_key), + .final_const(RndCnstDigestConst[sel_flash])); + exp_key = {exp_key_higher, exp_key_lower}; + `DV_CHECK_EQ(key, exp_key, $sformatf("flash %s key mismatch", sel_flash.name())) + + if (cfg.en_cov) cov.flash_req_cg.sample(sel_flash, part_locked); + end + join_none; + end + endtask + + virtual task check_sram_rsps(); + for (int i = 0; i < NumSramKeyReqSlots; i++) begin + automatic int index = i; + fork + forever begin + push_pull_item#(.DeviceDataWidth(SRAM_DATA_SIZE)) rcv_item; + sram_key_t key, exp_key; + sram_nonce_t nonce, exp_nonce; + bit seed_valid, part_locked; + bit [SCRAMBLE_KEY_SIZE-1:0] edn_key2, edn_key1; + bit [SCRAMBLE_KEY_SIZE-1:0] sram_key; // key used as input to present algo + bit [SCRAMBLE_DATA_SIZE-1:0] exp_key_lower, exp_key_higher; + + sram_fifos[index].get(rcv_item); + seed_valid = rcv_item.d_data[0]; + nonce = rcv_item.d_data[1+:SramNonceWidth]; + key = rcv_item.d_data[SramNonceWidth+1+:SramKeyWidth]; + part_locked = {`gmv(ral.secret1_digest[0]), `gmv(ral.secret1_digest[1])} != '0; + + // seed is valid as long as secret1 is locked + `DV_CHECK_EQ(seed_valid, part_locked, $sformatf("sram_%0d seed_valid mismatch", index)) + + // If edn_data_q matches the OTBN requested size, check OTBN outputs + if (edn_data_q.size() == NUM_SRAM_EDN_REQ) begin + {exp_nonce, edn_key2, edn_key1} = {<<32{edn_data_q}}; + + // check nonce value + `DV_CHECK_EQ(nonce, exp_nonce, $sformatf("sram_%0d nonce mismatch", index)) + + // calculate key + sram_key = get_key_from_otp(part_locked, SramDataKeySeedOffset / 4); + exp_key_lower = present_encode_with_final_const( + .data(RndCnstDigestIV[SramDataKey]), + .key(sram_key), + .final_const(RndCnstDigestConst[SramDataKey]), + .second_key(edn_key1), + .num_round(2)); + + exp_key_higher = present_encode_with_final_const( + .data(RndCnstDigestIV[SramDataKey]), + .key(sram_key), + .final_const(RndCnstDigestConst[SramDataKey]), + .second_key(edn_key2), + .num_round(2)); + exp_key = {exp_key_higher, exp_key_lower}; + `DV_CHECK_EQ(key, exp_key, $sformatf("sram_%0d key mismatch", index)) + if (cfg.en_cov) cov.sram_req_cg.sample(index, part_locked); + + end else if ((edn_data_q.size() - NUM_SRAM_EDN_REQ) % 2 != 0) begin + `uvm_error(`gfn, $sformatf("Unexpected edn_data_q size (%0d) during SRAM request", + edn_data_q.size())) + end + edn_data_q.delete(); + end + join_none + end + endtask + + virtual task process_tl_access(tl_seq_item item, tl_channels_e channel, string ral_name); + bit write = item.is_write(); + uvm_reg_addr_t csr_addr = cfg.ral_models[ral_name].get_word_aligned_addr(item.a_addr); + bit [TL_AW-1:0] addr_mask = ral.get_addr_mask(); + + bit addr_phase_read = (!write && channel == AddrChannel); + bit addr_phase_write = (write && channel == AddrChannel); + bit data_phase_read = (!write && channel == DataChannel); + bit data_phase_write = (write && channel == DataChannel); + + if (ral_name != "otp_ctrl_prim_reg_block") begin + process_core_tl_access(item, csr_addr, ral_name, addr_mask, + addr_phase_read, addr_phase_write, data_phase_read, data_phase_write); + end else begin + process_prim_tl_access(item, csr_addr, ral_name, addr_phase_write, data_phase_read); + end + endtask + + virtual function void process_prim_tl_access(tl_seq_item item, uvm_reg_addr_t csr_addr, + string ral_name, bit addr_phase_write, bit data_phase_read); + + uvm_reg csr; + dv_base_reg dv_reg; + csr = cfg.ral_models[ral_name].default_map.get_reg_by_offset(csr_addr); + `DV_CHECK_NE_FATAL(csr, null) + `downcast(dv_reg, csr) + + if (addr_phase_write) begin + void'(csr.predict(.value(item.a_data), .kind(UVM_PREDICT_WRITE), .be(item.a_mask))); + end else if (data_phase_read) begin + `DV_CHECK_EQ((csr.get_mirrored_value() | status_mask), (item.d_data | status_mask), + $sformatf("reg name: status, compare_mask %0h", status_mask)) + end + endfunction + + virtual function void process_core_tl_access(tl_seq_item item, uvm_reg_addr_t csr_addr, + string ral_name, bit [TL_AW-1:0] addr_mask, bit addr_phase_read, bit addr_phase_write, + bit data_phase_read, bit data_phase_write); + + bit do_read_check = 1; + uvm_reg csr; + dv_base_reg dv_reg; + string csr_name; + + `uvm_info(`gfn, $sformatf("sw state %d, reg state %d", direct_access_regwen_state, + `gmv(ral.direct_access_regwen)), UVM_LOW); + + // if access was to a valid csr, get the csr handle + if (csr_addr inside {cfg.ral_models[ral_name].csr_addrs}) begin + csr = cfg.ral_models[ral_name].default_map.get_reg_by_offset(csr_addr); + `DV_CHECK_NE_FATAL(csr, null) + `downcast(dv_reg, csr) + // SW CFG window + end else if ((csr_addr & addr_mask) inside + {[SW_WINDOW_BASE_ADDR : SW_WINDOW_BASE_ADDR + SW_WINDOW_SIZE]}) begin + if (data_phase_read) begin + bit [TL_AW-1:0] dai_addr = (csr_addr & addr_mask - SW_WINDOW_BASE_ADDR); + bit [TL_AW-1:0] otp_addr = dai_addr >> 2; + int part_idx = get_part_index(dai_addr); + bit [TL_DW-1:0] read_out; + int ecc_err = OtpNoEccErr; + + // We can't get an ECC error if the partition does not have integrity. + if (part_has_integrity(part_idx)) begin + ecc_err = read_a_word_with_ecc(dai_addr, read_out); + end else begin + ecc_err = read_a_word_with_ecc_raw(dai_addr, read_out); + end + + if (part_has_digest(part_idx) && cfg.en_cov) begin + cov.unbuf_access_lock_cg_wrap[part_idx].sample(.read_lock(0), + .write_lock(get_digest_reg_val(part_idx) != 0), .is_write(0)); + end + + // Any alert that indicates the OTP block is in the final error state should not enter the + // logic here, but gated at `is_tl_mem_access_allowed` function. + `DV_CHECK_EQ(cfg.otp_ctrl_vif.alert_reqs, 0) + + // ECC uncorrectable errors are gated by `is_tl_mem_access_allowed` function. + if (ecc_err != OtpNoEccErr && part_has_integrity(part_idx)) begin + + predict_err(otp_status_e'(part_idx), OtpMacroEccCorrError); + if (ecc_err == OtpEccCorrErr) begin + `DV_CHECK_EQ(item.d_data, otp_a[otp_addr], + $sformatf("mem read mismatch at TLUL addr %0h, csr_addr %0h", + csr_addr, dai_addr)) + end else begin + // Only check the first 16 bits because if ECC readout detects uncorrectable error, it + // won't continue read the remaining 16 bits. + `DV_CHECK_EQ(item.d_data & 16'hffff, read_out & 16'hffff, + $sformatf("mem read mismatch at TLUL addr %0h, csr_addr %0h", + csr_addr, dai_addr)) + end + // If there is an injected error, but the partition cannot detect it, we have to compare + // to the value read via the backdoor instead of otp_a[otp_addr] since otherwise the + // perturbed value does not get modelled correctly. + end else if (ecc_err != OtpNoEccErr && !part_has_integrity(part_idx)) begin + `DV_CHECK_EQ(item.d_data, read_out, + $sformatf("mem read mismatch at TLUL addr %0h, csr_addr %0h", + csr_addr, dai_addr)) + predict_no_err(otp_status_e'(part_idx)); + end else if (ecc_err == OtpNoEccErr) begin + `DV_CHECK_EQ(item.d_data, otp_a[otp_addr], + $sformatf("mem read mismatch at TLUL addr %0h, csr_addr %0h", + csr_addr, dai_addr)) + predict_no_err(otp_status_e'(part_idx)); + end + end + return; + end else begin + `uvm_fatal(`gfn, $sformatf("Access unexpected addr 0x%0h", csr_addr)) + end + + csr_name = csr.get_name(); + + if (addr_phase_write) begin + if (cfg.en_cov && cfg.otp_ctrl_vif.alert_reqs && csr_name == "direct_access_cmd") begin + cov.req_dai_access_after_alert_cg.sample(item.a_data); + end + + // Skip predict if the register is locked by `direct_access_regwen`. + // An exception is the direct_access_regwen which may always be written. + if (ral.direct_access_regwen.locks_reg_or_fld(dv_reg) && + `gmv(ral.direct_access_regwen) == 0 && + csr_name != "direct_access_regwen") return; + + void'(csr.predict(.value(item.a_data), .kind(UVM_PREDICT_WRITE), .be(item.a_mask))); + end + + // process the csr req + // for write, update local variable and fifo at address phase + // for read, update predication at address phase and compare at data phase + case (csr_name) + // add individual case item for each csr + "intr_state": begin + if (data_phase_read) begin + // Disable intr_state checking when lc_program is in progress, because scb cannot + // accurately predict when program_error will be triggered. + // We will check the intr_state after lc_program request is done, and the error bit will + // be checked in the `process_lc_prog_req` task. + if (cfg.otp_ctrl_vif.lc_prog_no_sta_check) do_read_check = 0; + if (do_read_check) begin + bit [TL_DW-1:0] intr_en = `gmv(ral.intr_enable); + bit [NumOtpCtrlIntr-1:0] intr_exp = `gmv(ral.intr_state); + + foreach (intr_exp[i]) begin + otp_intr_e intr = otp_intr_e'(i); + `DV_CHECK_CASE_EQ(cfg.intr_vif.pins[i], (intr_en[i] & intr_exp[i]), + $sformatf("Interrupt_pin: %0s", intr.name)); + if (cfg.en_cov) begin + cov.intr_cg.sample(i, intr_en[i], item.d_data[i]); + cov.intr_pins_cg.sample(i, cfg.intr_vif.pins[i]); + end + end + end + end + end + "intr_test": begin + if (addr_phase_write) begin + bit [TL_DW-1:0] intr_en = `gmv(ral.intr_enable); + bit [NumOtpCtrlIntr-1:0] intr_exp = `gmv(ral.intr_state) | item.a_data; + + void'(ral.intr_state.predict(.value(intr_exp))); + if (cfg.en_cov) begin + foreach (intr_exp[i]) begin + cov.intr_test_cg.sample(i, item.a_data[i], intr_en[i], intr_exp[i]); + end + end + end + end + "direct_access_cmd": begin + if (addr_phase_write && !cfg.otp_ctrl_vif.under_error_states()) begin + // here only normalize to 2 lsb, if is secret, will be reduced further + bit [TL_AW-1:0] dai_addr = normalize_dai_addr(`gmv(ral.direct_access_address)); + int part_idx = get_part_index(dai_addr); + bit sw_read_lock = 0; + void'(ral.direct_access_regwen.predict(0)); + under_dai_access = 1; + + // Check if it is sw partition read lock - this can be used in `DaiRead` branch and also + // coverage collection. + if (part_idx == VendorTestIdx) begin + sw_read_lock = `gmv(ral.vendor_test_read_lock) == 0; + end else if (part_idx == CreatorSwCfgIdx) begin + sw_read_lock = `gmv(ral.creator_sw_cfg_read_lock) == 0; + end else if (part_idx == OwnerSwCfgIdx) begin + sw_read_lock = `gmv(ral.owner_sw_cfg_read_lock) == 0; + end else if (part_idx == OwnershipSlotStateIdx) begin + sw_read_lock = `gmv(ral.ownership_slot_state_read_lock) == 0; + end else if (part_idx == RotCreatorAuthIdx) begin + sw_read_lock = `gmv(ral.rot_creator_auth_read_lock) == 0; + end else if (part_idx == RotOwnerAuthSlot0Idx) begin + sw_read_lock = `gmv(ral.rot_owner_auth_slot0_read_lock) == 0; + end else if (part_idx == RotOwnerAuthSlot1Idx) begin + sw_read_lock = `gmv(ral.rot_owner_auth_slot1_read_lock) == 0; + end else if (part_idx == PlatIntegAuthSlot0Idx) begin + sw_read_lock = `gmv(ral.plat_integ_auth_slot0_read_lock) == 0; + end else if (part_idx == PlatIntegAuthSlot1Idx) begin + sw_read_lock = `gmv(ral.plat_integ_auth_slot1_read_lock) == 0; + end else if (part_idx == PlatOwnerAuthSlot0Idx) begin + sw_read_lock = `gmv(ral.plat_owner_auth_slot0_read_lock) == 0; + end else if (part_idx == PlatOwnerAuthSlot1Idx) begin + sw_read_lock = `gmv(ral.plat_owner_auth_slot1_read_lock) == 0; + end else if (part_idx == PlatOwnerAuthSlot2Idx) begin + sw_read_lock = `gmv(ral.plat_owner_auth_slot2_read_lock) == 0; + end else if (part_idx == PlatOwnerAuthSlot3Idx) begin + sw_read_lock = `gmv(ral.plat_owner_auth_slot3_read_lock) == 0; + end else if (part_idx == ExtNvmIdx) begin + sw_read_lock = `gmv(ral.ext_nvm_read_lock) == 0; + end else if (part_idx == RomPatchIdx) begin + sw_read_lock = `gmv(ral.rom_patch_read_lock) == 0; + end + + // LC partition cannot be access via DAI + if (part_idx == LifeCycleIdx) begin + predict_err(OtpDaiErrIdx, OtpAccessError); + if (item.a_data == DaiRead) predict_rdata(is_secret(dai_addr), 0, 0); + end else begin + // Collect coverage. + if (cfg.en_cov) begin + if (part_idx == Secret2Idx) begin + cov.dai_access_secret2_cg.sample( + !(cfg.otp_ctrl_vif.lc_creator_seed_sw_rw_en_i != lc_ctrl_pkg::On), + dai_cmd_e'(item.a_data)); + end else if (is_sw_part_idx(part_idx) && part_has_digest(part_idx) && + item.a_data inside {DaiRead, DaiWrite}) begin + cov.unbuf_access_lock_cg_wrap[part_idx].sample(.read_lock(sw_read_lock), + .write_lock(get_digest_reg_val(part_idx) != 0), + .is_write(item.a_data == DaiWrite)); + + end + end + + case (item.a_data) + DaiDigest: cal_digest_val(part_idx); + DaiRead: begin + // Check if it is sw partition read lock + check_dai_rd_data = 1; + + // SW partitions write read_lock_csr can lock read access. + if (sw_read_lock || + // Secret partitions cal digest can also lock read access. + // However, digest is always readable except SW partitions (Issue #5752). + (is_secret(dai_addr) && get_digest_reg_val(part_idx) != 0 && + !is_digest(dai_addr)) || + // If the partition has creator key material and lc_creator_seed_sw_rw is + // disable, then return access error. + (PartInfo[part_idx].iskeymgr_creator && !is_digest(dai_addr) && + cfg.otp_ctrl_vif.lc_creator_seed_sw_rw_en_i != lc_ctrl_pkg::On)) begin + predict_err(OtpDaiErrIdx, OtpAccessError); + predict_rdata(is_secret(dai_addr) || is_digest(dai_addr), 0, 0); + end else if (sw_read_lock || + // Secret partitions cal digest can also lock read access. + // However, digest is always readable except SW partitions (Issue #5752). + (is_secret(dai_addr) && get_digest_reg_val(part_idx) != 0 && + !is_digest(dai_addr)) || + // If the partition has owner key material and lc_owner_seed_sw_rw is disable, + // then return access error. + (PartInfo[part_idx].iskeymgr_owner && !is_digest(dai_addr) && + cfg.otp_ctrl_vif.lc_owner_seed_sw_rw_en_i != lc_ctrl_pkg::On)) begin + predict_err(OtpDaiErrIdx, OtpAccessError); + predict_rdata(is_secret(dai_addr) || is_digest(dai_addr), 0, 0); + + end else begin + bit [TL_DW-1:0] read_out0, read_out1; + bit [TL_AW-1:0] otp_addr = get_scb_otp_addr(); + int ecc_err = 0; + + // Backdoor read to check if there is any ECC error. + if (part_has_integrity(part_idx)) begin + ecc_err = read_a_word_with_ecc(dai_addr, read_out0); + if (is_secret(dai_addr) || is_digest(dai_addr)) begin + ecc_err = max2(read_a_word_with_ecc(dai_addr + 4, read_out1), ecc_err); + end + end else begin + ecc_err = read_a_word_with_ecc_raw(dai_addr, read_out0); + if (is_secret(dai_addr) || is_digest(dai_addr)) begin + ecc_err = max2(read_a_word_with_ecc_raw(dai_addr + 4, read_out1), ecc_err); + end + end + + if (ecc_err == OtpEccCorrErr && part_has_integrity(part_idx)) begin + predict_err(OtpDaiErrIdx, OtpMacroEccCorrError); + backdoor_update_otp_array(dai_addr); + predict_rdata(is_secret(dai_addr) || is_digest(dai_addr), + otp_a[otp_addr], otp_a[otp_addr+1]); + end else if (ecc_err == OtpEccUncorrErr && part_has_integrity(part_idx)) begin + predict_err(OtpDaiErrIdx, OtpMacroEccUncorrError); + // Max wait 20 clock cycles because scb did not know when exactly OTP will + // finish reading and reporting the uncorrectable error. + set_exp_alert("fatal_macro_error", 1, 20); + predict_rdata(1, 0, 0); + // Some partitions do not interpret/report ECC errors. In those cases + // we still need to model the read data correctly if it has been perturbed. + end else if (ecc_err inside {OtpEccCorrErr, OtpEccUncorrErr} && + !part_has_integrity(part_idx)) begin + predict_no_err(OtpDaiErrIdx); + predict_rdata(is_secret(dai_addr) || is_digest(dai_addr), + read_out0, read_out1); + // do not check direct_access_rdata_* on ECC errors in + // non-integrity partitions + check_dai_rd_data = 0; + end else begin + predict_no_err(OtpDaiErrIdx); + predict_rdata(is_secret(dai_addr) || is_digest(dai_addr), + otp_a[otp_addr], otp_a[otp_addr+1]); + end + end + end + DaiWrite: begin + bit[TL_AW-1:0] otp_addr = get_scb_otp_addr(); + bit is_write_locked; + // check if write locked + if (part_has_digest(part_idx)) begin + is_write_locked = get_digest_reg_val(part_idx) != 0; + end else begin + is_write_locked = 0; + end + + if (is_write_locked || (PartInfo[part_idx].iskeymgr_creator && + !is_digest(dai_addr) && + cfg.otp_ctrl_vif.lc_creator_seed_sw_rw_en_i != lc_ctrl_pkg::On)) begin + predict_err(OtpDaiErrIdx, OtpAccessError); + end else if (is_write_locked || (PartInfo[part_idx].iskeymgr_owner && + !is_digest(dai_addr) && + cfg.otp_ctrl_vif.lc_owner_seed_sw_rw_en_i != lc_ctrl_pkg::On)) begin + predict_err(OtpDaiErrIdx, OtpAccessError); + end else begin + predict_no_err(OtpDaiErrIdx); + // write digest + if (is_sw_digest(dai_addr)) begin + bit [TL_DW*2-1:0] curr_digest, prev_digest; + curr_digest = {`gmv(ral.direct_access_wdata[1]), + `gmv(ral.direct_access_wdata[0])}; + prev_digest = {otp_a[otp_addr+1], otp_a[otp_addr]}; + dai_wr_ip = 1; + // allow bit write + if ((prev_digest & curr_digest) == prev_digest) begin + update_digest_to_otp(part_idx, curr_digest); + end else begin + predict_err(OtpDaiErrIdx, OtpMacroWriteBlankError); + end + end else if (is_digest(dai_addr)) begin + predict_err(OtpDaiErrIdx, OtpAccessError); + // write OTP memory + end else begin + dai_wr_ip = 1; + if (!is_secret(dai_addr)) begin + bit [TL_DW-1:0] wr_data = `gmv(ral.direct_access_wdata[0]); + // allow bit write + if ((otp_a[otp_addr] & wr_data) == otp_a[otp_addr]) begin + otp_a[otp_addr] = wr_data; + check_otp_idle(.val(0), .wait_clks(3)); + end else begin + predict_err(OtpDaiErrIdx, OtpMacroWriteBlankError); + end + end else begin + bit [SCRAMBLE_DATA_SIZE-1:0] secret_data = {otp_a[otp_addr + 1], + otp_a[otp_addr]}; + bit [SCRAMBLE_DATA_SIZE-1:0] wr_data = {`gmv(ral.direct_access_wdata[1]), + `gmv(ral.direct_access_wdata[0])}; + wr_data = scramble_data(wr_data, part_idx); + secret_data = scramble_data(secret_data, part_idx); + if ((secret_data & wr_data) == secret_data) begin + otp_a[otp_addr] = `gmv(ral.direct_access_wdata[0]); + otp_a[otp_addr + 1] = `gmv(ral.direct_access_wdata[1]); + // wait until secret scrambling is done + check_otp_idle(.val(0), .wait_clks(34)); + end else begin + predict_err(OtpDaiErrIdx, OtpMacroWriteBlankError); + end + end + end + end + end + default: begin + `uvm_fatal(`gfn, $sformatf("invalid cmd: %0d", item.a_data)) + end + endcase + // regwen is set to 0 only if the dai operation is successfully + if (`gmv(ral.intr_state.otp_error) == 0) void'(ral.direct_access_regwen.predict(0)); + end + end + end + "status": begin + if (addr_phase_read) begin + void'(ral.status.predict(.value(exp_status), .kind(UVM_PREDICT_READ))); + + // update status mask + status_mask = 0; + // Mask out check_pending field - we do not know how long it takes to process checks. + // Check failure can trigger all kinds of errors. + if (under_chk) status_mask = '1; + + // Mask out otp_dai access related field - we do not know how long it takes to finish + // DAI access. + if (under_dai_access) begin + status_mask[OtpDaiIdleIdx] = 1; + status_mask[OtpDaiErrIdx] = 1; + end + + // Mask out LCI error bit if lc_req is set. + if (cfg.otp_ctrl_vif.lc_prog_no_sta_check) status_mask[OtpLciErrIdx] = 1; + + end else if (data_phase_read) begin + if (cfg.en_cov) begin + cov.collect_status_cov(item.d_data); + if (cfg.otp_ctrl_vif.alert_reqs) begin + cov.csr_rd_after_alert_cg_wrap.sample(csr.get_offset()); + end + end + + if (item.d_data[OtpDaiIdleIdx]) begin + check_otp_idle(1); + dai_wr_ip = 0; + dai_digest_ip = LifeCycleIdx; + end + + // STATUS register check with mask + if (do_read_check) begin + `DV_CHECK_EQ((csr.get_mirrored_value() | status_mask), (item.d_data | status_mask), + $sformatf("reg name: status, compare_mask %0h", status_mask)) + end + + // Check if OtpCheckPending is set correctly, then ignore checking until check is done + if (under_chk) begin + if (item.d_data[OtpCheckPendingIdx] == 0) begin + exp_status[OtpCheckPendingIdx] = 0; + under_chk = 0; + end + end + + if (under_dai_access && !cfg.otp_ctrl_vif.under_error_states()) begin + if (item.d_data[OtpDaiIdleIdx]) begin + under_dai_access = 0; + void'(ral.direct_access_regwen.predict(direct_access_regwen_state)); + void'(ral.intr_state.otp_operation_done.predict(1)); + end + end + end + // checked in this block above + do_read_check = 0; + end + "check_trigger": begin + if (addr_phase_write && cfg.en_cov && cfg.otp_ctrl_vif.alert_reqs) begin + cov.issue_checks_after_alert_cg.sample(item.a_data); + end + + if (addr_phase_write && `gmv(ral.check_trigger_regwen) && item.a_data inside {[1:3]}) begin + bit [TL_DW-1:0] check_timeout = `gmv(ral.check_timeout) == 0 ? '1 : + `gmv(ral.check_timeout); + exp_status[OtpCheckPendingIdx] = 1; + under_chk = 1; + if (check_timeout <= CHK_TIMEOUT_CYC) begin + set_exp_alert("fatal_check_error", 1, `gmv(ral.check_timeout)); + predict_err(OtpTimeoutErrIdx); + end else begin + if (get_field_val(ral.check_trigger.consistency, item.a_data)) begin + foreach (cfg.ecc_chk_err[i]) begin + if (cfg.ecc_chk_err[i] == OtpEccCorrErr && part_has_integrity(i)) begin + predict_err(otp_status_e'(i), OtpMacroEccCorrError); + end else if (cfg.ecc_chk_err[i] == OtpEccUncorrErr && + part_has_integrity(i)) begin + set_exp_alert("fatal_macro_error", 1, 40_000); + predict_err(otp_status_e'(i), OtpMacroEccUncorrError); + end + end + end + end + end + end + "direct_access_regwen": begin + if (addr_phase_write) begin + // This locks the DAI until the next reset. + if (!item.a_data[0]) begin + direct_access_regwen_state = 0; + void'(ral.direct_access_regwen.predict(0)); + end + end + end + // For error codes, if lc_prog in progress, err_code might update anytime in DUT. Ignore + // checking until req is acknowledged. + + "err_code_0": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(0, item.d_data, access_part_idx); + end + end + "err_code_1": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(1, item.d_data, access_part_idx); + end + end + "err_code_2": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(2, item.d_data, access_part_idx); + end + end + "err_code_3": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(3, item.d_data, access_part_idx); + end + end + "err_code_4": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(4, item.d_data, access_part_idx); + end + end + "err_code_5": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(5, item.d_data, access_part_idx); + end + end + "err_code_6": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(6, item.d_data, access_part_idx); + end + end + "err_code_7": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(7, item.d_data, access_part_idx); + end + end + "err_code_8": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(8, item.d_data, access_part_idx); + end + end + "err_code_9": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(9, item.d_data, access_part_idx); + end + end + "err_code_10": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(10, item.d_data, access_part_idx); + end + end + "err_code_11": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(11, item.d_data, access_part_idx); + end + end + "err_code_12": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(12, item.d_data, access_part_idx); + end + end + "err_code_13": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(13, item.d_data, access_part_idx); + end + end + "err_code_14": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(14, item.d_data, access_part_idx); + end + end + "err_code_15": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(15, item.d_data, access_part_idx); + end + end + "err_code_16": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(16, item.d_data, access_part_idx); + end + end + "err_code_17": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(17, item.d_data, access_part_idx); + end + end + "err_code_18": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(18, item.d_data, access_part_idx); + end + end + "err_code_19": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(19, item.d_data, access_part_idx); + end + end + "err_code_20": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(20, item.d_data, access_part_idx); + end + end + "err_code_21": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(21, item.d_data, access_part_idx); + end + end + "err_code_22": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(22, item.d_data, access_part_idx); + end + end + "err_code_23": begin + if (cfg.m_lc_prog_pull_agent_cfg.vif.req) do_read_check = 0; + if (cfg.en_cov && do_read_check && data_phase_read) begin + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address) >> 2 << 2; + int access_part_idx = get_part_index(dai_addr); + cov.collect_err_code_cov(23, item.d_data, access_part_idx); + end + end + "vendor_test_digest_0", "vendor_test_digest_1", + "creator_sw_cfg_digest_0", "creator_sw_cfg_digest_1", + "owner_sw_cfg_digest_0", "owner_sw_cfg_digest_1", + "rot_creator_auth_digest_0", "rot_creator_auth_digest_1", + "rot_owner_auth_slot0_digest_0", "rot_owner_auth_slot0_digest_1", + "rot_owner_auth_slot1_digest_0", "rot_owner_auth_slot1_digest_1", + "plat_integ_auth_slot0_digest_0", "plat_integ_auth_slot0_digest_1", + "plat_integ_auth_slot1_digest_0", "plat_integ_auth_slot1_digest_1", + "plat_owner_auth_slot0_digest_0", "plat_owner_auth_slot0_digest_1", + "plat_owner_auth_slot1_digest_0", "plat_owner_auth_slot1_digest_1", + "plat_owner_auth_slot2_digest_0", "plat_owner_auth_slot2_digest_1", + "plat_owner_auth_slot3_digest_0", "plat_owner_auth_slot3_digest_1", + "rom_patch_digest_0", "rom_patch_digest_1", + "hw_cfg0_digest_0", "hw_cfg0_digest_1", + "hw_cfg1_digest_0", "hw_cfg1_digest_1", + "secret0_digest_0", "secret0_digest_1", + "secret1_digest_0", "secret1_digest_1", + "secret2_digest_0", "secret2_digest_1", + "secret3_digest_0", "secret3_digest_1": begin + if (ignore_digest_chk) do_read_check = 0; + end + "vendor_test_read_lock", + "creator_sw_cfg_read_lock", + "owner_sw_cfg_read_lock", + "ownership_slot_state_read_lock", + "rot_creator_auth_read_lock", + "rot_owner_auth_slot0_read_lock", + "rot_owner_auth_slot1_read_lock", + "plat_integ_auth_slot0_read_lock", + "plat_integ_auth_slot1_read_lock", + "plat_owner_auth_slot0_read_lock", + "plat_owner_auth_slot1_read_lock", + "plat_owner_auth_slot2_read_lock", + "plat_owner_auth_slot3_read_lock", + "ext_nvm_read_lock", + "rom_patch_read_lock", + "direct_access_wdata_0", + "direct_access_wdata_1", + "direct_access_address", + "check_regwen", + "check_trigger_regwen", + "check_trigger", + "check_timeout", + "intr_enable", + "integrity_check_period", + "consistency_check_period", + "alert_test": begin + // Do nothing + end + // DAI read data + "direct_access_rdata_0", "direct_access_rdata_1": do_read_check = check_dai_rd_data; + default: begin + `uvm_fatal(`gfn, $sformatf("invalid csr: %0s", csr.get_full_name())) + end + endcase + + // On reads, if do_read_check, is set, then check mirrored_value against item.d_data + if (data_phase_read) begin + if (do_read_check) begin + `DV_CHECK_EQ(csr.get_mirrored_value(), item.d_data, + $sformatf("reg name: %0s", csr.get_full_name())) + if (cfg.en_cov && cfg.otp_ctrl_vif.alert_reqs) begin + cov.csr_rd_after_alert_cg_wrap.sample(csr.get_offset()); + end + end + void'(csr.predict(.value(item.d_data), .kind(UVM_PREDICT_READ))); + end + endfunction + + // If reset or lc_escalate_en is issued during otp program, this function will backdoor update + // otp memory write value because scb did not know how many cells haven been written. + // We won't update csr `direct_access_address` after fatal alert happened, so in this function + // we can directly call method `get_scb_otp_addr` to get the interrupted dai address. + virtual function void recover_interrupted_op(); + if (dai_wr_ip) begin + bit [TL_DW-1:0] otp_addr = get_scb_otp_addr(); + bit [TL_DW-1:0] dai_addr = otp_addr << 2; + backdoor_update_otp_array(dai_addr); + dai_wr_ip = 0; + end + endfunction + + virtual function void backdoor_update_otp_array(bit [TL_DW-1:0] dai_addr); + bit [TL_DW-1:0] otp_addr = dai_addr >> 2; + bit [TL_DW-1:0] readout_word, readout_word1; + int part_idx = get_part_index(dai_addr); + if (part_has_integrity(part_idx)) begin + void'(read_a_word_with_ecc(dai_addr, readout_word)); + void'(read_a_word_with_ecc(dai_addr + 4, readout_word1)); + end else begin + void'(read_a_word_with_ecc_raw(dai_addr, readout_word)); + void'(read_a_word_with_ecc_raw(dai_addr + 4, readout_word1)); + end + + otp_a[otp_addr] = readout_word; + + if (is_digest(dai_addr)) begin + otp_a[otp_addr+1] = readout_word1; + end else if (is_secret(dai_addr)) begin + bit [TL_DW*2-1:0] mem_rd_val, descrambled_val; + mem_rd_val = {readout_word1 ,readout_word}; + descrambled_val = descramble_data(mem_rd_val, part_idx); + otp_a[otp_addr+1] = descrambled_val[TL_DW*2-1:TL_DW]; + otp_a[otp_addr] = descrambled_val[TL_DW-1:0]; + end + endfunction + + virtual function bit [1:0] read_a_word_with_ecc(bit [TL_DW-1:0] dai_addr, + ref bit [TL_DW-1:0] readout_word); + prim_secded_pkg::secded_22_16_t ecc_rd_data0 = cfg.mem_bkdr_util_h.ecc_read16(dai_addr); + prim_secded_pkg::secded_22_16_t ecc_rd_data1 = cfg.mem_bkdr_util_h.ecc_read16(dai_addr + 2); + readout_word[15:0] = ecc_rd_data0.data; + readout_word[31:16] = ecc_rd_data1.data; + return max2(ecc_rd_data0.err, ecc_rd_data1.err); + endfunction + + // Returns the ECC error but does not correct the data bits (i.e. returns the raw data). + virtual function bit [1:0] read_a_word_with_ecc_raw(bit [TL_DW-1:0] dai_addr, + ref bit [TL_DW-1:0] readout_word); + prim_secded_pkg::secded_22_16_t ecc_rd_data0 = cfg.mem_bkdr_util_h.ecc_read16(dai_addr); + prim_secded_pkg::secded_22_16_t ecc_rd_data1 = cfg.mem_bkdr_util_h.ecc_read16(dai_addr + 2); + readout_word[15:0] = 16'hFFFF & cfg.mem_bkdr_util_h.read(dai_addr); + readout_word[31:16] = 16'hFFFF & cfg.mem_bkdr_util_h.read(dai_addr + 2); + return max2(ecc_rd_data0.err, ecc_rd_data1.err); + endfunction + + + virtual function void reset(string kind = "HARD"); + recover_interrupted_op(); + super.reset(kind); + // flush fifos + otbn_fifo.flush(); + flash_addr_fifo.flush(); + flash_data_fifo.flush(); + lc_prog_fifo.flush(); + for (int i = 0; i < NumSramKeyReqSlots; i++) begin + sram_fifos[i].flush(); + end + + direct_access_regwen_state = 1; + under_chk = 0; + under_dai_access = 0; + ignore_digest_chk = 0; + exp_status = `gmv(ral.status); + exp_alert = OtpNoAlert; + + edn_data_q.delete(); + + // Out of reset: lock dai access until power init is done + if (cfg.en_scb) void'(ral.direct_access_regwen.predict(0)); + endfunction + + virtual function void check_otp_idle(bit val, int wait_clks = 0); + fork + begin + fork + begin + // use negedge to avoid race condition + cfg.clk_rst_vif.wait_n_clks(wait_clks + 1); + `uvm_error(`gfn, + $sformatf("pwr_otp_idle output is %0b while expect %0b within %0d cycles", + cfg.otp_ctrl_vif.pwr_otp_idle_o, val, wait_clks)) + end + begin + wait(cfg.under_reset || cfg.otp_ctrl_vif.pwr_otp_idle_o == val || + // Due to OTP access arbitration, any KDI request during DAI access might block + // write secret until KDI request is completed. Since the KDI process time could + // vary depends on the push-pull-agent, we are going to ignore the checking if + // this scenario happens. + cfg.m_otbn_pull_agent_cfg.vif.req || + cfg.m_flash_data_pull_agent_cfg.vif.req || + cfg.m_flash_addr_pull_agent_cfg.vif.req || + cfg.m_sram_pull_agent_cfg[0].vif.req || + cfg.m_sram_pull_agent_cfg[1].vif.req || + cfg.m_sram_pull_agent_cfg[2].vif.req || + cfg.m_sram_pull_agent_cfg[3].vif.req || + cfg.m_lc_prog_pull_agent_cfg.vif.req || + // When lc_escalation is on, the DAI interface goes to ErrorSt, so ignore + // otp_idle checking. + cfg.otp_ctrl_vif.alert_reqs || + // Check timeout will keep doing background check, issue #5616 + exp_status[OtpTimeoutErrIdx]); + end + join_any + disable fork; + end + join_none + endfunction + + // predict digest registers + virtual function void predict_digest_csrs(); + void'(ral.vendor_test_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[VendorTestIdx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.vendor_test_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[VendorTestIdx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.creator_sw_cfg_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[CreatorSwCfgIdx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.creator_sw_cfg_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[CreatorSwCfgIdx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.owner_sw_cfg_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[OwnerSwCfgIdx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.owner_sw_cfg_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[OwnerSwCfgIdx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.rot_creator_auth_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[RotCreatorAuthIdx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.rot_creator_auth_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[RotCreatorAuthIdx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.rot_owner_auth_slot0_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[RotOwnerAuthSlot0Idx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.rot_owner_auth_slot0_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[RotOwnerAuthSlot0Idx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.rot_owner_auth_slot1_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[RotOwnerAuthSlot1Idx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.rot_owner_auth_slot1_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[RotOwnerAuthSlot1Idx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.plat_integ_auth_slot0_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[PlatIntegAuthSlot0Idx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.plat_integ_auth_slot0_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[PlatIntegAuthSlot0Idx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.plat_integ_auth_slot1_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[PlatIntegAuthSlot1Idx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.plat_integ_auth_slot1_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[PlatIntegAuthSlot1Idx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.plat_owner_auth_slot0_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[PlatOwnerAuthSlot0Idx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.plat_owner_auth_slot0_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[PlatOwnerAuthSlot0Idx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.plat_owner_auth_slot1_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[PlatOwnerAuthSlot1Idx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.plat_owner_auth_slot1_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[PlatOwnerAuthSlot1Idx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.plat_owner_auth_slot2_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[PlatOwnerAuthSlot2Idx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.plat_owner_auth_slot2_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[PlatOwnerAuthSlot2Idx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.plat_owner_auth_slot3_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[PlatOwnerAuthSlot3Idx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.plat_owner_auth_slot3_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[PlatOwnerAuthSlot3Idx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.rom_patch_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[RomPatchIdx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.rom_patch_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[RomPatchIdx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.hw_cfg0_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[HwCfg0Idx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.hw_cfg0_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[HwCfg0Idx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.hw_cfg1_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[HwCfg1Idx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.hw_cfg1_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[HwCfg1Idx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.secret0_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[Secret0Idx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.secret0_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[Secret0Idx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.secret1_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[Secret1Idx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.secret1_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[Secret1Idx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.secret2_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[Secret2Idx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.secret2_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[Secret2Idx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + + void'(ral.secret3_digest[0].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[Secret3Idx]]), + .kind(UVM_PREDICT_DIRECT))); + void'(ral.secret3_digest[1].predict( + .value(otp_a[PART_OTP_DIGEST_ADDRS[Secret3Idx] + 1]), + .kind(UVM_PREDICT_DIRECT))); + endfunction + + function void update_digest_to_otp(int part_idx, bit [TL_DW*2-1:0] digest); + otp_a[PART_OTP_DIGEST_ADDRS[part_idx]] = digest[31:0]; + otp_a[PART_OTP_DIGEST_ADDRS[part_idx] + 1] = digest[63:32]; + endfunction + + function void check_phase(uvm_phase phase); + super.check_phase(phase); + // post test checks - ensure that all local fifos and queues are empty + endfunction + + // Calculate digest value for each partition + // According to the design spec, the calculation is based on 64-rounds of PRESENT cipher + // The 64-bit data_in state is initialized with a silicon creator constant, and each 128 bit + // chunk of partition data are fed in as keys + // The last 64-round PRESENT calculation will use a global digest constant as key input + function void cal_digest_val(int part_idx); + bit [TL_DW-1:0] mem_q[$]; + int array_size; + bit [SCRAMBLE_DATA_SIZE-1:0] digest; + + if (cfg.otp_ctrl_vif.under_error_states()) return; + + if (!part_has_hw_digest(part_idx) || get_digest_reg_val(part_idx) != 0) begin + predict_err(OtpDaiErrIdx, OtpAccessError); + return; + end else if (PartInfo[part_idx].iskeymgr_creator && + cfg.otp_ctrl_vif.lc_creator_seed_sw_rw_en_i != lc_ctrl_pkg::On) begin + predict_err(OtpDaiErrIdx, OtpAccessError); + return; + end else if (PartInfo[part_idx].iskeymgr_owner && + cfg.otp_ctrl_vif.lc_owner_seed_sw_rw_en_i != lc_ctrl_pkg::On) begin + predict_err(OtpDaiErrIdx, OtpAccessError); + return; + end else begin + predict_no_err(OtpDaiErrIdx); + dai_digest_ip = part_idx; + end + case (part_idx) + HwCfg0Idx: mem_q = otp_a[HwCfg0Offset / TL_SIZE : HwCfg0DigestOffset / TL_SIZE - 1]; + HwCfg1Idx: mem_q = otp_a[HwCfg1Offset / TL_SIZE : HwCfg1DigestOffset / TL_SIZE - 1]; + Secret0Idx: mem_q = otp_a[Secret0Offset / TL_SIZE : Secret0DigestOffset / TL_SIZE - 1]; + Secret1Idx: mem_q = otp_a[Secret1Offset / TL_SIZE : Secret1DigestOffset / TL_SIZE - 1]; + Secret2Idx: mem_q = otp_a[Secret2Offset / TL_SIZE : Secret2DigestOffset / TL_SIZE - 1]; + Secret3Idx: mem_q = otp_a[Secret3Offset / TL_SIZE : Secret3DigestOffset / TL_SIZE - 1]; + default: begin + `uvm_fatal(`gfn, $sformatf("Access unexpected partition %0d", part_idx)) + end + endcase + + array_size = mem_q.size(); + + // for secret partitions, need to use otp scrambled value as data input + if (PartInfo[part_idx].secret) begin + bit [TL_DW-1:0] scrambled_mem_q[$]; + for (int i = 0; i < array_size/2; i++) begin + bit [SCRAMBLE_DATA_SIZE-1:0] scrambled_data; + scrambled_data = scramble_data({mem_q[i*2+1], mem_q[i*2]}, part_idx); + scrambled_mem_q.push_back(scrambled_data[TL_DW-1:0]); + scrambled_mem_q.push_back(scrambled_data[SCRAMBLE_DATA_SIZE-1:TL_DW]); + end + mem_q = scrambled_mem_q; + end + + digest = otp_scrambler_pkg::cal_digest(part_idx, mem_q); + update_digest_to_otp(part_idx, digest); + endfunction + + + // this function go through present encode algo two or three iterations: + // first iteration with input key, + // second iteration with second_key, this iteration only happens if num_round is 2 + // third iteration with a final constant as key + // this is mainly used for unlock token hashing, key derivation + virtual function bit [SCRAMBLE_DATA_SIZE-1:0] present_encode_with_final_const( + bit [SCRAMBLE_DATA_SIZE-1:0] data, + bit [SCRAMBLE_KEY_SIZE-1:0] key, + bit [SCRAMBLE_KEY_SIZE-1:0] final_const, + bit [SCRAMBLE_KEY_SIZE-1:0] second_key = '0, + int num_round = 1); + bit [SCRAMBLE_DATA_SIZE-1:0] enc_data; + bit [SCRAMBLE_DATA_SIZE-1:0] intermediate_state; + crypto_dpi_present_pkg::sv_dpi_present_encrypt(data, key, + SCRAMBLE_KEY_SIZE, NUM_ROUND, enc_data); + // XOR the previous state into the digest result according to the Davies-Meyer scheme. + intermediate_state = data ^ enc_data; + + if (num_round == 2) begin + crypto_dpi_present_pkg::sv_dpi_present_encrypt(intermediate_state, second_key, + SCRAMBLE_KEY_SIZE, NUM_ROUND, enc_data); + intermediate_state = intermediate_state ^ enc_data; + end else if (num_round > 2) begin + `uvm_fatal(`gfn, $sformatf("does not support num_round: %0d > 2", num_round)) + end + + crypto_dpi_present_pkg::sv_dpi_present_encrypt(intermediate_state, final_const, + SCRAMBLE_KEY_SIZE, NUM_ROUND, enc_data); + // XOR the previous state into the digest result according to the Davies-Meyer scheme. + present_encode_with_final_const = intermediate_state ^ enc_data; + endfunction + + // Get address for scoreboard's otp_a array from the `direct_access_address` CSR + function bit [TL_DW-1:0] get_scb_otp_addr(); + bit [TL_DW-1:0] dai_addr = `gmv(ral.direct_access_address); + get_scb_otp_addr = normalize_dai_addr(dai_addr) >> 2; + endfunction + + // This function predict OTP error related registers: intr_state, status, and err_code + virtual function void predict_err(otp_status_e status_err_idx, + otp_err_code_e err_code = OtpNoError, + bit update_esc_err = 0); + if (cfg.otp_ctrl_vif.under_error_states() && !update_esc_err) return; + + // Update intr_state + void'(ral.intr_state.otp_error.predict(.value(1), .kind(UVM_PREDICT_READ))); + // Update status + exp_status[status_err_idx] = 1; + + // Only first status errors up to the LCI have corresponding err_code + if (status_err_idx <= OtpLciErrIdx) begin + dv_base_reg_field err_code_flds[$]; + if (err_code == OtpNoError) begin + `uvm_error(`gfn, $sformatf("please set status error: %0s error code", status_err_idx.name)) + end + ral.err_code[status_err_idx].get_dv_base_reg_fields(err_code_flds); + + if (`gmv(err_code_flds[0]) inside {OTP_TERMINAL_ERRS}) begin + `uvm_info(`gfn, "terminal error cannot be updated", UVM_HIGH) + end else if (status_err_idx == OtpLciErrIdx && + `gmv(err_code_flds[0]) != OtpNoError) begin + `uvm_info(`gfn, "For LC partition, all errors are terminal error!", UVM_HIGH) + end else begin + void'(err_code_flds[0].predict(.value(err_code), .kind(UVM_PREDICT_READ))); + end + end + + endfunction + + virtual function void predict_no_err(otp_status_e status_err_idx); + if (cfg.otp_ctrl_vif.under_error_states()) return; + + exp_status[status_err_idx] = 0; + if (status_err_idx == OtpDaiErrIdx) exp_status[OtpDaiIdleIdx] = 1; + + if (status_err_idx <= OtpLciErrIdx) begin + dv_base_reg_field err_code_flds[$]; + ral.err_code[status_err_idx].get_dv_base_reg_fields(err_code_flds); + void'(err_code_flds[0].predict(OtpNoError)); + end + endfunction + + virtual function void predict_rdata(bit is_64_bits, bit [TL_DW-1:0] rdata0, + bit [TL_DW-1:0] rdata1 = 0); + void'(ral.direct_access_rdata[0].predict(.value(rdata0), .kind(UVM_PREDICT_READ))); + if (is_64_bits) begin + void'(ral.direct_access_rdata[1].predict(.value(rdata1), .kind(UVM_PREDICT_READ))); + end + endfunction + + // this function retrieves keys (128 bits) from scb's otp_array with a starting address + // if not locked, it will return 0 + // this is mainly used for scrambling key algo + virtual function bit [SCRAMBLE_KEY_SIZE-1:0] get_key_from_otp(bit locked, int start_i); + bit [SCRAMBLE_KEY_SIZE-1:0] key; + if (!locked) return 0; + for (int i = 0; i < 4; i++) key |= otp_a[i + start_i] << (TL_DW * i); + return key; + endfunction + + // The following two methods are all retrieving digest val. + // get_otp_digest_val: is the digest value from OTP memory + // get_digest_reg_val: is the digest value in register. This value is identical to OTP + // memory's digest value after a power cycle reset. + virtual function bit [TL_DW*2-1:0] get_otp_digest_val(int part_idx); + get_otp_digest_val[31:0] = otp_a[PART_OTP_DIGEST_ADDRS[part_idx]]; + get_otp_digest_val[63:32] = otp_a[PART_OTP_DIGEST_ADDRS[part_idx] + 1]; + endfunction + + virtual function bit [TL_DW*2-1:0] get_digest_reg_val(int part_idx); + bit [TL_DW*2-1:0] digest; + case (part_idx) + VendorTestIdx: begin + digest = {`gmv(ral.vendor_test_digest[1]), + `gmv(ral.vendor_test_digest[0])}; + end + CreatorSwCfgIdx: begin + digest = {`gmv(ral.creator_sw_cfg_digest[1]), + `gmv(ral.creator_sw_cfg_digest[0])}; + end + OwnerSwCfgIdx: begin + digest = {`gmv(ral.owner_sw_cfg_digest[1]), + `gmv(ral.owner_sw_cfg_digest[0])}; + end + RotCreatorAuthIdx: begin + digest = {`gmv(ral.rot_creator_auth_digest[1]), + `gmv(ral.rot_creator_auth_digest[0])}; + end + RotOwnerAuthSlot0Idx: begin + digest = {`gmv(ral.rot_owner_auth_slot0_digest[1]), + `gmv(ral.rot_owner_auth_slot0_digest[0])}; + end + RotOwnerAuthSlot1Idx: begin + digest = {`gmv(ral.rot_owner_auth_slot1_digest[1]), + `gmv(ral.rot_owner_auth_slot1_digest[0])}; + end + PlatIntegAuthSlot0Idx: begin + digest = {`gmv(ral.plat_integ_auth_slot0_digest[1]), + `gmv(ral.plat_integ_auth_slot0_digest[0])}; + end + PlatIntegAuthSlot1Idx: begin + digest = {`gmv(ral.plat_integ_auth_slot1_digest[1]), + `gmv(ral.plat_integ_auth_slot1_digest[0])}; + end + PlatOwnerAuthSlot0Idx: begin + digest = {`gmv(ral.plat_owner_auth_slot0_digest[1]), + `gmv(ral.plat_owner_auth_slot0_digest[0])}; + end + PlatOwnerAuthSlot1Idx: begin + digest = {`gmv(ral.plat_owner_auth_slot1_digest[1]), + `gmv(ral.plat_owner_auth_slot1_digest[0])}; + end + PlatOwnerAuthSlot2Idx: begin + digest = {`gmv(ral.plat_owner_auth_slot2_digest[1]), + `gmv(ral.plat_owner_auth_slot2_digest[0])}; + end + PlatOwnerAuthSlot3Idx: begin + digest = {`gmv(ral.plat_owner_auth_slot3_digest[1]), + `gmv(ral.plat_owner_auth_slot3_digest[0])}; + end + RomPatchIdx: begin + digest = {`gmv(ral.rom_patch_digest[1]), + `gmv(ral.rom_patch_digest[0])}; + end + HwCfg0Idx: begin + digest = {`gmv(ral.hw_cfg0_digest[1]), + `gmv(ral.hw_cfg0_digest[0])}; + end + HwCfg1Idx: begin + digest = {`gmv(ral.hw_cfg1_digest[1]), + `gmv(ral.hw_cfg1_digest[0])}; + end + Secret0Idx: begin + digest = {`gmv(ral.secret0_digest[1]), + `gmv(ral.secret0_digest[0])}; + end + Secret1Idx: begin + digest = {`gmv(ral.secret1_digest[1]), + `gmv(ral.secret1_digest[0])}; + end + Secret2Idx: begin + digest = {`gmv(ral.secret2_digest[1]), + `gmv(ral.secret2_digest[0])}; + end + Secret3Idx: begin + digest = {`gmv(ral.secret3_digest[1]), + `gmv(ral.secret3_digest[0])}; + end + default: `uvm_fatal(`gfn, $sformatf("Partition %0d does not have digest", part_idx)) + endcase + return digest; + endfunction + + virtual function bit is_tl_mem_access_allowed(input tl_seq_item item, input string ral_name, + output bit mem_byte_access_err, + output bit mem_wo_err, + output bit mem_ro_err, + output bit custom_err); + + uvm_reg_addr_t addr = cfg.ral_models[ral_name].get_word_aligned_addr(item.a_addr); + uvm_reg_addr_t csr_addr = cfg.ral_models[ral_name].get_word_aligned_addr(item.a_addr); + bit [TL_AW-1:0] addr_mask = ral.get_addr_mask(); + bit [TL_AW-1:0] dai_addr = (csr_addr & addr_mask - SW_WINDOW_BASE_ADDR); + + bit mem_access_allowed = super.is_tl_mem_access_allowed(item, ral_name, mem_byte_access_err, + mem_wo_err, mem_ro_err, custom_err); + + if (ral_name == "otp_ctrl_prim_reg_block") return mem_access_allowed; + + // Ensure the address is within the memory window range. + // Also will skip checking if memory access is not allowed due to TLUL bus error. + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr : + cfg.ral_models[ral_name].mem_ranges[0].end_addr]} && + mem_access_allowed) begin + + // If sw partition is read locked, then access policy changes from RO to no access + if (`gmv(ral.vendor_test_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + VendorTestOffset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + VendorTestOffset + + VendorTestSize - 1]}) begin + predict_err(OtpVendorTestErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + cov.unbuf_access_lock_cg_wrap[VendorTestIdx].sample(.read_lock(1), + .write_lock(get_digest_reg_val(VendorTestIdx) != 0), .is_write(0)); + end + return 0; + end + end + if (`gmv(ral.creator_sw_cfg_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + CreatorSwCfgOffset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + CreatorSwCfgOffset + + CreatorSwCfgSize - 1]}) begin + predict_err(OtpCreatorSwCfgErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + cov.unbuf_access_lock_cg_wrap[CreatorSwCfgIdx].sample(.read_lock(1), + .write_lock(get_digest_reg_val(CreatorSwCfgIdx) != 0), .is_write(0)); + end + return 0; + end + end + if (`gmv(ral.owner_sw_cfg_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + OwnerSwCfgOffset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + OwnerSwCfgOffset + + OwnerSwCfgSize - 1]}) begin + predict_err(OtpOwnerSwCfgErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + cov.unbuf_access_lock_cg_wrap[OwnerSwCfgIdx].sample(.read_lock(1), + .write_lock(get_digest_reg_val(OwnerSwCfgIdx) != 0), .is_write(0)); + end + return 0; + end + end + if (`gmv(ral.ownership_slot_state_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + OwnershipSlotStateOffset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + OwnershipSlotStateOffset + + OwnershipSlotStateSize - 1]}) begin + predict_err(OtpOwnershipSlotStateErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + // TODO: we should probably create a different covergroup + // for unbuffered partitions without digest. + cov.unbuf_access_lock_cg_wrap[OwnershipSlotStateIdx].sample(.read_lock(1), + .write_lock(0), .is_write(0)); + end + return 0; + end + end + if (`gmv(ral.rot_creator_auth_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + RotCreatorAuthOffset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + RotCreatorAuthOffset + + RotCreatorAuthSize - 1]}) begin + predict_err(OtpRotCreatorAuthErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + cov.unbuf_access_lock_cg_wrap[RotCreatorAuthIdx].sample(.read_lock(1), + .write_lock(get_digest_reg_val(RotCreatorAuthIdx) != 0), .is_write(0)); + end + return 0; + end + end + if (`gmv(ral.rot_owner_auth_slot0_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + RotOwnerAuthSlot0Offset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + RotOwnerAuthSlot0Offset + + RotOwnerAuthSlot0Size - 1]}) begin + predict_err(OtpRotOwnerAuthSlot0ErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + cov.unbuf_access_lock_cg_wrap[RotOwnerAuthSlot0Idx].sample(.read_lock(1), + .write_lock(get_digest_reg_val(RotOwnerAuthSlot0Idx) != 0), .is_write(0)); + end + return 0; + end + end + if (`gmv(ral.rot_owner_auth_slot1_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + RotOwnerAuthSlot1Offset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + RotOwnerAuthSlot1Offset + + RotOwnerAuthSlot1Size - 1]}) begin + predict_err(OtpRotOwnerAuthSlot1ErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + cov.unbuf_access_lock_cg_wrap[RotOwnerAuthSlot1Idx].sample(.read_lock(1), + .write_lock(get_digest_reg_val(RotOwnerAuthSlot1Idx) != 0), .is_write(0)); + end + return 0; + end + end + if (`gmv(ral.plat_integ_auth_slot0_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + PlatIntegAuthSlot0Offset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + PlatIntegAuthSlot0Offset + + PlatIntegAuthSlot0Size - 1]}) begin + predict_err(OtpPlatIntegAuthSlot0ErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + cov.unbuf_access_lock_cg_wrap[PlatIntegAuthSlot0Idx].sample(.read_lock(1), + .write_lock(get_digest_reg_val(PlatIntegAuthSlot0Idx) != 0), .is_write(0)); + end + return 0; + end + end + if (`gmv(ral.plat_integ_auth_slot1_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + PlatIntegAuthSlot1Offset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + PlatIntegAuthSlot1Offset + + PlatIntegAuthSlot1Size - 1]}) begin + predict_err(OtpPlatIntegAuthSlot1ErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + cov.unbuf_access_lock_cg_wrap[PlatIntegAuthSlot1Idx].sample(.read_lock(1), + .write_lock(get_digest_reg_val(PlatIntegAuthSlot1Idx) != 0), .is_write(0)); + end + return 0; + end + end + if (`gmv(ral.plat_owner_auth_slot0_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + PlatOwnerAuthSlot0Offset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + PlatOwnerAuthSlot0Offset + + PlatOwnerAuthSlot0Size - 1]}) begin + predict_err(OtpPlatOwnerAuthSlot0ErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + cov.unbuf_access_lock_cg_wrap[PlatOwnerAuthSlot0Idx].sample(.read_lock(1), + .write_lock(get_digest_reg_val(PlatOwnerAuthSlot0Idx) != 0), .is_write(0)); + end + return 0; + end + end + if (`gmv(ral.plat_owner_auth_slot1_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + PlatOwnerAuthSlot1Offset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + PlatOwnerAuthSlot1Offset + + PlatOwnerAuthSlot1Size - 1]}) begin + predict_err(OtpPlatOwnerAuthSlot1ErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + cov.unbuf_access_lock_cg_wrap[PlatOwnerAuthSlot1Idx].sample(.read_lock(1), + .write_lock(get_digest_reg_val(PlatOwnerAuthSlot1Idx) != 0), .is_write(0)); + end + return 0; + end + end + if (`gmv(ral.plat_owner_auth_slot2_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + PlatOwnerAuthSlot2Offset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + PlatOwnerAuthSlot2Offset + + PlatOwnerAuthSlot2Size - 1]}) begin + predict_err(OtpPlatOwnerAuthSlot2ErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + cov.unbuf_access_lock_cg_wrap[PlatOwnerAuthSlot2Idx].sample(.read_lock(1), + .write_lock(get_digest_reg_val(PlatOwnerAuthSlot2Idx) != 0), .is_write(0)); + end + return 0; + end + end + if (`gmv(ral.plat_owner_auth_slot3_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + PlatOwnerAuthSlot3Offset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + PlatOwnerAuthSlot3Offset + + PlatOwnerAuthSlot3Size - 1]}) begin + predict_err(OtpPlatOwnerAuthSlot3ErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + cov.unbuf_access_lock_cg_wrap[PlatOwnerAuthSlot3Idx].sample(.read_lock(1), + .write_lock(get_digest_reg_val(PlatOwnerAuthSlot3Idx) != 0), .is_write(0)); + end + return 0; + end + end + if (`gmv(ral.ext_nvm_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + ExtNvmOffset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + ExtNvmOffset + + ExtNvmSize - 1]}) begin + predict_err(OtpExtNvmErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + // TODO: we should probably create a different covergroup + // for unbuffered partitions without digest. + cov.unbuf_access_lock_cg_wrap[ExtNvmIdx].sample(.read_lock(1), + .write_lock(0), .is_write(0)); + end + return 0; + end + end + if (`gmv(ral.rom_patch_read_lock) == 0 || + cfg.otp_ctrl_vif.under_error_states()) begin + if (addr inside { + [cfg.ral_models[ral_name].mem_ranges[0].start_addr + RomPatchOffset : + cfg.ral_models[ral_name].mem_ranges[0].start_addr + RomPatchOffset + + RomPatchSize - 1]}) begin + predict_err(OtpRomPatchErrIdx, OtpAccessError); + custom_err = 1; + if (cfg.en_cov) begin + cov.unbuf_access_lock_cg_wrap[RomPatchIdx].sample(.read_lock(1), + .write_lock(get_digest_reg_val(RomPatchIdx) != 0), .is_write(0)); + end + return 0; + end + end + + // Check ECC uncorrectable fatal error. + if (dai_addr < LifeCycleOffset) begin + int part_idx = get_part_index(dai_addr); + bit [TL_DW-1:0] read_out; + int ecc_err = read_a_word_with_ecc(dai_addr, read_out); + if (ecc_err == OtpEccUncorrErr && part_has_integrity(part_idx)) begin + predict_err(otp_status_e'(part_idx), OtpMacroEccUncorrError); + set_exp_alert("fatal_macro_error", 1, 20); + custom_err = 1; + return 0; + end + end + end + + return mem_access_allowed; + endfunction + + virtual function bit predict_tl_err(tl_seq_item item, tl_channels_e channel, string ral_name); + if (ral_name == "otp_ctrl_prim_reg_block" && + cfg.otp_ctrl_vif.lc_dft_en_i != lc_ctrl_pkg::On) begin + if (channel == DataChannel) begin + `DV_CHECK_EQ(item.d_error, 1, + $sformatf({"On interface %0s, TL item: %0s, access gated by lc_dft_en_i"}, + ral_name, item.sprint(uvm_default_line_printer))) + + // In data read phase, check d_data when d_error = 1. + if (item.d_error && (item.d_opcode == tlul_pkg::AccessAckData)) begin + check_tl_read_value_after_error(item, ral_name); + end + end + return 1; + end + return super.predict_tl_err(item, channel, ral_name); + endfunction + + virtual function void set_exp_alert(string alert_name, bit is_fatal = 0, int max_delay = 0); + exp_alert = alert_name == "fatal_check_error" ? OtpCheckAlert : OtpMacroAlert; + super.set_exp_alert(alert_name, is_fatal, max_delay); + endfunction + +endclass diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_virtual_sequencer.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_virtual_sequencer.sv new file mode 100644 index 00000000000000..4780a806599403 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/otp_ctrl_virtual_sequencer.sv @@ -0,0 +1,19 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +class otp_ctrl_virtual_sequencer extends cip_base_virtual_sequencer #( + .CFG_T(otp_ctrl_env_cfg), + .COV_T(otp_ctrl_env_cov) + ); + `uvm_component_utils(otp_ctrl_virtual_sequencer) + + `uvm_component_new + + push_pull_sequencer#(.DeviceDataWidth(SRAM_DATA_SIZE)) sram_pull_sequencer_h[NumSramKeyReqSlots]; + push_pull_sequencer#(.DeviceDataWidth(OTBN_DATA_SIZE)) otbn_pull_sequencer_h; + push_pull_sequencer#(.DeviceDataWidth(FLASH_DATA_SIZE)) flash_data_pull_sequencer_h; + push_pull_sequencer#(.DeviceDataWidth(FLASH_DATA_SIZE)) flash_addr_pull_sequencer_h; + push_pull_sequencer#(.DeviceDataWidth(1), .HostDataWidth(LC_PROG_DATA_SIZE)) + lc_prog_pull_sequencer_h; +endclass diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_background_chks_vseq.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_background_chks_vseq.sv new file mode 100644 index 00000000000000..d316e1a12a5218 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_background_chks_vseq.sv @@ -0,0 +1,73 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// This simple sequence checks if the background check can be triggered once the period is set. + +class otp_ctrl_background_chks_vseq extends otp_ctrl_dai_lock_vseq; + `uvm_object_utils(otp_ctrl_background_chks_vseq) + + `uvm_object_new + + rand bit [1:0] trigger_chks; + rand uint check_period; + + constraint regwens_c {check_regwen_val == 1;} + + // At least one check will be triggered + constraint check_triggers_c {trigger_chks > 0;} + + constraint check_period_c { + check_period < 20; + check_period > 0; + } + + task body(); + int check_wait_cycles; + super.body(); + + // For stress_all_with_rand_reset test, if previous lc_esc_en is not cleared, then skip the + // background check. + if (cfg.otp_ctrl_vif.lc_esc_on == 0) begin + + // Write background check + if (trigger_chks[0]) csr_wr(ral.integrity_check_period, check_period); + if (trigger_chks[1]) csr_wr(ral.consistency_check_period, check_period); + `uvm_info(`gfn, $sformatf("trigger background check %0h", trigger_chks), UVM_LOW) + + cfg.en_scb = 0; + // According to spec, check period will append an 'hFF from the LSF. Add 10 cycle buffers for + // register updates. + check_wait_cycles = (check_period + 1) << 8 + 10; + + // Wait for first check done + repeat($countones(trigger_chks)) begin + csr_spinwait(.ptr(ral.status.check_pending), .exp_data(1), + .timeout_ns(cfg.clk_rst_vif.clk_period_ps / 1000 * check_wait_cycles)); + + csr_spinwait(.ptr(ral.status.check_pending), .exp_data(0)); + end + + // Configure timeout settings to trigger check error + csr_wr(ral.check_timeout, $urandom_range(1, 5)); + `uvm_info(`gfn, "trigger check timeout error", UVM_LOW) + + // Wait for fatal alert + `DV_SPINWAIT_EXIT( + wait(cfg.m_alert_agent_cfgs["fatal_check_error"].vif.alert_tx_final.alert_p);, + cfg.clk_rst_vif.wait_clks(check_wait_cycles);, + $sformatf("Timeout waiting for alert %0s", "fatal_check_error")) + check_fatal_alert_nonblocking("fatal_check_error"); + + cfg.clk_rst_vif.wait_clks($urandom_range(50, 1000)); + csr_rd_check(.ptr(ral.status.timeout_error), .compare_value(1)); + end + endtask + + // Enable scoreboard is done in stress_all sequence and `apply_resets_concurrently` task to + // avoid otp_ctrl_scoreboard reporting failures when reset has not been issued. + virtual task post_start(); + expect_fatal_alerts = 1; + super.post_start(); + endtask +endclass diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_base_vseq.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_base_vseq.sv new file mode 100644 index 00000000000000..673e1a6443c3a5 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_base_vseq.sv @@ -0,0 +1,842 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +class otp_ctrl_base_vseq extends cip_base_vseq #( + .RAL_T (otp_ctrl_core_reg_block), + .CFG_T (otp_ctrl_env_cfg), + .COV_T (otp_ctrl_env_cov), + .VIRTUAL_SEQUENCER_T (otp_ctrl_virtual_sequencer) + ); + `uvm_object_utils(otp_ctrl_base_vseq) + `uvm_object_new + + // various knobs to enable certain routines + bit do_otp_ctrl_init = 1'b1; + bit do_otp_pwr_init = 1'b1; + + // To only write unused OTP address, sequence will collect all the written addresses to an + // associative array to avoid `write_blank_addr_error`. + bit write_unused_addr = 1; + static bit used_dai_addrs[bit [OTP_ADDR_WIDTH - 1 : 0]]; + + rand bit [NumOtpCtrlIntr-1:0] en_intr; + + rand int apply_reset_during_pwr_init_cycles; + + bit is_valid_dai_op = 1; + + // According to spec, the period between digest calculation and reset should not issue any write. + bit [NumPart-2:0] digest_calculated; + + // For stress_all_with_rand reset sequence to issue reset during OTP operations. + bit do_digest_cal, do_otp_rd, do_otp_wr; + + // LC program request will use a separate variable to automatically set to non-blocking setting + // when LC error bit is set. + bit default_req_blocking = 1; + bit lc_prog_blocking = 1; + bit dai_wr_inprogress = 0; + uint32_t op_done_spinwait_timeout_ns = 20_000_000; + + // Collect current lc_state and lc_cnt. This is used to create next lc_state and lc_cnt without + // error. + lc_ctrl_state_pkg::lc_state_e lc_state; + lc_ctrl_state_pkg::lc_cnt_e lc_cnt; + + otp_ctrl_callback_vseq callback_vseq; + + constraint apply_reset_during_pwr_init_cycles_c { + apply_reset_during_pwr_init_cycles == 0; + } + + virtual task pre_start(); + `uvm_create_on(callback_vseq, p_sequencer); + super.pre_start(); + endtask + + virtual task dut_init(string reset_kind = "HARD"); + // OTP has dut and edn reset. If assign OTP values after `super.dut_init()`, and if dut reset + // deasserts earlier than edn reset, some OTP outputs might remain X or Z when dut clock is + // running. + otp_ctrl_vif_init(); + super.dut_init(reset_kind); + callback_vseq.dut_init_callback(); + + cfg.backdoor_clear_mem = 0; + // reset power init pin and lc pins + if (do_otp_ctrl_init && do_apply_reset) otp_ctrl_init(); + cfg.clk_rst_vif.wait_clks($urandom_range(0, 10)); + if (do_otp_pwr_init && do_apply_reset) otp_pwr_init(); + callback_vseq.post_otp_pwr_init(); + endtask + + // Cfg errors are cleared after reset + virtual task apply_reset(string kind = "HARD"); + super.apply_reset(kind); + cfg.otp_ctrl_vif.release_part_access_mubi(); + clear_seq_flags(); + endtask + + virtual function void clear_seq_flags(); + do_digest_cal = 0; + do_otp_rd = 0; + do_otp_wr = 0; + endfunction + + virtual task otp_ctrl_vif_init(); + cfg.otp_ctrl_vif.drive_lc_creator_seed_sw_rw_en(lc_ctrl_pkg::On); + cfg.otp_ctrl_vif.drive_lc_owner_seed_sw_rw_en(lc_ctrl_pkg::On); + cfg.otp_ctrl_vif.drive_lc_seed_hw_rd_en(get_rand_lc_tx_val()); + cfg.otp_ctrl_vif.drive_lc_dft_en(get_rand_lc_tx_val(.t_weight(0))); + cfg.otp_ctrl_vif.drive_lc_escalate_en(lc_ctrl_pkg::Off); + cfg.otp_ctrl_vif.drive_pwr_otp_init(0); + cfg.otp_ctrl_vif.drive_ext_voltage_h_io(1'bz); + + // Unused signals in open sourced OTP memory + `DV_CHECK_RANDOMIZE_FATAL(cfg.dut_cfg) + cfg.otp_ctrl_vif.otp_ast_pwr_seq_h_i = cfg.dut_cfg.otp_ast_pwr_seq_h; + cfg.otp_ctrl_vif.scan_en_i = cfg.dut_cfg.scan_en; + cfg.otp_ctrl_vif.scan_rst_ni = cfg.dut_cfg.scan_rst_n; + cfg.otp_ctrl_vif.scanmode_i = cfg.dut_cfg.scanmode; + cfg.otp_ctrl_vif.otp_vendor_test_ctrl_i = cfg.dut_cfg.otp_vendor_test_ctrl; + endtask + + // drive otp_pwr req pin to initialize OTP, and wait until init is done + virtual task otp_pwr_init(); + cfg.otp_ctrl_vif.drive_pwr_otp_init(1); + if (apply_reset_during_pwr_init_cycles > 0) begin + `DV_SPINWAIT_EXIT( + cfg.clk_rst_vif.wait_clks(apply_reset_during_pwr_init_cycles);, + wait (cfg.otp_ctrl_vif.pwr_otp_done_o == 1);) + if (cfg.otp_ctrl_vif.pwr_otp_done_o == 0) begin + cfg.otp_ctrl_vif.drive_pwr_otp_init(0); + apply_reset(); + cfg.otp_ctrl_vif.drive_pwr_otp_init(1); + end + end + wait (cfg.otp_ctrl_vif.pwr_otp_done_o == 1); + cfg.otp_ctrl_vif.drive_pwr_otp_init(0); + digest_calculated = 0; + endtask + + // setup basic otp_ctrl features + virtual task otp_ctrl_init(); + // reset memory to avoid readout X + clear_otp_memory(); + lc_state = lc_state_e'(0); + lc_cnt = lc_cnt_e'(0); + endtask + + virtual function void clear_otp_memory(); + cfg.mem_bkdr_util_h.clear_mem(); + cfg.backdoor_clear_mem = 1; + used_dai_addrs.delete(); + endfunction + + // Overide this task for otp_ctrl_common_vseq and otp_ctrl_stress_all_with_rand_reset_vseq + // because some registers won't set to default value until otp_init is done. + virtual task read_and_check_all_csrs_after_reset(); + cfg.otp_ctrl_vif.drive_lc_escalate_en(lc_ctrl_pkg::Off); + otp_pwr_init(); + super.read_and_check_all_csrs_after_reset(); + endtask + + // this task triggers an OTP write sequence via the DAI interface + virtual task dai_wr(bit [TL_DW-1:0] addr, + bit [TL_DW-1:0] wdata0, + bit [TL_DW-1:0] wdata1 = 0); + bit [TL_DW-1:0] val; + dai_wr_inprogress = 1; + if (write_unused_addr) begin + if (used_dai_addrs.exists(addr[OTP_ADDR_WIDTH - 1 : 0])) begin + `uvm_info(`gfn, $sformatf("addr %0h is already written!", addr), UVM_MEDIUM) + dai_wr_inprogress = 0; + return; + end else begin + used_dai_addrs[addr] = 1; + end + end + addr = randomize_dai_addr(addr); + `uvm_info(`gfn, $sformatf("dai write addr %0h, data %0h", addr, wdata0), UVM_HIGH) + csr_wr(ral.direct_access_address, addr); + csr_wr(ral.direct_access_wdata[0], wdata0); + if (is_secret(addr) || is_sw_digest(addr)) csr_wr(ral.direct_access_wdata[1], wdata1); + + do_otp_wr = 1; + csr_wr(ral.direct_access_cmd, int'(otp_ctrl_top_specific_pkg::DaiWrite)); + `uvm_info(`gfn, $sformatf("DAI write, address %0h, data0 %0h data1 %0h, is_secret = %0b", + addr, wdata0, wdata1, is_secret(addr)), UVM_DEBUG) + + // Direct_access_regwen and dai_idle are checked only when following conditions are met: + // - the dai operation is valid, otherwise it is hard to predict which cycle the error is + // detected + // - zero delays in TLUL interface, otherwise dai operation might be finished before reading + // these two CSRs + if (cfg.zero_delays && is_valid_dai_op && + cfg.otp_ctrl_vif.lc_escalate_en_i == lc_ctrl_pkg::Off) begin + csr_rd_check(ral.status.dai_idle, .compare_value(0), .backdoor(1)); + if ($urandom_range(0, 1)) csr_rd(.ptr(ral.direct_access_regwen), .value(val)); + end + wait_dai_op_done(); + rd_and_clear_intrs(); + dai_wr_inprogress = 0; + endtask : dai_wr + + // This task triggers an OTP readout sequence via the DAI interface + virtual task dai_rd(input bit [TL_DW-1:0] addr, + output bit [TL_DW-1:0] rdata0, + output bit [TL_DW-1:0] rdata1); + bit [TL_DW-1:0] val; + addr = randomize_dai_addr(addr); + + csr_wr(ral.direct_access_address, addr); + do_otp_rd = 1; + csr_wr(ral.direct_access_cmd, int'(otp_ctrl_top_specific_pkg::DaiRead)); + + if (cfg.zero_delays && is_valid_dai_op && + cfg.otp_ctrl_vif.lc_escalate_en_i == lc_ctrl_pkg::Off) begin + csr_rd_check(ral.status.dai_idle, .compare_value(0), .backdoor(1)); + if ($urandom_range(0, 1)) csr_rd(.ptr(ral.direct_access_regwen), .value(val)); + end + + wait_dai_op_done(); + csr_rd(ral.direct_access_rdata[0], rdata0); + if (is_secret(addr) || is_digest(addr)) csr_rd(ral.direct_access_rdata[1], rdata1); + rd_and_clear_intrs(); + endtask : dai_rd + + virtual task dai_rd_check(bit [TL_DW-1:0] addr, + bit [TL_DW-1:0] exp_data0, + bit [TL_DW-1:0] exp_data1 = 0); + bit [TL_DW-1:0] rdata0, rdata1; + dai_rd(addr, rdata0, rdata1); + if (!cfg.under_reset) begin + `DV_CHECK_EQ(rdata0, exp_data0, $sformatf("dai addr %0h rdata0 readout mismatch", addr)) + if (is_secret(addr) || is_digest(addr)) begin + `DV_CHECK_EQ(rdata1, exp_data1, $sformatf("dai addr %0h rdata1 readout mismatch", addr)) + end + end + endtask: dai_rd_check + + // this task exercises an OTP digest calculation via the DAI interface + virtual task cal_digest(int part_idx); + bit [TL_DW-1:0] val; + csr_wr(ral.direct_access_address, PART_BASE_ADDRS[part_idx]); + csr_wr(ral.direct_access_cmd, otp_ctrl_top_specific_pkg::DaiDigest); + + if (cfg.zero_delays && is_valid_dai_op && + cfg.otp_ctrl_vif.lc_escalate_en_i == lc_ctrl_pkg::Off) begin + csr_rd_check(ral.status.dai_idle, .compare_value(0), .backdoor(1)); + if ($urandom_range(0, 1)) csr_rd(.ptr(ral.direct_access_regwen), .value(val)); + end + do_digest_cal = 1; + wait_dai_op_done(); + digest_calculated[part_idx] = 1; + rd_and_clear_intrs(); + endtask + + // this task provisions all HW partitions + // SW partitions could not be provisioned via DAI interface + // LC partitions cannot be locked + virtual task cal_hw_digests(bit [NumPart-1:0] trigger_digest = $urandom()); + foreach (PartInfo[i]) begin + if (PartInfo[i].hw_digest && trigger_digest[i]) begin + cal_digest(i); + end + end + endtask + + // SW digest data are calculated in sw and won't be checked in OTP. + // Here to simplify testbench, write random data to sw digest. + virtual task write_sw_digests(bit [NumPartUnbuf-1:0] wr_digest = $urandom()); + bit [TL_DW*2-1:0] wdata; + if (wr_digest[VendorTestIdx]) begin + `DV_CHECK_STD_RANDOMIZE_FATAL(wdata); + dai_wr(VendorTestDigestOffset, wdata[TL_DW-1:0], wdata[TL_DW*2-1:TL_DW]); + end + if (wr_digest[CreatorSwCfgIdx]) begin + `DV_CHECK_STD_RANDOMIZE_FATAL(wdata); + dai_wr(CreatorSwCfgDigestOffset, wdata[TL_DW-1:0], wdata[TL_DW*2-1:TL_DW]); + end + if (wr_digest[OwnerSwCfgIdx]) begin + `DV_CHECK_STD_RANDOMIZE_FATAL(wdata); + dai_wr(OwnerSwCfgDigestOffset, wdata[TL_DW-1:0], wdata[TL_DW*2-1:TL_DW]); + end + if (wr_digest[RotCreatorAuthIdx]) begin + `DV_CHECK_STD_RANDOMIZE_FATAL(wdata); + dai_wr(RotCreatorAuthDigestOffset, wdata[TL_DW-1:0], wdata[TL_DW*2-1:TL_DW]); + end + if (wr_digest[RotOwnerAuthSlot0Idx]) begin + `DV_CHECK_STD_RANDOMIZE_FATAL(wdata); + dai_wr(RotOwnerAuthSlot0DigestOffset, wdata[TL_DW-1:0], wdata[TL_DW*2-1:TL_DW]); + end + if (wr_digest[RotOwnerAuthSlot1Idx]) begin + `DV_CHECK_STD_RANDOMIZE_FATAL(wdata); + dai_wr(RotOwnerAuthSlot1DigestOffset, wdata[TL_DW-1:0], wdata[TL_DW*2-1:TL_DW]); + end + if (wr_digest[PlatIntegAuthSlot0Idx]) begin + `DV_CHECK_STD_RANDOMIZE_FATAL(wdata); + dai_wr(PlatIntegAuthSlot0DigestOffset, wdata[TL_DW-1:0], wdata[TL_DW*2-1:TL_DW]); + end + if (wr_digest[PlatIntegAuthSlot1Idx]) begin + `DV_CHECK_STD_RANDOMIZE_FATAL(wdata); + dai_wr(PlatIntegAuthSlot1DigestOffset, wdata[TL_DW-1:0], wdata[TL_DW*2-1:TL_DW]); + end + if (wr_digest[PlatOwnerAuthSlot0Idx]) begin + `DV_CHECK_STD_RANDOMIZE_FATAL(wdata); + dai_wr(PlatOwnerAuthSlot0DigestOffset, wdata[TL_DW-1:0], wdata[TL_DW*2-1:TL_DW]); + end + if (wr_digest[PlatOwnerAuthSlot1Idx]) begin + `DV_CHECK_STD_RANDOMIZE_FATAL(wdata); + dai_wr(PlatOwnerAuthSlot1DigestOffset, wdata[TL_DW-1:0], wdata[TL_DW*2-1:TL_DW]); + end + if (wr_digest[PlatOwnerAuthSlot2Idx]) begin + `DV_CHECK_STD_RANDOMIZE_FATAL(wdata); + dai_wr(PlatOwnerAuthSlot2DigestOffset, wdata[TL_DW-1:0], wdata[TL_DW*2-1:TL_DW]); + end + if (wr_digest[PlatOwnerAuthSlot3Idx]) begin + `DV_CHECK_STD_RANDOMIZE_FATAL(wdata); + dai_wr(PlatOwnerAuthSlot3DigestOffset, wdata[TL_DW-1:0], wdata[TL_DW*2-1:TL_DW]); + end + if (wr_digest[RomPatchIdx]) begin + `DV_CHECK_STD_RANDOMIZE_FATAL(wdata); + dai_wr(RomPatchDigestOffset, wdata[TL_DW-1:0], wdata[TL_DW*2-1:TL_DW]); + end + endtask + + virtual task write_sw_rd_locks(bit [NumPartUnbuf-1:0] do_rd_lock= $urandom()); + if (do_rd_lock[VendorTestIdx]) csr_wr(ral.vendor_test_read_lock, 0); + if (do_rd_lock[CreatorSwCfgIdx]) csr_wr(ral.creator_sw_cfg_read_lock, 0); + if (do_rd_lock[OwnerSwCfgIdx]) csr_wr(ral.owner_sw_cfg_read_lock, 0); + if (do_rd_lock[OwnershipSlotStateIdx]) csr_wr(ral.ownership_slot_state_read_lock, 0); + if (do_rd_lock[RotCreatorAuthIdx]) csr_wr(ral.rot_creator_auth_read_lock, 0); + if (do_rd_lock[RotOwnerAuthSlot0Idx]) csr_wr(ral.rot_owner_auth_slot0_read_lock, 0); + if (do_rd_lock[RotOwnerAuthSlot1Idx]) csr_wr(ral.rot_owner_auth_slot1_read_lock, 0); + if (do_rd_lock[PlatIntegAuthSlot0Idx]) csr_wr(ral.plat_integ_auth_slot0_read_lock, 0); + if (do_rd_lock[PlatIntegAuthSlot1Idx]) csr_wr(ral.plat_integ_auth_slot1_read_lock, 0); + if (do_rd_lock[PlatOwnerAuthSlot0Idx]) csr_wr(ral.plat_owner_auth_slot0_read_lock, 0); + if (do_rd_lock[PlatOwnerAuthSlot1Idx]) csr_wr(ral.plat_owner_auth_slot1_read_lock, 0); + if (do_rd_lock[PlatOwnerAuthSlot2Idx]) csr_wr(ral.plat_owner_auth_slot2_read_lock, 0); + if (do_rd_lock[PlatOwnerAuthSlot3Idx]) csr_wr(ral.plat_owner_auth_slot3_read_lock, 0); + if (do_rd_lock[ExtNvmIdx]) csr_wr(ral.ext_nvm_read_lock, 0); + if (do_rd_lock[RomPatchIdx]) csr_wr(ral.rom_patch_read_lock, 0); + endtask + + // The digest CSR values are verified in otp_ctrl_scoreboard + virtual task rd_digests(); + bit [TL_DW-1:0] val; + csr_rd(.ptr(ral.vendor_test_digest[0]), .value(val)); + csr_rd(.ptr(ral.vendor_test_digest[1]), .value(val)); + csr_rd(.ptr(ral.creator_sw_cfg_digest[0]), .value(val)); + csr_rd(.ptr(ral.creator_sw_cfg_digest[1]), .value(val)); + csr_rd(.ptr(ral.owner_sw_cfg_digest[0]), .value(val)); + csr_rd(.ptr(ral.owner_sw_cfg_digest[1]), .value(val)); + csr_rd(.ptr(ral.rot_creator_auth_digest[0]), .value(val)); + csr_rd(.ptr(ral.rot_creator_auth_digest[1]), .value(val)); + csr_rd(.ptr(ral.rot_owner_auth_slot0_digest[0]), .value(val)); + csr_rd(.ptr(ral.rot_owner_auth_slot0_digest[1]), .value(val)); + csr_rd(.ptr(ral.rot_owner_auth_slot1_digest[0]), .value(val)); + csr_rd(.ptr(ral.rot_owner_auth_slot1_digest[1]), .value(val)); + csr_rd(.ptr(ral.plat_integ_auth_slot0_digest[0]), .value(val)); + csr_rd(.ptr(ral.plat_integ_auth_slot0_digest[1]), .value(val)); + csr_rd(.ptr(ral.plat_integ_auth_slot1_digest[0]), .value(val)); + csr_rd(.ptr(ral.plat_integ_auth_slot1_digest[1]), .value(val)); + csr_rd(.ptr(ral.plat_owner_auth_slot0_digest[0]), .value(val)); + csr_rd(.ptr(ral.plat_owner_auth_slot0_digest[1]), .value(val)); + csr_rd(.ptr(ral.plat_owner_auth_slot1_digest[0]), .value(val)); + csr_rd(.ptr(ral.plat_owner_auth_slot1_digest[1]), .value(val)); + csr_rd(.ptr(ral.plat_owner_auth_slot2_digest[0]), .value(val)); + csr_rd(.ptr(ral.plat_owner_auth_slot2_digest[1]), .value(val)); + csr_rd(.ptr(ral.plat_owner_auth_slot3_digest[0]), .value(val)); + csr_rd(.ptr(ral.plat_owner_auth_slot3_digest[1]), .value(val)); + csr_rd(.ptr(ral.rom_patch_digest[0]), .value(val)); + csr_rd(.ptr(ral.rom_patch_digest[1]), .value(val)); + csr_rd(.ptr(ral.hw_cfg0_digest[0]), .value(val)); + csr_rd(.ptr(ral.hw_cfg0_digest[1]), .value(val)); + csr_rd(.ptr(ral.hw_cfg1_digest[0]), .value(val)); + csr_rd(.ptr(ral.hw_cfg1_digest[1]), .value(val)); + csr_rd(.ptr(ral.secret0_digest[0]), .value(val)); + csr_rd(.ptr(ral.secret0_digest[1]), .value(val)); + csr_rd(.ptr(ral.secret1_digest[0]), .value(val)); + csr_rd(.ptr(ral.secret1_digest[1]), .value(val)); + csr_rd(.ptr(ral.secret2_digest[0]), .value(val)); + csr_rd(.ptr(ral.secret2_digest[1]), .value(val)); + csr_rd(.ptr(ral.secret3_digest[0]), .value(val)); + csr_rd(.ptr(ral.secret3_digest[1]), .value(val)); + endtask + + // If the partition is read/write locked, there is 20% chance we will force the internal mubi + // access signal to the values other than mubi::true or mubi::false. + virtual task force_mubi_part_access(); + // Stress_all_with_rand_reset seq will issue reset and wait until reset is done then kill the + // parallel sequence. This gating logic avoid injecting error during reset active. + if (cfg.otp_ctrl_vif.alert_reqs == 0 && !cfg.under_reset) begin + otp_part_access_lock_t forced_mubi_part_access[NumPart-1]; + + // Digest write locks + if ((`gmv(ral.vendor_test_digest[0]) || + `gmv(ral.vendor_test_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[VendorTestIdx].write_lock = 1; + end + if ((`gmv(ral.creator_sw_cfg_digest[0]) || + `gmv(ral.creator_sw_cfg_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[CreatorSwCfgIdx].write_lock = 1; + end + if ((`gmv(ral.owner_sw_cfg_digest[0]) || + `gmv(ral.owner_sw_cfg_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[OwnerSwCfgIdx].write_lock = 1; + end + if ((`gmv(ral.rot_creator_auth_digest[0]) || + `gmv(ral.rot_creator_auth_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[RotCreatorAuthIdx].write_lock = 1; + end + if ((`gmv(ral.rot_owner_auth_slot0_digest[0]) || + `gmv(ral.rot_owner_auth_slot0_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[RotOwnerAuthSlot0Idx].write_lock = 1; + end + if ((`gmv(ral.rot_owner_auth_slot1_digest[0]) || + `gmv(ral.rot_owner_auth_slot1_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[RotOwnerAuthSlot1Idx].write_lock = 1; + end + if ((`gmv(ral.plat_integ_auth_slot0_digest[0]) || + `gmv(ral.plat_integ_auth_slot0_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[PlatIntegAuthSlot0Idx].write_lock = 1; + end + if ((`gmv(ral.plat_integ_auth_slot1_digest[0]) || + `gmv(ral.plat_integ_auth_slot1_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[PlatIntegAuthSlot1Idx].write_lock = 1; + end + if ((`gmv(ral.plat_owner_auth_slot0_digest[0]) || + `gmv(ral.plat_owner_auth_slot0_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[PlatOwnerAuthSlot0Idx].write_lock = 1; + end + if ((`gmv(ral.plat_owner_auth_slot1_digest[0]) || + `gmv(ral.plat_owner_auth_slot1_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[PlatOwnerAuthSlot1Idx].write_lock = 1; + end + if ((`gmv(ral.plat_owner_auth_slot2_digest[0]) || + `gmv(ral.plat_owner_auth_slot2_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[PlatOwnerAuthSlot2Idx].write_lock = 1; + end + if ((`gmv(ral.plat_owner_auth_slot3_digest[0]) || + `gmv(ral.plat_owner_auth_slot3_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[PlatOwnerAuthSlot3Idx].write_lock = 1; + end + if ((`gmv(ral.rom_patch_digest[0]) || + `gmv(ral.rom_patch_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[RomPatchIdx].write_lock = 1; + end + if ((`gmv(ral.hw_cfg0_digest[0]) || + `gmv(ral.hw_cfg0_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[HwCfg0Idx].write_lock = 1; + end + if ((`gmv(ral.hw_cfg1_digest[0]) || + `gmv(ral.hw_cfg1_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[HwCfg1Idx].write_lock = 1; + end + if ((`gmv(ral.secret0_digest[0]) || + `gmv(ral.secret0_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[Secret0Idx].write_lock = 1; + end + if ((`gmv(ral.secret1_digest[0]) || + `gmv(ral.secret1_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[Secret1Idx].write_lock = 1; + end + if ((`gmv(ral.secret2_digest[0]) || + `gmv(ral.secret2_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[Secret2Idx].write_lock = 1; + end + if ((`gmv(ral.secret3_digest[0]) || + `gmv(ral.secret3_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[Secret3Idx].write_lock = 1; + end + + // CSR read locks + if ((`gmv(ral.vendor_test_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[VendorTestIdx].read_lock = 1; + end + if ((`gmv(ral.creator_sw_cfg_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[CreatorSwCfgIdx].read_lock = 1; + end + if ((`gmv(ral.owner_sw_cfg_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[OwnerSwCfgIdx].read_lock = 1; + end + if ((`gmv(ral.ownership_slot_state_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[OwnershipSlotStateIdx].read_lock = 1; + end + if ((`gmv(ral.rot_creator_auth_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[RotCreatorAuthIdx].read_lock = 1; + end + if ((`gmv(ral.rot_owner_auth_slot0_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[RotOwnerAuthSlot0Idx].read_lock = 1; + end + if ((`gmv(ral.rot_owner_auth_slot1_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[RotOwnerAuthSlot1Idx].read_lock = 1; + end + if ((`gmv(ral.plat_integ_auth_slot0_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[PlatIntegAuthSlot0Idx].read_lock = 1; + end + if ((`gmv(ral.plat_integ_auth_slot1_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[PlatIntegAuthSlot1Idx].read_lock = 1; + end + if ((`gmv(ral.plat_owner_auth_slot0_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[PlatOwnerAuthSlot0Idx].read_lock = 1; + end + if ((`gmv(ral.plat_owner_auth_slot1_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[PlatOwnerAuthSlot1Idx].read_lock = 1; + end + if ((`gmv(ral.plat_owner_auth_slot2_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[PlatOwnerAuthSlot2Idx].read_lock = 1; + end + if ((`gmv(ral.plat_owner_auth_slot3_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[PlatOwnerAuthSlot3Idx].read_lock = 1; + end + if ((`gmv(ral.ext_nvm_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[ExtNvmIdx].read_lock = 1; + end + if ((`gmv(ral.rom_patch_read_lock) == 0) && !$urandom_range(0, 4)) begin + forced_mubi_part_access[RomPatchIdx].read_lock = 1; + end + + + // Digest read locks + if ((`gmv(ral.secret0_digest[0]) || + `gmv(ral.secret0_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[Secret0Idx].read_lock = 1; + end + if ((`gmv(ral.secret1_digest[0]) || + `gmv(ral.secret1_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[Secret1Idx].read_lock = 1; + end + if ((`gmv(ral.secret2_digest[0]) || + `gmv(ral.secret2_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[Secret2Idx].read_lock = 1; + end + if ((`gmv(ral.secret3_digest[0]) || + `gmv(ral.secret3_digest[1])) && + !$urandom_range(0, 4)) begin + forced_mubi_part_access[Secret3Idx].read_lock = 1; + end + + foreach (forced_mubi_part_access[i]) begin + `uvm_info(`gfn, $sformatf("partition %0d inject mubi value: read=%0b, write=%0b", i, + forced_mubi_part_access[i].read_lock, forced_mubi_part_access[i].write_lock), UVM_HIGH) + end + + cfg.otp_ctrl_vif.force_part_access_mubi(forced_mubi_part_access); + end + endtask + + // This function backdoor inject error according to ecc_err: + // - for OtpEccUncorrErr it injects a 2 bit eror + // - for OtpEccCorrErr it injects a 1 bit eror + // This function will output original backdoor read data for the given address + // so the error can be cleared. + virtual function bit [TL_DW-1:0] backdoor_inject_ecc_err(bit [TL_DW-1:0] addr, + otp_ecc_err_e ecc_err); + bit [TL_DW-1:0] val; + addr = {addr[TL_DW-1:2], 2'b00}; + val = cfg.mem_bkdr_util_h.read32(addr); + if (ecc_err == OtpNoEccErr || addr >= (LifeCycleOffset + LifeCycleSize)) return val; + + // Backdoor read and write back with error bits + cfg.mem_bkdr_util_h.inject_errors(addr, (ecc_err == OtpEccUncorrErr) ? 2 : 1); + `uvm_info(`gfn, $sformatf("original val %0h, addr %0h, err_type %0s", + val, addr, ecc_err.name), UVM_HIGH) + return val; + endfunction + + virtual task trigger_checks(bit [1:0] val, + bit wait_done = 1, + otp_ecc_err_e ecc_err = OtpNoEccErr); + bit [TL_DW-1:0] backdoor_rd_val, addr; + + // If ECC and check error happens in the same consistency check, the scb cannot predict which + // error will happen first, so it cannot correctly predict the error status and alert + // triggered. + // So the sequence only allows one error at a time. + if (get_field_val(ral.check_trigger.consistency, val) && + `gmv(ral.check_timeout) > 0 && `gmv(ral.check_timeout) <= CHK_TIMEOUT_CYC) begin + ecc_err = OtpNoEccErr; + end + + // Backdoor write ECC errors + if (ecc_err != OtpNoEccErr) begin + int part_idx = $urandom_range(HwCfg0Idx, LifeCycleIdx); + + // Only HW cfgs check digest correctness + if (part_idx != LifeCycleIdx) begin + addr = $urandom_range(0, 1) ? PART_OTP_DIGEST_ADDRS[part_idx] << 2 : + (PART_OTP_DIGEST_ADDRS[part_idx] + 1) << 2; + end else begin + addr = $urandom_range(LifeCycleOffset, LifeCycleOffset + LifeCycleSize - 1); + addr = {addr[TL_DW-1:2], 2'b00}; + end + backdoor_rd_val = backdoor_inject_ecc_err(addr, ecc_err); + cfg.ecc_chk_err[part_idx] = ecc_err; + end + + csr_wr(ral.check_trigger, val); + if (wait_done && val) csr_spinwait(ral.status.check_pending, 0); + + if (ecc_err != OtpNoEccErr) begin + cfg.mem_bkdr_util_h.write32(addr, backdoor_rd_val); + cfg.ecc_chk_err = '{default: OtpNoEccErr}; + end + endtask + + // For a DAI interface operation to finish, either way until status dai_idle is set, or check + // err_code and see if fatal error happened. In any case, break out of this wait if there + // is a need to stop transaction generators, since a spinwait will otherwise just stop + // when it times-out. + virtual task wait_dai_op_done(); + if (cfg.stop_transaction_generators()) return; + fork begin + fork + begin + csr_spinwait(.ptr(ral.status.dai_idle), + .exp_data(1), + .timeout_ns(op_done_spinwait_timeout_ns), + .spinwait_delay_ns($urandom_range(0, 5))); + end + begin + forever begin + bit [TL_DW-1:0] err_val; + cfg.clk_rst_vif.wait_clks(1); + csr_rd(.ptr(ral.err_code[DaiIdx].err_code), .value(err_val), .backdoor(1)); + // Break if error will cause fatal alerts + if (err_val inside {OTP_TERMINAL_ERRS}) break; + end + end + begin + forever begin + cfg.clk_rst_vif.wait_clks(1); + if (cfg.stop_transaction_generators()) break; + end + end + join_any + wait_no_outstanding_access(); + disable fork; + end join + endtask + + virtual task rd_and_clear_intrs(); + bit [TL_DW-1:0] val; + if (cfg.otp_ctrl_vif.lc_prog_no_sta_check == 0) begin + csr_rd(ral.intr_state, val); + // In case lc_program request is issued after intr_state read + if (cfg.otp_ctrl_vif.lc_prog_no_sta_check == 0) csr_wr(ral.intr_state, val); + end + endtask + + // first two or three LSB bits of DAI address can be randomized based on if it is secret + virtual function bit [TL_AW-1:0] randomize_dai_addr(bit [TL_AW-1:0] dai_addr); + if (is_secret(dai_addr)) begin + bit [2:0] rand_addr = $urandom(); + randomize_dai_addr = {dai_addr[TL_DW-1:3], rand_addr}; + end else begin + bit [1:0] rand_addr = $urandom(); + randomize_dai_addr = {dai_addr[TL_DW-1:2], rand_addr}; + end + endfunction + + // The following interface requests are separated to blocking and non-blocking accesses. + // The non-blocking access is mainly used when lc_escalate_en is On, which acts like a reset and + // move all design state machines to ErrorSt. Thus pending request will never get a response + // until reset. + virtual task req_sram_key(int index, bit blocking = default_req_blocking); + // Return if the request is already high, this is mainly due to lc_escalate_en On. + if (cfg.m_sram_pull_agent_cfg[index].vif.req === 1'b1) return; + + if (blocking) begin + req_sram_key_sub(index); + end else begin + fork + begin + req_sram_key_sub(index); + end + join_none; + // Add #0 to ensure that this thread starts executing before any subsequent call + #0; + end + endtask + + virtual task req_sram_key_sub(int index); + push_pull_host_seq#(.DeviceDataWidth(SRAM_DATA_SIZE)) sram_pull_seq; + wait(cfg.under_reset == 0); + `uvm_create_on(sram_pull_seq, p_sequencer.sram_pull_sequencer_h[index]); + `DV_CHECK_RANDOMIZE_FATAL(sram_pull_seq) + `uvm_send(sram_pull_seq) + endtask + + virtual task req_all_sram_keys(bit blocking = default_req_blocking); + for (int i = 0; i < NumSramKeyReqSlots; i++) req_sram_key(i, blocking); + endtask + + virtual task req_otbn_key(bit blocking = default_req_blocking); + if (cfg.m_otbn_pull_agent_cfg.vif.req === 1'b1) return; + + if (blocking) begin + req_otbn_key_sub(); + end else begin + fork + begin + req_otbn_key_sub(); + end + join_none; + // Add #0 to ensure that this thread starts executing before any subsequent call + #0; + end + endtask + + virtual task req_otbn_key_sub(); + push_pull_host_seq#(.DeviceDataWidth(OTBN_DATA_SIZE)) otbn_pull_seq; + wait(cfg.under_reset == 0); + `uvm_create_on(otbn_pull_seq, p_sequencer.otbn_pull_sequencer_h); + `DV_CHECK_RANDOMIZE_FATAL(otbn_pull_seq) + `uvm_send(otbn_pull_seq) + endtask + + virtual task req_flash_addr_key(bit blocking = default_req_blocking); + if (cfg.m_flash_addr_pull_agent_cfg.vif.req === 1'b1) return; + + if (blocking) begin + req_flash_addr_key_sub(); + end else begin + fork + begin + req_flash_addr_key_sub(); + end + join_none; + // Add #0 to ensure that this thread starts executing before any subsequent call + #0; + end + endtask + + virtual task req_flash_addr_key_sub(); + push_pull_host_seq#(.DeviceDataWidth(FLASH_DATA_SIZE)) flash_addr_pull_seq; + wait(cfg.under_reset == 0); + `uvm_create_on(flash_addr_pull_seq, p_sequencer.flash_addr_pull_sequencer_h); + `DV_CHECK_RANDOMIZE_FATAL(flash_addr_pull_seq) + `uvm_send(flash_addr_pull_seq) + endtask + + virtual task req_flash_data_key(bit blocking = default_req_blocking); + if (cfg.m_flash_data_pull_agent_cfg.vif.req === 1'b1) return; + + if (blocking) begin + req_flash_data_key_sub(); + end else begin + fork + begin + req_flash_data_key_sub(); + end + join_none; + // Add #0 to ensure that this thread starts executing before any subsequent call + #0; + end + endtask + + virtual task req_flash_data_key_sub(); + push_pull_host_seq#(.DeviceDataWidth(FLASH_DATA_SIZE)) flash_data_pull_seq; + wait(cfg.under_reset == 0); + `uvm_create_on(flash_data_pull_seq, p_sequencer.flash_data_pull_sequencer_h); + `DV_CHECK_RANDOMIZE_FATAL(flash_data_pull_seq) + `uvm_send(flash_data_pull_seq) + endtask + + virtual task req_lc_transition(bit check_intr = 0, + bit blocking = default_req_blocking, + bit wr_blank_err = !write_unused_addr); + if (cfg.m_lc_prog_pull_agent_cfg.vif.req === 1'b1) return; + + if (blocking) begin + req_lc_transition_sub(check_intr, wr_blank_err); + end else begin + fork + begin + req_lc_transition_sub(check_intr, wr_blank_err); + end + join_none; + // Add #0 to ensure that this thread starts executing before any subsequent call + #0; + end + endtask + + virtual task req_lc_transition_sub(bit check_intr = 0, bit wr_blank_err = !write_unused_addr); + lc_ctrl_state_pkg::lc_cnt_e next_lc_cnt; + lc_ctrl_state_pkg::dec_lc_state_e next_lc_state, lc_state_dec; + bit [TL_DW-1:0] intr_val; + push_pull_host_seq#(.HostDataWidth(LC_PROG_DATA_SIZE), .DeviceDataWidth(1)) + lc_prog_pull_seq; + wait(cfg.under_reset == 0); + `uvm_create_on(lc_prog_pull_seq, p_sequencer.lc_prog_pull_sequencer_h); + + if (!wr_blank_err) begin + // Find valid next state and next cnt using lc_ctrl_dv_utils_pkg. + // If terminal state or max LcCnt reaches, will not program any new data. + if ((lc_state != LcStScrap) && (lc_cnt != LcCnt24)) begin + lc_state_dec = lc_ctrl_dv_utils_pkg::dec_lc_state(lc_state); + `DV_CHECK_STD_RANDOMIZE_WITH_FATAL(next_lc_state, + next_lc_state inside {VALID_NEXT_STATES[lc_state_dec]};) + `DV_CHECK_STD_RANDOMIZE_WITH_FATAL(next_lc_cnt, next_lc_cnt > lc_cnt;) + lc_state = lc_ctrl_dv_utils_pkg::encode_lc_state(next_lc_state); + lc_cnt = next_lc_cnt; + end + cfg.m_lc_prog_pull_agent_cfg.add_h_user_data({lc_cnt, lc_state}); + end + + `DV_CHECK_RANDOMIZE_FATAL(lc_prog_pull_seq) + `uvm_send(lc_prog_pull_seq) + + if (check_intr) rd_and_clear_intrs(); + endtask + + // This test access OTP_CTRL's test_access memory. The open-sourced code only test if the access + // is valid. Please override this task in proprietary OTP. + virtual task otp_test_access(); + if (`PRIM_DEFAULT_IMPL == prim_pkg::ImplGeneric) begin + repeat (10) begin + bit [TL_DW-1:0] data; + bit test_access_en; + bit [TL_AW-1:0] rand_addr = $urandom_range(0, NUM_PRIM_REG - 1) * 4; + bit [TL_AW-1:0] tlul_addr = + cfg.ral_models["otp_ctrl_prim_reg_block"].get_addr_from_offset(rand_addr); + if (cfg.stop_transaction_generators()) break; + rand_drive_dft_en(); + `DV_CHECK_STD_RANDOMIZE_FATAL(data) + test_access_en = cfg.otp_ctrl_vif.lc_dft_en_i == lc_ctrl_pkg::On; + tl_access(.addr(tlul_addr), .write(1), .data(data), .exp_err_rsp(~test_access_en), + .tl_sequencer_h(p_sequencer.tl_sequencer_hs["otp_ctrl_prim_reg_block"])); + tl_access(.addr(tlul_addr), .write(0), .data(data), .exp_err_rsp(~test_access_en), + .tl_sequencer_h(p_sequencer.tl_sequencer_hs["otp_ctrl_prim_reg_block"])); + end + end + endtask + + // Empty task, only drive it under `otp_ctrl_test_access_vseq` + virtual task rand_drive_dft_en(); + endtask +endclass : otp_ctrl_base_vseq diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_callback_vseq.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_callback_vseq.sv new file mode 100644 index 00000000000000..155e6f5e3615ca --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_callback_vseq.sv @@ -0,0 +1,22 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// A sequence hook to attach to otp_ctrl_base_vseq. +class otp_ctrl_callback_vseq extends cip_base_vseq #( + .RAL_T (otp_ctrl_core_reg_block), + .CFG_T (otp_ctrl_env_cfg), + .COV_T (otp_ctrl_env_cov), + .VIRTUAL_SEQUENCER_T (otp_ctrl_virtual_sequencer) + ); + `uvm_object_utils(otp_ctrl_callback_vseq) + `uvm_object_new + + virtual task dut_init_callback(); + // Do nothing but can be overridden in closed source environment. + endtask + + virtual task post_otp_pwr_init(); + // Do nothing but can be overridden in closed source environment. + endtask : post_otp_pwr_init +endclass : otp_ctrl_callback_vseq diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_check_fail_vseq.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_check_fail_vseq.sv new file mode 100644 index 00000000000000..061ebe96a30a08 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_check_fail_vseq.sv @@ -0,0 +1,32 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// This sequence creates the following check failure scenarios: +// 1. Check timeout +// 2. Correctable ECC check error +// 3. Uncorrectable ECC error +class otp_ctrl_check_fail_vseq extends otp_ctrl_dai_lock_vseq; + `uvm_object_utils(otp_ctrl_check_fail_vseq) + + `uvm_object_new + + constraint ecc_otp_err_c { + ecc_otp_err inside {OtpEccCorrErr, OtpNoEccErr}; + } + + constraint ecc_chk_err_c { + ecc_chk_err dist {OtpNoEccErr :/ 1, + OtpEccCorrErr :/ 1, + OtpEccUncorrErr :/1 }; + } + + // 50% chance of having a check timeout + // Because of the regwen, even though we constrain the timeout value, it might not apply to the + // DUT. + constraint check_timeout_val_c { + check_timeout_val dist {[1 : CHK_TIMEOUT_CYC] :/ 1, + [100_000 :'1] :/ 1}; + } + +endclass diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_common_vseq.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_common_vseq.sv new file mode 100644 index 00000000000000..58f419c77c0ad5 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_common_vseq.sv @@ -0,0 +1,232 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +class otp_ctrl_common_vseq extends otp_ctrl_base_vseq; + `uvm_object_utils(otp_ctrl_common_vseq) + + rand bit [TL_DW-1:0] dai_addr, wdata0, wdata1; + rand port_drive_condition_e reset_drive_cond; + + string prim_otp_alert_name = "fatal_prim_otp_alert"; + string integ_err_alert_name = "fatal_bus_integ_error"; + + // This flag is used to identify if the sec_cm or tl_intg_err uses prim_otp_tl_i/o. + protected bit is_prim_otp; + + constraint dai_addr_c { + dai_addr dist { + [0 : (PartInfo[LifeCycleIdx].offset - 1)] :/ 1, + [PartInfo[LifeCycleIdx].offset : {OTP_ADDR_WIDTH{1'b1}}] :/ 1}; + } + + constraint reset_drive_cond_c { + reset_drive_cond dist { + DriveRandomly :/ 7, + DuringOTPDaiBusy :/ 1, + DuringOTPRead :/ 1, + DuringOTPDaiDigest :/ 1 + }; + } + + constraint num_trans_c { + num_trans inside {[1:2]}; + } + `uvm_object_new + + virtual task dut_init(string reset_kind = "HARD"); + super.dut_init(reset_kind); + // drive dft_en pins to access the test_access memory + cfg.otp_ctrl_vif.drive_lc_dft_en(lc_ctrl_pkg::On); + // once turn on lc_dft_en regiser, will need some time to update the state register + // two clock cycles for lc_async mode, one clock cycle for driving dft_en, one more clock cycle + // so there is no racing condition. + if (cfg.en_dv_cdc) cfg.clk_rst_vif.wait_clks(5); + else cfg.clk_rst_vif.wait_clks(3); + endtask + + virtual task body(); + if (common_seq_type == "sec_cm_fi") begin + // OTP_CTRL has many sec_cm items, so too many iterations of this test will consume too much + // simulation time and eventually causes timeout. So we reduce to 10 iterations. + run_sec_cm_fi_vseq(10); + end else begin + run_common_vseq_wrapper(num_trans); + end + endtask : body + + virtual task apply_resets_concurrently(int reset_duration_ps = 0); + // For stress_all_with_rand_reset test only - backdoor clear OTP memory, + // and re-initialize OTP_ctrl after reset. + if (common_seq_type == "stress_all_with_rand_reset") begin + cfg.otp_ctrl_vif.release_part_access_mubi(); + cfg.otp_ctrl_vif.drive_lc_escalate_en(lc_ctrl_pkg::Off); + // Set dft_en to On to allow the csr_test to check all registers' default value after reset. + cfg.otp_ctrl_vif.drive_lc_dft_en(lc_ctrl_pkg::On); + otp_ctrl_init(); + otp_pwr_init(); + super.apply_resets_concurrently(reset_duration_ps); + cfg.en_scb = 1; + end else begin + super.apply_resets_concurrently(reset_duration_ps); + end + clear_seq_flags(); + endtask + + virtual task wait_to_issue_reset(uint reset_delay_bound = 10_000_000); + `DV_CHECK_MEMBER_RANDOMIZE_FATAL(reset_drive_cond) + case (reset_drive_cond) + DriveRandomly: begin + super.wait_to_issue_reset(reset_delay_bound); + end + DuringOTPDaiBusy: begin + `DV_SPINWAIT_EXIT( + wait (do_otp_wr); + cfg.clk_rst_vif.wait_clks($urandom_range(1, 70));, + super.wait_to_issue_reset(reset_delay_bound);) + #($urandom_range(0, cfg.clk_rst_vif.clk_period_ps) * 1ps); + end + DuringOTPDaiDigest: begin + `DV_SPINWAIT_EXIT( + wait (do_digest_cal); + cfg.clk_rst_vif.wait_clks($urandom_range(1, 350));, + super.wait_to_issue_reset(reset_delay_bound);) + #($urandom_range(0, cfg.clk_rst_vif.clk_period_ps) * 1ps); + end + DuringOTPRead: begin + `DV_SPINWAIT_EXIT( + wait (do_otp_rd); + cfg.clk_rst_vif.wait_clks($urandom_range(1, 10));, + super.wait_to_issue_reset(reset_delay_bound);) + #($urandom_range(0, cfg.clk_rst_vif.clk_period_ps) * 1ps); + end + default: `uvm_fatal(`gfn, $sformatf("Unsupported reset_drive_cond %0d", reset_drive_cond)) + endcase + endtask : wait_to_issue_reset + + // This task overrides the check for `prim_onehot_check` and `tl_intg_error`. + // Alerts coming from the `prim_otp` module will only bypass OTP_CTRL, it won't affect the + // OTP_CTRL and will fire its own alerts. + virtual task check_tl_intg_error_response(); + if (is_prim_otp) begin + repeat ($urandom_range(5, 20)) begin + wait_alert_trigger(prim_otp_alert_name, .wait_complete(1)); + end + end else begin + super.check_tl_intg_error_response(); + end + endtask + + virtual task check_sec_cm_alert(string sec_type_name, string alert_name); + `uvm_info(`gfn, $sformatf("expected fatal alert is triggered for %s", + sec_type_name), UVM_LOW) + + // This is a fatal alert and design keeps sending it until reset is issued. + // Check alerts are triggered for a few times + repeat (5) begin + wait_alert_trigger(alert_name, .wait_complete(1)); + end + endtask + + // In tl_intg_err test, override this task to set is_prim_otp flag. + virtual task run_tl_intg_err_vseq_sub(string ral_name); + if (ral_name == "otp_ctrl_prim_reg_block") is_prim_otp = 1; + else is_prim_otp = 0; + super.run_tl_intg_err_vseq_sub(ral_name); + endtask + + virtual task check_sec_cm_fi_resp(sec_cm_base_if_proxy if_proxy); + bit [TL_DW-1:0] exp_status_val, rdata0, rdata1; + string prim_otp_alert_name = "fatal_prim_otp_alert"; + string integ_err_alert_name = "fatal_bus_integ_error"; + + // Alerts coming from the `prim_otp` module will only bypass OTP_CTRL, it won't affect the + // OTP_CTRL and will fire its own alerts. + if (is_prim_otp) begin + check_sec_cm_alert(if_proxy.sec_cm_type.name, prim_otp_alert_name); + + // Alerts coming from the `u_tlul_lc_gate` module will only trigger bus_integrity alerts, and + // bus_integrity related status. + // This error won't local escalate to OTP partitions. + end else if (!uvm_re_match("*.u_tlul_lc_gate*", if_proxy.path)) begin + check_sec_cm_alert(if_proxy.sec_cm_type.name, integ_err_alert_name); + + exp_status_val[OtpBusIntegErrorIdx] = 1; + exp_status_val[OtpDaiIdleIdx] = 1; + + // All other errors triggers normal fatal alerts, and will locally escalate to other + // partitions. + end else begin + super.check_sec_cm_fi_resp(if_proxy); + + // Set expected status error val. + for (int i = 0; i <= OtpLciErrIdx; i++) exp_status_val[i] = 1; + if (!uvm_re_match("*.u_otp_ctrl_lfsr_timer*", if_proxy.path)) begin + exp_status_val[OtpLfsrFsmErrIdx] = 1; + end else if (!uvm_re_match("*u_otp_ctrl_kdi*", if_proxy.path)) begin + exp_status_val[OtpDerivKeyFsmErrIdx] = 1; + end else if (!uvm_re_match("*u_otp_ctrl_scrmbl*", if_proxy.path)) begin + exp_status_val[OtpScramblingFsmErrIdx] = 1; + end + + csr_rd_check(.ptr(ral.status), .compare_value(exp_status_val), .err_msg( + $sformatf("cm_fi status failed at injection %0s", if_proxy.sec_cm_type.name))); + + // Check OTP is locked after fault error. + `DV_CHECK_RANDOMIZE_FATAL(this) + is_valid_dai_op = 0; + + // Access OTP via DAI interface. + dai_wr(dai_addr, wdata0, wdata1); + dai_rd(dai_addr, rdata0, rdata1); + `DV_CHECK_EQ(rdata0, 0) + `DV_CHECK_EQ(rdata1, 0) + if (is_sw_part(dai_addr)) begin + uvm_reg_addr_t tlul_addr = cfg.ral.get_addr_from_offset(get_sw_window_offset(dai_addr)); + tl_access(.addr(tlul_addr), .write(0), .data(rdata0), .blocking(1), .check_rsp(1), + .exp_err_rsp(1), .exp_data('1)); + end + cal_hw_digests(); + write_sw_digests(); + + // Access OTP via app interface. + if ($urandom_range(0, 1)) req_otbn_key(0); + if ($urandom_range(0, 1)) req_flash_addr_key(0); + if ($urandom_range(0, 1)) req_flash_data_key(0); + if ($urandom_range(0, 1)) req_all_sram_keys(0); + cfg.clk_rst_vif.wait_clks($urandom_range(10, 20)); + + csr_rd_check(.ptr(ral.status), .compare_value(exp_status_val), + .err_msg("status failure after OTP fatal fault error")); + end + endtask : check_sec_cm_fi_resp + + virtual function void sec_cm_fi_ctrl_svas(sec_cm_base_if_proxy if_proxy, bit enable); + case (if_proxy.sec_cm_type) + SecCmPrimCount: begin + if (!enable) begin + $assertoff(0, "tb.dut.gen_partitions[3].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $assertoff(0, "tb.dut.gen_partitions[4].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $assertoff(0, "tb.dut.gen_partitions[5].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $assertoff(0, "tb.dut.gen_partitions[6].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $assertoff(0, "tb.dut.gen_partitions[7].gen_lifecycle.u_part_buf.ScrmblDataKnown_A"); + end else begin + $asserton(0, "tb.dut.gen_partitions[3].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $asserton(0, "tb.dut.gen_partitions[4].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $asserton(0, "tb.dut.gen_partitions[5].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $asserton(0, "tb.dut.gen_partitions[6].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $asserton(0, "tb.dut.gen_partitions[7].gen_lifecycle.u_part_buf.ScrmblDataKnown_A"); + end + end + SecCmPrimSparseFsmFlop, SecCmPrimDoubleLfsr, SecCmPrimOnehot: begin + // No assertion error. + end + default: `uvm_fatal(`gfn, $sformatf("unexpected sec_cm_type %s", if_proxy.sec_cm_type.name)) + endcase + + // Set the flag to store if the error injection is on prim_tlul_if or core_tlul_if. + if (!uvm_re_match("*.u_otp.*", if_proxy.path)) is_prim_otp = 1; + else is_prim_otp = 0; + endfunction: sec_cm_fi_ctrl_svas + +endclass diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv new file mode 100644 index 00000000000000..c8ea43b32a6bcb --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv @@ -0,0 +1,51 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// otp_ctrl_dai_errs_vseq is developed to randomly read/write to any address within OTP: +// - A writeblank error will be triggered if write to a non-empty address +// - An access error will be triggered if write to lc partition via DAI interface, or if DAI write +// to digest addrs for non-sw partitions +class otp_ctrl_dai_errs_vseq extends otp_ctrl_dai_lock_vseq; + `uvm_object_utils(otp_ctrl_dai_errs_vseq) + + bit[31:0] exp_status; + `uvm_object_new + + // Only run one transition to avoid dut_init in the sequence. Because write-blank-error can cause + // otp_init failure. + constraint num_trans_c { + num_trans == 1; + num_dai_op inside {[100:500]}; + } + + constraint regwens_c { + check_trigger_regwen_val == 0; + } + + constraint rd_check_after_wr_c { + rand_wr == rand_rd; + } + + function void pre_randomize(); + this.dai_wr_blank_addr_c.constraint_mode(0); + this.no_access_err_c.constraint_mode(0); + this.dai_wr_digests_c.constraint_mode(0); + write_unused_addr = 0; + endfunction + + task body(); + do_apply_reset = 0; + if (do_lc_trans && !cfg.otp_ctrl_vif.alert_reqs) begin + req_lc_transition(do_lc_trans, lc_prog_blocking); + end + super.body(); + endtask + + virtual task post_start(); + expect_fatal_alerts = 1; + do_apply_reset = 1; + do_otp_ctrl_init = 1; + super.post_start(); + endtask +endclass diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv new file mode 100644 index 00000000000000..a482363f8506f7 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv @@ -0,0 +1,146 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// otp_ctrl_dai_lock_vseq is developed to read/write lock DAI interface by partitions, and request +// read/write access to check if correct status and error code is triggered + +// Partition's legal range covers offset to digest addresses, dai_rd/dai_wr function will +// randomize the address based on the granularity. +`define PART_ADDR_RANGE(i) \ + {[PartInfo[``i``].offset : (PartInfo[``i``].offset + PartInfo[``i``].size - 8)]} + +class otp_ctrl_dai_lock_vseq extends otp_ctrl_smoke_vseq; + `uvm_object_utils(otp_ctrl_dai_lock_vseq) + + `uvm_object_new + + // enable access_err for each cycle + constraint no_access_err_c {access_locked_parts == 1;} + + constraint num_trans_c { + num_trans inside {[1:10]}; + num_dai_op inside {[1:50]}; + } + + // the LC partition is always the last one + constraint partition_index_c {part_idx inside {[0:LifeCycleIdx]};} + + constraint dai_wr_legal_addr_c { + if (part_idx == VendorTestIdx) { + dai_addr inside `PART_ADDR_RANGE(VendorTestIdx); + } + if (part_idx == CreatorSwCfgIdx) { + dai_addr inside `PART_ADDR_RANGE(CreatorSwCfgIdx); + } + if (part_idx == OwnerSwCfgIdx) { + dai_addr inside `PART_ADDR_RANGE(OwnerSwCfgIdx); + } + if (part_idx == OwnershipSlotStateIdx) { + dai_addr inside `PART_ADDR_RANGE(OwnershipSlotStateIdx); + } + if (part_idx == RotCreatorAuthIdx) { + dai_addr inside `PART_ADDR_RANGE(RotCreatorAuthIdx); + } + if (part_idx == RotOwnerAuthSlot0Idx) { + dai_addr inside `PART_ADDR_RANGE(RotOwnerAuthSlot0Idx); + } + if (part_idx == RotOwnerAuthSlot1Idx) { + dai_addr inside `PART_ADDR_RANGE(RotOwnerAuthSlot1Idx); + } + if (part_idx == PlatIntegAuthSlot0Idx) { + dai_addr inside `PART_ADDR_RANGE(PlatIntegAuthSlot0Idx); + } + if (part_idx == PlatIntegAuthSlot1Idx) { + dai_addr inside `PART_ADDR_RANGE(PlatIntegAuthSlot1Idx); + } + if (part_idx == PlatOwnerAuthSlot0Idx) { + dai_addr inside `PART_ADDR_RANGE(PlatOwnerAuthSlot0Idx); + } + if (part_idx == PlatOwnerAuthSlot1Idx) { + dai_addr inside `PART_ADDR_RANGE(PlatOwnerAuthSlot1Idx); + } + if (part_idx == PlatOwnerAuthSlot2Idx) { + dai_addr inside `PART_ADDR_RANGE(PlatOwnerAuthSlot2Idx); + } + if (part_idx == PlatOwnerAuthSlot3Idx) { + dai_addr inside `PART_ADDR_RANGE(PlatOwnerAuthSlot3Idx); + } + if (part_idx == ExtNvmIdx) { + dai_addr inside `PART_ADDR_RANGE(ExtNvmIdx); + } + if (part_idx == RomPatchIdx) { + dai_addr inside `PART_ADDR_RANGE(RomPatchIdx); + } + if (part_idx == HwCfg0Idx) { + dai_addr inside `PART_ADDR_RANGE(HwCfg0Idx); + } + if (part_idx == HwCfg1Idx) { + dai_addr inside `PART_ADDR_RANGE(HwCfg1Idx); + } + if (part_idx == Secret0Idx) { + dai_addr inside `PART_ADDR_RANGE(Secret0Idx); + } + if (part_idx == Secret1Idx) { + dai_addr inside `PART_ADDR_RANGE(Secret1Idx); + } + if (part_idx == Secret2Idx) { + dai_addr inside `PART_ADDR_RANGE(Secret2Idx); + } + if (part_idx == Secret3Idx) { + dai_addr inside `PART_ADDR_RANGE(Secret3Idx); + } + if (part_idx == LifeCycleIdx) { + if (write_unused_addr) { + dai_addr inside {[PartInfo[LifeCycleIdx].offset : {OTP_ADDR_WIDTH{1'b1}}]}; + } else { + dai_addr inside `PART_ADDR_RANGE(LifeCycleIdx); + } + } + solve part_idx before dai_addr; + } + + constraint dai_wr_digests_c { + {dai_addr[TL_AW-1:2], 2'b0} dist { + { + VendorTestDigestOffset, + CreatorSwCfgDigestOffset, + OwnerSwCfgDigestOffset, + RotCreatorAuthDigestOffset, + RotOwnerAuthSlot0DigestOffset, + RotOwnerAuthSlot1DigestOffset, + PlatIntegAuthSlot0DigestOffset, + PlatIntegAuthSlot1DigestOffset, + PlatOwnerAuthSlot0DigestOffset, + PlatOwnerAuthSlot1DigestOffset, + PlatOwnerAuthSlot2DigestOffset, + PlatOwnerAuthSlot3DigestOffset, + RomPatchDigestOffset, + HwCfg0DigestOffset, + HwCfg1DigestOffset, + Secret0DigestOffset, + Secret1DigestOffset, + Secret2DigestOffset, + Secret3DigestOffset + } :/ 1, + [VendorTestOffset : '1] :/ 9 + }; + } + + virtual task pre_start(); + super.pre_start(); + is_valid_dai_op = 0; + endtask + + virtual task dut_init(string reset_kind = "HARD"); + super.dut_init(reset_kind); + if ($urandom_range(0, 1)) begin + cfg.otp_ctrl_vif.drive_lc_creator_seed_sw_rw_en(get_rand_lc_tx_val(.t_weight(0))); + end + if ($urandom_range(0, 1)) begin + cfg.otp_ctrl_vif.drive_lc_owner_seed_sw_rw_en(get_rand_lc_tx_val(.t_weight(0))); + end + endtask + +endclass + +`undef PART_ADDR_RANGE diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv new file mode 100644 index 00000000000000..6a5a2526f4ed41 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv @@ -0,0 +1,235 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// Otp_ctrl_init_fail_vseq is developed to check if otp_ctrl reacts correctly if initialization +// failed. +// Note that scoreboard is disabled in this test and all checks are done within this sequence. +// This test writes and reads to OTP_memory via DAI interface, then triggers digest calculations. +// Afterwards instead of issuing reset, this sequence continues to write to DAI interface. +// If any of the hardware partition is updated, then in next power cycle, the initialization will +// fail. +// If no check failure, we will random inject ECC errors to create init macro errors. +// We will also trigger sw partition ECC reg failure by forcing the ECC reg error output. +// +// This sequence will check the following items if OTP init failed with fatal error: +// - Otp_initialization failure triggers fatal alert +// - Status register reflect the correct error +// - Otp_ctrl's power init output stays 0 +// This sequence will check the following items if OTP init failed with correctable error: +// - Otp_initialtion passed with power init output changes to 1 +// - Otp status and interrupt reflect the correct error message + +class otp_ctrl_init_fail_vseq extends otp_ctrl_smoke_vseq; + `uvm_object_utils(otp_ctrl_init_fail_vseq) + + `uvm_object_new + + rand uint num_to_lock_digests; + bit [NumPart-1:0] init_chk_err; + bit part_locked; + + // If num_to_lock_digests is larger than num_dai_op, that means there won't be OTP init check + // error, so this sequence will trigger ECC error instead. + // We set 25% possibility that OTP init check fails due to writing OTP after digest is locked. + constraint lock_digest_c {num_to_lock_digests < num_dai_op * 4;} + constraint num_iterations_c {num_dai_op inside {[20:100]};} + constraint ecc_otp_err_c { + $countones(ecc_otp_err) dist {OtpNoEccErr :/ 2, + OtpEccCorrErr :/ 2, + OtpEccUncorrErr :/ 1}; + } + + virtual task pre_start(); + super.pre_start(); + num_to_lock_digests.rand_mode(0); + endtask + + task body(); + bit [TL_DW-1:0] exp_status; + `uvm_info(`gfn, $sformatf("Number of dai operation is %0d, number to lock digest is %0d", + num_dai_op, num_to_lock_digests), UVM_MEDIUM) + + for (uint i = 0; i <= num_dai_op; i++) begin + bit [TL_DW-1:0] tlul_val; + if (cfg.stop_transaction_generators()) return; + + `DV_CHECK_RANDOMIZE_FATAL(this) + `uvm_info(`gfn, $sformatf("starting dai access seq %0d/%0d with addr %0h in partition %0d", + i, num_dai_op, dai_addr, part_idx), UVM_MEDIUM) + + if (i > num_to_lock_digests && PartInfo[part_idx].hw_digest && + !used_dai_addrs.exists(dai_addr)) begin + init_chk_err[part_idx] = 1; + end + + // OTP write via DAI + dai_wr(dai_addr, wdata0, wdata1); + + // OTP read via DAI, check data in scb + dai_rd(dai_addr, wdata0, wdata1); + + // If write sw partitions, check tlul window + if (is_sw_part(dai_addr)) begin + uvm_reg_addr_t tlul_addr = cfg.ral.get_addr_from_offset(get_sw_window_offset(dai_addr)); + tl_access(.addr(tlul_addr), .write(0), .data(tlul_val), .blocking(1), .check_rsp(0)); + end + + if (i == num_to_lock_digests) begin + cal_hw_digests('1); + part_locked = 1; + end + + csr_rd(ral.status, tlul_val); + end + + do_otp_ctrl_init = 0; + do_otp_pwr_init = 0; + cfg.en_scb = 0; + dut_init(); + + if (init_chk_err) begin + `uvm_info(`gfn, $sformatf("OTP_init check failure with init error = %0h", init_chk_err), + UVM_LOW) + foreach(init_chk_err[i]) begin + if (cfg.stop_transaction_generators()) break; + if (init_chk_err[i]) exp_status |= 1'b1 << i; + end + + check_otp_fatal_err("fatal_check_error", exp_status); + + // If not check error, force ECC correctable and uncorrectable error + end else begin + bit is_fatal, is_correctable; + bit [TL_DW-1:0] addr; + + for (int i = 0; i < NumPart; i++) begin + if (cfg.stop_transaction_generators()) return; + `DV_CHECK_RANDOMIZE_FATAL(this); + + if (PartInfo[i].sw_digest) begin + // During OTP init, SW partitions only read digest value + addr = PART_OTP_DIGEST_ADDRS[i] << 2; + end else begin + // During OTP init, non SW partitions read all value + addr = $urandom_range(PartInfo[i].offset, PartInfo[i].offset + PartInfo[i].size - 1); + end + + void'(backdoor_inject_ecc_err(addr, ecc_otp_err)); + // VendorTest partition's ECC error is not fatal. + if (!is_fatal && ecc_otp_err == OtpEccUncorrErr && part_has_integrity(i)) begin + is_fatal = 1; + end else if (!is_correctable && ecc_otp_err == OtpEccCorrErr && part_has_integrity(i)) begin + is_correctable = 1; + end + if (ecc_otp_err != OtpNoEccErr && part_has_integrity(i)) exp_status[i] = 1; + end + + if (is_fatal) begin + // ECC uncorrectable error. + `uvm_info(`gfn, "OTP_init macro ECC uncorrectable failure", UVM_LOW) + check_otp_fatal_err("fatal_macro_error", exp_status); + end else if ($urandom_range(0, 1)) begin + + // Randomly force ECC reg in sw partitions to create a check failure. + // Totaly three sw partitions, and each bit indexes a partition. + bit [NumPartUnbuf-1:0] sw_check_fail = $urandom_range(1, (1'b1< 1 && do_dut_init) dut_init(); + // after otp-init done, check status + cfg.clk_rst_vif.wait_clks(1); + if (!cfg.otp_ctrl_vif.lc_esc_on) begin + csr_rd_check(.ptr(ral.status.dai_idle), .compare_value(1)); + end + end + do_otp_ctrl_init = 0; + + `DV_CHECK_RANDOMIZE_FATAL(this) + // set consistency and integrity checks + csr_wr(ral.check_regwen, check_regwen_val); + csr_wr(ral.check_trigger_regwen, check_trigger_regwen_val); + csr_wr(ral.check_timeout, check_timeout_val); + trigger_checks(.val(check_trigger_val), .wait_done(1), .ecc_err(ecc_chk_err)); + + if (!$urandom_range(0, 9) && access_locked_parts) write_sw_rd_locks(); + + // Backdoor write mubi to values that are not true or false. + force_mubi_part_access(); + + if (do_req_keys && !cfg.otp_ctrl_vif.alert_reqs) begin + req_otbn_key(); + req_flash_addr_key(); + req_flash_data_key(); + req_all_sram_keys(); + end + if (do_lc_trans && !cfg.otp_ctrl_vif.alert_reqs) begin + req_lc_transition(do_lc_trans, lc_prog_blocking); + if (cfg.otp_ctrl_vif.lc_prog_req == 0) begin + for (int k = 0; k <= LciIdx; k++) begin + csr_rd(.ptr(ral.err_code[k]), .value(tlul_val)); + end + end + end + + for (int i = 0; i < num_dai_op; i++) begin + bit [TL_DW-1:0] rdata0, rdata1, backdoor_rd_val; + if (cfg.stop_transaction_generators()) break; + + `DV_CHECK_RANDOMIZE_FATAL(this) + // recalculate part_idx in case some test turn off constraint dai_wr_legal_addr_c + part_idx = part_idx_e'(get_part_index(dai_addr)); + `uvm_info(`gfn, $sformatf("starting dai access seq %0d/%0d with addr %0h in partition %0d", + i, num_dai_op, dai_addr, part_idx), UVM_HIGH) + + // OTP write via DAI + if (rand_wr && !digest_calculated[part_idx]) begin + dai_wr(dai_addr, wdata0, wdata1); + if (cfg.otp_ctrl_vif.lc_prog_req == 0) begin + for (int k = 0; k <= LciIdx; k++) begin + csr_rd(.ptr(ral.err_code[k]), .value(tlul_val)); + end + end + end + + // Inject ECC error. + if (ecc_otp_err != OtpNoEccErr && dai_addr < LifeCycleOffset) begin + `uvm_info(`gfn, $sformatf("Injecting ecc error %0d at 0x%x", ecc_otp_err, dai_addr), + UVM_HIGH) + backdoor_rd_val = backdoor_inject_ecc_err(dai_addr, ecc_otp_err); + end + + if (rand_rd) begin + // OTP read via DAI, check data in scb + dai_rd(dai_addr, rdata0, rdata1); + end + + // if write sw partitions, check tlul window + if (is_sw_part(dai_addr) && rd_sw_tlul_rd) begin + uvm_reg_addr_t tlul_addr = cfg.ral.get_addr_from_offset(get_sw_window_offset(dai_addr)); + // tlul error rsp is checked in scoreboard + do_otp_rd = 1; + tl_access(.addr(tlul_addr), .write(0), .data(tlul_val), .blocking(1), .check_rsp(0)); + end + + // Backdoor restore injected ECC error, but should not affect fatal alerts. + if (ecc_otp_err != OtpNoEccErr && dai_addr < LifeCycleOffset) begin + `uvm_info(`gfn, $sformatf("Injecting ecc error %0d at 0x%x", ecc_otp_err, dai_addr), + UVM_HIGH) + cfg.mem_bkdr_util_h.write32({dai_addr[TL_DW-3:2], 2'b00}, backdoor_rd_val); + // Wait for two lock cycles to make sure the local escalation error propagates to other + // patitions and err_code reg. + cfg.clk_rst_vif.wait_clks(2); + end + + // Random lock sw partitions + if (!$urandom_range(0, 9) && access_locked_parts) write_sw_rd_locks(); + if (!$urandom_range(0, 9) && access_locked_parts) write_sw_digests(); + if ($urandom_range(0, 1)) csr_rd(.ptr(ral.direct_access_regwen), .value(tlul_val)); + if ($urandom_range(0, 1)) csr_rd(.ptr(ral.status), .value(tlul_val)); + if (cfg.otp_ctrl_vif.lc_prog_req == 0) begin + for (int k = 0; k <= LciIdx; k++) begin + csr_rd(.ptr(ral.err_code[k]), .value(tlul_val)); + end + end + end + + // Read/write test access memory + otp_test_access(); + + // lock digests + `uvm_info(`gfn, "Trigger HW digest calculation", UVM_HIGH) + cal_hw_digests(); + if ($urandom_range(0, 1)) csr_rd(.ptr(ral.status), .value(tlul_val)); + + if (cfg.otp_ctrl_vif.lc_prog_req == 0) begin + for (int k = 0; k <= LciIdx; k++) begin + csr_rd(.ptr(ral.err_code[k]), .value(tlul_val)); + end + end + + if ($urandom_range(0, 1)) rd_digests(); + if (do_dut_init) dut_init(); + + // read and check digest in scb + rd_digests(); + + // send request to the interfaces again after partitions are locked + if (do_lc_trans && !cfg.otp_ctrl_vif.alert_reqs) begin + req_lc_transition(do_lc_trans, lc_prog_blocking); + if (cfg.otp_ctrl_vif.lc_prog_req == 0) begin + for (int k = 0; k <= LciIdx; k++) begin + csr_rd(.ptr(ral.err_code[k]), .value(tlul_val)); + end + end + end + + if (do_req_keys && !cfg.otp_ctrl_vif.alert_reqs && !cfg.smoke_test) begin + req_otbn_key(); + req_flash_addr_key(); + req_flash_data_key(); + req_all_sram_keys(); + end + + end + + endtask : body + +endclass : otp_ctrl_smoke_vseq + +`undef PART_CONTENT_RANGE diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_stress_all_vseq.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_stress_all_vseq.sv new file mode 100644 index 00000000000000..031f5a15d20534 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_stress_all_vseq.sv @@ -0,0 +1,81 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// Combine all otp_ctrl seqs (except below seqs) in one seq to run sequentially. +// Exception: - csr seq: requires scb to be disabled +// - regwen_vseq and parallel_lc_vseq: time sensitive thus require zero_delays +// - macro_errs_vseq and check_fail_vseq: require to write back to OTP once fatal +// error is triggered, thus does not handle random reset +// - partition_walk_vseq: assume OTP initial value is 0 +// - init_fail: requires resets in the middle of the sequence +class otp_ctrl_stress_all_vseq extends otp_ctrl_base_vseq; + `uvm_object_utils(otp_ctrl_stress_all_vseq) + + string seq_names[]; + + `uvm_object_new + + virtual function void assign_seq_names(); + seq_names = {"otp_ctrl_common_vseq", + "otp_ctrl_dai_lock_vseq", + "otp_ctrl_smoke_vseq", + "otp_ctrl_test_access_vseq", + "otp_ctrl_background_chks_vseq", + "otp_ctrl_parallel_lc_esc_vseq", + "otp_ctrl_parallel_lc_req_vseq", + "otp_ctrl_parallel_key_req_vseq", + "otp_ctrl_dai_errs_vseq", + "otp_ctrl_low_freq_read_vseq"}; + endfunction + + task body(); + assign_seq_names(); + + for (int i = 1; i <= num_trans; i++) begin + uvm_sequence seq; + otp_ctrl_base_vseq otp_ctrl_vseq; + uint seq_idx = $urandom_range(0, seq_names.size - 1); + + seq = create_seq_by_name(seq_names[seq_idx]); + `downcast(otp_ctrl_vseq, seq) + `uvm_info(`gfn, $sformatf("Starting sequence %s in stress_all", otp_ctrl_vseq.get_name()), + UVM_MEDIUM) + + // At the end of each vseq, design might enter terminal Error State, need to reset to + // recover. If upper seq disables do_apply_reset for this seq, then can't issue reset + // as upper seq may drive reset. + if (do_apply_reset) otp_ctrl_vseq.do_apply_reset = 1; + else otp_ctrl_vseq.do_apply_reset = 0; + + otp_ctrl_vseq.set_sequencer(p_sequencer); + `DV_CHECK_RANDOMIZE_FATAL(otp_ctrl_vseq) + if (seq_names[seq_idx] == "otp_ctrl_common_vseq") begin + otp_ctrl_common_vseq common_vseq; + `downcast(common_vseq, otp_ctrl_vseq); + common_vseq.common_seq_type = "intr_test"; + end + + // Pass local variables to next sequence due to randomly issued reset. + otp_ctrl_vseq.is_valid_dai_op = 0; + otp_ctrl_vseq.lc_prog_blocking = this.lc_prog_blocking; + otp_ctrl_vseq.digest_calculated = this.digest_calculated; + otp_ctrl_vseq.start(p_sequencer); + + this.lc_prog_blocking = otp_ctrl_vseq.lc_prog_blocking; + this.digest_calculated = otp_ctrl_vseq.digest_calculated; + + // This is for otp_ctrl_stress_all_with_rand_reset. + // We need to reset for each vseq, but in otp_ctrl_stress_all_with_rand_reset, reset should be + // issued in upper seq. So, wait forever until reset is issued and this vseq is killed by + // upper seq. + if (!do_apply_reset) wait(0); + + // This is only valid for stress_all sequence. + // For stress_all_with_rand_reset sequence, the logic will be gated at previous line and will + // enable scb again at `apply_resets_concurrently` task. + cfg.en_scb = 1; + end + endtask : body + +endclass diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_test_access_vseq.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_test_access_vseq.sv new file mode 100644 index 00000000000000..fbdb51f23a628d --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_test_access_vseq.sv @@ -0,0 +1,44 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +class otp_ctrl_test_access_vseq extends otp_ctrl_dai_lock_vseq; + `uvm_object_utils(otp_ctrl_test_access_vseq) + + `uvm_object_new + + virtual task dut_init(string reset_kind = "HARD"); + super.dut_init(reset_kind); + + // Drive dft_en pins to access the test_access memory. + cfg.otp_ctrl_vif.drive_lc_dft_en(lc_ctrl_pkg::On); + + // Once turn on lc_dft_en register, will need some time to update the state register + // two clock cycles for lc_async mode, one clock cycle for driving dft_en. + if (cfg.en_dv_cdc) cfg.clk_rst_vif.wait_clks(4); + else cfg.clk_rst_vif.wait_clks(3); + endtask + + // Avoid back-to-back lc_dft_en initializations with random values since CDC can cause + // temporary On values when transitioning between arbitrary values. This is not a realistic + // attack mode since + // - The value is always driven as On or Off in hardware, + // - Mubi cannot protect against all bits being glitched, + // - getting an ON value due to CDC requires the mubi values before and after the CDC transition + // to have close enough Hamming distance to On, and is as hard or harder than glitching all + // bits to On. + virtual task rand_drive_dft_en(); + static bit phase; + super.rand_drive_dft_en(); + // 25% chance drive lc_dft_en to a random value. + if ($urandom_range(0, 3) == 3) begin + cfg.otp_ctrl_vif.drive_lc_dft_en(get_rand_lc_tx_val( + .t_weight(1), .f_weight(1), .other_weight(phase))); + phase = !phase; + // Once turn on lc_dft_en regiser, will need some time to update the state register + // two clock cycles for lc_async mode, one clock cycle for driving dft_en. + if (cfg.en_dv_cdc) cfg.clk_rst_vif.wait_clks(4); + else cfg.clk_rst_vif.wait_clks(3); + end + endtask +endclass diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_vseq_list.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_vseq_list.sv new file mode 100644 index 00000000000000..3cf96c1b521805 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_vseq_list.sv @@ -0,0 +1,24 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +`include "otp_ctrl_callback_vseq.sv" +`include "otp_ctrl_base_vseq.sv" +`include "otp_ctrl_wake_up_vseq.sv" +`include "otp_ctrl_smoke_vseq.sv" +`include "otp_ctrl_common_vseq.sv" +`include "otp_ctrl_partition_walk_vseq.sv" +`include "otp_ctrl_low_freq_read_vseq.sv" +`include "otp_ctrl_init_fail_vseq.sv" +`include "otp_ctrl_dai_lock_vseq.sv" +`include "otp_ctrl_dai_errs_vseq.sv" +`include "otp_ctrl_macro_errs_vseq.sv" +`include "otp_ctrl_background_chks_vseq.sv" +`include "otp_ctrl_check_fail_vseq.sv" +`include "otp_ctrl_parallel_base_vseq.sv" +`include "otp_ctrl_parallel_key_req_vseq.sv" +`include "otp_ctrl_parallel_lc_req_vseq.sv" +`include "otp_ctrl_parallel_lc_esc_vseq.sv" +`include "otp_ctrl_regwen_vseq.sv" +`include "otp_ctrl_test_access_vseq.sv" +`include "otp_ctrl_stress_all_vseq.sv" diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_wake_up_vseq.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_wake_up_vseq.sv new file mode 100644 index 00000000000000..8a048e5e560f68 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_wake_up_vseq.sv @@ -0,0 +1,56 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +class otp_ctrl_wake_up_vseq extends otp_ctrl_base_vseq; + `uvm_object_utils(otp_ctrl_wake_up_vseq) + + `uvm_object_new + + virtual task otp_ctrl_init(); + super.otp_ctrl_init(); + csr_wr(ral.intr_enable, en_intr); + endtask + + task body(); + bit [TL_DW-1:0] rand_addr = $urandom_range(CreatorSwCfgOffset, + CreatorSwCfgOffset + CreatorSwCfgSize); + // check status + cfg.clk_rst_vif.wait_clks(1); + + // turn on all intr_enable + csr_wr(ral.intr_enable, (1'b1 << NumOtpCtrlIntr) - 1); + csr_rd_check(.ptr(ral.status.dai_idle), .compare_value(1)); + + // write seq + csr_wr(ral.direct_access_address, rand_addr); + csr_wr(ral.direct_access_wdata[0], '1); + csr_wr(ral.direct_access_cmd, 2); + wait(cfg.intr_vif.pins[OtpOperationDone] == 1); + csr_wr(ral.intr_state, 1'b1 << OtpOperationDone); + + // read seq + csr_wr(ral.direct_access_address, rand_addr); + csr_wr(ral.direct_access_cmd, 1); + wait(cfg.intr_vif.pins[OtpOperationDone] == 1); + csr_rd_check(.ptr(ral.direct_access_rdata[0]), .compare_value('1)); + csr_wr(ral.intr_state, 1'b1 << OtpOperationDone); + + // digest sw error seq + csr_wr(ral.direct_access_address, CreatorSwCfgOffset + 2); + csr_wr(ral.direct_access_cmd, 4); + wait(cfg.intr_vif.pins[OtpOperationDone] == 1); + wait(cfg.intr_vif.pins[OtpErr] == 1); + csr_wr(ral.intr_state, (1'b1 << NumOtpCtrlIntr) - 1); + + // digest hw seq + csr_wr(ral.direct_access_address, HwCfg0DigestOffset); + csr_wr(ral.direct_access_cmd, 4); + wait(cfg.intr_vif.pins[OtpOperationDone] == 1); + csr_wr(ral.intr_state, 1'b1 << OtpOperationDone); + + // check all interrupts are cleared + csr_rd_check(.ptr(ral.intr_state), .compare_value(0)); + endtask : body + +endclass : otp_ctrl_wake_up_vseq diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/otp_ctrl_sim.core b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/otp_ctrl_sim.core new file mode 100644 index 00000000000000..b1ff3ab09e0772 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/otp_ctrl_sim.core @@ -0,0 +1,37 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: lowrisc:opentitan:top_darjeeling_otp_ctrl_sim:0.1 +description: "OTP_CTRL DV sim target" +virtual: + - lowrisc:ip_interfaces:otp_ctrl_sim + +filesets: + files_rtl: + depend: + - lowrisc:opentitan:top_darjeeling_otp_ctrl + + files_dv: + depend: + - lowrisc:dv:mem_bkdr_util + - lowrisc:opentitan:top_darjeeling_otp_ctrl_test + - lowrisc:opentitan:top_darjeeling_otp_ctrl_sva + - lowrisc:opentitan:top_darjeeling_otp_ctrl_cov + files: + - tb.sv + file_type: systemVerilogSource + +targets: + default: &default_target + toplevel: tb + filesets: + - files_rtl + - files_dv + + sim: + <<: *default_target + default_tool: vcs + + lint: + <<: *default_target diff --git a/hw/ip/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson similarity index 90% rename from hw/ip/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson rename to hw/top_darjeeling/ip_autogen/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson index 2b2333f040dd9f..62f35bc4fa6f64 100644 --- a/hw/ip/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson @@ -15,13 +15,13 @@ tool: vcs // Fusesoc core file used for building the file list. - fusesoc_core: lowrisc:dv:otp_ctrl_sim:0.1 + fusesoc_core: lowrisc:opentitan:top_darjeeling_otp_ctrl_sim:0.1 // Testplan hjson file. - testplan: "{proj_root}/hw/ip/otp_ctrl/data/otp_ctrl_testplan.hjson" + testplan: "{self_dir}/../data/otp_ctrl_testplan.hjson" // RAL spec - used to generate the RAL model. - ral_spec: "{proj_root}/hw/ip/otp_ctrl/data/otp_ctrl.hjson" + ral_spec: "{self_dir}/../data/otp_ctrl.hjson" // Import additional common sim cfg files. import_cfgs: [// Project wide common sim cfg file @@ -44,7 +44,7 @@ // defined in `hw/dv/tools/dvsim/common_modes.hjson` for more details. { name: build_seed - pre_build_cmds: ["cd {proj_root} && ./util/design/gen-otp-mmap.py --seed {seed}"] + pre_build_cmds: ["cd {proj_root} && ./util/design/gen-otp-mmap.py --seed {seed} --topname darjeeling"] is_sim_mode: 1 } ] @@ -60,13 +60,13 @@ reseed: 50 // Add OTP_CTRL specific exclusion files. - vcs_cov_excl_files: ["{proj_root}/hw/ip/otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el", - "{proj_root}/hw/ip/otp_ctrl/dv/cov/otp_ctrl_cov_fsm_unr_excl.el"] + vcs_cov_excl_files: ["{self_dir}/cov/otp_ctrl_cov_unr_excl.el", + "{self_dir}/cov/otp_ctrl_cov_fsm_unr_excl.el"] overrides: [ { name: default_vcs_cov_cfg_file - value: "-cm_hier {dv_root}/tools/vcs/cover.cfg+{dv_root}/tools/vcs/common_cov_excl.cfg+{proj_root}/hw/ip/otp_ctrl/dv/cov/otp_ctrl_cover.cfg" + value: "-cm_hier {dv_root}/tools/vcs/cover.cfg+{dv_root}/tools/vcs/common_cov_excl.cfg+{self_dir}/cov/otp_ctrl_cover.cfg" } ] diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/sva/otp_ctrl_bind.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/sva/otp_ctrl_bind.sv new file mode 100644 index 00000000000000..833c710797e624 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/sva/otp_ctrl_bind.sv @@ -0,0 +1,32 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +module otp_ctrl_bind; + + bind otp_ctrl tlul_assert #( + .EndpointType("Device") + ) core_tlul_assert_device ( + .clk_i, + .rst_ni, + .h2d (core_tl_i), + .d2h (core_tl_o) + ); + + bind otp_ctrl tlul_assert #( + .EndpointType("Device") + ) prim_tlul_assert_device ( + .clk_i, + .rst_ni, + .h2d (prim_tl_i), + .d2h (prim_tl_o) + ); + + bind otp_ctrl otp_ctrl_core_csr_assert_fpv otp_ctrl_core_csr_assert ( + .clk_i, + .rst_ni, + .h2d (core_tl_i), + .d2h (core_tl_o) + ); + +endmodule diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/sva/otp_ctrl_sva.core b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/sva/otp_ctrl_sva.core new file mode 100644 index 00000000000000..8f1360da8ed0a8 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/sva/otp_ctrl_sva.core @@ -0,0 +1,41 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: lowrisc:opentitan:top_darjeeling_otp_ctrl_sva:0.1 +description: "OTP_CTRL assertion modules and bind file." +virtual: + - lowrisc:ip_interfaces:otp_ctrl_sva + +filesets: + files_dv: + depend: + - lowrisc:tlul:headers + - lowrisc:fpv:csr_assert_gen + files: + - otp_ctrl_bind.sv + file_type: systemVerilogSource + + files_formal: + depend: + - lowrisc:opentitan:top_darjeeling_otp_ctrl + +generate: + csr_assert_gen: + generator: csr_assert_gen + parameters: + spec: ../../data/otp_ctrl.hjson + +targets: + default: &default_target + filesets: + - files_dv + generate: + - csr_assert_gen + + formal: + <<: *default_target + filesets: + - files_formal + - files_dv + toplevel: otp_ctrl diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tb.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tb.sv new file mode 100644 index 00000000000000..1daafcc30e5595 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tb.sv @@ -0,0 +1,238 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +module tb; + // dep packages + import uvm_pkg::*; + import dv_utils_pkg::*; + import otp_ctrl_env_pkg::*; + import otp_ctrl_test_pkg::*; + import otp_ctrl_reg_pkg::*; + import mem_bkdr_util_pkg::mem_bkdr_util; + + // macro includes + `include "uvm_macros.svh" + `include "dv_macros.svh" + + // TB base test ENV_T & CFG_T specification + // + // Specify the parameters for the otp_ctrl_base_test + // This will invoke the UVM registry and link this test type to + // the name 'otp_ctrl_base_test' as a test name passed by UVM_TESTNAME + // + // This is done explicitly only for the prim_pkg::ImplGeneric implementation + // since partner base tests inherit from otp_ctrl_base_test#(CFG_T, ENV_T) and + // specify directly (CFG_T, ENV_T) via the class extension and use a different + // UVM_TESTNAME + if (`PRIM_DEFAULT_IMPL == prim_pkg::ImplGeneric) begin : gen_spec_base_test_params + typedef otp_ctrl_base_test #(.CFG_T(otp_ctrl_env_cfg), + .ENV_T(otp_ctrl_env)) otp_ctrl_base_test_t; + end + + wire clk, rst_n; + wire otp_ctrl_pkg::flash_otp_key_req_t flash_req; + wire otp_ctrl_pkg::flash_otp_key_rsp_t flash_rsp; + wire otp_ctrl_pkg::otbn_otp_key_req_t otbn_req; + wire otp_ctrl_pkg::otbn_otp_key_rsp_t otbn_rsp; + wire otp_ctrl_pkg::sram_otp_key_req_t[NumSramKeyReqSlots-1:0] sram_req; + wire otp_ctrl_pkg::sram_otp_key_rsp_t[NumSramKeyReqSlots-1:0] sram_rsp; + + wire [NUM_MAX_INTERRUPTS-1:0] interrupts; + wire intr_otp_operation_done, intr_otp_error; + + // Output from close-source OTP, not checked in open-source env. + wire otp_ctrl_pkg::otp_ast_req_t ast_req; + wire [7:0] otp_obs_o; + + tlul_pkg::tl_d2h_t prim_tl_o; + + // interfaces + clk_rst_if clk_rst_if(.clk(clk), .rst_n(rst_n)); + pins_if #(NUM_MAX_INTERRUPTS) intr_if(interrupts); + + // lc_otp interfaces + push_pull_if #(.HostDataWidth(LC_PROG_DATA_SIZE), .DeviceDataWidth(1)) + lc_prog_if(.clk(clk), .rst_n(rst_n)); + push_pull_if #(.DeviceDataWidth(SRAM_DATA_SIZE)) + sram_if[NumSramKeyReqSlots](.clk(clk), .rst_n(rst_n)); + push_pull_if #(.DeviceDataWidth(OTBN_DATA_SIZE)) otbn_if(.clk(clk), .rst_n(rst_n)); + push_pull_if #(.DeviceDataWidth(FLASH_DATA_SIZE)) flash_addr_if(.clk(clk), .rst_n(rst_n)); + push_pull_if #(.DeviceDataWidth(FLASH_DATA_SIZE)) flash_data_if(.clk(clk), .rst_n(rst_n)); + + tl_if tl_if(.clk(clk), .rst_n(rst_n)); + tl_if prim_tl_if(.clk(clk), .rst_n(rst_n)); + + otp_ctrl_if otp_ctrl_if(.clk_i(clk), .rst_ni(rst_n)); + + `DV_ALERT_IF_CONNECT() + + // edn_clk, edn_rst_n and edn_if are defined and driven in below macro + `DV_EDN_IF_CONNECT + + assign otp_ctrl_if.lc_prog_req = lc_prog_if.req; + assign otp_ctrl_if.lc_prog_err = lc_prog_if.d_data; + + // Assign to otp_ctrl_if for assertion checks. + assign otp_ctrl_if.lc_prog_ack = lc_prog_if.ack; + assign otp_ctrl_if.flash_acks = flash_data_if.ack; + assign otp_ctrl_if.otbn_ack = otbn_if.ack; + + // This signal probes design's alert request to avoid additional logic for triggering alert and + // disable assertions. + // Alert checkings are done independently in otp_ctrl's scb. + // The correctness of this probed signal is checked in otp_ctrl's scb as well. + assign otp_ctrl_if.alert_reqs = dut.alerts[0] | dut.alerts[1]; + + // connected to interface + wire otp_ext_voltage_h = otp_ctrl_if.ext_voltage_h_io; + + // dut + otp_ctrl dut ( + .clk_i (clk ), + .rst_ni (rst_n ), + // edn + .clk_edn_i (edn_clk ), + .rst_edn_ni (edn_rst_n ), + .edn_o (edn_if[0].req ), + .edn_i ({edn_if[0].ack, edn_if[0].d_data}), + // bus interfaces + .core_tl_i (tl_if.h2d ), + .core_tl_o (tl_if.d2h ), + .prim_tl_i (prim_tl_if.h2d), + .prim_tl_o (prim_tl_if.d2h), + // interrupt + .intr_otp_operation_done_o (intr_otp_operation_done), + .intr_otp_error_o (intr_otp_error), + // alert + .alert_rx_i (alert_rx ), + .alert_tx_o (alert_tx ), + // ast + .obs_ctrl_i (otp_ctrl_if.obs_ctrl_i), + .otp_obs_o (otp_obs_o), + .otp_ast_pwr_seq_o (ast_req), + .otp_ast_pwr_seq_h_i (otp_ctrl_if.otp_ast_pwr_seq_h_i), + // pwrmgr + .pwr_otp_i (otp_ctrl_if.pwr_otp_init_i), + .pwr_otp_o ({otp_ctrl_if.pwr_otp_done_o, otp_ctrl_if.pwr_otp_idle_o}), + // lc + .lc_otp_vendor_test_i (otp_ctrl_if.otp_vendor_test_ctrl_i), + .lc_otp_vendor_test_o (otp_ctrl_if.otp_vendor_test_status_o), + .lc_otp_program_i ({lc_prog_if.req, lc_prog_if.h_data}), + .lc_otp_program_o ({lc_prog_if.d_data, lc_prog_if.ack}), + .lc_creator_seed_sw_rw_en_i (otp_ctrl_if.lc_creator_seed_sw_rw_en_i), + .lc_owner_seed_sw_rw_en_i (otp_ctrl_if.lc_owner_seed_sw_rw_en_i), + .lc_seed_hw_rd_en_i (otp_ctrl_if.lc_seed_hw_rd_en_i), + .lc_dft_en_i (otp_ctrl_if.lc_dft_en_i), + .lc_escalate_en_i (otp_ctrl_if.lc_escalate_en_i), + .lc_check_byp_en_i (otp_ctrl_if.lc_check_byp_en_i), + .otp_lc_data_o (otp_ctrl_if.lc_data_o), + // keymgr + .otp_keymgr_key_o (otp_ctrl_if.keymgr_key_o), + // flash + .flash_otp_key_i (flash_req), + .flash_otp_key_o (flash_rsp), + // sram + .sram_otp_key_i (sram_req), + .sram_otp_key_o (sram_rsp), + // otbn + .otbn_otp_key_i (otbn_req), + .otbn_otp_key_o (otbn_rsp), + + .otp_broadcast_o (otp_ctrl_if.otp_broadcast_o), + .otp_ext_voltage_h_io (otp_ext_voltage_h), + + //scan + .scan_en_i (otp_ctrl_if.scan_en_i), + .scan_rst_ni (otp_ctrl_if.scan_rst_ni), + .scanmode_i (otp_ctrl_if.scanmode_i), + + // Test-related GPIO output + .cio_test_o (otp_ctrl_if.cio_test_o), + .cio_test_en_o (otp_ctrl_if.cio_test_en_o) + ); + + for (genvar i = 0; i < NumSramKeyReqSlots; i++) begin : gen_sram_pull_if + assign sram_req[i] = sram_if[i].req; + assign sram_if[i].ack = sram_rsp[i].ack; + assign sram_if[i].d_data = {sram_rsp[i].key, sram_rsp[i].nonce, sram_rsp[i].seed_valid}; + assign otp_ctrl_if.sram_acks[i] = sram_rsp[i].ack; + initial begin + uvm_config_db#(virtual push_pull_if#(.DeviceDataWidth(SRAM_DATA_SIZE)))::set(null, + $sformatf("*env.m_sram_pull_agent[%0d]*", i), "vif", sram_if[i]); + end + end + assign otbn_req = otbn_if.req; + assign otbn_if.ack = otbn_rsp.ack; + assign otbn_if.d_data = {otbn_rsp.key, otbn_rsp.nonce, otbn_rsp.seed_valid}; + + assign flash_req = {flash_data_if.req, flash_addr_if.req}; + assign flash_data_if.ack = flash_rsp.data_ack; + assign flash_addr_if.ack = flash_rsp.addr_ack; + assign flash_data_if.d_data = {flash_rsp.key, flash_rsp.seed_valid}; + assign flash_addr_if.d_data = {flash_rsp.key, flash_rsp.seed_valid}; + + assign interrupts[OtpOperationDone] = intr_otp_operation_done; + assign interrupts[OtpErr] = intr_otp_error; + + // Instantitate the memory backdoor util instance only for OS implementation + // Proprietary IP will instantiate their own backdoor util + + if (`PRIM_DEFAULT_IMPL == prim_pkg::ImplGeneric) begin : gen_impl_generic + `define MEM_MODULE_PATH \ + tb.dut.u_otp.gen_generic.u_impl_generic.u_prim_ram_1p_adv.gen_ram_inst[0] + + `define MEM_ARRAY_PATH \ + `MEM_MODULE_PATH.u_mem.gen_generic.u_impl_generic.mem + + initial begin : mem_bkdr_util_gen + mem_bkdr_util m_mem_bkdr_util; + m_mem_bkdr_util = new(.name("mem_bkdr_util"), + .path(`DV_STRINGIFY(`MEM_ARRAY_PATH)), + .depth($size(`MEM_ARRAY_PATH)), + .n_bits($bits(`MEM_ARRAY_PATH)), + .err_detection_scheme(mem_bkdr_util_pkg::EccHamming_22_16)); + + uvm_config_db#(mem_bkdr_util)::set(null, "*.env", "mem_bkdr_util", m_mem_bkdr_util); + end : mem_bkdr_util_gen + + `undef MEM_ARRAY_PATH + `undef MEM_MODULE_PATH + end : gen_impl_generic + + // DV forced otp_cmd_i to reach invalid state, thus violate the assertions + for (genvar idx = 0; idx < NumPart; idx++) begin : gen_assertoff_loop + if (is_hw_part_idx(idx)) begin : gen_assertoff + initial begin + $assertoff(0, tb.dut.gen_partitions[idx].gen_buffered.u_part_buf.OtpErrorState_A); + end + end + end + + initial begin + // drive clk and rst_n from clk_if + clk_rst_if.set_active(); + uvm_config_db#(virtual clk_rst_if)::set(null, "*.env", "clk_rst_vif", clk_rst_if); + uvm_config_db#(virtual clk_rst_if)::set(null, "*.env", + "clk_rst_vif_otp_ctrl_prim_reg_block", clk_rst_if); + uvm_config_db#(virtual tl_if)::set(null, "*.env.m_tl_agent_otp_ctrl_core_reg_block*", + "vif", tl_if); + uvm_config_db#(virtual tl_if)::set(null, "*.env.m_tl_agent_otp_ctrl_prim_reg_block", + "vif", prim_tl_if); + uvm_config_db#(virtual push_pull_if#(.DeviceDataWidth(OTBN_DATA_SIZE)))::set(null, + "*env.m_otbn_pull_agent*", "vif", otbn_if); + uvm_config_db#(virtual push_pull_if#(.DeviceDataWidth(FLASH_DATA_SIZE)))::set(null, + "*env.m_flash_data_pull_agent*", "vif", flash_data_if); + uvm_config_db#(virtual push_pull_if#(.DeviceDataWidth(FLASH_DATA_SIZE)))::set(null, + "*env.m_flash_addr_pull_agent*", "vif", flash_addr_if); + uvm_config_db#(virtual push_pull_if#(.HostDataWidth(LC_PROG_DATA_SIZE), .DeviceDataWidth(1))):: + set(null, "*env.m_lc_prog_pull_agent*", "vif", lc_prog_if); + + uvm_config_db#(intr_vif)::set(null, "*.env", "intr_vif", intr_if); + + uvm_config_db#(virtual otp_ctrl_if)::set(null, "*.env", "otp_ctrl_vif", otp_ctrl_if); + $timeformat(-12, 0, " ps", 12); + run_test(); + end + +endmodule diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tests/otp_ctrl_base_test.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tests/otp_ctrl_base_test.sv new file mode 100644 index 00000000000000..284c5e0cdfe724 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tests/otp_ctrl_base_test.sv @@ -0,0 +1,51 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +class otp_ctrl_base_test #( + type CFG_T = otp_ctrl_env_cfg, + type ENV_T = otp_ctrl_env + ) extends cip_base_test #( + .CFG_T(CFG_T), + .ENV_T(ENV_T) + ); + + // A prototype for the registry to associate the parameterized base test + // with the name 'otp_ctrl_base_test' + // + // Register the name 'otp_ctrl_base_test' with the UVM factory to be associated + // with the template base test class parameterized with the default types (see + // declaration. We cannot invoke the standard UVM factory automation macro t + // (uvm_component_param_utils) to register a parameterized test class with the + // factory because the creation of the test by name (via the UVM_TESTNAME + // plusarg) does not work. We expand the contents of the automation macro + // here instead. See the following paper for details: + // https://verificationacademy-news.s3.amazonaws.com/DVCon2016/Papers/ + // dvcon-2016_paramaters-uvm-coverage-and-emulation-take-two-and-call-me-in-the-morning_paper.pdf + typedef uvm_component_registry#(otp_ctrl_base_test#(CFG_T, ENV_T), "otp_ctrl_base_test") type_id; + + // functions to support the component registry above + static function type_id get_type(); + return type_id::get(); + endfunction : get_type + + virtual function uvm_object_wrapper get_object_type(); + return type_id::get(); + endfunction : get_object_type + + const static string type_name = "otp_ctrl_base_test"; + + virtual function string get_type_name(); + return type_name; + endfunction : get_type_name + + `uvm_component_new + + // the base class dv_base_test creates the following instances: + // otp_ctrl_env_cfg: cfg + // otp_ctrl_env: env + + // the base class also looks up UVM_TEST_SEQ plusarg to create and run that seq in + // the run_phase; as such, nothing more needs to be done + +endclass : otp_ctrl_base_test diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tests/otp_ctrl_test.core b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tests/otp_ctrl_test.core new file mode 100644 index 00000000000000..c2591a8478b776 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tests/otp_ctrl_test.core @@ -0,0 +1,22 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: lowrisc:opentitan:top_darjeeling_otp_ctrl_test:0.1 +description: "OTP_CTRL DV UVM test" +virtual: + - lowrisc:ip_interfaces:otp_ctrl_test + +filesets: + files_dv: + depend: + - lowrisc:opentitan:top_darjeeling_otp_ctrl_env + files: + - otp_ctrl_test_pkg.sv + - otp_ctrl_base_test.sv: {is_include_file: true} + file_type: systemVerilogSource + +targets: + default: + filesets: + - files_dv diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tests/otp_ctrl_test_pkg.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tests/otp_ctrl_test_pkg.sv new file mode 100644 index 00000000000000..3d5703d49e6e29 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/dv/tests/otp_ctrl_test_pkg.sv @@ -0,0 +1,22 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +package otp_ctrl_test_pkg; + // dep packages + import uvm_pkg::*; + import cip_base_pkg::*; + import otp_ctrl_env_pkg::*; + + // macro includes + `include "uvm_macros.svh" + `include "dv_macros.svh" + + // local types + + // functions + + // package sources + `include "otp_ctrl_base_test.sv" + +endpackage diff --git a/hw/ip/otp_ctrl/lint/otp_ctrl_pkg.vbl b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl.vbl similarity index 87% rename from hw/ip/otp_ctrl/lint/otp_ctrl_pkg.vbl rename to hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl.vbl index 01a79ec02a146c..33b74bcc9541c1 100644 --- a/hw/ip/otp_ctrl/lint/otp_ctrl_pkg.vbl +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl.vbl @@ -2,7 +2,7 @@ # Licensed under the Apache License, Version 2.0, see LICENSE for details. # SPDX-License-Identifier: Apache-2.0 # -# waiver file for OTP controller package +# waiver file for OTP controller # waive long line violations in generated code waive --rule=line-length --location="otp_ctrl_part_pkg.sv" diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl.vlt b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl.vlt new file mode 100644 index 00000000000000..0f4180cb5e3d56 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl.vlt @@ -0,0 +1,21 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// waiver file for OTP controller + +`verilator_config + +// Some code in this block checks that an address is greater than or equal to +// the offset of a fixed part_info_t (supplied via a parameter). If that +// part_info_t happens to have a zero offset, Verilator warns that the +// comparison is always true. Waive the warning. +lint_off -rule UNSIGNED -file "*/rtl/otp_ctrl.sv" +lint_off -rule UNSIGNED -file "*/rtl/otp_ctrl_dai.sv" +lint_off -rule UNSIGNED -file "*/rtl/otp_ctrl_part_unbuf.sv" + +// In otp_ctrl_scrmbl, there are some comparisons between the "sel_i" signal +// and LastDigestSet. If the parameter NumDigestSets is a power of 2 (which it +// is at the moment), these checks will always be true, causing a Verilator +// warning. +lint_off -rule CMPCONST -file "*/rtl/otp_ctrl_scrmbl.sv" diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl.waiver b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl.waiver new file mode 100644 index 00000000000000..b97042a7aec339 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl.waiver @@ -0,0 +1,26 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# waiver file for OTP controller + +waive -rules {TERMINAL_STATE} -location {otp_ctrl_dai.sv \ + otp_ctrl_lci.sv \ + otp_ctrl_lfsr_timer.sv \ + otp_ctrl_part_buf.sv \ + otp_ctrl_part_unbuf.sv \ + otp_ctrl_scrmbl.sv} \ + -msg {Terminal state 'ErrorSt' is detected. Next state register 'state_d' is not assigned in this state.} \ + -comment "All these FSMs have a valid, terminal error state." + +waive -rules {INVALID_COMPARE} -location {otp_ctrl_dai.sv} -regexp {.*dai_addr_i >= PartInfo\[0\]\.offset.*} \ + -comment "This invalid compare is due to the first partition offset being zero." + +waive -rules {INVALID_COMPARE} -location {otp_ctrl.sv} -regexp {.*tlul_addr.* >= PartInfo\[0\]\.offset.*} \ + -comment "This invalid compare is due to the first partition offset being zero." + +waive -rules {LOOP_VAR_OP FOR_LOOP_BOUNDS} -location {prim_cipher_pkg.sv} -regexp {Loop.*round_cnt.*constant.*} \ + -comment "This function needs to iterate over the key schedule function in order to derive the decryption key." + +waive -rules {ASSIGN_SIGN} -location {otp_ctrl_part_buf.sv} -regexp {Unsigned target 'step_i' assigned signed value 'CntWidth'\(1\)'} \ + -comment "This is fine as long as CntWidth is greater equal 2 which is checked using an SVA." diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vbl b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vbl new file mode 100644 index 00000000000000..709cf2bb305769 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vbl @@ -0,0 +1,5 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# verible waiver file for OTP controller generic package diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vlt b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vlt new file mode 100644 index 00000000000000..fefa0f75baad1d --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vlt @@ -0,0 +1,5 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// waiver file for OTP controller generic package diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.waiver b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.waiver new file mode 100644 index 00000000000000..e0846938574c63 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.waiver @@ -0,0 +1,5 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# ascentlint waiver file for OTP controller generic package diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vbl b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vbl new file mode 100644 index 00000000000000..dadab89d18ef1e --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vbl @@ -0,0 +1,8 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# waiver file for OTP controller top-specific package + +# waive long line violations in generated code +waive --rule=line-length --location="otp_ctrl_part_pkg.sv" diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vlt b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vlt new file mode 100644 index 00000000000000..06603a4760cf74 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vlt @@ -0,0 +1,5 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// waiver file for OTP controller top-specific package diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.waiver b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.waiver new file mode 100644 index 00000000000000..69259ce4b16170 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.waiver @@ -0,0 +1,9 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# waiver file for OTP controller top-specific package + +# waive line length violations in generated code +waive -rules LINE_LENGTH -location {otp_ctrl_part_pkg.sv} \ + -comment "Waive line length violations in generated code." diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl.core b/hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl.core new file mode 100644 index 00000000000000..0c473f039719b8 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl.core @@ -0,0 +1,106 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: lowrisc:opentitan:top_darjeeling_otp_ctrl:1.0 +description: "OTP Controller" +virtual: + - lowrisc:ip_interfaces:otp_ctrl + +filesets: + files_rtl: + depend: + - lowrisc:ip:otp_ctrl_pkg + - lowrisc:opentitan:top_darjeeling_otp_ctrl_top_specific_pkg + - lowrisc:ip:tlul + - lowrisc:prim:all + - lowrisc:prim:ram_1p + - lowrisc:prim:otp + - lowrisc:prim:double_lfsr + - lowrisc:prim:count + - lowrisc:prim:lc_sender + - lowrisc:prim:lc_sync + - lowrisc:prim:buf + - lowrisc:prim:flop + - lowrisc:prim:secded + - lowrisc:prim:edn_req + - lowrisc:prim:sec_anchor + - lowrisc:ip_interfaces:pwrmgr_pkg + - lowrisc:ip:edn_pkg + - lowrisc:prim:sparse_fsm + - "fileset_partner ? (partner:systems:ast_pkg)" + - "!fileset_partner ? (lowrisc:systems:ast_pkg)" + files: + - rtl/otp_ctrl_core_reg_top.sv + - rtl/otp_ctrl_ecc_reg.sv + - rtl/otp_ctrl_scrmbl.sv + - rtl/otp_ctrl_lfsr_timer.sv + - rtl/otp_ctrl_part_unbuf.sv + - rtl/otp_ctrl_part_buf.sv + - rtl/otp_ctrl_dai.sv + - rtl/otp_ctrl_kdi.sv + - rtl/otp_ctrl_lci.sv + - rtl/otp_ctrl.sv + file_type: systemVerilogSource + + files_verilator_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl.vlt + file_type: vlt + + files_ascentlint_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl.waiver + file_type: waiver + + files_veriblelint_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl.vbl + file_type: veribleLintWaiver + +parameters: + SYNTHESIS: + datatype: bool + paramtype: vlogdefine + + +targets: + default: &default_target + filesets: + - tool_verilator ? (files_verilator_waiver) + - tool_ascentlint ? (files_ascentlint_waiver) + - tool_veriblelint ? (files_veriblelint_waiver) + - files_rtl + toplevel: otp_ctrl + + lint: + <<: *default_target + default_tool: verilator + parameters: + - SYNTHESIS=true + tools: + verilator: + mode: lint-only + verilator_options: + - "-Wall" + + syn: + <<: *default_target + # TODO: set default to DC once + # this option is available + # olofk/edalize#89 + default_tool: icarus + parameters: + - SYNTHESIS=true diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl_pkg.core b/hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl_pkg.core new file mode 100644 index 00000000000000..6698fbdb8541fa --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl_pkg.core @@ -0,0 +1,67 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: lowrisc:ip:otp_ctrl_pkg:1.0 +description: "OTP Controller Package" +filesets: + files_rtl: + depend: + - lowrisc:ip:lc_ctrl_pkg + + files: + - rtl/otp_ctrl_pkg.sv + file_type: systemVerilogSource + + files_verilator_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_pkg.vlt + file_type: vlt + + files_ascentlint_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_pkg.waiver + file_type: waiver + + files_veriblelint_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_pkg.vbl + file_type: veribleLintWaiver + +parameters: + SYNTHESIS: + datatype: bool + paramtype: vlogdefine + + +targets: + default: &default_target + filesets: + - tool_verilator ? (files_verilator_waiver) + - tool_ascentlint ? (files_ascentlint_waiver) + - tool_veriblelint ? (files_veriblelint_waiver) + - files_rtl + toplevel: lc_ctrl + + lint: + <<: *default_target + default_tool: verilator + parameters: + - SYNTHESIS=true + tools: + verilator: + mode: lint-only + verilator_options: + - "-Wall" diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl_prim_reg_top.core b/hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl_prim_reg_top.core new file mode 100644 index 00000000000000..817b3e74434c60 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl_prim_reg_top.core @@ -0,0 +1,29 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: "lowrisc:ip:otp_ctrl_prim_reg_top:1.0" +description: "Generic register top for the OTP wrapper" +filesets: + files_rtl: + depend: + # otp_ctrl_prim_reg_top.sv should depend on generic items for now. + # This may change and will require reworking the flow to generate + # the otp prim. + - lowrisc:ip_interfaces:otp_ctrl_top_specific_pkg + files: + - rtl/otp_ctrl_prim_reg_top.sv + file_type: systemVerilogSource + + +parameters: + SYNTHESIS: + datatype: bool + paramtype: vlogdefine + + +targets: + default: &default_target + filesets: + - files_rtl + toplevel: lc_ctrl diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl_top_specific_pkg.core b/hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl_top_specific_pkg.core new file mode 100644 index 00000000000000..588bf79ed59c89 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/otp_ctrl_top_specific_pkg.core @@ -0,0 +1,75 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: lowrisc:opentitan:top_darjeeling_otp_ctrl_top_specific_pkg:1.0 +description: "OTP Controller Top Specific Packages" +virtual: + - lowrisc:ip_interfaces:otp_ctrl_top_specific_pkg + +filesets: + files_rtl: + depend: + - lowrisc:tlul:headers + - lowrisc:ip:lc_ctrl_pkg + - lowrisc:ip:otp_ctrl_pkg + - lowrisc:prim:mubi + + files: + - rtl/otp_ctrl_reg_pkg.sv + - rtl/otp_ctrl_top_specific_pkg.sv + - rtl/otp_ctrl_part_pkg.sv + file_type: systemVerilogSource + + files_verilator_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_top_specific_pkg.vlt + file_type: vlt + + files_ascentlint_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_top_specific_pkg.waiver + file_type: waiver + + files_veriblelint_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_top_specific_pkg.vbl + file_type: veribleLintWaiver + +parameters: + SYNTHESIS: + datatype: bool + paramtype: vlogdefine + + +targets: + default: &default_target + filesets: + - tool_verilator ? (files_verilator_waiver) + - tool_ascentlint ? (files_ascentlint_waiver) + - tool_veriblelint ? (files_veriblelint_waiver) + - files_rtl + toplevel: lc_ctrl + + lint: + <<: *default_target + default_tool: verilator + parameters: + - SYNTHESIS=true + tools: + verilator: + mode: lint-only + verilator_options: + - "-Wall" diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl.sv new file mode 100644 index 00000000000000..6d5647a989a41b --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl.sv @@ -0,0 +1,1572 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// OTP Controller top. +// + +`include "prim_assert.sv" + +module otp_ctrl + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; +#( + // Enable asynchronous transitions on alerts. + parameter logic [NumAlerts-1:0] AlertAsyncOn = {NumAlerts{1'b1}}, + // Compile time random constants, to be overriden by topgen. + parameter lfsr_seed_t RndCnstLfsrSeed = RndCnstLfsrSeedDefault, + parameter lfsr_perm_t RndCnstLfsrPerm = RndCnstLfsrPermDefault, + parameter scrmbl_key_init_t RndCnstScrmblKeyInit = RndCnstScrmblKeyInitDefault, + // Hexfile file to initialize the OTP macro. + // Note that the hexdump needs to account for ECC. + parameter MemInitFile = "" +) ( + // OTP clock + input clk_i, + input rst_ni, + // EDN clock and interface + logic clk_edn_i, + logic rst_edn_ni, + output edn_pkg::edn_req_t edn_o, + input edn_pkg::edn_rsp_t edn_i, + // Bus Interface + input tlul_pkg::tl_h2d_t core_tl_i, + output tlul_pkg::tl_d2h_t core_tl_o, + input tlul_pkg::tl_h2d_t prim_tl_i, + output tlul_pkg::tl_d2h_t prim_tl_o, + // Interrupt Requests + output logic intr_otp_operation_done_o, + output logic intr_otp_error_o, + // Alerts + input prim_alert_pkg::alert_rx_t [NumAlerts-1:0] alert_rx_i, + output prim_alert_pkg::alert_tx_t [NumAlerts-1:0] alert_tx_o, + // Observability to AST + input ast_pkg::ast_obs_ctrl_t obs_ctrl_i, + output logic [7:0] otp_obs_o, + // Macro-specific power sequencing signals to/from AST. + output otp_ast_req_t otp_ast_pwr_seq_o, + input otp_ast_rsp_t otp_ast_pwr_seq_h_i, + // Power manager interface (inputs are synced to OTP clock domain) + input pwrmgr_pkg::pwr_otp_req_t pwr_otp_i, + output pwrmgr_pkg::pwr_otp_rsp_t pwr_otp_o, + // Macro-specific test registers going to lifecycle TAP + input lc_otp_vendor_test_req_t lc_otp_vendor_test_i, + output lc_otp_vendor_test_rsp_t lc_otp_vendor_test_o, + // Lifecycle transition command interface + input lc_otp_program_req_t lc_otp_program_i, + output lc_otp_program_rsp_t lc_otp_program_o, + // Lifecycle broadcast inputs + // SEC_CM: LC_CTRL.INTERSIG.MUBI + input lc_ctrl_pkg::lc_tx_t lc_creator_seed_sw_rw_en_i, + input lc_ctrl_pkg::lc_tx_t lc_owner_seed_sw_rw_en_i, + input lc_ctrl_pkg::lc_tx_t lc_seed_hw_rd_en_i, + input lc_ctrl_pkg::lc_tx_t lc_dft_en_i, + input lc_ctrl_pkg::lc_tx_t lc_escalate_en_i, + input lc_ctrl_pkg::lc_tx_t lc_check_byp_en_i, + // OTP broadcast outputs + // SEC_CM: TOKEN_VALID.CTRL.MUBI + output otp_lc_data_t otp_lc_data_o, + output otp_keymgr_key_t otp_keymgr_key_o, + // Scrambling key requests + input flash_otp_key_req_t flash_otp_key_i, + output flash_otp_key_rsp_t flash_otp_key_o, + input sram_otp_key_req_t [NumSramKeyReqSlots-1:0] sram_otp_key_i, + output sram_otp_key_rsp_t [NumSramKeyReqSlots-1:0] sram_otp_key_o, + input otbn_otp_key_req_t otbn_otp_key_i, + output otbn_otp_key_rsp_t otbn_otp_key_o, + // Hardware config bits + output otp_broadcast_t otp_broadcast_o, + // External voltage for OTP + inout wire otp_ext_voltage_h_io, + // Scan + input scan_en_i, + input scan_rst_ni, + input prim_mubi_pkg::mubi4_t scanmode_i, + // Test-related GPIO output + output logic [OtpTestVectWidth-1:0] cio_test_o, + output logic [OtpTestVectWidth-1:0] cio_test_en_o +); + + import prim_mubi_pkg::*; + import prim_util_pkg::vbits; + + //////////////////////// + // Integration Checks // + //////////////////////// + + // This ensures that we can transfer scrambler data blocks in and out of OTP atomically. + `ASSERT_INIT(OtpIfWidth_A, OtpIfWidth == ScrmblBlockWidth) + + // These error codes need to be identical. + `ASSERT_INIT(ErrorCodeWidth_A, OtpErrWidth == prim_otp_pkg::ErrWidth) + `ASSERT_INIT(OtpErrorCode0_A, int'(NoError) == int'(prim_otp_pkg::NoError)) + `ASSERT_INIT(OtpErrorCode1_A, int'(MacroError) == int'(prim_otp_pkg::MacroError)) + `ASSERT_INIT(OtpErrorCode2_A, int'(MacroEccCorrError) == int'(prim_otp_pkg::MacroEccCorrError)) + `ASSERT_INIT(OtpErrorCode3_A, + int'(MacroEccUncorrError) == int'(prim_otp_pkg::MacroEccUncorrError)) + `ASSERT_INIT(OtpErrorCode4_A, + int'(MacroWriteBlankError) == int'(prim_otp_pkg::MacroWriteBlankError)) + + ///////////// + // Regfile // + ///////////// + + // We have one CSR node, one functional TL-UL window and a gate module for that window + logic [2:0] intg_error; + + tlul_pkg::tl_h2d_t tl_win_h2d; + tlul_pkg::tl_d2h_t tl_win_d2h; + + otp_ctrl_reg_pkg::otp_ctrl_core_reg2hw_t reg2hw; + otp_ctrl_reg_pkg::otp_ctrl_core_hw2reg_t hw2reg; + + // SEC_CM: DIRECT_ACCESS.CONFIG.REGWEN, CHECK_TRIGGER.CONFIG.REGWEN, CHECK.CONFIG.REGWEN + otp_ctrl_core_reg_top u_reg_core ( + .clk_i, + .rst_ni, + .tl_i ( core_tl_i ), + .tl_o ( core_tl_o ), + .tl_win_o ( tl_win_h2d ), + .tl_win_i ( tl_win_d2h ), + .reg2hw ( reg2hw ), + .hw2reg ( hw2reg ), + // SEC_CM: BUS.INTEGRITY + .intg_err_o( intg_error[0] ) + ); + + /////////////////////////////////////// + // Life Cycle Signal Synchronization // + /////////////////////////////////////// + + lc_ctrl_pkg::lc_tx_t lc_creator_seed_sw_rw_en, lc_owner_seed_sw_rw_en, + lc_seed_hw_rd_en, lc_check_byp_en; + lc_ctrl_pkg::lc_tx_t [2:0] lc_dft_en; + // NumAgents + lfsr timer and scrambling datapath. + lc_ctrl_pkg::lc_tx_t [NumAgentsIdx+1:0] lc_escalate_en, lc_escalate_en_synced; + // Single wire for gating assertions in arbitration and CDC primitives. + logic lc_escalate_en_any; + + prim_lc_sync #( + .NumCopies(NumAgentsIdx+2) + ) u_prim_lc_sync_escalate_en ( + .clk_i, + .rst_ni, + .lc_en_i(lc_escalate_en_i), + .lc_en_o(lc_escalate_en_synced) + ); + + prim_lc_sync #( + .NumCopies(1) + ) u_prim_lc_sync_creator_seed_sw_rw_en ( + .clk_i, + .rst_ni, + .lc_en_i(lc_creator_seed_sw_rw_en_i), + .lc_en_o({lc_creator_seed_sw_rw_en}) + ); + + prim_lc_sync #( + .NumCopies(1) + ) u_prim_lc_sync_owner_seed_sw_rw_en ( + .clk_i, + .rst_ni, + .lc_en_i(lc_owner_seed_sw_rw_en_i), + .lc_en_o({lc_owner_seed_sw_rw_en}) + ); + + prim_lc_sync #( + .NumCopies(1) + ) u_prim_lc_sync_seed_hw_rd_en ( + .clk_i, + .rst_ni, + .lc_en_i(lc_seed_hw_rd_en_i), + .lc_en_o({lc_seed_hw_rd_en}) + ); + + prim_lc_sync #( + .NumCopies(3) + ) u_prim_lc_sync_dft_en ( + .clk_i, + .rst_ni, + .lc_en_i(lc_dft_en_i), + .lc_en_o(lc_dft_en) + ); + + prim_lc_sync #( + .NumCopies(1) + ) u_prim_lc_sync_check_byp_en ( + .clk_i, + .rst_ni, + .lc_en_i(lc_check_byp_en_i), + .lc_en_o({lc_check_byp_en}) + ); + + ///////////////////////////////////// + // TL-UL SW partition select logic // + ///////////////////////////////////// + + // The SW partitions share the same TL-UL adapter. + logic tlul_req, tlul_gnt, tlul_rvalid; + logic [SwWindowAddrWidth-1:0] tlul_addr; + logic [1:0] tlul_rerror; + logic [31:0] tlul_rdata; + + import prim_mubi_pkg::MuBi4False; + tlul_adapter_sram #( + .SramAw ( SwWindowAddrWidth ), + .SramDw ( 32 ), + .Outstanding ( 1 ), + .ByteAccess ( 0 ), + .ErrOnWrite ( 1 ) // No write accesses allowed here. + ) u_tlul_adapter_sram ( + .clk_i, + .rst_ni, + .en_ifetch_i ( MuBi4False ), + .tl_i ( tl_win_h2d ), + .tl_o ( tl_win_d2h ), + .req_o ( tlul_req ), + .gnt_i ( tlul_gnt ), + .we_o ( ), // unused + .addr_o ( tlul_addr ), + .wdata_o ( ), // unused + .wmask_o ( ), // unused + // SEC_CM: BUS.INTEGRITY + .intg_error_o ( intg_error[1] ), + .user_rsvd_o ( ), + .rdata_i ( tlul_rdata ), + .rvalid_i ( tlul_rvalid ), + .rerror_i ( tlul_rerror ), + .req_type_o ( ), + .compound_txn_in_progress_o ( ), + .readback_en_i ( MuBi4False ), + .readback_error_o ( ), + .wr_collision_i ( 1'b0 ), + .write_pending_i ( 1'b0 ) + ); + + logic [NumPart-1:0] tlul_part_sel_oh; + for (genvar k = 0; k < NumPart; k++) begin : gen_part_sel + localparam logic [OtpByteAddrWidth:0] PartEnd = (OtpByteAddrWidth+1)'(PartInfo[k].offset) + + (OtpByteAddrWidth+1)'(PartInfo[k].size); + if (PartInfo[k].offset == 0) begin : gen_zero_offset + assign tlul_part_sel_oh[k] = ({1'b0, {tlul_addr, 2'b00}} < PartEnd); + end else begin : gen_nonzero_offset + assign tlul_part_sel_oh[k] = ({tlul_addr, 2'b00} >= PartInfo[k].offset) & + ({1'b0, {tlul_addr, 2'b00}} < PartEnd); + end + end + + `ASSERT(PartSelMustBeOnehot_A, $onehot0(tlul_part_sel_oh)) + + logic [NumPartWidth-1:0] tlul_part_idx; + prim_arbiter_fixed #( + .N(NumPart), + .EnDataPort(0) + ) u_part_sel_idx ( + .clk_i, + .rst_ni, + .req_i ( tlul_part_sel_oh ), + .data_i ( '{default: '0} ), + .gnt_o ( ), // unused + .idx_o ( tlul_part_idx ), + .valid_o ( ), // unused + .data_o ( ), // unused + .ready_i ( 1'b0 ) + ); + + logic tlul_oob_err_d, tlul_oob_err_q; + logic [NumPart-1:0] part_tlul_req, part_tlul_gnt, part_tlul_rvalid; + logic [SwWindowAddrWidth-1:0] part_tlul_addr; + logic [NumPart-1:0][1:0] part_tlul_rerror; + logic [NumPart-1:0][31:0] part_tlul_rdata; + + always_comb begin : p_tlul_assign + // Send request to the correct partition. + part_tlul_addr = tlul_addr; + part_tlul_req = '0; + tlul_oob_err_d = 1'b0; + if (tlul_req) begin + if (tlul_part_sel_oh != '0) begin + part_tlul_req[tlul_part_idx] = 1'b1; + end else begin + // Error out in the next cycle if address was out of bounds. + tlul_oob_err_d = 1'b1; + end + end + + // aggregate TL-UL responses + tlul_gnt = |part_tlul_gnt | tlul_oob_err_q; + tlul_rvalid = |part_tlul_rvalid | tlul_oob_err_q; + tlul_rerror = '0; + tlul_rdata = '0; + for (int k = 0; k < NumPart; k++) begin + tlul_rerror |= part_tlul_rerror[k]; + tlul_rdata |= part_tlul_rdata[k]; + end + end + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_tlul_reg + if (!rst_ni) begin + tlul_oob_err_q <= 1'b0; + end else begin + tlul_oob_err_q <= tlul_oob_err_d; + end + end + + ////////////////////////////// + // Access Defaults and CSRs // + ////////////////////////////// + + // SEC_CM: ACCESS.CTRL.MUBI + part_access_t [NumPart-1:0] part_access_pre, part_access; + always_comb begin : p_access_control + // Assigns default and extracts named CSR read enables for SW_CFG partitions. + // SEC_CM: PART.MEM.REGREN + part_access_pre = named_part_access_pre(reg2hw); + + // Permanently lock DAI write and read access to the life cycle partition. + // The LC partition can only be read from and written to via the LC controller. + // SEC_CM: LC_PART.MEM.SW_NOACCESS + part_access_pre[LifeCycleIdx].write_lock = MuBi8True; + part_access_pre[LifeCycleIdx].read_lock = MuBi8True; + + // Special partitions for keymgr material only become writable when + // provisioning is enabled. + if (lc_ctrl_pkg::lc_tx_test_false_loose(lc_creator_seed_sw_rw_en)) begin + for (int k = 0; k < NumPart; k++) begin + if (PartInfo[k].iskeymgr_creator) begin + part_access_pre[k] = {2{prim_mubi_pkg::MuBi8True}}; + end + end + end + if (lc_ctrl_pkg::lc_tx_test_false_loose(lc_owner_seed_sw_rw_en)) begin + for (int k = 0; k < NumPart; k++) begin + if (PartInfo[k].iskeymgr_owner) begin + part_access_pre[k] = {2{prim_mubi_pkg::MuBi8True}}; + end + end + end + end + + // This prevents the synthesis tool from optimizing the multibit signals. + for (genvar k = 0; k < NumPart; k++) begin : gen_bufs + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_write_lock ( + .clk_i, + .rst_ni, + .mubi_i(part_access_pre[k].write_lock), + .mubi_o(part_access[k].write_lock) + ); + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_read_lock ( + .clk_i, + .rst_ni, + .mubi_i(part_access_pre[k].read_lock), + .mubi_o(part_access[k].read_lock) + ); + end + + ////////////////////// + // DAI-related CSRs // + ////////////////////// + + logic dai_idle; + logic dai_req; + dai_cmd_e dai_cmd; + logic [OtpByteAddrWidth-1:0] dai_addr; + logic [NumDaiWords-1:0][31:0] dai_wdata, dai_rdata; + logic direct_access_regwen_d, direct_access_regwen_q; + + // This is the HWEXT implementation of a RW0C regwen bit. + assign direct_access_regwen_d = (reg2hw.direct_access_regwen.qe && + !reg2hw.direct_access_regwen.q) ? 1'b0 : direct_access_regwen_q; + + // Any write to this register triggers a DAI command. + assign dai_req = reg2hw.direct_access_cmd.digest.qe | + reg2hw.direct_access_cmd.wr.qe | + reg2hw.direct_access_cmd.rd.qe; + + assign dai_cmd = dai_cmd_e'({reg2hw.direct_access_cmd.digest.q, + reg2hw.direct_access_cmd.wr.q, + reg2hw.direct_access_cmd.rd.q}); + + assign dai_addr = reg2hw.direct_access_address.q; + assign dai_wdata = reg2hw.direct_access_wdata; + + // The DAI and the LCI can initiate write transactions, which + // are critical and we must not power down if such transactions + // are pending. Hence, we signal the LCI/DAI idle state to the + // power manager. This signal is flopped here as it has to + // cross a clock boundary to the power manager. + logic dai_prog_idle, lci_prog_idle, otp_idle_d, otp_idle_q; + assign otp_idle_d = lci_prog_idle & dai_prog_idle; + assign pwr_otp_o.otp_idle = otp_idle_q; + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_idle_regwen_regs + if (!rst_ni) begin + otp_idle_q <= 1'b0; + // The regwen bit has to reset to 1 so that CSR accesses are enabled by default. + direct_access_regwen_q <= 1'b1; + end else begin + otp_idle_q <= otp_idle_d; + direct_access_regwen_q <= direct_access_regwen_d; + end + end + + ////////////////////////////////////// + // Ctrl/Status CSRs, Errors, Alerts // + ////////////////////////////////////// + + // Status and error reporting CSRs, error interrupt generation and alerts. + otp_err_e [NumPart+1:0] part_error; + logic [NumAgents-1:0] part_fsm_err; + logic [NumPart+1:0] part_errors_reduced; + logic otp_operation_done, otp_error; + logic fatal_macro_error_d, fatal_macro_error_q; + logic fatal_check_error_d, fatal_check_error_q; + logic fatal_bus_integ_error_d, fatal_bus_integ_error_q; + logic chk_pending, chk_timeout; + logic lfsr_fsm_err, scrmbl_fsm_err; + always_comb begin : p_errors_alerts + // Note: since these are all fatal alert events, we latch them and keep on sending + // alert events via the alert senders. These regs can only be cleared via a system reset. + fatal_macro_error_d = fatal_macro_error_q; + fatal_check_error_d = fatal_check_error_q; + fatal_bus_integ_error_d = fatal_bus_integ_error_q | (|intg_error); + // These are the per-partition buffered escalation inputs + lc_escalate_en = lc_escalate_en_synced; + // Need a single wire for gating assertions in arbitration and CDC primitives. + lc_escalate_en_any = 1'b0; + + // Aggregate all the macro alerts from the partitions + for (int k = 0; k < NumPart; k++) begin + // Filter for critical error codes that should not occur in the field. + fatal_macro_error_d |= part_error[k] == MacroError; + // While uncorrectable ECC errors are always reported, they do not trigger a fatal alert + // event in some partitions like the VENDOR_TEST partition. + if (PartInfo[k].integrity) begin + fatal_macro_error_d |= part_error[k] == MacroEccUncorrError; + end + end + // Aggregate all the macro alerts from the DAI/LCI + for (int k = NumPart; k < NumPart+2; k++) begin + // Filter for critical error codes that should not occur in the field. + fatal_macro_error_d |= part_error[k] inside {MacroError, MacroEccUncorrError}; + end + + // Aggregate all the remaining errors / alerts from the partitions and the DAI/LCI + for (int k = 0; k < NumPart+2; k++) begin + // Set the error bit if the error status of the corresponding partition is nonzero. + // Need to reverse the order here since the field enumeration in hw2reg.status is reversed. + part_errors_reduced[NumPart+1-k] = |part_error[k]; + // Filter for integrity and consistency check failures. + fatal_check_error_d |= part_error[k] inside {CheckFailError, FsmStateError}; + + // If a fatal alert has been observed in any of the partitions/FSMs, + // we locally trigger escalation within OTP, which moves all FSMs + // to a terminal error state. + if (fatal_macro_error_q || fatal_check_error_q) begin + lc_escalate_en[k] = lc_ctrl_pkg::On; + end + if (lc_ctrl_pkg::lc_tx_test_true_strict(lc_escalate_en[k])) begin + lc_escalate_en_any = 1'b1; + end + end + + // Errors from other non-partition FSMs. + fatal_check_error_d |= chk_timeout | + lfsr_fsm_err | + scrmbl_fsm_err | + (|part_fsm_err); + end + + // If we got an error, we trigger an interrupt. + logic [$bits(part_errors_reduced)+4-1:0] interrupt_triggers_d, interrupt_triggers_q; + + // This makes sure that interrupts are not sticky. + assign interrupt_triggers_d = { + part_errors_reduced, + chk_timeout, + lfsr_fsm_err, + scrmbl_fsm_err, + |part_fsm_err + }; + + assign otp_error = |(interrupt_triggers_d & ~interrupt_triggers_q); + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_alert_regs + if (!rst_ni) begin + fatal_macro_error_q <= '0; + fatal_check_error_q <= '0; + fatal_bus_integ_error_q <= '0; + interrupt_triggers_q <= '0; + end else begin + fatal_macro_error_q <= fatal_macro_error_d; + fatal_check_error_q <= fatal_check_error_d; + fatal_bus_integ_error_q <= fatal_bus_integ_error_d; + interrupt_triggers_q <= interrupt_triggers_d; + end + end + + // CSR assignments are done in one combo process so that we can use + // the parameterized digest_assign task below without multiple driver issues. + logic unused_part_digest; + logic [NumPart-1:0][ScrmblBlockWidth-1:0] part_digest; + logic intr_state_otp_operation_done_d, intr_state_otp_operation_done_de; + logic intr_state_otp_error_d, intr_state_otp_error_de; + always_comb begin : p_csr_assign + // Not all partition digests are consumed, and assigning them to an unused_* signal in the + // function below does not seem to work for some linters. + unused_part_digest = ^part_digest; + // Assign named CSRs (like digests). + hw2reg = named_reg_assign(part_digest); + // DAI related CSRs + hw2reg.direct_access_rdata = dai_rdata; + // ANDing this state with dai_idle write-protects all DAI regs during pending operations. + hw2reg.direct_access_regwen.d = direct_access_regwen_q & dai_idle; + // Assign these to the status register. + hw2reg.status = {part_errors_reduced, + chk_timeout, + lfsr_fsm_err, + scrmbl_fsm_err, + part_fsm_err[KdiIdx], + fatal_bus_integ_error_q, + dai_idle, + chk_pending}; + // Error code registers. + hw2reg.err_code = part_error; + // Interrupt signals + hw2reg.intr_state.otp_operation_done.de = intr_state_otp_operation_done_de; + hw2reg.intr_state.otp_operation_done.d = intr_state_otp_operation_done_d; + hw2reg.intr_state.otp_error.de = intr_state_otp_error_de; + hw2reg.intr_state.otp_error.d = intr_state_otp_error_d; +end + + + ////////////////////////////////// + // Interrupts and Alert Senders // + ////////////////////////////////// + + prim_intr_hw #( + .Width(1) + ) u_intr_operation_done ( + .clk_i, + .rst_ni, + .event_intr_i ( otp_operation_done ), + .reg2hw_intr_enable_q_i ( reg2hw.intr_enable.otp_operation_done.q ), + .reg2hw_intr_test_q_i ( reg2hw.intr_test.otp_operation_done.q ), + .reg2hw_intr_test_qe_i ( reg2hw.intr_test.otp_operation_done.qe ), + .reg2hw_intr_state_q_i ( reg2hw.intr_state.otp_operation_done.q ), + .hw2reg_intr_state_de_o ( intr_state_otp_operation_done_de ), + .hw2reg_intr_state_d_o ( intr_state_otp_operation_done_d ), + .intr_o ( intr_otp_operation_done_o ) + ); + + prim_intr_hw #( + .Width(1) + ) u_intr_error ( + .clk_i, + .rst_ni, + .event_intr_i ( otp_error ), + .reg2hw_intr_enable_q_i ( reg2hw.intr_enable.otp_error.q ), + .reg2hw_intr_test_q_i ( reg2hw.intr_test.otp_error.q ), + .reg2hw_intr_test_qe_i ( reg2hw.intr_test.otp_error.qe ), + .reg2hw_intr_state_q_i ( reg2hw.intr_state.otp_error.q ), + .hw2reg_intr_state_de_o ( intr_state_otp_error_de ), + .hw2reg_intr_state_d_o ( intr_state_otp_error_d ), + .intr_o ( intr_otp_error_o ) + ); + + logic [NumAlerts-1:0] alerts; + logic [NumAlerts-1:0] alert_test; + logic fatal_prim_otp_alert, recov_prim_otp_alert; + + assign alerts = { + recov_prim_otp_alert, + fatal_prim_otp_alert, + fatal_bus_integ_error_q, + fatal_check_error_q, + fatal_macro_error_q + }; + + assign alert_test = { + reg2hw.alert_test.recov_prim_otp_alert.q & + reg2hw.alert_test.recov_prim_otp_alert.qe, + reg2hw.alert_test.fatal_prim_otp_alert.q & + reg2hw.alert_test.fatal_prim_otp_alert.qe, + reg2hw.alert_test.fatal_bus_integ_error.q & + reg2hw.alert_test.fatal_bus_integ_error.qe, + reg2hw.alert_test.fatal_check_error.q & + reg2hw.alert_test.fatal_check_error.qe, + reg2hw.alert_test.fatal_macro_error.q & + reg2hw.alert_test.fatal_macro_error.qe + }; + + localparam logic [NumAlerts-1:0] AlertIsFatal = { + 1'b0, // recov_prim_otp_alert + 1'b1, // fatal_prim_otp_alert + 1'b1, // fatal_bus_integ_error_q + 1'b1, // fatal_check_error_q + 1'b1 // fatal_macro_error_q + }; + + for (genvar k = 0; k < NumAlerts; k++) begin : gen_alert_tx + prim_alert_sender #( + .AsyncOn(AlertAsyncOn[k]), + .IsFatal(AlertIsFatal[k]) + ) u_prim_alert_sender ( + .clk_i, + .rst_ni, + .alert_test_i ( alert_test[k] ), + .alert_req_i ( alerts[k] ), + .alert_ack_o ( ), + .alert_state_o ( ), + .alert_rx_i ( alert_rx_i[k] ), + .alert_tx_o ( alert_tx_o[k] ) + ); + end + + //////////////////////////////// + // LFSR Timer and CSR mapping // + //////////////////////////////// + + logic integ_chk_trig, cnsty_chk_trig; + logic [NumPart-1:0] integ_chk_req, integ_chk_ack; + logic [NumPart-1:0] cnsty_chk_req, cnsty_chk_ack; + logic lfsr_edn_req, lfsr_edn_ack; + logic [EdnDataWidth-1:0] edn_data; + + assign integ_chk_trig = reg2hw.check_trigger.integrity.q & + reg2hw.check_trigger.integrity.qe; + assign cnsty_chk_trig = reg2hw.check_trigger.consistency.q & + reg2hw.check_trigger.consistency.qe; + + // SEC_CM: PART.DATA_REG.BKGN_CHK + otp_ctrl_lfsr_timer #( + .RndCnstLfsrSeed(RndCnstLfsrSeed), + .RndCnstLfsrPerm(RndCnstLfsrPerm) + ) u_otp_ctrl_lfsr_timer ( + .clk_i, + .rst_ni, + .edn_req_o ( lfsr_edn_req ), + .edn_ack_i ( lfsr_edn_ack ), + .edn_data_i ( edn_data ), + // We can enable the timer once OTP has initialized. + // Note that this is only the initial release that gets + // the timer FSM into an operational state. + // Whether or not the timers / background checks are + // activated depends on the CSR configuration (by default + // they are switched off). + .timer_en_i ( pwr_otp_o.otp_done ), + // This idle signal is the same that is output to the power + // manager, and indicates whether there is an ongoing OTP programming + // operation. It is used to pause the consistency check timeout + // counter in order to prevent spurious timeouts (OTP programming + // operations are very slow compared to readout operations and can + // hence interfere with the timeout mechanism). + .otp_prog_busy_i ( ~otp_idle_d ), + .integ_chk_trig_i ( integ_chk_trig ), + .cnsty_chk_trig_i ( cnsty_chk_trig ), + .chk_pending_o ( chk_pending ), + .timeout_i ( reg2hw.check_timeout.q ), + .integ_period_msk_i ( reg2hw.integrity_check_period.q ), + .cnsty_period_msk_i ( reg2hw.consistency_check_period.q ), + .integ_chk_req_o ( integ_chk_req ), + .cnsty_chk_req_o ( cnsty_chk_req ), + .integ_chk_ack_i ( integ_chk_ack ), + .cnsty_chk_ack_i ( cnsty_chk_ack ), + .escalate_en_i ( lc_escalate_en[NumAgents] ), + .chk_timeout_o ( chk_timeout ), + .fsm_err_o ( lfsr_fsm_err ) + ); + + /////////////////////////////////////// + // EDN Arbitration, Request and Sync // + /////////////////////////////////////// + + // Both the key derivation and LFSR reseeding are low bandwidth, + // hence they can share the same EDN interface. + logic edn_req, edn_ack; + logic key_edn_req, key_edn_ack; + prim_arbiter_tree #( + .N(2), + .EnDataPort(0) + ) u_edn_arb ( + .clk_i, + .rst_ni, + .req_chk_i ( ~lc_escalate_en_any ), + .req_i ( {lfsr_edn_req, key_edn_req} ), + .data_i ( '{default: '0} ), + .gnt_o ( {lfsr_edn_ack, key_edn_ack} ), + .idx_o ( ), // unused + .valid_o ( edn_req ), + .data_o ( ), // unused + .ready_i ( edn_ack ) + ); + + // This synchronizes the data coming from EDN and stacks the + // 32bit EDN words to achieve an internal entropy width of 64bit. + prim_edn_req #( + .OutWidth(EdnDataWidth) + ) u_prim_edn_req ( + .clk_i, + .rst_ni, + .req_chk_i ( ~lc_escalate_en_any ), + .req_i ( edn_req ), + .ack_o ( edn_ack ), + .data_o ( edn_data ), + .fips_o ( ), // unused + .err_o ( ), // unused + .clk_edn_i, + .rst_edn_ni, + .edn_o, + .edn_i + ); + + /////////////////////////////// + // OTP Macro and Arbitration // + /////////////////////////////// + + typedef struct packed { + prim_otp_pkg::cmd_e cmd; + logic [OtpSizeWidth-1:0] size; // Number of native words to write. + logic [OtpIfWidth-1:0] wdata; + logic [OtpAddrWidth-1:0] addr; // Halfword address. + } otp_bundle_t; + + logic [NumAgents-1:0] part_otp_arb_req, part_otp_arb_gnt; + otp_bundle_t part_otp_arb_bundle [NumAgents]; + logic otp_arb_valid, otp_arb_ready; + logic otp_prim_valid, otp_prim_ready; + logic otp_rsp_fifo_valid, otp_rsp_fifo_ready; + logic [vbits(NumAgents)-1:0] otp_arb_idx; + otp_bundle_t otp_arb_bundle; + + // The OTP interface is arbitrated on a per-cycle basis, meaning that back-to-back + // transactions can be completely independent. + prim_arbiter_tree #( + .N(NumAgents), + .DW($bits(otp_bundle_t)) + ) u_otp_arb ( + .clk_i, + .rst_ni, + .req_chk_i ( ~lc_escalate_en_any ), + .req_i ( part_otp_arb_req ), + .data_i ( part_otp_arb_bundle ), + .gnt_o ( part_otp_arb_gnt ), + .idx_o ( otp_arb_idx ), + .valid_o ( otp_arb_valid ), + .data_o ( otp_arb_bundle ), + .ready_i ( otp_arb_ready ) + ); + + // Don't issue more transactions than what the rsp_fifo can keep track of. + assign otp_arb_ready = otp_prim_ready & otp_rsp_fifo_ready; + assign otp_prim_valid = otp_arb_valid & otp_rsp_fifo_ready; + assign otp_rsp_fifo_valid = otp_prim_ready & otp_prim_valid; + + prim_otp_pkg::err_e part_otp_err; + logic [OtpIfWidth-1:0] part_otp_rdata; + logic otp_rvalid; + tlul_pkg::tl_h2d_t prim_tl_h2d_gated; + tlul_pkg::tl_d2h_t prim_tl_d2h_gated; + + // Life cycle qualification of TL-UL test interface. + // SEC_CM: TEST.BUS.LC_GATED + // SEC_CM: TEST_TL_LC_GATE.FSM.SPARSE + tlul_lc_gate #( + .NumGatesPerDirection(2) + ) u_tlul_lc_gate ( + .clk_i, + .rst_ni, + .tl_h2d_i(prim_tl_i), + .tl_d2h_o(prim_tl_o), + .tl_h2d_o(prim_tl_h2d_gated), + .tl_d2h_i(prim_tl_d2h_gated), + .lc_en_i (lc_dft_en[0]), + .flush_req_i('0), + .flush_ack_o(), + .resp_pending_o(), + .err_o (intg_error[2]) + ); + + // Test-related GPIOs. + // SEC_CM: TEST.BUS.LC_GATED + logic [OtpTestVectWidth-1:0] otp_test_vect; + assign cio_test_o = (lc_ctrl_pkg::lc_tx_test_true_strict(lc_dft_en[1])) ? + otp_test_vect : '0; + assign cio_test_en_o = (lc_ctrl_pkg::lc_tx_test_true_strict(lc_dft_en[2])) ? + {OtpTestVectWidth{1'b1}} : '0; + + // SEC_CM: MACRO.MEM.CM, MACRO.MEM.INTEGRITY + prim_otp #( + .Width ( OtpWidth ), + .Depth ( OtpDepth ), + .SizeWidth ( OtpSizeWidth ), + .PwrSeqWidth ( OtpPwrSeqWidth ), + .TestCtrlWidth ( OtpTestCtrlWidth ), + .TestStatusWidth ( OtpTestStatusWidth ), + .TestVectWidth ( OtpTestVectWidth ), + .MemInitFile ( MemInitFile ), + .VendorTestOffset ( VendorTestOffset ), + .VendorTestSize ( VendorTestSize ) + ) u_otp ( + .clk_i, + .rst_ni, + // Observability controls to/from AST + .obs_ctrl_i, + .otp_obs_o, + // Power sequencing signals to/from AST + .pwr_seq_o ( otp_ast_pwr_seq_o.pwr_seq ), + .pwr_seq_h_i ( otp_ast_pwr_seq_h_i.pwr_seq_h ), + .ext_voltage_io ( otp_ext_voltage_h_io ), + // Test interface + .test_ctrl_i ( lc_otp_vendor_test_i.ctrl ), + .test_status_o ( lc_otp_vendor_test_o.status ), + .test_vect_o ( otp_test_vect ), + .test_tl_i ( prim_tl_h2d_gated ), + .test_tl_o ( prim_tl_d2h_gated ), + // Other DFT signals + .scan_en_i, + .scan_rst_ni, + .scanmode_i, + // Alerts + .fatal_alert_o ( fatal_prim_otp_alert ), + .recov_alert_o ( recov_prim_otp_alert ), + // Read / Write command interface + .ready_o ( otp_prim_ready ), + .valid_i ( otp_prim_valid ), + .cmd_i ( otp_arb_bundle.cmd ), + .size_i ( otp_arb_bundle.size ), + .addr_i ( otp_arb_bundle.addr ), + .wdata_i ( otp_arb_bundle.wdata ), + // Read data out + .valid_o ( otp_rvalid ), + .rdata_o ( part_otp_rdata ), + .err_o ( part_otp_err ) + ); + + logic otp_fifo_valid; + logic [vbits(NumAgents)-1:0] otp_part_idx; + logic [NumAgents-1:0] part_otp_rvalid; + + // We can have up to two OTP commands in flight, hence we size this to be 2 deep. + // The partitions can unconditionally sink requested data. + prim_fifo_sync #( + .Width(vbits(NumAgents)), + .Depth(2) + ) u_otp_rsp_fifo ( + .clk_i, + .rst_ni, + .clr_i ( 1'b0 ), + .wvalid_i ( otp_rsp_fifo_valid ), + .wready_o ( otp_rsp_fifo_ready ), + .wdata_i ( otp_arb_idx ), + .rvalid_o ( otp_fifo_valid ), + .rready_i ( otp_rvalid ), + .rdata_o ( otp_part_idx ), + .depth_o ( ), + .full_o ( ), + .err_o ( ) + ); + + // Steer response back to the partition where this request originated. + always_comb begin : p_rvalid + part_otp_rvalid = '0; + part_otp_rvalid[otp_part_idx] = otp_rvalid & otp_fifo_valid; + end + + // Note that this must be true by construction. + `ASSERT(OtpRespFifoUnderflow_A, otp_rvalid |-> otp_fifo_valid) + + ///////////////////////////////////////// + // Scrambling Datapath and Arbitration // + ///////////////////////////////////////// + + // Note: as opposed to the OTP arbitration above, we do not perform cycle-wise arbitration, but + // transaction-wise arbitration. This is implemented using a RR arbiter that acts as a mutex. + // I.e., each agent (e.g. the DAI or a partition) can request a lock on the mutex. Once granted, + // the partition can keep the lock as long as needed for the transaction to complete. The + // partition must yield its lock by deasserting the request signal for the arbiter to proceed. + // Since this scheme does not have built-in preemtion, it must be ensured that the agents + // eventually release their locks for this to be fair. + // + // See also https://docs.opentitan.org/hw/ip/otp_ctrl/index.html#otp-controller-overview for + // details. + typedef struct packed { + otp_scrmbl_cmd_e cmd; + digest_mode_e mode; + logic [ConstSelWidth-1:0] sel; + logic [ScrmblBlockWidth-1:0] data; + logic valid; + } scrmbl_bundle_t; + + logic [NumAgents-1:0] part_scrmbl_mtx_req, part_scrmbl_mtx_gnt; + scrmbl_bundle_t part_scrmbl_req_bundle [NumAgents]; + scrmbl_bundle_t scrmbl_req_bundle; + logic [vbits(NumAgents)-1:0] scrmbl_mtx_idx; + logic scrmbl_mtx_valid; + + // Note that arbiter decisions do not change when backpressured. + // Hence, the idx_o signal is guaranteed to remain stable until ack'ed. + prim_arbiter_tree #( + .N(NumAgents), + .DW($bits(scrmbl_bundle_t)) + ) u_scrmbl_mtx ( + .clk_i, + .rst_ni, + .req_chk_i ( 1'b0 ), // REQ is allowed to go low again without ACK even + // during normal operation. + .req_i ( part_scrmbl_mtx_req ), + .data_i ( part_scrmbl_req_bundle ), + .gnt_o ( ), + .idx_o ( scrmbl_mtx_idx ), + .valid_o ( scrmbl_mtx_valid ), + .data_o ( scrmbl_req_bundle ), + .ready_i ( 1'b0 ) + ); + + // Since the ready_i signal of the arbiter is statically set to 1'b0 above, we are always in a + // "backpressure" situation, where the RR arbiter will automatically advance the internal RR state + // to give the current winner max priority in subsequent cycles in order to keep the decision + // stable. Rearbitration occurs once the winning agent deasserts its request. + always_comb begin : p_mutex + part_scrmbl_mtx_gnt = '0; + part_scrmbl_mtx_gnt[scrmbl_mtx_idx] = scrmbl_mtx_valid; + end + + logic [ScrmblBlockWidth-1:0] part_scrmbl_rsp_data; + logic scrmbl_arb_req_ready, scrmbl_arb_rsp_valid; + logic [NumAgents-1:0] part_scrmbl_req_ready, part_scrmbl_rsp_valid; + + // SEC_CM: SECRET.MEM.SCRAMBLE + // SEC_CM: PART.MEM.DIGEST + otp_ctrl_scrmbl u_otp_ctrl_scrmbl ( + .clk_i, + .rst_ni, + .cmd_i ( scrmbl_req_bundle.cmd ), + .mode_i ( scrmbl_req_bundle.mode ), + .sel_i ( scrmbl_req_bundle.sel ), + .data_i ( scrmbl_req_bundle.data ), + .valid_i ( scrmbl_req_bundle.valid ), + .ready_o ( scrmbl_arb_req_ready ), + .data_o ( part_scrmbl_rsp_data ), + .valid_o ( scrmbl_arb_rsp_valid ), + .escalate_en_i ( lc_escalate_en[NumAgents+1] ), + .fsm_err_o ( scrmbl_fsm_err ) + ); + + // steer back responses + always_comb begin : p_scmrbl_resp + part_scrmbl_req_ready = '0; + part_scrmbl_rsp_valid = '0; + part_scrmbl_req_ready[scrmbl_mtx_idx] = scrmbl_arb_req_ready; + part_scrmbl_rsp_valid[scrmbl_mtx_idx] = scrmbl_arb_rsp_valid; + end + + ///////////////////////////// + // Direct Access Interface // + ///////////////////////////// + + logic part_init_req; + logic [NumPart-1:0] part_init_done; + part_access_t [NumPart-1:0] part_access_dai; + + // The init request comes from the power manager, which lives in the AON clock domain. + logic pwr_otp_req_synced; + prim_flop_2sync #( + .Width(1) + ) u_otp_init_sync ( + .clk_i, + .rst_ni, + .d_i ( pwr_otp_i.otp_init ), + .q_o ( pwr_otp_req_synced ) + ); + + // Register this signal as it has to cross a clock boundary. + logic pwr_otp_rsp_d, pwr_otp_rsp_q; + assign pwr_otp_o.otp_done = pwr_otp_rsp_q; + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_init_reg + if (!rst_ni) begin + pwr_otp_rsp_q <= 1'b0; + end else begin + pwr_otp_rsp_q <= pwr_otp_rsp_d; + end + end + + otp_ctrl_dai u_otp_ctrl_dai ( + .clk_i, + .rst_ni, + .init_req_i ( pwr_otp_req_synced ), + .init_done_o ( pwr_otp_rsp_d ), + .part_init_req_o ( part_init_req ), + .part_init_done_i ( part_init_done ), + .escalate_en_i ( lc_escalate_en[DaiIdx] ), + .error_o ( part_error[DaiIdx] ), + .fsm_err_o ( part_fsm_err[DaiIdx] ), + .part_access_i ( part_access_dai ), + .dai_addr_i ( dai_addr ), + .dai_cmd_i ( dai_cmd ), + .dai_req_i ( dai_req ), + .dai_wdata_i ( dai_wdata ), + .dai_idle_o ( dai_idle ), + .dai_prog_idle_o ( dai_prog_idle ), + .dai_cmd_done_o ( otp_operation_done ), + .dai_rdata_o ( dai_rdata ), + .otp_req_o ( part_otp_arb_req[DaiIdx] ), + .otp_cmd_o ( part_otp_arb_bundle[DaiIdx].cmd ), + .otp_size_o ( part_otp_arb_bundle[DaiIdx].size ), + .otp_wdata_o ( part_otp_arb_bundle[DaiIdx].wdata ), + .otp_addr_o ( part_otp_arb_bundle[DaiIdx].addr ), + .otp_gnt_i ( part_otp_arb_gnt[DaiIdx] ), + .otp_rvalid_i ( part_otp_rvalid[DaiIdx] ), + .otp_rdata_i ( part_otp_rdata ), + .otp_err_i ( part_otp_err ), + .scrmbl_mtx_req_o ( part_scrmbl_mtx_req[DaiIdx] ), + .scrmbl_mtx_gnt_i ( part_scrmbl_mtx_gnt[DaiIdx] ), + .scrmbl_cmd_o ( part_scrmbl_req_bundle[DaiIdx].cmd ), + .scrmbl_mode_o ( part_scrmbl_req_bundle[DaiIdx].mode ), + .scrmbl_sel_o ( part_scrmbl_req_bundle[DaiIdx].sel ), + .scrmbl_data_o ( part_scrmbl_req_bundle[DaiIdx].data ), + .scrmbl_valid_o ( part_scrmbl_req_bundle[DaiIdx].valid ), + .scrmbl_ready_i ( part_scrmbl_req_ready[DaiIdx] ), + .scrmbl_valid_i ( part_scrmbl_rsp_valid[DaiIdx] ), + .scrmbl_data_i ( part_scrmbl_rsp_data ) + ); + + //////////////////////////////////// + // Lifecycle Transition Interface // + //////////////////////////////////// + + logic [PartInfo[LifeCycleIdx].size-1:0][7:0] lc_otp_program_data; + assign lc_otp_program_data[LcStateOffset-LifeCycleOffset +: LcStateSize] = + lc_otp_program_i.state; + assign lc_otp_program_data[LcTransitionCntOffset-LifeCycleOffset +: LcTransitionCntSize] = + lc_otp_program_i.count; + + otp_ctrl_lci #( + .Info(PartInfo[LifeCycleIdx]) + ) u_otp_ctrl_lci ( + .clk_i, + .rst_ni, + .lci_en_i ( pwr_otp_o.otp_done ), + .escalate_en_i ( lc_escalate_en[LciIdx] ), + .error_o ( part_error[LciIdx] ), + .fsm_err_o ( part_fsm_err[LciIdx] ), + .lci_prog_idle_o ( lci_prog_idle ), + .lc_req_i ( lc_otp_program_i.req ), + .lc_data_i ( lc_otp_program_data ), + .lc_ack_o ( lc_otp_program_o.ack ), + .lc_err_o ( lc_otp_program_o.err ), + .otp_req_o ( part_otp_arb_req[LciIdx] ), + .otp_cmd_o ( part_otp_arb_bundle[LciIdx].cmd ), + .otp_size_o ( part_otp_arb_bundle[LciIdx].size ), + .otp_wdata_o ( part_otp_arb_bundle[LciIdx].wdata ), + .otp_addr_o ( part_otp_arb_bundle[LciIdx].addr ), + .otp_gnt_i ( part_otp_arb_gnt[LciIdx] ), + .otp_rvalid_i ( part_otp_rvalid[LciIdx] ), + .otp_rdata_i ( part_otp_rdata ), + .otp_err_i ( part_otp_err ) + ); + + // Tie off unused connections. + assign part_scrmbl_mtx_req[LciIdx] = '0; + assign part_scrmbl_req_bundle[LciIdx] = '0; + + // This stops lint from complaining about unused signals. + logic unused_lci_scrmbl_sigs; + assign unused_lci_scrmbl_sigs = ^{part_scrmbl_mtx_gnt[LciIdx], + part_scrmbl_req_ready[LciIdx], + part_scrmbl_rsp_valid[LciIdx]}; + + //////////////////////////////////// + // Key Derivation Interface (KDI) // + //////////////////////////////////// + + logic scrmbl_key_seed_valid; + logic [SramKeySeedWidth-1:0] sram_data_key_seed; + logic [FlashKeySeedWidth-1:0] flash_data_key_seed, flash_addr_key_seed; + + otp_ctrl_kdi #( + .RndCnstScrmblKeyInit(RndCnstScrmblKeyInit) + ) u_otp_ctrl_kdi ( + .clk_i, + .rst_ni, + .kdi_en_i ( pwr_otp_o.otp_done ), + .escalate_en_i ( lc_escalate_en[KdiIdx] ), + .fsm_err_o ( part_fsm_err[KdiIdx] ), + .scrmbl_key_seed_valid_i ( scrmbl_key_seed_valid ), + .flash_data_key_seed_i ( flash_data_key_seed ), + .flash_addr_key_seed_i ( flash_addr_key_seed ), + .sram_data_key_seed_i ( sram_data_key_seed ), + .edn_req_o ( key_edn_req ), + .edn_ack_i ( key_edn_ack ), + .edn_data_i ( edn_data ), + .flash_otp_key_i, + .flash_otp_key_o, + .sram_otp_key_i, + .sram_otp_key_o, + .otbn_otp_key_i, + .otbn_otp_key_o, + .scrmbl_mtx_req_o ( part_scrmbl_mtx_req[KdiIdx] ), + .scrmbl_mtx_gnt_i ( part_scrmbl_mtx_gnt[KdiIdx] ), + .scrmbl_cmd_o ( part_scrmbl_req_bundle[KdiIdx].cmd ), + .scrmbl_mode_o ( part_scrmbl_req_bundle[KdiIdx].mode ), + .scrmbl_sel_o ( part_scrmbl_req_bundle[KdiIdx].sel ), + .scrmbl_data_o ( part_scrmbl_req_bundle[KdiIdx].data ), + .scrmbl_valid_o ( part_scrmbl_req_bundle[KdiIdx].valid ), + .scrmbl_ready_i ( part_scrmbl_req_ready[KdiIdx] ), + .scrmbl_valid_i ( part_scrmbl_rsp_valid[KdiIdx] ), + .scrmbl_data_i ( part_scrmbl_rsp_data ) + ); + + // Tie off OTP bus access, since this is not needed. + assign part_otp_arb_req[KdiIdx] = 1'b0; + assign part_otp_arb_bundle[KdiIdx] = '0; + + // This stops lint from complaining about unused signals. + logic unused_kdi_otp_sigs; + assign unused_kdi_otp_sigs = ^{part_otp_arb_gnt[KdiIdx], + part_otp_rvalid[KdiIdx]}; + + ///////////////////////// + // Partition Instances // + ///////////////////////// + + logic [$bits(PartInvDefault)/8-1:0][7:0] part_buf_data; + + for (genvar k = 0; k < NumPart; k ++) begin : gen_partitions + //////////////////////////////////////////////////////////////////////////////////////////////// + if (PartInfo[k].variant == Unbuffered) begin : gen_unbuffered + otp_ctrl_part_unbuf #( + .Info(PartInfo[k]) + ) u_part_unbuf ( + .clk_i, + .rst_ni, + .init_req_i ( part_init_req ), + .init_done_o ( part_init_done[k] ), + .escalate_en_i ( lc_escalate_en[k] ), + .error_o ( part_error[k] ), + .fsm_err_o ( part_fsm_err[k] ), + .access_i ( part_access[k] ), + .access_o ( part_access_dai[k] ), + .digest_o ( part_digest[k] ), + .tlul_req_i ( part_tlul_req[k] ), + .tlul_gnt_o ( part_tlul_gnt[k] ), + .tlul_addr_i ( part_tlul_addr ), + .tlul_rerror_o ( part_tlul_rerror[k] ), + .tlul_rvalid_o ( part_tlul_rvalid[k] ), + .tlul_rdata_o ( part_tlul_rdata[k] ), + .otp_req_o ( part_otp_arb_req[k] ), + .otp_cmd_o ( part_otp_arb_bundle[k].cmd ), + .otp_size_o ( part_otp_arb_bundle[k].size ), + .otp_wdata_o ( part_otp_arb_bundle[k].wdata ), + .otp_addr_o ( part_otp_arb_bundle[k].addr ), + .otp_gnt_i ( part_otp_arb_gnt[k] ), + .otp_rvalid_i ( part_otp_rvalid[k] ), + .otp_rdata_i ( part_otp_rdata ), + .otp_err_i ( part_otp_err ) + ); + + // Tie off unused connections. + assign part_scrmbl_mtx_req[k] = '0; + assign part_scrmbl_req_bundle[k] = '0; + // These checks do not exist in this partition type, + // so we always acknowledge the request. + assign integ_chk_ack[k] = 1'b1; + assign cnsty_chk_ack[k] = 1'b1; + + // No buffered data to expose. + assign part_buf_data[PartInfo[k].offset +: PartInfo[k].size] = '0; + + // This stops lint from complaining about unused signals. + logic unused_part_scrmbl_sigs; + assign unused_part_scrmbl_sigs = ^{part_scrmbl_mtx_gnt[k], + part_scrmbl_req_ready[k], + part_scrmbl_rsp_valid[k], + integ_chk_req[k], + cnsty_chk_req[k]}; + + // Alert assertion for sparse FSM. + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlPartUnbufFsmCheck_A, + u_part_unbuf.u_state_regs, alert_tx_o[1]) + //////////////////////////////////////////////////////////////////////////////////////////////// + end else if (PartInfo[k].variant == Buffered) begin : gen_buffered + otp_ctrl_part_buf #( + .Info(PartInfo[k]), + .DataDefault(PartInvDefault[PartInfo[k].offset*8 +: PartInfo[k].size*8]) + ) u_part_buf ( + .clk_i, + .rst_ni, + .init_req_i ( part_init_req ), + .init_done_o ( part_init_done[k] ), + .integ_chk_req_i ( integ_chk_req[k] ), + .integ_chk_ack_o ( integ_chk_ack[k] ), + .cnsty_chk_req_i ( cnsty_chk_req[k] ), + .cnsty_chk_ack_o ( cnsty_chk_ack[k] ), + .escalate_en_i ( lc_escalate_en[k] ), + // Only supported by life cycle partition (see further below). + .check_byp_en_i ( lc_ctrl_pkg::Off ), + .error_o ( part_error[k] ), + .fsm_err_o ( part_fsm_err[k] ), + .access_i ( part_access[k] ), + .access_o ( part_access_dai[k] ), + .digest_o ( part_digest[k] ), + .data_o ( part_buf_data[PartInfo[k].offset +: PartInfo[k].size] ), + .otp_req_o ( part_otp_arb_req[k] ), + .otp_cmd_o ( part_otp_arb_bundle[k].cmd ), + .otp_size_o ( part_otp_arb_bundle[k].size ), + .otp_wdata_o ( part_otp_arb_bundle[k].wdata ), + .otp_addr_o ( part_otp_arb_bundle[k].addr ), + .otp_gnt_i ( part_otp_arb_gnt[k] ), + .otp_rvalid_i ( part_otp_rvalid[k] ), + .otp_rdata_i ( part_otp_rdata ), + .otp_err_i ( part_otp_err ), + .scrmbl_mtx_req_o ( part_scrmbl_mtx_req[k] ), + .scrmbl_mtx_gnt_i ( part_scrmbl_mtx_gnt[k] ), + .scrmbl_cmd_o ( part_scrmbl_req_bundle[k].cmd ), + .scrmbl_mode_o ( part_scrmbl_req_bundle[k].mode ), + .scrmbl_sel_o ( part_scrmbl_req_bundle[k].sel ), + .scrmbl_data_o ( part_scrmbl_req_bundle[k].data ), + .scrmbl_valid_o ( part_scrmbl_req_bundle[k].valid ), + .scrmbl_ready_i ( part_scrmbl_req_ready[k] ), + .scrmbl_valid_i ( part_scrmbl_rsp_valid[k] ), + .scrmbl_data_i ( part_scrmbl_rsp_data ) + ); + + // Buffered partitions are not accessible via the TL-UL window. + logic unused_part_tlul_sigs; + assign unused_part_tlul_sigs = ^part_tlul_req[k]; + assign part_tlul_gnt[k] = 1'b0; + assign part_tlul_rerror[k] = '0; + assign part_tlul_rvalid[k] = 1'b0; + assign part_tlul_rdata[k] = '0; + + // Alert assertion for sparse FSM. + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlPartBufFsmCheck_A, + u_part_buf.u_state_regs, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntPartBufCheck_A, + u_part_buf.u_prim_count, alert_tx_o[1]) + //////////////////////////////////////////////////////////////////////////////////////////////// + end else if (PartInfo[k].variant == LifeCycle) begin : gen_lifecycle + otp_ctrl_part_buf #( + .Info(PartInfo[k]), + .DataDefault(PartInvDefault[PartInfo[k].offset*8 +: PartInfo[k].size*8]) + ) u_part_buf ( + .clk_i, + .rst_ni, + .init_req_i ( part_init_req ), + .init_done_o ( part_init_done[k] ), + .integ_chk_req_i ( integ_chk_req[k] ), + .integ_chk_ack_o ( integ_chk_ack[k] ), + .cnsty_chk_req_i ( cnsty_chk_req[k] ), + .cnsty_chk_ack_o ( cnsty_chk_ack[k] ), + .escalate_en_i ( lc_escalate_en[k] ), + // This is only supported by the life cycle partition. We need to prevent this partition + // from escalating once the life cycle state in memory is being updated (and hence not + // consistent with the values in the buffer regs anymore). + .check_byp_en_i ( lc_check_byp_en ), + .error_o ( part_error[k] ), + .fsm_err_o ( part_fsm_err[k] ), + .access_i ( part_access[k] ), + .access_o ( part_access_dai[k] ), + .digest_o ( part_digest[k] ), + .data_o ( part_buf_data[PartInfo[k].offset +: PartInfo[k].size] ), + .otp_req_o ( part_otp_arb_req[k] ), + .otp_cmd_o ( part_otp_arb_bundle[k].cmd ), + .otp_size_o ( part_otp_arb_bundle[k].size ), + .otp_wdata_o ( part_otp_arb_bundle[k].wdata ), + .otp_addr_o ( part_otp_arb_bundle[k].addr ), + .otp_gnt_i ( part_otp_arb_gnt[k] ), + .otp_rvalid_i ( part_otp_rvalid[k] ), + .otp_rdata_i ( part_otp_rdata ), + .otp_err_i ( part_otp_err ), + // The LC partition does not need any scrambling features. + .scrmbl_mtx_req_o ( ), + .scrmbl_mtx_gnt_i ( 1'b0 ), + .scrmbl_cmd_o ( ), + .scrmbl_mode_o ( ), + .scrmbl_sel_o ( ), + .scrmbl_data_o ( ), + .scrmbl_valid_o ( ), + .scrmbl_ready_i ( 1'b0 ), + .scrmbl_valid_i ( 1'b0 ), + .scrmbl_data_i ( '0 ) + ); + + // Buffered partitions are not accessible via the TL-UL window. + logic unused_part_tlul_sigs; + assign unused_part_tlul_sigs = ^part_tlul_req[k]; + assign part_tlul_gnt[k] = 1'b0; + assign part_tlul_rerror[k] = '0; + assign part_tlul_rvalid[k] = 1'b0; + assign part_tlul_rdata[k] = '0; + + // Tie off unused connections. + assign part_scrmbl_mtx_req[k] = '0; + assign part_scrmbl_req_bundle[k] = '0; + + // This stops lint from complaining about unused signals. + logic unused_part_scrmbl_sigs; + assign unused_part_scrmbl_sigs = ^{part_scrmbl_mtx_gnt[k], + part_scrmbl_req_ready[k], + part_scrmbl_rsp_valid[k]}; + // Alert assertion for sparse FSM. + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlPartLcFsmCheck_A, + u_part_buf.u_state_regs, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntPartLcCheck_A, + u_part_buf.u_prim_count, alert_tx_o[1]) + //////////////////////////////////////////////////////////////////////////////////////////////// + end else begin : gen_invalid + // This is invalid and should break elaboration + assert_static_in_generate_invalid assert_static_in_generate_invalid(); + end + //////////////////////////////////////////////////////////////////////////////////////////////// + end + + ////////////////////////////////// + // Buffered Data Output Mapping // + ////////////////////////////////// + + // Output complete hardware config partition. + // Actual mapping to other IPs is done via the intersignal topgen feature, + // selection of fields can be done using the otp_hw_cfg_t struct fields. + otp_broadcast_t otp_broadcast; + assign otp_broadcast = named_broadcast_assign(part_init_done, part_buf_data); + + // Make sure the broadcast valid is flopped before sending it out. + lc_ctrl_pkg::lc_tx_t otp_broadcast_valid_q; + prim_lc_sender u_prim_lc_sender_otp_broadcast_valid ( + .clk_i, + .rst_ni, + .lc_en_i(otp_broadcast.valid), + .lc_en_o(otp_broadcast_valid_q) + ); + + always_comb begin : p_otp_broadcast_valid + otp_broadcast_o = otp_broadcast; + otp_broadcast_o.valid = otp_broadcast_valid_q; + end + + // Root keys and seeds. + // This uses a generated function to assign all collateral that is marked with "iskeymgr" in + // the memory map. Note that in this case the type is static and represents a superset of all + // options so that we can maintain a stable interface with keymgr (otherwise keymgr will have + // to be templated as well. Unused key material will be tied off to '0. The keymgr has to be + // parameterized accordingly (via SV parameters) to consume the correct key material. + // + // The key material valid signals are set to true if the corresponding digest is nonzero and the + // partition is initialized. On top of that, the entire output is gated by lc_seed_hw_rd_en. + otp_keymgr_key_t otp_keymgr_key; + assign otp_keymgr_key = named_keymgr_key_assign(part_digest, + part_buf_data, + lc_seed_hw_rd_en); + + // Note regarding these breakouts: named_keymgr_key_assign will tie off unused key material / + // valid signals to '0. This is the case for instance in system configurations that keep the seed + // material in the flash instead of OTP. + logic creator_root_key_share0_valid_d, creator_root_key_share0_valid_q; + logic creator_root_key_share1_valid_d, creator_root_key_share1_valid_q; + logic creator_seed_valid_d, creator_seed_valid_q; + logic owner_seed_valid_d, owner_seed_valid_q; + prim_flop #( + .Width(4) + ) u_keygmr_key_valid ( + .clk_i, + .rst_ni, + .d_i ({creator_root_key_share0_valid_d, + creator_root_key_share1_valid_d, + creator_seed_valid_d, + owner_seed_valid_d}), + .q_o ({creator_root_key_share0_valid_q, + creator_root_key_share1_valid_q, + creator_seed_valid_q, + owner_seed_valid_q}) + ); + + always_comb begin : p_otp_keymgr_key_valid + // Valid reg inputs + creator_root_key_share0_valid_d = otp_keymgr_key.creator_root_key_share0_valid; + creator_root_key_share1_valid_d = otp_keymgr_key.creator_root_key_share1_valid; + creator_seed_valid_d = otp_keymgr_key.creator_seed_valid; + owner_seed_valid_d = otp_keymgr_key.owner_seed_valid; + // Output to keymgr + otp_keymgr_key_o = otp_keymgr_key; + otp_keymgr_key_o.creator_root_key_share0_valid = creator_root_key_share0_valid_q; + otp_keymgr_key_o.creator_root_key_share1_valid = creator_root_key_share1_valid_q; + otp_keymgr_key_o.creator_seed_valid = creator_seed_valid_q; + otp_keymgr_key_o.owner_seed_valid = owner_seed_valid_q; + end + + // Check that the lc_seed_hw_rd_en remains stable, once the key material is valid. + `ASSERT(LcSeedHwRdEnStable0_A, + $rose(creator_root_key_share0_valid_q) |=> $stable(lc_seed_hw_rd_en) [*1:$], + clk_i, !rst_ni || lc_ctrl_pkg::lc_tx_test_true_loose(lc_escalate_en_i) // Disable if escalating + ) + `ASSERT(LcSeedHwRdEnStable1_A, + $rose(creator_root_key_share1_valid_q) |=> $stable(lc_seed_hw_rd_en) [*1:$], + clk_i, !rst_ni || lc_ctrl_pkg::lc_tx_test_true_loose(lc_escalate_en_i) // Disable if escalating + ) + `ASSERT(LcSeedHwRdEnStable2_A, + $rose(creator_seed_valid_q) |=> $stable(lc_seed_hw_rd_en) [*1:$], + clk_i, !rst_ni || lc_ctrl_pkg::lc_tx_test_true_loose(lc_escalate_en_i) // Disable if escalating + ) + `ASSERT(LcSeedHwRdEnStable3_A, + $rose(owner_seed_valid_q) |=> $stable(lc_seed_hw_rd_en) [*1:$], + clk_i, !rst_ni || lc_ctrl_pkg::lc_tx_test_true_loose(lc_escalate_en_i) // Disable if escalating + ) + + // Scrambling Keys + assign scrmbl_key_seed_valid = part_digest[Secret1Idx] != '0; + assign sram_data_key_seed = part_buf_data[SramDataKeySeedOffset +: + SramDataKeySeedSize]; + assign flash_data_key_seed = part_buf_data[FlashDataKeySeedOffset +: + FlashDataKeySeedSize]; + assign flash_addr_key_seed = part_buf_data[FlashAddrKeySeedOffset +: + FlashAddrKeySeedSize]; + + // Test unlock and exit tokens and RMA token + assign otp_lc_data_o.test_exit_token = part_buf_data[TestExitTokenOffset +: + TestExitTokenSize]; + assign otp_lc_data_o.test_unlock_token = part_buf_data[TestUnlockTokenOffset +: + TestUnlockTokenSize]; + assign otp_lc_data_o.rma_token = part_buf_data[RmaTokenOffset +: + RmaTokenSize]; + + lc_ctrl_pkg::lc_tx_t test_tokens_valid, rma_token_valid, secrets_valid; + // The test tokens have been provisioned. + assign test_tokens_valid = (part_digest[Secret0Idx] != '0) ? lc_ctrl_pkg::On : lc_ctrl_pkg::Off; + // The rma token has been provisioned. + assign rma_token_valid = (part_digest[Secret2Idx] != '0) ? lc_ctrl_pkg::On : lc_ctrl_pkg::Off; + // The device is personalized if the root key has been provisioned and locked. + assign secrets_valid = (part_digest[Secret2Idx] != '0) ? lc_ctrl_pkg::On : lc_ctrl_pkg::Off; + + // Buffer these constants in order to ensure that synthesis does not try to optimize the encoding. + // SEC_CM: TOKEN_VALID.CTRL.MUBI + prim_lc_sender #( + .AsyncOn(0) + ) u_prim_lc_sender_test_tokens_valid ( + .clk_i, + .rst_ni, + .lc_en_i(test_tokens_valid), + .lc_en_o(otp_lc_data_o.test_tokens_valid) + ); + + prim_lc_sender #( + .AsyncOn(0) + ) u_prim_lc_sender_rma_token_valid ( + .clk_i, + .rst_ni, + .lc_en_i(rma_token_valid), + .lc_en_o(otp_lc_data_o.rma_token_valid) + ); + + prim_lc_sender #( + .AsyncOn(0) + ) u_prim_lc_sender_secrets_valid ( + .clk_i, + .rst_ni, + .lc_en_i(secrets_valid), + .lc_en_o(otp_lc_data_o.secrets_valid) + ); + + // Lifecycle state + assign otp_lc_data_o.state = lc_ctrl_state_pkg::lc_state_e'(part_buf_data[LcStateOffset +: + LcStateSize]); + assign otp_lc_data_o.count = lc_ctrl_state_pkg::lc_cnt_e'(part_buf_data[LcTransitionCntOffset +: + LcTransitionCntSize]); + + // Assert life cycle state valid signal only when all partitions have initialized. + assign otp_lc_data_o.valid = &part_init_done; + // Signal whether there are any errors in the life cycle partition (both correctable and + // uncorrectable ones). This bit is made available via the JTAG TAP, which is useful for + // production testing in RAW life cycle state where the OTP regs are not accessible. + assign otp_lc_data_o.error = |part_error[LifeCycleIdx]; + + // Not all bits of part_buf_data are used here. + logic unused_buf_data; + assign unused_buf_data = ^part_buf_data; + + //////////////// + // Assertions // + //////////////// + + `ASSERT_INIT(CreatorRootKeyShare0Size_A, KeyMgrKeyWidth == CreatorRootKeyShare0Size * 8) + `ASSERT_INIT(CreatorRootKeyShare1Size_A, KeyMgrKeyWidth == CreatorRootKeyShare1Size * 8) + `ASSERT_INIT(FlashDataKeySeedSize_A, FlashKeySeedWidth == FlashDataKeySeedSize * 8) + `ASSERT_INIT(FlashAddrKeySeedSize_A, FlashKeySeedWidth == FlashAddrKeySeedSize * 8) + `ASSERT_INIT(SramDataKeySeedSize_A, SramKeySeedWidth == SramDataKeySeedSize * 8) + + `ASSERT_INIT(RmaTokenSize_A, lc_ctrl_state_pkg::LcTokenWidth == RmaTokenSize * 8) + `ASSERT_INIT(TestUnlockTokenSize_A, lc_ctrl_state_pkg::LcTokenWidth == TestUnlockTokenSize * 8) + `ASSERT_INIT(TestExitTokenSize_A, lc_ctrl_state_pkg::LcTokenWidth == TestExitTokenSize * 8) + `ASSERT_INIT(LcStateSize_A, lc_ctrl_state_pkg::LcStateWidth == LcStateSize * 8) + `ASSERT_INIT(LcTransitionCntSize_A, lc_ctrl_state_pkg::LcCountWidth == LcTransitionCntSize * 8) + + `ASSERT_KNOWN(OtpAstPwrSeqKnown_A, otp_ast_pwr_seq_o) + `ASSERT_KNOWN(CoreTlOutKnown_A, core_tl_o) + `ASSERT_KNOWN(PrimTlOutKnown_A, prim_tl_o) + `ASSERT_KNOWN(IntrOtpOperationDoneKnown_A, intr_otp_operation_done_o) + `ASSERT_KNOWN(IntrOtpErrorKnown_A, intr_otp_error_o) + `ASSERT_KNOWN(AlertTxKnown_A, alert_tx_o) + `ASSERT_KNOWN(PwrOtpInitRspKnown_A, pwr_otp_o) + `ASSERT_KNOWN(LcOtpProgramRspKnown_A, lc_otp_program_o) + `ASSERT_KNOWN(OtpLcDataKnown_A, otp_lc_data_o) + `ASSERT_KNOWN(OtpKeymgrKeyKnown_A, otp_keymgr_key_o) + `ASSERT_KNOWN(FlashOtpKeyRspKnown_A, flash_otp_key_o) + `ASSERT_KNOWN(OtpSramKeyKnown_A, sram_otp_key_o) + `ASSERT_KNOWN(OtpOtgnKeyKnown_A, otbn_otp_key_o) + `ASSERT_KNOWN(OtpBroadcastKnown_A, otp_broadcast_o) + + // Alert assertions for sparse FSMs. + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlDaiFsmCheck_A, + u_otp_ctrl_dai.u_state_regs, alert_tx_o[1]) + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlKdiFsmCheck_A, + u_otp_ctrl_kdi.u_state_regs, alert_tx_o[1]) + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlLciFsmCheck_A, + u_otp_ctrl_lci.u_state_regs, alert_tx_o[1]) + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlLfsrTimerFsmCheck_A, + u_otp_ctrl_lfsr_timer.u_state_regs, alert_tx_o[1]) + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlScrambleFsmCheck_A, + u_otp_ctrl_scrmbl.u_state_regs, alert_tx_o[1]) + + // Alert assertions for redundant counters. + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntIntegCheck_A, + u_otp_ctrl_lfsr_timer.u_prim_count_integ, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntCnstyCheck_A, + u_otp_ctrl_lfsr_timer.u_prim_count_cnsty, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntDaiCheck_A, + u_otp_ctrl_dai.u_prim_count, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntKdiSeedCheck_A, + u_otp_ctrl_kdi.u_prim_count_seed, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntKdiEntropyCheck_A, + u_otp_ctrl_kdi.u_prim_count_entropy, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntLciCheck_A, + u_otp_ctrl_lci.u_prim_count, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntScrmblCheck_A, + u_otp_ctrl_scrmbl.u_prim_count, alert_tx_o[1]) + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(TlLcGateFsm_A, + u_tlul_lc_gate.u_state_regs, alert_tx_o[2]) + + // Alert assertions for double LFSR. + `ASSERT_PRIM_DOUBLE_LFSR_ERROR_TRIGGER_ALERT(DoubleLfsrCheck_A, + u_otp_ctrl_lfsr_timer.u_prim_double_lfsr, alert_tx_o[1]) + + // Alert assertions for reg_we onehot check + `ASSERT_PRIM_REG_WE_ONEHOT_ERROR_TRIGGER_ALERT(RegWeOnehotCheck_A, u_reg_core, alert_tx_o[2]) + + // Assertions for countermeasures inside prim_otp + `ifndef PRIM_DEFAULT_IMPL + `define PRIM_DEFAULT_IMPL prim_pkg::ImplGeneric + `endif + if (`PRIM_DEFAULT_IMPL == prim_pkg::ImplGeneric) begin : gen_reg_we_assert_generic + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(PrimFsmCheck_A, + u_otp.gen_generic.u_impl_generic.u_state_regs, alert_tx_o[3]) + `ASSERT_PRIM_REG_WE_ONEHOT_ERROR_TRIGGER_ALERT(PrimRegWeOnehotCheck_A, + u_otp.gen_generic.u_impl_generic.u_reg_top, alert_tx_o[3]) + end +endmodule : otp_ctrl diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_core_reg_top.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_core_reg_top.sv new file mode 100644 index 00000000000000..ae63afe41d6496 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_core_reg_top.sv @@ -0,0 +1,3997 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Register Top module auto-generated by `reggen` + +`include "prim_assert.sv" + +module otp_ctrl_core_reg_top ( + input clk_i, + input rst_ni, + input tlul_pkg::tl_h2d_t tl_i, + output tlul_pkg::tl_d2h_t tl_o, + + // Output port for window + output tlul_pkg::tl_h2d_t tl_win_o, + input tlul_pkg::tl_d2h_t tl_win_i, + + // To HW + output otp_ctrl_reg_pkg::otp_ctrl_core_reg2hw_t reg2hw, // Write + input otp_ctrl_reg_pkg::otp_ctrl_core_hw2reg_t hw2reg, // Read + + // Integrity check errors + output logic intg_err_o +); + + import otp_ctrl_reg_pkg::* ; + + localparam int AW = 15; + localparam int DW = 32; + localparam int DBW = DW/8; // Byte Width + + // register signals + logic reg_we; + logic reg_re; + logic [AW-1:0] reg_addr; + logic [DW-1:0] reg_wdata; + logic [DBW-1:0] reg_be; + logic [DW-1:0] reg_rdata; + logic reg_error; + + logic addrmiss, wr_err; + + logic [DW-1:0] reg_rdata_next; + logic reg_busy; + + tlul_pkg::tl_h2d_t tl_reg_h2d; + tlul_pkg::tl_d2h_t tl_reg_d2h; + + + // incoming payload check + logic intg_err; + tlul_cmd_intg_chk u_chk ( + .tl_i(tl_i), + .err_o(intg_err) + ); + + // also check for spurious write enables + logic reg_we_err; + logic [94:0] reg_we_check; + prim_reg_we_check #( + .OneHotWidth(95) + ) u_prim_reg_we_check ( + .clk_i(clk_i), + .rst_ni(rst_ni), + .oh_i (reg_we_check), + .en_i (reg_we && !addrmiss), + .err_o (reg_we_err) + ); + + logic err_q; + always_ff @(posedge clk_i or negedge rst_ni) begin + if (!rst_ni) begin + err_q <= '0; + end else if (intg_err || reg_we_err) begin + err_q <= 1'b1; + end + end + + // integrity error output is permanent and should be used for alert generation + // register errors are transactional + assign intg_err_o = err_q | intg_err | reg_we_err; + + // outgoing integrity generation + tlul_pkg::tl_d2h_t tl_o_pre; + tlul_rsp_intg_gen #( + .EnableRspIntgGen(1), + .EnableDataIntgGen(1) + ) u_rsp_intg_gen ( + .tl_i(tl_o_pre), + .tl_o(tl_o) + ); + + tlul_pkg::tl_h2d_t tl_socket_h2d [2]; + tlul_pkg::tl_d2h_t tl_socket_d2h [2]; + + logic [0:0] reg_steer; + + // socket_1n connection + assign tl_reg_h2d = tl_socket_h2d[1]; + assign tl_socket_d2h[1] = tl_reg_d2h; + + assign tl_win_o = tl_socket_h2d[0]; + assign tl_socket_d2h[0] = tl_win_i; + + // Create Socket_1n + tlul_socket_1n #( + .N (2), + .HReqPass (1'b1), + .HRspPass (1'b1), + .DReqPass ({2{1'b1}}), + .DRspPass ({2{1'b1}}), + .HReqDepth (4'h0), + .HRspDepth (4'h0), + .DReqDepth ({2{4'h0}}), + .DRspDepth ({2{4'h0}}), + .ExplicitErrs (1'b0) + ) u_socket ( + .clk_i (clk_i), + .rst_ni (rst_ni), + .tl_h_i (tl_i), + .tl_h_o (tl_o_pre), + .tl_d_o (tl_socket_h2d), + .tl_d_i (tl_socket_d2h), + .dev_select_i (reg_steer) + ); + + // Create steering logic + always_comb begin + reg_steer = + tl_i.a_address[AW-1:0] inside {[16384:32767]} ? 1'd0 : + // Default set to register + 1'd1; + + // Override this in case of an integrity error + if (intg_err) begin + reg_steer = 1'd1; + end + end + + tlul_adapter_reg #( + .RegAw(AW), + .RegDw(DW), + .EnableDataIntgGen(0) + ) u_reg_if ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + .tl_i (tl_reg_h2d), + .tl_o (tl_reg_d2h), + + .en_ifetch_i(prim_mubi_pkg::MuBi4False), + .intg_error_o(), + + .we_o (reg_we), + .re_o (reg_re), + .addr_o (reg_addr), + .wdata_o (reg_wdata), + .be_o (reg_be), + .busy_i (reg_busy), + .rdata_i (reg_rdata), + .error_i (reg_error) + ); + + // cdc oversampling signals + + assign reg_rdata = reg_rdata_next ; + assign reg_error = addrmiss | wr_err | intg_err; + + // Define SW related signals + // Format: __{wd|we|qs} + // or _{wd|we|qs} if field == 1 or 0 + logic intr_state_we; + logic intr_state_otp_operation_done_qs; + logic intr_state_otp_operation_done_wd; + logic intr_state_otp_error_qs; + logic intr_state_otp_error_wd; + logic intr_enable_we; + logic intr_enable_otp_operation_done_qs; + logic intr_enable_otp_operation_done_wd; + logic intr_enable_otp_error_qs; + logic intr_enable_otp_error_wd; + logic intr_test_we; + logic intr_test_otp_operation_done_wd; + logic intr_test_otp_error_wd; + logic alert_test_we; + logic alert_test_fatal_macro_error_wd; + logic alert_test_fatal_check_error_wd; + logic alert_test_fatal_bus_integ_error_wd; + logic alert_test_fatal_prim_otp_alert_wd; + logic alert_test_recov_prim_otp_alert_wd; + logic status_re; + logic status_vendor_test_error_qs; + logic status_creator_sw_cfg_error_qs; + logic status_owner_sw_cfg_error_qs; + logic status_ownership_slot_state_error_qs; + logic status_rot_creator_auth_error_qs; + logic status_rot_owner_auth_slot0_error_qs; + logic status_rot_owner_auth_slot1_error_qs; + logic status_plat_integ_auth_slot0_error_qs; + logic status_plat_integ_auth_slot1_error_qs; + logic status_plat_owner_auth_slot0_error_qs; + logic status_plat_owner_auth_slot1_error_qs; + logic status_plat_owner_auth_slot2_error_qs; + logic status_plat_owner_auth_slot3_error_qs; + logic status_ext_nvm_error_qs; + logic status_rom_patch_error_qs; + logic status_hw_cfg0_error_qs; + logic status_hw_cfg1_error_qs; + logic status_secret0_error_qs; + logic status_secret1_error_qs; + logic status_secret2_error_qs; + logic status_secret3_error_qs; + logic status_life_cycle_error_qs; + logic status_dai_error_qs; + logic status_lci_error_qs; + logic status_timeout_error_qs; + logic status_lfsr_fsm_error_qs; + logic status_scrambling_fsm_error_qs; + logic status_key_deriv_fsm_error_qs; + logic status_bus_integ_error_qs; + logic status_dai_idle_qs; + logic status_check_pending_qs; + logic err_code_0_re; + logic [2:0] err_code_0_qs; + logic err_code_1_re; + logic [2:0] err_code_1_qs; + logic err_code_2_re; + logic [2:0] err_code_2_qs; + logic err_code_3_re; + logic [2:0] err_code_3_qs; + logic err_code_4_re; + logic [2:0] err_code_4_qs; + logic err_code_5_re; + logic [2:0] err_code_5_qs; + logic err_code_6_re; + logic [2:0] err_code_6_qs; + logic err_code_7_re; + logic [2:0] err_code_7_qs; + logic err_code_8_re; + logic [2:0] err_code_8_qs; + logic err_code_9_re; + logic [2:0] err_code_9_qs; + logic err_code_10_re; + logic [2:0] err_code_10_qs; + logic err_code_11_re; + logic [2:0] err_code_11_qs; + logic err_code_12_re; + logic [2:0] err_code_12_qs; + logic err_code_13_re; + logic [2:0] err_code_13_qs; + logic err_code_14_re; + logic [2:0] err_code_14_qs; + logic err_code_15_re; + logic [2:0] err_code_15_qs; + logic err_code_16_re; + logic [2:0] err_code_16_qs; + logic err_code_17_re; + logic [2:0] err_code_17_qs; + logic err_code_18_re; + logic [2:0] err_code_18_qs; + logic err_code_19_re; + logic [2:0] err_code_19_qs; + logic err_code_20_re; + logic [2:0] err_code_20_qs; + logic err_code_21_re; + logic [2:0] err_code_21_qs; + logic err_code_22_re; + logic [2:0] err_code_22_qs; + logic err_code_23_re; + logic [2:0] err_code_23_qs; + logic direct_access_regwen_re; + logic direct_access_regwen_we; + logic direct_access_regwen_qs; + logic direct_access_regwen_wd; + logic direct_access_cmd_we; + logic direct_access_cmd_rd_wd; + logic direct_access_cmd_wr_wd; + logic direct_access_cmd_digest_wd; + logic direct_access_address_we; + logic [13:0] direct_access_address_qs; + logic [13:0] direct_access_address_wd; + logic direct_access_wdata_0_we; + logic [31:0] direct_access_wdata_0_qs; + logic [31:0] direct_access_wdata_0_wd; + logic direct_access_wdata_1_we; + logic [31:0] direct_access_wdata_1_qs; + logic [31:0] direct_access_wdata_1_wd; + logic direct_access_rdata_0_re; + logic [31:0] direct_access_rdata_0_qs; + logic direct_access_rdata_1_re; + logic [31:0] direct_access_rdata_1_qs; + logic check_trigger_regwen_we; + logic check_trigger_regwen_qs; + logic check_trigger_regwen_wd; + logic check_trigger_we; + logic check_trigger_integrity_wd; + logic check_trigger_consistency_wd; + logic check_regwen_we; + logic check_regwen_qs; + logic check_regwen_wd; + logic check_timeout_we; + logic [31:0] check_timeout_qs; + logic [31:0] check_timeout_wd; + logic integrity_check_period_we; + logic [31:0] integrity_check_period_qs; + logic [31:0] integrity_check_period_wd; + logic consistency_check_period_we; + logic [31:0] consistency_check_period_qs; + logic [31:0] consistency_check_period_wd; + logic vendor_test_read_lock_we; + logic vendor_test_read_lock_qs; + logic vendor_test_read_lock_wd; + logic creator_sw_cfg_read_lock_we; + logic creator_sw_cfg_read_lock_qs; + logic creator_sw_cfg_read_lock_wd; + logic owner_sw_cfg_read_lock_we; + logic owner_sw_cfg_read_lock_qs; + logic owner_sw_cfg_read_lock_wd; + logic ownership_slot_state_read_lock_we; + logic ownership_slot_state_read_lock_qs; + logic ownership_slot_state_read_lock_wd; + logic rot_creator_auth_read_lock_we; + logic rot_creator_auth_read_lock_qs; + logic rot_creator_auth_read_lock_wd; + logic rot_owner_auth_slot0_read_lock_we; + logic rot_owner_auth_slot0_read_lock_qs; + logic rot_owner_auth_slot0_read_lock_wd; + logic rot_owner_auth_slot1_read_lock_we; + logic rot_owner_auth_slot1_read_lock_qs; + logic rot_owner_auth_slot1_read_lock_wd; + logic plat_integ_auth_slot0_read_lock_we; + logic plat_integ_auth_slot0_read_lock_qs; + logic plat_integ_auth_slot0_read_lock_wd; + logic plat_integ_auth_slot1_read_lock_we; + logic plat_integ_auth_slot1_read_lock_qs; + logic plat_integ_auth_slot1_read_lock_wd; + logic plat_owner_auth_slot0_read_lock_we; + logic plat_owner_auth_slot0_read_lock_qs; + logic plat_owner_auth_slot0_read_lock_wd; + logic plat_owner_auth_slot1_read_lock_we; + logic plat_owner_auth_slot1_read_lock_qs; + logic plat_owner_auth_slot1_read_lock_wd; + logic plat_owner_auth_slot2_read_lock_we; + logic plat_owner_auth_slot2_read_lock_qs; + logic plat_owner_auth_slot2_read_lock_wd; + logic plat_owner_auth_slot3_read_lock_we; + logic plat_owner_auth_slot3_read_lock_qs; + logic plat_owner_auth_slot3_read_lock_wd; + logic ext_nvm_read_lock_we; + logic ext_nvm_read_lock_qs; + logic ext_nvm_read_lock_wd; + logic rom_patch_read_lock_we; + logic rom_patch_read_lock_qs; + logic rom_patch_read_lock_wd; + logic vendor_test_digest_0_re; + logic [31:0] vendor_test_digest_0_qs; + logic vendor_test_digest_1_re; + logic [31:0] vendor_test_digest_1_qs; + logic creator_sw_cfg_digest_0_re; + logic [31:0] creator_sw_cfg_digest_0_qs; + logic creator_sw_cfg_digest_1_re; + logic [31:0] creator_sw_cfg_digest_1_qs; + logic owner_sw_cfg_digest_0_re; + logic [31:0] owner_sw_cfg_digest_0_qs; + logic owner_sw_cfg_digest_1_re; + logic [31:0] owner_sw_cfg_digest_1_qs; + logic rot_creator_auth_digest_0_re; + logic [31:0] rot_creator_auth_digest_0_qs; + logic rot_creator_auth_digest_1_re; + logic [31:0] rot_creator_auth_digest_1_qs; + logic rot_owner_auth_slot0_digest_0_re; + logic [31:0] rot_owner_auth_slot0_digest_0_qs; + logic rot_owner_auth_slot0_digest_1_re; + logic [31:0] rot_owner_auth_slot0_digest_1_qs; + logic rot_owner_auth_slot1_digest_0_re; + logic [31:0] rot_owner_auth_slot1_digest_0_qs; + logic rot_owner_auth_slot1_digest_1_re; + logic [31:0] rot_owner_auth_slot1_digest_1_qs; + logic plat_integ_auth_slot0_digest_0_re; + logic [31:0] plat_integ_auth_slot0_digest_0_qs; + logic plat_integ_auth_slot0_digest_1_re; + logic [31:0] plat_integ_auth_slot0_digest_1_qs; + logic plat_integ_auth_slot1_digest_0_re; + logic [31:0] plat_integ_auth_slot1_digest_0_qs; + logic plat_integ_auth_slot1_digest_1_re; + logic [31:0] plat_integ_auth_slot1_digest_1_qs; + logic plat_owner_auth_slot0_digest_0_re; + logic [31:0] plat_owner_auth_slot0_digest_0_qs; + logic plat_owner_auth_slot0_digest_1_re; + logic [31:0] plat_owner_auth_slot0_digest_1_qs; + logic plat_owner_auth_slot1_digest_0_re; + logic [31:0] plat_owner_auth_slot1_digest_0_qs; + logic plat_owner_auth_slot1_digest_1_re; + logic [31:0] plat_owner_auth_slot1_digest_1_qs; + logic plat_owner_auth_slot2_digest_0_re; + logic [31:0] plat_owner_auth_slot2_digest_0_qs; + logic plat_owner_auth_slot2_digest_1_re; + logic [31:0] plat_owner_auth_slot2_digest_1_qs; + logic plat_owner_auth_slot3_digest_0_re; + logic [31:0] plat_owner_auth_slot3_digest_0_qs; + logic plat_owner_auth_slot3_digest_1_re; + logic [31:0] plat_owner_auth_slot3_digest_1_qs; + logic rom_patch_digest_0_re; + logic [31:0] rom_patch_digest_0_qs; + logic rom_patch_digest_1_re; + logic [31:0] rom_patch_digest_1_qs; + logic hw_cfg0_digest_0_re; + logic [31:0] hw_cfg0_digest_0_qs; + logic hw_cfg0_digest_1_re; + logic [31:0] hw_cfg0_digest_1_qs; + logic hw_cfg1_digest_0_re; + logic [31:0] hw_cfg1_digest_0_qs; + logic hw_cfg1_digest_1_re; + logic [31:0] hw_cfg1_digest_1_qs; + logic secret0_digest_0_re; + logic [31:0] secret0_digest_0_qs; + logic secret0_digest_1_re; + logic [31:0] secret0_digest_1_qs; + logic secret1_digest_0_re; + logic [31:0] secret1_digest_0_qs; + logic secret1_digest_1_re; + logic [31:0] secret1_digest_1_qs; + logic secret2_digest_0_re; + logic [31:0] secret2_digest_0_qs; + logic secret2_digest_1_re; + logic [31:0] secret2_digest_1_qs; + logic secret3_digest_0_re; + logic [31:0] secret3_digest_0_qs; + logic secret3_digest_1_re; + logic [31:0] secret3_digest_1_qs; + + // Register instances + // R[intr_state]: V(False) + // F[otp_operation_done]: 0:0 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW1C), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_intr_state_otp_operation_done ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (intr_state_we), + .wd (intr_state_otp_operation_done_wd), + + // from internal hardware + .de (hw2reg.intr_state.otp_operation_done.de), + .d (hw2reg.intr_state.otp_operation_done.d), + + // to internal hardware + .qe (), + .q (reg2hw.intr_state.otp_operation_done.q), + .ds (), + + // to register interface (read) + .qs (intr_state_otp_operation_done_qs) + ); + + // F[otp_error]: 1:1 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW1C), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_intr_state_otp_error ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (intr_state_we), + .wd (intr_state_otp_error_wd), + + // from internal hardware + .de (hw2reg.intr_state.otp_error.de), + .d (hw2reg.intr_state.otp_error.d), + + // to internal hardware + .qe (), + .q (reg2hw.intr_state.otp_error.q), + .ds (), + + // to register interface (read) + .qs (intr_state_otp_error_qs) + ); + + + // R[intr_enable]: V(False) + // F[otp_operation_done]: 0:0 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_intr_enable_otp_operation_done ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (intr_enable_we), + .wd (intr_enable_otp_operation_done_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.intr_enable.otp_operation_done.q), + .ds (), + + // to register interface (read) + .qs (intr_enable_otp_operation_done_qs) + ); + + // F[otp_error]: 1:1 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_intr_enable_otp_error ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (intr_enable_we), + .wd (intr_enable_otp_error_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.intr_enable.otp_error.q), + .ds (), + + // to register interface (read) + .qs (intr_enable_otp_error_qs) + ); + + + // R[intr_test]: V(True) + logic intr_test_qe; + logic [1:0] intr_test_flds_we; + assign intr_test_qe = &intr_test_flds_we; + // F[otp_operation_done]: 0:0 + prim_subreg_ext #( + .DW (1) + ) u_intr_test_otp_operation_done ( + .re (1'b0), + .we (intr_test_we), + .wd (intr_test_otp_operation_done_wd), + .d ('0), + .qre (), + .qe (intr_test_flds_we[0]), + .q (reg2hw.intr_test.otp_operation_done.q), + .ds (), + .qs () + ); + assign reg2hw.intr_test.otp_operation_done.qe = intr_test_qe; + + // F[otp_error]: 1:1 + prim_subreg_ext #( + .DW (1) + ) u_intr_test_otp_error ( + .re (1'b0), + .we (intr_test_we), + .wd (intr_test_otp_error_wd), + .d ('0), + .qre (), + .qe (intr_test_flds_we[1]), + .q (reg2hw.intr_test.otp_error.q), + .ds (), + .qs () + ); + assign reg2hw.intr_test.otp_error.qe = intr_test_qe; + + + // R[alert_test]: V(True) + logic alert_test_qe; + logic [4:0] alert_test_flds_we; + assign alert_test_qe = &alert_test_flds_we; + // F[fatal_macro_error]: 0:0 + prim_subreg_ext #( + .DW (1) + ) u_alert_test_fatal_macro_error ( + .re (1'b0), + .we (alert_test_we), + .wd (alert_test_fatal_macro_error_wd), + .d ('0), + .qre (), + .qe (alert_test_flds_we[0]), + .q (reg2hw.alert_test.fatal_macro_error.q), + .ds (), + .qs () + ); + assign reg2hw.alert_test.fatal_macro_error.qe = alert_test_qe; + + // F[fatal_check_error]: 1:1 + prim_subreg_ext #( + .DW (1) + ) u_alert_test_fatal_check_error ( + .re (1'b0), + .we (alert_test_we), + .wd (alert_test_fatal_check_error_wd), + .d ('0), + .qre (), + .qe (alert_test_flds_we[1]), + .q (reg2hw.alert_test.fatal_check_error.q), + .ds (), + .qs () + ); + assign reg2hw.alert_test.fatal_check_error.qe = alert_test_qe; + + // F[fatal_bus_integ_error]: 2:2 + prim_subreg_ext #( + .DW (1) + ) u_alert_test_fatal_bus_integ_error ( + .re (1'b0), + .we (alert_test_we), + .wd (alert_test_fatal_bus_integ_error_wd), + .d ('0), + .qre (), + .qe (alert_test_flds_we[2]), + .q (reg2hw.alert_test.fatal_bus_integ_error.q), + .ds (), + .qs () + ); + assign reg2hw.alert_test.fatal_bus_integ_error.qe = alert_test_qe; + + // F[fatal_prim_otp_alert]: 3:3 + prim_subreg_ext #( + .DW (1) + ) u_alert_test_fatal_prim_otp_alert ( + .re (1'b0), + .we (alert_test_we), + .wd (alert_test_fatal_prim_otp_alert_wd), + .d ('0), + .qre (), + .qe (alert_test_flds_we[3]), + .q (reg2hw.alert_test.fatal_prim_otp_alert.q), + .ds (), + .qs () + ); + assign reg2hw.alert_test.fatal_prim_otp_alert.qe = alert_test_qe; + + // F[recov_prim_otp_alert]: 4:4 + prim_subreg_ext #( + .DW (1) + ) u_alert_test_recov_prim_otp_alert ( + .re (1'b0), + .we (alert_test_we), + .wd (alert_test_recov_prim_otp_alert_wd), + .d ('0), + .qre (), + .qe (alert_test_flds_we[4]), + .q (reg2hw.alert_test.recov_prim_otp_alert.q), + .ds (), + .qs () + ); + assign reg2hw.alert_test.recov_prim_otp_alert.qe = alert_test_qe; + + + // R[status]: V(True) + // F[vendor_test_error]: 0:0 + prim_subreg_ext #( + .DW (1) + ) u_status_vendor_test_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.vendor_test_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_vendor_test_error_qs) + ); + + // F[creator_sw_cfg_error]: 1:1 + prim_subreg_ext #( + .DW (1) + ) u_status_creator_sw_cfg_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.creator_sw_cfg_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_creator_sw_cfg_error_qs) + ); + + // F[owner_sw_cfg_error]: 2:2 + prim_subreg_ext #( + .DW (1) + ) u_status_owner_sw_cfg_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.owner_sw_cfg_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_owner_sw_cfg_error_qs) + ); + + // F[ownership_slot_state_error]: 3:3 + prim_subreg_ext #( + .DW (1) + ) u_status_ownership_slot_state_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.ownership_slot_state_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_ownership_slot_state_error_qs) + ); + + // F[rot_creator_auth_error]: 4:4 + prim_subreg_ext #( + .DW (1) + ) u_status_rot_creator_auth_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.rot_creator_auth_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_rot_creator_auth_error_qs) + ); + + // F[rot_owner_auth_slot0_error]: 5:5 + prim_subreg_ext #( + .DW (1) + ) u_status_rot_owner_auth_slot0_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.rot_owner_auth_slot0_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_rot_owner_auth_slot0_error_qs) + ); + + // F[rot_owner_auth_slot1_error]: 6:6 + prim_subreg_ext #( + .DW (1) + ) u_status_rot_owner_auth_slot1_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.rot_owner_auth_slot1_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_rot_owner_auth_slot1_error_qs) + ); + + // F[plat_integ_auth_slot0_error]: 7:7 + prim_subreg_ext #( + .DW (1) + ) u_status_plat_integ_auth_slot0_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.plat_integ_auth_slot0_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_plat_integ_auth_slot0_error_qs) + ); + + // F[plat_integ_auth_slot1_error]: 8:8 + prim_subreg_ext #( + .DW (1) + ) u_status_plat_integ_auth_slot1_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.plat_integ_auth_slot1_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_plat_integ_auth_slot1_error_qs) + ); + + // F[plat_owner_auth_slot0_error]: 9:9 + prim_subreg_ext #( + .DW (1) + ) u_status_plat_owner_auth_slot0_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.plat_owner_auth_slot0_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_plat_owner_auth_slot0_error_qs) + ); + + // F[plat_owner_auth_slot1_error]: 10:10 + prim_subreg_ext #( + .DW (1) + ) u_status_plat_owner_auth_slot1_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.plat_owner_auth_slot1_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_plat_owner_auth_slot1_error_qs) + ); + + // F[plat_owner_auth_slot2_error]: 11:11 + prim_subreg_ext #( + .DW (1) + ) u_status_plat_owner_auth_slot2_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.plat_owner_auth_slot2_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_plat_owner_auth_slot2_error_qs) + ); + + // F[plat_owner_auth_slot3_error]: 12:12 + prim_subreg_ext #( + .DW (1) + ) u_status_plat_owner_auth_slot3_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.plat_owner_auth_slot3_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_plat_owner_auth_slot3_error_qs) + ); + + // F[ext_nvm_error]: 13:13 + prim_subreg_ext #( + .DW (1) + ) u_status_ext_nvm_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.ext_nvm_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_ext_nvm_error_qs) + ); + + // F[rom_patch_error]: 14:14 + prim_subreg_ext #( + .DW (1) + ) u_status_rom_patch_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.rom_patch_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_rom_patch_error_qs) + ); + + // F[hw_cfg0_error]: 15:15 + prim_subreg_ext #( + .DW (1) + ) u_status_hw_cfg0_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.hw_cfg0_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_hw_cfg0_error_qs) + ); + + // F[hw_cfg1_error]: 16:16 + prim_subreg_ext #( + .DW (1) + ) u_status_hw_cfg1_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.hw_cfg1_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_hw_cfg1_error_qs) + ); + + // F[secret0_error]: 17:17 + prim_subreg_ext #( + .DW (1) + ) u_status_secret0_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.secret0_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_secret0_error_qs) + ); + + // F[secret1_error]: 18:18 + prim_subreg_ext #( + .DW (1) + ) u_status_secret1_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.secret1_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_secret1_error_qs) + ); + + // F[secret2_error]: 19:19 + prim_subreg_ext #( + .DW (1) + ) u_status_secret2_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.secret2_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_secret2_error_qs) + ); + + // F[secret3_error]: 20:20 + prim_subreg_ext #( + .DW (1) + ) u_status_secret3_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.secret3_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_secret3_error_qs) + ); + + // F[life_cycle_error]: 21:21 + prim_subreg_ext #( + .DW (1) + ) u_status_life_cycle_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.life_cycle_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_life_cycle_error_qs) + ); + + // F[dai_error]: 22:22 + prim_subreg_ext #( + .DW (1) + ) u_status_dai_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.dai_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_dai_error_qs) + ); + + // F[lci_error]: 23:23 + prim_subreg_ext #( + .DW (1) + ) u_status_lci_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.lci_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_lci_error_qs) + ); + + // F[timeout_error]: 24:24 + prim_subreg_ext #( + .DW (1) + ) u_status_timeout_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.timeout_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_timeout_error_qs) + ); + + // F[lfsr_fsm_error]: 25:25 + prim_subreg_ext #( + .DW (1) + ) u_status_lfsr_fsm_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.lfsr_fsm_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_lfsr_fsm_error_qs) + ); + + // F[scrambling_fsm_error]: 26:26 + prim_subreg_ext #( + .DW (1) + ) u_status_scrambling_fsm_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.scrambling_fsm_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_scrambling_fsm_error_qs) + ); + + // F[key_deriv_fsm_error]: 27:27 + prim_subreg_ext #( + .DW (1) + ) u_status_key_deriv_fsm_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.key_deriv_fsm_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_key_deriv_fsm_error_qs) + ); + + // F[bus_integ_error]: 28:28 + prim_subreg_ext #( + .DW (1) + ) u_status_bus_integ_error ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.bus_integ_error.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_bus_integ_error_qs) + ); + + // F[dai_idle]: 29:29 + prim_subreg_ext #( + .DW (1) + ) u_status_dai_idle ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.dai_idle.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_dai_idle_qs) + ); + + // F[check_pending]: 30:30 + prim_subreg_ext #( + .DW (1) + ) u_status_check_pending ( + .re (status_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.status.check_pending.d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (status_check_pending_qs) + ); + + + // Subregister 0 of Multireg err_code + // R[err_code_0]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_0 ( + .re (err_code_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_0_qs) + ); + + + // Subregister 1 of Multireg err_code + // R[err_code_1]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_1 ( + .re (err_code_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_1_qs) + ); + + + // Subregister 2 of Multireg err_code + // R[err_code_2]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_2 ( + .re (err_code_2_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[2].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_2_qs) + ); + + + // Subregister 3 of Multireg err_code + // R[err_code_3]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_3 ( + .re (err_code_3_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[3].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_3_qs) + ); + + + // Subregister 4 of Multireg err_code + // R[err_code_4]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_4 ( + .re (err_code_4_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[4].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_4_qs) + ); + + + // Subregister 5 of Multireg err_code + // R[err_code_5]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_5 ( + .re (err_code_5_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[5].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_5_qs) + ); + + + // Subregister 6 of Multireg err_code + // R[err_code_6]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_6 ( + .re (err_code_6_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[6].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_6_qs) + ); + + + // Subregister 7 of Multireg err_code + // R[err_code_7]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_7 ( + .re (err_code_7_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[7].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_7_qs) + ); + + + // Subregister 8 of Multireg err_code + // R[err_code_8]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_8 ( + .re (err_code_8_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[8].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_8_qs) + ); + + + // Subregister 9 of Multireg err_code + // R[err_code_9]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_9 ( + .re (err_code_9_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[9].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_9_qs) + ); + + + // Subregister 10 of Multireg err_code + // R[err_code_10]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_10 ( + .re (err_code_10_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[10].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_10_qs) + ); + + + // Subregister 11 of Multireg err_code + // R[err_code_11]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_11 ( + .re (err_code_11_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[11].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_11_qs) + ); + + + // Subregister 12 of Multireg err_code + // R[err_code_12]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_12 ( + .re (err_code_12_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[12].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_12_qs) + ); + + + // Subregister 13 of Multireg err_code + // R[err_code_13]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_13 ( + .re (err_code_13_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[13].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_13_qs) + ); + + + // Subregister 14 of Multireg err_code + // R[err_code_14]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_14 ( + .re (err_code_14_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[14].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_14_qs) + ); + + + // Subregister 15 of Multireg err_code + // R[err_code_15]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_15 ( + .re (err_code_15_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[15].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_15_qs) + ); + + + // Subregister 16 of Multireg err_code + // R[err_code_16]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_16 ( + .re (err_code_16_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[16].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_16_qs) + ); + + + // Subregister 17 of Multireg err_code + // R[err_code_17]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_17 ( + .re (err_code_17_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[17].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_17_qs) + ); + + + // Subregister 18 of Multireg err_code + // R[err_code_18]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_18 ( + .re (err_code_18_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[18].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_18_qs) + ); + + + // Subregister 19 of Multireg err_code + // R[err_code_19]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_19 ( + .re (err_code_19_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[19].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_19_qs) + ); + + + // Subregister 20 of Multireg err_code + // R[err_code_20]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_20 ( + .re (err_code_20_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[20].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_20_qs) + ); + + + // Subregister 21 of Multireg err_code + // R[err_code_21]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_21 ( + .re (err_code_21_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[21].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_21_qs) + ); + + + // Subregister 22 of Multireg err_code + // R[err_code_22]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_22 ( + .re (err_code_22_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[22].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_22_qs) + ); + + + // Subregister 23 of Multireg err_code + // R[err_code_23]: V(True) + prim_subreg_ext #( + .DW (3) + ) u_err_code_23 ( + .re (err_code_23_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.err_code[23].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (err_code_23_qs) + ); + + + // R[direct_access_regwen]: V(True) + logic direct_access_regwen_qe; + logic [0:0] direct_access_regwen_flds_we; + assign direct_access_regwen_qe = &direct_access_regwen_flds_we; + prim_subreg_ext #( + .DW (1) + ) u_direct_access_regwen ( + .re (direct_access_regwen_re), + .we (direct_access_regwen_we), + .wd (direct_access_regwen_wd), + .d (hw2reg.direct_access_regwen.d), + .qre (), + .qe (direct_access_regwen_flds_we[0]), + .q (reg2hw.direct_access_regwen.q), + .ds (), + .qs (direct_access_regwen_qs) + ); + assign reg2hw.direct_access_regwen.qe = direct_access_regwen_qe; + + + // R[direct_access_cmd]: V(True) + logic direct_access_cmd_qe; + logic [2:0] direct_access_cmd_flds_we; + assign direct_access_cmd_qe = &direct_access_cmd_flds_we; + // Create REGWEN-gated WE signal + logic direct_access_cmd_gated_we; + assign direct_access_cmd_gated_we = direct_access_cmd_we & direct_access_regwen_qs; + // F[rd]: 0:0 + prim_subreg_ext #( + .DW (1) + ) u_direct_access_cmd_rd ( + .re (1'b0), + .we (direct_access_cmd_gated_we), + .wd (direct_access_cmd_rd_wd), + .d ('0), + .qre (), + .qe (direct_access_cmd_flds_we[0]), + .q (reg2hw.direct_access_cmd.rd.q), + .ds (), + .qs () + ); + assign reg2hw.direct_access_cmd.rd.qe = direct_access_cmd_qe; + + // F[wr]: 1:1 + prim_subreg_ext #( + .DW (1) + ) u_direct_access_cmd_wr ( + .re (1'b0), + .we (direct_access_cmd_gated_we), + .wd (direct_access_cmd_wr_wd), + .d ('0), + .qre (), + .qe (direct_access_cmd_flds_we[1]), + .q (reg2hw.direct_access_cmd.wr.q), + .ds (), + .qs () + ); + assign reg2hw.direct_access_cmd.wr.qe = direct_access_cmd_qe; + + // F[digest]: 2:2 + prim_subreg_ext #( + .DW (1) + ) u_direct_access_cmd_digest ( + .re (1'b0), + .we (direct_access_cmd_gated_we), + .wd (direct_access_cmd_digest_wd), + .d ('0), + .qre (), + .qe (direct_access_cmd_flds_we[2]), + .q (reg2hw.direct_access_cmd.digest.q), + .ds (), + .qs () + ); + assign reg2hw.direct_access_cmd.digest.qe = direct_access_cmd_qe; + + + // R[direct_access_address]: V(False) + // Create REGWEN-gated WE signal + logic direct_access_address_gated_we; + assign direct_access_address_gated_we = direct_access_address_we & direct_access_regwen_qs; + prim_subreg #( + .DW (14), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (14'h0), + .Mubi (1'b0) + ) u_direct_access_address ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (direct_access_address_gated_we), + .wd (direct_access_address_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.direct_access_address.q), + .ds (), + + // to register interface (read) + .qs (direct_access_address_qs) + ); + + + // Subregister 0 of Multireg direct_access_wdata + // R[direct_access_wdata_0]: V(False) + // Create REGWEN-gated WE signal + logic direct_access_wdata_0_gated_we; + assign direct_access_wdata_0_gated_we = direct_access_wdata_0_we & direct_access_regwen_qs; + prim_subreg #( + .DW (32), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (32'h0), + .Mubi (1'b0) + ) u_direct_access_wdata_0 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (direct_access_wdata_0_gated_we), + .wd (direct_access_wdata_0_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.direct_access_wdata[0].q), + .ds (), + + // to register interface (read) + .qs (direct_access_wdata_0_qs) + ); + + + // Subregister 1 of Multireg direct_access_wdata + // R[direct_access_wdata_1]: V(False) + // Create REGWEN-gated WE signal + logic direct_access_wdata_1_gated_we; + assign direct_access_wdata_1_gated_we = direct_access_wdata_1_we & direct_access_regwen_qs; + prim_subreg #( + .DW (32), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (32'h0), + .Mubi (1'b0) + ) u_direct_access_wdata_1 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (direct_access_wdata_1_gated_we), + .wd (direct_access_wdata_1_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.direct_access_wdata[1].q), + .ds (), + + // to register interface (read) + .qs (direct_access_wdata_1_qs) + ); + + + // Subregister 0 of Multireg direct_access_rdata + // R[direct_access_rdata_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_direct_access_rdata_0 ( + .re (direct_access_rdata_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.direct_access_rdata[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (direct_access_rdata_0_qs) + ); + + + // Subregister 1 of Multireg direct_access_rdata + // R[direct_access_rdata_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_direct_access_rdata_1 ( + .re (direct_access_rdata_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.direct_access_rdata[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (direct_access_rdata_1_qs) + ); + + + // R[check_trigger_regwen]: V(False) + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_check_trigger_regwen ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (check_trigger_regwen_we), + .wd (check_trigger_regwen_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (), + .ds (), + + // to register interface (read) + .qs (check_trigger_regwen_qs) + ); + + + // R[check_trigger]: V(True) + logic check_trigger_qe; + logic [1:0] check_trigger_flds_we; + assign check_trigger_qe = &check_trigger_flds_we; + // Create REGWEN-gated WE signal + logic check_trigger_gated_we; + assign check_trigger_gated_we = check_trigger_we & check_trigger_regwen_qs; + // F[integrity]: 0:0 + prim_subreg_ext #( + .DW (1) + ) u_check_trigger_integrity ( + .re (1'b0), + .we (check_trigger_gated_we), + .wd (check_trigger_integrity_wd), + .d ('0), + .qre (), + .qe (check_trigger_flds_we[0]), + .q (reg2hw.check_trigger.integrity.q), + .ds (), + .qs () + ); + assign reg2hw.check_trigger.integrity.qe = check_trigger_qe; + + // F[consistency]: 1:1 + prim_subreg_ext #( + .DW (1) + ) u_check_trigger_consistency ( + .re (1'b0), + .we (check_trigger_gated_we), + .wd (check_trigger_consistency_wd), + .d ('0), + .qre (), + .qe (check_trigger_flds_we[1]), + .q (reg2hw.check_trigger.consistency.q), + .ds (), + .qs () + ); + assign reg2hw.check_trigger.consistency.qe = check_trigger_qe; + + + // R[check_regwen]: V(False) + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_check_regwen ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (check_regwen_we), + .wd (check_regwen_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (), + .ds (), + + // to register interface (read) + .qs (check_regwen_qs) + ); + + + // R[check_timeout]: V(False) + // Create REGWEN-gated WE signal + logic check_timeout_gated_we; + assign check_timeout_gated_we = check_timeout_we & check_regwen_qs; + prim_subreg #( + .DW (32), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (32'h0), + .Mubi (1'b0) + ) u_check_timeout ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (check_timeout_gated_we), + .wd (check_timeout_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.check_timeout.q), + .ds (), + + // to register interface (read) + .qs (check_timeout_qs) + ); + + + // R[integrity_check_period]: V(False) + // Create REGWEN-gated WE signal + logic integrity_check_period_gated_we; + assign integrity_check_period_gated_we = integrity_check_period_we & check_regwen_qs; + prim_subreg #( + .DW (32), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (32'h0), + .Mubi (1'b0) + ) u_integrity_check_period ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (integrity_check_period_gated_we), + .wd (integrity_check_period_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.integrity_check_period.q), + .ds (), + + // to register interface (read) + .qs (integrity_check_period_qs) + ); + + + // R[consistency_check_period]: V(False) + // Create REGWEN-gated WE signal + logic consistency_check_period_gated_we; + assign consistency_check_period_gated_we = consistency_check_period_we & check_regwen_qs; + prim_subreg #( + .DW (32), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (32'h0), + .Mubi (1'b0) + ) u_consistency_check_period ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (consistency_check_period_gated_we), + .wd (consistency_check_period_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.consistency_check_period.q), + .ds (), + + // to register interface (read) + .qs (consistency_check_period_qs) + ); + + + // R[vendor_test_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic vendor_test_read_lock_gated_we; + assign vendor_test_read_lock_gated_we = vendor_test_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_vendor_test_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (vendor_test_read_lock_gated_we), + .wd (vendor_test_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.vendor_test_read_lock.q), + .ds (), + + // to register interface (read) + .qs (vendor_test_read_lock_qs) + ); + + + // R[creator_sw_cfg_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic creator_sw_cfg_read_lock_gated_we; + assign creator_sw_cfg_read_lock_gated_we = creator_sw_cfg_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_creator_sw_cfg_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (creator_sw_cfg_read_lock_gated_we), + .wd (creator_sw_cfg_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.creator_sw_cfg_read_lock.q), + .ds (), + + // to register interface (read) + .qs (creator_sw_cfg_read_lock_qs) + ); + + + // R[owner_sw_cfg_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic owner_sw_cfg_read_lock_gated_we; + assign owner_sw_cfg_read_lock_gated_we = owner_sw_cfg_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_owner_sw_cfg_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (owner_sw_cfg_read_lock_gated_we), + .wd (owner_sw_cfg_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.owner_sw_cfg_read_lock.q), + .ds (), + + // to register interface (read) + .qs (owner_sw_cfg_read_lock_qs) + ); + + + // R[ownership_slot_state_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic ownership_slot_state_read_lock_gated_we; + assign ownership_slot_state_read_lock_gated_we = + ownership_slot_state_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_ownership_slot_state_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (ownership_slot_state_read_lock_gated_we), + .wd (ownership_slot_state_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.ownership_slot_state_read_lock.q), + .ds (), + + // to register interface (read) + .qs (ownership_slot_state_read_lock_qs) + ); + + + // R[rot_creator_auth_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic rot_creator_auth_read_lock_gated_we; + assign rot_creator_auth_read_lock_gated_we = + rot_creator_auth_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_rot_creator_auth_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (rot_creator_auth_read_lock_gated_we), + .wd (rot_creator_auth_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.rot_creator_auth_read_lock.q), + .ds (), + + // to register interface (read) + .qs (rot_creator_auth_read_lock_qs) + ); + + + // R[rot_owner_auth_slot0_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic rot_owner_auth_slot0_read_lock_gated_we; + assign rot_owner_auth_slot0_read_lock_gated_we = + rot_owner_auth_slot0_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_rot_owner_auth_slot0_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (rot_owner_auth_slot0_read_lock_gated_we), + .wd (rot_owner_auth_slot0_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.rot_owner_auth_slot0_read_lock.q), + .ds (), + + // to register interface (read) + .qs (rot_owner_auth_slot0_read_lock_qs) + ); + + + // R[rot_owner_auth_slot1_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic rot_owner_auth_slot1_read_lock_gated_we; + assign rot_owner_auth_slot1_read_lock_gated_we = + rot_owner_auth_slot1_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_rot_owner_auth_slot1_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (rot_owner_auth_slot1_read_lock_gated_we), + .wd (rot_owner_auth_slot1_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.rot_owner_auth_slot1_read_lock.q), + .ds (), + + // to register interface (read) + .qs (rot_owner_auth_slot1_read_lock_qs) + ); + + + // R[plat_integ_auth_slot0_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic plat_integ_auth_slot0_read_lock_gated_we; + assign plat_integ_auth_slot0_read_lock_gated_we = + plat_integ_auth_slot0_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_plat_integ_auth_slot0_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (plat_integ_auth_slot0_read_lock_gated_we), + .wd (plat_integ_auth_slot0_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.plat_integ_auth_slot0_read_lock.q), + .ds (), + + // to register interface (read) + .qs (plat_integ_auth_slot0_read_lock_qs) + ); + + + // R[plat_integ_auth_slot1_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic plat_integ_auth_slot1_read_lock_gated_we; + assign plat_integ_auth_slot1_read_lock_gated_we = + plat_integ_auth_slot1_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_plat_integ_auth_slot1_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (plat_integ_auth_slot1_read_lock_gated_we), + .wd (plat_integ_auth_slot1_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.plat_integ_auth_slot1_read_lock.q), + .ds (), + + // to register interface (read) + .qs (plat_integ_auth_slot1_read_lock_qs) + ); + + + // R[plat_owner_auth_slot0_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic plat_owner_auth_slot0_read_lock_gated_we; + assign plat_owner_auth_slot0_read_lock_gated_we = + plat_owner_auth_slot0_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_plat_owner_auth_slot0_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (plat_owner_auth_slot0_read_lock_gated_we), + .wd (plat_owner_auth_slot0_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.plat_owner_auth_slot0_read_lock.q), + .ds (), + + // to register interface (read) + .qs (plat_owner_auth_slot0_read_lock_qs) + ); + + + // R[plat_owner_auth_slot1_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic plat_owner_auth_slot1_read_lock_gated_we; + assign plat_owner_auth_slot1_read_lock_gated_we = + plat_owner_auth_slot1_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_plat_owner_auth_slot1_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (plat_owner_auth_slot1_read_lock_gated_we), + .wd (plat_owner_auth_slot1_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.plat_owner_auth_slot1_read_lock.q), + .ds (), + + // to register interface (read) + .qs (plat_owner_auth_slot1_read_lock_qs) + ); + + + // R[plat_owner_auth_slot2_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic plat_owner_auth_slot2_read_lock_gated_we; + assign plat_owner_auth_slot2_read_lock_gated_we = + plat_owner_auth_slot2_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_plat_owner_auth_slot2_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (plat_owner_auth_slot2_read_lock_gated_we), + .wd (plat_owner_auth_slot2_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.plat_owner_auth_slot2_read_lock.q), + .ds (), + + // to register interface (read) + .qs (plat_owner_auth_slot2_read_lock_qs) + ); + + + // R[plat_owner_auth_slot3_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic plat_owner_auth_slot3_read_lock_gated_we; + assign plat_owner_auth_slot3_read_lock_gated_we = + plat_owner_auth_slot3_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_plat_owner_auth_slot3_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (plat_owner_auth_slot3_read_lock_gated_we), + .wd (plat_owner_auth_slot3_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.plat_owner_auth_slot3_read_lock.q), + .ds (), + + // to register interface (read) + .qs (plat_owner_auth_slot3_read_lock_qs) + ); + + + // R[ext_nvm_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic ext_nvm_read_lock_gated_we; + assign ext_nvm_read_lock_gated_we = ext_nvm_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_ext_nvm_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (ext_nvm_read_lock_gated_we), + .wd (ext_nvm_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.ext_nvm_read_lock.q), + .ds (), + + // to register interface (read) + .qs (ext_nvm_read_lock_qs) + ); + + + // R[rom_patch_read_lock]: V(False) + // Create REGWEN-gated WE signal + logic rom_patch_read_lock_gated_we; + assign rom_patch_read_lock_gated_we = rom_patch_read_lock_we & direct_access_regwen_qs; + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW0C), + .RESVAL (1'h1), + .Mubi (1'b0) + ) u_rom_patch_read_lock ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (rom_patch_read_lock_gated_we), + .wd (rom_patch_read_lock_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.rom_patch_read_lock.q), + .ds (), + + // to register interface (read) + .qs (rom_patch_read_lock_qs) + ); + + + // Subregister 0 of Multireg vendor_test_digest + // R[vendor_test_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_vendor_test_digest_0 ( + .re (vendor_test_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.vendor_test_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (vendor_test_digest_0_qs) + ); + + + // Subregister 1 of Multireg vendor_test_digest + // R[vendor_test_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_vendor_test_digest_1 ( + .re (vendor_test_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.vendor_test_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (vendor_test_digest_1_qs) + ); + + + // Subregister 0 of Multireg creator_sw_cfg_digest + // R[creator_sw_cfg_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_creator_sw_cfg_digest_0 ( + .re (creator_sw_cfg_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.creator_sw_cfg_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (creator_sw_cfg_digest_0_qs) + ); + + + // Subregister 1 of Multireg creator_sw_cfg_digest + // R[creator_sw_cfg_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_creator_sw_cfg_digest_1 ( + .re (creator_sw_cfg_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.creator_sw_cfg_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (creator_sw_cfg_digest_1_qs) + ); + + + // Subregister 0 of Multireg owner_sw_cfg_digest + // R[owner_sw_cfg_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_owner_sw_cfg_digest_0 ( + .re (owner_sw_cfg_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.owner_sw_cfg_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (owner_sw_cfg_digest_0_qs) + ); + + + // Subregister 1 of Multireg owner_sw_cfg_digest + // R[owner_sw_cfg_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_owner_sw_cfg_digest_1 ( + .re (owner_sw_cfg_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.owner_sw_cfg_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (owner_sw_cfg_digest_1_qs) + ); + + + // Subregister 0 of Multireg rot_creator_auth_digest + // R[rot_creator_auth_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_rot_creator_auth_digest_0 ( + .re (rot_creator_auth_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.rot_creator_auth_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (rot_creator_auth_digest_0_qs) + ); + + + // Subregister 1 of Multireg rot_creator_auth_digest + // R[rot_creator_auth_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_rot_creator_auth_digest_1 ( + .re (rot_creator_auth_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.rot_creator_auth_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (rot_creator_auth_digest_1_qs) + ); + + + // Subregister 0 of Multireg rot_owner_auth_slot0_digest + // R[rot_owner_auth_slot0_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_rot_owner_auth_slot0_digest_0 ( + .re (rot_owner_auth_slot0_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.rot_owner_auth_slot0_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (rot_owner_auth_slot0_digest_0_qs) + ); + + + // Subregister 1 of Multireg rot_owner_auth_slot0_digest + // R[rot_owner_auth_slot0_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_rot_owner_auth_slot0_digest_1 ( + .re (rot_owner_auth_slot0_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.rot_owner_auth_slot0_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (rot_owner_auth_slot0_digest_1_qs) + ); + + + // Subregister 0 of Multireg rot_owner_auth_slot1_digest + // R[rot_owner_auth_slot1_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_rot_owner_auth_slot1_digest_0 ( + .re (rot_owner_auth_slot1_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.rot_owner_auth_slot1_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (rot_owner_auth_slot1_digest_0_qs) + ); + + + // Subregister 1 of Multireg rot_owner_auth_slot1_digest + // R[rot_owner_auth_slot1_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_rot_owner_auth_slot1_digest_1 ( + .re (rot_owner_auth_slot1_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.rot_owner_auth_slot1_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (rot_owner_auth_slot1_digest_1_qs) + ); + + + // Subregister 0 of Multireg plat_integ_auth_slot0_digest + // R[plat_integ_auth_slot0_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_plat_integ_auth_slot0_digest_0 ( + .re (plat_integ_auth_slot0_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.plat_integ_auth_slot0_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (plat_integ_auth_slot0_digest_0_qs) + ); + + + // Subregister 1 of Multireg plat_integ_auth_slot0_digest + // R[plat_integ_auth_slot0_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_plat_integ_auth_slot0_digest_1 ( + .re (plat_integ_auth_slot0_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.plat_integ_auth_slot0_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (plat_integ_auth_slot0_digest_1_qs) + ); + + + // Subregister 0 of Multireg plat_integ_auth_slot1_digest + // R[plat_integ_auth_slot1_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_plat_integ_auth_slot1_digest_0 ( + .re (plat_integ_auth_slot1_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.plat_integ_auth_slot1_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (plat_integ_auth_slot1_digest_0_qs) + ); + + + // Subregister 1 of Multireg plat_integ_auth_slot1_digest + // R[plat_integ_auth_slot1_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_plat_integ_auth_slot1_digest_1 ( + .re (plat_integ_auth_slot1_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.plat_integ_auth_slot1_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (plat_integ_auth_slot1_digest_1_qs) + ); + + + // Subregister 0 of Multireg plat_owner_auth_slot0_digest + // R[plat_owner_auth_slot0_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_plat_owner_auth_slot0_digest_0 ( + .re (plat_owner_auth_slot0_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.plat_owner_auth_slot0_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (plat_owner_auth_slot0_digest_0_qs) + ); + + + // Subregister 1 of Multireg plat_owner_auth_slot0_digest + // R[plat_owner_auth_slot0_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_plat_owner_auth_slot0_digest_1 ( + .re (plat_owner_auth_slot0_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.plat_owner_auth_slot0_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (plat_owner_auth_slot0_digest_1_qs) + ); + + + // Subregister 0 of Multireg plat_owner_auth_slot1_digest + // R[plat_owner_auth_slot1_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_plat_owner_auth_slot1_digest_0 ( + .re (plat_owner_auth_slot1_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.plat_owner_auth_slot1_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (plat_owner_auth_slot1_digest_0_qs) + ); + + + // Subregister 1 of Multireg plat_owner_auth_slot1_digest + // R[plat_owner_auth_slot1_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_plat_owner_auth_slot1_digest_1 ( + .re (plat_owner_auth_slot1_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.plat_owner_auth_slot1_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (plat_owner_auth_slot1_digest_1_qs) + ); + + + // Subregister 0 of Multireg plat_owner_auth_slot2_digest + // R[plat_owner_auth_slot2_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_plat_owner_auth_slot2_digest_0 ( + .re (plat_owner_auth_slot2_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.plat_owner_auth_slot2_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (plat_owner_auth_slot2_digest_0_qs) + ); + + + // Subregister 1 of Multireg plat_owner_auth_slot2_digest + // R[plat_owner_auth_slot2_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_plat_owner_auth_slot2_digest_1 ( + .re (plat_owner_auth_slot2_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.plat_owner_auth_slot2_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (plat_owner_auth_slot2_digest_1_qs) + ); + + + // Subregister 0 of Multireg plat_owner_auth_slot3_digest + // R[plat_owner_auth_slot3_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_plat_owner_auth_slot3_digest_0 ( + .re (plat_owner_auth_slot3_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.plat_owner_auth_slot3_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (plat_owner_auth_slot3_digest_0_qs) + ); + + + // Subregister 1 of Multireg plat_owner_auth_slot3_digest + // R[plat_owner_auth_slot3_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_plat_owner_auth_slot3_digest_1 ( + .re (plat_owner_auth_slot3_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.plat_owner_auth_slot3_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (plat_owner_auth_slot3_digest_1_qs) + ); + + + // Subregister 0 of Multireg rom_patch_digest + // R[rom_patch_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_rom_patch_digest_0 ( + .re (rom_patch_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.rom_patch_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (rom_patch_digest_0_qs) + ); + + + // Subregister 1 of Multireg rom_patch_digest + // R[rom_patch_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_rom_patch_digest_1 ( + .re (rom_patch_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.rom_patch_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (rom_patch_digest_1_qs) + ); + + + // Subregister 0 of Multireg hw_cfg0_digest + // R[hw_cfg0_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_hw_cfg0_digest_0 ( + .re (hw_cfg0_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.hw_cfg0_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (hw_cfg0_digest_0_qs) + ); + + + // Subregister 1 of Multireg hw_cfg0_digest + // R[hw_cfg0_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_hw_cfg0_digest_1 ( + .re (hw_cfg0_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.hw_cfg0_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (hw_cfg0_digest_1_qs) + ); + + + // Subregister 0 of Multireg hw_cfg1_digest + // R[hw_cfg1_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_hw_cfg1_digest_0 ( + .re (hw_cfg1_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.hw_cfg1_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (hw_cfg1_digest_0_qs) + ); + + + // Subregister 1 of Multireg hw_cfg1_digest + // R[hw_cfg1_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_hw_cfg1_digest_1 ( + .re (hw_cfg1_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.hw_cfg1_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (hw_cfg1_digest_1_qs) + ); + + + // Subregister 0 of Multireg secret0_digest + // R[secret0_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_secret0_digest_0 ( + .re (secret0_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.secret0_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (secret0_digest_0_qs) + ); + + + // Subregister 1 of Multireg secret0_digest + // R[secret0_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_secret0_digest_1 ( + .re (secret0_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.secret0_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (secret0_digest_1_qs) + ); + + + // Subregister 0 of Multireg secret1_digest + // R[secret1_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_secret1_digest_0 ( + .re (secret1_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.secret1_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (secret1_digest_0_qs) + ); + + + // Subregister 1 of Multireg secret1_digest + // R[secret1_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_secret1_digest_1 ( + .re (secret1_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.secret1_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (secret1_digest_1_qs) + ); + + + // Subregister 0 of Multireg secret2_digest + // R[secret2_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_secret2_digest_0 ( + .re (secret2_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.secret2_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (secret2_digest_0_qs) + ); + + + // Subregister 1 of Multireg secret2_digest + // R[secret2_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_secret2_digest_1 ( + .re (secret2_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.secret2_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (secret2_digest_1_qs) + ); + + + // Subregister 0 of Multireg secret3_digest + // R[secret3_digest_0]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_secret3_digest_0 ( + .re (secret3_digest_0_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.secret3_digest[0].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (secret3_digest_0_qs) + ); + + + // Subregister 1 of Multireg secret3_digest + // R[secret3_digest_1]: V(True) + prim_subreg_ext #( + .DW (32) + ) u_secret3_digest_1 ( + .re (secret3_digest_1_re), + .we (1'b0), + .wd ('0), + .d (hw2reg.secret3_digest[1].d), + .qre (), + .qe (), + .q (), + .ds (), + .qs (secret3_digest_1_qs) + ); + + + + logic [94:0] addr_hit; + always_comb begin + addr_hit = '0; + addr_hit[ 0] = (reg_addr == OTP_CTRL_INTR_STATE_OFFSET); + addr_hit[ 1] = (reg_addr == OTP_CTRL_INTR_ENABLE_OFFSET); + addr_hit[ 2] = (reg_addr == OTP_CTRL_INTR_TEST_OFFSET); + addr_hit[ 3] = (reg_addr == OTP_CTRL_ALERT_TEST_OFFSET); + addr_hit[ 4] = (reg_addr == OTP_CTRL_STATUS_OFFSET); + addr_hit[ 5] = (reg_addr == OTP_CTRL_ERR_CODE_0_OFFSET); + addr_hit[ 6] = (reg_addr == OTP_CTRL_ERR_CODE_1_OFFSET); + addr_hit[ 7] = (reg_addr == OTP_CTRL_ERR_CODE_2_OFFSET); + addr_hit[ 8] = (reg_addr == OTP_CTRL_ERR_CODE_3_OFFSET); + addr_hit[ 9] = (reg_addr == OTP_CTRL_ERR_CODE_4_OFFSET); + addr_hit[10] = (reg_addr == OTP_CTRL_ERR_CODE_5_OFFSET); + addr_hit[11] = (reg_addr == OTP_CTRL_ERR_CODE_6_OFFSET); + addr_hit[12] = (reg_addr == OTP_CTRL_ERR_CODE_7_OFFSET); + addr_hit[13] = (reg_addr == OTP_CTRL_ERR_CODE_8_OFFSET); + addr_hit[14] = (reg_addr == OTP_CTRL_ERR_CODE_9_OFFSET); + addr_hit[15] = (reg_addr == OTP_CTRL_ERR_CODE_10_OFFSET); + addr_hit[16] = (reg_addr == OTP_CTRL_ERR_CODE_11_OFFSET); + addr_hit[17] = (reg_addr == OTP_CTRL_ERR_CODE_12_OFFSET); + addr_hit[18] = (reg_addr == OTP_CTRL_ERR_CODE_13_OFFSET); + addr_hit[19] = (reg_addr == OTP_CTRL_ERR_CODE_14_OFFSET); + addr_hit[20] = (reg_addr == OTP_CTRL_ERR_CODE_15_OFFSET); + addr_hit[21] = (reg_addr == OTP_CTRL_ERR_CODE_16_OFFSET); + addr_hit[22] = (reg_addr == OTP_CTRL_ERR_CODE_17_OFFSET); + addr_hit[23] = (reg_addr == OTP_CTRL_ERR_CODE_18_OFFSET); + addr_hit[24] = (reg_addr == OTP_CTRL_ERR_CODE_19_OFFSET); + addr_hit[25] = (reg_addr == OTP_CTRL_ERR_CODE_20_OFFSET); + addr_hit[26] = (reg_addr == OTP_CTRL_ERR_CODE_21_OFFSET); + addr_hit[27] = (reg_addr == OTP_CTRL_ERR_CODE_22_OFFSET); + addr_hit[28] = (reg_addr == OTP_CTRL_ERR_CODE_23_OFFSET); + addr_hit[29] = (reg_addr == OTP_CTRL_DIRECT_ACCESS_REGWEN_OFFSET); + addr_hit[30] = (reg_addr == OTP_CTRL_DIRECT_ACCESS_CMD_OFFSET); + addr_hit[31] = (reg_addr == OTP_CTRL_DIRECT_ACCESS_ADDRESS_OFFSET); + addr_hit[32] = (reg_addr == OTP_CTRL_DIRECT_ACCESS_WDATA_0_OFFSET); + addr_hit[33] = (reg_addr == OTP_CTRL_DIRECT_ACCESS_WDATA_1_OFFSET); + addr_hit[34] = (reg_addr == OTP_CTRL_DIRECT_ACCESS_RDATA_0_OFFSET); + addr_hit[35] = (reg_addr == OTP_CTRL_DIRECT_ACCESS_RDATA_1_OFFSET); + addr_hit[36] = (reg_addr == OTP_CTRL_CHECK_TRIGGER_REGWEN_OFFSET); + addr_hit[37] = (reg_addr == OTP_CTRL_CHECK_TRIGGER_OFFSET); + addr_hit[38] = (reg_addr == OTP_CTRL_CHECK_REGWEN_OFFSET); + addr_hit[39] = (reg_addr == OTP_CTRL_CHECK_TIMEOUT_OFFSET); + addr_hit[40] = (reg_addr == OTP_CTRL_INTEGRITY_CHECK_PERIOD_OFFSET); + addr_hit[41] = (reg_addr == OTP_CTRL_CONSISTENCY_CHECK_PERIOD_OFFSET); + addr_hit[42] = (reg_addr == OTP_CTRL_VENDOR_TEST_READ_LOCK_OFFSET); + addr_hit[43] = (reg_addr == OTP_CTRL_CREATOR_SW_CFG_READ_LOCK_OFFSET); + addr_hit[44] = (reg_addr == OTP_CTRL_OWNER_SW_CFG_READ_LOCK_OFFSET); + addr_hit[45] = (reg_addr == OTP_CTRL_OWNERSHIP_SLOT_STATE_READ_LOCK_OFFSET); + addr_hit[46] = (reg_addr == OTP_CTRL_ROT_CREATOR_AUTH_READ_LOCK_OFFSET); + addr_hit[47] = (reg_addr == OTP_CTRL_ROT_OWNER_AUTH_SLOT0_READ_LOCK_OFFSET); + addr_hit[48] = (reg_addr == OTP_CTRL_ROT_OWNER_AUTH_SLOT1_READ_LOCK_OFFSET); + addr_hit[49] = (reg_addr == OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_READ_LOCK_OFFSET); + addr_hit[50] = (reg_addr == OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_READ_LOCK_OFFSET); + addr_hit[51] = (reg_addr == OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_READ_LOCK_OFFSET); + addr_hit[52] = (reg_addr == OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_READ_LOCK_OFFSET); + addr_hit[53] = (reg_addr == OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_READ_LOCK_OFFSET); + addr_hit[54] = (reg_addr == OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_READ_LOCK_OFFSET); + addr_hit[55] = (reg_addr == OTP_CTRL_EXT_NVM_READ_LOCK_OFFSET); + addr_hit[56] = (reg_addr == OTP_CTRL_ROM_PATCH_READ_LOCK_OFFSET); + addr_hit[57] = (reg_addr == OTP_CTRL_VENDOR_TEST_DIGEST_0_OFFSET); + addr_hit[58] = (reg_addr == OTP_CTRL_VENDOR_TEST_DIGEST_1_OFFSET); + addr_hit[59] = (reg_addr == OTP_CTRL_CREATOR_SW_CFG_DIGEST_0_OFFSET); + addr_hit[60] = (reg_addr == OTP_CTRL_CREATOR_SW_CFG_DIGEST_1_OFFSET); + addr_hit[61] = (reg_addr == OTP_CTRL_OWNER_SW_CFG_DIGEST_0_OFFSET); + addr_hit[62] = (reg_addr == OTP_CTRL_OWNER_SW_CFG_DIGEST_1_OFFSET); + addr_hit[63] = (reg_addr == OTP_CTRL_ROT_CREATOR_AUTH_DIGEST_0_OFFSET); + addr_hit[64] = (reg_addr == OTP_CTRL_ROT_CREATOR_AUTH_DIGEST_1_OFFSET); + addr_hit[65] = (reg_addr == OTP_CTRL_ROT_OWNER_AUTH_SLOT0_DIGEST_0_OFFSET); + addr_hit[66] = (reg_addr == OTP_CTRL_ROT_OWNER_AUTH_SLOT0_DIGEST_1_OFFSET); + addr_hit[67] = (reg_addr == OTP_CTRL_ROT_OWNER_AUTH_SLOT1_DIGEST_0_OFFSET); + addr_hit[68] = (reg_addr == OTP_CTRL_ROT_OWNER_AUTH_SLOT1_DIGEST_1_OFFSET); + addr_hit[69] = (reg_addr == OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_DIGEST_0_OFFSET); + addr_hit[70] = (reg_addr == OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_DIGEST_1_OFFSET); + addr_hit[71] = (reg_addr == OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_DIGEST_0_OFFSET); + addr_hit[72] = (reg_addr == OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_DIGEST_1_OFFSET); + addr_hit[73] = (reg_addr == OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_DIGEST_0_OFFSET); + addr_hit[74] = (reg_addr == OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_DIGEST_1_OFFSET); + addr_hit[75] = (reg_addr == OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_DIGEST_0_OFFSET); + addr_hit[76] = (reg_addr == OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_DIGEST_1_OFFSET); + addr_hit[77] = (reg_addr == OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_DIGEST_0_OFFSET); + addr_hit[78] = (reg_addr == OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_DIGEST_1_OFFSET); + addr_hit[79] = (reg_addr == OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_DIGEST_0_OFFSET); + addr_hit[80] = (reg_addr == OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_DIGEST_1_OFFSET); + addr_hit[81] = (reg_addr == OTP_CTRL_ROM_PATCH_DIGEST_0_OFFSET); + addr_hit[82] = (reg_addr == OTP_CTRL_ROM_PATCH_DIGEST_1_OFFSET); + addr_hit[83] = (reg_addr == OTP_CTRL_HW_CFG0_DIGEST_0_OFFSET); + addr_hit[84] = (reg_addr == OTP_CTRL_HW_CFG0_DIGEST_1_OFFSET); + addr_hit[85] = (reg_addr == OTP_CTRL_HW_CFG1_DIGEST_0_OFFSET); + addr_hit[86] = (reg_addr == OTP_CTRL_HW_CFG1_DIGEST_1_OFFSET); + addr_hit[87] = (reg_addr == OTP_CTRL_SECRET0_DIGEST_0_OFFSET); + addr_hit[88] = (reg_addr == OTP_CTRL_SECRET0_DIGEST_1_OFFSET); + addr_hit[89] = (reg_addr == OTP_CTRL_SECRET1_DIGEST_0_OFFSET); + addr_hit[90] = (reg_addr == OTP_CTRL_SECRET1_DIGEST_1_OFFSET); + addr_hit[91] = (reg_addr == OTP_CTRL_SECRET2_DIGEST_0_OFFSET); + addr_hit[92] = (reg_addr == OTP_CTRL_SECRET2_DIGEST_1_OFFSET); + addr_hit[93] = (reg_addr == OTP_CTRL_SECRET3_DIGEST_0_OFFSET); + addr_hit[94] = (reg_addr == OTP_CTRL_SECRET3_DIGEST_1_OFFSET); + end + + assign addrmiss = (reg_re || reg_we) ? ~|addr_hit : 1'b0 ; + + // Check sub-word write is permitted + always_comb begin + wr_err = (reg_we & + ((addr_hit[ 0] & (|(OTP_CTRL_CORE_PERMIT[ 0] & ~reg_be))) | + (addr_hit[ 1] & (|(OTP_CTRL_CORE_PERMIT[ 1] & ~reg_be))) | + (addr_hit[ 2] & (|(OTP_CTRL_CORE_PERMIT[ 2] & ~reg_be))) | + (addr_hit[ 3] & (|(OTP_CTRL_CORE_PERMIT[ 3] & ~reg_be))) | + (addr_hit[ 4] & (|(OTP_CTRL_CORE_PERMIT[ 4] & ~reg_be))) | + (addr_hit[ 5] & (|(OTP_CTRL_CORE_PERMIT[ 5] & ~reg_be))) | + (addr_hit[ 6] & (|(OTP_CTRL_CORE_PERMIT[ 6] & ~reg_be))) | + (addr_hit[ 7] & (|(OTP_CTRL_CORE_PERMIT[ 7] & ~reg_be))) | + (addr_hit[ 8] & (|(OTP_CTRL_CORE_PERMIT[ 8] & ~reg_be))) | + (addr_hit[ 9] & (|(OTP_CTRL_CORE_PERMIT[ 9] & ~reg_be))) | + (addr_hit[10] & (|(OTP_CTRL_CORE_PERMIT[10] & ~reg_be))) | + (addr_hit[11] & (|(OTP_CTRL_CORE_PERMIT[11] & ~reg_be))) | + (addr_hit[12] & (|(OTP_CTRL_CORE_PERMIT[12] & ~reg_be))) | + (addr_hit[13] & (|(OTP_CTRL_CORE_PERMIT[13] & ~reg_be))) | + (addr_hit[14] & (|(OTP_CTRL_CORE_PERMIT[14] & ~reg_be))) | + (addr_hit[15] & (|(OTP_CTRL_CORE_PERMIT[15] & ~reg_be))) | + (addr_hit[16] & (|(OTP_CTRL_CORE_PERMIT[16] & ~reg_be))) | + (addr_hit[17] & (|(OTP_CTRL_CORE_PERMIT[17] & ~reg_be))) | + (addr_hit[18] & (|(OTP_CTRL_CORE_PERMIT[18] & ~reg_be))) | + (addr_hit[19] & (|(OTP_CTRL_CORE_PERMIT[19] & ~reg_be))) | + (addr_hit[20] & (|(OTP_CTRL_CORE_PERMIT[20] & ~reg_be))) | + (addr_hit[21] & (|(OTP_CTRL_CORE_PERMIT[21] & ~reg_be))) | + (addr_hit[22] & (|(OTP_CTRL_CORE_PERMIT[22] & ~reg_be))) | + (addr_hit[23] & (|(OTP_CTRL_CORE_PERMIT[23] & ~reg_be))) | + (addr_hit[24] & (|(OTP_CTRL_CORE_PERMIT[24] & ~reg_be))) | + (addr_hit[25] & (|(OTP_CTRL_CORE_PERMIT[25] & ~reg_be))) | + (addr_hit[26] & (|(OTP_CTRL_CORE_PERMIT[26] & ~reg_be))) | + (addr_hit[27] & (|(OTP_CTRL_CORE_PERMIT[27] & ~reg_be))) | + (addr_hit[28] & (|(OTP_CTRL_CORE_PERMIT[28] & ~reg_be))) | + (addr_hit[29] & (|(OTP_CTRL_CORE_PERMIT[29] & ~reg_be))) | + (addr_hit[30] & (|(OTP_CTRL_CORE_PERMIT[30] & ~reg_be))) | + (addr_hit[31] & (|(OTP_CTRL_CORE_PERMIT[31] & ~reg_be))) | + (addr_hit[32] & (|(OTP_CTRL_CORE_PERMIT[32] & ~reg_be))) | + (addr_hit[33] & (|(OTP_CTRL_CORE_PERMIT[33] & ~reg_be))) | + (addr_hit[34] & (|(OTP_CTRL_CORE_PERMIT[34] & ~reg_be))) | + (addr_hit[35] & (|(OTP_CTRL_CORE_PERMIT[35] & ~reg_be))) | + (addr_hit[36] & (|(OTP_CTRL_CORE_PERMIT[36] & ~reg_be))) | + (addr_hit[37] & (|(OTP_CTRL_CORE_PERMIT[37] & ~reg_be))) | + (addr_hit[38] & (|(OTP_CTRL_CORE_PERMIT[38] & ~reg_be))) | + (addr_hit[39] & (|(OTP_CTRL_CORE_PERMIT[39] & ~reg_be))) | + (addr_hit[40] & (|(OTP_CTRL_CORE_PERMIT[40] & ~reg_be))) | + (addr_hit[41] & (|(OTP_CTRL_CORE_PERMIT[41] & ~reg_be))) | + (addr_hit[42] & (|(OTP_CTRL_CORE_PERMIT[42] & ~reg_be))) | + (addr_hit[43] & (|(OTP_CTRL_CORE_PERMIT[43] & ~reg_be))) | + (addr_hit[44] & (|(OTP_CTRL_CORE_PERMIT[44] & ~reg_be))) | + (addr_hit[45] & (|(OTP_CTRL_CORE_PERMIT[45] & ~reg_be))) | + (addr_hit[46] & (|(OTP_CTRL_CORE_PERMIT[46] & ~reg_be))) | + (addr_hit[47] & (|(OTP_CTRL_CORE_PERMIT[47] & ~reg_be))) | + (addr_hit[48] & (|(OTP_CTRL_CORE_PERMIT[48] & ~reg_be))) | + (addr_hit[49] & (|(OTP_CTRL_CORE_PERMIT[49] & ~reg_be))) | + (addr_hit[50] & (|(OTP_CTRL_CORE_PERMIT[50] & ~reg_be))) | + (addr_hit[51] & (|(OTP_CTRL_CORE_PERMIT[51] & ~reg_be))) | + (addr_hit[52] & (|(OTP_CTRL_CORE_PERMIT[52] & ~reg_be))) | + (addr_hit[53] & (|(OTP_CTRL_CORE_PERMIT[53] & ~reg_be))) | + (addr_hit[54] & (|(OTP_CTRL_CORE_PERMIT[54] & ~reg_be))) | + (addr_hit[55] & (|(OTP_CTRL_CORE_PERMIT[55] & ~reg_be))) | + (addr_hit[56] & (|(OTP_CTRL_CORE_PERMIT[56] & ~reg_be))) | + (addr_hit[57] & (|(OTP_CTRL_CORE_PERMIT[57] & ~reg_be))) | + (addr_hit[58] & (|(OTP_CTRL_CORE_PERMIT[58] & ~reg_be))) | + (addr_hit[59] & (|(OTP_CTRL_CORE_PERMIT[59] & ~reg_be))) | + (addr_hit[60] & (|(OTP_CTRL_CORE_PERMIT[60] & ~reg_be))) | + (addr_hit[61] & (|(OTP_CTRL_CORE_PERMIT[61] & ~reg_be))) | + (addr_hit[62] & (|(OTP_CTRL_CORE_PERMIT[62] & ~reg_be))) | + (addr_hit[63] & (|(OTP_CTRL_CORE_PERMIT[63] & ~reg_be))) | + (addr_hit[64] & (|(OTP_CTRL_CORE_PERMIT[64] & ~reg_be))) | + (addr_hit[65] & (|(OTP_CTRL_CORE_PERMIT[65] & ~reg_be))) | + (addr_hit[66] & (|(OTP_CTRL_CORE_PERMIT[66] & ~reg_be))) | + (addr_hit[67] & (|(OTP_CTRL_CORE_PERMIT[67] & ~reg_be))) | + (addr_hit[68] & (|(OTP_CTRL_CORE_PERMIT[68] & ~reg_be))) | + (addr_hit[69] & (|(OTP_CTRL_CORE_PERMIT[69] & ~reg_be))) | + (addr_hit[70] & (|(OTP_CTRL_CORE_PERMIT[70] & ~reg_be))) | + (addr_hit[71] & (|(OTP_CTRL_CORE_PERMIT[71] & ~reg_be))) | + (addr_hit[72] & (|(OTP_CTRL_CORE_PERMIT[72] & ~reg_be))) | + (addr_hit[73] & (|(OTP_CTRL_CORE_PERMIT[73] & ~reg_be))) | + (addr_hit[74] & (|(OTP_CTRL_CORE_PERMIT[74] & ~reg_be))) | + (addr_hit[75] & (|(OTP_CTRL_CORE_PERMIT[75] & ~reg_be))) | + (addr_hit[76] & (|(OTP_CTRL_CORE_PERMIT[76] & ~reg_be))) | + (addr_hit[77] & (|(OTP_CTRL_CORE_PERMIT[77] & ~reg_be))) | + (addr_hit[78] & (|(OTP_CTRL_CORE_PERMIT[78] & ~reg_be))) | + (addr_hit[79] & (|(OTP_CTRL_CORE_PERMIT[79] & ~reg_be))) | + (addr_hit[80] & (|(OTP_CTRL_CORE_PERMIT[80] & ~reg_be))) | + (addr_hit[81] & (|(OTP_CTRL_CORE_PERMIT[81] & ~reg_be))) | + (addr_hit[82] & (|(OTP_CTRL_CORE_PERMIT[82] & ~reg_be))) | + (addr_hit[83] & (|(OTP_CTRL_CORE_PERMIT[83] & ~reg_be))) | + (addr_hit[84] & (|(OTP_CTRL_CORE_PERMIT[84] & ~reg_be))) | + (addr_hit[85] & (|(OTP_CTRL_CORE_PERMIT[85] & ~reg_be))) | + (addr_hit[86] & (|(OTP_CTRL_CORE_PERMIT[86] & ~reg_be))) | + (addr_hit[87] & (|(OTP_CTRL_CORE_PERMIT[87] & ~reg_be))) | + (addr_hit[88] & (|(OTP_CTRL_CORE_PERMIT[88] & ~reg_be))) | + (addr_hit[89] & (|(OTP_CTRL_CORE_PERMIT[89] & ~reg_be))) | + (addr_hit[90] & (|(OTP_CTRL_CORE_PERMIT[90] & ~reg_be))) | + (addr_hit[91] & (|(OTP_CTRL_CORE_PERMIT[91] & ~reg_be))) | + (addr_hit[92] & (|(OTP_CTRL_CORE_PERMIT[92] & ~reg_be))) | + (addr_hit[93] & (|(OTP_CTRL_CORE_PERMIT[93] & ~reg_be))) | + (addr_hit[94] & (|(OTP_CTRL_CORE_PERMIT[94] & ~reg_be))))); + end + + // Generate write-enables + assign intr_state_we = addr_hit[0] & reg_we & !reg_error; + + assign intr_state_otp_operation_done_wd = reg_wdata[0]; + + assign intr_state_otp_error_wd = reg_wdata[1]; + assign intr_enable_we = addr_hit[1] & reg_we & !reg_error; + + assign intr_enable_otp_operation_done_wd = reg_wdata[0]; + + assign intr_enable_otp_error_wd = reg_wdata[1]; + assign intr_test_we = addr_hit[2] & reg_we & !reg_error; + + assign intr_test_otp_operation_done_wd = reg_wdata[0]; + + assign intr_test_otp_error_wd = reg_wdata[1]; + assign alert_test_we = addr_hit[3] & reg_we & !reg_error; + + assign alert_test_fatal_macro_error_wd = reg_wdata[0]; + + assign alert_test_fatal_check_error_wd = reg_wdata[1]; + + assign alert_test_fatal_bus_integ_error_wd = reg_wdata[2]; + + assign alert_test_fatal_prim_otp_alert_wd = reg_wdata[3]; + + assign alert_test_recov_prim_otp_alert_wd = reg_wdata[4]; + assign status_re = addr_hit[4] & reg_re & !reg_error; + assign err_code_0_re = addr_hit[5] & reg_re & !reg_error; + assign err_code_1_re = addr_hit[6] & reg_re & !reg_error; + assign err_code_2_re = addr_hit[7] & reg_re & !reg_error; + assign err_code_3_re = addr_hit[8] & reg_re & !reg_error; + assign err_code_4_re = addr_hit[9] & reg_re & !reg_error; + assign err_code_5_re = addr_hit[10] & reg_re & !reg_error; + assign err_code_6_re = addr_hit[11] & reg_re & !reg_error; + assign err_code_7_re = addr_hit[12] & reg_re & !reg_error; + assign err_code_8_re = addr_hit[13] & reg_re & !reg_error; + assign err_code_9_re = addr_hit[14] & reg_re & !reg_error; + assign err_code_10_re = addr_hit[15] & reg_re & !reg_error; + assign err_code_11_re = addr_hit[16] & reg_re & !reg_error; + assign err_code_12_re = addr_hit[17] & reg_re & !reg_error; + assign err_code_13_re = addr_hit[18] & reg_re & !reg_error; + assign err_code_14_re = addr_hit[19] & reg_re & !reg_error; + assign err_code_15_re = addr_hit[20] & reg_re & !reg_error; + assign err_code_16_re = addr_hit[21] & reg_re & !reg_error; + assign err_code_17_re = addr_hit[22] & reg_re & !reg_error; + assign err_code_18_re = addr_hit[23] & reg_re & !reg_error; + assign err_code_19_re = addr_hit[24] & reg_re & !reg_error; + assign err_code_20_re = addr_hit[25] & reg_re & !reg_error; + assign err_code_21_re = addr_hit[26] & reg_re & !reg_error; + assign err_code_22_re = addr_hit[27] & reg_re & !reg_error; + assign err_code_23_re = addr_hit[28] & reg_re & !reg_error; + assign direct_access_regwen_re = addr_hit[29] & reg_re & !reg_error; + assign direct_access_regwen_we = addr_hit[29] & reg_we & !reg_error; + + assign direct_access_regwen_wd = reg_wdata[0]; + assign direct_access_cmd_we = addr_hit[30] & reg_we & !reg_error; + + assign direct_access_cmd_rd_wd = reg_wdata[0]; + + assign direct_access_cmd_wr_wd = reg_wdata[1]; + + assign direct_access_cmd_digest_wd = reg_wdata[2]; + assign direct_access_address_we = addr_hit[31] & reg_we & !reg_error; + + assign direct_access_address_wd = reg_wdata[13:0]; + assign direct_access_wdata_0_we = addr_hit[32] & reg_we & !reg_error; + + assign direct_access_wdata_0_wd = reg_wdata[31:0]; + assign direct_access_wdata_1_we = addr_hit[33] & reg_we & !reg_error; + + assign direct_access_wdata_1_wd = reg_wdata[31:0]; + assign direct_access_rdata_0_re = addr_hit[34] & reg_re & !reg_error; + assign direct_access_rdata_1_re = addr_hit[35] & reg_re & !reg_error; + assign check_trigger_regwen_we = addr_hit[36] & reg_we & !reg_error; + + assign check_trigger_regwen_wd = reg_wdata[0]; + assign check_trigger_we = addr_hit[37] & reg_we & !reg_error; + + assign check_trigger_integrity_wd = reg_wdata[0]; + + assign check_trigger_consistency_wd = reg_wdata[1]; + assign check_regwen_we = addr_hit[38] & reg_we & !reg_error; + + assign check_regwen_wd = reg_wdata[0]; + assign check_timeout_we = addr_hit[39] & reg_we & !reg_error; + + assign check_timeout_wd = reg_wdata[31:0]; + assign integrity_check_period_we = addr_hit[40] & reg_we & !reg_error; + + assign integrity_check_period_wd = reg_wdata[31:0]; + assign consistency_check_period_we = addr_hit[41] & reg_we & !reg_error; + + assign consistency_check_period_wd = reg_wdata[31:0]; + assign vendor_test_read_lock_we = addr_hit[42] & reg_we & !reg_error; + + assign vendor_test_read_lock_wd = reg_wdata[0]; + assign creator_sw_cfg_read_lock_we = addr_hit[43] & reg_we & !reg_error; + + assign creator_sw_cfg_read_lock_wd = reg_wdata[0]; + assign owner_sw_cfg_read_lock_we = addr_hit[44] & reg_we & !reg_error; + + assign owner_sw_cfg_read_lock_wd = reg_wdata[0]; + assign ownership_slot_state_read_lock_we = addr_hit[45] & reg_we & !reg_error; + + assign ownership_slot_state_read_lock_wd = reg_wdata[0]; + assign rot_creator_auth_read_lock_we = addr_hit[46] & reg_we & !reg_error; + + assign rot_creator_auth_read_lock_wd = reg_wdata[0]; + assign rot_owner_auth_slot0_read_lock_we = addr_hit[47] & reg_we & !reg_error; + + assign rot_owner_auth_slot0_read_lock_wd = reg_wdata[0]; + assign rot_owner_auth_slot1_read_lock_we = addr_hit[48] & reg_we & !reg_error; + + assign rot_owner_auth_slot1_read_lock_wd = reg_wdata[0]; + assign plat_integ_auth_slot0_read_lock_we = addr_hit[49] & reg_we & !reg_error; + + assign plat_integ_auth_slot0_read_lock_wd = reg_wdata[0]; + assign plat_integ_auth_slot1_read_lock_we = addr_hit[50] & reg_we & !reg_error; + + assign plat_integ_auth_slot1_read_lock_wd = reg_wdata[0]; + assign plat_owner_auth_slot0_read_lock_we = addr_hit[51] & reg_we & !reg_error; + + assign plat_owner_auth_slot0_read_lock_wd = reg_wdata[0]; + assign plat_owner_auth_slot1_read_lock_we = addr_hit[52] & reg_we & !reg_error; + + assign plat_owner_auth_slot1_read_lock_wd = reg_wdata[0]; + assign plat_owner_auth_slot2_read_lock_we = addr_hit[53] & reg_we & !reg_error; + + assign plat_owner_auth_slot2_read_lock_wd = reg_wdata[0]; + assign plat_owner_auth_slot3_read_lock_we = addr_hit[54] & reg_we & !reg_error; + + assign plat_owner_auth_slot3_read_lock_wd = reg_wdata[0]; + assign ext_nvm_read_lock_we = addr_hit[55] & reg_we & !reg_error; + + assign ext_nvm_read_lock_wd = reg_wdata[0]; + assign rom_patch_read_lock_we = addr_hit[56] & reg_we & !reg_error; + + assign rom_patch_read_lock_wd = reg_wdata[0]; + assign vendor_test_digest_0_re = addr_hit[57] & reg_re & !reg_error; + assign vendor_test_digest_1_re = addr_hit[58] & reg_re & !reg_error; + assign creator_sw_cfg_digest_0_re = addr_hit[59] & reg_re & !reg_error; + assign creator_sw_cfg_digest_1_re = addr_hit[60] & reg_re & !reg_error; + assign owner_sw_cfg_digest_0_re = addr_hit[61] & reg_re & !reg_error; + assign owner_sw_cfg_digest_1_re = addr_hit[62] & reg_re & !reg_error; + assign rot_creator_auth_digest_0_re = addr_hit[63] & reg_re & !reg_error; + assign rot_creator_auth_digest_1_re = addr_hit[64] & reg_re & !reg_error; + assign rot_owner_auth_slot0_digest_0_re = addr_hit[65] & reg_re & !reg_error; + assign rot_owner_auth_slot0_digest_1_re = addr_hit[66] & reg_re & !reg_error; + assign rot_owner_auth_slot1_digest_0_re = addr_hit[67] & reg_re & !reg_error; + assign rot_owner_auth_slot1_digest_1_re = addr_hit[68] & reg_re & !reg_error; + assign plat_integ_auth_slot0_digest_0_re = addr_hit[69] & reg_re & !reg_error; + assign plat_integ_auth_slot0_digest_1_re = addr_hit[70] & reg_re & !reg_error; + assign plat_integ_auth_slot1_digest_0_re = addr_hit[71] & reg_re & !reg_error; + assign plat_integ_auth_slot1_digest_1_re = addr_hit[72] & reg_re & !reg_error; + assign plat_owner_auth_slot0_digest_0_re = addr_hit[73] & reg_re & !reg_error; + assign plat_owner_auth_slot0_digest_1_re = addr_hit[74] & reg_re & !reg_error; + assign plat_owner_auth_slot1_digest_0_re = addr_hit[75] & reg_re & !reg_error; + assign plat_owner_auth_slot1_digest_1_re = addr_hit[76] & reg_re & !reg_error; + assign plat_owner_auth_slot2_digest_0_re = addr_hit[77] & reg_re & !reg_error; + assign plat_owner_auth_slot2_digest_1_re = addr_hit[78] & reg_re & !reg_error; + assign plat_owner_auth_slot3_digest_0_re = addr_hit[79] & reg_re & !reg_error; + assign plat_owner_auth_slot3_digest_1_re = addr_hit[80] & reg_re & !reg_error; + assign rom_patch_digest_0_re = addr_hit[81] & reg_re & !reg_error; + assign rom_patch_digest_1_re = addr_hit[82] & reg_re & !reg_error; + assign hw_cfg0_digest_0_re = addr_hit[83] & reg_re & !reg_error; + assign hw_cfg0_digest_1_re = addr_hit[84] & reg_re & !reg_error; + assign hw_cfg1_digest_0_re = addr_hit[85] & reg_re & !reg_error; + assign hw_cfg1_digest_1_re = addr_hit[86] & reg_re & !reg_error; + assign secret0_digest_0_re = addr_hit[87] & reg_re & !reg_error; + assign secret0_digest_1_re = addr_hit[88] & reg_re & !reg_error; + assign secret1_digest_0_re = addr_hit[89] & reg_re & !reg_error; + assign secret1_digest_1_re = addr_hit[90] & reg_re & !reg_error; + assign secret2_digest_0_re = addr_hit[91] & reg_re & !reg_error; + assign secret2_digest_1_re = addr_hit[92] & reg_re & !reg_error; + assign secret3_digest_0_re = addr_hit[93] & reg_re & !reg_error; + assign secret3_digest_1_re = addr_hit[94] & reg_re & !reg_error; + + // Assign write-enables to checker logic vector. + always_comb begin + reg_we_check = '0; + reg_we_check[0] = intr_state_we; + reg_we_check[1] = intr_enable_we; + reg_we_check[2] = intr_test_we; + reg_we_check[3] = alert_test_we; + reg_we_check[4] = 1'b0; + reg_we_check[5] = 1'b0; + reg_we_check[6] = 1'b0; + reg_we_check[7] = 1'b0; + reg_we_check[8] = 1'b0; + reg_we_check[9] = 1'b0; + reg_we_check[10] = 1'b0; + reg_we_check[11] = 1'b0; + reg_we_check[12] = 1'b0; + reg_we_check[13] = 1'b0; + reg_we_check[14] = 1'b0; + reg_we_check[15] = 1'b0; + reg_we_check[16] = 1'b0; + reg_we_check[17] = 1'b0; + reg_we_check[18] = 1'b0; + reg_we_check[19] = 1'b0; + reg_we_check[20] = 1'b0; + reg_we_check[21] = 1'b0; + reg_we_check[22] = 1'b0; + reg_we_check[23] = 1'b0; + reg_we_check[24] = 1'b0; + reg_we_check[25] = 1'b0; + reg_we_check[26] = 1'b0; + reg_we_check[27] = 1'b0; + reg_we_check[28] = 1'b0; + reg_we_check[29] = direct_access_regwen_we; + reg_we_check[30] = direct_access_cmd_gated_we; + reg_we_check[31] = direct_access_address_gated_we; + reg_we_check[32] = direct_access_wdata_0_gated_we; + reg_we_check[33] = direct_access_wdata_1_gated_we; + reg_we_check[34] = 1'b0; + reg_we_check[35] = 1'b0; + reg_we_check[36] = check_trigger_regwen_we; + reg_we_check[37] = check_trigger_gated_we; + reg_we_check[38] = check_regwen_we; + reg_we_check[39] = check_timeout_gated_we; + reg_we_check[40] = integrity_check_period_gated_we; + reg_we_check[41] = consistency_check_period_gated_we; + reg_we_check[42] = vendor_test_read_lock_gated_we; + reg_we_check[43] = creator_sw_cfg_read_lock_gated_we; + reg_we_check[44] = owner_sw_cfg_read_lock_gated_we; + reg_we_check[45] = ownership_slot_state_read_lock_gated_we; + reg_we_check[46] = rot_creator_auth_read_lock_gated_we; + reg_we_check[47] = rot_owner_auth_slot0_read_lock_gated_we; + reg_we_check[48] = rot_owner_auth_slot1_read_lock_gated_we; + reg_we_check[49] = plat_integ_auth_slot0_read_lock_gated_we; + reg_we_check[50] = plat_integ_auth_slot1_read_lock_gated_we; + reg_we_check[51] = plat_owner_auth_slot0_read_lock_gated_we; + reg_we_check[52] = plat_owner_auth_slot1_read_lock_gated_we; + reg_we_check[53] = plat_owner_auth_slot2_read_lock_gated_we; + reg_we_check[54] = plat_owner_auth_slot3_read_lock_gated_we; + reg_we_check[55] = ext_nvm_read_lock_gated_we; + reg_we_check[56] = rom_patch_read_lock_gated_we; + reg_we_check[57] = 1'b0; + reg_we_check[58] = 1'b0; + reg_we_check[59] = 1'b0; + reg_we_check[60] = 1'b0; + reg_we_check[61] = 1'b0; + reg_we_check[62] = 1'b0; + reg_we_check[63] = 1'b0; + reg_we_check[64] = 1'b0; + reg_we_check[65] = 1'b0; + reg_we_check[66] = 1'b0; + reg_we_check[67] = 1'b0; + reg_we_check[68] = 1'b0; + reg_we_check[69] = 1'b0; + reg_we_check[70] = 1'b0; + reg_we_check[71] = 1'b0; + reg_we_check[72] = 1'b0; + reg_we_check[73] = 1'b0; + reg_we_check[74] = 1'b0; + reg_we_check[75] = 1'b0; + reg_we_check[76] = 1'b0; + reg_we_check[77] = 1'b0; + reg_we_check[78] = 1'b0; + reg_we_check[79] = 1'b0; + reg_we_check[80] = 1'b0; + reg_we_check[81] = 1'b0; + reg_we_check[82] = 1'b0; + reg_we_check[83] = 1'b0; + reg_we_check[84] = 1'b0; + reg_we_check[85] = 1'b0; + reg_we_check[86] = 1'b0; + reg_we_check[87] = 1'b0; + reg_we_check[88] = 1'b0; + reg_we_check[89] = 1'b0; + reg_we_check[90] = 1'b0; + reg_we_check[91] = 1'b0; + reg_we_check[92] = 1'b0; + reg_we_check[93] = 1'b0; + reg_we_check[94] = 1'b0; + end + + // Read data return + always_comb begin + reg_rdata_next = '0; + unique case (1'b1) + addr_hit[0]: begin + reg_rdata_next[0] = intr_state_otp_operation_done_qs; + reg_rdata_next[1] = intr_state_otp_error_qs; + end + + addr_hit[1]: begin + reg_rdata_next[0] = intr_enable_otp_operation_done_qs; + reg_rdata_next[1] = intr_enable_otp_error_qs; + end + + addr_hit[2]: begin + reg_rdata_next[0] = '0; + reg_rdata_next[1] = '0; + end + + addr_hit[3]: begin + reg_rdata_next[0] = '0; + reg_rdata_next[1] = '0; + reg_rdata_next[2] = '0; + reg_rdata_next[3] = '0; + reg_rdata_next[4] = '0; + end + + addr_hit[4]: begin + reg_rdata_next[0] = status_vendor_test_error_qs; + reg_rdata_next[1] = status_creator_sw_cfg_error_qs; + reg_rdata_next[2] = status_owner_sw_cfg_error_qs; + reg_rdata_next[3] = status_ownership_slot_state_error_qs; + reg_rdata_next[4] = status_rot_creator_auth_error_qs; + reg_rdata_next[5] = status_rot_owner_auth_slot0_error_qs; + reg_rdata_next[6] = status_rot_owner_auth_slot1_error_qs; + reg_rdata_next[7] = status_plat_integ_auth_slot0_error_qs; + reg_rdata_next[8] = status_plat_integ_auth_slot1_error_qs; + reg_rdata_next[9] = status_plat_owner_auth_slot0_error_qs; + reg_rdata_next[10] = status_plat_owner_auth_slot1_error_qs; + reg_rdata_next[11] = status_plat_owner_auth_slot2_error_qs; + reg_rdata_next[12] = status_plat_owner_auth_slot3_error_qs; + reg_rdata_next[13] = status_ext_nvm_error_qs; + reg_rdata_next[14] = status_rom_patch_error_qs; + reg_rdata_next[15] = status_hw_cfg0_error_qs; + reg_rdata_next[16] = status_hw_cfg1_error_qs; + reg_rdata_next[17] = status_secret0_error_qs; + reg_rdata_next[18] = status_secret1_error_qs; + reg_rdata_next[19] = status_secret2_error_qs; + reg_rdata_next[20] = status_secret3_error_qs; + reg_rdata_next[21] = status_life_cycle_error_qs; + reg_rdata_next[22] = status_dai_error_qs; + reg_rdata_next[23] = status_lci_error_qs; + reg_rdata_next[24] = status_timeout_error_qs; + reg_rdata_next[25] = status_lfsr_fsm_error_qs; + reg_rdata_next[26] = status_scrambling_fsm_error_qs; + reg_rdata_next[27] = status_key_deriv_fsm_error_qs; + reg_rdata_next[28] = status_bus_integ_error_qs; + reg_rdata_next[29] = status_dai_idle_qs; + reg_rdata_next[30] = status_check_pending_qs; + end + + addr_hit[5]: begin + reg_rdata_next[2:0] = err_code_0_qs; + end + + addr_hit[6]: begin + reg_rdata_next[2:0] = err_code_1_qs; + end + + addr_hit[7]: begin + reg_rdata_next[2:0] = err_code_2_qs; + end + + addr_hit[8]: begin + reg_rdata_next[2:0] = err_code_3_qs; + end + + addr_hit[9]: begin + reg_rdata_next[2:0] = err_code_4_qs; + end + + addr_hit[10]: begin + reg_rdata_next[2:0] = err_code_5_qs; + end + + addr_hit[11]: begin + reg_rdata_next[2:0] = err_code_6_qs; + end + + addr_hit[12]: begin + reg_rdata_next[2:0] = err_code_7_qs; + end + + addr_hit[13]: begin + reg_rdata_next[2:0] = err_code_8_qs; + end + + addr_hit[14]: begin + reg_rdata_next[2:0] = err_code_9_qs; + end + + addr_hit[15]: begin + reg_rdata_next[2:0] = err_code_10_qs; + end + + addr_hit[16]: begin + reg_rdata_next[2:0] = err_code_11_qs; + end + + addr_hit[17]: begin + reg_rdata_next[2:0] = err_code_12_qs; + end + + addr_hit[18]: begin + reg_rdata_next[2:0] = err_code_13_qs; + end + + addr_hit[19]: begin + reg_rdata_next[2:0] = err_code_14_qs; + end + + addr_hit[20]: begin + reg_rdata_next[2:0] = err_code_15_qs; + end + + addr_hit[21]: begin + reg_rdata_next[2:0] = err_code_16_qs; + end + + addr_hit[22]: begin + reg_rdata_next[2:0] = err_code_17_qs; + end + + addr_hit[23]: begin + reg_rdata_next[2:0] = err_code_18_qs; + end + + addr_hit[24]: begin + reg_rdata_next[2:0] = err_code_19_qs; + end + + addr_hit[25]: begin + reg_rdata_next[2:0] = err_code_20_qs; + end + + addr_hit[26]: begin + reg_rdata_next[2:0] = err_code_21_qs; + end + + addr_hit[27]: begin + reg_rdata_next[2:0] = err_code_22_qs; + end + + addr_hit[28]: begin + reg_rdata_next[2:0] = err_code_23_qs; + end + + addr_hit[29]: begin + reg_rdata_next[0] = direct_access_regwen_qs; + end + + addr_hit[30]: begin + reg_rdata_next[0] = '0; + reg_rdata_next[1] = '0; + reg_rdata_next[2] = '0; + end + + addr_hit[31]: begin + reg_rdata_next[13:0] = direct_access_address_qs; + end + + addr_hit[32]: begin + reg_rdata_next[31:0] = direct_access_wdata_0_qs; + end + + addr_hit[33]: begin + reg_rdata_next[31:0] = direct_access_wdata_1_qs; + end + + addr_hit[34]: begin + reg_rdata_next[31:0] = direct_access_rdata_0_qs; + end + + addr_hit[35]: begin + reg_rdata_next[31:0] = direct_access_rdata_1_qs; + end + + addr_hit[36]: begin + reg_rdata_next[0] = check_trigger_regwen_qs; + end + + addr_hit[37]: begin + reg_rdata_next[0] = '0; + reg_rdata_next[1] = '0; + end + + addr_hit[38]: begin + reg_rdata_next[0] = check_regwen_qs; + end + + addr_hit[39]: begin + reg_rdata_next[31:0] = check_timeout_qs; + end + + addr_hit[40]: begin + reg_rdata_next[31:0] = integrity_check_period_qs; + end + + addr_hit[41]: begin + reg_rdata_next[31:0] = consistency_check_period_qs; + end + + addr_hit[42]: begin + reg_rdata_next[0] = vendor_test_read_lock_qs; + end + + addr_hit[43]: begin + reg_rdata_next[0] = creator_sw_cfg_read_lock_qs; + end + + addr_hit[44]: begin + reg_rdata_next[0] = owner_sw_cfg_read_lock_qs; + end + + addr_hit[45]: begin + reg_rdata_next[0] = ownership_slot_state_read_lock_qs; + end + + addr_hit[46]: begin + reg_rdata_next[0] = rot_creator_auth_read_lock_qs; + end + + addr_hit[47]: begin + reg_rdata_next[0] = rot_owner_auth_slot0_read_lock_qs; + end + + addr_hit[48]: begin + reg_rdata_next[0] = rot_owner_auth_slot1_read_lock_qs; + end + + addr_hit[49]: begin + reg_rdata_next[0] = plat_integ_auth_slot0_read_lock_qs; + end + + addr_hit[50]: begin + reg_rdata_next[0] = plat_integ_auth_slot1_read_lock_qs; + end + + addr_hit[51]: begin + reg_rdata_next[0] = plat_owner_auth_slot0_read_lock_qs; + end + + addr_hit[52]: begin + reg_rdata_next[0] = plat_owner_auth_slot1_read_lock_qs; + end + + addr_hit[53]: begin + reg_rdata_next[0] = plat_owner_auth_slot2_read_lock_qs; + end + + addr_hit[54]: begin + reg_rdata_next[0] = plat_owner_auth_slot3_read_lock_qs; + end + + addr_hit[55]: begin + reg_rdata_next[0] = ext_nvm_read_lock_qs; + end + + addr_hit[56]: begin + reg_rdata_next[0] = rom_patch_read_lock_qs; + end + + addr_hit[57]: begin + reg_rdata_next[31:0] = vendor_test_digest_0_qs; + end + + addr_hit[58]: begin + reg_rdata_next[31:0] = vendor_test_digest_1_qs; + end + + addr_hit[59]: begin + reg_rdata_next[31:0] = creator_sw_cfg_digest_0_qs; + end + + addr_hit[60]: begin + reg_rdata_next[31:0] = creator_sw_cfg_digest_1_qs; + end + + addr_hit[61]: begin + reg_rdata_next[31:0] = owner_sw_cfg_digest_0_qs; + end + + addr_hit[62]: begin + reg_rdata_next[31:0] = owner_sw_cfg_digest_1_qs; + end + + addr_hit[63]: begin + reg_rdata_next[31:0] = rot_creator_auth_digest_0_qs; + end + + addr_hit[64]: begin + reg_rdata_next[31:0] = rot_creator_auth_digest_1_qs; + end + + addr_hit[65]: begin + reg_rdata_next[31:0] = rot_owner_auth_slot0_digest_0_qs; + end + + addr_hit[66]: begin + reg_rdata_next[31:0] = rot_owner_auth_slot0_digest_1_qs; + end + + addr_hit[67]: begin + reg_rdata_next[31:0] = rot_owner_auth_slot1_digest_0_qs; + end + + addr_hit[68]: begin + reg_rdata_next[31:0] = rot_owner_auth_slot1_digest_1_qs; + end + + addr_hit[69]: begin + reg_rdata_next[31:0] = plat_integ_auth_slot0_digest_0_qs; + end + + addr_hit[70]: begin + reg_rdata_next[31:0] = plat_integ_auth_slot0_digest_1_qs; + end + + addr_hit[71]: begin + reg_rdata_next[31:0] = plat_integ_auth_slot1_digest_0_qs; + end + + addr_hit[72]: begin + reg_rdata_next[31:0] = plat_integ_auth_slot1_digest_1_qs; + end + + addr_hit[73]: begin + reg_rdata_next[31:0] = plat_owner_auth_slot0_digest_0_qs; + end + + addr_hit[74]: begin + reg_rdata_next[31:0] = plat_owner_auth_slot0_digest_1_qs; + end + + addr_hit[75]: begin + reg_rdata_next[31:0] = plat_owner_auth_slot1_digest_0_qs; + end + + addr_hit[76]: begin + reg_rdata_next[31:0] = plat_owner_auth_slot1_digest_1_qs; + end + + addr_hit[77]: begin + reg_rdata_next[31:0] = plat_owner_auth_slot2_digest_0_qs; + end + + addr_hit[78]: begin + reg_rdata_next[31:0] = plat_owner_auth_slot2_digest_1_qs; + end + + addr_hit[79]: begin + reg_rdata_next[31:0] = plat_owner_auth_slot3_digest_0_qs; + end + + addr_hit[80]: begin + reg_rdata_next[31:0] = plat_owner_auth_slot3_digest_1_qs; + end + + addr_hit[81]: begin + reg_rdata_next[31:0] = rom_patch_digest_0_qs; + end + + addr_hit[82]: begin + reg_rdata_next[31:0] = rom_patch_digest_1_qs; + end + + addr_hit[83]: begin + reg_rdata_next[31:0] = hw_cfg0_digest_0_qs; + end + + addr_hit[84]: begin + reg_rdata_next[31:0] = hw_cfg0_digest_1_qs; + end + + addr_hit[85]: begin + reg_rdata_next[31:0] = hw_cfg1_digest_0_qs; + end + + addr_hit[86]: begin + reg_rdata_next[31:0] = hw_cfg1_digest_1_qs; + end + + addr_hit[87]: begin + reg_rdata_next[31:0] = secret0_digest_0_qs; + end + + addr_hit[88]: begin + reg_rdata_next[31:0] = secret0_digest_1_qs; + end + + addr_hit[89]: begin + reg_rdata_next[31:0] = secret1_digest_0_qs; + end + + addr_hit[90]: begin + reg_rdata_next[31:0] = secret1_digest_1_qs; + end + + addr_hit[91]: begin + reg_rdata_next[31:0] = secret2_digest_0_qs; + end + + addr_hit[92]: begin + reg_rdata_next[31:0] = secret2_digest_1_qs; + end + + addr_hit[93]: begin + reg_rdata_next[31:0] = secret3_digest_0_qs; + end + + addr_hit[94]: begin + reg_rdata_next[31:0] = secret3_digest_1_qs; + end + + default: begin + reg_rdata_next = '1; + end + endcase + end + + // shadow busy + logic shadow_busy; + assign shadow_busy = 1'b0; + + // register busy + assign reg_busy = shadow_busy; + + // Unused signal tieoff + + // wdata / byte enable are not always fully used + // add a blanket unused statement to handle lint waivers + logic unused_wdata; + logic unused_be; + assign unused_wdata = ^reg_wdata; + assign unused_be = ^reg_be; + + // Assertions for Register Interface + `ASSERT_PULSE(wePulse, reg_we, clk_i, !rst_ni) + `ASSERT_PULSE(rePulse, reg_re, clk_i, !rst_ni) + + `ASSERT(reAfterRv, $rose(reg_re || reg_we) |=> tl_o_pre.d_valid, clk_i, !rst_ni) + + `ASSERT(en2addrHit, (reg_we || reg_re) |-> $onehot0(addr_hit), clk_i, !rst_ni) + + // this is formulated as an assumption such that the FPV testbenches do disprove this + // property by mistake + //`ASSUME(reqParity, tl_reg_h2d.a_valid |-> tl_reg_h2d.a_user.chk_en == tlul_pkg::CheckDis) + +endmodule diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_dai.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_dai.sv new file mode 100644 index 00000000000000..95240cfa9d165b --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_dai.sv @@ -0,0 +1,859 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Direct access interface for OTP controller. +// + +`include "prim_flop_macros.sv" + +module otp_ctrl_dai + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; +( + input clk_i, + input rst_ni, + // Init reqest from power manager + input init_req_i, + output logic init_done_o, + // Init request going to partitions + output logic part_init_req_o, + input [NumPart-1:0] part_init_done_i, + // Escalation input. This moves the FSM into a terminal state and locks down + // the DAI. + input lc_ctrl_pkg::lc_tx_t escalate_en_i, + // Output error state of DAI, to be consumed by OTP error/alert logic. + // Note that most errors are not recoverable and move the DAI FSM into + // a terminal error state. + output otp_err_e error_o, + // This error signal is pulsed high if the FSM has been glitched into an invalid state. + // Although it is somewhat redundant with the error code in error_o above, it is + // meant to cover cases where we already latched an error code while the FSM is + // glitched into an invalid state (since in that case, the error code will not be + // overridden with the FSM error code so that the original error code is still + // discoverable). + output logic fsm_err_o, + // Access/lock status from partitions + // SEC_CM: ACCESS.CTRL.MUBI + input part_access_t [NumPart-1:0] part_access_i, + // CSR interface + input [OtpByteAddrWidth-1:0] dai_addr_i, + input dai_cmd_e dai_cmd_i, + input logic dai_req_i, + input [NumDaiWords-1:0][31:0] dai_wdata_i, + output logic dai_idle_o, // wired to the status CSRs + output logic dai_prog_idle_o, // wired to lfsr timer and pwrmgr + output logic dai_cmd_done_o, // this is used to raise an IRQ + output logic [NumDaiWords-1:0][31:0] dai_rdata_o, + // OTP interface + output logic otp_req_o, + output prim_otp_pkg::cmd_e otp_cmd_o, + output logic [OtpSizeWidth-1:0] otp_size_o, + output logic [OtpIfWidth-1:0] otp_wdata_o, + output logic [OtpAddrWidth-1:0] otp_addr_o, + input otp_gnt_i, + input otp_rvalid_i, + input [ScrmblBlockWidth-1:0] otp_rdata_i, + input prim_otp_pkg::err_e otp_err_i, + // Scrambling mutex request + output logic scrmbl_mtx_req_o, + input scrmbl_mtx_gnt_i, + // Scrambling datapath interface + output otp_scrmbl_cmd_e scrmbl_cmd_o, + output digest_mode_e scrmbl_mode_o, + output logic [ConstSelWidth-1:0] scrmbl_sel_o, + output logic [ScrmblBlockWidth-1:0] scrmbl_data_o, + output logic scrmbl_valid_o, + input logic scrmbl_ready_i, + input logic scrmbl_valid_i, + input logic [ScrmblBlockWidth-1:0] scrmbl_data_i +); + + //////////////////////// + // Integration Checks // + //////////////////////// + + import prim_mubi_pkg::*; + import prim_util_pkg::vbits; + + localparam int CntWidth = OtpByteAddrWidth - $clog2(ScrmblBlockWidth/8); + + // Integration checks for parameters. + `ASSERT_INIT(CheckNativeOtpWidth0_A, (ScrmblBlockWidth % OtpWidth) == 0) + `ASSERT_INIT(CheckNativeOtpWidth1_A, (32 % OtpWidth) == 0) + + ///////////////////// + // DAI Control FSM // + ///////////////////// + + // SEC_CM: DAI.FSM.SPARSE + // Encoding generated with: + // $ ./util/design/sparse-fsm-encode.py -d 5 -m 20 -n 12 \ + // -s 3011551511 --language=sv + // + // Hamming distance histogram: + // + // 0: -- + // 1: -- + // 2: -- + // 3: -- + // 4: -- + // 5: |||||||||||||||| (31.05%) + // 6: |||||||||||||||||||| (36.84%) + // 7: |||||||| (15.26%) + // 8: |||| (8.95%) + // 9: || (5.26%) + // 10: (1.58%) + // 11: (1.05%) + // 12: -- + // + // Minimum Hamming distance: 5 + // Maximum Hamming distance: 11 + // Minimum Hamming weight: 2 + // Maximum Hamming weight: 9 + // + localparam int StateWidth = 12; + typedef enum logic [StateWidth-1:0] { + ResetSt = 12'b101111010100, + InitOtpSt = 12'b110000110010, + InitPartSt = 12'b000111111001, + IdleSt = 12'b111010000011, + ErrorSt = 12'b100010001110, + ReadSt = 12'b100101100110, + ReadWaitSt = 12'b001100000000, + DescrSt = 12'b011000101111, + DescrWaitSt = 12'b110101011111, + WriteSt = 12'b110111001000, + WriteWaitSt = 12'b111001111100, + ScrSt = 12'b000000010101, + ScrWaitSt = 12'b010110110100, + DigClrSt = 12'b001111001111, + DigReadSt = 12'b001001110011, + DigReadWaitSt = 12'b101110111010, + DigSt = 12'b011111100010, + DigPadSt = 12'b011010011000, + DigFinSt = 12'b110011100101, + DigWaitSt = 12'b100000101001 + } state_e; + + typedef enum logic [1:0] { + OtpData = 2'b00, + DaiData = 2'b01, + ScrmblData = 2'b10 + } data_sel_e; + + + typedef enum logic { + PartOffset = 1'b0, + DaiOffset = 1'b1 + } addr_sel_e; + + state_e state_d, state_q; + logic [CntWidth-1:0] cnt; + logic cnt_en, cnt_clr, cnt_err; + otp_err_e error_d, error_q; + logic data_en, data_clr; + data_sel_e data_sel; + addr_sel_e base_sel_d, base_sel_q; + logic [ScrmblBlockWidth-1:0] data_q; + logic [NumPartWidth-1:0] part_idx; + logic [NumPart-1:0][OtpAddrWidth-1:0] digest_addr_lut; + logic part_sel_valid; + + // Depending on the partition configuration, the wrapper is instructed to ignore integrity + // calculations and checks. To be on the safe side, the partition filters error responses at this + // point and does not report any integrity errors if integrity is disabled. + otp_err_e otp_err; + always_comb begin + otp_err = otp_err_e'(otp_err_i); + if (!PartInfo[part_idx].integrity && + otp_err_e'(otp_err_i) inside {MacroEccCorrError, MacroEccUncorrError}) begin + otp_err = NoError; + end + end + + // Output partition error state. + assign error_o = error_q; + // Working register is connected to data outputs. + assign otp_wdata_o = data_q; + assign scrmbl_data_o = data_q; + // Only expose this working register in IdleSt. + // The FSM below makes sure to clear this register + // after digest and write ops. + assign dai_rdata_o = (state_q == IdleSt) ? data_q : '0; + + always_comb begin : p_fsm + state_d = state_q; + + // Init signals + init_done_o = 1'b1; + part_init_req_o = 1'b0; + + // DAI signals + dai_idle_o = 1'b0; + dai_prog_idle_o = 1'b1; + dai_cmd_done_o = 1'b0; + + // OTP signals + otp_req_o = 1'b0; + otp_cmd_o = prim_otp_pkg::Init; + + // Scrambling mutex + scrmbl_mtx_req_o = 1'b0; + + // Scrambling datapath + scrmbl_cmd_o = LoadShadow; + scrmbl_sel_o = CnstyDigest; + scrmbl_mode_o = StandardMode; + scrmbl_valid_o = 1'b0; + + // Counter + cnt_en = 1'b0; + cnt_clr = 1'b0; + base_sel_d = base_sel_q; + + // Temporary data register + data_en = 1'b0; + data_clr = 1'b0; + data_sel = OtpData; + + // Error Register + error_d = error_q; + fsm_err_o = 1'b0; + + unique case (state_q) + /////////////////////////////////////////////////////////////////// + // We get here after reset and wait until the power manager + // requests OTP initialization. If initialization is requested, + // an init command is written to the OTP macro, and we move on + // to the InitOtpSt waiting state. + ResetSt: begin + init_done_o = 1'b0; + dai_prog_idle_o = 1'b0; + data_clr = 1'b1; + if (init_req_i) begin + otp_req_o = 1'b1; + if (otp_gnt_i) begin + state_d = InitOtpSt; + end + end + end + /////////////////////////////////////////////////////////////////// + // We wait here unitl the OTP macro has initialized without + // error. If an error occurred during this stage, we latch that + // error and move into a terminal error state. + InitOtpSt: begin + init_done_o = 1'b0; + dai_prog_idle_o = 1'b0; + if (otp_rvalid_i) begin + if ((!(otp_err inside {NoError, MacroEccCorrError}))) begin + state_d = ErrorSt; + error_d = otp_err; + end else begin + state_d = InitPartSt; + end + end + end + /////////////////////////////////////////////////////////////////// + // Since the OTP macro is now functional, we can send out an + // initialization request to all partitions and wait until they + // all have initialized. + InitPartSt: begin + init_done_o = 1'b0; + dai_prog_idle_o = 1'b0; + part_init_req_o = 1'b1; + if (part_init_done_i == {NumPart{1'b1}}) begin + state_d = IdleSt; + end + end + /////////////////////////////////////////////////////////////////// + // Idle state where we wait for incoming commands. + // Invalid commands trigger a CmdInvErr, which is recoverable. + IdleSt: begin + dai_idle_o = 1'b1; + if (dai_req_i) begin + // This clears previous (recoverable) and reset the counter. + error_d = NoError; + cnt_clr = 1'b1; + unique case (dai_cmd_i) + DaiRead: begin + state_d = ReadSt; + // Clear the temporary data register. + data_clr = 1'b1; + base_sel_d = DaiOffset; + end + DaiWrite: begin + data_sel = DaiData; + // Fetch data block. + data_en = 1'b1; + base_sel_d = DaiOffset; + // If this partition is scrambled, directly go to write scrambling first. + if (PartInfo[part_idx].secret) begin + state_d = ScrSt; + end else begin + state_d = WriteSt; + end + end + DaiDigest: begin + state_d = DigClrSt; + scrmbl_mtx_req_o = 1'b1; + base_sel_d = PartOffset; + end + default: ; // Ignore invalid commands + endcase // dai_cmd_i + end // dai_req_i + end + /////////////////////////////////////////////////////////////////// + // Each time we request a block of data from OTP, we re-check + // whether read access has been locked for this partition. If + // that is the case, we immediately bail out. Otherwise, we + // request a block of data from OTP. + ReadSt: begin + if (part_sel_valid && (mubi8_test_false_strict(part_access_i[part_idx].read_lock) || + // HW digests always remain readable. + PartInfo[part_idx].hw_digest && otp_addr_o == + digest_addr_lut[part_idx])) begin + otp_req_o = 1'b1; + // Depending on the partition configuration, + // the wrapper is instructed to ignore integrity errors. + if (PartInfo[part_idx].integrity) begin + otp_cmd_o = prim_otp_pkg::Read; + end else begin + otp_cmd_o = prim_otp_pkg::ReadRaw; + end + if (otp_gnt_i) begin + state_d = ReadWaitSt; + end + end else begin + state_d = IdleSt; + error_d = AccessError; // Signal this error, but do not go into terminal error state. + dai_cmd_done_o = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response and write to readout register. Check + // whether descrambling is required or not. In case an OTP + // transaction fails, latch the OTP error code, and jump to + // terminal error state. + ReadWaitSt: begin + // Continuously check read access and bail out if this is not consistent. + if (part_sel_valid && (mubi8_test_false_strict(part_access_i[part_idx].read_lock) || + // HW digests always remain readable. + PartInfo[part_idx].hw_digest && otp_addr_o == + digest_addr_lut[part_idx])) begin + if (otp_rvalid_i) begin + // Check OTP return code. + if (otp_err inside {NoError, MacroEccCorrError}) begin + data_en = 1'b1; + // We do not need to descramble the digest values. + if (PartInfo[part_idx].secret && otp_addr_o != digest_addr_lut[part_idx]) begin + state_d = DescrSt; + end else begin + state_d = IdleSt; + dai_cmd_done_o = 1'b1; + end + // At this point the only error that we could have gotten are correctable ECC errors. + if (otp_err != NoError) begin + error_d = MacroEccCorrError; + end + end else begin + state_d = ErrorSt; + error_d = otp_err; + end + end + // At this point, this check MUST succeed - otherwise this means that + // there was a tampering attempt. Hence we go into a terminal error state + // when this check fails. + end else begin + state_d = ErrorSt; + error_d = FsmStateError; + end + end + /////////////////////////////////////////////////////////////////// + // Descrambling state. This first acquires the scrambling + // datapath mutex. Note that once the mutex is acquired, we have + // exclusive access to the scrambling datapath until we release + // the mutex by deasserting scrmbl_mtx_req_o. + // SEC_CM: SECRET.MEM.SCRAMBLE + DescrSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = Decrypt; + scrmbl_sel_o = PartInfo[part_idx].key_sel; + if (scrmbl_mtx_gnt_i && scrmbl_ready_i) begin + state_d = DescrWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for the descrambled data to return. Note that we release + // the mutex lock upon leaving this state. + // SEC_CM: SECRET.MEM.SCRAMBLE + DescrWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_sel_o = PartInfo[part_idx].key_sel; + data_sel = ScrmblData; + if (scrmbl_valid_i) begin + state_d = IdleSt; + data_en = 1'b1; + dai_cmd_done_o = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // First, check whether write accesses are allowed to this + // partition, and error out otherwise. Note that for buffered + // partitions, we do not allow DAI writes to the digest offset. + // Unbuffered partitions have SW managed digests, hence that + // check is not needed in that case. The LC partition is + // permanently write locked and can hence not be written via the DAI. + WriteSt: begin + dai_prog_idle_o = 1'b0; + if (part_sel_valid && mubi8_test_false_strict(part_access_i[part_idx].write_lock) && + // If this is a HW digest write to a buffered partition. + ((PartInfo[part_idx].variant == Buffered && PartInfo[part_idx].hw_digest && + base_sel_q == PartOffset && otp_addr_o == digest_addr_lut[part_idx]) || + // If this is a non HW digest write to a buffered partition. + (PartInfo[part_idx].variant == Buffered && PartInfo[part_idx].hw_digest && + base_sel_q == DaiOffset && otp_addr_o < digest_addr_lut[part_idx]) || + // If this is a write to an unbuffered partition + (PartInfo[part_idx].variant != Buffered && base_sel_q == DaiOffset))) begin + otp_req_o = 1'b1; + // Depending on the partition configuration, + // the wrapper is instructed to ignore integrity errors. + if (PartInfo[part_idx].integrity) begin + otp_cmd_o = prim_otp_pkg::Write; + end else begin + otp_cmd_o = prim_otp_pkg::WriteRaw; + end + if (otp_gnt_i) begin + state_d = WriteWaitSt; + end + end else begin + // Clear working register state. + data_clr = 1'b1; + state_d = IdleSt; + error_d = AccessError; // Signal this error, but do not go into terminal error state. + dai_cmd_done_o = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response, and then go back to idle. In case an + // OTP transaction fails, latch the OTP error code, and jump to + // terminal error state. + WriteWaitSt: begin + dai_prog_idle_o = 1'b0; + // Continuously check write access and bail out if this is not consistent. + if (part_sel_valid && mubi8_test_false_strict(part_access_i[part_idx].write_lock) && + // If this is a HW digest write to a buffered partition. + ((PartInfo[part_idx].variant == Buffered && PartInfo[part_idx].hw_digest && + base_sel_q == PartOffset && otp_addr_o == digest_addr_lut[part_idx]) || + // If this is a non HW digest write to a buffered partition. + (PartInfo[part_idx].variant == Buffered && PartInfo[part_idx].hw_digest && + base_sel_q == DaiOffset && otp_addr_o < digest_addr_lut[part_idx]) || + // If this is a write to an unbuffered partition + (PartInfo[part_idx].variant != Buffered && base_sel_q == DaiOffset))) begin + + if (otp_rvalid_i) begin + // Check OTP return code. Note that non-blank errors are recoverable. + if ((!(otp_err inside {NoError, MacroWriteBlankError}))) begin + state_d = ErrorSt; + error_d = otp_err; + end else begin + // Clear working register state. + data_clr = 1'b1; + state_d = IdleSt; + dai_cmd_done_o = 1'b1; + // Signal non-blank state, but do not go to terminal error state. + if (otp_err == MacroWriteBlankError) begin + error_d = otp_err; + end + end + end + // At this point, this check MUST succeed - otherwise this means that + // there was a tampering attempt. Hence we go into a terminal error state + // when this check fails. + end else begin + state_d = ErrorSt; + error_d = FsmStateError; + end + end + /////////////////////////////////////////////////////////////////// + // Scrambling state. This first acquires the scrambling + // datapath mutex. Note that once the mutex is acquired, we have + // exclusive access to the scrambling datapath until we release + // the mutex by deasserting scrmbl_mtx_req_o. + // SEC_CM: SECRET.MEM.SCRAMBLE + ScrSt: begin + scrmbl_mtx_req_o = 1'b1; + // Check write access and bail out if this is not consistent. + if (part_sel_valid && mubi8_test_false_strict(part_access_i[part_idx].write_lock) && + // If this is a non HW digest write to a buffered partition. + (PartInfo[part_idx].variant == Buffered && PartInfo[part_idx].secret && + PartInfo[part_idx].hw_digest && base_sel_q == DaiOffset && + otp_addr_o < digest_addr_lut[part_idx])) begin + + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = Encrypt; + scrmbl_sel_o = PartInfo[part_idx].key_sel; + if (scrmbl_mtx_gnt_i && scrmbl_ready_i) begin + state_d = ScrWaitSt; + end + end else begin + state_d = IdleSt; + error_d = AccessError; // Signal this error, but do not go into terminal error state. + dai_cmd_done_o = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for the scrambled data to return. Note that we release + // the mutex lock upon leaving this state. + // SEC_CM: SECRET.MEM.SCRAMBLE + ScrWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + // Continously check write access and bail out if this is not consistent. + if (part_sel_valid && mubi8_test_false_strict(part_access_i[part_idx].write_lock) && + // If this is a non HW digest write to a buffered partition. + (PartInfo[part_idx].variant == Buffered && PartInfo[part_idx].secret && + PartInfo[part_idx].hw_digest && base_sel_q == DaiOffset && + otp_addr_o < digest_addr_lut[part_idx])) begin + data_sel = ScrmblData; + if (scrmbl_valid_i) begin + state_d = WriteSt; + data_en = 1'b1; + end + // At this point, this check MUST succeed - otherwise this means that + // there was a tampering attempt. Hence we go into a terminal error state + // when this check fails. + end else begin + state_d = ErrorSt; + error_d = FsmStateError; + end + end + /////////////////////////////////////////////////////////////////// + // First, acquire the mutex for the digest and clear the digest state. + // SEC_CM: PART.MEM.DIGEST + DigClrSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + // Need to reset the digest state and set digest mode to "standard". + scrmbl_cmd_o = DigestInit; + if (scrmbl_mtx_gnt_i && scrmbl_ready_i) begin + state_d = DigReadSt; + end + end + /////////////////////////////////////////////////////////////////// + // This requests a 64bit block to be pushed into the digest datapath. + // We also check here whether the partition has been write locked. + // SEC_CM: PART.MEM.DIGEST + DigReadSt: begin + scrmbl_mtx_req_o = 1'b1; + if (part_sel_valid && + mubi8_test_false_strict(part_access_i[part_idx].read_lock) && + mubi8_test_false_strict(part_access_i[part_idx].write_lock)) begin + otp_req_o = 1'b1; + // Depending on the partition configuration, + // the wrapper is instructed to ignore integrity errors. + if (PartInfo[part_idx].integrity) begin + otp_cmd_o = prim_otp_pkg::Read; + end else begin + otp_cmd_o = prim_otp_pkg::ReadRaw; + end + if (otp_gnt_i) begin + state_d = DigReadWaitSt; + end + end else begin + state_d = IdleSt; + error_d = AccessError; // Signal this error, but do not go into terminal error state. + dai_cmd_done_o = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response and write to readout register. Check + // whether descrambling is required or not. In case an OTP + // transaction fails, latch the OTP error code, and jump to + // terminal error state. + // SEC_CM: PART.MEM.DIGEST + DigReadWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + if (otp_rvalid_i) begin + cnt_en = 1'b1; + // Check OTP return code. + if ((!(otp_err inside {NoError, MacroEccCorrError}))) begin + state_d = ErrorSt; + error_d = otp_err; + end else begin + data_en = 1'b1; + state_d = DigSt; + // Signal soft ECC errors, but do not go into terminal error state. + if (otp_err == MacroEccCorrError) begin + error_d = otp_err; + end + end + end + end + /////////////////////////////////////////////////////////////////// + // Push the word read into the scrambling datapath. The last + // block is repeated in case the number blocks in this partition + // is odd. + // SEC_CM: PART.MEM.DIGEST + DigSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + // No need to digest the digest value itself + if (otp_addr_o == digest_addr_lut[part_idx]) begin + // Trigger digest round in case this is the second block in a row. + if (!cnt[0]) begin + scrmbl_cmd_o = Digest; + if (scrmbl_ready_i) begin + state_d = DigFinSt; + end + // Otherwise, just load low word and go to padding state. + end else if (scrmbl_ready_i) begin + state_d = DigPadSt; + end + end else begin + // Trigger digest round in case this is the second block in a row. + if (!cnt[0]) begin + scrmbl_cmd_o = Digest; + end + // Go back and fetch more data blocks. + if (scrmbl_ready_i) begin + state_d = DigReadSt; + end + end + end + /////////////////////////////////////////////////////////////////// + // Padding state, just repeat the last block and go to digest + // finalization. + // SEC_CM: PART.MEM.DIGEST + DigPadSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = Digest; + if (scrmbl_ready_i) begin + state_d = DigFinSt; + end + end + /////////////////////////////////////////////////////////////////// + // Trigger digest finalization and go wait for the result. + // SEC_CM: PART.MEM.DIGEST + DigFinSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = DigestFinalize; + if (scrmbl_ready_i) begin + state_d = DigWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for the digest to return, and write the result to OTP. + // Note that the write address will be correct in this state, + // since the counter has been stepped to the correct address as + // part of the readout sequence, and the correct size for this + // access has been loaded before. + // SEC_CM: PART.MEM.DIGEST + DigWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + data_sel = ScrmblData; + if (scrmbl_valid_i) begin + state_d = WriteSt; + data_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Terminal Error State. This locks access to the DAI. Make sure + // an FsmStateError error code is assigned here, in case no error code has + // been assigned yet. + ErrorSt: begin + if (error_q == NoError) begin + error_d = FsmStateError; + end + end + /////////////////////////////////////////////////////////////////// + // We should never get here. If we do (e.g. via a malicious + // glitch), error out immediately. + default: begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + endcase // state_q + + // Unconditionally jump into the terminal error state in case of escalation. + // SEC_CM: DAI.FSM.LOCAL_ESC, DAI.FSM.GLOBAL_ESC + if (lc_ctrl_pkg::lc_tx_test_true_loose(escalate_en_i) || cnt_err) begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + if (state_q != ErrorSt) begin + error_d = FsmStateError; + end + end + end + + //////////////////////////// + // Partition Select Logic // + //////////////////////////// + + // This checks which partition the address belongs to by comparing + // the incoming address to the partition address ranges. The onehot + // bitvector generated by the parallel comparisons is fed into a + // binary tree that determines the partition index with O(log(N)) delay. + + logic [NumPart-1:0] part_sel_oh; + for (genvar k = 0; k < NumPart; k++) begin : gen_part_sel + localparam int unsigned PartEndInt = 32'(PartInfo[k].offset) + 32'(PartInfo[k].size); + localparam int unsigned DigestOffsetInt = PartEndInt - ScrmblBlockWidth / 8; + localparam int unsigned DigestAddrLutInt = DigestOffsetInt >> OtpAddrShift; + + // PartEnd has an extra bit to cope with the case where offset + size overflows. However, we + // arrange the address map to make sure that PartEndInt is at most 1 << OtpByteAddrWidth. Check + // that here. + `ASSERT_INIT(PartEndMax_A, PartEndInt <= (1 << OtpByteAddrWidth)) + + // The shift right by OtpAddrShift drops exactly the bottom bits that are needed to convert + // between OtpAddrWidth and OtpByteAddrWidth, so we know that we can slice safely here. + localparam bit [OtpAddrWidth-1:0] DigestAddrLut = DigestAddrLutInt[OtpAddrWidth-1:0]; + + if (PartInfo[k].offset == 0) begin : gen_zero_offset + assign part_sel_oh[k] = ({1'b0, dai_addr_i} < PartEndInt[OtpByteAddrWidth:0]); + + end else begin : gen_nonzero_offset + assign part_sel_oh[k] = (dai_addr_i >= PartInfo[k].offset) & + ({1'b0, dai_addr_i} < PartEndInt[OtpByteAddrWidth:0]); + end + assign digest_addr_lut[k] = DigestAddrLut; + end + + `ASSERT(ScrmblBlockWidthGe8_A, ScrmblBlockWidth >= 8) + `ASSERT(PartSelMustBeOnehot_A, $onehot0(part_sel_oh)) + + prim_arbiter_fixed #( + .N(NumPart), + .EnDataPort(0) + ) u_part_sel_idx ( + .clk_i, + .rst_ni, + .req_i ( part_sel_oh ), + .data_i ( '{default: '0} ), + .gnt_o ( ), // unused + .idx_o ( part_idx ), + .valid_o ( part_sel_valid ), // used for detecting OOB addresses + .data_o ( ), // unused + .ready_i ( 1'b0 ) + ); + + ///////////////////////////////////// + // Address Calculations for Digest // + ///////////////////////////////////// + + // Depending on whether this is a 32bit or 64bit partition, we cut off the lower address bits. + // Access sizes are either 64bit or 32bit, depending on what region the access goes to. + logic [OtpByteAddrWidth-1:0] addr_base; + always_comb begin : p_size_sel + otp_size_o = OtpSizeWidth'(unsigned'(32 / OtpWidth - 1)); + addr_base = {dai_addr_i[OtpByteAddrWidth-1:2], 2'h0}; + + // 64bit transaction for scrambled partitions. + if (PartInfo[part_idx].secret) begin + otp_size_o = OtpSizeWidth'(unsigned'(ScrmblBlockWidth / OtpWidth - 1)); + addr_base = {dai_addr_i[OtpByteAddrWidth-1:3], 3'h0}; + // 64bit transaction if computing a digest. + end else if (PartInfo[part_idx].hw_digest && (base_sel_q == PartOffset)) begin + otp_size_o = OtpSizeWidth'(unsigned'(ScrmblBlockWidth / OtpWidth - 1)); + addr_base = PartInfo[part_idx].offset; + // 64bit transaction if the DAI address points to the partition's digest offset. + end else if ((PartInfo[part_idx].hw_digest || PartInfo[part_idx].sw_digest) && + (base_sel_q == DaiOffset) && + ({dai_addr_i[OtpByteAddrWidth-1:3], 2'b0} == digest_addr_lut[part_idx])) begin + otp_size_o = OtpSizeWidth'(unsigned'(ScrmblBlockWidth / OtpWidth - 1)); + addr_base = {dai_addr_i[OtpByteAddrWidth-1:3], 3'h0}; + end + end + + // Address counter - this is only used for computing a digest, hence the increment is + // fixed to 8 byte. + // SEC_CM: DAI.CTR.REDUN + prim_count #( + .Width(CntWidth) + ) u_prim_count ( + .clk_i, + .rst_ni, + .clr_i(cnt_clr), + .set_i(1'b0), + .set_cnt_i('0), + .incr_en_i(cnt_en), + .decr_en_i(1'b0), + .step_i(CntWidth'(1)), + .commit_i(1'b1), + .cnt_o(cnt), + .cnt_after_commit_o(), + .err_o(cnt_err) + ); + + // Note that OTP works on halfword (16bit) addresses, hence need to + // shift the addresses appropriately. + logic [OtpByteAddrWidth-1:0] addr_calc; + assign addr_calc = {cnt, {$clog2(ScrmblBlockWidth/8){1'b0}}} + addr_base; + assign otp_addr_o = OtpAddrWidth'(addr_calc >> OtpAddrShift); + + /////////////// + // Registers // + /////////////// + + `PRIM_FLOP_SPARSE_FSM(u_state_regs, state_d, state_q, state_e, ResetSt) + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + error_q <= NoError; + data_q <= '0; + base_sel_q <= DaiOffset; + end else begin + error_q <= error_d; + base_sel_q <= base_sel_d; + + // Working register + if (data_clr) begin + data_q <= '0; + end else if (data_en) begin + if (data_sel == ScrmblData) begin + data_q <= scrmbl_data_i; + end else if (data_sel == DaiData) begin + data_q <= dai_wdata_i; + end else begin + data_q <= otp_rdata_i; + end + end + end + end + + //////////////// + // Assertions // + //////////////// + + // Known assertions + `ASSERT_KNOWN(InitDoneKnown_A, init_done_o) + `ASSERT_KNOWN(PartInitReqKnown_A, part_init_req_o) + `ASSERT_KNOWN(ErrorKnown_A, error_o) + `ASSERT_KNOWN(DaiIdleKnown_A, dai_idle_o) + `ASSERT_KNOWN(DaiRdataKnown_A, dai_rdata_o) + `ASSERT_KNOWN(OtpReqKnown_A, otp_req_o) + `ASSERT_KNOWN(OtpCmdKnown_A, otp_cmd_o) + `ASSERT_KNOWN(OtpSizeKnown_A, otp_size_o) + `ASSERT_KNOWN(OtpWdataKnown_A, otp_wdata_o) + `ASSERT_KNOWN(OtpAddrKnown_A, otp_addr_o) + `ASSERT_KNOWN(ScrmblMtxReqKnown_A, scrmbl_mtx_req_o) + `ASSERT_KNOWN(ScrmblCmdKnown_A, scrmbl_cmd_o) + `ASSERT_KNOWN(ScrmblModeKnown_A, scrmbl_mode_o) + `ASSERT_KNOWN(ScrmblSelKnown_A, scrmbl_sel_o) + `ASSERT_KNOWN(ScrmblDataKnown_A, scrmbl_data_o) + `ASSERT_KNOWN(ScrmblValidKnown_A, scrmbl_valid_o) + + // OTP error response + `ASSERT(OtpErrorState_A, + state_q inside {InitOtpSt, ReadWaitSt, WriteWaitSt, DigReadWaitSt} && otp_rvalid_i && + !(otp_err inside {NoError, MacroEccCorrError, MacroWriteBlankError}) + |=> + state_q == ErrorSt && error_o == $past(otp_err)) + +endmodule : otp_ctrl_dai diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_ecc_reg.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_ecc_reg.sv new file mode 100644 index 00000000000000..2199a7cc6024d0 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_ecc_reg.sv @@ -0,0 +1,105 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Register file for buffered OTP partitions. ECC is used to detect up +// to two simultaneous errors within each 64bit word. + +`include "prim_assert.sv" + +module otp_ctrl_ecc_reg #( + parameter int Width = 64, // bit + parameter int Depth = 128, + localparam int Aw = prim_util_pkg::vbits(Depth) // derived parameter +) ( + input logic clk_i, + input logic rst_ni, + + input logic wren_i, + input logic [Aw-1:0] addr_i, + input logic [Width-1:0] wdata_i, + output logic [Width-1:0] rdata_o, + + // Concurrent output of the register state. + output logic [Depth-1:0][Width-1:0] data_o, + // Concurrent ECC check error is flagged via this signal. + output logic ecc_err_o +); + + // Integration checks for parameters. + `ASSERT_INIT(WidthMustBe64bit_A, Width == 64) + + localparam int EccWidth = 8; + + logic [Depth-1:0][Width-1:0] data_d, data_q; + logic [Depth-1:0][EccWidth-1:0] ecc_d, ecc_q; + logic [Width+EccWidth-1:0] ecc_enc; + + // Only one encoder is needed. + prim_secded_inv_72_64_enc u_prim_secded_inv_72_64_enc ( + .data_i(wdata_i), + .data_o(ecc_enc) + ); + + if (Depth == 1) begin : gen_one_word_only + always_comb begin : p_write + data_o = data_q; + data_d = data_q; + ecc_d = ecc_q; + + rdata_o = '0; + if (32'(addr_i) < Depth) begin + rdata_o = data_q[0]; + if (wren_i) begin + {ecc_d[0], data_d[0]} = ecc_enc; + end + end + end + end else begin : gen_multiple_words + always_comb begin : p_write + data_o = data_q; + data_d = data_q; + ecc_d = ecc_q; + + rdata_o = '0; + if (32'(addr_i) < Depth) begin + rdata_o = data_q[addr_i]; + if (wren_i) begin + {ecc_d[addr_i], data_d[addr_i]} = ecc_enc; + end + end + end + end + + // Concurrent ECC checks. + logic [Depth-1:0][1:0] err; + for (genvar k = 0; k < Depth; k++) begin : gen_ecc_dec + prim_secded_inv_72_64_dec u_prim_secded_inv_72_64_dec ( + .data_i({ecc_q[k], data_q[k]}), + // We only rely on the error detection mechanism, + // and not on error correction. + .data_o(), + .syndrome_o(), + .err_o(err[k]) + ); + end + + assign ecc_err_o = |err; + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + ecc_q <= {Depth{prim_secded_pkg::SecdedInv7264ZeroEcc}}; + data_q <= '0; + end else begin + ecc_q <= ecc_d; + data_q <= data_d; + end + end + + `ASSERT_KNOWN(EccKnown_A, ecc_q) + `ASSERT_KNOWN(DataKnown_A, data_q) + `ASSERT_KNOWN(RDataOutKnown_A, rdata_o) + `ASSERT_KNOWN(DataOutKnown_A, data_o) + `ASSERT_KNOWN(EccErrKnown_A, ecc_err_o) + +endmodule : otp_ctrl_ecc_reg diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_kdi.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_kdi.sv new file mode 100644 index 00000000000000..3b5d6b454f950f --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_kdi.sv @@ -0,0 +1,604 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Scrambling key derivation module for OTP. +// + +`include "prim_flop_macros.sv" + +module otp_ctrl_kdi + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; +#( + parameter scrmbl_key_init_t RndCnstScrmblKeyInit = RndCnstScrmblKeyInitDefault +) ( + input clk_i, + input rst_ni, + // Pulse to enable this module after OTP partitions have + // been initialized. + input kdi_en_i, + // Escalation input. This moves the FSM into a terminal state. + input lc_ctrl_pkg::lc_tx_t escalate_en_i, + // FSM is in error state + output logic fsm_err_o, + // Key seed inputs from OTP + input logic scrmbl_key_seed_valid_i, + input logic [FlashKeySeedWidth-1:0] flash_data_key_seed_i, + input logic [FlashKeySeedWidth-1:0] flash_addr_key_seed_i, + input logic [SramKeySeedWidth-1:0] sram_data_key_seed_i, + // EDN interface for requesting entropy + output logic edn_req_o, + input edn_ack_i, + input [EdnDataWidth-1:0] edn_data_i, + // Scrambling key requests + input flash_otp_key_req_t flash_otp_key_i, + output flash_otp_key_rsp_t flash_otp_key_o, + input sram_otp_key_req_t [NumSramKeyReqSlots-1:0] sram_otp_key_i, + output sram_otp_key_rsp_t [NumSramKeyReqSlots-1:0] sram_otp_key_o, + input otbn_otp_key_req_t otbn_otp_key_i, + output otbn_otp_key_rsp_t otbn_otp_key_o, + // Scrambling mutex request + output logic scrmbl_mtx_req_o, + input scrmbl_mtx_gnt_i, + // Scrambling datapath interface + output otp_scrmbl_cmd_e scrmbl_cmd_o, + output digest_mode_e scrmbl_mode_o, + output logic [ConstSelWidth-1:0] scrmbl_sel_o, + output logic [ScrmblBlockWidth-1:0] scrmbl_data_o, + output logic scrmbl_valid_o, + input logic scrmbl_ready_i, + input logic scrmbl_valid_i, + input logic [ScrmblBlockWidth-1:0] scrmbl_data_i +); + + import prim_util_pkg::vbits; + + //////////////////////// + // Integration Checks // + //////////////////////// + + // 2xFlash, OTBN + SRAM slots + localparam int NumReq = 3 + NumSramKeyReqSlots; + // Make sure key sizes in the system are multiples of 64bit and not larger than 256bit. + `ASSERT_INIT(KeyNonceSize0_A, (FlashKeySeedWidth <= 256) && ((FlashKeySeedWidth % 64) == 0)) + `ASSERT_INIT(KeyNonceSize1_A, (SramKeySeedWidth <= 256) && ((SramKeySeedWidth % 64) == 0)) + `ASSERT_INIT(KeyNonceSize2_A, (FlashKeyWidth <= 256) && ((FlashKeyWidth % 64) == 0)) + `ASSERT_INIT(KeyNonceSize3_A, (SramKeyWidth <= 256) && ((SramKeyWidth % 64) == 0)) + `ASSERT_INIT(KeyNonceSize4_A, (SramNonceWidth <= 256) && ((SramNonceWidth % 64) == 0)) + `ASSERT_INIT(KeyNonceSize5_A, (OtbnKeyWidth <= 256) && ((OtbnKeyWidth % 64) == 0)) + `ASSERT_INIT(KeyNonceSize6_A, (OtbnNonceWidth <= 256) && ((OtbnNonceWidth % 64) == 0)) + + // Make sure EDN interface has compatible width. + `ASSERT_INIT(EntropyWidthDividesDigestBlockWidth_A, (ScrmblKeyWidth % EdnDataWidth) == 0) + + // Currently the assumption is that the SRAM nonce is the widest. + `ASSERT_INIT(NonceWidth_A, NumNonceChunks * ScrmblBlockWidth == SramNonceWidth) + + /////////////////////////////////// + // Input Mapping and Arbitration // + /////////////////////////////////// + + // The key derivation and token hashing functions are aligned such that 2 x 128bit key + // seeds / token blocks are processed in two subsequent steps using the digest primitive. + // This effectively compresses these blocks down into 2 x 64bit blocks, thereby creating + // one 128bit key or token output. + // + // The same FSM is shared among the different flavors of key derivation and token + // hashing functions, and the following configuration options are available: + // + // 1) ingest an additional 128bit entropy block after ingesting a 128bit key seed. + // 2) keep digest state after producing the first 64bit block instead of reverting to the IV. + // 3) netlist constant index. + // 4) fetch additional entropy for the nonce output. + // 5) whether or not the key seed is valid. if not, it will be defaulted to '0. + // 6) 256bit key seed / token input. + // + // The configuration options are set further below, depending on the request type. + + typedef struct packed { + logic ingest_entropy; // 1) + logic chained_digest; // 2) + digest_sel_e digest_sel; // 3) + logic fetch_nonce; // 4) + logic [1:0] nonce_size; // 4) + logic seed_valid; // 5) + logic [3:0][ScrmblBlockWidth-1:0] seed; // 6) + } req_bundle_t; + + logic [NumReq-1:0] req, gnt; + req_bundle_t req_bundles [NumReq]; + + assign req[0] = flash_otp_key_i.data_req; + assign req[1] = flash_otp_key_i.addr_req; + assign req[2] = otbn_otp_key_i.req; + + assign flash_otp_key_o.data_ack = gnt[0]; + assign flash_otp_key_o.addr_ack = gnt[1]; + assign otbn_otp_key_o.ack = gnt[2]; + + // anchored seeds + logic [FlashKeySeedWidth-1:0] flash_data_key_seed; + logic [FlashKeySeedWidth-1:0] flash_addr_key_seed; + logic [SramKeySeedWidth-1:0] sram_data_key_seed; + + prim_sec_anchor_buf #( + .Width(FlashKeySeedWidth) + ) u_flash_data_key_anchor ( + .in_i(flash_data_key_seed_i), + .out_o(flash_data_key_seed) + ); + + prim_sec_anchor_buf #( + .Width(FlashKeySeedWidth) + ) u_flash_addr_key_anchor ( + .in_i(flash_addr_key_seed_i), + .out_o(flash_addr_key_seed) + ); + + prim_sec_anchor_buf #( + .Width(SramKeySeedWidth) + ) u_sram_data_key_anchor ( + .in_i(sram_data_key_seed_i), + .out_o(sram_data_key_seed) + ); + + // Flash data key + assign req_bundles[0] = '{ingest_entropy: 1'b0, // no random entropy added + chained_digest: 1'b0, // revert to netlist IV between blocks + digest_sel: FlashDataKey, + fetch_nonce: 1'b1, + nonce_size: 2'(FlashKeyWidth/EdnDataWidth-1), + seed_valid: scrmbl_key_seed_valid_i, + seed: flash_data_key_seed}; // 2x128bit + // Flash addr key + assign req_bundles[1] = '{ingest_entropy: 1'b0, // no random entropy added + chained_digest: 1'b0, // revert to netlist IV between blocks + digest_sel: FlashAddrKey, + fetch_nonce: 1'b1, + nonce_size: '0, + seed_valid: scrmbl_key_seed_valid_i, + seed: flash_addr_key_seed}; // 2x128bit + // OTBN key + assign req_bundles[2] = '{ingest_entropy: 1'b1, // ingest random data + chained_digest: 1'b0, // revert to netlist IV between blocks + digest_sel: SramDataKey, + fetch_nonce: 1'b1, // fetch nonce + nonce_size: 2'(OtbnNonceWidth/EdnDataWidth-1), + seed_valid: scrmbl_key_seed_valid_i, + seed: {sram_data_key_seed, // reuse same seed + sram_data_key_seed}}; + + // SRAM keys + for (genvar k = 3; k < NumReq; k++) begin : gen_req_assign + assign req[k] = sram_otp_key_i[k-3].req; + assign sram_otp_key_o[k-3].ack = gnt[k]; + assign req_bundles[k] = '{ingest_entropy: 1'b1, // ingest random data + chained_digest: 1'b0, // revert to netlist IV between blocks + digest_sel: SramDataKey, + fetch_nonce: 1'b1, // fetch nonce + nonce_size: 2'(SramNonceWidth/EdnDataWidth-1), + seed_valid: scrmbl_key_seed_valid_i, + seed: {sram_data_key_seed, // reuse same seed + sram_data_key_seed}}; + end + + // This arbitrates among incoming key derivation requests on a + // round robin basis to prevent deadlock. + logic req_valid, req_ready; + req_bundle_t req_bundle; + + prim_arbiter_tree #( + .N(NumReq), + .DW($bits(req_bundle_t))) + u_req_arb ( + .clk_i, + .rst_ni, + .req_chk_i ( 1'b1 ), + .req_i ( req ), + .data_i ( req_bundles ), + .gnt_o ( gnt ), + .idx_o ( ), + .valid_o ( req_valid ), + .data_o ( req_bundle ), + .ready_i ( req_ready ) + ); + + ////////////////////////////// + // Temporary Regs and Muxes // + ////////////////////////////// + + localparam int CntWidth = 2; + logic seed_cnt_clr, seed_cnt_en, entropy_cnt_clr, entropy_cnt_en, seed_cnt_err, entropy_cnt_err; + logic [CntWidth-1:0] seed_cnt, entropy_cnt; + + // SEC_CM: KDI_SEED.CTR.REDUN + prim_count #( + .Width(CntWidth) + ) u_prim_count_seed ( + .clk_i, + .rst_ni, + .clr_i(seed_cnt_clr), + .set_i(1'b0), + .set_cnt_i('0), + .incr_en_i(seed_cnt_en), + .decr_en_i(1'b0), + .step_i(CntWidth'(1)), + .commit_i(1'b1), + .cnt_o(seed_cnt), + .cnt_after_commit_o(), + .err_o(seed_cnt_err) + ); + + // SEC_CM: KDI_ENTROPY.CTR.REDUN + prim_count #( + .Width(CntWidth) + ) u_prim_count_entropy ( + .clk_i, + .rst_ni, + .clr_i(entropy_cnt_clr), + .set_i(1'b0), + .set_cnt_i('0), + .incr_en_i(entropy_cnt_en), + .decr_en_i(1'b0), + .step_i(CntWidth'(1)), + .commit_i(1'b1), + .cnt_o(entropy_cnt), + .cnt_after_commit_o(), + .err_o(entropy_cnt_err) + ); + + logic seed_valid_reg_en; + logic key_reg_en, nonce_reg_en; + logic seed_valid_d, seed_valid_q; + logic [ScrmblKeyWidth/ScrmblBlockWidth-1:0][ScrmblBlockWidth-1:0] key_out_d, key_out_q; + logic [NumNonceChunks-1:0][ScrmblBlockWidth-1:0] nonce_out_d, nonce_out_q; + + always_comb begin : p_outregs + key_out_d = key_out_q; + nonce_out_d = nonce_out_q; + seed_valid_d = seed_valid_q; + if (key_reg_en) begin + key_out_d[seed_cnt[1]] = scrmbl_data_i; + end + if (nonce_reg_en) begin + nonce_out_d[entropy_cnt[$clog2(NumNonceChunks)-1:0]] = edn_data_i; + end + if (seed_valid_reg_en) begin + seed_valid_d = req_bundle.seed_valid; + end + end + + // Connect keys/nonce outputs to output regs. + prim_sec_anchor_flop #( + .Width(ScrmblKeyWidth), + .ResetValue(RndCnstScrmblKeyInit.key) + ) u_key_out_anchor ( + .clk_i, + .rst_ni, + .d_i(key_out_d), + .q_o(key_out_q) + ); + + assign otbn_otp_key_o.key = key_out_q; + assign otbn_otp_key_o.nonce = nonce_out_q[OtbnNonceSel-1:0]; + assign otbn_otp_key_o.seed_valid = seed_valid_q; + + assign flash_otp_key_o.key = key_out_q; + assign flash_otp_key_o.rand_key = nonce_out_q[FlashNonceSel-1:0]; + assign flash_otp_key_o.seed_valid = seed_valid_q; + + for (genvar k = 0; k < NumSramKeyReqSlots; k++) begin : gen_out_assign + assign sram_otp_key_o[k].key = key_out_q; + assign sram_otp_key_o[k].nonce = nonce_out_q[SramNonceSel-1:0]; + assign sram_otp_key_o[k].seed_valid = seed_valid_q; + end + + typedef enum logic { + SeedData, + EntropyData + } data_sel_e; + + // Select correct 64bit block. + data_sel_e data_sel; + assign scrmbl_data_o = (data_sel == EntropyData) ? nonce_out_q[entropy_cnt[0]] : + // Gate seed value to '0 if invalid. + (req_bundle.seed_valid) ? req_bundle.seed[seed_cnt] : '0; + + ///////////////// + // Control FSM // + ///////////////// + + // SEC_CM: KDI.FSM.SPARSE + // Encoding generated with: + // $ ./util/design/sparse-fsm-encode.py -d 5 -m 11 -n 10 \ + // -s 2544133835 --language=sv + // + // Hamming distance histogram: + // + // 0: -- + // 1: -- + // 2: -- + // 3: -- + // 4: -- + // 5: |||||||||||||||||||| (54.55%) + // 6: |||||||||||||||| (45.45%) + // 7: -- + // 8: -- + // 9: -- + // 10: -- + // + // Minimum Hamming distance: 5 + // Maximum Hamming distance: 6 + // Minimum Hamming weight: 3 + // Maximum Hamming weight: 9 + // + localparam int StateWidth = 10; + typedef enum logic [StateWidth-1:0] { + ResetSt = 10'b0101100001, + IdleSt = 10'b0001011011, + DigClrSt = 10'b1101010110, + DigLoadSt = 10'b0010110111, + FetchEntropySt = 10'b1000001101, + DigEntropySt = 10'b0100111100, + DigFinSt = 10'b1000100010, + DigWaitSt = 10'b1110010001, + FetchNonceSt = 10'b0011000100, + FinishSt = 10'b1011111000, + ErrorSt = 10'b1111101111 + } state_e; + + state_e state_d, state_q; + logic edn_req_d, edn_req_q; + assign edn_req_o = edn_req_q; + + always_comb begin : p_fsm + state_d = state_q; + + // FSM Error output + fsm_err_o = 1'b0; + + // Counters + seed_cnt_en = 1'b0; + seed_cnt_clr = 1'b0; + entropy_cnt_en = 1'b0; + entropy_cnt_clr = 1'b0; + + // EDN 128bit block fetch request. + // This keeps the request alive until it has + // been acked to adhere to the req/ack protocol + // even in cases where the FSM jumps into + // an error state while waiting for a request. + edn_req_d = edn_req_q & ~edn_ack_i; + + // Data selection and temp registers + data_sel = SeedData; + key_reg_en = 1'b0; + nonce_reg_en = 1'b0; + seed_valid_reg_en = 1'b0; + + // Scrambling datapath + scrmbl_mtx_req_o = 1'b0; + scrmbl_sel_o = req_bundle.digest_sel; + scrmbl_cmd_o = LoadShadow; + scrmbl_mode_o = StandardMode; + + scrmbl_valid_o = 1'b0; + + // Request acknowledgement + req_ready = 1'b0; + + unique case (state_q) + /////////////////////////////////////////////////////////////////// + // State right after reset. Wait here until KDI gets enabled. + ResetSt: begin + if (kdi_en_i) begin + state_d = IdleSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for a request, then go and acquire the mutex. + IdleSt: begin + if (req_valid) begin + state_d = DigClrSt; + seed_cnt_clr = 1'b1; + entropy_cnt_clr = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // First, acquire the mutex for the digest and clear the digest state. + DigClrSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + // Need to reset the digest state and set digest mode to "standard". + scrmbl_cmd_o = DigestInit; + if (scrmbl_mtx_gnt_i && scrmbl_ready_i) begin + state_d = DigLoadSt; + end + end + /////////////////////////////////////////////////////////////////// + // Load two 64bit blocks of the seed, and trigger digest calculation. + DigLoadSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + // Trigger digest round in case this is the second block in a row. + if (seed_cnt[0]) begin + scrmbl_cmd_o = Digest; + if (scrmbl_ready_i) begin + // Go and ingest a block of entropy if required. + if (req_bundle.ingest_entropy) begin + state_d = FetchEntropySt; + // Otherwise go to digest finalization state. + end else begin + state_d = DigFinSt; + end + end + // Just load first 64bit block and stay here. + end else if (scrmbl_ready_i) begin + seed_cnt_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Fetch random data to ingest for key derivation. + FetchEntropySt: begin + scrmbl_mtx_req_o = 1'b1; + edn_req_d = 1'b1; + if (edn_ack_i) begin + nonce_reg_en = 1'b1; + // Finished, go and acknowledge this request. + if (entropy_cnt == 2'h1) begin + state_d = DigEntropySt; + entropy_cnt_clr = 1'b1; + // Keep on requesting entropy. + end else begin + entropy_cnt_en = 1'b1; + end + end + end + /////////////////////////////////////////////////////////////////// + // Load two 64bit blocks of entropy data. + DigEntropySt: begin + scrmbl_mtx_req_o = 1'b1; + data_sel = EntropyData; + scrmbl_valid_o = 1'b1; + // Trigger digest round in case this is the second block in a row, + // and go to digest finalization. + if (entropy_cnt[0]) begin + scrmbl_cmd_o = Digest; + if (scrmbl_ready_i) begin + state_d = DigFinSt; + entropy_cnt_clr = 1'b1; + end + // Just load first 64bit block and stay here. + end else if (scrmbl_ready_i) begin + entropy_cnt_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Trigger digest finalization and go wait for the result. + DigFinSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = DigestFinalize; + if (scrmbl_ready_i) begin + state_d = DigWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for the digest to return, and write the result to the key + // output register. Go back and process the second part of the + // input seed if needed. + DigWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + if (scrmbl_valid_i) begin + key_reg_en = 1'b1; + // Not finished yet, need to go back and produce second 64bit block. + if (seed_cnt == 2'h1) begin + seed_cnt_en = 1'b1; + // In this case the previous digest state is kept, + // which leads to a chained digest. + if (req_bundle.chained_digest) begin + state_d = DigLoadSt; + // In this case we revert the digest state to the netlist IV. + end else begin + state_d = DigClrSt; + end + // This was the second 64bit output block. + end else begin + seed_cnt_clr = 1'b1; + // Make sure we output the status of the key seed in OTP. + seed_valid_reg_en = 1'b1; + // Check whether we need to fetch additional nonce data. + if (req_bundle.fetch_nonce) begin + state_d = FetchNonceSt; + end else begin + // Finished, go and acknowledge this request. + state_d = FinishSt; + end + end + end + end + /////////////////////////////////////////////////////////////////// + // Fetch additional nonce data. Note that the mutex is released in + // this state. + FetchNonceSt: begin + edn_req_d = 1'b1; + if (edn_ack_i) begin + nonce_reg_en = 1'b1; + // Finished, go and acknowledge this request. + if (entropy_cnt == req_bundle.nonce_size) begin + state_d = FinishSt; + entropy_cnt_clr = 1'b1; + // Keep on requesting entropy. + end else begin + entropy_cnt_en = 1'b1; + end + end + end + /////////////////////////////////////////////////////////////////// + // Acknowledge request and go back to IdleSt. + FinishSt: begin + state_d = IdleSt; + req_ready = 1'b1; + end + /////////////////////////////////////////////////////////////////// + // Terminal error state. This raises an alert. + ErrorSt: begin + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + // This should never happen, hence we directly jump into the + // error state, where an alert will be triggered. + default: begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + endcase // state_q + + // Unconditionally jump into the terminal error state in case of escalation. + // SEC_CM: KDI.FSM.LOCAL_ESC, KDI.FSM.GLOBAL_ESC + if (lc_ctrl_pkg::lc_tx_test_true_loose(escalate_en_i) || + seed_cnt_err || entropy_cnt_err) begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + end + + /////////////// + // Registers // + /////////////// + + `PRIM_FLOP_SPARSE_FSM(u_state_regs, state_d, state_q, state_e, ResetSt) + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + nonce_out_q <= RndCnstScrmblKeyInit.nonce; + seed_valid_q <= 1'b0; + edn_req_q <= 1'b0; + end else begin + nonce_out_q <= nonce_out_d; + seed_valid_q <= seed_valid_d; + edn_req_q <= edn_req_d; + end + end + + //////////////// + // Assertions // + //////////////// + + `ASSERT_KNOWN(FsmErrKnown_A, fsm_err_o) + `ASSERT_KNOWN(EdnReqKnown_A, edn_req_o) + `ASSERT_KNOWN(FlashOtpKeyRspKnown_A, flash_otp_key_o) + `ASSERT_KNOWN(SramOtpKeyRspKnown_A, sram_otp_key_o) + `ASSERT_KNOWN(OtbnOtpKeyRspKnown_A, otbn_otp_key_o) + `ASSERT_KNOWN(ScrmblMtxReqKnown_A, scrmbl_mtx_req_o) + `ASSERT_KNOWN(ScrmblCmdKnown_A, scrmbl_cmd_o) + `ASSERT_KNOWN(ScrmblModeKnown_A, scrmbl_mode_o) + `ASSERT_KNOWN(ScrmblSelKnown_A, scrmbl_sel_o) + `ASSERT_KNOWN(ScrmblDataKnown_A, scrmbl_data_o) + `ASSERT_KNOWN(ScrmblValidKnown_A, scrmbl_valid_o) + +endmodule : otp_ctrl_kdi diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_lci.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_lci.sv new file mode 100644 index 00000000000000..b2f98d41be9d06 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_lci.sv @@ -0,0 +1,299 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Life cycle interface for performing life cycle transitions in OTP. +// + +`include "prim_flop_macros.sv" + +module otp_ctrl_lci + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; +#( + // Lifecycle partition information + parameter part_info_t Info = PartInfoDefault +) ( + input clk_i, + input rst_ni, + input lci_en_i, + // Escalation input. This moves the FSM into a terminal state and locks down + // the partition. + input lc_ctrl_pkg::lc_tx_t escalate_en_i, + // Life cycle transition request. In order to perform a state transition, + // the LC controller signals the new count and state. The OTP wrapper then + // only programs bits that have not been programmed before. + // Note that a transition request will fail if the request attempts to + // clear already programmed bits within OTP. + input lc_req_i, + input logic [Info.size*8-1:0] lc_data_i, + output logic lc_ack_o, + output logic lc_err_o, + // Output error state of partition, to be consumed by OTP error/alert logic. + // Note that most errors are not recoverable and move the partition FSM into + // a terminal error state. + output otp_err_e error_o, + // This error signal is pulsed high if the FSM has been glitched into an invalid state. + // Although it is somewhat redundant with the error code in error_o above, it is + // meant to cover cases where we already latched an error code while the FSM is + // glitched into an invalid state (since in that case, the error code will not be + // overridden with the FSM error code so that the original error code is still + // discoverable). + output logic fsm_err_o, + output logic lci_prog_idle_o, + // OTP interface + output logic otp_req_o, + output prim_otp_pkg::cmd_e otp_cmd_o, + output logic [OtpSizeWidth-1:0] otp_size_o, + output logic [OtpIfWidth-1:0] otp_wdata_o, + output logic [OtpAddrWidth-1:0] otp_addr_o, + input otp_gnt_i, + input otp_rvalid_i, + input [ScrmblBlockWidth-1:0] otp_rdata_i, + input prim_otp_pkg::err_e otp_err_i +); + + //////////////////////// + // Integration Checks // + //////////////////////// + + import prim_util_pkg::vbits; + + localparam int NumLcOtpWords = int'(Info.size) >> OtpAddrShift; + localparam int CntWidth = vbits(NumLcOtpWords); + + localparam int unsigned LastLcOtpWordInt = NumLcOtpWords - 1; + localparam bit [CntWidth-1:0] LastLcOtpWord = LastLcOtpWordInt[CntWidth-1:0]; + + // This is required, since each native OTP word can only be programmed once. + `ASSERT_INIT(LcValueMustBeWiderThanNativeOtpWidth_A, lc_ctrl_state_pkg::LcValueWidth >= OtpWidth) + + //////////////////// + // Controller FSM // + //////////////////// + + // SEC_CM: LCI.FSM.SPARSE + // Encoding generated with: + // $ ./util/design/sparse-fsm-encode.py -d 5 -m 5 -n 9 \ + // -s 558234734 --language=sv + // + // Hamming distance histogram: + // + // 0: -- + // 1: -- + // 2: -- + // 3: -- + // 4: -- + // 5: |||||||||||||||||||| (60.00%) + // 6: ||||||||||||| (40.00%) + // 7: -- + // 8: -- + // 9: -- + // + // Minimum Hamming distance: 5 + // Maximum Hamming distance: 6 + // Minimum Hamming weight: 1 + // Maximum Hamming weight: 7 + // + localparam int StateWidth = 9; + typedef enum logic [StateWidth-1:0] { + ResetSt = 9'b000101011, + IdleSt = 9'b110011110, + WriteSt = 9'b101010001, + WriteWaitSt = 9'b010000000, + ErrorSt = 9'b011111101 + } state_e; + + state_e state_d, state_q; + logic cnt_clr, cnt_en, cnt_err; + logic [CntWidth-1:0] cnt; + otp_err_e error_d, error_q; + + // Output LCI errors + assign error_o = error_q; + + always_comb begin : p_fsm + state_d = state_q; + + // Counter + cnt_en = 1'b0; + cnt_clr = 1'b0; + + // Idle status + lci_prog_idle_o = 1'b1; + + // OTP signals + otp_req_o = 1'b0; + otp_cmd_o = prim_otp_pkg::Read; + + // Response to LC controller + lc_err_o = 1'b0; + lc_ack_o = 1'b0; + + // Error Register + error_d = error_q; + fsm_err_o = 1'b0; + + unique case (state_q) + /////////////////////////////////////////////////////////////////// + // State right after reset. Wait here until LCI gets enabled. + ResetSt: begin + lci_prog_idle_o = 1'b0; + if (lci_en_i) begin + state_d = IdleSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for a request from the life cycle controller + IdleSt: begin + if (lc_req_i) begin + state_d = WriteSt; + cnt_clr = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Loop through the lifecycle sate and burn in all words. + // If the write data contains a 0 bit in a position where a bit has already been + // programmed to 1 before, the OTP errors out. + WriteSt: begin + otp_req_o = 1'b1; + otp_cmd_o = prim_otp_pkg::Write; + lci_prog_idle_o = 1'b0; + if (otp_gnt_i) begin + state_d = WriteWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response, and check whether there are more words to burn in. + // In case an OTP transaction fails, latch the OTP error code, and jump to + // terminal error state. + WriteWaitSt: begin + lci_prog_idle_o = 1'b0; + if (otp_rvalid_i) begin + // Check OTP return code. + // Note that if errors occur, we aggregate the error code + // but still attempt to program all remaining words. + // This is done to ensure that a life cycle state with + // ECC correctable errors in some words can still be scrapped. + if (otp_err_e'(otp_err_i) != NoError) begin + error_d = otp_err_e'(otp_err_i); + end + + // Check whether we programmed all OTP words. + // If yes, we are done and can go back to idle. + if (cnt == LastLcOtpWord) begin + state_d = IdleSt; + lc_ack_o = 1'b1; + // If in any of the words a programming error has occurred, + // we signal that accordingly and go to the error state. + if (error_d != NoError) begin + lc_err_o = 1'b1; + state_d = ErrorSt; + end + // Otherwise we increase the OTP word counter. + end else begin + state_d = WriteSt; + cnt_en = 1'b1; + end + end + end + /////////////////////////////////////////////////////////////////// + // Terminal Error State. This locks access to the partition. + // Make sure the partition signals an error state if no error + // code has been latched so far, and lock the buffer regs down. + ErrorSt: begin + if (error_q == NoError) begin + error_d = FsmStateError; + end + end + /////////////////////////////////////////////////////////////////// + // We should never get here. If we do (e.g. via a malicious + // glitch), error out immediately. + default: begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + endcase // state_q + + // Unconditionally jump into the terminal error state in case of escalation. + // SEC_CM: LCI.FSM.LOCAL_ESC, LCI.FSM.GLOBAL_ESC + if (lc_ctrl_pkg::lc_tx_test_true_loose(escalate_en_i) || cnt_err) begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + if (error_q == NoError) begin + error_d = FsmStateError; + end + end + + end + + ////////////////////////////// + // Counter and address calc // + ////////////////////////////// + + // Native OTP word counter + // SEC_CM: LCI.CTR.REDUN + prim_count #( + .Width(CntWidth) + ) u_prim_count ( + .clk_i, + .rst_ni, + .clr_i(cnt_clr), + .set_i(1'b0), + .set_cnt_i('0), + .incr_en_i(cnt_en), + .decr_en_i(1'b0), + .step_i(CntWidth'(1)), + .commit_i(1'b1), + .cnt_o(cnt), + .cnt_after_commit_o(), + .err_o(cnt_err) + ); + + // The output address is "offset + count", but we have to convert Info.offset from a byte address + // to a halfword (16-bit) address by discarding the bottom OtpAddrShift bits. We also make the + // zero-extension of cnt explicit (to avoid width mismatch warnings). + assign otp_addr_o = Info.offset[OtpByteAddrWidth-1:OtpAddrShift] + OtpAddrWidth'(cnt); + + // Always transfer 16bit blocks. + assign otp_size_o = '0; + + logic [NumLcOtpWords-1:0][OtpWidth-1:0] data; + assign data = lc_data_i; + assign otp_wdata_o = (otp_req_o) ? OtpIfWidth'(data[cnt]) : '0; + + logic unused_rdata; + assign unused_rdata = ^otp_rdata_i; + + /////////////// + // Registers // + /////////////// + + `PRIM_FLOP_SPARSE_FSM(u_state_regs, state_d, state_q, state_e, ResetSt) + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + error_q <= NoError; + end else begin + error_q <= error_d; + end + end + + //////////////// + // Assertions // + //////////////// + + `ASSERT_KNOWN(LcAckKnown_A, lc_ack_o) + `ASSERT_KNOWN(LcErrKnown_A, lc_err_o) + `ASSERT_KNOWN(ErrorKnown_A, error_o) + `ASSERT_KNOWN(LciIdleKnown_A, lci_prog_idle_o) + `ASSERT_KNOWN(OtpReqKnown_A, otp_req_o) + `ASSERT_KNOWN(OtpCmdKnown_A, otp_cmd_o) + `ASSERT_KNOWN(OtpSizeKnown_A, otp_size_o) + `ASSERT_KNOWN(OtpWdataKnown_A, otp_wdata_o) + `ASSERT_KNOWN(OtpAddrKnown_A, otp_addr_o) + +endmodule : otp_ctrl_lci diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv new file mode 100644 index 00000000000000..775c8586cfcaf4 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv @@ -0,0 +1,398 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// This module implements the LFSR timer for triggering periodic consistency and integrity checks in +// OTP. In particular, this module contains two 40bit counters (one for the consistency and one +// for the integrity checks) and a 40bit LFSR to draw pseudo random wait counts. +// +// The integ_period_msk_i and cnsty_period_msk_i mask signals are used to mask off the LFSR outputs +// and hence determine the maximum wait count that can be drawn. If these values are set to +// zero, the corresponding timer is disabled. +// +// Once a particular check timer has expired, the module will send out a check request to all +// partitions and wait for an acknowledgment. If a particular partition encounters an integrity or +// consistency mismatch, this will be directly reported via the error and alert logic. +// +// In order to guard against wedged partition controllers or arbitration lock ups due to tampering +// attempts, this check timer module also supports a 32bit timeout that can optionally be +// programmed. If a particular check times out, chk_timeout_o will be asserted, which will raise +// an alert via the error logic. +// +// The EntropyWidth LSBs of the LFSR are reseeded with fresh entropy from EDN once +// LfsrUsageThreshold values have been drawn from the LFSR. +// +// It is also possible to trigger one-off checks via integ_chk_trig_i and cnsty_chk_trig_i. +// This can be useful if SW chooses to leave the periodic checks disabled. +// + +`include "prim_flop_macros.sv" + +module otp_ctrl_lfsr_timer + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_top_specific_pkg::*; +#( + // Compile time random constants, to be overriden by topgen. + parameter lfsr_seed_t RndCnstLfsrSeed = RndCnstLfsrSeedDefault, + parameter lfsr_perm_t RndCnstLfsrPerm = RndCnstLfsrPermDefault +) ( + input clk_i, + input rst_ni, + output logic edn_req_o, // request to EDN + input edn_ack_i, // ack from EDN + input [EdnDataWidth-1:0] edn_data_i, // from EDN + input timer_en_i, // enable timer + input otp_prog_busy_i, // indicates whether prog ops are in progress + input integ_chk_trig_i, // one-off trigger for integrity check + input cnsty_chk_trig_i, // one-off trigger for consistency check + output logic chk_pending_o, // indicates whether there are pending checks + input [31:0] timeout_i, // check timeout + input [31:0] integ_period_msk_i, // maximum integrity check mask + input [31:0] cnsty_period_msk_i, // maximum consistency check mask + output logic [NumPart-1:0] integ_chk_req_o, // request to all partitions + output logic [NumPart-1:0] cnsty_chk_req_o, // request to all partitions + input [NumPart-1:0] integ_chk_ack_i, // response from partitions + input [NumPart-1:0] cnsty_chk_ack_i, // response from partitions + input lc_ctrl_pkg::lc_tx_t escalate_en_i, // escalation input, moves FSM into ErrorSt + output logic chk_timeout_o, // a check has timed out + output logic fsm_err_o // the FSM has reached an invalid state +); + + //////////////////// + // Reseed counter // + //////////////////// + + // Count how many times the LFSR has been used to generate a value. + // Once we've reached the limit, we request new entropy from EDN to reseed + // the LFSR. Note that this is not a blocking operation for the timer below. + // I.e., the timer is allowed to continue its operation, and may draw more + // values, even if the EDN reseed request is still in progress. + logic reseed_en, lfsr_en; + logic [$clog2(LfsrUsageThreshold+1)-1:0] reseed_cnt_d, reseed_cnt_q; + assign reseed_cnt_d = (reseed_en) ? '0 : + (edn_req_o) ? reseed_cnt_q : + (lfsr_en) ? reseed_cnt_q + 1'b1 : + reseed_cnt_q; + + assign edn_req_o = (reseed_cnt_q >= LfsrUsageThreshold); + assign reseed_en = edn_req_o & edn_ack_i; + + /////////////////////////// + // Tandem LFSR Instances // + /////////////////////////// + + logic lfsr_err; + logic [LfsrWidth-1:0] entropy; + logic [LfsrWidth-1:0] lfsr_state; + assign entropy = (reseed_en) ? edn_data_i[LfsrWidth-1:0] : '0; + + // We employ two redundant LFSRs to guard against FI attacks. + // If any of the two is glitched and the two LFSR states do not agree, + // the FSM below is moved into a terminal error state. + // SEC_CM: TIMER.LFSR.REDUN + prim_double_lfsr #( + .LfsrDw ( LfsrWidth ), + .EntropyDw ( LfsrWidth ), + .StateOutDw ( LfsrWidth ), + .DefaultSeed ( RndCnstLfsrSeed ), + .StatePermEn ( 1'b1 ), + .StatePerm ( RndCnstLfsrPerm ), + .ExtSeedSVA ( 1'b0 ) + ) u_prim_double_lfsr ( + .clk_i, + .rst_ni, + .seed_en_i ( 1'b0 ), + .seed_i ( '0 ), + .lfsr_en_i ( reseed_en || lfsr_en ), + .entropy_i ( entropy ), + .state_o ( lfsr_state ), + .err_o ( lfsr_err ) + ); + + // Not all entropy bits are used. + logic unused_seed; + assign unused_seed = ^edn_data_i; + + `ASSERT_INIT(EdnIsWideEnough_A, EdnDataWidth >= LfsrWidth) + + ////////////////////////////// + // Tandem Counter Instances // + ////////////////////////////// + + // We employ redundant counters to guard against FI attacks. + // If any of them is glitched and the redundant counter states do not agree, + // the FSM below is moved into a terminal error state. + logic [LfsrWidth-1:0] integ_cnt, cnsty_cnt, integ_cnt_set_val, cnsty_cnt_set_val; + logic [LfsrWidth-1:0] integ_mask, cnsty_mask; + logic integ_set_period, integ_set_timeout, integ_cnt_zero; + logic cnsty_set_period, cnsty_set_timeout, cnsty_cnt_zero; + logic integ_cnt_set, cnsty_cnt_set, integ_cnt_err, cnsty_cnt_err; + logic timeout_zero, integ_msk_zero, cnsty_msk_zero, cnsty_cnt_pause; + + assign timeout_zero = (timeout_i == '0); + assign integ_msk_zero = (integ_period_msk_i == '0); + assign cnsty_msk_zero = (cnsty_period_msk_i == '0); + assign integ_cnt_zero = (integ_cnt == '0); + assign cnsty_cnt_zero = (cnsty_cnt == '0); + + assign integ_cnt_set = integ_set_period || integ_set_timeout; + assign cnsty_cnt_set = cnsty_set_period || cnsty_set_timeout; + + assign integ_mask = {integ_period_msk_i, {LfsrWidth-32{1'b1}}}; + assign cnsty_mask = {cnsty_period_msk_i, {LfsrWidth-32{1'b1}}}; + assign integ_cnt_set_val = (integ_set_period) ? (lfsr_state & integ_mask) : LfsrWidth'(timeout_i); + assign cnsty_cnt_set_val = (cnsty_set_period) ? (lfsr_state & cnsty_mask) : LfsrWidth'(timeout_i); + + // SEC_CM: TIMER_INTEG.CTR.REDUN + prim_count #( + .Width(LfsrWidth) + ) u_prim_count_integ ( + .clk_i, + .rst_ni, + .clr_i(1'b0), + .set_i(integ_cnt_set), + .set_cnt_i(integ_cnt_set_val), + .incr_en_i(1'b0), + .decr_en_i(!integ_cnt_zero), + .step_i(LfsrWidth'(1)), + .commit_i(1'b1), + .cnt_o(integ_cnt), + .cnt_after_commit_o(), + .err_o(integ_cnt_err) + ); + + // SEC_CM: TIMER_CNSTY.CTR.REDUN + prim_count #( + .Width(LfsrWidth) + ) u_prim_count_cnsty ( + .clk_i, + .rst_ni, + .clr_i(1'b0), + .set_i(cnsty_cnt_set), + .set_cnt_i(cnsty_cnt_set_val), + .incr_en_i(1'b0), + .decr_en_i(!cnsty_cnt_zero && !cnsty_cnt_pause), + .step_i(LfsrWidth'(1)), + .commit_i(1'b1), + .cnt_o(cnsty_cnt), + .cnt_after_commit_o(), + .err_o(cnsty_cnt_err) + ); + + ///////////////////// + // Request signals // + ///////////////////// + + logic set_all_integ_reqs, set_all_cnsty_reqs; + logic [NumPart-1:0] integ_chk_req_d, integ_chk_req_q; + logic [NumPart-1:0] cnsty_chk_req_d, cnsty_chk_req_q; + assign integ_chk_req_o = integ_chk_req_q; + assign cnsty_chk_req_o = cnsty_chk_req_q; + assign integ_chk_req_d = (set_all_integ_reqs) ? {NumPart{1'b1}} : + integ_chk_req_q & ~integ_chk_ack_i; + assign cnsty_chk_req_d = (set_all_cnsty_reqs) ? {NumPart{1'b1}} : + cnsty_chk_req_q & ~cnsty_chk_ack_i; + + + // external triggers + logic clr_integ_chk_trig, clr_cnsty_chk_trig; + logic integ_chk_trig_d, integ_chk_trig_q; + logic cnsty_chk_trig_d, cnsty_chk_trig_q; + assign integ_chk_trig_d = (integ_chk_trig_q & ~clr_integ_chk_trig) | integ_chk_trig_i; + assign cnsty_chk_trig_d = (cnsty_chk_trig_q & ~clr_cnsty_chk_trig) | cnsty_chk_trig_i; + + //////////////////////////// + // Ping and Timeout Logic // + //////////////////////////// + + // SEC_CM: TIMER.FSM.SPARSE + // Encoding generated with: + // $ ./util/design/sparse-fsm-encode.py -d 5 -m 5 -n 9 \ + // -s 628816752 --language=sv + // + // Hamming distance histogram: + // + // 0: -- + // 1: -- + // 2: -- + // 3: -- + // 4: -- + // 5: |||||||||||||||||||| (60.00%) + // 6: ||||||||||||| (40.00%) + // 7: -- + // 8: -- + // 9: -- + // + // Minimum Hamming distance: 5 + // Maximum Hamming distance: 6 + // Minimum Hamming weight: 4 + // Maximum Hamming weight: 6 + // + localparam int StateWidth = 9; + typedef enum logic [StateWidth-1:0] { + ResetSt = 9'b100100101, + IdleSt = 9'b001101110, + IntegWaitSt = 9'b010110011, + CnstyWaitSt = 9'b111010110, + ErrorSt = 9'b001011001 + } state_e; + + state_e state_d, state_q; + logic chk_timeout_d, chk_timeout_q; + + assign chk_timeout_o = chk_timeout_q; + + always_comb begin : p_fsm + state_d = state_q; + + // LFSR and counter signals + lfsr_en = 1'b0; + integ_set_period = 1'b0; + cnsty_set_period = 1'b0; + integ_set_timeout = 1'b0; + cnsty_set_timeout = 1'b0; + cnsty_cnt_pause = 1'b0; + + // Requests going to partitions. + set_all_integ_reqs = '0; + set_all_cnsty_reqs = '0; + + // Status signals going to CSRs and error logic. + chk_timeout_d = chk_timeout_q; + chk_pending_o = cnsty_chk_trig_q || integ_chk_trig_q; + fsm_err_o = 1'b0; + + // Clear signals for external triggers + clr_integ_chk_trig = 1'b0; + clr_cnsty_chk_trig = 1'b0; + + unique case (state_q) + /////////////////////////////////////////////////////////////////// + // Wait until enabled. We never return to this state + // once enabled! + ResetSt: begin + if (timer_en_i) begin + state_d = IdleSt; + lfsr_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Wait here until one of the two timers expires (if enabled) or if + // a check is triggered externally. + IdleSt: begin + if ((!integ_msk_zero && integ_cnt_zero) || integ_chk_trig_q) begin + state_d = IntegWaitSt; + integ_set_timeout = 1'b1; + set_all_integ_reqs = 1'b1; + clr_integ_chk_trig = integ_chk_trig_q; + end else if ((!cnsty_msk_zero && cnsty_cnt_zero) || cnsty_chk_trig_q) begin + state_d = CnstyWaitSt; + cnsty_set_timeout = 1'b1; + set_all_cnsty_reqs = 1'b1; + clr_cnsty_chk_trig = cnsty_chk_trig_q; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for all the partitions to respond and go back to idle. + // If the timeout is enabled, bail out into terminal error state + // if the timeout counter expires (this will raise an alert). + IntegWaitSt: begin + chk_pending_o = 1'b1; + if (!timeout_zero && integ_cnt_zero) begin + state_d = ErrorSt; + chk_timeout_d = 1'b1; + end else if (integ_chk_req_q == '0) begin + state_d = IdleSt; + // This draws the next wait period. + integ_set_period = 1'b1; + lfsr_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for all the partitions to respond and go back to idle. + // If the timeout is enabled, bail out into terminal error state + // if the timeout counter expires (this will raise an alert). + CnstyWaitSt: begin + chk_pending_o = 1'b1; + // Note that consistency checks go back and read from OTP. Hence, + // life cycle transitions and DAI programming operations + // may interfere with these checks and cause them to take longer + // than typically expected. Therefore, the timeout counter is stopped + // during programming operations. + cnsty_cnt_pause = otp_prog_busy_i; + if (!timeout_zero && cnsty_cnt_zero) begin + state_d = ErrorSt; + chk_timeout_d = 1'b1; + end else if (cnsty_chk_req_q == '0) begin + state_d = IdleSt; + // This draws the next wait period. + cnsty_set_period = 1'b1; + lfsr_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Terminal error state. This raises an alert. + ErrorSt: begin + // Continuously clear pending checks. + clr_integ_chk_trig = 1'b1; + clr_cnsty_chk_trig = 1'b1; + if (!chk_timeout_q) begin + fsm_err_o = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // This should never happen, hence we directly jump into the + // error state, where an alert will be triggered. + default: begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + endcase // state_q + + // Unconditionally jump into the terminal error state in case of escalation, + // or if the two LFSR or counter states do not agree. + // SEC_CM: TIMER.FSM.LOCAL_ESC, TIMER.FSM.GLOBAL_ESC + if (lfsr_err || integ_cnt_err || cnsty_cnt_err || + lc_ctrl_pkg::lc_tx_test_true_loose(escalate_en_i)) begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + end + + /////////////// + // Registers // + /////////////// + + `PRIM_FLOP_SPARSE_FSM(u_state_regs, state_d, state_q, state_e, ResetSt) + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + integ_chk_req_q <= '0; + cnsty_chk_req_q <= '0; + chk_timeout_q <= 1'b0; + reseed_cnt_q <= '0; + integ_chk_trig_q <= 1'b0; + cnsty_chk_trig_q <= 1'b0; + end else begin + integ_chk_req_q <= integ_chk_req_d; + cnsty_chk_req_q <= cnsty_chk_req_d; + chk_timeout_q <= chk_timeout_d; + reseed_cnt_q <= reseed_cnt_d; + integ_chk_trig_q <= integ_chk_trig_d; + cnsty_chk_trig_q <= cnsty_chk_trig_d; + end + end + + //////////////// + // Assertions // + //////////////// + + `ASSERT_KNOWN(EdnReqKnown_A, edn_req_o) + `ASSERT_KNOWN(ChkPendingKnown_A, chk_pending_o) + `ASSERT_KNOWN(IntegChkReqKnown_A, integ_chk_req_o) + `ASSERT_KNOWN(CnstyChkReqKnown_A, cnsty_chk_req_o) + `ASSERT_KNOWN(ChkTimeoutKnown_A, chk_timeout_o) + +endmodule : otp_ctrl_lfsr_timer diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_buf.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_buf.sv new file mode 100644 index 00000000000000..c4da5a89302840 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_buf.sv @@ -0,0 +1,822 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Buffered partition for OTP controller. +// + +`include "prim_flop_macros.sv" + +module otp_ctrl_part_buf + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; +#( + // Partition information. + parameter part_info_t Info = PartInfoDefault, + parameter logic [Info.size*8-1:0] DataDefault = '0 +) ( + input clk_i, + input rst_ni, + // Pulse to start partition initialisation (required once per power cycle). + input init_req_i, + output logic init_done_o, + // Integrity check requests + input integ_chk_req_i, + output logic integ_chk_ack_o, + // Consistency check requests + input cnsty_chk_req_i, + output logic cnsty_chk_ack_o, + // Escalation input. This moves the FSM into a terminal state and locks down + // the partition. + input lc_ctrl_pkg::lc_tx_t escalate_en_i, + // Check bypass enable. This bypasses integrity and consistency checks and + // acknowledges all incoming check requests (only used by life cycle). + input lc_ctrl_pkg::lc_tx_t check_byp_en_i, + // Output error state of partition, to be consumed by OTP error/alert logic. + // Note that most errors are not recoverable and move the partition FSM into + // a terminal error state. + output otp_err_e error_o, + // This error signal is pulsed high if the FSM has been glitched into an invalid state. + // Although it is somewhat redundant with the error code in error_o above, it is + // meant to cover cases where we already latched an error code while the FSM is + // glitched into an invalid state (since in that case, the error code will not be + // overridden with the FSM error code so that the original error code is still + // discoverable). + output logic fsm_err_o, + // Access/lock status + // SEC_CM: ACCESS.CTRL.MUBI + input part_access_t access_i, // runtime lock from CSRs + output part_access_t access_o, + // Buffered 64bit digest output. + output logic [ScrmblBlockWidth-1:0] digest_o, + output logic [Info.size*8-1:0] data_o, + // OTP interface + output logic otp_req_o, + output prim_otp_pkg::cmd_e otp_cmd_o, + output logic [OtpSizeWidth-1:0] otp_size_o, + output logic [OtpIfWidth-1:0] otp_wdata_o, + output logic [OtpAddrWidth-1:0] otp_addr_o, + input otp_gnt_i, + input otp_rvalid_i, + input [ScrmblBlockWidth-1:0] otp_rdata_i, + input prim_otp_pkg::err_e otp_err_i, + // Scrambling mutex request + output logic scrmbl_mtx_req_o, + input scrmbl_mtx_gnt_i, + // Scrambling datapath interface + output otp_scrmbl_cmd_e scrmbl_cmd_o, + output digest_mode_e scrmbl_mode_o, + output logic [ConstSelWidth-1:0] scrmbl_sel_o, + output logic [ScrmblBlockWidth-1:0] scrmbl_data_o, + output logic scrmbl_valid_o, + input logic scrmbl_ready_i, + input logic scrmbl_valid_i, + input logic [ScrmblBlockWidth-1:0] scrmbl_data_i +); + + //////////////////////// + // Integration Checks // + //////////////////////// + + import prim_mubi_pkg::*; + import prim_util_pkg::vbits; + + localparam int unsigned DigestOffsetInt = (int'(Info.offset) + + int'(Info.size) - ScrmblBlockWidth/8); + localparam int NumScrmblBlocks = int'(Info.size) / (ScrmblBlockWidth/8); + localparam int CntWidth = vbits(NumScrmblBlocks); + + localparam bit [OtpByteAddrWidth-1:0] DigestOffset = DigestOffsetInt[OtpByteAddrWidth-1:0]; + + localparam int unsigned LastScrmblBlockInt = NumScrmblBlocks - 1; + localparam int unsigned PenultimateScrmblBlockInt = NumScrmblBlocks - 2; + localparam bit [CntWidth-1:0] LastScrmblBlock = LastScrmblBlockInt[CntWidth-1:0]; + localparam bit [CntWidth-1:0] PenultimateScrmblBlock = PenultimateScrmblBlockInt[CntWidth-1:0]; + + // Integration checks for parameters. + `ASSERT_INIT(OffsetMustBeBlockAligned_A, (Info.offset % (ScrmblBlockWidth/8)) == 0) + `ASSERT_INIT(SizeMustBeBlockAligned_A, (Info.size % (ScrmblBlockWidth/8)) == 0) + `ASSERT_INIT(DigestOffsetMustBeRepresentable_A, DigestOffsetInt == int'(DigestOffset)) + `ASSERT(ScrambledImpliesDigest_A, Info.secret |-> Info.hw_digest) + `ASSERT(WriteLockImpliesDigest_A, Info.read_lock |-> Info.hw_digest) + `ASSERT(ReadLockImpliesDigest_A, Info.write_lock |-> Info.hw_digest) + + // This feature is only supposed to be used with partitions that are not scrambled + // and that do not have a digest. + `ASSERT(BypassEnable0_A, Info.secret |-> lc_ctrl_pkg::lc_tx_test_false_strict(check_byp_en_i)) + `ASSERT(BypassEnable1_A, Info.hw_digest |-> lc_ctrl_pkg::lc_tx_test_false_strict(check_byp_en_i)) + + /////////////////////// + // OTP Partition FSM // + /////////////////////// + + // SEC_CM: PART.FSM.SPARSE + // Encoding generated with: + // $ ./util/design/sparse-fsm-encode.py -d 5 -m 16 -n 12 \ + // -s 3370657881 --language=sv + // + // Hamming distance histogram: + // + // 0: -- + // 1: -- + // 2: -- + // 3: -- + // 4: -- + // 5: |||||||||||||| (28.33%) + // 6: |||||||||||||||||||| (38.33%) + // 7: |||||||||| (19.17%) + // 8: ||| (5.83%) + // 9: || (4.17%) + // 10: | (2.50%) + // 11: (0.83%) + // 12: (0.83%) + // + // Minimum Hamming distance: 5 + // Maximum Hamming distance: 12 + // Minimum Hamming weight: 4 + // Maximum Hamming weight: 8 + // + localparam int StateWidth = 12; + typedef enum logic [StateWidth-1:0] { + ResetSt = 12'b011000001110, + InitSt = 12'b110100100111, + InitWaitSt = 12'b001110110001, + InitDescrSt = 12'b110010000100, + InitDescrWaitSt = 12'b100110101000, + IdleSt = 12'b010101001101, + IntegScrSt = 12'b110101011010, + IntegScrWaitSt = 12'b100010011111, + IntegDigClrSt = 12'b101001000001, + IntegDigSt = 12'b011101100010, + IntegDigPadSt = 12'b001101010111, + IntegDigFinSt = 12'b011011100101, + IntegDigWaitSt = 12'b100011110010, + CnstyReadSt = 12'b000001101011, + CnstyReadWaitSt = 12'b101001111100, + ErrorSt = 12'b010110111110 + } state_e; + + typedef enum logic { + ScrmblData, + OtpData + } data_sel_e; + + typedef enum logic { + PartOffset, + DigOffset + } base_sel_e; + + state_e state_d, state_q; + otp_err_e error_d, error_q; + data_sel_e data_sel; + base_sel_e base_sel; + mubi8_t dout_locked_d, dout_locked_q; + logic [CntWidth-1:0] cnt; + logic cnt_en, cnt_clr, cnt_err; + logic ecc_err; + logic buffer_reg_en; + logic [ScrmblBlockWidth-1:0] data_mux; + + // Output partition error state. + assign error_o = error_q; + + // This partition cannot do any write accesses, hence we tie this + // constantly off. + assign otp_wdata_o = '0; + // Depending on the partition configuration, the wrapper is instructed to ignore integrity + // calculations and checks. To be on the safe side, the partition filters error responses at this + // point and does not report any integrity errors if integrity is disabled. + otp_err_e otp_err; + if (Info.integrity) begin : gen_integrity + assign otp_cmd_o = prim_otp_pkg::Read; + assign otp_err = otp_err_e'(otp_err_i); + end else begin : gen_no_integrity + assign otp_cmd_o = prim_otp_pkg::ReadRaw; + always_comb begin + if (otp_err_e'(otp_err_i) inside {MacroEccCorrError, MacroEccUncorrError}) begin + otp_err = NoError; + end else begin + otp_err = otp_err_e'(otp_err_i); + end + end + end + + always_comb begin : p_fsm + state_d = state_q; + + // Redundantly encoded lock signal for buffer regs. + dout_locked_d = dout_locked_q; + + // OTP signals + otp_req_o = 1'b0; + + // Scrambling mutex + scrmbl_mtx_req_o = 1'b0; + + // Scrambling datapath + scrmbl_cmd_o = LoadShadow; + scrmbl_sel_o = CnstyDigest; + scrmbl_mode_o = StandardMode; + scrmbl_valid_o = 1'b0; + + // Counter + cnt_en = 1'b0; + cnt_clr = 1'b0; + base_sel = PartOffset; + + // Buffer register + buffer_reg_en = 1'b0; + data_sel = OtpData; + + // Error Register + error_d = error_q; + fsm_err_o = 1'b0; + + // Integrity/Consistency check responses + cnsty_chk_ack_o = 1'b0; + integ_chk_ack_o = 1'b0; + + unique case (state_q) + /////////////////////////////////////////////////////////////////// + // State right after reset. Wait here until we get a an + // initialization request. + ResetSt: begin + if (init_req_i) begin + state_d = InitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Initialization reads out the digest only in unbuffered + // partitions. Wait here until the OTP request has been granted. + // And then wait until the OTP word comes back. + InitSt: begin + otp_req_o = 1'b1; + if (otp_gnt_i) begin + state_d = InitWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response and write to buffer register, then go to + // descrambling state. In case an OTP transaction fails, latch the + // OTP error code and jump to a + // terminal error state. + InitWaitSt: begin + if (otp_rvalid_i) begin + buffer_reg_en = 1'b1; + if (otp_err inside {NoError, MacroEccCorrError}) begin + // Once we've read and descrambled the whole partition, we can go to integrity + // verification. Note that the last block is the digest value, which does not + // have to be descrambled. + if (cnt == LastScrmblBlock) begin + state_d = IntegDigClrSt; + // Only need to descramble if this is a scrambled partition. + // Otherwise, we can just go back to InitSt and read the next block. + end else if (Info.secret) begin + state_d = InitDescrSt; + end else begin + state_d = InitSt; + cnt_en = 1'b1; + end + // At this point the only error that we could have gotten are correctable ECC errors. + if (otp_err != NoError) begin + error_d = MacroEccCorrError; + end + end else begin + state_d = ErrorSt; + error_d = otp_err; + end + end + end + /////////////////////////////////////////////////////////////////// + // Descrambling state. This first acquires the scrambling + // datapath mutex. Note that once the mutex is acquired, we have + // exclusive access to the scrambling datapath until we release + // the mutex by deasserting scrmbl_mtx_req_o. + // SEC_CM: SECRET.MEM.SCRAMBLE + InitDescrSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = Decrypt; + scrmbl_sel_o = Info.key_sel; + if (scrmbl_mtx_gnt_i && scrmbl_ready_i) begin + state_d = InitDescrWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for the descrambled data to return. Note that we release + // the mutex lock upon leaving this state. + // SEC_CM: SECRET.MEM.SCRAMBLE + InitDescrWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_sel_o = Info.key_sel; + data_sel = ScrmblData; + if (scrmbl_valid_i) begin + state_d = InitSt; + buffer_reg_en = 1'b1; + cnt_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Idle state. We basically wait for integrity and consistency check + // triggers in this state. + IdleSt: begin + if (integ_chk_req_i) begin + if (Info.hw_digest) begin + state_d = IntegDigClrSt; + // In case there is nothing to check we can just + // acknowledge the request right away, without going to the + // integrity check. + end else begin + integ_chk_ack_o = 1'b1; + end + end else if (cnsty_chk_req_i) begin + state_d = CnstyReadSt; + cnt_clr = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Read the digest. Wait here until the OTP request has been granted. + // And then wait until the OTP word comes back. + // SEC_CM: PART.DATA_REG.BKGN_CHK + CnstyReadSt: begin + otp_req_o = 1'b1; + // In case this partition has a hardware digest, we only have to read + // and compare the digest value. In that case we select the digest offset here. + // Otherwise we have to read and compare the whole partition, in which case we + // select the partition offset, which is the default assignment of base_sel. + if (Info.hw_digest) begin + base_sel = DigOffset; + end + if (otp_gnt_i) begin + state_d = CnstyReadWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response and compare the digest. In case there is + // a mismatch, lock down the partition and go into the terminal error + // state. In case an OTP transaction fails, latch the OTP error code + // and jump to a terminal error state. + // SEC_CM: PART.DATA_REG.BKGN_CHK + CnstyReadWaitSt: begin + if (otp_rvalid_i) begin + if (otp_err inside {NoError, MacroEccCorrError}) begin + // Check whether we need to compare the digest or the full partition + // contents here. + if (Info.hw_digest) begin + // Note that we ignore this check if the digest is still blank. + if (digest_o == data_mux || digest_o == '0) begin + state_d = IdleSt; + cnsty_chk_ack_o = 1'b1; + // Error out and lock the partition if this check fails. + end else begin + state_d = ErrorSt; + error_d = CheckFailError; + // The check has finished and found an error. + cnsty_chk_ack_o = 1'b1; + end + end else begin + // Check whether the read data corresponds with the data buffered in regs. + // Note that this particular check can be bypassed in case a transition is ongoing. + if (scrmbl_data_o == data_mux || + lc_ctrl_pkg::lc_tx_test_true_strict(check_byp_en_i)) begin + // Can go back to idle and acknowledge the + // request if this is the last block. + if (cnt == LastScrmblBlock) begin + state_d = IdleSt; + cnsty_chk_ack_o = 1'b1; + // Need to go back and read out more blocks. + end else begin + state_d = CnstyReadSt; + cnt_en = 1'b1; + end + end else begin + state_d = ErrorSt; + error_d = CheckFailError; + // The check has finished and found an error. + cnsty_chk_ack_o = 1'b1; + end + end + // At this point the only error that we could have gotten are correctable ECC errors. + if (otp_err != NoError) begin + error_d = MacroEccCorrError; + end + end else begin + state_d = ErrorSt; + error_d = otp_err; + // The check has finished and found an error. + cnsty_chk_ack_o = 1'b1; + end + end + end + /////////////////////////////////////////////////////////////////// + // First, acquire the mutex for the digest and clear the digest state. + // SEC_CM: PART.DATA_REG.BKGN_CHK + IntegDigClrSt: begin + // Check whether this partition requires checking at all. + if (Info.hw_digest) begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + cnt_clr = 1'b1; + // Need to reset the digest state and set it to chained + // mode if this partition is scrambled. + scrmbl_cmd_o = DigestInit; + if (Info.secret) begin + scrmbl_mode_o = ChainedMode; + if (scrmbl_mtx_gnt_i && scrmbl_ready_i) begin + state_d = IntegScrSt; + end + // If this partition is not scrambled, we can just directly + // jump to the digest state. + end else begin + scrmbl_mode_o = StandardMode; + if (scrmbl_mtx_gnt_i && scrmbl_ready_i) begin + state_d = IntegDigSt; + end + end + // Otherwise, if this partition is not digest protected, + // we can just go to idle, since there is nothing to check. + // Note that we do not come back to this state in case there is no + // digest, and hence it is safe to unlock the buffer regs at this point. + // This is the only way the buffer regs can get unlocked. + end else begin + state_d = IdleSt; + if (mubi8_test_true_strict(dout_locked_q)) begin + dout_locked_d = MuBi8False; + end + end + end + /////////////////////////////////////////////////////////////////// + // Scramble buffered data (which is held in plaintext form). + // This moves the previous scrambling result into the shadow reg + // for later use. + // SEC_CM: PART.DATA_REG.BKGN_CHK + IntegScrSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = Encrypt; + scrmbl_sel_o = Info.key_sel; + if (scrmbl_ready_i) begin + state_d = IntegScrWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for the scrambled data to return. + // SEC_CM: PART.DATA_REG.BKGN_CHK + IntegScrWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_sel_o = Info.key_sel; + if (scrmbl_valid_i) begin + state_d = IntegDigSt; + end + end + /////////////////////////////////////////////////////////////////// + // Push the word read into the scrambling datapath. The last + // block is repeated in case the number blocks in this partition + // is odd. + // SEC_CM: PART.MEM.DIGEST + // SEC_CM: PART.DATA_REG.BKGN_CHK + IntegDigSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + if (scrmbl_ready_i) begin + cnt_en = 1'b1; + // No need to digest the digest value itself + if (cnt == PenultimateScrmblBlock) begin + // Note that the digest operates on 128bit blocks since the data is fed in via the + // PRESENT key input. Therefore, we only trigger a digest update on every second + // 64bit block that is pushed into the scrambling datapath. + if (cnt[0]) begin + scrmbl_cmd_o = Digest; + state_d = IntegDigFinSt; + end else begin + state_d = IntegDigPadSt; + cnt_en = 1'b0; + end + end else begin + // Trigger digest round in case this is the second block in a row. + if (cnt[0]) begin + scrmbl_cmd_o = Digest; + end + // Go back and scramble the next data block if this is + // a scrambled partition. Otherwise just stay here. + if (Info.secret) begin + state_d = IntegScrSt; + end + end + end + end + /////////////////////////////////////////////////////////////////// + // Padding state. When we get here, we've copied the last encryption + // result into the shadow register such that we've effectively + // repeated the last block twice in order to pad the data to 128bit. + // SEC_CM: PART.MEM.DIGEST + // SEC_CM: PART.DATA_REG.BKGN_CHK + IntegDigPadSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = Digest; + if (scrmbl_ready_i) begin + state_d = IntegDigFinSt; + end + end + /////////////////////////////////////////////////////////////////// + // Trigger digest finalization and go wait for the result. + // SEC_CM: PART.MEM.DIGEST + // SEC_CM: PART.DATA_REG.BKGN_CHK + IntegDigFinSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = DigestFinalize; + if (scrmbl_ready_i) begin + state_d = IntegDigWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for the digest to return, and double check whether the digest + // matches. If yes, unlock the partition. Otherwise, go into the terminal + // error state, where the partition will be locked down. + // SEC_CM: PART.MEM.DIGEST + // SEC_CM: PART.DATA_REG.BKGN_CHK + IntegDigWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + data_sel = ScrmblData; + if (scrmbl_valid_i) begin + // This is the only way the buffer regs can get unlocked. + // Note that we ignore this check if the digest is still blank. + if (digest_o == data_mux || digest_o == '0) begin + state_d = IdleSt; + // If the partition is still locked, this is the first integrity check after + // initialization. This is the only way the buffer regs can get unlocked. + if (mubi8_test_true_strict(dout_locked_q)) begin + dout_locked_d = MuBi8False; + // Otherwise, this integrity check has requested by the LFSR timer, and we have + // to acknowledge its completion. + end else begin + integ_chk_ack_o = 1'b1; + end + // Error out and lock the partition if this check fails. + end else begin + state_d = ErrorSt; + error_d = CheckFailError; + // The check has finished and found an error. + integ_chk_ack_o = 1'b1; + end + end + end + /////////////////////////////////////////////////////////////////// + // Terminal Error State. This locks access to the partition. + // Make sure the partition signals an error state if no error + // code has been latched so far, and lock the buffer regs down. + ErrorSt: begin + dout_locked_d = MuBi8True; + if (error_q == NoError) begin + error_d = FsmStateError; + end + // If we are in error state, we cannot execute the checks anymore. + // Hence the acknowledgements are returned immediately. + cnsty_chk_ack_o = 1'b1; + integ_chk_ack_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + // We should never get here. If we do (e.g. via a malicious + // glitch), error out immediately. + default: begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + endcase // state_q + + + // Unconditionally jump into the terminal error state in case of + // an ECC error or escalation, and lock access to the partition down. + // SEC_CM: PART.FSM.LOCAL_ESC + if (ecc_err) begin + state_d = ErrorSt; + if (state_q != ErrorSt) begin + error_d = CheckFailError; + end + end + // SEC_CM: PART.FSM.LOCAL_ESC, PART.FSM.GLOBAL_ESC + if (lc_ctrl_pkg::lc_tx_test_true_loose(escalate_en_i) || cnt_err) begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + if (state_q != ErrorSt) begin + error_d = FsmStateError; + end + end + end + + //////////////////////////// + // Address Calc and Muxes // + //////////////////////////// + + // Address counter - this is only used for computing a digest, hence the increment is + // fixed to 8 byte. + // SEC_CM: PART.CTR.REDUN + prim_count #( + .Width(CntWidth) + ) u_prim_count ( + .clk_i, + .rst_ni, + .clr_i(cnt_clr), + .set_i(1'b0), + .set_cnt_i('0), + .incr_en_i(cnt_en), + .decr_en_i(1'b0), + .step_i(CntWidth'(1)), + .commit_i(1'b1), + .cnt_o(cnt), + .cnt_after_commit_o(), + .err_o(cnt_err) + ); + + logic [OtpByteAddrWidth-1:0] addr_base; + assign addr_base = (base_sel == DigOffset) ? DigestOffset : Info.offset; + + // Note that OTP works on halfword (16bit) addresses, hence need to + // shift the addresses appropriately. + logic [OtpByteAddrWidth-1:0] addr_calc; + assign addr_calc = OtpByteAddrWidth'({cnt, {$clog2(ScrmblBlockWidth/8){1'b0}}}) + addr_base; + assign otp_addr_o = addr_calc[OtpByteAddrWidth-1:OtpAddrShift]; + + if (OtpAddrShift > 0) begin : gen_unused + logic unused_bits; + assign unused_bits = ^addr_calc[OtpAddrShift-1:0]; + end + + // Always transfer 64bit blocks. + assign otp_size_o = OtpSizeWidth'(unsigned'(ScrmblBlockWidth / OtpWidth) - 1); + + assign data_mux = (data_sel == ScrmblData) ? scrmbl_data_i : otp_rdata_i; + + ///////////////// + // Buffer Regs // + ///////////////// + + // SEC_CM: PART.DATA_REG.INTEGRITY + logic [Info.size*8-1:0] data; + otp_ctrl_ecc_reg #( + .Width ( ScrmblBlockWidth ), + .Depth ( NumScrmblBlocks ) + ) u_otp_ctrl_ecc_reg ( + .clk_i, + .rst_ni, + .wren_i ( buffer_reg_en ), + .addr_i ( cnt ), + .wdata_i ( data_mux ), + .rdata_o ( scrmbl_data_o ), + .data_o ( data ), + .ecc_err_o ( ecc_err ) + ); + + // We have successfully initialized the partition once it has been unlocked. + assign init_done_o = mubi8_test_false_strict(dout_locked_q); + // Hardware output gating. + // Note that this is decoupled from the DAI access rules further below. + assign data_o = (init_done_o) ? data : DataDefault; + // The digest does not have to be gated. + assign digest_o = data[$high(data_o) -: ScrmblBlockWidth]; + + //////////////////////// + // DAI Access Control // + //////////////////////// + + // Aggregate all possible DAI write /readlocks. The partition is also locked when uninitialized. + // Note that the locks are redundantly encoded values. + part_access_t access_pre; + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_write_lock_pre ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(dout_locked_q, access_i.write_lock)), + .mubi_o(access_pre.write_lock) + ); + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_read_lock_pre ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(dout_locked_q, access_i.read_lock)), + .mubi_o(access_pre.read_lock) + ); + + // SEC_CM: PART.MEM.SW_UNWRITABLE + if (Info.write_lock) begin : gen_digest_write_lock + mubi8_t digest_locked; + assign digest_locked = (digest_o != '0) ? MuBi8True : MuBi8False; + + // This prevents the synthesis tool from optimizing the multibit signal. + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_write_lock ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(access_pre.write_lock, digest_locked)), + .mubi_o(access_o.write_lock) + ); + + `ASSERT(DigestWriteLocksPartition_A, digest_o |-> mubi8_test_true_loose(access_o.write_lock)) + end else begin : gen_no_digest_write_lock + assign access_o.write_lock = access_pre.write_lock; + end + + // SEC_CM: PART.MEM.SW_UNREADABLE + if (Info.read_lock) begin : gen_digest_read_lock + mubi8_t digest_locked; + assign digest_locked = (digest_o != '0) ? MuBi8True : MuBi8False; + + // This prevents the synthesis tool from optimizing the multibit signal. + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_read_lock ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(access_pre.read_lock, digest_locked)), + .mubi_o(access_o.read_lock) + ); + + `ASSERT(DigestReadLocksPartition_A, digest_o |-> mubi8_test_true_loose(access_o.read_lock)) + end else begin : gen_no_digest_read_lock + assign access_o.read_lock = access_pre.read_lock; + end + + /////////////// + // Registers // + /////////////// + + `PRIM_FLOP_SPARSE_FSM(u_state_regs, state_d, state_q, state_e, ResetSt) + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + error_q <= NoError; + // data output is locked by default + dout_locked_q <= MuBi8True; + end else begin + error_q <= error_d; + dout_locked_q <= dout_locked_d; + end + end + + //////////////// + // Assertions // + //////////////// + + // Known assertions + `ASSERT_KNOWN(InitDoneKnown_A, init_done_o) + `ASSERT_KNOWN(IntegChkAckKnown_A, integ_chk_ack_o) + `ASSERT_KNOWN(CnstyChkAckKnown_A, cnsty_chk_ack_o) + `ASSERT_KNOWN(ErrorKnown_A, error_o) + `ASSERT_KNOWN(AccessKnown_A, access_o) + `ASSERT_KNOWN(DigestKnown_A, digest_o) + `ASSERT_KNOWN(DataKnown_A, data_o) + `ASSERT_KNOWN(OtpReqKnown_A, otp_req_o) + `ASSERT_KNOWN(OtpCmdKnown_A, otp_cmd_o) + `ASSERT_KNOWN(OtpSizeKnown_A, otp_size_o) + `ASSERT_KNOWN(OtpWdataKnown_A, otp_wdata_o) + `ASSERT_KNOWN(OtpAddrKnown_A, otp_addr_o) + `ASSERT_KNOWN(ScrmblMtxReqKnown_A, scrmbl_mtx_req_o) + `ASSERT_KNOWN(ScrmblCmdKnown_A, scrmbl_cmd_o) + `ASSERT_KNOWN(ScrmblModeKnown_A, scrmbl_mode_o) + `ASSERT_KNOWN(ScrmblSelKnown_A, scrmbl_sel_o) + `ASSERT_KNOWN(ScrmblDataKnown_A, scrmbl_data_o) + `ASSERT_KNOWN(ScrmblValidKnown_A, scrmbl_valid_o) + + // Uninitialized partitions should always be locked, no matter what. + `ASSERT(InitWriteLocksPartition_A, + mubi8_test_true_loose(dout_locked_q) + |-> + mubi8_test_true_loose(access_o.write_lock)) + `ASSERT(InitReadLocksPartition_A, + mubi8_test_true_loose(dout_locked_q) + |-> + mubi8_test_true_loose(access_o.read_lock)) + // Incoming Lock propagation + `ASSERT(WriteLockPropagation_A, + mubi8_test_true_loose(access_i.write_lock) + |-> + mubi8_test_true_loose(access_o.write_lock)) + `ASSERT(ReadLockPropagation_A, + mubi8_test_true_loose(access_i.read_lock) + |-> + mubi8_test_true_loose(access_o.read_lock)) + // ECC error in buffer regs + `ASSERT(EccErrorState_A, + ecc_err + |=> + state_q == ErrorSt) + // OTP error response + `ASSERT(OtpErrorState_A, + state_q inside {InitWaitSt, CnstyReadWaitSt} && otp_rvalid_i && + !(otp_err inside {NoError, MacroEccCorrError}) && !ecc_err + |=> + state_q == ErrorSt && error_o == $past(otp_err)) + + // The partition size must be greater than one scrambling block for the address calculation + // and muxing to work correctly. + `ASSERT_INIT(OtpPartBufSize_A, Info.size > (ScrmblBlockWidth/8)) + +endmodule : otp_ctrl_part_buf diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_pkg.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_pkg.sv new file mode 100644 index 00000000000000..36826899af8dc6 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_pkg.sv @@ -0,0 +1,997 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Package partition metadata. +// +package otp_ctrl_part_pkg; + + import prim_util_pkg::vbits; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_pkg::*; + import otp_ctrl_top_specific_pkg::*; + + //////////////////////////////////// + // Scrambling Constants and Types // + //////////////////////////////////// + + parameter int NumScrmblKeys = 4; + parameter int NumDigestSets = 4; + + parameter int ScrmblKeySelWidth = vbits(NumScrmblKeys); + parameter int DigestSetSelWidth = vbits(NumDigestSets); + parameter int ConstSelWidth = (ScrmblKeySelWidth > DigestSetSelWidth) ? + ScrmblKeySelWidth : + DigestSetSelWidth; + + typedef enum logic [ConstSelWidth-1:0] { + StandardMode, + ChainedMode + } digest_mode_e; + + typedef logic [NumScrmblKeys-1:0][ScrmblKeyWidth-1:0] key_array_t; + typedef logic [NumDigestSets-1:0][ScrmblKeyWidth-1:0] digest_const_array_t; + typedef logic [NumDigestSets-1:0][ScrmblBlockWidth-1:0] digest_iv_array_t; + + typedef enum logic [ConstSelWidth-1:0] { + Secret0Key, + Secret1Key, + Secret2Key, + Secret3Key + } key_sel_e; + + typedef enum logic [ConstSelWidth-1:0] { + CnstyDigest, + FlashDataKey, + FlashAddrKey, + SramDataKey + } digest_sel_e; + + // SEC_CM: SECRET.MEM.SCRAMBLE + parameter key_array_t RndCnstKey = { + 128'hBEAD91D5FA4E09150E95F517CB98955B, + 128'h85A9E830BC059BA9286D6E2856A05CC3, + 128'hEFFA6D736C5EFF49AE7B70F9C46E5A62, + 128'h3BA121C5E097DDEB7768B4C666E9C3DA + }; + + // SEC_CM: PART.MEM.DIGEST + // Note: digest set 0 is used for computing the partition digests. Constants at + // higher indices are used to compute the scrambling keys. + parameter digest_const_array_t RndCnstDigestConst = { + 128'h63B9485A3856C417CF7A50A9A91EF7F7, + 128'hF98C48B1F93772844A22D4B78FE0266F, + 128'hB7474D640F8A7F5D60822E1FAEC5C72, + 128'hE048B657396B4B83277195FC471E4B26 + }; + + parameter digest_iv_array_t RndCnstDigestIV = { + 64'hBEE3958332F2939B, + 64'h90C7F21F6224F027, + 64'hB6641214B61D1B43, + 64'h4D5A89AA9109294A + }; + + + ///////////////////////////////////// + // Typedefs for Partition Metadata // + ///////////////////////////////////// + + typedef enum logic [1:0] { + Unbuffered, + Buffered, + LifeCycle + } part_variant_e; + + typedef struct packed { + part_variant_e variant; + // Offset and size within the OTP array, in Bytes. + logic [OtpByteAddrWidth-1:0] offset; + logic [OtpByteAddrWidth-1:0] size; + // Key index to use for scrambling. + key_sel_e key_sel; + // Attributes + logic secret; // Whether the partition is secret (and hence scrambled) + logic sw_digest; // Whether the partition has a software digest + logic hw_digest; // Whether the partition has a hardware digest + logic write_lock; // Whether the partition is write lockable (via digest) + logic read_lock; // Whether the partition is read lockable (via digest) + logic integrity; // Whether the partition is integrity protected + logic iskeymgr_creator; // Whether the partition has any creator key material + logic iskeymgr_owner; // Whether the partition has any owner key material + } part_info_t; + + parameter part_info_t PartInfoDefault = '{ + variant: Unbuffered, + offset: '0, + size: OtpByteAddrWidth'('hFF), + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b0, + hw_digest: 1'b0, + write_lock: 1'b0, + read_lock: 1'b0, + integrity: 1'b0, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }; + + //////////////////////// + // Partition Metadata // + //////////////////////// + + localparam part_info_t PartInfo [NumPart] = '{ + // VENDOR_TEST + '{ + variant: Unbuffered, + offset: 14'd0, + size: 64, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b1, + hw_digest: 1'b0, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b0, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // CREATOR_SW_CFG + '{ + variant: Unbuffered, + offset: 14'd64, + size: 320, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b1, + hw_digest: 1'b0, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // OWNER_SW_CFG + '{ + variant: Unbuffered, + offset: 14'd384, + size: 632, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b1, + hw_digest: 1'b0, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // OWNERSHIP_SLOT_STATE + '{ + variant: Unbuffered, + offset: 14'd1016, + size: 48, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b0, + hw_digest: 1'b0, + write_lock: 1'b0, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // ROT_CREATOR_AUTH + '{ + variant: Unbuffered, + offset: 14'd1064, + size: 1424, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b1, + hw_digest: 1'b0, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // ROT_OWNER_AUTH_SLOT0 + '{ + variant: Unbuffered, + offset: 14'd2488, + size: 328, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b1, + hw_digest: 1'b0, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // ROT_OWNER_AUTH_SLOT1 + '{ + variant: Unbuffered, + offset: 14'd2816, + size: 328, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b1, + hw_digest: 1'b0, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // PLAT_INTEG_AUTH_SLOT0 + '{ + variant: Unbuffered, + offset: 14'd3144, + size: 328, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b1, + hw_digest: 1'b0, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // PLAT_INTEG_AUTH_SLOT1 + '{ + variant: Unbuffered, + offset: 14'd3472, + size: 328, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b1, + hw_digest: 1'b0, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // PLAT_OWNER_AUTH_SLOT0 + '{ + variant: Unbuffered, + offset: 14'd3800, + size: 328, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b1, + hw_digest: 1'b0, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // PLAT_OWNER_AUTH_SLOT1 + '{ + variant: Unbuffered, + offset: 14'd4128, + size: 328, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b1, + hw_digest: 1'b0, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // PLAT_OWNER_AUTH_SLOT2 + '{ + variant: Unbuffered, + offset: 14'd4456, + size: 328, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b1, + hw_digest: 1'b0, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // PLAT_OWNER_AUTH_SLOT3 + '{ + variant: Unbuffered, + offset: 14'd4784, + size: 328, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b1, + hw_digest: 1'b0, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // EXT_NVM + '{ + variant: Unbuffered, + offset: 14'd5112, + size: 1024, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b0, + hw_digest: 1'b0, + write_lock: 1'b0, + read_lock: 1'b0, + integrity: 1'b0, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // ROM_PATCH + '{ + variant: Unbuffered, + offset: 14'd6136, + size: 9784, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b1, + hw_digest: 1'b0, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // HW_CFG0 + '{ + variant: Buffered, + offset: 14'd15920, + size: 72, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b0, + hw_digest: 1'b1, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // HW_CFG1 + '{ + variant: Buffered, + offset: 14'd15992, + size: 16, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b0, + hw_digest: 1'b1, + write_lock: 1'b1, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // SECRET0 + '{ + variant: Buffered, + offset: 14'd16008, + size: 40, + key_sel: Secret0Key, + secret: 1'b1, + sw_digest: 1'b0, + hw_digest: 1'b1, + write_lock: 1'b1, + read_lock: 1'b1, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // SECRET1 + '{ + variant: Buffered, + offset: 14'd16048, + size: 88, + key_sel: Secret1Key, + secret: 1'b1, + sw_digest: 1'b0, + hw_digest: 1'b1, + write_lock: 1'b1, + read_lock: 1'b1, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + }, + // SECRET2 + '{ + variant: Buffered, + offset: 14'd16136, + size: 120, + key_sel: Secret2Key, + secret: 1'b1, + sw_digest: 1'b0, + hw_digest: 1'b1, + write_lock: 1'b1, + read_lock: 1'b1, + integrity: 1'b1, + iskeymgr_creator: 1'b1, + iskeymgr_owner: 1'b0 + }, + // SECRET3 + '{ + variant: Buffered, + offset: 14'd16256, + size: 40, + key_sel: Secret3Key, + secret: 1'b1, + sw_digest: 1'b0, + hw_digest: 1'b1, + write_lock: 1'b1, + read_lock: 1'b1, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b1 + }, + // LIFE_CYCLE + '{ + variant: LifeCycle, + offset: 14'd16296, + size: 88, + key_sel: key_sel_e'('0), + secret: 1'b0, + sw_digest: 1'b0, + hw_digest: 1'b0, + write_lock: 1'b0, + read_lock: 1'b0, + integrity: 1'b1, + iskeymgr_creator: 1'b0, + iskeymgr_owner: 1'b0 + } + }; + + typedef enum { + VendorTestIdx, + CreatorSwCfgIdx, + OwnerSwCfgIdx, + OwnershipSlotStateIdx, + RotCreatorAuthIdx, + RotOwnerAuthSlot0Idx, + RotOwnerAuthSlot1Idx, + PlatIntegAuthSlot0Idx, + PlatIntegAuthSlot1Idx, + PlatOwnerAuthSlot0Idx, + PlatOwnerAuthSlot1Idx, + PlatOwnerAuthSlot2Idx, + PlatOwnerAuthSlot3Idx, + ExtNvmIdx, + RomPatchIdx, + HwCfg0Idx, + HwCfg1Idx, + Secret0Idx, + Secret1Idx, + Secret2Idx, + Secret3Idx, + LifeCycleIdx, + // These are not "real partitions", but in terms of implementation it is convenient to + // add these at the end of certain arrays. + DaiIdx, + LciIdx, + KdiIdx, + // Number of agents is the last idx+1. + NumAgentsIdx + } part_idx_e; + + parameter int NumAgents = int'(NumAgentsIdx); + + // Breakout types for easier access of individual items. + typedef struct packed { + logic [63:0] hw_cfg0_digest; + logic [255:0] manuf_state; + logic [255:0] device_id; + } otp_hw_cfg0_data_t; + + // default value used for intermodule + parameter otp_hw_cfg0_data_t OTP_HW_CFG0_DATA_DEFAULT = '{ + hw_cfg0_digest: 64'hB7A0C53617A6A31C, + manuf_state: 256'h40119A3C6E63CDF358840E458E4029A6B5AC1F53D00A08C3B28B5C0FEE5F4C02, + device_id: 256'hB3A5B4421F462370FFF698183664DC7EDF3888886BD10DC67ABB319BDA0529AE + }; + typedef struct packed { + logic [63:0] hw_cfg1_digest; + logic [23:0] unallocated; + prim_mubi_pkg::mubi8_t en_sram_ifetch; + logic [31:0] soc_dbg_state; + } otp_hw_cfg1_data_t; + + // default value used for intermodule + parameter otp_hw_cfg1_data_t OTP_HW_CFG1_DATA_DEFAULT = '{ + hw_cfg1_digest: 64'hB8138A3BBDAAE552, + unallocated: 24'h0, + en_sram_ifetch: prim_mubi_pkg::mubi8_t'(8'h69), + soc_dbg_state: 32'h0 + }; + typedef struct packed { + // This reuses the same encoding as the life cycle signals for indicating valid status. + lc_ctrl_pkg::lc_tx_t valid; + otp_hw_cfg1_data_t hw_cfg1_data; + otp_hw_cfg0_data_t hw_cfg0_data; + } otp_broadcast_t; + + // default value for intermodule + parameter otp_broadcast_t OTP_BROADCAST_DEFAULT = '{ + valid: lc_ctrl_pkg::Off, + hw_cfg1_data: OTP_HW_CFG1_DATA_DEFAULT, + hw_cfg0_data: OTP_HW_CFG0_DATA_DEFAULT + }; + + + // OTP invalid partition default for buffered partitions. + parameter logic [131071:0] PartInvDefault = 131072'({ + 704'({ + 320'hF87BED95CFBA3727BBF4A76885E754F2BE193854E9CA60A0C469C593E5DC0DA88CBBAD02BB4CA928, + 384'hA445C3C29F71A2564947DD361344767A0340A5B93BB19342E29749216775E8A515F164D7930C9D1920440F25BB053FB5 + }), + 320'({ + 64'hDD3C869E21D220A3, + 256'h93B61DE417B9FB339605F051E74379CBCC6596C7174EBA643E725E464F593C87 + }), + 960'({ + 64'h3E01C22789430178, + 256'h1136C663A36C3E3E817E760B27AE937BFCDF15A3429452A851B80674A2B6FBE, + 256'h279FC51CC7C626E315FD2B871D88819A0D1E90E8C9FDDFA01E46311FD36D954, + 256'h5E58D0AA97A0F8F6D3D58610F4851667D68C96F0B3D1FEED688098A43C33459F, + 128'hD0BAC511D08ECE0E2C0DBDDEDF7A854D + }), + 704'({ + 64'h725A4C748BE1317E, + 128'h94CD3DED94B578192A4D8B51F5D41C8A, + 256'h55D0320379A0D260426D99D374E699CAE00E9680BD9B70291C752824C7DDC896, + 256'hDBB9844327F20FB5D396D1CE085BDC31105733EAA3880C5A234729143F97B62A + }), + 320'({ + 64'h50DE28C64D4C187, + 128'hDBC827839FE2DCC27E17D06B5D4E0DDD, + 128'h711D135F59A50322B6711DB6F5D40A37 + }), + 128'({ + 64'hB8138A3BBDAAE552, + 24'h0, // unallocated space + 8'h69, + 32'h0 + }), + 576'({ + 64'hB7A0C53617A6A31C, + 256'h40119A3C6E63CDF358840E458E4029A6B5AC1F53D00A08C3B28B5C0FEE5F4C02, + 256'hB3A5B4421F462370FFF698183664DC7EDF3888886BD10DC67ABB319BDA0529AE + }), + 78272'({ + 64'h4D104B5B0B3D8FDD, + 4672'h0, // unallocated space + 73536'h0 + }), + 8192'({ + 8192'h0 + }), + 2624'({ + 64'h47508BAB4DC75216, + 1280'h0, + 1280'h0 + }), + 2624'({ + 64'h563C0C2920F6372, + 1280'h0, + 1280'h0 + }), + 2624'({ + 64'h644C4723CF740F6A, + 1280'h0, + 1280'h0 + }), + 2624'({ + 64'hBF1F41B783B6DB8C, + 1280'h0, + 1280'h0 + }), + 2624'({ + 64'hB8DE43EDFF17AA86, + 1280'h0, + 1280'h0 + }), + 2624'({ + 64'h60BAE4A876D70627, + 1280'h0, + 1280'h0 + }), + 2624'({ + 64'h2DCDD92FA5B24BF3, + 1280'h0, + 1280'h0 + }), + 2624'({ + 64'h3BF7D79A9FF747F6, + 1280'h0, + 1280'h0 + }), + 11392'({ + 64'h41837480464544A1, + 32'h0, // unallocated space + 6144'h0, + 1280'h0, + 1280'h0, + 1280'h0, + 32'h0, + 1280'h0 + }), + 384'({ + 128'h0, + 128'h0, + 128'h0 + }), + 5056'({ + 64'h6FDFE93D3146B0F, + 128'h0, // unallocated space + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 512'h0, + 128'h0, + 128'h0, + 512'h0, + 3200'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0 + }), + 2560'({ + 64'h67BBE3B4555DF35C, + 160'h0, // unallocated space + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 32'h0, + 64'h0, + 32'h0, + 64'h0, + 32'h0, + 32'h0, + 256'h0, + 32'h0, + 992'h0 + }), + 512'({ + 64'hAA3F4C71234F097C, + 448'h0 + })}); + + /////////////////////////////////////////////// + // Parameterized Assignment Helper Functions // + /////////////////////////////////////////////// + + function automatic otp_ctrl_core_hw2reg_t named_reg_assign( + logic [NumPart-1:0][ScrmblBlockWidth-1:0] part_digest); + otp_ctrl_core_hw2reg_t hw2reg; + logic unused_sigs; + unused_sigs = ^part_digest; + hw2reg = '0; + hw2reg.vendor_test_digest = part_digest[VendorTestIdx]; + hw2reg.creator_sw_cfg_digest = part_digest[CreatorSwCfgIdx]; + hw2reg.owner_sw_cfg_digest = part_digest[OwnerSwCfgIdx]; + hw2reg.rot_creator_auth_digest = part_digest[RotCreatorAuthIdx]; + hw2reg.rot_owner_auth_slot0_digest = part_digest[RotOwnerAuthSlot0Idx]; + hw2reg.rot_owner_auth_slot1_digest = part_digest[RotOwnerAuthSlot1Idx]; + hw2reg.plat_integ_auth_slot0_digest = part_digest[PlatIntegAuthSlot0Idx]; + hw2reg.plat_integ_auth_slot1_digest = part_digest[PlatIntegAuthSlot1Idx]; + hw2reg.plat_owner_auth_slot0_digest = part_digest[PlatOwnerAuthSlot0Idx]; + hw2reg.plat_owner_auth_slot1_digest = part_digest[PlatOwnerAuthSlot1Idx]; + hw2reg.plat_owner_auth_slot2_digest = part_digest[PlatOwnerAuthSlot2Idx]; + hw2reg.plat_owner_auth_slot3_digest = part_digest[PlatOwnerAuthSlot3Idx]; + hw2reg.rom_patch_digest = part_digest[RomPatchIdx]; + hw2reg.hw_cfg0_digest = part_digest[HwCfg0Idx]; + hw2reg.hw_cfg1_digest = part_digest[HwCfg1Idx]; + hw2reg.secret0_digest = part_digest[Secret0Idx]; + hw2reg.secret1_digest = part_digest[Secret1Idx]; + hw2reg.secret2_digest = part_digest[Secret2Idx]; + hw2reg.secret3_digest = part_digest[Secret3Idx]; + return hw2reg; + endfunction : named_reg_assign + + function automatic part_access_t [NumPart-1:0] named_part_access_pre( + otp_ctrl_core_reg2hw_t reg2hw); + part_access_t [NumPart-1:0] part_access_pre; + logic unused_sigs; + unused_sigs = ^reg2hw; + // Default (this will be overridden by partition-internal settings). + part_access_pre = {{32'(2*NumPart)}{prim_mubi_pkg::MuBi8False}}; + // Note: these could be made a MuBi CSRs in the future. + // The main thing that is missing right now is proper support for W0C. + // VENDOR_TEST + if (!reg2hw.vendor_test_read_lock) begin + part_access_pre[VendorTestIdx].read_lock = prim_mubi_pkg::MuBi8True; + end + // CREATOR_SW_CFG + if (!reg2hw.creator_sw_cfg_read_lock) begin + part_access_pre[CreatorSwCfgIdx].read_lock = prim_mubi_pkg::MuBi8True; + end + // OWNER_SW_CFG + if (!reg2hw.owner_sw_cfg_read_lock) begin + part_access_pre[OwnerSwCfgIdx].read_lock = prim_mubi_pkg::MuBi8True; + end + // OWNERSHIP_SLOT_STATE + if (!reg2hw.ownership_slot_state_read_lock) begin + part_access_pre[OwnershipSlotStateIdx].read_lock = prim_mubi_pkg::MuBi8True; + end + // ROT_CREATOR_AUTH + if (!reg2hw.rot_creator_auth_read_lock) begin + part_access_pre[RotCreatorAuthIdx].read_lock = prim_mubi_pkg::MuBi8True; + end + // ROT_OWNER_AUTH_SLOT0 + if (!reg2hw.rot_owner_auth_slot0_read_lock) begin + part_access_pre[RotOwnerAuthSlot0Idx].read_lock = prim_mubi_pkg::MuBi8True; + end + // ROT_OWNER_AUTH_SLOT1 + if (!reg2hw.rot_owner_auth_slot1_read_lock) begin + part_access_pre[RotOwnerAuthSlot1Idx].read_lock = prim_mubi_pkg::MuBi8True; + end + // PLAT_INTEG_AUTH_SLOT0 + if (!reg2hw.plat_integ_auth_slot0_read_lock) begin + part_access_pre[PlatIntegAuthSlot0Idx].read_lock = prim_mubi_pkg::MuBi8True; + end + // PLAT_INTEG_AUTH_SLOT1 + if (!reg2hw.plat_integ_auth_slot1_read_lock) begin + part_access_pre[PlatIntegAuthSlot1Idx].read_lock = prim_mubi_pkg::MuBi8True; + end + // PLAT_OWNER_AUTH_SLOT0 + if (!reg2hw.plat_owner_auth_slot0_read_lock) begin + part_access_pre[PlatOwnerAuthSlot0Idx].read_lock = prim_mubi_pkg::MuBi8True; + end + // PLAT_OWNER_AUTH_SLOT1 + if (!reg2hw.plat_owner_auth_slot1_read_lock) begin + part_access_pre[PlatOwnerAuthSlot1Idx].read_lock = prim_mubi_pkg::MuBi8True; + end + // PLAT_OWNER_AUTH_SLOT2 + if (!reg2hw.plat_owner_auth_slot2_read_lock) begin + part_access_pre[PlatOwnerAuthSlot2Idx].read_lock = prim_mubi_pkg::MuBi8True; + end + // PLAT_OWNER_AUTH_SLOT3 + if (!reg2hw.plat_owner_auth_slot3_read_lock) begin + part_access_pre[PlatOwnerAuthSlot3Idx].read_lock = prim_mubi_pkg::MuBi8True; + end + // EXT_NVM + if (!reg2hw.ext_nvm_read_lock) begin + part_access_pre[ExtNvmIdx].read_lock = prim_mubi_pkg::MuBi8True; + end + // ROM_PATCH + if (!reg2hw.rom_patch_read_lock) begin + part_access_pre[RomPatchIdx].read_lock = prim_mubi_pkg::MuBi8True; + end + return part_access_pre; + endfunction : named_part_access_pre + + function automatic otp_broadcast_t named_broadcast_assign( + logic [NumPart-1:0] part_init_done, + logic [$bits(PartInvDefault)/8-1:0][7:0] part_buf_data); + otp_broadcast_t otp_broadcast; + logic valid, unused; + unused = 1'b0; + valid = 1'b1; + // VENDOR_TEST + unused ^= ^{part_init_done[VendorTestIdx], + part_buf_data[VendorTestOffset +: VendorTestSize]}; + // CREATOR_SW_CFG + unused ^= ^{part_init_done[CreatorSwCfgIdx], + part_buf_data[CreatorSwCfgOffset +: CreatorSwCfgSize]}; + // OWNER_SW_CFG + unused ^= ^{part_init_done[OwnerSwCfgIdx], + part_buf_data[OwnerSwCfgOffset +: OwnerSwCfgSize]}; + // OWNERSHIP_SLOT_STATE + unused ^= ^{part_init_done[OwnershipSlotStateIdx], + part_buf_data[OwnershipSlotStateOffset +: OwnershipSlotStateSize]}; + // ROT_CREATOR_AUTH + unused ^= ^{part_init_done[RotCreatorAuthIdx], + part_buf_data[RotCreatorAuthOffset +: RotCreatorAuthSize]}; + // ROT_OWNER_AUTH_SLOT0 + unused ^= ^{part_init_done[RotOwnerAuthSlot0Idx], + part_buf_data[RotOwnerAuthSlot0Offset +: RotOwnerAuthSlot0Size]}; + // ROT_OWNER_AUTH_SLOT1 + unused ^= ^{part_init_done[RotOwnerAuthSlot1Idx], + part_buf_data[RotOwnerAuthSlot1Offset +: RotOwnerAuthSlot1Size]}; + // PLAT_INTEG_AUTH_SLOT0 + unused ^= ^{part_init_done[PlatIntegAuthSlot0Idx], + part_buf_data[PlatIntegAuthSlot0Offset +: PlatIntegAuthSlot0Size]}; + // PLAT_INTEG_AUTH_SLOT1 + unused ^= ^{part_init_done[PlatIntegAuthSlot1Idx], + part_buf_data[PlatIntegAuthSlot1Offset +: PlatIntegAuthSlot1Size]}; + // PLAT_OWNER_AUTH_SLOT0 + unused ^= ^{part_init_done[PlatOwnerAuthSlot0Idx], + part_buf_data[PlatOwnerAuthSlot0Offset +: PlatOwnerAuthSlot0Size]}; + // PLAT_OWNER_AUTH_SLOT1 + unused ^= ^{part_init_done[PlatOwnerAuthSlot1Idx], + part_buf_data[PlatOwnerAuthSlot1Offset +: PlatOwnerAuthSlot1Size]}; + // PLAT_OWNER_AUTH_SLOT2 + unused ^= ^{part_init_done[PlatOwnerAuthSlot2Idx], + part_buf_data[PlatOwnerAuthSlot2Offset +: PlatOwnerAuthSlot2Size]}; + // PLAT_OWNER_AUTH_SLOT3 + unused ^= ^{part_init_done[PlatOwnerAuthSlot3Idx], + part_buf_data[PlatOwnerAuthSlot3Offset +: PlatOwnerAuthSlot3Size]}; + // EXT_NVM + unused ^= ^{part_init_done[ExtNvmIdx], + part_buf_data[ExtNvmOffset +: ExtNvmSize]}; + // ROM_PATCH + unused ^= ^{part_init_done[RomPatchIdx], + part_buf_data[RomPatchOffset +: RomPatchSize]}; + // HW_CFG0 + valid &= part_init_done[HwCfg0Idx]; + otp_broadcast.hw_cfg0_data = otp_hw_cfg0_data_t'(part_buf_data[HwCfg0Offset +: HwCfg0Size]); + // HW_CFG1 + valid &= part_init_done[HwCfg1Idx]; + otp_broadcast.hw_cfg1_data = otp_hw_cfg1_data_t'(part_buf_data[HwCfg1Offset +: HwCfg1Size]); + // SECRET0 + unused ^= ^{part_init_done[Secret0Idx], + part_buf_data[Secret0Offset +: Secret0Size]}; + // SECRET1 + unused ^= ^{part_init_done[Secret1Idx], + part_buf_data[Secret1Offset +: Secret1Size]}; + // SECRET2 + unused ^= ^{part_init_done[Secret2Idx], + part_buf_data[Secret2Offset +: Secret2Size]}; + // SECRET3 + unused ^= ^{part_init_done[Secret3Idx], + part_buf_data[Secret3Offset +: Secret3Size]}; + // LIFE_CYCLE + unused ^= ^{part_init_done[LifeCycleIdx], + part_buf_data[LifeCycleOffset +: LifeCycleSize]}; + otp_broadcast.valid = lc_ctrl_pkg::lc_tx_bool_to_lc_tx(valid); + return otp_broadcast; + endfunction : named_broadcast_assign + + function automatic otp_keymgr_key_t named_keymgr_key_assign( + logic [NumPart-1:0][ScrmblBlockWidth-1:0] part_digest, + logic [$bits(PartInvDefault)/8-1:0][7:0] part_buf_data, + lc_ctrl_pkg::lc_tx_t lc_seed_hw_rd_en); + otp_keymgr_key_t otp_keymgr_key; + logic valid, unused; + unused = 1'b0; + // For now we use a fixed struct type here so that the + // interface to the keymgr remains stable. The type contains + // a superset of all options, so we have to initialize it to '0 here. + otp_keymgr_key = '0; + // VENDOR_TEST + unused ^= ^{part_digest[VendorTestIdx], + part_buf_data[VendorTestOffset +: VendorTestSize]}; + // CREATOR_SW_CFG + unused ^= ^{part_digest[CreatorSwCfgIdx], + part_buf_data[CreatorSwCfgOffset +: CreatorSwCfgSize]}; + // OWNER_SW_CFG + unused ^= ^{part_digest[OwnerSwCfgIdx], + part_buf_data[OwnerSwCfgOffset +: OwnerSwCfgSize]}; + // OWNERSHIP_SLOT_STATE + unused ^= ^{part_digest[OwnershipSlotStateIdx], + part_buf_data[OwnershipSlotStateOffset +: OwnershipSlotStateSize]}; + // ROT_CREATOR_AUTH + unused ^= ^{part_digest[RotCreatorAuthIdx], + part_buf_data[RotCreatorAuthOffset +: RotCreatorAuthSize]}; + // ROT_OWNER_AUTH_SLOT0 + unused ^= ^{part_digest[RotOwnerAuthSlot0Idx], + part_buf_data[RotOwnerAuthSlot0Offset +: RotOwnerAuthSlot0Size]}; + // ROT_OWNER_AUTH_SLOT1 + unused ^= ^{part_digest[RotOwnerAuthSlot1Idx], + part_buf_data[RotOwnerAuthSlot1Offset +: RotOwnerAuthSlot1Size]}; + // PLAT_INTEG_AUTH_SLOT0 + unused ^= ^{part_digest[PlatIntegAuthSlot0Idx], + part_buf_data[PlatIntegAuthSlot0Offset +: PlatIntegAuthSlot0Size]}; + // PLAT_INTEG_AUTH_SLOT1 + unused ^= ^{part_digest[PlatIntegAuthSlot1Idx], + part_buf_data[PlatIntegAuthSlot1Offset +: PlatIntegAuthSlot1Size]}; + // PLAT_OWNER_AUTH_SLOT0 + unused ^= ^{part_digest[PlatOwnerAuthSlot0Idx], + part_buf_data[PlatOwnerAuthSlot0Offset +: PlatOwnerAuthSlot0Size]}; + // PLAT_OWNER_AUTH_SLOT1 + unused ^= ^{part_digest[PlatOwnerAuthSlot1Idx], + part_buf_data[PlatOwnerAuthSlot1Offset +: PlatOwnerAuthSlot1Size]}; + // PLAT_OWNER_AUTH_SLOT2 + unused ^= ^{part_digest[PlatOwnerAuthSlot2Idx], + part_buf_data[PlatOwnerAuthSlot2Offset +: PlatOwnerAuthSlot2Size]}; + // PLAT_OWNER_AUTH_SLOT3 + unused ^= ^{part_digest[PlatOwnerAuthSlot3Idx], + part_buf_data[PlatOwnerAuthSlot3Offset +: PlatOwnerAuthSlot3Size]}; + // EXT_NVM + unused ^= ^{part_digest[ExtNvmIdx], + part_buf_data[ExtNvmOffset +: ExtNvmSize]}; + // ROM_PATCH + unused ^= ^{part_digest[RomPatchIdx], + part_buf_data[RomPatchOffset +: RomPatchSize]}; + // HW_CFG0 + unused ^= ^{part_digest[HwCfg0Idx], + part_buf_data[HwCfg0Offset +: HwCfg0Size]}; + // HW_CFG1 + unused ^= ^{part_digest[HwCfg1Idx], + part_buf_data[HwCfg1Offset +: HwCfg1Size]}; + // SECRET0 + unused ^= ^{part_digest[Secret0Idx], + part_buf_data[Secret0Offset +: Secret0Size]}; + // SECRET1 + unused ^= ^{part_digest[Secret1Idx], + part_buf_data[Secret1Offset +: Secret1Size]}; + // SECRET2 + valid = (part_digest[Secret2Idx] != 0); + unused ^= ^part_buf_data[RmaTokenOffset +: RmaTokenSize]; + otp_keymgr_key.creator_root_key_share0_valid = valid; + if (lc_ctrl_pkg::lc_tx_test_true_strict(lc_seed_hw_rd_en)) begin + otp_keymgr_key.creator_root_key_share0 = + part_buf_data[CreatorRootKeyShare0Offset +: CreatorRootKeyShare0Size]; + end else begin + otp_keymgr_key.creator_root_key_share0 = + PartInvDefault[CreatorRootKeyShare0Offset*8 +: CreatorRootKeyShare0Size*8]; + end + otp_keymgr_key.creator_root_key_share1_valid = valid; + if (lc_ctrl_pkg::lc_tx_test_true_strict(lc_seed_hw_rd_en)) begin + otp_keymgr_key.creator_root_key_share1 = + part_buf_data[CreatorRootKeyShare1Offset +: CreatorRootKeyShare1Size]; + end else begin + otp_keymgr_key.creator_root_key_share1 = + PartInvDefault[CreatorRootKeyShare1Offset*8 +: CreatorRootKeyShare1Size*8]; + end + otp_keymgr_key.creator_seed_valid = valid; + if (lc_ctrl_pkg::lc_tx_test_true_strict(lc_seed_hw_rd_en)) begin + otp_keymgr_key.creator_seed = + part_buf_data[CreatorSeedOffset +: CreatorSeedSize]; + end else begin + otp_keymgr_key.creator_seed = + PartInvDefault[CreatorSeedOffset*8 +: CreatorSeedSize*8]; + end + // This is not used since we consume the + // ungated digest values from the part_digest array. + unused ^= ^part_buf_data[Secret2DigestOffset +: Secret2DigestSize]; + // SECRET3 + valid = (part_digest[Secret3Idx] != 0); + otp_keymgr_key.owner_seed_valid = valid; + if (lc_ctrl_pkg::lc_tx_test_true_strict(lc_seed_hw_rd_en)) begin + otp_keymgr_key.owner_seed = + part_buf_data[OwnerSeedOffset +: OwnerSeedSize]; + end else begin + otp_keymgr_key.owner_seed = + PartInvDefault[OwnerSeedOffset*8 +: OwnerSeedSize*8]; + end + // This is not used since we consume the + // ungated digest values from the part_digest array. + unused ^= ^part_buf_data[Secret3DigestOffset +: Secret3DigestSize]; + // LIFE_CYCLE + unused ^= ^{part_digest[LifeCycleIdx], + part_buf_data[LifeCycleOffset +: LifeCycleSize]}; + unused ^= valid; + return otp_keymgr_key; + endfunction : named_keymgr_key_assign + +endpackage : otp_ctrl_part_pkg diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv new file mode 100644 index 00000000000000..09eb0fe0d73fa5 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv @@ -0,0 +1,532 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Unbuffered partition for OTP controller. +// + +`include "prim_flop_macros.sv" + +module otp_ctrl_part_unbuf + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; +#( + // Partition information. + parameter part_info_t Info = PartInfoDefault +) ( + input clk_i, + input rst_ni, + // Pulse to start partition initialisation (required once per power cycle). + input init_req_i, + output logic init_done_o, + // Escalation input. This moves the FSM into a terminal state and locks down + // the partition. + input lc_ctrl_pkg::lc_tx_t escalate_en_i, + // Output error state of partition, to be consumed by OTP error/alert logic. + // Note that most errors are not recoverable and move the partition FSM into + // a terminal error state. + output otp_err_e error_o, + // This error signal is pulsed high if the FSM has been glitched into an invalid state. + // Although it is somewhat redundant with the error code in error_o above, it is + // meant to cover cases where we already latched an error code while the FSM is + // glitched into an invalid state (since in that case, the error code will not be + // overridden with the FSM error code so that the original error code is still + // discoverable). + output logic fsm_err_o, + // Access/lock status + // SEC_CM: ACCESS.CTRL.MUBI + input part_access_t access_i, // runtime lock from CSRs + output part_access_t access_o, + // Buffered 64bit digest output. + output logic [ScrmblBlockWidth-1:0] digest_o, + // Interface to TL-UL adapter + input logic tlul_req_i, + output logic tlul_gnt_o, + input [SwWindowAddrWidth-1:0] tlul_addr_i, + output logic [1:0] tlul_rerror_o, + output logic tlul_rvalid_o, + output logic [31:0] tlul_rdata_o, + // OTP interface + output logic otp_req_o, + output prim_otp_pkg::cmd_e otp_cmd_o, + output logic [OtpSizeWidth-1:0] otp_size_o, + output logic [OtpIfWidth-1:0] otp_wdata_o, + output logic [OtpAddrWidth-1:0] otp_addr_o, + input otp_gnt_i, + input otp_rvalid_i, + input [ScrmblBlockWidth-1:0] otp_rdata_i, + input prim_otp_pkg::err_e otp_err_i +); + + //////////////////////// + // Integration Checks // + //////////////////////// + + import prim_mubi_pkg::*; + import prim_util_pkg::vbits; + + localparam logic [OtpByteAddrWidth:0] PartEnd = (OtpByteAddrWidth+1)'(Info.offset) + + (OtpByteAddrWidth+1)'(Info.size); + localparam int unsigned DigestOffsetInt = int'(PartEnd) - ScrmblBlockWidth/8; + + localparam bit [OtpByteAddrWidth-1:0] DigestOffset = DigestOffsetInt[OtpByteAddrWidth-1:0]; + + // Integration checks for parameters. + `ASSERT_INIT(OffsetMustBeBlockAligned_A, (Info.offset % (ScrmblBlockWidth/8)) == 0) + `ASSERT_INIT(SizeMustBeBlockAligned_A, (Info.size % (ScrmblBlockWidth/8)) == 0) + `ASSERT_INIT(DigestOffsetMustBeRepresentable_A, DigestOffsetInt == int'(DigestOffset)) + + /////////////////////// + // OTP Partition FSM // + /////////////////////// + + // SEC_CM: PART.FSM.SPARSE + // Encoding generated with: + // $ ./util/design/sparse-fsm-encode.py -d 5 -m 7 -n 10 \ + // -s 4247417884 --language=sv + // + // Hamming distance histogram: + // + // 0: -- + // 1: -- + // 2: -- + // 3: -- + // 4: -- + // 5: |||||||||||||||||||| (52.38%) + // 6: |||||||||||| (33.33%) + // 7: | (4.76%) + // 8: ||| (9.52%) + // 9: -- + // 10: -- + // + // Minimum Hamming distance: 5 + // Maximum Hamming distance: 8 + // Minimum Hamming weight: 3 + // Maximum Hamming weight: 9 + // + localparam int StateWidth = 10; + typedef enum logic [StateWidth-1:0] { + ResetSt = 10'b1010110110, + InitSt = 10'b0100010011, + InitWaitSt = 10'b0001011000, + IdleSt = 10'b1011101001, + ReadSt = 10'b0101101110, + ReadWaitSt = 10'b0110100101, + ErrorSt = 10'b1111011111 + } state_e; + + typedef enum logic { + DigestAddrSel = 1'b0, + DataAddrSel = 1'b1 + } addr_sel_e; + + state_e state_d, state_q; + addr_sel_e otp_addr_sel; + otp_err_e error_d, error_q; + + logic digest_reg_en; + logic ecc_err; + + logic tlul_addr_in_range; + logic [SwWindowAddrWidth-1:0] tlul_addr_d, tlul_addr_q; + + // This is only used to return bus errors when the FSM is in ErrorSt. + logic pending_tlul_error_d, pending_tlul_error_q; + + // Output partition error state. + assign error_o = error_q; + + // This partition cannot do any write accesses, hence we tie this + // constantly off. + assign otp_wdata_o = '0; + // Depending on the partition configuration, the wrapper is instructed to ignore integrity + // calculations and checks. To be on the safe side, the partition filters error responses at this + // point and does not report any integrity errors if integrity is disabled. + otp_err_e otp_err; + if (Info.integrity) begin : gen_integrity + assign otp_cmd_o = prim_otp_pkg::Read; + assign otp_err = otp_err_e'(otp_err_i); + end else begin : gen_no_integrity + assign otp_cmd_o = prim_otp_pkg::ReadRaw; + always_comb begin + if (otp_err_e'(otp_err_i) inside {MacroEccCorrError, MacroEccUncorrError}) begin + otp_err = NoError; + end else begin + otp_err = otp_err_e'(otp_err_i); + end + end + end + + `ASSERT_KNOWN(FsmStateKnown_A, state_q) + always_comb begin : p_fsm + // Default assignments + state_d = state_q; + + // Response to init request + init_done_o = 1'b0; + + // OTP signals + otp_req_o = 1'b0; + otp_addr_sel = DigestAddrSel; + + // TL-UL signals + tlul_gnt_o = 1'b0; + tlul_rvalid_o = 1'b0; + tlul_rerror_o = '0; + + // Enable for buffered digest register + digest_reg_en = 1'b0; + + // Error Register + error_d = error_q; + pending_tlul_error_d = 1'b0; + fsm_err_o = 1'b0; + + unique case (state_q) + /////////////////////////////////////////////////////////////////// + // State right after reset. Wait here until we get a an + // initialization request. + ResetSt: begin + if (init_req_i) begin + // If the partition does not have a digest, no initialization is necessary. + if (Info.sw_digest) begin + state_d = InitSt; + end else begin + state_d = IdleSt; + end + end + end + /////////////////////////////////////////////////////////////////// + // Initialization reads out the digest only in unbuffered + // partitions. Wait here until the OTP request has been granted. + // And then wait until the OTP word comes back. + InitSt: begin + otp_req_o = 1'b1; + if (otp_gnt_i) begin + state_d = InitWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response and write to digest buffer register. In + // case an OTP transaction fails, latch the OTP error code and + // jump to a terminal error state. + InitWaitSt: begin + if (otp_rvalid_i) begin + digest_reg_en = 1'b1; + if (otp_err inside {NoError, MacroEccCorrError}) begin + state_d = IdleSt; + // At this point the only error that we could have gotten are correctable ECC errors. + if (otp_err != NoError) begin + error_d = MacroEccCorrError; + end + end else begin + state_d = ErrorSt; + error_d = otp_err; + end + end + end + /////////////////////////////////////////////////////////////////// + // Wait for TL-UL requests coming in. + // Then latch address and go to readout state. + IdleSt: begin + init_done_o = 1'b1; + if (tlul_req_i) begin + error_d = NoError; // clear recoverable soft errors. + state_d = ReadSt; + tlul_gnt_o = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // If the address is out of bounds, or if the partition is + // locked, signal back a bus error. Note that such an error does + // not cause the partition to go into error state. Otherwise if + // these checks pass, an OTP word is requested. + ReadSt: begin + init_done_o = 1'b1; + // Double check the address range. + if (tlul_addr_in_range && mubi8_test_false_strict(access_o.read_lock)) begin + otp_req_o = 1'b1; + otp_addr_sel = DataAddrSel; + if (otp_gnt_i) begin + state_d = ReadWaitSt; + end + end else begin + state_d = IdleSt; + error_d = AccessError; // Signal this error, but do not go into terminal error state. + tlul_rvalid_o = 1'b1; + tlul_rerror_o = 2'b11; // This causes the TL-UL adapter to return a bus error. + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response and release the TL-UL response. In + // case an OTP transaction fails, latch the OTP error code, + // signal a TL-Ul bus error and jump to a terminal error state. + ReadWaitSt: begin + init_done_o = 1'b1; + if (otp_rvalid_i) begin + tlul_rvalid_o = 1'b1; + if (otp_err inside {NoError, MacroEccCorrError}) begin + state_d = IdleSt; + // At this point the only error that we could have gotten are correctable ECC errors. + if (otp_err != NoError) begin + error_d = MacroEccCorrError; + end + end else begin + state_d = ErrorSt; + error_d = otp_err; + // This causes the TL-UL adapter to return a bus error. + tlul_rerror_o = 2'b11; + end + end + end + /////////////////////////////////////////////////////////////////// + // Terminal Error State. This locks access to the partition. + // Make sure the partition signals an error state if no error + // code has been latched so far. + ErrorSt: begin + if (error_q == NoError) begin + error_d = FsmStateError; + end + + // Return bus errors if there are pending TL-UL requests. + if (pending_tlul_error_q) begin + tlul_rerror_o = 2'b11; + tlul_rvalid_o = 1'b1; + end else if (tlul_req_i) begin + tlul_gnt_o = 1'b1; + pending_tlul_error_d = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // We should never get here. If we do (e.g. via a malicious + // glitch), error out immediately. + default: begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + endcase // state_q + + // Unconditionally jump into the terminal error state in case of + // an ECC error or escalation, and lock access to the partition down. + // SEC_CM: PART.FSM.LOCAL_ESC + if (ecc_err) begin + state_d = ErrorSt; + if (state_q != ErrorSt) begin + error_d = CheckFailError; + end + end + // SEC_CM: PART.FSM.GLOBAL_ESC + if (lc_ctrl_pkg::lc_tx_test_true_loose(escalate_en_i)) begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + if (state_q != ErrorSt) begin + error_d = FsmStateError; + end + end + end + + /////////////////////////////////// + // Signals to/from TL-UL Adapter // + /////////////////////////////////// + + assign tlul_addr_d = tlul_addr_i; + // Do not forward data in case of an error. + assign tlul_rdata_o = (tlul_rvalid_o && tlul_rerror_o == '0) ? otp_rdata_i[31:0] : '0; + + if (Info.offset == 0) begin : gen_zero_offset + assign tlul_addr_in_range = {1'b0, tlul_addr_q, 2'b00} < PartEnd; + + end else begin : gen_nonzero_offset + assign tlul_addr_in_range = {tlul_addr_q, 2'b00} >= Info.offset && + {1'b0, tlul_addr_q, 2'b00} < PartEnd; + end + + // Note that OTP works on halfword (16bit) addresses, hence need to + // shift the addresses appropriately. + logic [OtpByteAddrWidth-1:0] addr_calc; + assign addr_calc = (otp_addr_sel == DigestAddrSel) ? DigestOffset : {tlul_addr_q, 2'b00}; + assign otp_addr_o = addr_calc[OtpByteAddrWidth-1:OtpAddrShift]; + + if (OtpAddrShift > 0) begin : gen_unused + logic unused_bits; + assign unused_bits = ^addr_calc[OtpAddrShift-1:0]; + end + + // Request 32bit except in case of the digest. + assign otp_size_o = (otp_addr_sel == DigestAddrSel) ? + OtpSizeWidth'(unsigned'(ScrmblBlockWidth / OtpWidth - 1)) : + OtpSizeWidth'(unsigned'(32 / OtpWidth - 1)); + + //////////////// + // Digest Reg // + //////////////// + + if (Info.sw_digest) begin : gen_ecc_reg + // SEC_CM: PART.DATA_REG.INTEGRITY + otp_ctrl_ecc_reg #( + .Width ( ScrmblBlockWidth ), + .Depth ( 1 ) + ) u_otp_ctrl_ecc_reg ( + .clk_i, + .rst_ni, + .wren_i ( digest_reg_en ), + .addr_i ( '0 ), + .wdata_i ( otp_rdata_i ), + .rdata_o ( ), + .data_o ( digest_o ), + .ecc_err_o ( ecc_err ) + ); + end else begin : gen_no_ecc_reg + logic unused_digest_reg_en; + logic unused_rdata; + assign unused_digest_reg_en = digest_reg_en; + assign unused_rdata = ^otp_rdata_i[32 +: 32]; // Upper word is not connected in this case. + assign digest_o = '0; + assign ecc_err = 1'b0; + end + + //////////////////////// + // DAI Access Control // + //////////////////////// + + mubi8_t init_locked; + assign init_locked = (~init_done_o) ? MuBi8True : MuBi8False; + + // Aggregate all possible DAI write locks. The partition is also locked when uninitialized. + // Note that the locks are redundantly encoded values. + part_access_t access_pre; + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_write_lock_pre ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(init_locked, access_i.write_lock)), + .mubi_o(access_pre.write_lock) + ); + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_read_lock_pre ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(init_locked, access_i.read_lock)), + .mubi_o(access_pre.read_lock) + ); + + // SEC_CM: PART.MEM.SW_UNWRITABLE + if (Info.write_lock) begin : gen_digest_write_lock + mubi8_t digest_locked; + assign digest_locked = (digest_o != '0) ? MuBi8True : MuBi8False; + + // This prevents the synthesis tool from optimizing the multibit signal. + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_write_lock ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(access_pre.write_lock, digest_locked)), + .mubi_o(access_o.write_lock) + ); + + `ASSERT(DigestWriteLocksPartition_A, digest_o |-> mubi8_test_true_loose(access_o.write_lock)) + end else begin : gen_no_digest_write_lock + assign access_o.write_lock = access_pre.write_lock; + end + + // SEC_CM: PART.MEM.SW_UNREADABLE + if (Info.read_lock) begin : gen_digest_read_lock + mubi8_t digest_locked; + assign digest_locked = (digest_o != '0) ? MuBi8True : MuBi8False; + + // This prevents the synthesis tool from optimizing the multibit signal. + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_read_lock ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(access_pre.read_lock, digest_locked)), + .mubi_o(access_o.read_lock) + ); + + `ASSERT(DigestReadLocksPartition_A, digest_o |-> mubi8_test_true_loose(access_o.read_lock)) + end else begin : gen_no_digest_read_lock + assign access_o.read_lock = access_pre.read_lock; + end + + /////////////// + // Registers // + /////////////// + + `PRIM_FLOP_SPARSE_FSM(u_state_regs, state_d, state_q, state_e, ResetSt) + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + error_q <= NoError; + tlul_addr_q <= '0; + pending_tlul_error_q <= 1'b0; + end else begin + error_q <= error_d; + pending_tlul_error_q <= pending_tlul_error_d; + if (tlul_gnt_o) begin + tlul_addr_q <= tlul_addr_d; + end + end + end + + //////////////// + // Assertions // + //////////////// + + // Known assertions + `ASSERT_KNOWN(InitDoneKnown_A, init_done_o) + `ASSERT_KNOWN(ErrorKnown_A, error_o) + `ASSERT_KNOWN(AccessKnown_A, access_o) + `ASSERT_KNOWN(DigestKnown_A, digest_o) + `ASSERT_KNOWN(TlulGntKnown_A, tlul_gnt_o) + `ASSERT_KNOWN(TlulRerrorKnown_A, tlul_rerror_o) + `ASSERT_KNOWN(TlulRvalidKnown_A, tlul_rvalid_o) + `ASSERT_KNOWN(TlulRdataKnown_A, tlul_rdata_o) + `ASSERT_KNOWN(OtpReqKnown_A, otp_req_o) + `ASSERT_KNOWN(OtpCmdKnown_A, otp_cmd_o) + `ASSERT_KNOWN(OtpSizeKnown_A, otp_size_o) + `ASSERT_KNOWN(OtpWdataKnown_A, otp_wdata_o) + `ASSERT_KNOWN(OtpAddrKnown_A, otp_addr_o) + + // Uninitialized partitions should always be locked, no matter what. + `ASSERT(InitWriteLocksPartition_A, + ~init_done_o + |-> + mubi8_test_true_loose(access_o.write_lock)) + `ASSERT(InitReadLocksPartition_A, + ~init_done_o + |-> + mubi8_test_true_loose(access_o.read_lock)) + // Incoming Lock propagation + `ASSERT(WriteLockPropagation_A, + mubi8_test_true_loose(access_i.write_lock) + |-> + mubi8_test_true_loose(access_o.write_lock)) + `ASSERT(ReadLockPropagation_A, + mubi8_test_true_loose(access_i.read_lock) + |-> + mubi8_test_true_loose(access_o.read_lock)) + // If the partition is read locked, the TL-UL access must error out + `ASSERT(TlulReadOnReadLock_A, + tlul_req_i && tlul_gnt_o ##1 mubi8_test_true_loose(access_o.read_lock) + |-> + tlul_rerror_o > '0 && tlul_rvalid_o) + // ECC error in buffer regs. + `ASSERT(EccErrorState_A, + ecc_err + |=> + state_q == ErrorSt) + // OTP error response + `ASSERT(OtpErrorState_A, + state_q inside {InitWaitSt, ReadWaitSt} && otp_rvalid_i && + !(otp_err inside {NoError, MacroEccCorrError}) && !ecc_err + |=> + state_q == ErrorSt && error_o == $past(otp_err)) + +endmodule : otp_ctrl_part_unbuf diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_pkg.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_pkg.sv new file mode 100644 index 00000000000000..0450b103b07741 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_pkg.sv @@ -0,0 +1,232 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// This package can be imported by generic IPs: +// - It does not import otp_ctrl_reg_pkg, which is generated and top-specific. + +package otp_ctrl_pkg; + + //////////////////////// + // General Parameters // + //////////////////////// + + // Number of vendor-specific test CSR bits coming from and going to + // the life cycle TAP registers. + parameter int OtpTestCtrlWidth = 32; + parameter int OtpTestStatusWidth = 32; + parameter int OtpTestVectWidth = 8; + + parameter int DeviceIdWidth = 256; + typedef logic [DeviceIdWidth-1:0] otp_device_id_t; + + parameter int ManufStateWidth = 256; + typedef logic [ManufStateWidth-1:0] otp_manuf_state_t; + + // OTP-macro specific + parameter int OtpPwrSeqWidth = 2; + + ///////////////////////////////// + // Typedefs for OTP Scrambling // + ///////////////////////////////// + + parameter int ScrmblKeyWidth = 128; + parameter int ScrmblBlockWidth = 64; + + /////////////////////////////// + // Typedefs for LC Interface // + /////////////////////////////// + + // The tokens below are all hash post-images + typedef struct packed { + logic valid; + logic error; + // Use lc_state_t and lc_cnt_t here as very wide enumerations ( > 64 bits ) + // are not supported for virtual interfaces by Excelium yet + // https://github.com/lowRISC/opentitan/issues/8884 (Cadence issue: cds_46570160) + // The enumeration types lc_state_e and lc_cnt_e are still ok in other circumstances + lc_ctrl_state_pkg::lc_state_t state; + lc_ctrl_state_pkg::lc_cnt_t count; + // This is set to "On" if the partition containing the + // root secrets have been locked. In that case, the device + // is considered "personalized". + lc_ctrl_pkg::lc_tx_t secrets_valid; + // This is set to "On" if the partition containing the + // test tokens has been locked. + lc_ctrl_pkg::lc_tx_t test_tokens_valid; + lc_ctrl_state_pkg::lc_token_t test_unlock_token; + lc_ctrl_state_pkg::lc_token_t test_exit_token; + // This is set to "On" if the partition containing the + // rma token has been locked. + lc_ctrl_pkg::lc_tx_t rma_token_valid; + lc_ctrl_state_pkg::lc_token_t rma_token; + } otp_lc_data_t; + + // Default for dangling connection. + // Note that we put the life cycle into + // TEST_UNLOCKED0 by default such that top levels without + // the OTP controller can still function. + parameter otp_lc_data_t OTP_LC_DATA_DEFAULT = '{ + valid: 1'b1, + error: 1'b0, + state: lc_ctrl_state_pkg::LcStTestUnlocked0, + count: lc_ctrl_state_pkg::LcCnt1, + secrets_valid: lc_ctrl_pkg::Off, + test_tokens_valid: lc_ctrl_pkg::Off, + test_unlock_token: '0, + test_exit_token: '0, + rma_token_valid: lc_ctrl_pkg::Off, + rma_token: '0 + }; + + typedef struct packed { + logic req; + lc_ctrl_state_pkg::lc_state_e state; + lc_ctrl_state_pkg::lc_cnt_e count; + } lc_otp_program_req_t; + + typedef struct packed { + logic err; + logic ack; + } lc_otp_program_rsp_t; + + typedef struct packed { + logic [OtpTestCtrlWidth-1:0] ctrl; + } lc_otp_vendor_test_req_t; + + typedef struct packed { + logic [OtpTestStatusWidth-1:0] status; + } lc_otp_vendor_test_rsp_t; + + //////////////////////////////// + // Typedefs for Key Broadcast // + //////////////////////////////// + + parameter int FlashKeySeedWidth = 256; + parameter int SramKeySeedWidth = 128; + parameter int KeyMgrKeyWidth = 256; + parameter int FlashKeyWidth = 128; + parameter int SramKeyWidth = 128; + parameter int SramNonceWidth = 128; + parameter int OtbnKeyWidth = 128; + parameter int OtbnNonceWidth = 64; + + typedef logic [SramKeyWidth-1:0] sram_key_t; + typedef logic [SramNonceWidth-1:0] sram_nonce_t; + typedef logic [OtbnKeyWidth-1:0] otbn_key_t; + typedef logic [OtbnNonceWidth-1:0] otbn_nonce_t; + + localparam int OtbnNonceSel = OtbnNonceWidth / ScrmblBlockWidth; + localparam int FlashNonceSel = FlashKeyWidth / ScrmblBlockWidth; + localparam int SramNonceSel = SramNonceWidth / ScrmblBlockWidth; + + // Get maximum nonce width + localparam int NumNonceChunks = + (OtbnNonceWidth > FlashKeyWidth) ? + ((OtbnNonceWidth > SramNonceSel) ? OtbnNonceSel : SramNonceSel) : + ((FlashKeyWidth > SramNonceSel) ? FlashNonceSel : SramNonceSel); + + typedef struct packed { + logic [KeyMgrKeyWidth-1:0] creator_root_key_share0; + logic creator_root_key_share0_valid; + logic [KeyMgrKeyWidth-1:0] creator_root_key_share1; + logic creator_root_key_share1_valid; + logic [KeyMgrKeyWidth-1:0] creator_seed; + logic creator_seed_valid; + logic [KeyMgrKeyWidth-1:0] owner_seed; + logic owner_seed_valid; + } otp_keymgr_key_t; + + parameter otp_keymgr_key_t OTP_KEYMGR_KEY_DEFAULT = '{ + creator_root_key_share0: 256'hefb7ea7ee90093cf4affd9aaa2d6c0ec446cfdf5f2d5a0bfd7e2d93edc63a102, + creator_root_key_share0_valid: 1'b1, + creator_root_key_share1: 256'h56d24a00181de99e0f690b447a8dde2a1ffb8bc306707107aa6e2410f15cfc37, + creator_root_key_share1_valid: 1'b1, + creator_seed: 256'hc7c50b38655cc87f821e5b07fed85d2c07e222a9e00bef308b3eccba0ba406fa, + creator_seed_valid: 1'b1, + owner_seed: 256'hf5052c0f14782d8b066be9f49c0b2000d3643ff3723ea7db972f69cd3e2e3e68, + owner_seed_valid: 1'b1 + }; + + typedef struct packed { + logic data_req; // Requests static key for data scrambling. + logic addr_req; // Requests static key for address scrambling. + } flash_otp_key_req_t; + + typedef struct packed { + logic req; // Requests ephemeral scrambling key and nonce. + } sram_otp_key_req_t; + + typedef struct packed { + logic req; // Requests ephemeral scrambling key and nonce. + } otbn_otp_key_req_t; + + typedef struct packed { + logic data_ack; // Ack for data key. + logic addr_ack; // Ack for address key. + logic [FlashKeyWidth-1:0] key; // 128bit static scrambling key. + logic [FlashKeyWidth-1:0] rand_key; + logic seed_valid; // Set to 1 if the key seed has been provisioned and is + // valid. + } flash_otp_key_rsp_t; + + // Default for dangling connection + parameter flash_otp_key_rsp_t FLASH_OTP_KEY_RSP_DEFAULT = '{ + data_ack: 1'b1, + addr_ack: 1'b1, + key: '0, + rand_key: '0, + seed_valid: 1'b1 + }; + + typedef struct packed { + logic ack; // Ack for key. + sram_key_t key; // 128bit ephemeral scrambling key. + sram_nonce_t nonce; // 128bit nonce. + logic seed_valid; // Set to 1 if the key seed has been provisioned and is valid. + } sram_otp_key_rsp_t; + + // Default for dangling connection + parameter sram_otp_key_rsp_t SRAM_OTP_KEY_RSP_DEFAULT = '{ + ack: 1'b1, + key: '0, + nonce: '0, + seed_valid: 1'b1 + }; + + typedef struct packed { + logic ack; // Ack for key. + otbn_key_t key; // 128bit ephemeral scrambling key. + otbn_nonce_t nonce; // 256bit nonce. + logic seed_valid; // Set to 1 if the key seed has been provisioned and is valid. + } otbn_otp_key_rsp_t; + + //////////////////////////////// + // Power/Reset Ctrl Interface // + //////////////////////////////// + + typedef struct packed { + logic init; + } pwr_otp_init_req_t; + + typedef struct packed { + logic done; + } pwr_otp_init_rsp_t; + + typedef struct packed { + logic idle; + } otp_pwr_state_t; + + /////////////////// + // AST Interface // + /////////////////// + + typedef struct packed { + logic [OtpPwrSeqWidth-1:0] pwr_seq; + } otp_ast_req_t; + + typedef struct packed { + logic [OtpPwrSeqWidth-1:0] pwr_seq_h; + } otp_ast_rsp_t; + +endpackage : otp_ctrl_pkg diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl_prim_reg_top.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_prim_reg_top.sv similarity index 100% rename from hw/ip/otp_ctrl/rtl/otp_ctrl_prim_reg_top.sv rename to hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_prim_reg_top.sv diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_reg_pkg.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_reg_pkg.sv new file mode 100644 index 00000000000000..a0e892cdbec8a5 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_reg_pkg.sv @@ -0,0 +1,1527 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Register Package auto-generated by `reggen` containing data structure + +package otp_ctrl_reg_pkg; + + // Param list + parameter int NumSramKeyReqSlots = 4; + parameter int OtpByteAddrWidth = 14; + parameter int NumErrorEntries = 24; + parameter int NumDaiWords = 2; + parameter int NumDigestWords = 2; + parameter int NumSwCfgWindowWords = 4096; + parameter int NumPart = 22; + parameter int NumPartUnbuf = 15; + parameter int NumPartBuf = 7; + parameter int VendorTestOffset = 0; + parameter int VendorTestSize = 64; + parameter int ScratchOffset = 0; + parameter int ScratchSize = 56; + parameter int VendorTestDigestOffset = 56; + parameter int VendorTestDigestSize = 8; + parameter int CreatorSwCfgOffset = 64; + parameter int CreatorSwCfgSize = 320; + parameter int CreatorSwCfgAstCfgOffset = 64; + parameter int CreatorSwCfgAstCfgSize = 124; + parameter int CreatorSwCfgAstInitEnOffset = 188; + parameter int CreatorSwCfgAstInitEnSize = 4; + parameter int CreatorSwCfgOverridesOffset = 192; + parameter int CreatorSwCfgOverridesSize = 32; + parameter int CreatorSwCfgRomExtSkuOffset = 224; + parameter int CreatorSwCfgRomExtSkuSize = 4; + parameter int CreatorSwCfgSigverifyRsaModExpIbexEnOffset = 228; + parameter int CreatorSwCfgSigverifyRsaModExpIbexEnSize = 4; + parameter int CreatorSwCfgSigverifyRsaKeyEnOffset = 232; + parameter int CreatorSwCfgSigverifyRsaKeyEnSize = 8; + parameter int CreatorSwCfgSigverifySpxEnOffset = 240; + parameter int CreatorSwCfgSigverifySpxEnSize = 4; + parameter int CreatorSwCfgSigverifySpxKeyEnOffset = 244; + parameter int CreatorSwCfgSigverifySpxKeyEnSize = 8; + parameter int CreatorSwCfgFlashDataDefaultCfgOffset = 252; + parameter int CreatorSwCfgFlashDataDefaultCfgSize = 4; + parameter int CreatorSwCfgFlashInfoBootDataCfgOffset = 256; + parameter int CreatorSwCfgFlashInfoBootDataCfgSize = 4; + parameter int CreatorSwCfgFlashHwInfoCfgOverrideOffset = 260; + parameter int CreatorSwCfgFlashHwInfoCfgOverrideSize = 4; + parameter int CreatorSwCfgRngEnOffset = 264; + parameter int CreatorSwCfgRngEnSize = 4; + parameter int CreatorSwCfgJitterEnOffset = 268; + parameter int CreatorSwCfgJitterEnSize = 4; + parameter int CreatorSwCfgRetRamResetMaskOffset = 272; + parameter int CreatorSwCfgRetRamResetMaskSize = 4; + parameter int CreatorSwCfgManufStateOffset = 276; + parameter int CreatorSwCfgManufStateSize = 4; + parameter int CreatorSwCfgRomExecEnOffset = 280; + parameter int CreatorSwCfgRomExecEnSize = 4; + parameter int CreatorSwCfgCpuctrlOffset = 284; + parameter int CreatorSwCfgCpuctrlSize = 4; + parameter int CreatorSwCfgMinSecVerRomExtOffset = 288; + parameter int CreatorSwCfgMinSecVerRomExtSize = 4; + parameter int CreatorSwCfgMinSecVerBl0Offset = 292; + parameter int CreatorSwCfgMinSecVerBl0Size = 4; + parameter int CreatorSwCfgDefaultBootDataInProdEnOffset = 296; + parameter int CreatorSwCfgDefaultBootDataInProdEnSize = 4; + parameter int CreatorSwCfgRmaSpinEnOffset = 300; + parameter int CreatorSwCfgRmaSpinEnSize = 4; + parameter int CreatorSwCfgRmaSpinCyclesOffset = 304; + parameter int CreatorSwCfgRmaSpinCyclesSize = 4; + parameter int CreatorSwCfgRngRepcntThresholdsOffset = 308; + parameter int CreatorSwCfgRngRepcntThresholdsSize = 4; + parameter int CreatorSwCfgRngRepcntsThresholdsOffset = 312; + parameter int CreatorSwCfgRngRepcntsThresholdsSize = 4; + parameter int CreatorSwCfgRngAdaptpHiThresholdsOffset = 316; + parameter int CreatorSwCfgRngAdaptpHiThresholdsSize = 4; + parameter int CreatorSwCfgRngAdaptpLoThresholdsOffset = 320; + parameter int CreatorSwCfgRngAdaptpLoThresholdsSize = 4; + parameter int CreatorSwCfgRngBucketThresholdsOffset = 324; + parameter int CreatorSwCfgRngBucketThresholdsSize = 4; + parameter int CreatorSwCfgRngMarkovHiThresholdsOffset = 328; + parameter int CreatorSwCfgRngMarkovHiThresholdsSize = 4; + parameter int CreatorSwCfgRngMarkovLoThresholdsOffset = 332; + parameter int CreatorSwCfgRngMarkovLoThresholdsSize = 4; + parameter int CreatorSwCfgRngExthtHiThresholdsOffset = 336; + parameter int CreatorSwCfgRngExthtHiThresholdsSize = 4; + parameter int CreatorSwCfgRngExthtLoThresholdsOffset = 340; + parameter int CreatorSwCfgRngExthtLoThresholdsSize = 4; + parameter int CreatorSwCfgRngAlertThresholdOffset = 344; + parameter int CreatorSwCfgRngAlertThresholdSize = 4; + parameter int CreatorSwCfgRngHealthConfigDigestOffset = 348; + parameter int CreatorSwCfgRngHealthConfigDigestSize = 4; + parameter int CreatorSwCfgSramKeyRenewEnOffset = 352; + parameter int CreatorSwCfgSramKeyRenewEnSize = 4; + parameter int CreatorSwCfgDigestOffset = 376; + parameter int CreatorSwCfgDigestSize = 8; + parameter int OwnerSwCfgOffset = 384; + parameter int OwnerSwCfgSize = 632; + parameter int OwnerSwCfgRomErrorReportingOffset = 384; + parameter int OwnerSwCfgRomErrorReportingSize = 4; + parameter int OwnerSwCfgRomBootstrapDisOffset = 388; + parameter int OwnerSwCfgRomBootstrapDisSize = 4; + parameter int OwnerSwCfgRomAlertClassEnOffset = 392; + parameter int OwnerSwCfgRomAlertClassEnSize = 4; + parameter int OwnerSwCfgRomAlertEscalationOffset = 396; + parameter int OwnerSwCfgRomAlertEscalationSize = 4; + parameter int OwnerSwCfgRomAlertClassificationOffset = 400; + parameter int OwnerSwCfgRomAlertClassificationSize = 400; + parameter int OwnerSwCfgRomLocalAlertClassificationOffset = 800; + parameter int OwnerSwCfgRomLocalAlertClassificationSize = 64; + parameter int OwnerSwCfgRomAlertAccumThreshOffset = 864; + parameter int OwnerSwCfgRomAlertAccumThreshSize = 16; + parameter int OwnerSwCfgRomAlertTimeoutCyclesOffset = 880; + parameter int OwnerSwCfgRomAlertTimeoutCyclesSize = 16; + parameter int OwnerSwCfgRomAlertPhaseCyclesOffset = 896; + parameter int OwnerSwCfgRomAlertPhaseCyclesSize = 64; + parameter int OwnerSwCfgRomAlertDigestProdOffset = 960; + parameter int OwnerSwCfgRomAlertDigestProdSize = 4; + parameter int OwnerSwCfgRomAlertDigestProdEndOffset = 964; + parameter int OwnerSwCfgRomAlertDigestProdEndSize = 4; + parameter int OwnerSwCfgRomAlertDigestDevOffset = 968; + parameter int OwnerSwCfgRomAlertDigestDevSize = 4; + parameter int OwnerSwCfgRomAlertDigestRmaOffset = 972; + parameter int OwnerSwCfgRomAlertDigestRmaSize = 4; + parameter int OwnerSwCfgRomWatchdogBiteThresholdCyclesOffset = 976; + parameter int OwnerSwCfgRomWatchdogBiteThresholdCyclesSize = 4; + parameter int OwnerSwCfgRomKeymgrRomExtMeasEnOffset = 980; + parameter int OwnerSwCfgRomKeymgrRomExtMeasEnSize = 4; + parameter int OwnerSwCfgManufStateOffset = 984; + parameter int OwnerSwCfgManufStateSize = 4; + parameter int OwnerSwCfgRomRstmgrInfoEnOffset = 988; + parameter int OwnerSwCfgRomRstmgrInfoEnSize = 4; + parameter int OwnerSwCfgDigestOffset = 1008; + parameter int OwnerSwCfgDigestSize = 8; + parameter int OwnershipSlotStateOffset = 1016; + parameter int OwnershipSlotStateSize = 48; + parameter int OwnershipSlotStateRotOwnerAuthOffset = 1016; + parameter int OwnershipSlotStateRotOwnerAuthSize = 16; + parameter int OwnershipSlotStatePlatIntegAuthOffset = 1032; + parameter int OwnershipSlotStatePlatIntegAuthSize = 16; + parameter int OwnershipSlotStatePlatOwnerAuthOffset = 1048; + parameter int OwnershipSlotStatePlatOwnerAuthSize = 16; + parameter int RotCreatorAuthOffset = 1064; + parameter int RotCreatorAuthSize = 1424; + parameter int RotCreatorAuthNonRawMfwCodesignKeyOffset = 1064; + parameter int RotCreatorAuthNonRawMfwCodesignKeySize = 160; + parameter int RotCreatorAuthOwnershipStateOffset = 1224; + parameter int RotCreatorAuthOwnershipStateSize = 4; + parameter int RotCreatorAuthRom2PatchSigverifyKeyOffset = 1228; + parameter int RotCreatorAuthRom2PatchSigverifyKeySize = 160; + parameter int RotCreatorAuthKeymanifestKeyOffset = 1388; + parameter int RotCreatorAuthKeymanifestKeySize = 160; + parameter int RotCreatorAuthUnlock4xferKeyOffset = 1548; + parameter int RotCreatorAuthUnlock4xferKeySize = 160; + parameter int RotCreatorAuthIdentityCertOffset = 1708; + parameter int RotCreatorAuthIdentityCertSize = 768; + parameter int RotCreatorAuthDigestOffset = 2480; + parameter int RotCreatorAuthDigestSize = 8; + parameter int RotOwnerAuthSlot0Offset = 2488; + parameter int RotOwnerAuthSlot0Size = 328; + parameter int RotOwnerAuthSlot0KeymanifestKeyOffset = 2488; + parameter int RotOwnerAuthSlot0KeymanifestKeySize = 160; + parameter int RotOwnerAuthSlot0Unlock4xferKeyOffset = 2648; + parameter int RotOwnerAuthSlot0Unlock4xferKeySize = 160; + parameter int RotOwnerAuthSlot0DigestOffset = 2808; + parameter int RotOwnerAuthSlot0DigestSize = 8; + parameter int RotOwnerAuthSlot1Offset = 2816; + parameter int RotOwnerAuthSlot1Size = 328; + parameter int RotOwnerAuthSlot1KeymanifestKeyOffset = 2816; + parameter int RotOwnerAuthSlot1KeymanifestKeySize = 160; + parameter int RotOwnerAuthSlot1Unlock4xferKeyOffset = 2976; + parameter int RotOwnerAuthSlot1Unlock4xferKeySize = 160; + parameter int RotOwnerAuthSlot1DigestOffset = 3136; + parameter int RotOwnerAuthSlot1DigestSize = 8; + parameter int PlatIntegAuthSlot0Offset = 3144; + parameter int PlatIntegAuthSlot0Size = 328; + parameter int PlatIntegAuthSlot0KeymanifestKeyOffset = 3144; + parameter int PlatIntegAuthSlot0KeymanifestKeySize = 160; + parameter int PlatIntegAuthSlot0Unlock4xferKeyOffset = 3304; + parameter int PlatIntegAuthSlot0Unlock4xferKeySize = 160; + parameter int PlatIntegAuthSlot0DigestOffset = 3464; + parameter int PlatIntegAuthSlot0DigestSize = 8; + parameter int PlatIntegAuthSlot1Offset = 3472; + parameter int PlatIntegAuthSlot1Size = 328; + parameter int PlatIntegAuthSlot1KeymanifestKeyOffset = 3472; + parameter int PlatIntegAuthSlot1KeymanifestKeySize = 160; + parameter int PlatIntegAuthSlot1Unlock4xferKeyOffset = 3632; + parameter int PlatIntegAuthSlot1Unlock4xferKeySize = 160; + parameter int PlatIntegAuthSlot1DigestOffset = 3792; + parameter int PlatIntegAuthSlot1DigestSize = 8; + parameter int PlatOwnerAuthSlot0Offset = 3800; + parameter int PlatOwnerAuthSlot0Size = 328; + parameter int PlatOwnerAuthSlot0KeymanifestKeyOffset = 3800; + parameter int PlatOwnerAuthSlot0KeymanifestKeySize = 160; + parameter int PlatOwnerAuthSlot0Unlock4xferKeyOffset = 3960; + parameter int PlatOwnerAuthSlot0Unlock4xferKeySize = 160; + parameter int PlatOwnerAuthSlot0DigestOffset = 4120; + parameter int PlatOwnerAuthSlot0DigestSize = 8; + parameter int PlatOwnerAuthSlot1Offset = 4128; + parameter int PlatOwnerAuthSlot1Size = 328; + parameter int PlatOwnerAuthSlot1KeymanifestKeyOffset = 4128; + parameter int PlatOwnerAuthSlot1KeymanifestKeySize = 160; + parameter int PlatOwnerAuthSlot1Unlock4xferKeyOffset = 4288; + parameter int PlatOwnerAuthSlot1Unlock4xferKeySize = 160; + parameter int PlatOwnerAuthSlot1DigestOffset = 4448; + parameter int PlatOwnerAuthSlot1DigestSize = 8; + parameter int PlatOwnerAuthSlot2Offset = 4456; + parameter int PlatOwnerAuthSlot2Size = 328; + parameter int PlatOwnerAuthSlot2KeymanifestKeyOffset = 4456; + parameter int PlatOwnerAuthSlot2KeymanifestKeySize = 160; + parameter int PlatOwnerAuthSlot2Unlock4xferKeyOffset = 4616; + parameter int PlatOwnerAuthSlot2Unlock4xferKeySize = 160; + parameter int PlatOwnerAuthSlot2DigestOffset = 4776; + parameter int PlatOwnerAuthSlot2DigestSize = 8; + parameter int PlatOwnerAuthSlot3Offset = 4784; + parameter int PlatOwnerAuthSlot3Size = 328; + parameter int PlatOwnerAuthSlot3KeymanifestKeyOffset = 4784; + parameter int PlatOwnerAuthSlot3KeymanifestKeySize = 160; + parameter int PlatOwnerAuthSlot3Unlock4xferKeyOffset = 4944; + parameter int PlatOwnerAuthSlot3Unlock4xferKeySize = 160; + parameter int PlatOwnerAuthSlot3DigestOffset = 5104; + parameter int PlatOwnerAuthSlot3DigestSize = 8; + parameter int ExtNvmOffset = 5112; + parameter int ExtNvmSize = 1024; + parameter int ExtNvmAntireplayFreshnessCntOffset = 5112; + parameter int ExtNvmAntireplayFreshnessCntSize = 1024; + parameter int RomPatchOffset = 6136; + parameter int RomPatchSize = 9784; + parameter int RomPatchDataOffset = 6136; + parameter int RomPatchDataSize = 9192; + parameter int RomPatchDigestOffset = 15912; + parameter int RomPatchDigestSize = 8; + parameter int HwCfg0Offset = 15920; + parameter int HwCfg0Size = 72; + parameter int DeviceIdOffset = 15920; + parameter int DeviceIdSize = 32; + parameter int ManufStateOffset = 15952; + parameter int ManufStateSize = 32; + parameter int HwCfg0DigestOffset = 15984; + parameter int HwCfg0DigestSize = 8; + parameter int HwCfg1Offset = 15992; + parameter int HwCfg1Size = 16; + parameter int SocDbgStateOffset = 15992; + parameter int SocDbgStateSize = 4; + parameter int EnSramIfetchOffset = 15996; + parameter int EnSramIfetchSize = 1; + parameter int HwCfg1DigestOffset = 16000; + parameter int HwCfg1DigestSize = 8; + parameter int Secret0Offset = 16008; + parameter int Secret0Size = 40; + parameter int TestUnlockTokenOffset = 16008; + parameter int TestUnlockTokenSize = 16; + parameter int TestExitTokenOffset = 16024; + parameter int TestExitTokenSize = 16; + parameter int Secret0DigestOffset = 16040; + parameter int Secret0DigestSize = 8; + parameter int Secret1Offset = 16048; + parameter int Secret1Size = 88; + parameter int FlashAddrKeySeedOffset = 16048; + parameter int FlashAddrKeySeedSize = 32; + parameter int FlashDataKeySeedOffset = 16080; + parameter int FlashDataKeySeedSize = 32; + parameter int SramDataKeySeedOffset = 16112; + parameter int SramDataKeySeedSize = 16; + parameter int Secret1DigestOffset = 16128; + parameter int Secret1DigestSize = 8; + parameter int Secret2Offset = 16136; + parameter int Secret2Size = 120; + parameter int RmaTokenOffset = 16136; + parameter int RmaTokenSize = 16; + parameter int CreatorRootKeyShare0Offset = 16152; + parameter int CreatorRootKeyShare0Size = 32; + parameter int CreatorRootKeyShare1Offset = 16184; + parameter int CreatorRootKeyShare1Size = 32; + parameter int CreatorSeedOffset = 16216; + parameter int CreatorSeedSize = 32; + parameter int Secret2DigestOffset = 16248; + parameter int Secret2DigestSize = 8; + parameter int Secret3Offset = 16256; + parameter int Secret3Size = 40; + parameter int OwnerSeedOffset = 16256; + parameter int OwnerSeedSize = 32; + parameter int Secret3DigestOffset = 16288; + parameter int Secret3DigestSize = 8; + parameter int LifeCycleOffset = 16296; + parameter int LifeCycleSize = 88; + parameter int LcTransitionCntOffset = 16296; + parameter int LcTransitionCntSize = 48; + parameter int LcStateOffset = 16344; + parameter int LcStateSize = 40; + parameter int NumAlerts = 5; + + // Address widths within the block + parameter int CoreAw = 15; + parameter int PrimAw = 5; + + /////////////////////////////////////////////// + // Typedefs for registers for core interface // + /////////////////////////////////////////////// + + typedef struct packed { + struct packed { + logic q; + } otp_error; + struct packed { + logic q; + } otp_operation_done; + } otp_ctrl_reg2hw_intr_state_reg_t; + + typedef struct packed { + struct packed { + logic q; + } otp_error; + struct packed { + logic q; + } otp_operation_done; + } otp_ctrl_reg2hw_intr_enable_reg_t; + + typedef struct packed { + struct packed { + logic q; + logic qe; + } otp_error; + struct packed { + logic q; + logic qe; + } otp_operation_done; + } otp_ctrl_reg2hw_intr_test_reg_t; + + typedef struct packed { + struct packed { + logic q; + logic qe; + } recov_prim_otp_alert; + struct packed { + logic q; + logic qe; + } fatal_prim_otp_alert; + struct packed { + logic q; + logic qe; + } fatal_bus_integ_error; + struct packed { + logic q; + logic qe; + } fatal_check_error; + struct packed { + logic q; + logic qe; + } fatal_macro_error; + } otp_ctrl_reg2hw_alert_test_reg_t; + + typedef struct packed { + logic q; + logic qe; + } otp_ctrl_reg2hw_direct_access_regwen_reg_t; + + typedef struct packed { + struct packed { + logic q; + logic qe; + } digest; + struct packed { + logic q; + logic qe; + } wr; + struct packed { + logic q; + logic qe; + } rd; + } otp_ctrl_reg2hw_direct_access_cmd_reg_t; + + typedef struct packed { + logic [13:0] q; + } otp_ctrl_reg2hw_direct_access_address_reg_t; + + typedef struct packed { + logic [31:0] q; + } otp_ctrl_reg2hw_direct_access_wdata_mreg_t; + + typedef struct packed { + struct packed { + logic q; + logic qe; + } consistency; + struct packed { + logic q; + logic qe; + } integrity; + } otp_ctrl_reg2hw_check_trigger_reg_t; + + typedef struct packed { + logic [31:0] q; + } otp_ctrl_reg2hw_check_timeout_reg_t; + + typedef struct packed { + logic [31:0] q; + } otp_ctrl_reg2hw_integrity_check_period_reg_t; + + typedef struct packed { + logic [31:0] q; + } otp_ctrl_reg2hw_consistency_check_period_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_vendor_test_read_lock_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_creator_sw_cfg_read_lock_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_owner_sw_cfg_read_lock_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_ownership_slot_state_read_lock_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_rot_creator_auth_read_lock_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_rot_owner_auth_slot0_read_lock_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_rot_owner_auth_slot1_read_lock_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_plat_integ_auth_slot0_read_lock_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_plat_integ_auth_slot1_read_lock_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_plat_owner_auth_slot0_read_lock_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_plat_owner_auth_slot1_read_lock_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_plat_owner_auth_slot2_read_lock_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_plat_owner_auth_slot3_read_lock_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_ext_nvm_read_lock_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_rom_patch_read_lock_reg_t; + + typedef struct packed { + struct packed { + logic d; + logic de; + } otp_operation_done; + struct packed { + logic d; + logic de; + } otp_error; + } otp_ctrl_hw2reg_intr_state_reg_t; + + typedef struct packed { + struct packed { + logic d; + } vendor_test_error; + struct packed { + logic d; + } creator_sw_cfg_error; + struct packed { + logic d; + } owner_sw_cfg_error; + struct packed { + logic d; + } ownership_slot_state_error; + struct packed { + logic d; + } rot_creator_auth_error; + struct packed { + logic d; + } rot_owner_auth_slot0_error; + struct packed { + logic d; + } rot_owner_auth_slot1_error; + struct packed { + logic d; + } plat_integ_auth_slot0_error; + struct packed { + logic d; + } plat_integ_auth_slot1_error; + struct packed { + logic d; + } plat_owner_auth_slot0_error; + struct packed { + logic d; + } plat_owner_auth_slot1_error; + struct packed { + logic d; + } plat_owner_auth_slot2_error; + struct packed { + logic d; + } plat_owner_auth_slot3_error; + struct packed { + logic d; + } ext_nvm_error; + struct packed { + logic d; + } rom_patch_error; + struct packed { + logic d; + } hw_cfg0_error; + struct packed { + logic d; + } hw_cfg1_error; + struct packed { + logic d; + } secret0_error; + struct packed { + logic d; + } secret1_error; + struct packed { + logic d; + } secret2_error; + struct packed { + logic d; + } secret3_error; + struct packed { + logic d; + } life_cycle_error; + struct packed { + logic d; + } dai_error; + struct packed { + logic d; + } lci_error; + struct packed { + logic d; + } timeout_error; + struct packed { + logic d; + } lfsr_fsm_error; + struct packed { + logic d; + } scrambling_fsm_error; + struct packed { + logic d; + } key_deriv_fsm_error; + struct packed { + logic d; + } bus_integ_error; + struct packed { + logic d; + } dai_idle; + struct packed { + logic d; + } check_pending; + } otp_ctrl_hw2reg_status_reg_t; + + typedef struct packed { + logic [2:0] d; + } otp_ctrl_hw2reg_err_code_mreg_t; + + typedef struct packed { + logic d; + } otp_ctrl_hw2reg_direct_access_regwen_reg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_direct_access_rdata_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_vendor_test_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_creator_sw_cfg_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_owner_sw_cfg_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_rot_creator_auth_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_rot_owner_auth_slot0_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_rot_owner_auth_slot1_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_plat_integ_auth_slot0_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_plat_integ_auth_slot1_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_plat_owner_auth_slot0_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_plat_owner_auth_slot1_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_plat_owner_auth_slot2_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_plat_owner_auth_slot3_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_rom_patch_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_hw_cfg0_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_hw_cfg1_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_secret0_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_secret1_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_secret2_digest_mreg_t; + + typedef struct packed { + logic [31:0] d; + } otp_ctrl_hw2reg_secret3_digest_mreg_t; + + // Register -> HW type for core interface + typedef struct packed { + otp_ctrl_reg2hw_intr_state_reg_t intr_state; // [218:217] + otp_ctrl_reg2hw_intr_enable_reg_t intr_enable; // [216:215] + otp_ctrl_reg2hw_intr_test_reg_t intr_test; // [214:211] + otp_ctrl_reg2hw_alert_test_reg_t alert_test; // [210:201] + otp_ctrl_reg2hw_direct_access_regwen_reg_t direct_access_regwen; // [200:199] + otp_ctrl_reg2hw_direct_access_cmd_reg_t direct_access_cmd; // [198:193] + otp_ctrl_reg2hw_direct_access_address_reg_t direct_access_address; // [192:179] + otp_ctrl_reg2hw_direct_access_wdata_mreg_t [1:0] direct_access_wdata; // [178:115] + otp_ctrl_reg2hw_check_trigger_reg_t check_trigger; // [114:111] + otp_ctrl_reg2hw_check_timeout_reg_t check_timeout; // [110:79] + otp_ctrl_reg2hw_integrity_check_period_reg_t integrity_check_period; // [78:47] + otp_ctrl_reg2hw_consistency_check_period_reg_t consistency_check_period; // [46:15] + otp_ctrl_reg2hw_vendor_test_read_lock_reg_t vendor_test_read_lock; // [14:14] + otp_ctrl_reg2hw_creator_sw_cfg_read_lock_reg_t creator_sw_cfg_read_lock; // [13:13] + otp_ctrl_reg2hw_owner_sw_cfg_read_lock_reg_t owner_sw_cfg_read_lock; // [12:12] + otp_ctrl_reg2hw_ownership_slot_state_read_lock_reg_t ownership_slot_state_read_lock; // [11:11] + otp_ctrl_reg2hw_rot_creator_auth_read_lock_reg_t rot_creator_auth_read_lock; // [10:10] + otp_ctrl_reg2hw_rot_owner_auth_slot0_read_lock_reg_t rot_owner_auth_slot0_read_lock; // [9:9] + otp_ctrl_reg2hw_rot_owner_auth_slot1_read_lock_reg_t rot_owner_auth_slot1_read_lock; // [8:8] + otp_ctrl_reg2hw_plat_integ_auth_slot0_read_lock_reg_t plat_integ_auth_slot0_read_lock; // [7:7] + otp_ctrl_reg2hw_plat_integ_auth_slot1_read_lock_reg_t plat_integ_auth_slot1_read_lock; // [6:6] + otp_ctrl_reg2hw_plat_owner_auth_slot0_read_lock_reg_t plat_owner_auth_slot0_read_lock; // [5:5] + otp_ctrl_reg2hw_plat_owner_auth_slot1_read_lock_reg_t plat_owner_auth_slot1_read_lock; // [4:4] + otp_ctrl_reg2hw_plat_owner_auth_slot2_read_lock_reg_t plat_owner_auth_slot2_read_lock; // [3:3] + otp_ctrl_reg2hw_plat_owner_auth_slot3_read_lock_reg_t plat_owner_auth_slot3_read_lock; // [2:2] + otp_ctrl_reg2hw_ext_nvm_read_lock_reg_t ext_nvm_read_lock; // [1:1] + otp_ctrl_reg2hw_rom_patch_read_lock_reg_t rom_patch_read_lock; // [0:0] + } otp_ctrl_core_reg2hw_t; + + // HW -> register type for core interface + typedef struct packed { + otp_ctrl_hw2reg_intr_state_reg_t intr_state; // [1387:1384] + otp_ctrl_hw2reg_status_reg_t status; // [1383:1353] + otp_ctrl_hw2reg_err_code_mreg_t [23:0] err_code; // [1352:1281] + otp_ctrl_hw2reg_direct_access_regwen_reg_t direct_access_regwen; // [1280:1280] + otp_ctrl_hw2reg_direct_access_rdata_mreg_t [1:0] direct_access_rdata; // [1279:1216] + otp_ctrl_hw2reg_vendor_test_digest_mreg_t [1:0] vendor_test_digest; // [1215:1152] + otp_ctrl_hw2reg_creator_sw_cfg_digest_mreg_t [1:0] creator_sw_cfg_digest; // [1151:1088] + otp_ctrl_hw2reg_owner_sw_cfg_digest_mreg_t [1:0] owner_sw_cfg_digest; // [1087:1024] + otp_ctrl_hw2reg_rot_creator_auth_digest_mreg_t [1:0] rot_creator_auth_digest; // [1023:960] + otp_ctrl_hw2reg_rot_owner_auth_slot0_digest_mreg_t [1:0] + rot_owner_auth_slot0_digest; // [959:896] + otp_ctrl_hw2reg_rot_owner_auth_slot1_digest_mreg_t [1:0] + rot_owner_auth_slot1_digest; // [895:832] + otp_ctrl_hw2reg_plat_integ_auth_slot0_digest_mreg_t [1:0] + plat_integ_auth_slot0_digest; // [831:768] + otp_ctrl_hw2reg_plat_integ_auth_slot1_digest_mreg_t [1:0] + plat_integ_auth_slot1_digest; // [767:704] + otp_ctrl_hw2reg_plat_owner_auth_slot0_digest_mreg_t [1:0] + plat_owner_auth_slot0_digest; // [703:640] + otp_ctrl_hw2reg_plat_owner_auth_slot1_digest_mreg_t [1:0] + plat_owner_auth_slot1_digest; // [639:576] + otp_ctrl_hw2reg_plat_owner_auth_slot2_digest_mreg_t [1:0] + plat_owner_auth_slot2_digest; // [575:512] + otp_ctrl_hw2reg_plat_owner_auth_slot3_digest_mreg_t [1:0] + plat_owner_auth_slot3_digest; // [511:448] + otp_ctrl_hw2reg_rom_patch_digest_mreg_t [1:0] rom_patch_digest; // [447:384] + otp_ctrl_hw2reg_hw_cfg0_digest_mreg_t [1:0] hw_cfg0_digest; // [383:320] + otp_ctrl_hw2reg_hw_cfg1_digest_mreg_t [1:0] hw_cfg1_digest; // [319:256] + otp_ctrl_hw2reg_secret0_digest_mreg_t [1:0] secret0_digest; // [255:192] + otp_ctrl_hw2reg_secret1_digest_mreg_t [1:0] secret1_digest; // [191:128] + otp_ctrl_hw2reg_secret2_digest_mreg_t [1:0] secret2_digest; // [127:64] + otp_ctrl_hw2reg_secret3_digest_mreg_t [1:0] secret3_digest; // [63:0] + } otp_ctrl_core_hw2reg_t; + + // Register offsets for core interface + parameter logic [CoreAw-1:0] OTP_CTRL_INTR_STATE_OFFSET = 15'h 0; + parameter logic [CoreAw-1:0] OTP_CTRL_INTR_ENABLE_OFFSET = 15'h 4; + parameter logic [CoreAw-1:0] OTP_CTRL_INTR_TEST_OFFSET = 15'h 8; + parameter logic [CoreAw-1:0] OTP_CTRL_ALERT_TEST_OFFSET = 15'h c; + parameter logic [CoreAw-1:0] OTP_CTRL_STATUS_OFFSET = 15'h 10; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_0_OFFSET = 15'h 14; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_1_OFFSET = 15'h 18; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_2_OFFSET = 15'h 1c; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_3_OFFSET = 15'h 20; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_4_OFFSET = 15'h 24; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_5_OFFSET = 15'h 28; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_6_OFFSET = 15'h 2c; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_7_OFFSET = 15'h 30; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_8_OFFSET = 15'h 34; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_9_OFFSET = 15'h 38; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_10_OFFSET = 15'h 3c; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_11_OFFSET = 15'h 40; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_12_OFFSET = 15'h 44; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_13_OFFSET = 15'h 48; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_14_OFFSET = 15'h 4c; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_15_OFFSET = 15'h 50; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_16_OFFSET = 15'h 54; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_17_OFFSET = 15'h 58; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_18_OFFSET = 15'h 5c; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_19_OFFSET = 15'h 60; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_20_OFFSET = 15'h 64; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_21_OFFSET = 15'h 68; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_22_OFFSET = 15'h 6c; + parameter logic [CoreAw-1:0] OTP_CTRL_ERR_CODE_23_OFFSET = 15'h 70; + parameter logic [CoreAw-1:0] OTP_CTRL_DIRECT_ACCESS_REGWEN_OFFSET = 15'h 74; + parameter logic [CoreAw-1:0] OTP_CTRL_DIRECT_ACCESS_CMD_OFFSET = 15'h 78; + parameter logic [CoreAw-1:0] OTP_CTRL_DIRECT_ACCESS_ADDRESS_OFFSET = 15'h 7c; + parameter logic [CoreAw-1:0] OTP_CTRL_DIRECT_ACCESS_WDATA_0_OFFSET = 15'h 80; + parameter logic [CoreAw-1:0] OTP_CTRL_DIRECT_ACCESS_WDATA_1_OFFSET = 15'h 84; + parameter logic [CoreAw-1:0] OTP_CTRL_DIRECT_ACCESS_RDATA_0_OFFSET = 15'h 88; + parameter logic [CoreAw-1:0] OTP_CTRL_DIRECT_ACCESS_RDATA_1_OFFSET = 15'h 8c; + parameter logic [CoreAw-1:0] OTP_CTRL_CHECK_TRIGGER_REGWEN_OFFSET = 15'h 90; + parameter logic [CoreAw-1:0] OTP_CTRL_CHECK_TRIGGER_OFFSET = 15'h 94; + parameter logic [CoreAw-1:0] OTP_CTRL_CHECK_REGWEN_OFFSET = 15'h 98; + parameter logic [CoreAw-1:0] OTP_CTRL_CHECK_TIMEOUT_OFFSET = 15'h 9c; + parameter logic [CoreAw-1:0] OTP_CTRL_INTEGRITY_CHECK_PERIOD_OFFSET = 15'h a0; + parameter logic [CoreAw-1:0] OTP_CTRL_CONSISTENCY_CHECK_PERIOD_OFFSET = 15'h a4; + parameter logic [CoreAw-1:0] OTP_CTRL_VENDOR_TEST_READ_LOCK_OFFSET = 15'h a8; + parameter logic [CoreAw-1:0] OTP_CTRL_CREATOR_SW_CFG_READ_LOCK_OFFSET = 15'h ac; + parameter logic [CoreAw-1:0] OTP_CTRL_OWNER_SW_CFG_READ_LOCK_OFFSET = 15'h b0; + parameter logic [CoreAw-1:0] OTP_CTRL_OWNERSHIP_SLOT_STATE_READ_LOCK_OFFSET = 15'h b4; + parameter logic [CoreAw-1:0] OTP_CTRL_ROT_CREATOR_AUTH_READ_LOCK_OFFSET = 15'h b8; + parameter logic [CoreAw-1:0] OTP_CTRL_ROT_OWNER_AUTH_SLOT0_READ_LOCK_OFFSET = 15'h bc; + parameter logic [CoreAw-1:0] OTP_CTRL_ROT_OWNER_AUTH_SLOT1_READ_LOCK_OFFSET = 15'h c0; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_READ_LOCK_OFFSET = 15'h c4; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_READ_LOCK_OFFSET = 15'h c8; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_READ_LOCK_OFFSET = 15'h cc; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_READ_LOCK_OFFSET = 15'h d0; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_READ_LOCK_OFFSET = 15'h d4; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_READ_LOCK_OFFSET = 15'h d8; + parameter logic [CoreAw-1:0] OTP_CTRL_EXT_NVM_READ_LOCK_OFFSET = 15'h dc; + parameter logic [CoreAw-1:0] OTP_CTRL_ROM_PATCH_READ_LOCK_OFFSET = 15'h e0; + parameter logic [CoreAw-1:0] OTP_CTRL_VENDOR_TEST_DIGEST_0_OFFSET = 15'h e4; + parameter logic [CoreAw-1:0] OTP_CTRL_VENDOR_TEST_DIGEST_1_OFFSET = 15'h e8; + parameter logic [CoreAw-1:0] OTP_CTRL_CREATOR_SW_CFG_DIGEST_0_OFFSET = 15'h ec; + parameter logic [CoreAw-1:0] OTP_CTRL_CREATOR_SW_CFG_DIGEST_1_OFFSET = 15'h f0; + parameter logic [CoreAw-1:0] OTP_CTRL_OWNER_SW_CFG_DIGEST_0_OFFSET = 15'h f4; + parameter logic [CoreAw-1:0] OTP_CTRL_OWNER_SW_CFG_DIGEST_1_OFFSET = 15'h f8; + parameter logic [CoreAw-1:0] OTP_CTRL_ROT_CREATOR_AUTH_DIGEST_0_OFFSET = 15'h fc; + parameter logic [CoreAw-1:0] OTP_CTRL_ROT_CREATOR_AUTH_DIGEST_1_OFFSET = 15'h 100; + parameter logic [CoreAw-1:0] OTP_CTRL_ROT_OWNER_AUTH_SLOT0_DIGEST_0_OFFSET = 15'h 104; + parameter logic [CoreAw-1:0] OTP_CTRL_ROT_OWNER_AUTH_SLOT0_DIGEST_1_OFFSET = 15'h 108; + parameter logic [CoreAw-1:0] OTP_CTRL_ROT_OWNER_AUTH_SLOT1_DIGEST_0_OFFSET = 15'h 10c; + parameter logic [CoreAw-1:0] OTP_CTRL_ROT_OWNER_AUTH_SLOT1_DIGEST_1_OFFSET = 15'h 110; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_DIGEST_0_OFFSET = 15'h 114; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_DIGEST_1_OFFSET = 15'h 118; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_DIGEST_0_OFFSET = 15'h 11c; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_DIGEST_1_OFFSET = 15'h 120; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_DIGEST_0_OFFSET = 15'h 124; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_DIGEST_1_OFFSET = 15'h 128; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_DIGEST_0_OFFSET = 15'h 12c; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_DIGEST_1_OFFSET = 15'h 130; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_DIGEST_0_OFFSET = 15'h 134; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_DIGEST_1_OFFSET = 15'h 138; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_DIGEST_0_OFFSET = 15'h 13c; + parameter logic [CoreAw-1:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_DIGEST_1_OFFSET = 15'h 140; + parameter logic [CoreAw-1:0] OTP_CTRL_ROM_PATCH_DIGEST_0_OFFSET = 15'h 144; + parameter logic [CoreAw-1:0] OTP_CTRL_ROM_PATCH_DIGEST_1_OFFSET = 15'h 148; + parameter logic [CoreAw-1:0] OTP_CTRL_HW_CFG0_DIGEST_0_OFFSET = 15'h 14c; + parameter logic [CoreAw-1:0] OTP_CTRL_HW_CFG0_DIGEST_1_OFFSET = 15'h 150; + parameter logic [CoreAw-1:0] OTP_CTRL_HW_CFG1_DIGEST_0_OFFSET = 15'h 154; + parameter logic [CoreAw-1:0] OTP_CTRL_HW_CFG1_DIGEST_1_OFFSET = 15'h 158; + parameter logic [CoreAw-1:0] OTP_CTRL_SECRET0_DIGEST_0_OFFSET = 15'h 15c; + parameter logic [CoreAw-1:0] OTP_CTRL_SECRET0_DIGEST_1_OFFSET = 15'h 160; + parameter logic [CoreAw-1:0] OTP_CTRL_SECRET1_DIGEST_0_OFFSET = 15'h 164; + parameter logic [CoreAw-1:0] OTP_CTRL_SECRET1_DIGEST_1_OFFSET = 15'h 168; + parameter logic [CoreAw-1:0] OTP_CTRL_SECRET2_DIGEST_0_OFFSET = 15'h 16c; + parameter logic [CoreAw-1:0] OTP_CTRL_SECRET2_DIGEST_1_OFFSET = 15'h 170; + parameter logic [CoreAw-1:0] OTP_CTRL_SECRET3_DIGEST_0_OFFSET = 15'h 174; + parameter logic [CoreAw-1:0] OTP_CTRL_SECRET3_DIGEST_1_OFFSET = 15'h 178; + + // Reset values for hwext registers and their fields for core interface + parameter logic [1:0] OTP_CTRL_INTR_TEST_RESVAL = 2'h 0; + parameter logic [0:0] OTP_CTRL_INTR_TEST_OTP_OPERATION_DONE_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_INTR_TEST_OTP_ERROR_RESVAL = 1'h 0; + parameter logic [4:0] OTP_CTRL_ALERT_TEST_RESVAL = 5'h 0; + parameter logic [0:0] OTP_CTRL_ALERT_TEST_FATAL_MACRO_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_ALERT_TEST_FATAL_CHECK_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_ALERT_TEST_FATAL_BUS_INTEG_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_ALERT_TEST_FATAL_PRIM_OTP_ALERT_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_ALERT_TEST_RECOV_PRIM_OTP_ALERT_RESVAL = 1'h 0; + parameter logic [30:0] OTP_CTRL_STATUS_RESVAL = 31'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_VENDOR_TEST_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_CREATOR_SW_CFG_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_OWNER_SW_CFG_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_OWNERSHIP_SLOT_STATE_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_ROT_CREATOR_AUTH_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_ROT_OWNER_AUTH_SLOT0_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_ROT_OWNER_AUTH_SLOT1_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_PLAT_INTEG_AUTH_SLOT0_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_PLAT_INTEG_AUTH_SLOT1_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_PLAT_OWNER_AUTH_SLOT0_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_PLAT_OWNER_AUTH_SLOT1_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_PLAT_OWNER_AUTH_SLOT2_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_PLAT_OWNER_AUTH_SLOT3_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_EXT_NVM_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_ROM_PATCH_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_HW_CFG0_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_HW_CFG1_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_SECRET0_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_SECRET1_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_SECRET2_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_SECRET3_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_LIFE_CYCLE_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_DAI_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_LCI_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_TIMEOUT_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_LFSR_FSM_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_SCRAMBLING_FSM_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_KEY_DERIV_FSM_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_BUS_INTEG_ERROR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_DAI_IDLE_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_STATUS_CHECK_PENDING_RESVAL = 1'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_0_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_0_ERR_CODE_0_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_1_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_1_ERR_CODE_1_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_2_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_2_ERR_CODE_2_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_3_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_3_ERR_CODE_3_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_4_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_4_ERR_CODE_4_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_5_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_5_ERR_CODE_5_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_6_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_6_ERR_CODE_6_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_7_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_7_ERR_CODE_7_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_8_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_8_ERR_CODE_8_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_9_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_9_ERR_CODE_9_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_10_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_10_ERR_CODE_10_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_11_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_11_ERR_CODE_11_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_12_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_12_ERR_CODE_12_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_13_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_13_ERR_CODE_13_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_14_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_14_ERR_CODE_14_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_15_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_15_ERR_CODE_15_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_16_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_16_ERR_CODE_16_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_17_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_17_ERR_CODE_17_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_18_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_18_ERR_CODE_18_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_19_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_19_ERR_CODE_19_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_20_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_20_ERR_CODE_20_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_21_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_21_ERR_CODE_21_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_22_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_22_ERR_CODE_22_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_23_RESVAL = 3'h 0; + parameter logic [2:0] OTP_CTRL_ERR_CODE_23_ERR_CODE_23_RESVAL = 3'h 0; + parameter logic [0:0] OTP_CTRL_DIRECT_ACCESS_REGWEN_RESVAL = 1'h 1; + parameter logic [0:0] OTP_CTRL_DIRECT_ACCESS_REGWEN_DIRECT_ACCESS_REGWEN_RESVAL = 1'h 1; + parameter logic [2:0] OTP_CTRL_DIRECT_ACCESS_CMD_RESVAL = 3'h 0; + parameter logic [0:0] OTP_CTRL_DIRECT_ACCESS_CMD_RD_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_DIRECT_ACCESS_CMD_WR_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_DIRECT_ACCESS_CMD_DIGEST_RESVAL = 1'h 0; + parameter logic [31:0] OTP_CTRL_DIRECT_ACCESS_RDATA_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_DIRECT_ACCESS_RDATA_0_DIRECT_ACCESS_RDATA_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_DIRECT_ACCESS_RDATA_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_DIRECT_ACCESS_RDATA_1_DIRECT_ACCESS_RDATA_1_RESVAL = 32'h 0; + parameter logic [1:0] OTP_CTRL_CHECK_TRIGGER_RESVAL = 2'h 0; + parameter logic [0:0] OTP_CTRL_CHECK_TRIGGER_INTEGRITY_RESVAL = 1'h 0; + parameter logic [0:0] OTP_CTRL_CHECK_TRIGGER_CONSISTENCY_RESVAL = 1'h 0; + parameter logic [31:0] OTP_CTRL_VENDOR_TEST_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_VENDOR_TEST_DIGEST_0_VENDOR_TEST_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_VENDOR_TEST_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_VENDOR_TEST_DIGEST_1_VENDOR_TEST_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_CREATOR_SW_CFG_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_CREATOR_SW_CFG_DIGEST_0_CREATOR_SW_CFG_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_CREATOR_SW_CFG_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_CREATOR_SW_CFG_DIGEST_1_CREATOR_SW_CFG_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_OWNER_SW_CFG_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_OWNER_SW_CFG_DIGEST_0_OWNER_SW_CFG_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_OWNER_SW_CFG_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_OWNER_SW_CFG_DIGEST_1_OWNER_SW_CFG_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_ROT_CREATOR_AUTH_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_ROT_CREATOR_AUTH_DIGEST_0_ROT_CREATOR_AUTH_DIGEST_0_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_ROT_CREATOR_AUTH_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_ROT_CREATOR_AUTH_DIGEST_1_ROT_CREATOR_AUTH_DIGEST_1_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_ROT_OWNER_AUTH_SLOT0_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_ROT_OWNER_AUTH_SLOT0_DIGEST_0_ROT_OWNER_AUTH_SLOT0_DIGEST_0_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_ROT_OWNER_AUTH_SLOT0_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_ROT_OWNER_AUTH_SLOT0_DIGEST_1_ROT_OWNER_AUTH_SLOT0_DIGEST_1_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_ROT_OWNER_AUTH_SLOT1_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_ROT_OWNER_AUTH_SLOT1_DIGEST_0_ROT_OWNER_AUTH_SLOT1_DIGEST_0_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_ROT_OWNER_AUTH_SLOT1_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_ROT_OWNER_AUTH_SLOT1_DIGEST_1_ROT_OWNER_AUTH_SLOT1_DIGEST_1_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_DIGEST_0_PLAT_INTEG_AUTH_SLOT0_DIGEST_0_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_DIGEST_1_PLAT_INTEG_AUTH_SLOT0_DIGEST_1_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_DIGEST_0_PLAT_INTEG_AUTH_SLOT1_DIGEST_0_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_DIGEST_1_PLAT_INTEG_AUTH_SLOT1_DIGEST_1_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_DIGEST_0_PLAT_OWNER_AUTH_SLOT0_DIGEST_0_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_DIGEST_1_PLAT_OWNER_AUTH_SLOT0_DIGEST_1_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_DIGEST_0_PLAT_OWNER_AUTH_SLOT1_DIGEST_0_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_DIGEST_1_PLAT_OWNER_AUTH_SLOT1_DIGEST_1_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_DIGEST_0_PLAT_OWNER_AUTH_SLOT2_DIGEST_0_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_DIGEST_1_PLAT_OWNER_AUTH_SLOT2_DIGEST_1_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_DIGEST_0_PLAT_OWNER_AUTH_SLOT3_DIGEST_0_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] + OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_DIGEST_1_PLAT_OWNER_AUTH_SLOT3_DIGEST_1_RESVAL = + 32'h 0; + parameter logic [31:0] OTP_CTRL_ROM_PATCH_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_ROM_PATCH_DIGEST_0_ROM_PATCH_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_ROM_PATCH_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_ROM_PATCH_DIGEST_1_ROM_PATCH_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_HW_CFG0_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_HW_CFG0_DIGEST_0_HW_CFG0_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_HW_CFG0_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_HW_CFG0_DIGEST_1_HW_CFG0_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_HW_CFG1_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_HW_CFG1_DIGEST_0_HW_CFG1_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_HW_CFG1_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_HW_CFG1_DIGEST_1_HW_CFG1_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET0_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET0_DIGEST_0_SECRET0_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET0_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET0_DIGEST_1_SECRET0_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET1_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET1_DIGEST_0_SECRET1_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET1_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET1_DIGEST_1_SECRET1_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET2_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET2_DIGEST_0_SECRET2_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET2_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET2_DIGEST_1_SECRET2_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET3_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET3_DIGEST_0_SECRET3_DIGEST_0_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET3_DIGEST_1_RESVAL = 32'h 0; + parameter logic [31:0] OTP_CTRL_SECRET3_DIGEST_1_SECRET3_DIGEST_1_RESVAL = 32'h 0; + + // Window parameters for core interface + parameter logic [CoreAw-1:0] OTP_CTRL_SW_CFG_WINDOW_OFFSET = 15'h 4000; + parameter int unsigned OTP_CTRL_SW_CFG_WINDOW_SIZE = 'h 4000; + parameter int unsigned OTP_CTRL_SW_CFG_WINDOW_IDX = 0; + + // Register index for core interface + typedef enum int { + OTP_CTRL_INTR_STATE, + OTP_CTRL_INTR_ENABLE, + OTP_CTRL_INTR_TEST, + OTP_CTRL_ALERT_TEST, + OTP_CTRL_STATUS, + OTP_CTRL_ERR_CODE_0, + OTP_CTRL_ERR_CODE_1, + OTP_CTRL_ERR_CODE_2, + OTP_CTRL_ERR_CODE_3, + OTP_CTRL_ERR_CODE_4, + OTP_CTRL_ERR_CODE_5, + OTP_CTRL_ERR_CODE_6, + OTP_CTRL_ERR_CODE_7, + OTP_CTRL_ERR_CODE_8, + OTP_CTRL_ERR_CODE_9, + OTP_CTRL_ERR_CODE_10, + OTP_CTRL_ERR_CODE_11, + OTP_CTRL_ERR_CODE_12, + OTP_CTRL_ERR_CODE_13, + OTP_CTRL_ERR_CODE_14, + OTP_CTRL_ERR_CODE_15, + OTP_CTRL_ERR_CODE_16, + OTP_CTRL_ERR_CODE_17, + OTP_CTRL_ERR_CODE_18, + OTP_CTRL_ERR_CODE_19, + OTP_CTRL_ERR_CODE_20, + OTP_CTRL_ERR_CODE_21, + OTP_CTRL_ERR_CODE_22, + OTP_CTRL_ERR_CODE_23, + OTP_CTRL_DIRECT_ACCESS_REGWEN, + OTP_CTRL_DIRECT_ACCESS_CMD, + OTP_CTRL_DIRECT_ACCESS_ADDRESS, + OTP_CTRL_DIRECT_ACCESS_WDATA_0, + OTP_CTRL_DIRECT_ACCESS_WDATA_1, + OTP_CTRL_DIRECT_ACCESS_RDATA_0, + OTP_CTRL_DIRECT_ACCESS_RDATA_1, + OTP_CTRL_CHECK_TRIGGER_REGWEN, + OTP_CTRL_CHECK_TRIGGER, + OTP_CTRL_CHECK_REGWEN, + OTP_CTRL_CHECK_TIMEOUT, + OTP_CTRL_INTEGRITY_CHECK_PERIOD, + OTP_CTRL_CONSISTENCY_CHECK_PERIOD, + OTP_CTRL_VENDOR_TEST_READ_LOCK, + OTP_CTRL_CREATOR_SW_CFG_READ_LOCK, + OTP_CTRL_OWNER_SW_CFG_READ_LOCK, + OTP_CTRL_OWNERSHIP_SLOT_STATE_READ_LOCK, + OTP_CTRL_ROT_CREATOR_AUTH_READ_LOCK, + OTP_CTRL_ROT_OWNER_AUTH_SLOT0_READ_LOCK, + OTP_CTRL_ROT_OWNER_AUTH_SLOT1_READ_LOCK, + OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_READ_LOCK, + OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_READ_LOCK, + OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_READ_LOCK, + OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_READ_LOCK, + OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_READ_LOCK, + OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_READ_LOCK, + OTP_CTRL_EXT_NVM_READ_LOCK, + OTP_CTRL_ROM_PATCH_READ_LOCK, + OTP_CTRL_VENDOR_TEST_DIGEST_0, + OTP_CTRL_VENDOR_TEST_DIGEST_1, + OTP_CTRL_CREATOR_SW_CFG_DIGEST_0, + OTP_CTRL_CREATOR_SW_CFG_DIGEST_1, + OTP_CTRL_OWNER_SW_CFG_DIGEST_0, + OTP_CTRL_OWNER_SW_CFG_DIGEST_1, + OTP_CTRL_ROT_CREATOR_AUTH_DIGEST_0, + OTP_CTRL_ROT_CREATOR_AUTH_DIGEST_1, + OTP_CTRL_ROT_OWNER_AUTH_SLOT0_DIGEST_0, + OTP_CTRL_ROT_OWNER_AUTH_SLOT0_DIGEST_1, + OTP_CTRL_ROT_OWNER_AUTH_SLOT1_DIGEST_0, + OTP_CTRL_ROT_OWNER_AUTH_SLOT1_DIGEST_1, + OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_DIGEST_0, + OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_DIGEST_1, + OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_DIGEST_0, + OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_DIGEST_1, + OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_DIGEST_0, + OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_DIGEST_1, + OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_DIGEST_0, + OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_DIGEST_1, + OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_DIGEST_0, + OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_DIGEST_1, + OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_DIGEST_0, + OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_DIGEST_1, + OTP_CTRL_ROM_PATCH_DIGEST_0, + OTP_CTRL_ROM_PATCH_DIGEST_1, + OTP_CTRL_HW_CFG0_DIGEST_0, + OTP_CTRL_HW_CFG0_DIGEST_1, + OTP_CTRL_HW_CFG1_DIGEST_0, + OTP_CTRL_HW_CFG1_DIGEST_1, + OTP_CTRL_SECRET0_DIGEST_0, + OTP_CTRL_SECRET0_DIGEST_1, + OTP_CTRL_SECRET1_DIGEST_0, + OTP_CTRL_SECRET1_DIGEST_1, + OTP_CTRL_SECRET2_DIGEST_0, + OTP_CTRL_SECRET2_DIGEST_1, + OTP_CTRL_SECRET3_DIGEST_0, + OTP_CTRL_SECRET3_DIGEST_1 + } otp_ctrl_core_id_e; + + // Register width information to check illegal writes for core interface + parameter logic [3:0] OTP_CTRL_CORE_PERMIT [95] = '{ + 4'b 0001, // index[ 0] OTP_CTRL_INTR_STATE + 4'b 0001, // index[ 1] OTP_CTRL_INTR_ENABLE + 4'b 0001, // index[ 2] OTP_CTRL_INTR_TEST + 4'b 0001, // index[ 3] OTP_CTRL_ALERT_TEST + 4'b 1111, // index[ 4] OTP_CTRL_STATUS + 4'b 0001, // index[ 5] OTP_CTRL_ERR_CODE_0 + 4'b 0001, // index[ 6] OTP_CTRL_ERR_CODE_1 + 4'b 0001, // index[ 7] OTP_CTRL_ERR_CODE_2 + 4'b 0001, // index[ 8] OTP_CTRL_ERR_CODE_3 + 4'b 0001, // index[ 9] OTP_CTRL_ERR_CODE_4 + 4'b 0001, // index[10] OTP_CTRL_ERR_CODE_5 + 4'b 0001, // index[11] OTP_CTRL_ERR_CODE_6 + 4'b 0001, // index[12] OTP_CTRL_ERR_CODE_7 + 4'b 0001, // index[13] OTP_CTRL_ERR_CODE_8 + 4'b 0001, // index[14] OTP_CTRL_ERR_CODE_9 + 4'b 0001, // index[15] OTP_CTRL_ERR_CODE_10 + 4'b 0001, // index[16] OTP_CTRL_ERR_CODE_11 + 4'b 0001, // index[17] OTP_CTRL_ERR_CODE_12 + 4'b 0001, // index[18] OTP_CTRL_ERR_CODE_13 + 4'b 0001, // index[19] OTP_CTRL_ERR_CODE_14 + 4'b 0001, // index[20] OTP_CTRL_ERR_CODE_15 + 4'b 0001, // index[21] OTP_CTRL_ERR_CODE_16 + 4'b 0001, // index[22] OTP_CTRL_ERR_CODE_17 + 4'b 0001, // index[23] OTP_CTRL_ERR_CODE_18 + 4'b 0001, // index[24] OTP_CTRL_ERR_CODE_19 + 4'b 0001, // index[25] OTP_CTRL_ERR_CODE_20 + 4'b 0001, // index[26] OTP_CTRL_ERR_CODE_21 + 4'b 0001, // index[27] OTP_CTRL_ERR_CODE_22 + 4'b 0001, // index[28] OTP_CTRL_ERR_CODE_23 + 4'b 0001, // index[29] OTP_CTRL_DIRECT_ACCESS_REGWEN + 4'b 0001, // index[30] OTP_CTRL_DIRECT_ACCESS_CMD + 4'b 0011, // index[31] OTP_CTRL_DIRECT_ACCESS_ADDRESS + 4'b 1111, // index[32] OTP_CTRL_DIRECT_ACCESS_WDATA_0 + 4'b 1111, // index[33] OTP_CTRL_DIRECT_ACCESS_WDATA_1 + 4'b 1111, // index[34] OTP_CTRL_DIRECT_ACCESS_RDATA_0 + 4'b 1111, // index[35] OTP_CTRL_DIRECT_ACCESS_RDATA_1 + 4'b 0001, // index[36] OTP_CTRL_CHECK_TRIGGER_REGWEN + 4'b 0001, // index[37] OTP_CTRL_CHECK_TRIGGER + 4'b 0001, // index[38] OTP_CTRL_CHECK_REGWEN + 4'b 1111, // index[39] OTP_CTRL_CHECK_TIMEOUT + 4'b 1111, // index[40] OTP_CTRL_INTEGRITY_CHECK_PERIOD + 4'b 1111, // index[41] OTP_CTRL_CONSISTENCY_CHECK_PERIOD + 4'b 0001, // index[42] OTP_CTRL_VENDOR_TEST_READ_LOCK + 4'b 0001, // index[43] OTP_CTRL_CREATOR_SW_CFG_READ_LOCK + 4'b 0001, // index[44] OTP_CTRL_OWNER_SW_CFG_READ_LOCK + 4'b 0001, // index[45] OTP_CTRL_OWNERSHIP_SLOT_STATE_READ_LOCK + 4'b 0001, // index[46] OTP_CTRL_ROT_CREATOR_AUTH_READ_LOCK + 4'b 0001, // index[47] OTP_CTRL_ROT_OWNER_AUTH_SLOT0_READ_LOCK + 4'b 0001, // index[48] OTP_CTRL_ROT_OWNER_AUTH_SLOT1_READ_LOCK + 4'b 0001, // index[49] OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_READ_LOCK + 4'b 0001, // index[50] OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_READ_LOCK + 4'b 0001, // index[51] OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_READ_LOCK + 4'b 0001, // index[52] OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_READ_LOCK + 4'b 0001, // index[53] OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_READ_LOCK + 4'b 0001, // index[54] OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_READ_LOCK + 4'b 0001, // index[55] OTP_CTRL_EXT_NVM_READ_LOCK + 4'b 0001, // index[56] OTP_CTRL_ROM_PATCH_READ_LOCK + 4'b 1111, // index[57] OTP_CTRL_VENDOR_TEST_DIGEST_0 + 4'b 1111, // index[58] OTP_CTRL_VENDOR_TEST_DIGEST_1 + 4'b 1111, // index[59] OTP_CTRL_CREATOR_SW_CFG_DIGEST_0 + 4'b 1111, // index[60] OTP_CTRL_CREATOR_SW_CFG_DIGEST_1 + 4'b 1111, // index[61] OTP_CTRL_OWNER_SW_CFG_DIGEST_0 + 4'b 1111, // index[62] OTP_CTRL_OWNER_SW_CFG_DIGEST_1 + 4'b 1111, // index[63] OTP_CTRL_ROT_CREATOR_AUTH_DIGEST_0 + 4'b 1111, // index[64] OTP_CTRL_ROT_CREATOR_AUTH_DIGEST_1 + 4'b 1111, // index[65] OTP_CTRL_ROT_OWNER_AUTH_SLOT0_DIGEST_0 + 4'b 1111, // index[66] OTP_CTRL_ROT_OWNER_AUTH_SLOT0_DIGEST_1 + 4'b 1111, // index[67] OTP_CTRL_ROT_OWNER_AUTH_SLOT1_DIGEST_0 + 4'b 1111, // index[68] OTP_CTRL_ROT_OWNER_AUTH_SLOT1_DIGEST_1 + 4'b 1111, // index[69] OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_DIGEST_0 + 4'b 1111, // index[70] OTP_CTRL_PLAT_INTEG_AUTH_SLOT0_DIGEST_1 + 4'b 1111, // index[71] OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_DIGEST_0 + 4'b 1111, // index[72] OTP_CTRL_PLAT_INTEG_AUTH_SLOT1_DIGEST_1 + 4'b 1111, // index[73] OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_DIGEST_0 + 4'b 1111, // index[74] OTP_CTRL_PLAT_OWNER_AUTH_SLOT0_DIGEST_1 + 4'b 1111, // index[75] OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_DIGEST_0 + 4'b 1111, // index[76] OTP_CTRL_PLAT_OWNER_AUTH_SLOT1_DIGEST_1 + 4'b 1111, // index[77] OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_DIGEST_0 + 4'b 1111, // index[78] OTP_CTRL_PLAT_OWNER_AUTH_SLOT2_DIGEST_1 + 4'b 1111, // index[79] OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_DIGEST_0 + 4'b 1111, // index[80] OTP_CTRL_PLAT_OWNER_AUTH_SLOT3_DIGEST_1 + 4'b 1111, // index[81] OTP_CTRL_ROM_PATCH_DIGEST_0 + 4'b 1111, // index[82] OTP_CTRL_ROM_PATCH_DIGEST_1 + 4'b 1111, // index[83] OTP_CTRL_HW_CFG0_DIGEST_0 + 4'b 1111, // index[84] OTP_CTRL_HW_CFG0_DIGEST_1 + 4'b 1111, // index[85] OTP_CTRL_HW_CFG1_DIGEST_0 + 4'b 1111, // index[86] OTP_CTRL_HW_CFG1_DIGEST_1 + 4'b 1111, // index[87] OTP_CTRL_SECRET0_DIGEST_0 + 4'b 1111, // index[88] OTP_CTRL_SECRET0_DIGEST_1 + 4'b 1111, // index[89] OTP_CTRL_SECRET1_DIGEST_0 + 4'b 1111, // index[90] OTP_CTRL_SECRET1_DIGEST_1 + 4'b 1111, // index[91] OTP_CTRL_SECRET2_DIGEST_0 + 4'b 1111, // index[92] OTP_CTRL_SECRET2_DIGEST_1 + 4'b 1111, // index[93] OTP_CTRL_SECRET3_DIGEST_0 + 4'b 1111 // index[94] OTP_CTRL_SECRET3_DIGEST_1 + }; + + /////////////////////////////////////////////// + // Typedefs for registers for prim interface // + /////////////////////////////////////////////// + + typedef struct packed { + struct packed { + logic [10:0] q; + } field4; + struct packed { + logic [9:0] q; + } field3; + struct packed { + logic q; + } field2; + struct packed { + logic q; + } field1; + struct packed { + logic q; + } field0; + } otp_ctrl_reg2hw_csr0_reg_t; + + typedef struct packed { + struct packed { + logic [15:0] q; + } field4; + struct packed { + logic q; + } field3; + struct packed { + logic [6:0] q; + } field2; + struct packed { + logic q; + } field1; + struct packed { + logic [6:0] q; + } field0; + } otp_ctrl_reg2hw_csr1_reg_t; + + typedef struct packed { + logic q; + } otp_ctrl_reg2hw_csr2_reg_t; + + typedef struct packed { + struct packed { + logic q; + } field8; + struct packed { + logic q; + } field7; + struct packed { + logic q; + } field6; + struct packed { + logic q; + } field5; + struct packed { + logic q; + } field4; + struct packed { + logic q; + } field3; + struct packed { + logic q; + } field2; + struct packed { + logic [9:0] q; + } field1; + struct packed { + logic [2:0] q; + } field0; + } otp_ctrl_reg2hw_csr3_reg_t; + + typedef struct packed { + struct packed { + logic q; + } field3; + struct packed { + logic q; + } field2; + struct packed { + logic q; + } field1; + struct packed { + logic [9:0] q; + } field0; + } otp_ctrl_reg2hw_csr4_reg_t; + + typedef struct packed { + struct packed { + logic [15:0] q; + } field6; + struct packed { + logic q; + } field5; + struct packed { + logic q; + } field4; + struct packed { + logic [2:0] q; + } field3; + struct packed { + logic q; + } field2; + struct packed { + logic [1:0] q; + } field1; + struct packed { + logic [5:0] q; + } field0; + } otp_ctrl_reg2hw_csr5_reg_t; + + typedef struct packed { + struct packed { + logic [15:0] q; + } field3; + struct packed { + logic q; + } field2; + struct packed { + logic q; + } field1; + struct packed { + logic [9:0] q; + } field0; + } otp_ctrl_reg2hw_csr6_reg_t; + + typedef struct packed { + struct packed { + logic q; + } field3; + struct packed { + logic q; + } field2; + struct packed { + logic [2:0] q; + } field1; + struct packed { + logic [5:0] q; + } field0; + } otp_ctrl_reg2hw_csr7_reg_t; + + typedef struct packed { + struct packed { + logic [2:0] d; + logic de; + } field0; + struct packed { + logic [9:0] d; + logic de; + } field1; + struct packed { + logic d; + logic de; + } field2; + struct packed { + logic d; + logic de; + } field3; + struct packed { + logic d; + logic de; + } field4; + struct packed { + logic d; + logic de; + } field5; + struct packed { + logic d; + logic de; + } field6; + struct packed { + logic d; + logic de; + } field7; + struct packed { + logic d; + logic de; + } field8; + } otp_ctrl_hw2reg_csr3_reg_t; + + typedef struct packed { + struct packed { + logic [5:0] d; + logic de; + } field0; + struct packed { + logic [1:0] d; + logic de; + } field1; + struct packed { + logic d; + logic de; + } field2; + struct packed { + logic [2:0] d; + logic de; + } field3; + struct packed { + logic d; + logic de; + } field4; + struct packed { + logic d; + logic de; + } field5; + struct packed { + logic [15:0] d; + logic de; + } field6; + } otp_ctrl_hw2reg_csr5_reg_t; + + typedef struct packed { + struct packed { + logic [5:0] d; + logic de; + } field0; + struct packed { + logic [2:0] d; + logic de; + } field1; + struct packed { + logic d; + logic de; + } field2; + struct packed { + logic d; + logic de; + } field3; + } otp_ctrl_hw2reg_csr7_reg_t; + + // Register -> HW type for prim interface + typedef struct packed { + otp_ctrl_reg2hw_csr0_reg_t csr0; // [158:135] + otp_ctrl_reg2hw_csr1_reg_t csr1; // [134:103] + otp_ctrl_reg2hw_csr2_reg_t csr2; // [102:102] + otp_ctrl_reg2hw_csr3_reg_t csr3; // [101:82] + otp_ctrl_reg2hw_csr4_reg_t csr4; // [81:69] + otp_ctrl_reg2hw_csr5_reg_t csr5; // [68:39] + otp_ctrl_reg2hw_csr6_reg_t csr6; // [38:11] + otp_ctrl_reg2hw_csr7_reg_t csr7; // [10:0] + } otp_ctrl_prim_reg2hw_t; + + // HW -> register type for prim interface + typedef struct packed { + otp_ctrl_hw2reg_csr3_reg_t csr3; // [80:52] + otp_ctrl_hw2reg_csr5_reg_t csr5; // [51:15] + otp_ctrl_hw2reg_csr7_reg_t csr7; // [14:0] + } otp_ctrl_prim_hw2reg_t; + + // Register offsets for prim interface + parameter logic [PrimAw-1:0] OTP_CTRL_CSR0_OFFSET = 5'h 0; + parameter logic [PrimAw-1:0] OTP_CTRL_CSR1_OFFSET = 5'h 4; + parameter logic [PrimAw-1:0] OTP_CTRL_CSR2_OFFSET = 5'h 8; + parameter logic [PrimAw-1:0] OTP_CTRL_CSR3_OFFSET = 5'h c; + parameter logic [PrimAw-1:0] OTP_CTRL_CSR4_OFFSET = 5'h 10; + parameter logic [PrimAw-1:0] OTP_CTRL_CSR5_OFFSET = 5'h 14; + parameter logic [PrimAw-1:0] OTP_CTRL_CSR6_OFFSET = 5'h 18; + parameter logic [PrimAw-1:0] OTP_CTRL_CSR7_OFFSET = 5'h 1c; + + // Register index for prim interface + typedef enum int { + OTP_CTRL_CSR0, + OTP_CTRL_CSR1, + OTP_CTRL_CSR2, + OTP_CTRL_CSR3, + OTP_CTRL_CSR4, + OTP_CTRL_CSR5, + OTP_CTRL_CSR6, + OTP_CTRL_CSR7 + } otp_ctrl_prim_id_e; + + // Register width information to check illegal writes for prim interface + parameter logic [3:0] OTP_CTRL_PRIM_PERMIT [8] = '{ + 4'b 1111, // index[0] OTP_CTRL_CSR0 + 4'b 1111, // index[1] OTP_CTRL_CSR1 + 4'b 0001, // index[2] OTP_CTRL_CSR2 + 4'b 0111, // index[3] OTP_CTRL_CSR3 + 4'b 0011, // index[4] OTP_CTRL_CSR4 + 4'b 1111, // index[5] OTP_CTRL_CSR5 + 4'b 1111, // index[6] OTP_CTRL_CSR6 + 4'b 0011 // index[7] OTP_CTRL_CSR7 + }; + +endpackage diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_scrmbl.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_scrmbl.sv new file mode 100644 index 00000000000000..7c5296929b6f85 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_scrmbl.sv @@ -0,0 +1,512 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// This module contains the scrambling datapath for the OTP controller. It basically consists of +// two single-round PRESENT primitives (one for encryption and one for decryption mode), a counter +// with a simple FSM and four working registers, as listed below. +// +// key_state_q (128bit): working register to hold the round key (needed for the key schedule). +// +// data_state_q (64bit): working register to hold the data state in between rounds. +// +// data_shadow_q (64bit): shadow register for holding a second 64bit block of input data. This is +// used to form a 128bit data block for the digest mode, which has a block +// size of 128bit. +// +// digest_state_q (64bit): register to hold the digest state in between digest updates. Technically, +// this is not needed when the data for the digest is fed into this block +// back-to-back. However, the partition integrity checks require that it is +// possible to interleave encryption operations and digest update steps, +// hence an additional state register is needed, as otherwise the digest +// state would be lost. +// +// The scrambling datapath is arranged such that it can also be used for calculating a digest using +// the encryption primitive in a Merkle-Damgard construction. To that end, the PRESENT block cipher +// is turned into a one way function according to the Davies-Meyer scheme. Note however that this +// makes the digest block size 128bit wide, since the Merkle-Damgard construction leverages the +// cipher key input to ingest data. +// +// The scrambling datapath exposes a few simple commands and the FSM hides the complexity +// of steering the appropriate muxes and keeping track of the cipher rounds. These commands are +// briefly explained below. +// +// Decrypt: This decrypts the data block provided via data_i with the key at index sel_i. +// +// Encrypt: This encrypts the data block provided via data_i with the key at index sel_i. +// In addition, this command copies the prvious result into a shadow register before +// the first encryption round for later use in the digest (see description further below). +// This enables interleaved encrypt/digest operation needed for the integrity checks of +// the secret partitions. +// +// LoadShadow: In "StandardMode", the LoadShadow command loads the data provided via data_i into a +// shadow register that is mapped to the lower 64bit of the 128bit digest input data +// block. In "ChainedMode", this command copies the contents of the data state register +// into the shadow register. +// +// DigestInit: This ensures that the digest initialization vector (IV) is selected upon the next +// call of the Digest command. Also, mode_i can be used to set the digest mode. If +// mode_i is set to "StandardMode", the data to be digested has to be provided via +// data_i and LoadShadow. If mode_i is set to "ChainedMode", the digest input is formed +// by concatenating the results of the revious two encryption commands. +// +// Digest: In "StandardMode", this command concatenates the data input supplied via data_i with +// the shadow register in order to form a 128bit block ({data_i, data_shadow_q}). This block +// is then used to encrypt the digest state. In "ChainedMode" digest mode, the 128bit block +// to be digested is formed by concatenating {data_state_q, data_shadow_q}. If a DigestInit +// command has been executed right before calling Digest, the IV selected with sel_i is +// used to initialize the state. +// +// DigestFinalize: This command encrypts the digest state with the finalization constant selected +// by sel_i in order to form the final digest. +// +// References: +// - https://docs.opentitan.org/hw/ip/otp_ctrl/doc/theory_of_operation.html#block-diagram +// - https://docs.opentitan.org/hw/ip/prim/doc/prim_present/ +// - https://en.wikipedia.org/wiki/Merkle-Damgard_construction +// - https://en.wikipedia.org/wiki/One-way_compression_function#Davies%E2%80%93Meyer +// - https://en.wikipedia.org/wiki/PRESENT +// - http://www.lightweightcrypto.org/present/present_ches2007.pdf +// + +`include "prim_flop_macros.sv" + +module otp_ctrl_scrmbl + import otp_ctrl_pkg::*; + import otp_ctrl_top_specific_pkg::*; + import otp_ctrl_part_pkg::*; +( + input clk_i, + input rst_ni, + // input data and command + input otp_scrmbl_cmd_e cmd_i, + input digest_mode_e mode_i, + input [ConstSelWidth-1:0] sel_i, + input [ScrmblBlockWidth-1:0] data_i, + input valid_i, + output logic ready_o, + // output data + output logic [ScrmblBlockWidth-1:0] data_o, + output logic valid_o, + // escalation input and FSM error indication + input lc_ctrl_pkg::lc_tx_t escalate_en_i, + output logic fsm_err_o +); + + import prim_util_pkg::vbits; + + //////////////////////// + // Decryption Key LUT // + //////////////////////// + + // Anchor keys, constants and IVs + key_array_t rnd_cnst_key_anchor; + digest_const_array_t rnd_cnst_digest_anchor; + digest_iv_array_t rnd_cnst_digest_iv_anchor; + + for (genvar i = 0; i < NumScrmblKeys; i++) begin : gen_anchor_keys + prim_sec_anchor_buf #( + .Width(ScrmblKeyWidth) + ) u_key_anchor_buf ( + .in_i(RndCnstKey[i]), + .out_o(rnd_cnst_key_anchor[i]) + ); + end + + for (genvar i = 0; i < NumDigestSets; i++) begin : gen_anchor_digests + prim_sec_anchor_buf #( + .Width(ScrmblKeyWidth) + ) u_const_anchor_buf ( + .in_i(RndCnstDigestConst[i]), + .out_o(rnd_cnst_digest_anchor[i]) + ); + + prim_sec_anchor_buf #( + .Width(ScrmblBlockWidth) + ) u_iv_anchor_buf ( + .in_i(RndCnstDigestIV[i]), + .out_o(rnd_cnst_digest_iv_anchor[i]) + ); + end + + + // Align these arrays to power of 2's to prevent X's in the muxing operations further below. + logic [2**$clog2(NumScrmblKeys)-1:0][ScrmblKeyWidth-1:0] otp_enc_key_lut; + logic [2**$clog2(NumScrmblKeys)-1:0][ScrmblKeyWidth-1:0] otp_dec_key_lut; + logic [2**$clog2(NumDigestSets)-1:0][ScrmblKeyWidth-1:0] digest_const_lut; + logic [2**$clog2(NumDigestSets)-1:0][ScrmblBlockWidth-1:0] digest_iv_lut; + + // This pre-calculates the inverse scrambling keys at elab time. + `ASSERT_INIT(NumMaxPresentRounds_A, NumPresentRounds <= 31) + + always_comb begin : p_luts + otp_enc_key_lut = '0; + otp_dec_key_lut = '0; + digest_const_lut = '0; + digest_iv_lut = '0; + + for (int k = 0; k < NumScrmblKeys; k++) begin + localparam logic [4:0] NumRounds = 5'(unsigned'(NumPresentRounds)); + otp_enc_key_lut[k] = rnd_cnst_key_anchor[k]; + // Due to the PRESENT key schedule, we have to step the key schedule function by + // NumPresentRounds forwards to get the decryption key. + otp_dec_key_lut[k] = + prim_cipher_pkg::present_get_dec_key128(rnd_cnst_key_anchor[k], NumRounds); + end + + for (int k = 0; k < NumDigestSets; k++) begin + digest_const_lut[k] = rnd_cnst_digest_anchor[k]; + digest_iv_lut[k] = rnd_cnst_digest_iv_anchor[k]; + end + end + `ASSERT_KNOWN(EncKeyLutKnown_A, otp_enc_key_lut) + `ASSERT_KNOWN(DecKeyLutKnown_A, otp_dec_key_lut) + `ASSERT_KNOWN(DigestConstLutKnown_A, digest_const_lut) + `ASSERT_KNOWN(DigestIvLutKnown_A, digest_iv_lut) + + ////////////// + // Datapath // + ////////////// + + logic [4:0] idx_state_d, idx_state_q; + logic [ScrmblKeyWidth-1:0] key_state_d, key_state_q; + logic [ScrmblBlockWidth-1:0] data_state_d, data_state_q, data_shadow_q; + logic [ScrmblBlockWidth-1:0] digest_state_d, digest_state_q; + logic [ScrmblBlockWidth-1:0] enc_data_out, enc_data_out_xor, dec_data_out; + logic [ScrmblKeyWidth-1:0] dec_key_out, enc_key_out; + logic [4:0] dec_idx_out, enc_idx_out; + logic [ScrmblKeyWidth-1:0] otp_digest_const_mux, otp_enc_key_mux, otp_dec_key_mux; + logic [ScrmblBlockWidth-1:0] otp_digest_iv_mux; + + typedef enum logic [2:0] {SelEncDataOut, + SelDecDataOut, + SelDigestState, + SelEncDataOutXor, + SelDataInput} data_state_sel_e; + + typedef enum logic [2:0] {SelDecKeyOut, + SelEncKeyOut, + SelDecKeyInit, + SelEncKeyInit, + SelDigestConst, + SelDigestInput, + SelDigestChained} key_state_sel_e; + + logic digest_init; + data_state_sel_e data_state_sel; + key_state_sel_e key_state_sel; + logic data_state_en, data_shadow_copy, data_shadow_load, digest_state_en, key_state_en; + digest_mode_e digest_mode_d, digest_mode_q; + + assign otp_enc_key_mux = otp_enc_key_lut[ScrmblKeySelWidth'(sel_i)]; + assign otp_dec_key_mux = otp_dec_key_lut[ScrmblKeySelWidth'(sel_i)]; + assign otp_digest_const_mux = digest_const_lut[DigestSetSelWidth'(sel_i)]; + assign otp_digest_iv_mux = digest_iv_lut[DigestSetSelWidth'(sel_i)]; + + // Make sure we always select a valid key / digest constant. + `ASSERT(CheckNumEncKeys_A, key_state_sel == SelEncKeyInit |-> sel_i < NumScrmblKeys) + `ASSERT(CheckNumDecKeys_A, key_state_sel == SelDecKeyInit |-> sel_i < NumScrmblKeys) + `ASSERT(CheckNumDigest1_A, key_state_sel == SelDigestConst |-> sel_i < NumDigestSets) + + assign data_state_d = (data_state_sel == SelEncDataOut) ? enc_data_out : + (data_state_sel == SelDecDataOut) ? dec_data_out : + (data_state_sel == SelDigestState) ? digest_state_q : + (data_state_sel == SelEncDataOutXor) ? enc_data_out_xor : + data_i; + + assign key_state_d = (key_state_sel == SelDecKeyOut) ? dec_key_out : + (key_state_sel == SelEncKeyOut) ? enc_key_out : + (key_state_sel == SelDecKeyInit) ? otp_dec_key_mux : + (key_state_sel == SelEncKeyInit) ? otp_enc_key_mux : + (key_state_sel == SelDigestConst) ? otp_digest_const_mux : + (key_state_sel == SelDigestChained) ? {data_state_q, data_shadow_q} : + {data_i, data_shadow_q}; + + // Initialize the round index state with 1 in all cases, except for the decrypt operation. + assign idx_state_d = (key_state_sel == SelDecKeyOut) ? dec_idx_out : + (key_state_sel == SelEncKeyOut) ? enc_idx_out : + (key_state_sel == SelDecKeyInit) ? unsigned'(5'(NumPresentRounds)) : + 5'd1; + + // The XOR is for the Davies-Mayer one-way function construction. + assign enc_data_out_xor = enc_data_out ^ digest_state_q; + assign digest_state_d = (digest_init) ? otp_digest_iv_mux : enc_data_out_xor; + + logic valid_q; //valid_d defined below + assign data_o = (valid_q) ? data_state_q : 0; + + ///////// + // FSM // + ///////// + + // SEC_CM: SCRMBL.FSM.SPARSE + // Encoding generated with: + // $ ./util/design/sparse-fsm-encode.py -d 5 -m 5 -n 9 \ + // -s 2193087944 --language=sv + // + // Hamming distance histogram: + // + // 0: -- + // 1: -- + // 2: -- + // 3: -- + // 4: -- + // 5: |||||||||||||||||||| (60.00%) + // 6: ||||||||||||| (40.00%) + // 7: -- + // 8: -- + // 9: -- + // + // Minimum Hamming distance: 5 + // Maximum Hamming distance: 6 + // Minimum Hamming weight: 4 + // Maximum Hamming weight: 7 + // + localparam int StateWidth = 9; + typedef enum logic [StateWidth-1:0] { + IdleSt = 9'b100011001, + DecryptSt = 9'b101101111, + EncryptSt = 9'b010010111, + DigestSt = 9'b111000010, + ErrorSt = 9'b011111000 + } state_e; + + localparam int CntWidth = $clog2(NumPresentRounds+1); + localparam int unsigned LastPresentRoundInt = NumPresentRounds - 1; + localparam bit [CntWidth-1:0] LastPresentRound = LastPresentRoundInt[CntWidth-1:0]; + + state_e state_d, state_q; + logic [CntWidth-1:0] cnt; + logic cnt_clr, cnt_en, cnt_err; + logic valid_d; //valid_q defined above + + assign valid_o = valid_q; + + // SEC_CM: SCRMBL.CTR.REDUN + prim_count #( + .Width(CntWidth) + ) u_prim_count ( + .clk_i, + .rst_ni, + .clr_i(cnt_clr), + .set_i(1'b0), + .set_cnt_i('0), + .incr_en_i(cnt_en), + .decr_en_i(1'b0), + .step_i(CntWidth'(1)), + .commit_i(1'b1), + .cnt_o(cnt), + .cnt_after_commit_o(), + .err_o(cnt_err) + ); + + always_comb begin : p_fsm + state_d = state_q; + digest_mode_d = digest_mode_q; + data_state_sel = SelDataInput; + key_state_sel = SelDigestInput; + digest_init = 1'b0; + data_state_en = 1'b0; + data_shadow_copy = 1'b0; + data_shadow_load = 1'b0; + key_state_en = 1'b0; + digest_state_en = 1'b0; + cnt_en = 1'b0; + cnt_clr = 1'b0; + valid_d = 1'b0; + ready_o = 1'b0; + fsm_err_o = 1'b0; + + unique case (state_q) + /////////////////////////////////////////////////////////////////// + // Idle State: decode command and + // load working regs accordingly + IdleSt: begin + cnt_clr = 1'b1; + ready_o = 1'b1; + + if (valid_i) begin + unique case (cmd_i) + Decrypt: begin + state_d = DecryptSt; + key_state_sel = SelDecKeyInit; + data_state_en = 1'b1; + key_state_en = 1'b1; + end + Encrypt: begin + state_d = EncryptSt; + key_state_sel = SelEncKeyInit; + data_state_en = 1'b1; + key_state_en = 1'b1; + end + LoadShadow: begin + if (digest_mode_q == ChainedMode) begin + data_shadow_copy = 1'b1; + end else begin + data_shadow_load = 1'b1; + end + end + Digest: begin + state_d = DigestSt; + data_state_sel = SelDigestState; + key_state_sel = (digest_mode_q == ChainedMode) ? SelDigestChained : SelDigestInput; + data_state_en = 1'b1; + key_state_en = 1'b1; + end + DigestInit: begin + digest_mode_d = mode_i; + digest_init = 1'b1; + digest_state_en = 1'b1; + end + DigestFinalize: begin + state_d = DigestSt; + data_state_sel = SelDigestState; + key_state_sel = SelDigestConst; + data_state_en = 1'b1; + key_state_en = 1'b1; + digest_mode_d = StandardMode; + end + default: ; // ignore + endcase // cmd_i + end + end + /////////////////////////////////////////////////////////////////// + // Perform decrypt rounds. + DecryptSt: begin + data_state_sel = SelDecDataOut; + key_state_sel = SelDecKeyOut; + data_state_en = 1'b1; + key_state_en = 1'b1; + cnt_en = 1'b1; + if (cnt == LastPresentRound) begin + state_d = IdleSt; + valid_d = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Perform encrypt rounds. + EncryptSt: begin + data_state_sel = SelEncDataOut; + key_state_sel = SelEncKeyOut; + data_state_en = 1'b1; + key_state_en = 1'b1; + cnt_en = 1'b1; + if (cnt == LastPresentRound) begin + state_d = IdleSt; + valid_d = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // The digest is calculated with a Merkle-Damgard construction that + // employs the PRESENT encryption datapath. + DigestSt: begin + data_state_sel = SelEncDataOut; + key_state_sel = SelEncKeyOut; + data_state_en = 1'b1; + key_state_en = 1'b1; + cnt_en = 1'b1; + if (cnt == LastPresentRound) begin + state_d = IdleSt; + valid_d = 1'b1; + // Apply XOR for Davies-Meyer construction. + data_state_sel = SelEncDataOutXor; + // Backup digest state for next round of updates. We can't keep this state in the + // data state register as a digest may be calculated together with encryption + // operations in an interleaved way. + digest_state_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Terminal error state. This raises an alert. + ErrorSt: begin + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + // This should never happen, hence we directly jump into the + // error state, where an alert will be triggered. + default: begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + endcase // state_q + + // Unconditionally jump into the terminal error state in case of escalation. + // SEC_CM: SCRMBL.FSM.LOCAL_ESC, SCRMBL.FSM.GLOBAL_ESC + if (lc_ctrl_pkg::lc_tx_test_true_loose(escalate_en_i) || cnt_err) begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + end + + ///////////////////////////// + // PRESENT DEC/ENC Modules // + ///////////////////////////// + + prim_present #( + .KeyWidth(128), + .NumRounds(NumPresentRounds), + .NumPhysRounds(1) + ) u_prim_present_enc ( + .data_i ( data_state_q ), + .key_i ( key_state_q ), + .idx_i ( idx_state_q ), + .data_o ( enc_data_out ), + .key_o ( enc_key_out ), + .idx_o ( enc_idx_out ) + ); + + prim_present #( + .KeyWidth(128), + // We are using an iterative full-round implementation here. + .NumRounds(NumPresentRounds), + .NumPhysRounds(1), + .Decrypt(1) + ) u_prim_present_dec ( + .data_i ( data_state_q ), + .key_i ( key_state_q ), + .idx_i ( idx_state_q ), + .data_o ( dec_data_out ), + .key_o ( dec_key_out ), + .idx_o ( dec_idx_out ) + ); + + /////////////// + // Registers // + /////////////// + + `PRIM_FLOP_SPARSE_FSM(u_state_regs, state_d, state_q, state_e, IdleSt) + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + key_state_q <= '0; + idx_state_q <= '0; + data_state_q <= '0; + data_shadow_q <= '0; + digest_state_q <= '0; + valid_q <= 1'b0; + digest_mode_q <= StandardMode; + end else begin + valid_q <= valid_d; + digest_mode_q <= digest_mode_d; + + // enable regs + if (key_state_en) begin + key_state_q <= key_state_d; + idx_state_q <= idx_state_d; + end + if (data_state_en) begin + data_state_q <= data_state_d; + end + if (data_shadow_copy) begin + data_shadow_q <= data_state_q; + end else if (data_shadow_load) begin + data_shadow_q <= data_state_d; + end + if (digest_state_en) begin + digest_state_q <= digest_state_d; + end + end + end + +endmodule : otp_ctrl_scrmbl diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_token_const.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_token_const.sv new file mode 100644 index 00000000000000..2443cd8a11d61a --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_token_const.sv @@ -0,0 +1,65 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// This module contains the hash post-image constants for the all-zero and raw unlock tokens. +// This implementation relies on constant propagation to precompute these constants from the +// random netlist constants at compile time, and hence does not contain any "real" logic. + +module otp_ctrl_token_const import otp_ctrl_pkg::*; #( + // Compile time random constants, to be overriden by topgen. + parameter digest_const_array_t RndCnstDigestConst = RndCnstDigestConstDefault, + parameter digest_iv_array_t RndCnstDigestIV = RndCnstDigestIVDefault, + parameter lc_ctrl_pkg::lc_token_t RndCnstRawUnlockToken = RndCnstRawUnlockTokenDefault +) ( + output lc_ctrl_pkg::lc_token_t all_zero_token_hashed_o, + output lc_ctrl_pkg::lc_token_t raw_unlock_token_hashed_o +); + + localparam int NumHashes = 2; + localparam int AllZeroIdx = 0; + localparam int RawUnlockIdx = 1; + + logic [NumHashes-1:0][1:0][ScrmblKeyWidth-1:0] data; + logic [NumHashes-1:0][4:0][ScrmblBlockWidth-1:0] state; + + // First digest is for the all zero token, the second is for the raw unlock token. + assign data[AllZeroIdx][0] = '0; + assign data[RawUnlockIdx][0] = RndCnstRawUnlockToken; + + // Repeat for all precomputed hashes. + for (genvar j = 0; j < NumHashes; j++) begin : gen_hashes + // Initialize all hashes with digest IV. + assign state[j][0] = RndCnstDigestIV[LcRawDigest]; + // Second data block is always the digest finalization constant. + assign data[j][1] = RndCnstDigestConst[LcRawDigest]; + + // Each hash takes four invocations, see diagram c) on + // https://docs.opentitan.org/hw/ip/otp_ctrl/doc/theory_of_operation.html#scrambling-datapath + for (genvar k = 0; k < 4; k++) begin : gen_invocations + logic [ScrmblBlockWidth-1:0] next_state; + + // This relies on constant propagation to + // statically precompute the hashed token values. + prim_present #( + .KeyWidth(128), + .NumRounds(NumPresentRounds) + ) u_prim_present_enc_0 ( + .data_i ( state[j][k] ), + .key_i ( data[j][k%2] ), + .idx_i ( 5'h1 ), + .data_o ( next_state ), + .key_o ( ), + .idx_o ( ) + ); + + // XOR in last state according to the Davies-Meyer scheme. + assign state[j][k+1] = next_state ^ state[j][k]; + end + end + + // Concatenate the two 64bit hash results to form the final digests. + assign all_zero_token_hashed_o = {state[AllZeroIdx][4], state[AllZeroIdx][2]}; + assign raw_unlock_token_hashed_o = {state[RawUnlockIdx][4], state[RawUnlockIdx][2]}; + +endmodule : otp_ctrl_token_const diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv new file mode 100644 index 00000000000000..e9c41a99d920a2 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv @@ -0,0 +1,108 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// + +// This package contains top specific and implementation items. +// +// It imports otp_ctrl_reg_pkg, which is generated from the top specific hjson file. +// The items that are generic and used by top-independent IPs are placed in otp_ctrl_pkg. +package otp_ctrl_top_specific_pkg; + + import prim_util_pkg::vbits; + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + + //////////////////////// + // General Parameters // + //////////////////////// + + // Width of entropy input + parameter int EdnDataWidth = 64; + + parameter int NumPartWidth = vbits(NumPart); + + parameter int SwWindowAddrWidth = vbits(NumSwCfgWindowWords); + + // Background check timer LFSR width. + parameter int LfsrWidth = 40; + // The LFSR will be reseeded once LfsrUsageThreshold + // values have been drawn from it. + parameter int LfsrUsageThreshold = 16; + + // Redundantly encoded and complementary values are used to for signalling to the partition + // controller FSMs and the DAI whether a partition is locked or not. Any other value than + // "Mubi8Lo" is interpreted as "Locked" in those FSMs. + typedef struct packed { + prim_mubi_pkg::mubi8_t read_lock; + prim_mubi_pkg::mubi8_t write_lock; + } part_access_t; + + parameter int DaiCmdWidth = 3; + typedef enum logic [DaiCmdWidth-1:0] { + DaiRead = 3'b001, + DaiWrite = 3'b010, + DaiDigest = 3'b100 + } dai_cmd_e; + + ////////////////////////////////////// + // Typedefs for OTP Macro Interface // + ////////////////////////////////////// + + // OTP-macro specific + parameter int OtpWidth = 16; + parameter int OtpAddrWidth = OtpByteAddrWidth - $clog2(OtpWidth/8); + parameter int OtpDepth = 2**OtpAddrWidth; + parameter int OtpSizeWidth = 2; // Allows to transfer up to 4 native OTP words at once. + parameter int OtpErrWidth = 3; + parameter int OtpIfWidth = 2**OtpSizeWidth*OtpWidth; + // Number of Byte address bits to cut off in order to get the native OTP word address. + parameter int OtpAddrShift = OtpByteAddrWidth - OtpAddrWidth; + + typedef enum logic [OtpErrWidth-1:0] { + NoError = 3'h0, + MacroError = 3'h1, + MacroEccCorrError = 3'h2, + MacroEccUncorrError = 3'h3, + MacroWriteBlankError = 3'h4, + AccessError = 3'h5, + CheckFailError = 3'h6, + FsmStateError = 3'h7 + } otp_err_e; + + ///////////////////////////////// + // Typedefs for OTP Scrambling // + ///////////////////////////////// + + parameter int NumPresentRounds = 31; + parameter int ScrmblBlockHalfWords = ScrmblBlockWidth / OtpWidth; + + typedef enum logic [2:0] { + Decrypt, + Encrypt, + LoadShadow, + Digest, + DigestInit, + DigestFinalize + } otp_scrmbl_cmd_e; + + /////////////////////////////////////////// + // Defaults for random netlist constants // + /////////////////////////////////////////// + + // These LFSR parameters have been generated with + // $ util/design/gen-lfsr-seed.py --width 40 --seed 4247488366 + typedef logic [LfsrWidth-1:0] lfsr_seed_t; + typedef logic [LfsrWidth-1:0][$clog2(LfsrWidth)-1:0] lfsr_perm_t; + localparam lfsr_seed_t RndCnstLfsrSeedDefault = 40'h453d28ea98; + localparam lfsr_perm_t RndCnstLfsrPermDefault = + 240'h4235171482c225f79289b32181a0163a760355d3447063d16661e44c12a5; + + typedef struct packed { + sram_key_t key; + sram_nonce_t nonce; + } scrmbl_key_init_t; + localparam scrmbl_key_init_t RndCnstScrmblKeyInitDefault = + 256'hcebeb96ffe0eced795f8b2cfe23c1e519e4fa08047a6bcfb811b04f0a479006e; + +endpackage : otp_ctrl_top_specific_pkg diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/syn/constraints.sdc b/hw/top_darjeeling/ip_autogen/otp_ctrl/syn/constraints.sdc new file mode 100644 index 00000000000000..0b119da9708326 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/syn/constraints.sdc @@ -0,0 +1,50 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# Generic constraints file for simple testsynthesis flow + +# note that we do not fix hold timing in this flow +set SETUP_CLOCK_UNCERTAINTY 0.5 + +##################### +# main clock # +##################### +set MAIN_CLK_PIN clk_i +set MAIN_RST_PIN rst_ni +# set main clock to 125 MHz +set MAIN_TCK 8.0 +set_ideal_network ${MAIN_CLK_PIN} +set_ideal_network ${MAIN_RST_PIN} +set_clock_uncertainty ${SETUP_CLOCK_UNCERTAINTY} ${MAIN_CLK_PIN} + +# other timing constraint in ns +set IN_DEL 1.0 +set OUT_DEL 1.0 +set DELAY ${MAIN_TCK} + +create_clock ${MAIN_CLK_PIN} -period ${MAIN_TCK} + +# in to out +set_max_delay ${DELAY} -from [all_inputs] -to [all_outputs] +# in to reg / reg to out +set_input_delay ${IN_DEL} [remove_from_collection [all_inputs] {${MAIN_CLK_PIN}}] -clock ${MAIN_CLK_PIN} +set_output_delay ${OUT_DEL} [all_outputs] -clock ${MAIN_CLK_PIN} + +##################### +# I/O drive/load # +##################### + +# attach load and drivers to IOs to get a more realistic estimate +set_driving_cell -no_design_rule -lib_cell ${DRIVING_CELL} -pin ${DRIVING_CELL_PIN} [all_inputs] +set_load [load_of ${LOAD_CELL_LIB}/${LOAD_CELL}/${LOAD_CELL_PIN}] [all_outputs] + +# set a nonzero critical range to be able to spot the violating paths better +# in the report +set_critical_range 0.5 ${DUT} + +##################### +# Size Only Cells # +##################### + +set_size_only -all_instances [get_cells -h *u_size_only*] true diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson b/hw/top_darjeeling/ip_autogen/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson new file mode 100644 index 00000000000000..46a3dff7d11025 --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson @@ -0,0 +1,24 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +{ + // Top level dut name (sv module). + name: otp_ctrl + + // Fusesoc core file used for building the file list. + fusesoc_core: lowrisc:opentitan:top_darjeeling_{name}:0.1 + + import_cfgs: [// Project wide common GTECH synthesis config file + "{proj_root}/hw/syn/tools/dvsim/common_gtech_syn_cfg.hjson"] + + overrides: [ + { + name: design_level + value: "top" + } + { // Deletes black-boxed hierarchies before writing out the unmapped netlist + name: post_elab_script + value: "{proj_root}/hw/top_darjeeling/ip_autogen/{name}/syn/post_elab_gtech.tcl" + } + ] +} diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson b/hw/top_darjeeling/ip_autogen/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson new file mode 100644 index 00000000000000..2afd7ca053bd8b --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson @@ -0,0 +1,26 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +{ + // Top level dut name (sv module). + name: otp_ctrl + + // Fusesoc core file used for building the file list. + fusesoc_core: lowrisc:opentitan:top_darjeeling_{name}:0.1 + + import_cfgs: [// Project wide common synthesis config file + "{proj_root}/hw/syn/tools/dvsim/common_syn_cfg.hjson"] + + overrides: [ + { + name: design_level + value: "top" + } + ] + + // Timing constraints for this module + sdc_file: "{proj_root}/hw/top_darjeeling/ip_autogen/{name}/syn/constraints.sdc" + + // This is not needed for this module + foundry_sdc_file: "" +} diff --git a/hw/top_darjeeling/ip_autogen/otp_ctrl/syn/post_elab_gtech.tcl b/hw/top_darjeeling/ip_autogen/otp_ctrl/syn/post_elab_gtech.tcl new file mode 100644 index 00000000000000..abe1a8c181f27e --- /dev/null +++ b/hw/top_darjeeling/ip_autogen/otp_ctrl/syn/post_elab_gtech.tcl @@ -0,0 +1,9 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# Post elab script, used in GTECH runs to modify the unmapped netlist before +# writing it out. + +# Remove generic views of ram macros +remove_design prim_generic_ram_1p_Width* diff --git a/hw/top_darjeeling/ip_autogen/pwrmgr/doc/theory_of_operation.md b/hw/top_darjeeling/ip_autogen/pwrmgr/doc/theory_of_operation.md index 5f197a894c9e36..c46bc74cd1697f 100644 --- a/hw/top_darjeeling/ip_autogen/pwrmgr/doc/theory_of_operation.md +++ b/hw/top_darjeeling/ip_autogen/pwrmgr/doc/theory_of_operation.md @@ -69,7 +69,7 @@ Instead the system goes into a terminal non-responsive state where a user or hos The fast clock domain FSM (referred to as fast FSM from here on) resets to `Low Power` state and waits for a power-up request from the slow FSM. Once received, the fast FSM releases the life cycle reset stage (see [reset controller](../../rstmgr/README.md) for more details). -This allows the [OTP](../../../../ip/otp_ctrl/README.md) to begin sensing. +This allows the [OTP](../../otp_ctrl/README.md) to begin sensing. Once OTP sensing completes, the life cycle controller is initialized. The initialization of the life cycle controller puts the device into its allowed operating state (see [life cycle controller](../../../../ip/lc_ctrl/README.md) for more details). diff --git a/hw/top_darjeeling/lint/top_darjeeling_dv_lint_cfgs.hjson b/hw/top_darjeeling/lint/top_darjeeling_dv_lint_cfgs.hjson index cb2b6f6fb67630..dae7527b0c4c76 100644 --- a/hw/top_darjeeling/lint/top_darjeeling_dv_lint_cfgs.hjson +++ b/hw/top_darjeeling/lint/top_darjeeling_dv_lint_cfgs.hjson @@ -111,9 +111,9 @@ rel_path: "hw/ip/lc_ctrl/dv/lint/{tool}" }, { name: otp_ctrl - fusesoc_core: lowrisc:dv:otp_ctrl_sim + fusesoc_core: lowrisc:opentitan:top_darjeeling_otp_ctrl_sim import_cfgs: ["{proj_root}/hw/lint/tools/dvsim/common_lint_cfg.hjson"] - rel_path: "hw/ip/otp_ctrl/dv/lint/{tool}" + rel_path: "hw/top_darjeeling/ip_autogen/otp_ctrl/dv/lint/{tool}" }, { name: pattgen fusesoc_core: lowrisc:,dv:pattgen_sim, diff --git a/hw/top_darjeeling/lint/top_darjeeling_lint_cfgs.hjson b/hw/top_darjeeling/lint/top_darjeeling_lint_cfgs.hjson index ab9df523c9f629..1de13e54183a17 100644 --- a/hw/top_darjeeling/lint/top_darjeeling_lint_cfgs.hjson +++ b/hw/top_darjeeling/lint/top_darjeeling_lint_cfgs.hjson @@ -142,9 +142,9 @@ rel_path: "hw/ip/otbn/lint/{tool}" }, { name: otp_ctrl - fusesoc_core: lowrisc:ip:otp_ctrl + fusesoc_core: lowrisc:opentitan:top_darjeeling_otp_ctrl import_cfgs: ["{proj_root}/hw/lint/tools/dvsim/common_lint_cfg.hjson"] - rel_path: "hw/ip/otp_ctrl/lint/{tool}" + rel_path: "hw/top_darjeeling/ip_autogen/otp_ctrl/lint/{tool}" overrides: [ { name: design_level diff --git a/hw/top_darjeeling/rtl/autogen/top_darjeeling_pkg.sv b/hw/top_darjeeling/rtl/autogen/top_darjeeling_pkg.sv index 2bca37fe28d6d8..af0c3a8768b05f 100644 --- a/hw/top_darjeeling/rtl/autogen/top_darjeeling_pkg.sv +++ b/hw/top_darjeeling/rtl/autogen/top_darjeeling_pkg.sv @@ -69,7 +69,7 @@ package top_darjeeling_pkg; /** * Peripheral size in bytes for core device on otp_ctrl in top darjeeling. */ - parameter int unsigned TOP_DARJEELING_OTP_CTRL_CORE_SIZE_BYTES = 32'h1000; + parameter int unsigned TOP_DARJEELING_OTP_CTRL_CORE_SIZE_BYTES = 32'h8000; /** * Peripheral base address for prim device on otp_ctrl in top darjeeling. diff --git a/hw/top_darjeeling/rtl/autogen/top_darjeeling_rnd_cnst_pkg.sv b/hw/top_darjeeling/rtl/autogen/top_darjeeling_rnd_cnst_pkg.sv index 21c7f2e32754a4..eb1ac2ff87325e 100644 --- a/hw/top_darjeeling/rtl/autogen/top_darjeeling_rnd_cnst_pkg.sv +++ b/hw/top_darjeeling/rtl/autogen/top_darjeeling_rnd_cnst_pkg.sv @@ -17,18 +17,18 @@ package top_darjeeling_rnd_cnst_pkg; // otp_ctrl //////////////////////////////////////////// // Compile-time random bits for initial LFSR seed - parameter otp_ctrl_pkg::lfsr_seed_t RndCnstOtpCtrlLfsrSeed = { - 40'h50_8E576E43 + parameter otp_ctrl_top_specific_pkg::lfsr_seed_t RndCnstOtpCtrlLfsrSeed = { + 40'h54_3A0130B3 }; // Compile-time random permutation for LFSR output - parameter otp_ctrl_pkg::lfsr_perm_t RndCnstOtpCtrlLfsrPerm = { - 240'h7DA5_DD443190_81848F24_210E8DE8_A498830A_0015672C_74CD6E11_6559C654 + parameter otp_ctrl_top_specific_pkg::lfsr_perm_t RndCnstOtpCtrlLfsrPerm = { + 240'h0576_43886004_60C7159A_11CF2964_8D91B4DE_41120E0A_31677E52_E050969D }; // Compile-time random permutation for scrambling key/nonce register reset value - parameter otp_ctrl_pkg::scrmbl_key_init_t RndCnstOtpCtrlScrmblKeyInit = { - 256'h986313D6_12FDCD62_AFD0C3A5_FC772CEB_91C16F5D_6E17DFF0_661BFBA6_F4E0571E + parameter otp_ctrl_top_specific_pkg::scrmbl_key_init_t RndCnstOtpCtrlScrmblKeyInit = { + 256'hBE84F303_80655EE2_6D62DEE6_AE95AAFB_7A44E26D_F9CBEDBC_3D512527_FE67E8B2 }; //////////////////////////////////////////// @@ -36,35 +36,35 @@ package top_darjeeling_rnd_cnst_pkg; //////////////////////////////////////////// // Diversification value used for all invalid life cycle states. parameter lc_ctrl_pkg::lc_keymgr_div_t RndCnstLcCtrlLcKeymgrDivInvalid = { - 128'hAE0FC6E6_A1A665F0_42709B54_CB121F70 + 128'hE07D5551_0F7DF200_E9385195_DE4F2FB2 }; // Diversification value used for the TEST_UNLOCKED* life cycle states. parameter lc_ctrl_pkg::lc_keymgr_div_t RndCnstLcCtrlLcKeymgrDivTestUnlocked = { - 128'h02A405EC_DCF1FEB0_FA7A1487_7EC88637 + 128'h08A71E82_207A34AB_D439B462_F7FDB8FC }; // Diversification value used for the DEV life cycle state. parameter lc_ctrl_pkg::lc_keymgr_div_t RndCnstLcCtrlLcKeymgrDivDev = { - 128'h87AF272E_FAA51F0D_E710C891_D47FF720 + 128'h20E2960F_A1C63758_8C08C383_74903B58 }; // Diversification value used for the PROD/PROD_END life cycle states. parameter lc_ctrl_pkg::lc_keymgr_div_t RndCnstLcCtrlLcKeymgrDivProduction = { - 128'h9F95D614_848E3836_1CD9B7EB_23D532C0 + 128'hBC3484DF_9B0E9C2F_06F0694C_EDAB6BE2 }; // Diversification value used for the RMA life cycle state. parameter lc_ctrl_pkg::lc_keymgr_div_t RndCnstLcCtrlLcKeymgrDivRma = { - 128'hAB44E1E9_2CCFEBAD_AD9193A3_EE4ECD8C + 128'h563D7C0C_31CB754A_523EAFE9_0D67C2C5 }; // Compile-time random bits used for invalid tokens in the token mux parameter lc_ctrl_pkg::lc_token_mux_t RndCnstLcCtrlInvalidTokens = { - 256'h032462B3_E18549CB_1E70D5A8_3DE6673D_2C1C0A7B_55176264_E55C329C_8AE4725A, - 256'hEAF67281_50127AC7_55D70063_277642B5_309A1639_90B966CD_494444C3_BCDF8087, - 256'hB7FACD65_E9654CD8_5BC66D10_208C4FB5_1B97BBF4_D12C378C_CFD6A5A5_BF3032DB, - 256'h51FA1EB9_2F6CE10E_90F9C5C0_6F733DC9_11A321DD_4C0AC240_6D467215_DAB3F2B5 + 256'h5F3D8CE4_0AF0ADBB_E93A5F9B_AB97F2A9_139A0FFA_956AEADB_13BAAA10_D2336E39, + 256'h9E5F1AEB_58C2A1BA_65D13A0F_E39B01C9_5A626001_CC969493_D06CDB45_0C26A87F, + 256'h7BF58DDA_1149EFDC_5F023299_DFB44D70_A9F90685_9E02C051_85213FCF_029CB3E6, + 256'h2CE6FDDC_BA7F6C9D_2519EA1A_ACC8E192_2AF7B82D_7479CD41_D20282EB_0D61AA54 }; //////////////////////////////////////////// @@ -72,12 +72,12 @@ package top_darjeeling_rnd_cnst_pkg; //////////////////////////////////////////// // Compile-time random bits for initial LFSR seed parameter alert_handler_pkg::lfsr_seed_t RndCnstAlertHandlerLfsrSeed = { - 32'h4A52E672 + 32'hCC696D54 }; // Compile-time random permutation for LFSR output parameter alert_handler_pkg::lfsr_perm_t RndCnstAlertHandlerLfsrPerm = { - 160'h1626CA23_5B3B6C12_4BFE5E54_EBF4A934_783055F0 + 160'h2E8C1E35_32551D3F_32FF7F62_75C336DE_060412A2 }; //////////////////////////////////////////// @@ -85,22 +85,22 @@ package top_darjeeling_rnd_cnst_pkg; //////////////////////////////////////////// // Compile-time random reset value for SRAM scrambling key. parameter otp_ctrl_pkg::sram_key_t RndCnstSramCtrlRetAonSramKey = { - 128'h55E0F815_0242862E_B20EEA9A_1F754471 + 128'h224F25DA_B6226BF5_45B7FC56_675745FE }; // Compile-time random reset value for SRAM scrambling nonce. parameter otp_ctrl_pkg::sram_nonce_t RndCnstSramCtrlRetAonSramNonce = { - 128'h6551512A_112DED67_8CE824A0_8E29C772 + 128'hEC587DCB_2A925376_9D73F614_F297C5F7 }; // Compile-time random bits for initial LFSR seed parameter sram_ctrl_pkg::lfsr_seed_t RndCnstSramCtrlRetAonLfsrSeed = { - 32'h795A358E + 32'h024F5AD9 }; // Compile-time random permutation for LFSR output parameter sram_ctrl_pkg::lfsr_perm_t RndCnstSramCtrlRetAonLfsrPerm = { - 160'h44CFC8B5_8B712090_29F2B7FA_22DE86DE_7D5D0C38 + 160'hD1EF5544_82059A9C_7F7C28FC_87519D5C_B2684C2F }; //////////////////////////////////////////// @@ -108,34 +108,34 @@ package top_darjeeling_rnd_cnst_pkg; //////////////////////////////////////////// // Default seed of the PRNG used for register clearing. parameter aes_pkg::clearing_lfsr_seed_t RndCnstAesClearingLfsrSeed = { - 64'hD1095889_FBD8A7B5 + 64'hBEE8C0A3_F22571AC }; // Permutation applied to the LFSR of the PRNG used for clearing. parameter aes_pkg::clearing_lfsr_perm_t RndCnstAesClearingLfsrPerm = { - 128'h392FA90D_E595FF71_3B4289DD_4E117922, - 256'h432F2804_49DB5525_06281AC8_698B62BB_B13CDF02_B73D2A71_B8D7F7AC_191F8B1F + 128'h967DD7A7_A90EDA29_8C67E0D0_E05D0907, + 256'h9B9F013D_42C2A6B1_80446F21_877B6E17_0F49EFED_F0754637_2ACAB23A_33F4B594 }; // Permutation applied to the clearing PRNG output for clearing the second share of registers. parameter aes_pkg::clearing_lfsr_perm_t RndCnstAesClearingSharePerm = { - 128'h03E5AB51_C865BF6D_4DDC1EB8_653F3233, - 256'h3B7DFCA2_4814FA1F_11D61B09_5A971EC8_101B4C0E_8D14A793_92E689E6_A9C6EEC3 + 128'h62A78D67_CCC2D1BB_CC2B6049_53A9A011, + 256'hF916757D_77469C5C_A3F0D0C2_3B1AE7E2_E5F67E3B_AED1D3C8_E3C5BA11_A2488540 }; // Default seed of the PRNG used for masking. parameter aes_pkg::masking_lfsr_seed_t RndCnstAesMaskingLfsrSeed = { - 32'h82FA14A8, - 256'hF2B75C26_5C9CB3B2_F1C44A72_87AB4C06_18822E92_A084A048_FFE9FB86_140A79D3 + 32'h812B74CC, + 256'h981F89B9_B95175A8_F02EA42F_7883754A_0EAA875B_D05EE93E_58E0E429_86CC76AC }; // Permutation applied to the output of the PRNG used for masking. parameter aes_pkg::masking_lfsr_perm_t RndCnstAesMaskingLfsrPerm = { - 256'h70666885_3241427A_1D566559_0D530C47_458C2397_5C21863C_917C5160_37078324, - 256'h8B9A7F93_73627706_5A392F1B_8E7E6F4E_81118A1C_79589D74_042B0817_710F252D, - 256'h48648819_2A2E4900_4610129E_1A319228_8257843F_894C090A_50695440_616A1E96, - 256'h6C362038_144D1595_35346D02_875D9C18_754A268F_0B015B1F_78137629_9827220E, - 256'h9F7B8D90_437D4F6B_3D723A05_63996716_9B30446E_52942C33_5E033E3B_5F4B5580 + 256'h3E319D0B_58181078_00733C3B_12749486_79110A7A_892C5E9A_60398527_87045C5F, + 256'h018C0D16_4942214F_72335B38_5D829509_255A7522_7C8E5937_07083A1D_973D8126, + 256'h8A1B6220_76998424_301E0C23_962F9B03_7F19454C_9E513291_4E356D4D_71772D4B, + 256'h474A8D69_4363366A_667E6F9C_46569050_531F2A83_54801A1C_9F931561_13689892, + 256'h3F677006_287B2940_2B574402_41052E65_52640E0F_5588176B_7D6E8B6C_8F483414 }; //////////////////////////////////////////// @@ -143,58 +143,58 @@ package top_darjeeling_rnd_cnst_pkg; //////////////////////////////////////////// // Compile-time random data for PRNG default seed parameter kmac_pkg::lfsr_seed_t RndCnstKmacLfsrSeed = { - 32'hE933D88E, - 256'h2E1CF654_60F23FB7_80499E6F_CCE64CEA_FD282C0E_33FD2C07_986C2A51_1755F072 + 32'h534D7AC4, + 256'hBD0CA8B0_3546A981_88556C8C_BF9B24D1_D32C9BD8_6DFED26F_1FE72FFB_B4B9762D }; // Compile-time random permutation for PRNG output parameter kmac_pkg::lfsr_perm_t RndCnstKmacLfsrPerm = { - 64'h99E8E296_3E89488B, - 256'h131D0C87_2324F634_25F3CEDE_85EF7490_761B0EC0_AD251FC8_163069A6_5313C5AD, - 256'hA542E455_D553AEB3_7506368D_F89C1876_C38C3186_C6AA35AC_ED838342_B42D8471, - 256'h88A8C868_5C27895E_B4905481_AF5DE7E8_7210799F_63A6AD13_88D0561B_49C2A1E9, - 256'h9B876359_2CAA899B_B52E072E_92BA9A20_ABD6235D_97AA5183_C30F09CF_F5BCDE2D, - 256'h5B6AE2E1_7B9347DD_7C97164A_698E620C_B22E087C_2318B809_0190182A_A8262847, - 256'hC4F77240_A1453AB1_8BF07857_400BE5BB_0D07C504_FE02279E_C0FC901A_03EC0C33, - 256'h9A18B255_4D68DCB1_9079BEEC_0C1C169B_D569B628_26ACE962_B26FCBB3_E4B11019, - 256'hEBE0A119_B149E27D_A4222AC4_5F423012_3306A30E_E28244BB_52CA91B2_7F56C759, - 256'hB0625D13_1A005F6F_F0956988_6C5932F6_B70D2345_CF83128C_7C9426E6_B049D957, - 256'h5EB1A899_5D16A728_E190ADB5_547B2E66_17E6D750_DA619DA7_216BD1B6_5D6AD934, - 256'h8F1699A5_43925E3B_DD8039A5_180002BE_B529C963_8B000C19_271EE67B_5C65C140, - 256'hB94D65E5_1B602125_9CC0A71E_5476C36E_9ACB0AB9_60504A79_1C6426C9_D1B4A1CE, - 256'h8ADC622B_1D7C0A50_195B377C_6B4D824A_FE9391AC_1B4DD07D_4A621871_8B56151D, - 256'hE7868FC3_8F17C142_9106E97F_1CD398B0_8D0A4BEA_FD13C217_490C355A_09048A6A, - 256'hAFE0A1C8_04DBC939_CC98F102_1CC45DA9_2AF8A3A2_CFE6E670_6C6D4A9E_3C804925, - 256'h40CA1571_911D2252_A83AEA43_44603C11_60A089F3_615C82EE_88B0D370_D2AA5A49, - 256'h161E14E4_802BAEF6_0A8FBE6D_912A6094_4CC0C68A_1F6B44CE_CD074491_79A8516C, - 256'h9825E403_CFBC557B_A75FB314_C25D4526_6910A857_1900C3B8_19D2F19C_8DB0A658, - 256'h0B7E9097_A1993CA4_7860B358_C4885037_0047E2D8_8474DD42_8B1E1D4D_C88E5927, - 256'h1F49D408_8A44BBB0_0D991126_54E72DE2_FB68FD3E_A3A9E50D_B7C35521_29ADA65B, - 256'hA93C118C_7512056F_90A65093_0650B532_F56B93B2_CA4D008A_883EB3B8_8898CA45, - 256'h49966275_7FBC2CA6_D58D1B81_07C83D41_E7C43E94_B6224A61_5A544363_645C4F82, - 256'h660DF8B1_EBBB2F13_69054A44_058FE1FA_AA2B9ED4_9779E94C_15F6DDDD_1D1CE8ED, - 256'hF3B3F08A_7C5AB40C_F08B1BA9_43B06DB8_6897395C_5BB20B89_88429683_C20985A1, - 256'hA89712F4_92914564_637BCEB8_102CC3A7_A7A98F6F_4CD9B92E_7FA55485_547FD3D7, - 256'h06523D1E_31E23323_A1962903_C4A0605D_3208E885_18CE662F_28361554_97245CE5, - 256'h9207353C_F5132691_5AEE9D80_CB90A99A_87725ED1_702304D6_AE4E16D9_C64FAF21, - 256'h30E941A3_5DC47E72_20CB4A5C_04C26D58_DD166101_C14EE740_99F61D13_114436A4, - 256'h70174ABB_5A5B94E9_B1BEF156_0208F6F3_5E091AAD_19596391_851E3827_F00C591F, - 256'h0C1069E0_3E2B03A9_42768D4A_0039B8E7_723D1A87_A85A81C5_B31370BD_2F4B44DB, - 256'h765601CB_D04673AB_16885AA2_55D7E54D_915E8143_6F1CC2D0_EBF15BA0_444C4638 + 64'h3F69D7B0_444F62FB, + 256'h85E1BBCC_E9A23D25_29FC2CA6_57AA713B_124DA452_1A14C7D2_97617534_41488B05, + 256'h85D2C74C_7F2C23C8_CC1179EC_9B714548_9BB894F5_C7A124A9_73566537_DC7C3CDD, + 256'hBB06B6B1_C3847142_16DF2AFD_F0211C39_B9824695_C3E8D792_1A37A8B7_87506C10, + 256'hF8399465_BD559956_913590C4_C3100C18_9C5A7A38_E4AA87E2_D56DA015_AA9037AD, + 256'h08B54D84_7EC2041C_3E73A368_02CB6B6C_D7B4450C_14C312C8_34AF80A9_66150CFA, + 256'hB0B20341_97A16E92_6E8835A5_8B53F751_203CAFAA_DC852C69_30088D78_0FB86CD3, + 256'h096EB4BD_0C8A232A_408304CE_187C0753_ADA40A9C_6012D4CE_3B174A81_6ACF1182, + 256'h17DA2080_9A60F579_559442ED_962FD8CB_971659CA_7C25CBA9_ABC548C6_4DE76106, + 256'h39AE68C0_E46E0156_AA35A923_C084A23E_90962714_3A076097_F5F1A203_2231ABB6, + 256'hC800984B_92C1A0CC_A0BC6A53_F8EB45E4_31601536_D710D475_78D87816_A1B5AE7B, + 256'h0C599846_251D34C2_2DF3B1D2_5971B971_9B1A2159_1ADE77DC_7BBA844E_0A576A77, + 256'h44914C04_E24169E8_BA9257D8_B2B0DD86_A555E2A8_142C2306_F2A6E86A_5149C680, + 256'hA0D13BC7_2403A57E_910978D0_BA98AC03_01408059_305B19A0_976400F0_71070332, + 256'h7E222F02_B8DFC740_E42A634C_509A5C78_19702B38_632B4F19_E53C8440_16FDAA43, + 256'h89C13E0A_238A9AAC_9308E593_E5D017EC_4279FF31_E789B5E2_1B6B05DE_8AC0C0B0, + 256'h9E52812B_E4663137_0E0146D7_4981BAA6_4C6E4B68_2ECCBDA7_46D13905_E87BFCA4, + 256'h5C83BBCD_884C1D19_80871B02_E2AA7F22_61087B10_AF5A11BD_B5906663_6918205B, + 256'h22757A9B_C9555237_724CB356_7C026729_470E9C1F_4C603C28_557898CD_5C26C0BC, + 256'h297F1952_CD6BC344_063DF9BB_5CC6E028_3D2D5B1A_D044C0CA_589F9CD8_D7642B86, + 256'h135822EA_A890F7A6_98ABC478_58D007CE_FAB93EA8_3000419D_B8E4F5A0_6983D436, + 256'h7A5BAD5F_E19A29B7_70C80CDE_C1A5E91D_C85D2F51_846475C1_28E1153F_D037996C, + 256'hA694B03D_CD63A24A_2D9984D8_1AB4571A_8B9646F4_3096466A_BA940569_29A22468, + 256'hF9A6588C_01C5FC9B_26223359_F846C7B6_0E6B5EEE_06606B41_25F5770F_12472288, + 256'hC88302DA_CCBE61A3_F6487D8A_A4D9D4BD_B292789C_821705E3_F1F3871C_0362D19F, + 256'h46747523_44076B51_1E67DB3B_64E830E2_C39D4824_0AF53EB8_4E81E7AD_EE325B09, + 256'hA9BE52A8_9BF1500D_873495C1_C1CC510D_611289AA_C4ED7752_142CE31C_5CA3B855, + 256'hB4C53492_99C5E19A_0C742D23_AA54ABA8_D97616D2_96911010_2401AEC6_DE644CB0, + 256'hAA15582F_0D962621_352567D4_E5BEDDCB_8A4E0FC7_A6F5BA79_67A791EF_838A36B9, + 256'hDE950009_5D4173E3_52296032_97D6A0B8_12267AC9_D6459FAB_F84C3DD2_B628F675, + 256'h508B9AE9_B912E435_A4683151_DD51BF56_3BDD1F40_6BBB157C_1E71181C_EAB625D6, + 256'hB19C8ADB_16F1B610_9C69EF14_53422258_99249C2B_E5AFA042_C14A0665_15BA4460 }; // Compile-time random data for PRNG buffer default seed parameter kmac_pkg::buffer_lfsr_seed_t RndCnstKmacBufferLfsrSeed = { - 32'h46C49F15, - 256'hA16A2A1F_D01F8E09_D92DD13E_0298799D_91C54BFD_2A51F45A_ADF344C2_62BB0ABD, - 256'hE9B240D7_63DA6C42_50BA5A44_A7D4F4D7_F99B4A64_80DA2D53_E7CA1520_1A8F5653, - 256'h0010117F_58C9E994_6C111804_07FEB471_0F404E6E_4FACE59B_972D1FDC_AF711C05 + 32'hB7480D9D, + 256'h5001B775_5DE3EF26_18E28371_DD02B1DB_04469E57_70D10B00_69E332BE_0F35A726, + 256'hB339D953_A1DD95A3_F78E1438_9414ADC1_9300E093_4269B3A3_0B61EA4D_9790DE0F, + 256'hB09D2792_D26BF956_5775A284_3FFB7D42_993C0A1C_3D4D2937_F997A3B5_1AAFEBC6 }; // Compile-time random permutation for LFSR Message output parameter kmac_pkg::msg_perm_t RndCnstKmacMsgPerm = { - 128'h1ABBA9F8_CE16CF92_D4B5C3C2_6435ECD4, - 256'hDDE06154_85C5D6E8_9BB244D3_D7F0ABC0_641D3467_86323688_ABC0EB2F_E541E39A + 128'h2A89790D_B63EFF5A_927C7D96_C2F0B23B, + 256'hA17CAD7E_DE662023_B24CCCE2_E804A4B6_89B48510_F749A719_C4C1DDD3_D5851C4D }; //////////////////////////////////////////// @@ -202,17 +202,17 @@ package top_darjeeling_rnd_cnst_pkg; //////////////////////////////////////////// // Default seed of the PRNG used for URND. parameter otbn_pkg::urnd_prng_seed_t RndCnstOtbnUrndPrngSeed = { - 256'hE24BE7E1_C96737B7_26A4C6F0_4993B3E6_AAE70F22_AAF748B5_7C320264_E157E2C7 + 256'h4569B54E_4CC3468B_A78749A7_093FE9EB_381742E6_95130A4F_B5238EBB_74B4F311 }; // Compile-time random reset value for IMem/DMem scrambling key. parameter otp_ctrl_pkg::otbn_key_t RndCnstOtbnOtbnKey = { - 128'h4A280332_61EE6CEC_10075C55_A820F842 + 128'h8274EA0B_356B0515_35EAC4E0_98EA131F }; // Compile-time random reset value for IMem/DMem scrambling nonce. parameter otp_ctrl_pkg::otbn_nonce_t RndCnstOtbnOtbnNonce = { - 64'hD43EE1AB_1B3D78F9 + 64'h95C233E6_70A4321B }; //////////////////////////////////////////// @@ -220,53 +220,53 @@ package top_darjeeling_rnd_cnst_pkg; //////////////////////////////////////////// // Compile-time random bits for initial LFSR seed parameter keymgr_pkg::lfsr_seed_t RndCnstKeymgrDpeLfsrSeed = { - 64'h92D22346_9898EB2A + 64'hAD599D4D_F689612A }; // Compile-time random permutation for LFSR output parameter keymgr_pkg::lfsr_perm_t RndCnstKeymgrDpeLfsrPerm = { - 128'hBA87C33D_D8319DE0_A103ECEC_EAD96923, - 256'h98EFC95D_9B938350_5FC4B550_B9B459CA_A265A3B0_311F5824_12AB4C5E_C71371BF + 128'h961D0532_D78D8ACE_1FF82767_391F0F51, + 256'h5416EBF4_A96B95FB_20BC2E1C_472064AF_75C406EF_AD88D3DF_3903076C_A8266A86 }; // Compile-time random permutation for entropy used in share overriding parameter keymgr_pkg::rand_perm_t RndCnstKeymgrDpeRandPerm = { - 160'hBCE26820_3CC807B4_B242F394_F262A5B5_35F3F5AB + 160'h95CA1A67_1B6B87F5_F2CFD226_A0445E21_BB53C12C }; // Compile-time random bits for revision seed parameter keymgr_pkg::seed_t RndCnstKeymgrDpeRevisionSeed = { - 256'h5E456F3B_26D15F55_84BB9879_BD249785_2C23D0FA_289C3348_CE364ED5_34970E1F + 256'h46794CFC_A2B0C6CB_FB61F1BD_7ED0A8F9_26DA506E_D059049D_C9A6E2C1_826AA5BF }; // Compile-time random bits for software generation seed parameter keymgr_pkg::seed_t RndCnstKeymgrDpeSoftOutputSeed = { - 256'hD45EC606_BE134FC0_8318BD29_B22B08DF_647BDC30_068E7513_200FA828_BFEE3ECC + 256'hDF596FB9_A391C672_2B8B5CAD_051F3EDB_286E5540_01528F8E_9A5E55EB_B82A171C }; // Compile-time random bits for hardware generation seed parameter keymgr_pkg::seed_t RndCnstKeymgrDpeHardOutputSeed = { - 256'hF02417FA_68D00453_4267F60B_34A8F072_E77EE485_9B243F92_CEB098BE_6E65C359 + 256'hF1E411AD_FE91411B_1D8EE016_1ECD6990_02EF8026_E474E857_39EF0277_414266B4 }; // Compile-time random bits for generation seed when aes destination selected parameter keymgr_pkg::seed_t RndCnstKeymgrDpeAesSeed = { - 256'h480858C3_9611E658_C97FB8D3_E9CC0E7F_8FA87252_00C726C8_F90EF5E2_B74BEEE2 + 256'hF811EBD0_1E7EA4AC_F08B65A5_CF51AE52_8473DD46_61D013FB_44105783_0DD1B830 }; // Compile-time random bits for generation seed when kmac destination selected parameter keymgr_pkg::seed_t RndCnstKeymgrDpeKmacSeed = { - 256'h7CFA19EA_576CA8A8_225C8E17_053E44B6_8DD7822A_6859F2CF_52A55FD1_81B082D0 + 256'hBF365BBF_C1DEB892_1D7A6896_FD1E6013_622D6614_2EE83239_BA8E92C6_1DE956A5 }; // Compile-time random bits for generation seed when otbn destination selected parameter keymgr_pkg::seed_t RndCnstKeymgrDpeOtbnSeed = { - 256'h3C583756_F11AA5B3_7F9544C4_AA90EB97_E1C0F808_6F627A28_1C5681D6_566641EE + 256'h2768B6D5_F1650F1B_01969710_A957DBFB_2E2443F4_FE2D0A76_0F83C131_3D3EF78F }; // Compile-time random bits for generation seed when no destination selected parameter keymgr_pkg::seed_t RndCnstKeymgrDpeNoneSeed = { - 256'hCCCC8887_6AD113C5_0AD1BE4D_94DCE8FC_A54128A3_860AE1D3_493FBEF6_B9BA5DFE + 256'hEA4D3D48_54921724_8330D048_F78E497E_63710E97_20AC0F46_C68E74CA_B245BE89 }; //////////////////////////////////////////// @@ -274,14 +274,14 @@ package top_darjeeling_rnd_cnst_pkg; //////////////////////////////////////////// // Compile-time random bits for csrng state group diversification value parameter csrng_pkg::cs_keymgr_div_t RndCnstCsrngCsKeymgrDivNonProduction = { - 128'h4463A461_5EF46762_A794A8C2_597E69FB, - 256'h259B9619_14CD75BB_FD36BA48_E780D27C_D582540C_7A68FF25_4EDFD852_829FECCC + 128'h11081C18_0F034806_61762D84_BFB80313, + 256'h0EA592F6_3F968D16_D49748CB_C79F398E_21395417_31F82F0E_5E0F2EA9_61C615FC }; // Compile-time random bits for csrng state group diversification value parameter csrng_pkg::cs_keymgr_div_t RndCnstCsrngCsKeymgrDivProduction = { - 128'hABD4AF9C_9032DA20_BDAF59FE_2AE209B8, - 256'h325D2C8C_35FC9606_79B106A8_7E59E6AE_B1B5302D_33401070_251696FB_B4BA169A + 128'h96398950_EDED5E99_9CBB1667_0229D5A2, + 256'h4DF595BE_24C50547_533E86EF_45DC765B_C3FC1C44_92B8CD26_4903BA93_E2074570 }; //////////////////////////////////////////// @@ -289,22 +289,22 @@ package top_darjeeling_rnd_cnst_pkg; //////////////////////////////////////////// // Compile-time random reset value for SRAM scrambling key. parameter otp_ctrl_pkg::sram_key_t RndCnstSramCtrlMainSramKey = { - 128'h6CD82FAD_46A0A5C0_4009BB93_4C7EF7B8 + 128'h4FC7F5A2_63F7641C_180B43FF_715AA5A2 }; // Compile-time random reset value for SRAM scrambling nonce. parameter otp_ctrl_pkg::sram_nonce_t RndCnstSramCtrlMainSramNonce = { - 128'h3B6E4061_0B4309FC_B9DC54D4_276137F9 + 128'hBD793053_46EBE883_E11EEA60_C36419A2 }; // Compile-time random bits for initial LFSR seed parameter sram_ctrl_pkg::lfsr_seed_t RndCnstSramCtrlMainLfsrSeed = { - 32'h901D09A7 + 32'h27F0CBB5 }; // Compile-time random permutation for LFSR output parameter sram_ctrl_pkg::lfsr_perm_t RndCnstSramCtrlMainLfsrPerm = { - 160'h1036ED2B_E8B646C8_24A692F9_5F1031B9_DF89D3AD + 160'h8E38B367_E30F41AB_4C553C0A_E96E8A62_1B74BC9E }; //////////////////////////////////////////// @@ -312,22 +312,22 @@ package top_darjeeling_rnd_cnst_pkg; //////////////////////////////////////////// // Compile-time random reset value for SRAM scrambling key. parameter otp_ctrl_pkg::sram_key_t RndCnstSramCtrlMboxSramKey = { - 128'hE927F6D9_E4ABAC39_8D42C745_EEF646C1 + 128'h7C5F7FB3_9AFB3660_BABA1F1B_AC202D76 }; // Compile-time random reset value for SRAM scrambling nonce. parameter otp_ctrl_pkg::sram_nonce_t RndCnstSramCtrlMboxSramNonce = { - 128'h464DCA86_DAFD7C7C_71E6058D_DFD871C5 + 128'hE627CBB3_0AD1BC1E_181DB312_B8D5A947 }; // Compile-time random bits for initial LFSR seed parameter sram_ctrl_pkg::lfsr_seed_t RndCnstSramCtrlMboxLfsrSeed = { - 32'h1CACBAF4 + 32'h9C07C534 }; // Compile-time random permutation for LFSR output parameter sram_ctrl_pkg::lfsr_perm_t RndCnstSramCtrlMboxLfsrPerm = { - 160'h2C32AE93_87FDA299_E9CCF52A_BBC8627B_4D8D8028 + 160'h0EEBF897_27773140_0C88D2ED_0175EC6D_E69578D2 }; //////////////////////////////////////////// @@ -335,12 +335,12 @@ package top_darjeeling_rnd_cnst_pkg; //////////////////////////////////////////// // Fixed nonce used for address / data scrambling parameter bit [63:0] RndCnstRomCtrl0ScrNonce = { - 64'h6FD468A7_7EFDE3DE + 64'hFA5BA5CF_4337A897 }; // Randomised constant used as a scrambling key for ROM data parameter bit [127:0] RndCnstRomCtrl0ScrKey = { - 128'h5B4CAF47_76A247BA_BA4C9908_ED16BC54 + 128'h7E31C7B9_4C2BCF03_781EE0D8_B1BA84CB }; //////////////////////////////////////////// @@ -348,12 +348,12 @@ package top_darjeeling_rnd_cnst_pkg; //////////////////////////////////////////// // Fixed nonce used for address / data scrambling parameter bit [63:0] RndCnstRomCtrl1ScrNonce = { - 64'h15EC16D2_8C535513 + 64'hB565E8F2_F8BE62FE }; // Randomised constant used as a scrambling key for ROM data parameter bit [127:0] RndCnstRomCtrl1ScrKey = { - 128'h12FCEDCF_2832A66C_EACF8ED4_D5B61617 + 128'h9D2D4204_B2F54BE8_75AE4802_77680CBA }; //////////////////////////////////////////// @@ -361,22 +361,22 @@ package top_darjeeling_rnd_cnst_pkg; //////////////////////////////////////////// // Default seed of the PRNG used for random instructions. parameter ibex_pkg::lfsr_seed_t RndCnstRvCoreIbexLfsrSeed = { - 32'h7DD43B56 + 32'h442BAA3A }; // Permutation applied to the LFSR of the PRNG used for random instructions. parameter ibex_pkg::lfsr_perm_t RndCnstRvCoreIbexLfsrPerm = { - 160'h1440B451_B8D70817_96663DFD_BEE6D283_2AA1A5DF + 160'h959DE8F2_0436D6D5_31FFC06F_D48EA0D0_A8839679 }; // Default icache scrambling key parameter logic [ibex_pkg::SCRAMBLE_KEY_W-1:0] RndCnstRvCoreIbexIbexKeyDefault = { - 128'h74F35C79_F397C4E4_C7E22B75_81848A90 + 128'hA690C16D_D41B3D1F_A05F7ECC_83245CCC }; // Default icache scrambling nonce parameter logic [ibex_pkg::SCRAMBLE_NONCE_W-1:0] RndCnstRvCoreIbexIbexNonceDefault = { - 64'hA1254B22_76BEC9FC + 64'h8577DAAF_B97C26A7 }; endpackage : top_darjeeling_rnd_cnst_pkg diff --git a/hw/top_darjeeling/sw/autogen/chip/top_darjeeling.rs b/hw/top_darjeeling/sw/autogen/chip/top_darjeeling.rs index 6783cc7194f053..9daf6db76e6938 100644 --- a/hw/top_darjeeling/sw/autogen/chip/top_darjeeling.rs +++ b/hw/top_darjeeling/sw/autogen/chip/top_darjeeling.rs @@ -103,7 +103,7 @@ pub const OTP_CTRL_CORE_BASE_ADDR: usize = 0x30130000; /// memory-mapped registers associated with this peripheral should have an /// address between #OTP_CTRL_CORE_BASE_ADDR and /// `OTP_CTRL_CORE_BASE_ADDR + OTP_CTRL_CORE_SIZE_BYTES`. -pub const OTP_CTRL_CORE_SIZE_BYTES: usize = 0x1000; +pub const OTP_CTRL_CORE_SIZE_BYTES: usize = 0x8000; /// Peripheral base address for prim device on otp_ctrl in top darjeeling. /// diff --git a/hw/top_darjeeling/sw/autogen/top_darjeeling.h b/hw/top_darjeeling/sw/autogen/top_darjeeling.h index f73ae479e1001e..fe62a3aaeecfe8 100644 --- a/hw/top_darjeeling/sw/autogen/top_darjeeling.h +++ b/hw/top_darjeeling/sw/autogen/top_darjeeling.h @@ -136,7 +136,7 @@ extern "C" { * address between #TOP_DARJEELING_OTP_CTRL_CORE_BASE_ADDR and * `TOP_DARJEELING_OTP_CTRL_CORE_BASE_ADDR + TOP_DARJEELING_OTP_CTRL_CORE_SIZE_BYTES`. */ -#define TOP_DARJEELING_OTP_CTRL_CORE_SIZE_BYTES 0x1000u +#define TOP_DARJEELING_OTP_CTRL_CORE_SIZE_BYTES 0x8000u /** * Peripheral base address for prim device on otp_ctrl in top darjeeling. diff --git a/hw/top_darjeeling/sw/autogen/top_darjeeling_memory.h b/hw/top_darjeeling/sw/autogen/top_darjeeling_memory.h index 91498a49cdadfc..3583936afef4a2 100644 --- a/hw/top_darjeeling/sw/autogen/top_darjeeling_memory.h +++ b/hw/top_darjeeling/sw/autogen/top_darjeeling_memory.h @@ -189,7 +189,7 @@ * address between #TOP_DARJEELING_OTP_CTRL_CORE_BASE_ADDR and * `TOP_DARJEELING_OTP_CTRL_CORE_BASE_ADDR + TOP_DARJEELING_OTP_CTRL_CORE_SIZE_BYTES`. */ -#define TOP_DARJEELING_OTP_CTRL_CORE_SIZE_BYTES 0x1000 +#define TOP_DARJEELING_OTP_CTRL_CORE_SIZE_BYTES 0x8000 /** * Peripheral base address for prim device on otp_ctrl in top darjeeling. * diff --git a/hw/top_darjeeling/top_darjeeling.core b/hw/top_darjeeling/top_darjeeling.core index 6656be8457b109..ab4f8b9dbb0529 100644 --- a/hw/top_darjeeling/top_darjeeling.core +++ b/hw/top_darjeeling/top_darjeeling.core @@ -35,7 +35,6 @@ filesets: - lowrisc:constants:top_darjeeling_top_pkg - lowrisc:constants:top_darjeeling_jtag_id_pkg - lowrisc:constants:top_darjeeling_ibex_pmp_reset_pkg - - lowrisc:ip:otp_ctrl - lowrisc:ip:lc_ctrl - lowrisc:ip:mbx - lowrisc:top_darjeeling:xbar_dbg @@ -44,10 +43,11 @@ filesets: - lowrisc:top_darjeeling:xbar_peri - lowrisc:opentitan:top_darjeeling_alert_handler - lowrisc:opentitan:top_darjeeling_clkmgr - - lowrisc:opentitan:top_darjeeling_rstmgr - - lowrisc:opentitan:top_darjeeling_rv_plic + - lowrisc:opentitan:top_darjeeling_otp_ctrl - lowrisc:opentitan:top_darjeeling_pinmux - lowrisc:opentitan:top_darjeeling_pwrmgr + - lowrisc:opentitan:top_darjeeling_rstmgr + - lowrisc:opentitan:top_darjeeling_rv_plic - lowrisc:ip:aon_timer - lowrisc:ip:adc_ctrl - lowrisc:ip:sysrst_ctrl diff --git a/hw/top_earlgrey/BUILD b/hw/top_earlgrey/BUILD index 16d7594b7f4a97..4a3f3d4bb7a3b0 100644 --- a/hw/top_earlgrey/BUILD +++ b/hw/top_earlgrey/BUILD @@ -59,7 +59,7 @@ fpga_cw310( exec_env = "fpga_cw310_test_rom", mmi = "//hw/bitstream:cw310_mmi", openocd_adapter_config = "//third_party/openocd:jtag_olimex_cfg", - otp = "//hw/ip/otp_ctrl/data:img_rma", + otp = "//hw/top_earlgrey/data/otp:img_rma", param = { "interface": "cw310", "exit_success": DEFAULT_TEST_SUCCESS_MSG, @@ -123,7 +123,7 @@ fpga_cw310( exec_env = "fpga_hyper310_rom_with_fake_keys", manifest = "//sw/device/silicon_creator/rom_ext:manifest", mmi = "//hw/bitstream/hyperdebug:mmi", - otp = "//hw/ip/otp_ctrl/data:img_rma", + otp = "//hw/top_earlgrey/data/otp:img_rma", param = { "interface": "hyper310", "exit_success": DEFAULT_TEST_SUCCESS_MSG, @@ -177,7 +177,7 @@ fpga_cw310( # binary. ecdsa_key = {"//sw/device/silicon_creator/rom/keys/fake/ecdsa:prod_key_0_ecdsa_p256": "prod_key_0"}, exec_env = "fpga_cw310_sival", - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_prod_manuf_personalized", + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_prod_manuf_personalized", spx_key = {"//sw/device/silicon_creator/rom/keys/fake/spx:prod_key_0_spx": "prod_key_0"}, tags = ["cw310_sival"], ) @@ -196,7 +196,7 @@ fpga_cw310( "//sw/device/lib/arch:boot_stage_owner", "//sw/device/lib/arch:fpga_cw310", ], - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_prod_manuf_personalized", + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_prod_manuf_personalized", rom_ext = "//sw/device/silicon_creator/rom_ext:rom_ext_slot_a", rsa_key = {"//sw/device/silicon_creator/rom_ext/keys/fake:rom_ext_prod_private_key_0": "prod_key_0"}, tags = ["cw310_sival_rom_ext"], @@ -262,7 +262,7 @@ fpga_cw340( base_bitstream = "//hw/bitstream/cw340:bitstream", exec_env = "fpga_cw340_test_rom", mmi = "//hw/bitstream/cw340:mmi", - otp = "//hw/ip/otp_ctrl/data:img_rma", + otp = "//hw/top_earlgrey/data/otp:img_rma", rom = "//sw/device/lib/testing/test_rom:test_rom", test_cmd = """ --exec="transport init" @@ -282,7 +282,7 @@ fpga_cw340( exec_env = "fpga_cw340_rom_with_fake_keys", manifest = "//sw/device/silicon_creator/rom_ext:manifest", mmi = "//hw/bitstream/cw340:mmi", - otp = "//hw/ip/otp_ctrl/data:img_rma", + otp = "//hw/top_earlgrey/data/otp:img_rma", rom = "//sw/device/silicon_creator/rom:mask_rom", test_cmd = """ --exec="transport init" @@ -331,7 +331,7 @@ fpga_cw340( # binary. ecdsa_key = {"//sw/device/silicon_creator/rom/keys/fake/ecdsa:prod_key_0_ecdsa_p256": "prod_key_0"}, exec_env = "fpga_cw340_sival", - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_prod_manuf_personalized", + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_prod_manuf_personalized", spx_key = {"//sw/device/silicon_creator/rom/keys/fake/spx:prod_key_0_spx": "prod_key_0"}, tags = ["cw340_sival"], ) @@ -350,7 +350,7 @@ fpga_cw340( "//sw/device/lib/arch:boot_stage_owner", "//sw/device/lib/arch:fpga_cw340", ], - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_prod_manuf_personalized", + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_prod_manuf_personalized", rom_ext = "//sw/device/silicon_creator/rom_ext:rom_ext_slot_a", rsa_key = {"//sw/device/silicon_creator/rom_ext/keys/fake:rom_ext_prod_private_key_0": "prod_key_0"}, tags = ["cw340_sival_rom_ext"], @@ -452,7 +452,7 @@ sim_verilator( "//sw/device/lib/arch:sim_verilator", ], linker_script = "//sw/device/lib/testing/test_framework:ottf_ld_silicon_creator_slot_a", - otp = "//hw/ip/otp_ctrl/data:img_rma", + otp = "//hw/top_earlgrey/data/otp:img_rma", rom_scramble_config = "//hw/top_earlgrey/data/autogen:top_earlgrey.gen.hjson", test_cmd = "testing-not-supported", ) @@ -512,10 +512,10 @@ sim_dv( "//sw/device/lib/arch:sim_dv", ], linker_script = "//sw/device/lib/testing/test_framework:ottf_ld_silicon_creator_slot_a", - otp = "//hw/ip/otp_ctrl/data:img_rma", - otp_data_perm = "//hw/ip/otp_ctrl/data:data_perm", - otp_mmap = "//hw/ip/otp_ctrl/data:otp_ctrl_mmap.hjson", - otp_seed = "//hw/ip/otp_ctrl/data:otp_seed", + otp = "//hw/top_earlgrey/data/otp:img_rma", + otp_data_perm = "//util/design/data:data_perm", + otp_mmap = "//hw/top_earlgrey/data/otp:otp_ctrl_mmap.hjson", + otp_seed = "//util/design/data:otp_seed", rom_scramble_config = "//hw/top_earlgrey/data/autogen:top_earlgrey.gen.hjson", ) diff --git a/hw/top_earlgrey/data/autogen/defs.bzl b/hw/top_earlgrey/data/autogen/defs.bzl index 4ffb0e9367e9ae..76b3aedd31cc12 100644 --- a/hw/top_earlgrey/data/autogen/defs.bzl +++ b/hw/top_earlgrey/data/autogen/defs.bzl @@ -25,7 +25,7 @@ load("//hw/ip/keymgr:defs.bzl", "KEYMGR") load("//hw/ip/kmac:defs.bzl", "KMAC") load("//hw/ip/lc_ctrl:defs.bzl", "LC_CTRL") load("//hw/ip/otbn:defs.bzl", "OTBN") -load("//hw/ip/otp_ctrl:defs.bzl", "OTP_CTRL") +load("//hw/top_earlgrey/ip_autogen/otp_ctrl:defs.bzl", "OTP_CTRL") load("//hw/ip/pattgen:defs.bzl", "PATTGEN") load("//hw/top_earlgrey/ip_autogen/pinmux:defs.bzl", "PINMUX") load("//hw/ip/pwm:defs.bzl", "PWM") diff --git a/hw/top_earlgrey/data/autogen/top_earlgrey.gen.hjson b/hw/top_earlgrey/data/autogen/top_earlgrey.gen.hjson index 872633bda8c220..7aa78a1a92d96a 100644 --- a/hw/top_earlgrey/data/autogen/top_earlgrey.gen.hjson +++ b/hw/top_earlgrey/data/autogen/top_earlgrey.gen.hjson @@ -1535,6 +1535,7 @@ hart: 0x40138000 } } + attr: ipgen clock_connections: { clk_i: clkmgr_aon_clocks.clk_io_div4_secure @@ -1560,31 +1561,31 @@ { name: RndCnstLfsrSeed desc: Compile-time random bits for initial LFSR seed - type: otp_ctrl_pkg::lfsr_seed_t + type: otp_ctrl_top_specific_pkg::lfsr_seed_t randcount: 40 randtype: data name_top: RndCnstOtpCtrlLfsrSeed - default: 0x23d532c0ab + default: 0xaa3f4c7123 randwidth: 40 } { name: RndCnstLfsrPerm desc: Compile-time random permutation for LFSR output - type: otp_ctrl_pkg::lfsr_perm_t + type: otp_ctrl_top_specific_pkg::lfsr_perm_t randcount: 40 randtype: perm name_top: RndCnstOtpCtrlLfsrPerm - default: 0x35f9e27449828087a351b19228f05a15730754e0d98507166090139642d1 + default: 0x3d644385d00978610b29c6cc1671d48e460e40880d99a4a50625d565f093 randwidth: 240 } { name: RndCnstScrmblKeyInit desc: Compile-time random permutation for scrambling key/nonce register reset value - type: otp_ctrl_pkg::scrmbl_key_init_t + type: otp_ctrl_top_specific_pkg::scrmbl_key_init_t randcount: 256 randtype: data name_top: RndCnstOtpCtrlScrmblKeyInit - default: 0x55d70063277642b5309a163990b966cd494444c3bcdf8087b7facd65e9654cd8 + default: 0x86bf1f41b783b6db8c644c4723cf740f6a0563c0c2920f637247508bab4dc752 randwidth: 256 } ] @@ -2021,7 +2022,7 @@ randcount: 128 randtype: data name_top: RndCnstLcCtrlLcKeymgrDivInvalid - default: 0x5bc66d10208c4fb51b97bbf4d12c378c + default: 0x164d104b5b0b3d8fddb7a0c53617a6a3 randwidth: 128 } { @@ -2031,7 +2032,7 @@ randcount: 128 randtype: data name_top: RndCnstLcCtrlLcKeymgrDivTestUnlocked - default: 0xcfd6a5a5bf3032db51fa1eb92f6ce10e + default: 0x1cb8138a3bbdaae552050de28c64d4c1 randwidth: 128 } { @@ -2041,7 +2042,7 @@ randcount: 128 randtype: data name_top: RndCnstLcCtrlLcKeymgrDivDev - default: 0x90f9c5c06f733dc911a321dd4c0ac240 + default: 0x87725a4c748be1317e3e01c227894301 randwidth: 128 } { @@ -2051,7 +2052,7 @@ randcount: 128 randtype: data name_top: RndCnstLcCtrlLcKeymgrDivProduction - default: 0x6d467215dab3f2b54a52e6728678af07 + default: 0x78dd3c869e21d220a3543a0130b3bcfe randwidth: 128 } { @@ -2061,7 +2062,7 @@ randcount: 128 randtype: data name_top: RndCnstLcCtrlLcKeymgrDivRma - default: 0x18068e344d2eae4d73c8fa54de8aa4a4 + default: 0xc376d4bac76ac7f32652ff80cc92daef randwidth: 128 } { @@ -2071,7 +2072,7 @@ randcount: 1024 randtype: data name_top: RndCnstLcCtrlInvalidTokens - default: 0xf258f3b2a145fd1950ae55d693e57968117e66dc634155e0f8150242862eb20eea9a1f7544716551512a112ded678ce824a08e29c772795a358ec30f1bedd1af0ec9d9a9cf36a2b130d32afe121af734f628363aaa0e93d45c423690ed05bd1e710f22d1095889fbd8a7b57fb1e11df367c1c1ede3f35cfc8f6e72abdd73a9ee + default: 0x2ddfebb9b7267fef2cef1772418f8629c29c75aba36d93ffe068f655fa7990d5897e2d8d259f101a8bf1e2b746be84f30380655ee26d62dee6ae95aafb7a44e26df9cbedbc3d512527fe67e8b2e07d55510f7df200e9385195de4f2fb208a71e82207a34abd439b462f7fdb8fc20e2960fa1c637588c08c38374903b58bc3484 randwidth: 1024 } { @@ -2630,7 +2631,7 @@ randcount: 32 randtype: data name_top: RndCnstAlertHandlerLfsrSeed - default: 0xfbf8b708 + default: 0xdf9b0e9c randwidth: 32 } { @@ -2640,7 +2641,7 @@ randcount: 32 randtype: perm name_top: RndCnstAlertHandlerLfsrPerm - default: 0xd07af5ec5681152045096bb132d19ecfea7e1ae3 + default: 0x84e32be8825f903fd19bb633c7182f3abb54b405 randwidth: 160 } ] @@ -5233,7 +5234,7 @@ randcount: 128 randtype: data name_top: RndCnstSramCtrlRetAonSramKey - default: 0x42692b089a6f99b1a7ac084224ade461 + default: 0x9a0ffa956aeadb13baaa10d2336e399e randwidth: 128 } { @@ -5243,7 +5244,7 @@ randcount: 128 randtype: data name_top: RndCnstSramCtrlRetAonSramNonce - default: 0x2980d48e6c6493a50eedbac6a46b7a8a + default: 0x5f1aeb58c2a1ba65d13a0fe39b01c95a randwidth: 128 } { @@ -5253,7 +5254,7 @@ randcount: 32 randtype: data name_top: RndCnstSramCtrlRetAonLfsrSeed - default: 0x9b2c7aec + default: 0x626001cc randwidth: 32 } { @@ -5263,7 +5264,7 @@ randcount: 32 randtype: perm name_top: RndCnstSramCtrlRetAonLfsrPerm - default: 0x791b870b95b872c300aaa2fb686c7fd1e284cfd2 + default: 0x2e6e654dcce40f8a6fa302d228d9f52050dd7bf2 randwidth: 160 } { @@ -5514,7 +5515,7 @@ randcount: 128 randtype: data name_top: RndCnstFlashCtrlAddrKey - default: 0x11e64066908adfd5918f8c563c92184f + default: 0x29cb3e62ce6fddcba7f6c9d2519ea1a randwidth: 128 } { @@ -5524,7 +5525,7 @@ randcount: 128 randtype: data name_top: RndCnstFlashCtrlDataKey - default: 0x1c76a61e112578bf9f469fc482fa14a8 + default: 0xacc8e1922af7b82d7479cd41d20282eb randwidth: 128 } { @@ -5534,7 +5535,7 @@ randcount: 512 randtype: data name_top: RndCnstFlashCtrlAllSeeds - default: 0xf2b75c265c9cb3b2f1c44a7287ab4c0618822e92a084a048ffe9fb86140a79d38055b24b5f3b3ea79bd7f203d55eabf9f733d9b0cc2ca2b394f2526e4430c2af + default: 0xd61aa54cc696d5416af2545001dc3dbca07b204875b3a8be8207d2509c0d9257addc105a37be10e82f69c5d067ac23a86224f25dab6226bf545b7fc56675745 randwidth: 512 } { @@ -5544,7 +5545,7 @@ randcount: 32 randtype: data name_top: RndCnstFlashCtrlLfsrSeed - default: 0xb4f5b43b + default: 0xfeec587d randwidth: 32 } { @@ -5554,7 +5555,7 @@ randcount: 32 randtype: perm name_top: RndCnstFlashCtrlLfsrPerm - default: 0xda8c1fc48584aba1e2cb43d2dcd1c7e5fbe98182 + default: 0xa206dc5f5621f91ffaac3402f5a41d16e6e548b9 randwidth: 160 } { @@ -6450,7 +6451,7 @@ randcount: 64 randtype: data name_top: RndCnstAesClearingLfsrSeed - default: 0xec27f13eb602172c + default: 0x95e25e95d30f4592 randwidth: 64 } { @@ -6460,7 +6461,7 @@ randcount: 64 randtype: perm name_top: RndCnstAesClearingLfsrPerm - default: 0x352b301e82238b40c93bbf82ad9c4b5fd798caa906f1d45552d9e17ff105e60dcf5b86d073398aa766ba075baf300953 + default: 0x153fe9b41c646aa6607c0d73ba69d044222212561730e8436f43d27b2f87546e4ab23f772d652bf3aa30bcd278736edd randwidth: 384 } { @@ -6470,7 +6471,7 @@ randcount: 64 randtype: perm name_top: RndCnstAesClearingSharePerm - default: 0x347b55e5dfa46903140e8241db2fc0f67bcb7a38520972b01383c5d6b3a5d16ba07c69b3c6cddb66262aa5121c4fbf3a + default: 0x97226882bfbd6420f834a7b77509e96fac140588dcd84d576b42eeb394ce3c5846cd221502dee2bfc44c7c75b1aa4fe6 randwidth: 384 } { @@ -6480,7 +6481,7 @@ randcount: 288 randtype: data name_top: RndCnstAesMaskingLfsrSeed - default: 0x80499e6fcce64ceafd282c0e33fd2c07986c2a511755f072ee7a8e2fc45e111df07fa067 + default: 0x784a0c438ab14de7812b74cc981f89b9b95175a8f02ea42f7883754a0eaa875bd05ee93e randwidth: 288 } { @@ -6490,7 +6491,7 @@ randcount: 160 randtype: perm name_top: RndCnstAesMaskingLfsrPerm - default: 0x5227992e22693a163f8b5d8d58914a1e3283106a866c4c2d760e497831847c29673779240a710d9581424147066639488702639f7a144f178009209d0834447b7d612c332a0c6f5b901c007f890b1b0f351803573c151f6d9e3e6082774d28198a74941a1d73729a53884e6865114623459b8f503b07923d387e596b266e97968e4b628c702b21015c2f510413125a9c935f5455645e98052536754385305640 + default: 0x3c314f0b95181072006d3b3a1277908273110a74912c5b879d395d2742045999011b3e165c83216397815e945a498409257c3d229c8a853707081d609f266f0d5f20708024301e0c23922f667e7819454c9a5132754e35674d93712d4b474a89627a367f699646568c5053792a06543f1a1c9b98156113688e1f43336a38038d287b9e402b57440241052e6552640e0f5588176b7d6e8b6c8f48341476862958 randwidth: 1280 } ] @@ -6751,7 +6752,7 @@ randcount: 288 randtype: data name_top: RndCnstKmacLfsrSeed - default: 0xb666a5e22d3320f89caa47e5ced5d29cc9cdb468773ebacec14da34c505aedc70ea184c7 + default: 0x534d7ac4bd0ca8b03546a98188556c8cbf9b24d1d32c9bd86dfed26f1fe72ffbb4b9762d randwidth: 288 } { @@ -6761,7 +6762,7 @@ randcount: 800 randtype: perm name_top: RndCnstKmacLfsrPerm - default: 0x3181458084094d29114aaf9e6620f99adec77c9ba71ac27ac3b24c369a2380eee7ad87899d20e92f178ba48cc15c4ad288ee8389c0c94d23069974b6781ca7947d5580d156ef1f2687c2918bc2bb2852789c6c2ab686d75a79d9a0cdc32dd448a504159b466ed4aced492ac3430b878bb97d2d654f75d2897309911dd65a319b345f9dce184636ba4889e16f4b19734c388b5906d0c25f0ff1c88dbcd5c09f09a8e90a871e06312c0d451fc11aa10da15f4bc6c19ff06800801923d2cca0770e7b9ece1000ab184455451678bf2466f1e609995c90b4c5424aec7c2a6c17e0549f8c4570497a7f2320cde16bd5812d31081779b495a6ac71a2e2c04495384e7d9064394b808500fd5e944e990416c628f2619a165b67b515a66ff91aa6659b935dd19ceaf9e9432c4abafc09b22e0681781b27d6a64b9d313078bd9d62c401ba5021d19e07c30836a1aa46aeb49e643b1613c672c5406c3e3e3cf00c6ad42e6b19788b0f8a1b01921f8d11982e3260c3217bc09d58dcdda5c81332d96f0be2d518631d091c6a8e6288a57d625cb268da4a1251eacb999e73da7b0d076216131b6176fe585ad6c36d204eaab58ce44dd33908ef3c9c203801180002bdd829e4038a594083060510185f99e5397c59da1d34be8d67a11b64dab73cc062eea4740495aecbf2e1a2ed1592609270841304471118d4086016a074b73a048852382b0e00bea459c6558a080dc6b4d8246eef8734b41c1fda11122988668c713edc751ecf862e638eada39a11368a106dd6725bb72d34aaba8960aa08a4806b7b91c513e963610c356bd5a610b7e702284ac5c28c1c136080f70ed28bab99855721cc45d9d38711f28dada04fe7201c0c5dbad6438f05dc368c6964c7b9b224950884ca2a88f1d40b4fdb15c31669e222c0fc4055d8d2931609489d99188f03eab2970bc0e58e8a81f7dc297eaf0b58a5880e6619796ce7cbd8de84ad07907ba56f52406f94cfdc7108ae2feb8d9fc6dd7c382125b1dace7ea956eb78df7edf5869420d5488e929b32bb82c034f04631d44815be4299424c1942d4c351ae4ecb2a80022a20fc531122680a61265989d5fcf814db56346e041f20f507b610fbe6a789263bf55a54436366541713e099839b1f8459419151e645a4152f4938058fe4d87f108aed7525de7a53057db799ddd1d1ce8edf344c07c145a116a933c2256ef20ec1b6e1a2a197395c5b610b8ac29498a9b10a5a0f082616e5ba2bc711da92ada4515918def3ae040b216b55eab118f6f4cda766e4b9fe95692485547feb44f5c1948f478c788cc8e866a2b169f62903b095381adb05ef74c823a2146339983aaaa836155497245ce59207353cf5132691581a9d80c594a99a87725c43702304d6aebed385b67193e8d + default: 0x3f69d7b0444f62fb85e1bbcce9a23d2529fc2ca657aa713b124da4521a14c7d29761753441488b0585d2c74c7f2c23c8cc1179ec9b7145489bb894f5c7a124a973566537dc7c3cddbb06b6b1c384714216df2afdf0211c39b9824695c3e8d7921a37a8b787506c10f8399465bd559956913590c4c3100c189c5a7a38e4aa87e2d56da015aa9037ad08b54d847ec2041c3e73a36802cb6b6cd7b4450c14c312c834af80a966150cfab0b2034197a16e926e8835a58b53f751203cafaadc852c6930088d780fb86cd3096eb4bd0c8a232a408304ce187c0753ada40a9c6012d4ce3b174a816acf118217da20809a60f579559442ed962fd8cb971659ca7c25cba9abc548c64de7610639ae68c0e46e0156aa35a923c084a23e909627143a076097f5f1a2032231abb6c800984b92c1a0cca0bc6a53f8eb45e431601536d710d47578d87816a1b5ae7b0c599846251d34c22df3b1d25971b9719b1a21591ade77dc7bba844e0a576a7744914c04e24169e8ba9257d8b2b0dd86a555e2a8142c2306f2a6e86a5149c680a0d13bc72403a57e910978d0ba98ac0301408059305b19a0976400f0710703327e222f02b8dfc740e42a634c509a5c7819702b38632b4f19e53c844016fdaa4389c13e0a238a9aac9308e593e5d017ec4279ff31e789b5e21b6b05de8ac0c0b09e52812be46631370e0146d74981baa64c6e4b682eccbda746d13905e87bfca45c83bbcd884c1d1980871b02e2aa7f2261087b10af5a11bdb59066636918205b22757a9bc9555237724cb3567c026729470e9c1f4c603c28557898cd5c26c0bc297f1952cd6bc344063df9bb5cc6e0283d2d5b1ad044c0ca589f9cd8d7642b86135822eaa890f7a698abc47858d007cefab93ea83000419db8e4f5a06983d4367a5bad5fe19a29b770c80cdec1a5e91dc85d2f51846475c128e1153fd037996ca694b03dcd63a24a2d9984d81ab4571a8b9646f43096466aba94056929a22468f9a6588c01c5fc9b26223359f846c7b60e6b5eee06606b4125f5770f12472288c88302daccbe61a3f6487d8aa4d9d4bdb292789c821705e3f1f3871c0362d19f4674752344076b511e67db3b64e830e2c39d48240af53eb84e81e7adee325b09a9be52a89bf1500d873495c1c1cc510d611289aac4ed7752142ce31c5ca3b855b4c5349299c5e19a0c742d23aa54aba8d97616d2969110102401aec6de644cb0aa15582f0d962621352567d4e5beddcb8a4e0fc7a6f5ba7967a791ef838a36b9de9500095d4173e35229603297d6a0b812267ac9d6459fabf84c3dd2b628f675508b9ae9b912e435a4683151dd51bf563bdd1f406bbb157c1e71181ceab625d6b19c8adb16f1b6109c69ef145342225899249c2be5afa042c14a066515ba4460 randwidth: 8000 } { @@ -6771,7 +6772,7 @@ randcount: 800 randtype: data name_top: RndCnstKmacBufferLfsrSeed - default: 0xac29042563ae4733313ad2696788a8735b27982f5007a24a125f876c9ca6ddeb52ac92f5600d2a41bc465efe57aa3b59bd6b0a40216363420f041ed29a965b68faea3e59d4f2e2a0b42eb9a8c3ab227b50766d2eaec89eb13b07e9bf4ee42784136d42c8 + default: 0xb7480d9d5001b7755de3ef2618e28371dd02b1db04469e5770d10b0069e332be0f35a726b339d953a1dd95a3f78e14389414adc19300e0934269b3a30b61ea4d9790de0fb09d2792d26bf9565775a2843ffb7d42993c0a1c3d4d2937f997a3b51aafebc6 randwidth: 800 } { @@ -6781,7 +6782,7 @@ randcount: 64 randtype: perm name_top: RndCnstKmacMsgPerm - default: 0x73a93266a6b6e1fb1dce0b54c02f3e0063eb8bd5858f5418a040791c3c694cd48c9ca20bde5eef7a6ba157fd0939b457 + default: 0x2a89790db63eff5a927c7d96c2f0b23ba17cad7ede662023b24ccce2e804a4b689b48510f749a719c4c1ddd3d5851c4d randwidth: 384 } ] @@ -6975,7 +6976,7 @@ randcount: 256 randtype: data name_top: RndCnstOtbnUrndPrngSeed - default: 0xa8f072e77ee4859b243f92ceb098be6e65c359480858c39611e658c97fb8d3e9 + default: 0x4569b54e4cc3468ba78749a7093fe9eb381742e695130a4fb5238ebb74b4f311 randwidth: 256 } { @@ -7013,7 +7014,7 @@ randcount: 128 randtype: data name_top: RndCnstOtbnOtbnKey - default: 0xcc0e7f8fa8725200c726c8f90ef5e2b7 + default: 0x8274ea0b356b051535eac4e098ea131f randwidth: 128 } { @@ -7023,7 +7024,7 @@ randcount: 64 randtype: data name_top: RndCnstOtbnOtbnNonce - default: 0x4beee27cfa19ea57 + default: 0x95c233e670a4321b randwidth: 64 } ] @@ -7258,7 +7259,7 @@ randcount: 64 randtype: data name_top: RndCnstKeymgrLfsrSeed - default: 0x6ca8a8225c8e1705 + default: 0xad599d4df689612a randwidth: 64 } { @@ -7268,7 +7269,7 @@ randcount: 64 randtype: perm name_top: RndCnstKeymgrLfsrPerm - default: 0xb931fb3b12efadcdaa1008bc159489066f5d9c8321e10c03e5e61b0a893e95fd0654ddffcace97a54cd668a8358ed44f + default: 0x961d0532d78d8ace1ff82767391f0f515416ebf4a96b95fb20bc2e1c472064af75c406efad88d3df3903076ca8266a86 randwidth: 384 } { @@ -7278,7 +7279,7 @@ randcount: 32 randtype: perm name_top: RndCnstKeymgrRandPerm - default: 0xb87fdde58a44756a1c05f72093384394c8d7af15 + default: 0x95ca1a671b6b87f5f2cfd226a0445e21bb53c12c randwidth: 160 } { @@ -7288,7 +7289,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrRevisionSeed - default: 0x20bdaf59fe2ae209b8325d2c8c35fc960679b106a87e59e6aeb1b5302d334010 + default: 0x46794cfca2b0c6cbfb61f1bd7ed0a8f926da506ed059049dc9a6e2c1826aa5bf randwidth: 256 } { @@ -7298,7 +7299,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrCreatorIdentitySeed - default: 0x70251696fbb4ba169a6cd82fad46a0a5c04009bb934c7ef7b83b6e40610b4309 + default: 0xdf596fb9a391c6722b8b5cad051f3edb286e554001528f8e9a5e55ebb82a171c randwidth: 256 } { @@ -7308,7 +7309,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrOwnerIntIdentitySeed - default: 0xfcb9dc54d4276137f9901d09a76aeaa19de5c579fd38bdf38f0c21d6a52226b6 + default: 0xf1e411adfe91411b1d8ee0161ecd699002ef8026e474e85739ef0277414266b4 randwidth: 256 } { @@ -7318,7 +7319,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrOwnerIdentitySeed - default: 0xa4a7bdb05fe921615bf2ff984540f7d43ece76b4eb13363774ed2ed45545e927 + default: 0xf811ebd01e7ea4acf08b65a5cf51ae528473dd4661d013fb441057830dd1b830 randwidth: 256 } { @@ -7328,7 +7329,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrSoftOutputSeed - default: 0xf6d9e4abac398d42c745eef646c1464dca86dafd7c7c71e6058ddfd871c51cac + default: 0xbf365bbfc1deb8921d7a6896fd1e6013622d66142ee83239ba8e92c61de956a5 randwidth: 256 } { @@ -7338,7 +7339,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrHardOutputSeed - default: 0xbaf4410f06dcc036fcd16fcde97d171891105dd895e3a1d0a19a16d6dbd8b20f + default: 0x2768b6d5f1650f1b01969710a957dbfb2e2443f4fe2d0a760f83c1313d3ef78f randwidth: 256 } { @@ -7348,7 +7349,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrAesSeed - default: 0xc9e662e1e1b4982b3e8eff63890dbeae926fd468a77efde3de5b4caf4776a247 + default: 0xea4d3d48549217248330d048f78e497e63710e9720ac0f46c68e74cab245be89 randwidth: 256 } { @@ -7358,7 +7359,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrKmacSeed - default: 0xbaba4c9908ed16bc5415ec16d28c53551312fcedcf2832a66ceacf8ed4d5b616 + default: 0x11081c180f03480661762d84bfb803130ea592f63f968d16d49748cbc79f398e randwidth: 256 } { @@ -7368,7 +7369,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrOtbnSeed - default: 0x177dd43b56fa754e1f53ad658193b563d9bdc6d2aee84852f0cc7371ed3a6faa + default: 0x2139541731f82f0e5e0f2ea961c615fc96398950eded5e999cbb16670229d5a2 randwidth: 256 } { @@ -7378,7 +7379,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrCdi - default: 0x516c144b34c96dfabb6f6e79208a74f35c79f397c4e4c7e22b7581848a90a125 + default: 0x4df595be24c50547533e86ef45dc765bc3fc1c4492b8cd264903ba93e2074570 randwidth: 256 } { @@ -7388,7 +7389,7 @@ randcount: 256 randtype: data name_top: RndCnstKeymgrNoneSeed - default: 0x4b2276bec9fc1a7a5722a9aaca4db84a32e5c6985a1a6435118b5f5f3fd3b931 + default: 0x4fc7f5a263f7641c180b43ff715aa5a2bd79305346ebe883e11eea60c36419a2 randwidth: 256 } ] @@ -7602,7 +7603,7 @@ randcount: 384 randtype: data name_top: RndCnstCsrngCsKeymgrDivNonProduction - default: 0x3116b67192d856dd742d40e072e417ad9a5f261204cbcc69a2147819b290a4bb0264347c0f91b0cec93c0129d85b9580 + default: 0x27f0cbb5f22278f04abe69416755d0a65694cb702f93938638ac23e229502708c3001f75853897b4e8ef27d17c5f7fb3 randwidth: 384 } { @@ -7612,7 +7613,7 @@ randcount: 384 randtype: data name_top: RndCnstCsrngCsKeymgrDivProduction - default: 0x1dbe03f17e69dc260a79998eff5c710b67838adfb99a5eca671672cf19bca6adc1ebfab611f19e7262e0a491970e51b2 + default: 0x9afb3660baba1f1bac202d76e627cbb30ad1bc1e181db312b8d5a9479c07c53490303552ea4b9fbde86f64da79331682 randwidth: 384 } { @@ -8132,7 +8133,7 @@ randcount: 128 randtype: data name_top: RndCnstSramCtrlMainSramKey - default: 0x8e7026d0e5dce9e265a416a812231cfa + default: 0x785d8680f24b3700db2a343fe142fe18 randwidth: 128 } { @@ -8142,7 +8143,7 @@ randcount: 128 randtype: data name_top: RndCnstSramCtrlMainSramNonce - default: 0x76e3d982f8fdcec9ae1af0b8cd9e1760 + default: 0xb7568e2e1afa5ba5cf4337a8977e31c7 randwidth: 128 } { @@ -8152,7 +8153,7 @@ randcount: 32 randtype: data name_top: RndCnstSramCtrlMainLfsrSeed - default: 0x6ec98106 + default: 0xb94c2bcf randwidth: 32 } { @@ -8162,7 +8163,7 @@ randcount: 32 randtype: perm name_top: RndCnstSramCtrlMainLfsrPerm - default: 0x85097395a09dabee6ddd67d1948b1a899435be41 + default: 0xacb1d6845459bd1228ee4fd059e59a85edbe0de0 randwidth: 160 } { @@ -8402,7 +8403,7 @@ randcount: 64 randtype: data name_top: RndCnstRomCtrlScrNonce - default: 0xfee457dee82b6e06 + default: 0x3f40a5e816a506c2 randwidth: 64 } { @@ -8412,7 +8413,7 @@ randcount: 128 randtype: data name_top: RndCnstRomCtrlScrKey - default: 0x663c291739ff0e7d644758fee1c58564 + default: 0xab1f4a2f01cbe908a34c2a20fecda9a5 randwidth: 128 } { @@ -8615,7 +8616,7 @@ randcount: 32 randtype: data name_top: RndCnstRvCoreIbexLfsrSeed - default: 0xcf346d96 + default: 0xfceba680 randwidth: 32 } { @@ -8625,7 +8626,7 @@ randcount: 32 randtype: perm name_top: RndCnstRvCoreIbexLfsrPerm - default: 0xae357a7a3b2cd2063d7f3ab3043ac11b7b211b84 + default: 0x791c0627955e85ec954138f72a5a283426dc77f7 randwidth: 160 } { @@ -8635,7 +8636,7 @@ randcount: 128 randtype: data name_top: RndCnstRvCoreIbexIbexKeyDefault - default: 0xc652d7372432fcc4d3092bd6383fdba3 + default: 0x1988481250c2c82deb519d69c7051aab randwidth: 128 } { @@ -8645,7 +8646,7 @@ randcount: 64 randtype: data name_top: RndCnstRvCoreIbexIbexNonceDefault - default: 0x5279fcbcd2bd2c13 + default: 0x9fc45a1658fa9063 randwidth: 64 } { diff --git a/hw/ip/otp_ctrl/data/BUILD b/hw/top_earlgrey/data/otp/BUILD similarity index 94% rename from hw/ip/otp_ctrl/data/BUILD rename to hw/top_earlgrey/data/otp/BUILD index 5f742a53819946..3bb3b66d572c87 100644 --- a/hw/ip/otp_ctrl/data/BUILD +++ b/hw/top_earlgrey/data/otp/BUILD @@ -2,7 +2,6 @@ # Licensed under the Apache License, Version 2.0, see LICENSE for details. # SPDX-License-Identifier: Apache-2.0 -load("@bazel_skylib//rules:common_settings.bzl", "int_flag", "string_flag") load("@rules_pkg//pkg:mappings.bzl", "pkg_files") load("//rules:const.bzl", "CONST", "EARLGREY_ALERTS", "EARLGREY_LOC_ALERTS") load( @@ -22,47 +21,10 @@ load( package(default_visibility = ["//visibility:public"]) -# These configurations expose the OTP image generation tool's command line -# arguments to enable dvsim to pass this through Bazel to the underlying OTP -# image generation script. This is required to enable dvsim to invoke OTP image -# generation as part of the Bazel build process, while still enabling the use of -# multiple seeds needed to achieve DV coverage. -int_flag( - name = "img_seed", - build_setting_default = 0, -) - -string_flag( - name = "lc_seed", - # Default must match value in hw/ip/lc_ctrl/data/lc_ctrl.hjson. - build_setting_default = "40182201019264397688411770949626922549663256047001778394918990008320537410392", -) - -string_flag( - name = "otp_seed", - # Default must match value in hw/ip/otp_ctrl/data/otp_ctrl_mmap.hjson. - build_setting_default = "36021179872380457113239299468132194022238108125576166239904535336103582949069", -) - -string_flag( - name = "data_perm", - build_setting_default = "", -) - # This package must be kept in sync with get_otp_images() from //rules:otp.bzl. # That is, each OTP image referenced by the macro should have a definition in # this BUILD file. -filegroup( - name = "all_files", - srcs = glob(["**"]), -) - -exports_files([ - "otp_ctrl_img.h.tpl", - "otp_ctrl_img.c.tpl", -]) - exports_files(["otp_ctrl_mmap.hjson"]) otp_json( diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_img_creator_sw_cfg.hjson b/hw/top_earlgrey/data/otp/otp_ctrl_img_creator_sw_cfg.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_img_creator_sw_cfg.hjson rename to hw/top_earlgrey/data/otp/otp_ctrl_img_creator_sw_cfg.hjson diff --git a/hw/top_earlgrey/data/otp/otp_ctrl_img_dev.hjson b/hw/top_earlgrey/data/otp/otp_ctrl_img_dev.hjson new file mode 100644 index 00000000000000..0f1928a4185afc --- /dev/null +++ b/hw/top_earlgrey/data/otp/otp_ctrl_img_dev.hjson @@ -0,0 +1,77 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Use the gen-otp-img.py script to convert this configuration into +// a MEM file for preloading the OTP in FPGA synthesis or simulation. +// + +{ + // 256 bit seed to be used for generation of partition randomized values. + // Can be overridden on the command line with the --seed switch. + // The default seed was generated using secrets.py module. + seed: 85452983286950371191603618368782861611109037138182535346147818831008789508651 + + // The partition and item names must correspond with the OTP memory map. + partitions: [ + { + name: "SECRET0", + lock: "True", + items: [ + { + name: "TEST_UNLOCK_TOKEN", + value: "", + } + { + name: "TEST_EXIT_TOKEN", + value: "", + } + ], + } + { + name: "SECRET1", + lock: "True", + items: [ + { + name: "FLASH_ADDR_KEY_SEED", + value: "", + } + { + name: "FLASH_DATA_KEY_SEED", + value: "", + } + { + name: "SRAM_DATA_KEY_SEED", + value: "", + } + ], + } + { + name: "SECRET2", + lock: "False", + items: [ + { + name: "RMA_TOKEN", + value: "", + } + { + name: "CREATOR_ROOT_KEY_SHARE0", + value: "", + } + { + name: "CREATOR_ROOT_KEY_SHARE1", + value: "", + } + ], + } + { + name: "LIFE_CYCLE", + // Can be one of the following strings: + // RAW, TEST_UNLOCKED0-3, TEST_LOCKED0-2, DEV, PROD, PROD_END, RMA, SCRAP + state: "DEV", + // Can range from 0 to 16. + // Note that a value of 0 is only permissible in RAW state. + count: "5" + } + ] +} diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_img_hw_cfg.hjson b/hw/top_earlgrey/data/otp/otp_ctrl_img_hw_cfg.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_img_hw_cfg.hjson rename to hw/top_earlgrey/data/otp/otp_ctrl_img_hw_cfg.hjson diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_img_owner_sw_cfg.hjson b/hw/top_earlgrey/data/otp/otp_ctrl_img_owner_sw_cfg.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_img_owner_sw_cfg.hjson rename to hw/top_earlgrey/data/otp/otp_ctrl_img_owner_sw_cfg.hjson diff --git a/hw/top_earlgrey/data/otp/otp_ctrl_img_prod.hjson b/hw/top_earlgrey/data/otp/otp_ctrl_img_prod.hjson new file mode 100644 index 00000000000000..0a772f007e95f8 --- /dev/null +++ b/hw/top_earlgrey/data/otp/otp_ctrl_img_prod.hjson @@ -0,0 +1,77 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Use the gen-otp-img.py script to convert this configuration into +// a MEM file for preloading the OTP in FPGA synthesis or simulation. +// + +{ + // 256 bit seed to be used for generation of partition randomized values. + // Can be overridden on the command line with the --seed switch. + // The default seed was generated using secrets.py module. + seed: 113517944176559405110937879233240229311794601727326023435899657066678782830485 + + // The partition and item names must correspond with the OTP memory map. + partitions: [ + { + name: "SECRET0", + lock: "True", + items: [ + { + name: "TEST_UNLOCK_TOKEN", + value: "", + } + { + name: "TEST_EXIT_TOKEN", + value: "", + } + ], + } + { + name: "SECRET1", + lock: "True", + items: [ + { + name: "FLASH_ADDR_KEY_SEED", + value: "", + } + { + name: "FLASH_DATA_KEY_SEED", + value: "", + } + { + name: "SRAM_DATA_KEY_SEED", + value: "", + } + ], + } + { + name: "SECRET2", + lock: "False", + items: [ + { + name: "RMA_TOKEN", + value: "", + } + { + name: "CREATOR_ROOT_KEY_SHARE0", + value: "", + } + { + name: "CREATOR_ROOT_KEY_SHARE1", + value: "", + } + ], + } + { + name: "LIFE_CYCLE", + // Can be one of the following strings: + // RAW, TEST_UNLOCKED0-3, TEST_LOCKED0-2, DEV, PROD, PROD_END, RMA, SCRAP + state: "PROD", + // Can range from 0 to 16. + // Note that a value of 0 is only permissible in RAW state. + count: "5" + } + ] +} diff --git a/hw/top_earlgrey/data/otp/otp_ctrl_img_raw.hjson b/hw/top_earlgrey/data/otp/otp_ctrl_img_raw.hjson new file mode 100644 index 00000000000000..e5a94be7cabfb5 --- /dev/null +++ b/hw/top_earlgrey/data/otp/otp_ctrl_img_raw.hjson @@ -0,0 +1,27 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Use the gen-otp-img.py script to convert this configuration into +// a MEM file for preloading the OTP in FPGA synthesis or simulation. +// + +{ + // Seed to be used for generation of partition randomized values. + // Can be overridden on the command line with the --seed switch. + // The default seed was generated using secrets.py module. + seed: 52408960416235844780753299194502148156786072650816676092165912261205302331741 + + // The partition and item names must correspond with the OTP memory map. + partitions: [ + { + name: "LIFE_CYCLE", + // Can be one of the following strings: + // RAW, TEST_UNLOCKED0-3, TEST_LOCKED0-2, DEV, PROD, PROD_END, RMA, SCRAP + state: "RAW", + // Can range from 0 to 16. + // Note that a value of 0 is only permissible in RAW state. + count: 0 + } + ] +} diff --git a/hw/top_earlgrey/data/otp/otp_ctrl_img_rma.hjson b/hw/top_earlgrey/data/otp/otp_ctrl_img_rma.hjson new file mode 100644 index 00000000000000..f28e69cb61cf32 --- /dev/null +++ b/hw/top_earlgrey/data/otp/otp_ctrl_img_rma.hjson @@ -0,0 +1,77 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Use the gen-otp-img.py script to convert this configuration into +// a MEM file for preloading the OTP in FPGA synthesis or simulation. +// + +{ + // Seed to be used for generation of partition randomized values. + // Can be overridden on the command line with the --seed switch. + // The default seed was generated using secrets.py module. + seed: 52408960416235844780753299194502148156786072650816676092165912261205302331741 + + // The partition and item names must correspond with the OTP memory map. + partitions: [ + { + name: "SECRET0", + lock: "True", + items: [ + { + name: "TEST_UNLOCK_TOKEN", + value: "", + } + { + name: "TEST_EXIT_TOKEN", + value: "", + } + ], + } + { + name: "SECRET1", + lock: "True", + items: [ + { + name: "FLASH_ADDR_KEY_SEED", + value: "", + } + { + name: "FLASH_DATA_KEY_SEED", + value: "", + } + { + name: "SRAM_DATA_KEY_SEED", + value: "", + } + ], + } + { + name: "SECRET2", + lock: "False", + items: [ + { + name: "RMA_TOKEN", + value: "", + } + { + name: "CREATOR_ROOT_KEY_SHARE0", + value: "", + } + { + name: "CREATOR_ROOT_KEY_SHARE1", + value: "", + } + ], + } + { + name: "LIFE_CYCLE", + // Can be one of the following strings: + // RAW, TEST_UNLOCKED0-3, TEST_LOCKED0-2, DEV, PROD, PROD_END, RMA, SCRAP + state: "RMA", + // Can range from 0 to 16. + // Note that a value of 0 is only permissible in RAW state. + count: "8" + } + ] +} diff --git a/hw/top_earlgrey/data/otp/otp_ctrl_img_test_locked0.hjson b/hw/top_earlgrey/data/otp/otp_ctrl_img_test_locked0.hjson new file mode 100644 index 00000000000000..9f93b53d48e39f --- /dev/null +++ b/hw/top_earlgrey/data/otp/otp_ctrl_img_test_locked0.hjson @@ -0,0 +1,37 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Use the gen-otp-img.py script to convert this configuration into +// a MEM file for preloading the OTP in FPGA synthesis or simulation. +// + +{ + // Seed to be used for generation of partition randomized values. + // Can be overridden on the command line with the --seed switch. + seed: 52408960416235844780753299194502148156786072650816676092165912261205302331741 + + // The partition and item names must correspond with the OTP memory map. + partitions: [ + { + name: "CREATOR_SW_CFG", + items: [ + { + name: "CREATOR_SW_CFG_ROM_EXEC_EN", + // ROM execution is enabled if this item is set to a + // non-zero value. + value: "0xffffffff", + }, + ], + } + { + name: "LIFE_CYCLE", + // Can be one of the following strings: + // RAW, TEST_UNLOCKED0-3, TEST_LOCKED0-2, DEV, PROD, PROD_END, RMA, SCRAP + state: "TEST_LOCKED0", + // Can range from 0 to 16. + // Note that a value of 0 is only permissible in RAW state. + count: 2 + } + ] +} diff --git a/hw/top_earlgrey/data/otp/otp_ctrl_img_test_locked1.hjson b/hw/top_earlgrey/data/otp/otp_ctrl_img_test_locked1.hjson new file mode 100644 index 00000000000000..92c9da20054127 --- /dev/null +++ b/hw/top_earlgrey/data/otp/otp_ctrl_img_test_locked1.hjson @@ -0,0 +1,37 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Use the gen-otp-img.py script to convert this configuration into +// a MEM file for preloading the OTP in FPGA synthesis or simulation. +// + +{ + // Seed to be used for generation of partition randomized values. + // Can be overridden on the command line with the --seed switch. + seed: 52408960416235844780753299194502148156786072650816676092165912261205302331741 + + // The partition and item names must correspond with the OTP memory map. + partitions: [ + { + name: "CREATOR_SW_CFG", + items: [ + { + name: "CREATOR_SW_CFG_ROM_EXEC_EN", + // ROM execution is enabled if this item is set to a + // mnon-zero value. + value: "0xffffffff", + }, + ], + } + { + name: "LIFE_CYCLE", + // Can be one of the following strings: + // RAW, TEST_UNLOCKED0-3, TEST_LOCKED0-2, DEV, PROD, PROD_END, RMA, SCRAP + state: "TEST_LOCKED1", + // Can range from 0 to 16. + // Note that a value of 0 is only permissible in RAW state. + count: 2 + } + ] +} diff --git a/hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked0.hjson b/hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked0.hjson new file mode 100644 index 00000000000000..c6cd69f81e168f --- /dev/null +++ b/hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked0.hjson @@ -0,0 +1,37 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Use the gen-otp-img.py script to convert this configuration into +// a MEM file for preloading the OTP in FPGA synthesis or simulation. +// + +{ + // Seed to be used for generation of partition randomized values. + // Can be overridden on the command line with the --seed switch. + seed: 52408960416235844780753299194502148156786072650816676092165912261205302331741 + + // The partition and item names must correspond with the OTP memory map. + partitions: [ + { + name: "CREATOR_SW_CFG", + items: [ + { + name: "CREATOR_SW_CFG_ROM_EXEC_EN", + // ROM execution is enabled if this item is set to a + // non-zero value. + value: "0xffffffff", + }, + ], + } + { + name: "LIFE_CYCLE", + // Can be one of the following strings: + // RAW, TEST_UNLOCKED0-3, TEST_LOCKED0-2, DEV, PROD, PROD_END, RMA, SCRAP + state: "TEST_UNLOCKED0", + // Can range from 0 to 16. + // Note that a value of 0 is only permissible in RAW state. + count: 1 + } + ] +} diff --git a/hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked1.hjson b/hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked1.hjson new file mode 100644 index 00000000000000..f8d5a975f6917a --- /dev/null +++ b/hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked1.hjson @@ -0,0 +1,37 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Use the gen-otp-img.py script to convert this configuration into +// a MEM file for preloading the OTP in FPGA synthesis or simulation. +// + +{ + // Seed to be used for generation of partition randomized values. + // Can be overridden on the command line with the --seed switch. + seed: 52408960416235844780753299194502148156786072650816676092165912261205302331741 + + // The partition and item names must correspond with the OTP memory map. + partitions: [ + { + name: "CREATOR_SW_CFG", + items: [ + { + name: "CREATOR_SW_CFG_ROM_EXEC_EN", + // ROM execution is enabled if this item is set to a + // non-zero value. + value: "0xffffffff", + }, + ], + } + { + name: "LIFE_CYCLE", + // Can be one of the following strings: + // RAW, TEST_UNLOCKED0-3, TEST_LOCKED0-2, DEV, PROD, PROD_END, RMA, SCRAP + state: "TEST_UNLOCKED1", + // Can range from 0 to 16. + // Note that a value of 0 is only permissible in RAW state. + count: 3 + } + ] +} diff --git a/hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked2.hjson b/hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked2.hjson new file mode 100644 index 00000000000000..07228a53a45a30 --- /dev/null +++ b/hw/top_earlgrey/data/otp/otp_ctrl_img_test_unlocked2.hjson @@ -0,0 +1,37 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Use the gen-otp-img.py script to convert this configuration into +// a MEM file for preloading the OTP in FPGA synthesis or simulation. +// + +{ + // Seed to be used for generation of partition randomized values. + // Can be overridden on the command line with the --seed switch. + seed: 52408960416235844780753299194502148156786072650816676092165912261205302331741 + + // The partition and item names must correspond with the OTP memory map. + partitions: [ + { + name: "CREATOR_SW_CFG", + items: [ + { + name: "CREATOR_SW_CFG_ROM_EXEC_EN", + // ROM execution is enabled if this item is set to a + // non-zero value. + value: "0xffffffff", + }, + ], + } + { + name: "LIFE_CYCLE", + // Can be one of the following strings: + // RAW, TEST_UNLOCKED0-3, TEST_LOCKED0-2, DEV, PROD, PROD_END, RMA, SCRAP + state: "TEST_UNLOCKED2", + // Can range from 0 to 16. + // Note that a value of 0 is only permissible in RAW state. + count: 4 + } + ] +} diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_mmap.hjson b/hw/top_earlgrey/data/otp/otp_ctrl_mmap.hjson similarity index 100% rename from hw/ip/otp_ctrl/data/otp_ctrl_mmap.hjson rename to hw/top_earlgrey/data/otp/otp_ctrl_mmap.hjson diff --git a/hw/ip/otp_ctrl/data/earlgrey_skus/sival/BUILD b/hw/top_earlgrey/data/otp/sival_skus/BUILD similarity index 94% rename from hw/ip/otp_ctrl/data/earlgrey_skus/sival/BUILD rename to hw/top_earlgrey/data/otp/sival_skus/BUILD index f063bc7c14c7be..b1603dc65621b6 100644 --- a/hw/ip/otp_ctrl/data/earlgrey_skus/sival/BUILD +++ b/hw/top_earlgrey/data/otp/sival_skus/BUILD @@ -235,7 +235,7 @@ LC_MISSION_STATES = get_lc_items( # This profile represents the OTP state of a device that has completed CP # provisioning. MANUF_INITIALIZED = [ - "//hw/ip/otp_ctrl/data:otp_json_fixed_secret0", + "//hw/top_earlgrey/data/otp:otp_json_fixed_secret0", ] # The `MANUF_SW_INITIALIZED` OTP profile configures the following partitions: @@ -257,23 +257,24 @@ MANUF_SW_INITIALIZED = [ # It also includes the `MANUF_INITIALIZED` profile defined above. It represents # the OTP state of a device that has completed FT individualize provisioning. MANUF_INDIVIDUALIZED = MANUF_INITIALIZED + MANUF_SW_INITIALIZED + OTP_SIGVERIFY_FAKE_KEYS + [ - "//hw/ip/otp_ctrl/data:otp_json_hw_cfg0", - "//hw/ip/otp_ctrl/data:otp_json_hw_cfg1", + "//hw/top_earlgrey/data/otp:otp_json_hw_cfg0", + "//hw/top_earlgrey/data/otp:otp_json_hw_cfg1", ] # The `MANUF_PERSONALIZED` OTP profile configures the SECRET1 and SECRET2 OTP # partitions. It also includes the `MANUF_INDIVIDUALIZED` profile. It represents # the OTP state of a device that has completed all provisioning steps. MANUF_PERSONALIZED = MANUF_INDIVIDUALIZED + [ - "//hw/ip/otp_ctrl/data:otp_json_secret1", - "//hw/ip/otp_ctrl/data:otp_json_fixed_secret2", + "//hw/top_earlgrey/data/otp:otp_json_secret1", + "//hw/top_earlgrey/data/otp:otp_json_fixed_secret2", ] # OTP *_SW_CFG and ROT_CREATOR_AUTH_* constants used to generate an FT # individualization binary. otp_image_consts( name = "otp_consts_c_file", - src = "//hw/ip/otp_ctrl/data:otp_json_baseline", + src = "//hw/top_earlgrey/data/otp:otp_json_baseline", + mmap_def = "//hw/top_earlgrey/data/otp:otp_ctrl_mmap.hjson", # Do not add additional overlays here. Update the `MANUF_SW_INITIALIZED` # OTP profile instead. overlays = MANUF_SW_INITIALIZED + OTP_SIGVERIFY_FAKE_KEYS, @@ -297,7 +298,7 @@ cc_library( # See sw/device/tests/doc/sival/devguide.md for more details. otp_image( name = "otp_img_test_unlocked0_manuf_empty", - src = "//hw/ip/otp_ctrl/data:otp_json_test_unlocked0", + src = "//hw/top_earlgrey/data/otp:otp_json_test_unlocked0", ) # `MANUF_INITIALIZED` configuration. This configuration will be generally used @@ -305,7 +306,7 @@ otp_image( # See sw/device/tests/doc/sival/devguide.md for more details. otp_image( name = "otp_img_test_locked0_manuf_initialized", - src = "//hw/ip/otp_ctrl/data:otp_json_test_locked0", + src = "//hw/top_earlgrey/data/otp:otp_json_test_locked0", overlays = MANUF_INITIALIZED, ) @@ -314,7 +315,7 @@ otp_image( [ otp_image( name = "otp_img_test_unlocked{}_manuf_initialized".format(i), - src = "//hw/ip/otp_ctrl/data:otp_json_test_unlocked{}".format(i), + src = "//hw/top_earlgrey/data/otp:otp_json_test_unlocked{}".format(i), overlays = MANUF_INITIALIZED, ) for i in range(1, 8) @@ -328,7 +329,7 @@ otp_image( [ otp_image( name = "otp_img_{}_manuf_individualized".format(lc_state), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = MANUF_INDIVIDUALIZED, ) for lc_state, _ in get_lc_items( @@ -351,7 +352,7 @@ otp_image( [ otp_image( name = "otp_img_{}_manuf_personalized".format(lc_state), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = MANUF_PERSONALIZED, ) for lc_state, _ in LC_MISSION_STATES @@ -359,13 +360,13 @@ otp_image( otp_image( name = "otp_img_dev_manuf_personalized_enable_rv_dm_late_debug_enable", - src = "//hw/ip/otp_ctrl/data:otp_json_dev", + src = "//hw/top_earlgrey/data/otp:otp_json_dev", overlays = MANUF_PERSONALIZED + [":otp_json_hw_cfg1_enable_rv_dm_late_debug"], ) # `MANUF_PERSONALIZED` configuration for RMA. Only available in secure environments. otp_image( name = "otp_img_rma_manuf_personalized", - src = "//hw/ip/otp_ctrl/data:otp_json_rma", + src = "//hw/top_earlgrey/data/otp:otp_json_rma", overlays = MANUF_PERSONALIZED, ) diff --git a/hw/top_earlgrey/data/top_earlgrey.hjson b/hw/top_earlgrey/data/top_earlgrey.hjson index 3adf456afa8f98..b7b85ee89586c5 100644 --- a/hw/top_earlgrey/data/top_earlgrey.hjson +++ b/hw/top_earlgrey/data/top_earlgrey.hjson @@ -401,6 +401,7 @@ core: {hart: "0x40130000"}, prim: {hart: "0x40138000"}, } + attr: "ipgen" }, { name: "lc_ctrl", type: "lc_ctrl", diff --git a/hw/top_earlgrey/dv/chip_sim_cfg.hjson b/hw/top_earlgrey/dv/chip_sim_cfg.hjson index ecee7eaa3a8b4a..d4076af0629207 100644 --- a/hw/top_earlgrey/dv/chip_sim_cfg.hjson +++ b/hw/top_earlgrey/dv/chip_sim_cfg.hjson @@ -230,7 +230,7 @@ "{tool}_dpi_build_opts"] // Setup for generating OTP images. - gen_otp_images_cfg_dir: "{proj_root}/hw/ip/otp_ctrl/data" + gen_otp_images_cfg_dir: "{proj_root}/hw/top_earlgrey/data/otp" gen_otp_images_cmd: "{proj_root}/util/design/gen-otp-img.py" gen_otp_images_cmd_opts: ["--quiet", "--img-seed {seed}", diff --git a/hw/top_earlgrey/dv/top_earlgrey_sim_cfgs.hjson b/hw/top_earlgrey/dv/top_earlgrey_sim_cfgs.hjson index 9a1d2da25d271b..5bcd0f5bdfd891 100644 --- a/hw/top_earlgrey/dv/top_earlgrey_sim_cfgs.hjson +++ b/hw/top_earlgrey/dv/top_earlgrey_sim_cfgs.hjson @@ -35,7 +35,6 @@ "{proj_root}/hw/ip/lc_ctrl/dv/lc_ctrl_volatile_unlock_disabled_sim_cfg.hjson", "{proj_root}/hw/ip/lc_ctrl/dv/lc_ctrl_volatile_unlock_enabled_sim_cfg.hjson", "{proj_root}/hw/ip/otbn/dv/uvm/otbn_sim_cfg.hjson", - "{proj_root}/hw/ip/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson", "{proj_root}/hw/ip/pattgen/dv/pattgen_sim_cfg.hjson", "{proj_root}/hw/ip/prim/dv/prim_alert/prim_alert_sim_cfg.hjson", "{proj_root}/hw/ip/prim/dv/prim_esc/prim_esc_sim_cfg.hjson", @@ -59,6 +58,7 @@ "{proj_root}/hw/top_earlgrey/ip_autogen/alert_handler/dv/alert_handler_sim_cfg.hjson", "{proj_root}/hw/top_earlgrey/ip_autogen/clkmgr/dv/clkmgr_sim_cfg.hjson", "{proj_root}/hw/top_earlgrey/ip_autogen/flash_ctrl/dv/flash_ctrl_sim_cfg.hjson", + "{proj_root}/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson", "{proj_root}/hw/top_earlgrey/ip_autogen/pwrmgr/dv/pwrmgr_sim_cfg.hjson", "{proj_root}/hw/top_earlgrey/ip_autogen/rstmgr/dv/rstmgr_cnsty_chk/rstmgr_cnsty_chk_sim_cfg.hjson", "{proj_root}/hw/top_earlgrey/ip_autogen/rstmgr/dv/rstmgr_sim_cfg.hjson", diff --git a/hw/top_earlgrey/dv/verilator/verilator_sim_cfg.hjson b/hw/top_earlgrey/dv/verilator/verilator_sim_cfg.hjson index d04e45b5836354..a392474de7b521 100644 --- a/hw/top_earlgrey/dv/verilator/verilator_sim_cfg.hjson +++ b/hw/top_earlgrey/dv/verilator/verilator_sim_cfg.hjson @@ -57,7 +57,7 @@ { name: sw_test_mode sw_images: ["//sw/device/lib/testing/test_rom:test_rom:0", - "//hw/ip/otp_ctrl/data:img_rma:3"] + "//hw/top_earlgrey/data/otp:img_rma:3"] run_opts: [ // The following shell snippet converts the SW images specification to what's // needed as a run time switch to Verilator. diff --git a/hw/top_earlgrey/formal/top_earlgrey_fpv_sec_cm_cfgs.hjson b/hw/top_earlgrey/formal/top_earlgrey_fpv_sec_cm_cfgs.hjson index e514f5546511bc..eff7e9d11d5893 100644 --- a/hw/top_earlgrey/formal/top_earlgrey_fpv_sec_cm_cfgs.hjson +++ b/hw/top_earlgrey/formal/top_earlgrey_fpv_sec_cm_cfgs.hjson @@ -133,9 +133,9 @@ { name: otp_ctrl_sec_cm dut: otp_ctrl - fusesoc_core: lowrisc:dv:otp_ctrl_sva + fusesoc_core: lowrisc:opentitan:top_earlgrey_otp_ctrl_sva import_cfgs: ["{proj_root}/hw/formal/tools/dvsim/common_fpv_cfg.hjson"] - rel_path: "hw/ip/otp_ctrl/{sub_flow}/{tool}" + rel_path: "hw/top_earlgrey/ip_autogen/otp_ctrl/{sub_flow}/{tool}" stopats: ["*u_state_regs.state_o"] task: "FpvSecCm" } diff --git a/hw/top_earlgrey/ip/BUILD b/hw/top_earlgrey/ip/BUILD index 52008956b9e489..73c6d79ad26043 100644 --- a/hw/top_earlgrey/ip/BUILD +++ b/hw/top_earlgrey/ip/BUILD @@ -15,6 +15,7 @@ filegroup( "//hw/top_earlgrey/ip_autogen/alert_handler:all_files", "//hw/top_earlgrey/ip_autogen/clkmgr:all_files", "//hw/top_earlgrey/ip_autogen/flash_ctrl:all_files", + "//hw/top_earlgrey/ip_autogen/otp_ctrl:all_files", "//hw/top_earlgrey/ip_autogen/pinmux:all_files", "//hw/top_earlgrey/ip_autogen/pwrmgr:all_files", "//hw/top_earlgrey/ip_autogen/rstmgr:all_files", diff --git a/hw/top_earlgrey/ip_autogen/flash_ctrl/doc/programmers_guide.md b/hw/top_earlgrey/ip_autogen/flash_ctrl/doc/programmers_guide.md index 8f57609cf38640..c47051fbe75a08 100644 --- a/hw/top_earlgrey/ip_autogen/flash_ctrl/doc/programmers_guide.md +++ b/hw/top_earlgrey/ip_autogen/flash_ctrl/doc/programmers_guide.md @@ -43,7 +43,7 @@ Once the address is discovered, further steps can be taken to triage the issue. ### Hardware Initiated Reads -If the root secrets have been provisioned in OTP and the life cycle state is in either DEV, PROD* or RMA, the special info pages holding the creator and owner seeds will be read out automatically by the flash controller and sent to the keymanager upon flash initialization (see [life cycle controller documentation](../../../../ip/lc_ctrl/doc/theory_of_operation.md#life-cycle-access-control-signals) and [OTP controller documentation](../../../../ip/otp_ctrl/doc/theory_of_operation.md#life-cycle-interfaces) for more details). +If the root secrets have been provisioned in OTP and the life cycle state is in either DEV, PROD* or RMA, the special info pages holding the creator and owner seeds will be read out automatically by the flash controller and sent to the keymanager upon flash initialization (see [life cycle controller documentation](../../../../ip/lc_ctrl/doc/theory_of_operation.md#life-cycle-access-control-signals) and [OTP controller documentation](../../otp_ctrl/doc/theory_of_operation.md#life-cycle-interfaces) for more details). Hence, it is important that these pages are initialized with valid data, since otherwise the hardware will likely encounter ECC errors during the automatic readout. Note that by default, hardware assumes that scrambling and ECC is enabled on these special info pages. diff --git a/hw/top_earlgrey/ip_autogen/flash_ctrl/doc/theory_of_operation.md b/hw/top_earlgrey/ip_autogen/flash_ctrl/doc/theory_of_operation.md index be6f32e03c0b56..b95837cf3f98e1 100644 --- a/hw/top_earlgrey/ip_autogen/flash_ctrl/doc/theory_of_operation.md +++ b/hw/top_earlgrey/ip_autogen/flash_ctrl/doc/theory_of_operation.md @@ -39,7 +39,7 @@ It is expected that after an RMA transition, the entire system will be rebooted. #### Initialization The flash protocol controller is initialized through [`INIT`](registers.md#init). -When initialization is invoked, the flash controller requests the address and data scrambling keys from an external entity, [otp_ctrl](../../../../ip/otp_ctrl/README.md#interface-to-flash-scrambler) in this case. +When initialization is invoked, the flash controller requests the address and data scrambling keys from an external entity, [otp_ctrl](../../otp_ctrl/README.md#interface-to-flash-scrambler) in this case. After the scrambling keys are requested, the flash protocol controller reads the root seeds out of the [secret partitions](#secret-information-partitions) and sends them to the key manager. Once the above steps are completed, the read buffers in the flash physical controller are enabled for operation. diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/BUILD b/hw/top_earlgrey/ip_autogen/otp_ctrl/BUILD new file mode 100644 index 00000000000000..dda56de039e27f --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/BUILD @@ -0,0 +1,10 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 + +package(default_visibility = ["//visibility:public"]) + +filegroup( + name = "all_files", + srcs = glob(["**"]), +) diff --git a/hw/ip/otp_ctrl/README.md b/hw/top_earlgrey/ip_autogen/otp_ctrl/README.md similarity index 87% rename from hw/ip/otp_ctrl/README.md rename to hw/top_earlgrey/ip_autogen/otp_ctrl/README.md index 343f2d3a2e4991..ee38ef5efb4a37 100644 --- a/hw/ip/otp_ctrl/README.md +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/README.md @@ -1,6 +1,6 @@ # OTP Controller Technical Specification -[`otp_ctrl`](https://reports.opentitan.org/hw/ip/otp_ctrl/dv/latest/report.html): +[`otp_ctrl`](https://reports.opentitan.org/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/latest/report.html): ![](https://dashboards.lowrisc.org/badges/dv/otp_ctrl/test.svg) ![](https://dashboards.lowrisc.org/badges/dv/otp_ctrl/passing.svg) ![](https://dashboards.lowrisc.org/badges/dv/otp_ctrl/functional.svg) @@ -9,7 +9,7 @@ # Overview This document specifies the functionality of the one time programmable (OTP) memory controller. -The OTP controller is a module that is a peripheral on the chip interconnect bus, and thus follows the [Comportability Specification](../../../doc/contributing/hw/comportability/README.md). +The OTP controller is a module that is a peripheral on the chip interconnect bus, and thus follows the [Comportability Specification](../../../../doc/contributing/hw/comportability/README.md). The OTP is a module that provides a device with one-time-programming functionality. The result of this programming is non-volatile, and unlike flash, cannot be reversed. @@ -17,7 +17,7 @@ The OTP functionality is constructed through an open-source OTP controller and a The OTP controller provides: - An open-source abstraction interface that software can use to interact with a proprietary OTP block underneath. -- An open-source abstraction interface that hardware components (for example [life cycle controller](../lc_ctrl/README.md) and [key manager](../keymgr/README.md)) can use to interact with a proprietary OTP block underneath. +- An open-source abstraction interface that hardware components (for example [life cycle controller](../../../ip/lc_ctrl/README.md) and [key manager](../../../ip/keymgr/README.md)) can use to interact with a proprietary OTP block underneath. - High level logical security protection, such as integrity checks and scrambling of sensitive content. - Software isolation for when OTP contents are readable and programmable. @@ -58,6 +58,6 @@ The proprietary IP on the other hand translates a common access interface to the This split implies that every proprietary OTP IP must implement a translation layer from a standardized OpenTitan interface to the module underneath. It also implies that no matter how the OTP storage or word size may change underneath, the open-source controller must present a consistent and coherent software and hardware interface. -This standardized interface is defined further below, and the wrapper leverages the same [technology primitive mechanism](../prim/README.md) that is employed in other parts of OpenTitan in order to wrap and abstract technology-specific macros (such as memories and clocking cells) that are potentially closed-source. +This standardized interface is defined further below, and the wrapper leverages the same [technology primitive mechanism](../../../ip/prim/README.md) that is employed in other parts of OpenTitan in order to wrap and abstract technology-specific macros (such as memories and clocking cells) that are potentially closed-source. In order to enable simulation and FPGA emulation of the OTP controller even without access to the proprietary OTP IP, a generalized and synthesizable model of the OTP IP is provided in the form of a [generic technology primitive](https://github.com/lowRISC/opentitan/blob/master/hw/ip/prim_generic/rtl/prim_generic_otp.sv). diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet b/hw/top_earlgrey/ip_autogen/otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet new file mode 100644 index 00000000000000..766a1a333d5840 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/data/dif_otp_ctrl.c.tpl.not_yet @@ -0,0 +1,684 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +<% +from topgen.lib import Name + +parts = otp_mmap["partitions"] +digest_parts = [part for part in parts if + part["hw_digest"] or part["sw_digest"]] +read_locked_csr_parts = [part for part in parts if part["read_lock"] == "CSR"] +secret_parts = [part for part in parts if part["secret"]] +%>\ +#include "sw/device/lib/dif/dif_otp_ctrl.h" + +#include + +#include "sw/device/lib/base/bitfield.h" +#include "sw/device/lib/base/macros.h" +#include "sw/device/lib/dif/dif_base.h" + +#include "otp_ctrl_regs.h" // Generated. + +/** + * Checks if integrity/consistency-check-related operations are locked. + * + * This is a convenience function to avoid superfluous error-checking in all the + * functions that can be locked out by this register. + * + * @param check_config True to check the config regwen. False to check the + * trigger regwen. + */ +static bool checks_are_locked(const dif_otp_ctrl_t *otp, bool check_config) { + ptrdiff_t reg_offset = check_config + ? OTP_CTRL_CHECK_REGWEN_REG_OFFSET + : OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET; + size_t regwen_bit = + check_config ? OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT + : OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT; + uint32_t locked = mmio_region_read32(otp->base_addr, reg_offset); + return !bitfield_bit32_read(locked, regwen_bit); +} + +dif_result_t dif_otp_ctrl_configure(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_config_t config) { + if (otp == NULL) { + return kDifBadArg; + } + if (checks_are_locked(otp, /*check_config=*/true)) { + return kDifLocked; + } + + mmio_region_write32(otp->base_addr, OTP_CTRL_CHECK_TIMEOUT_REG_OFFSET, + config.check_timeout); + mmio_region_write32(otp->base_addr, + OTP_CTRL_INTEGRITY_CHECK_PERIOD_REG_OFFSET, + config.integrity_period_mask); + mmio_region_write32(otp->base_addr, + OTP_CTRL_CONSISTENCY_CHECK_PERIOD_REG_OFFSET, + config.consistency_period_mask); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_check_integrity(const dif_otp_ctrl_t *otp) { + if (otp == NULL) { + return kDifBadArg; + } + if (checks_are_locked(otp, /*check_config=*/false)) { + return kDifLocked; + } + + uint32_t reg = + bitfield_bit32_write(0, OTP_CTRL_CHECK_TRIGGER_INTEGRITY_BIT, true); + mmio_region_write32(otp->base_addr, OTP_CTRL_CHECK_TRIGGER_REG_OFFSET, reg); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_check_consistency(const dif_otp_ctrl_t *otp) { + if (otp == NULL) { + return kDifBadArg; + } + if (checks_are_locked(otp, /*check_config=*/false)) { + return kDifLocked; + } + + uint32_t reg = + bitfield_bit32_write(0, OTP_CTRL_CHECK_TRIGGER_CONSISTENCY_BIT, true); + mmio_region_write32(otp->base_addr, OTP_CTRL_CHECK_TRIGGER_REG_OFFSET, reg); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_lock(const dif_otp_ctrl_t *otp) { + if (otp == NULL) { + return kDifBadArg; + } + + uint32_t reg = bitfield_bit32_write( + 0, OTP_CTRL_DIRECT_ACCESS_REGWEN_DIRECT_ACCESS_REGWEN_BIT, false); + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_REGWEN_REG_OFFSET, + reg); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_is_locked(const dif_otp_ctrl_t *otp, + bool *is_locked) { + if (otp == NULL || is_locked == NULL) { + return kDifBadArg; + } + + uint32_t reg = mmio_region_read32(otp->base_addr, + OTP_CTRL_DIRECT_ACCESS_REGWEN_REG_OFFSET); + *is_locked = !bitfield_bit32_read( + reg, OTP_CTRL_DIRECT_ACCESS_REGWEN_DIRECT_ACCESS_REGWEN_BIT); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_lock_config(const dif_otp_ctrl_t *otp) { + if (otp == NULL) { + return kDifBadArg; + } + + uint32_t reg = + bitfield_bit32_write(0, OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT, false); + mmio_region_write32(otp->base_addr, OTP_CTRL_CHECK_REGWEN_REG_OFFSET, reg); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_config_is_locked(const dif_otp_ctrl_t *otp, + bool *is_locked) { + if (otp == NULL || is_locked == NULL) { + return kDifBadArg; + } + + *is_locked = checks_are_locked(otp, /*check_config=*/true); + return kDifOk; +} + +dif_result_t dif_otp_ctrl_lock_check_trigger(const dif_otp_ctrl_t *otp) { + if (otp == NULL) { + return kDifBadArg; + } + + uint32_t reg = bitfield_bit32_write( + 0, OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT, false); + mmio_region_write32(otp->base_addr, OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET, + reg); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_check_trigger_is_locked(const dif_otp_ctrl_t *otp, + bool *is_locked) { + if (otp == NULL || is_locked == NULL) { + return kDifBadArg; + } + + *is_locked = checks_are_locked(otp, /*check_config=*/false); + return kDifOk; +} + +static bool sw_read_lock_reg_offset(dif_otp_ctrl_partition_t partition, + ptrdiff_t *reg_offset, + bitfield_bit32_index_t *index) { + switch (partition) { +% for part in read_locked_csr_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_define = part_name.as_c_define() + index_line = f"*index = OTP_CTRL_{part_name_define}_READ_LOCK_{part_name_define}_READ_LOCK_BIT;" +%>\ + case kDifOtpCtrlPartition${part_name.as_camel_case()}: + *reg_offset = OTP_CTRL_${part_name_define}_READ_LOCK_REG_OFFSET; + % if len(index_line) > 80 - 6: + *index = + OTP_CTRL_${part_name_define}_READ_LOCK_${part_name_define}_READ_LOCK_BIT; + % else: + ${index_line} + % endif + break; +% endfor + default: + return false; + } + return true; +} + +dif_result_t dif_otp_ctrl_lock_reading(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition) { + if (otp == NULL) { + return kDifBadArg; + } + + ptrdiff_t offset; + bitfield_bit32_index_t index; + if (!sw_read_lock_reg_offset(partition, &offset, &index)) { + return kDifBadArg; + } + + uint32_t reg = bitfield_bit32_write(0, index, false); + mmio_region_write32(otp->base_addr, offset, reg); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_reading_is_locked(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + bool *is_locked) { + if (otp == NULL || is_locked == NULL) { + return kDifBadArg; + } + + ptrdiff_t offset; + bitfield_bit32_index_t index; + if (!sw_read_lock_reg_offset(partition, &offset, &index)) { + return kDifBadArg; + } + + uint32_t reg = mmio_region_read32(otp->base_addr, offset); + *is_locked = !bitfield_bit32_read(reg, index); + return kDifOk; +} + +dif_result_t dif_otp_ctrl_get_status(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_status_t *status) { + if (otp == NULL || status == NULL) { + return kDifBadArg; + } + + static const bitfield_bit32_index_t kIndices[] = { +% for part in parts: +<% + part_name = Name.from_snake_case(part["name"]) + lhs = f'[kDifOtpCtrlStatusCode{part_name.as_camel_case()}Error]' + rhs = f'OTP_CTRL_STATUS_{part_name.as_c_define()}_ERROR_BIT' + line = f'{lhs} = {rhs},' +%>\ + % if len(line) > 80 - 6: + ${lhs} = + ${rhs}, + % else: + ${line} + % endif +% endfor + [kDifOtpCtrlStatusCodeDaiError] = OTP_CTRL_STATUS_DAI_ERROR_BIT, + [kDifOtpCtrlStatusCodeLciError] = OTP_CTRL_STATUS_LCI_ERROR_BIT, + [kDifOtpCtrlStatusCodeTimeoutError] = OTP_CTRL_STATUS_TIMEOUT_ERROR_BIT, + [kDifOtpCtrlStatusCodeLfsrError] = OTP_CTRL_STATUS_LFSR_FSM_ERROR_BIT, + [kDifOtpCtrlStatusCodeScramblingError] = + OTP_CTRL_STATUS_SCRAMBLING_FSM_ERROR_BIT, + [kDifOtpCtrlStatusCodeKdfError] = OTP_CTRL_STATUS_KEY_DERIV_FSM_ERROR_BIT, + [kDifOtpCtrlStatusCodeBusIntegError] = + OTP_CTRL_STATUS_BUS_INTEG_ERROR_BIT, + [kDifOtpCtrlStatusCodeDaiIdle] = OTP_CTRL_STATUS_DAI_IDLE_BIT, + [kDifOtpCtrlStatusCodeCheckPending] = OTP_CTRL_STATUS_CHECK_PENDING_BIT, + }; + + status->codes = 0; + uint32_t status_code = + mmio_region_read32(otp->base_addr, OTP_CTRL_STATUS_REG_OFFSET); + for (int i = 0; i < ARRAYSIZE(kIndices); ++i) { + // If the error is not present at all, we clear its cause bit if relevant, + // and bail immediately. + if (!bitfield_bit32_read(status_code, kIndices[i])) { + if (i <= kDifOtpCtrlStatusCodeHasCauseLast) { + status->causes[i] = kDifOtpCtrlErrorOk; + } + continue; + } + + status->codes = + bitfield_bit32_write(status->codes, (bitfield_bit32_index_t)i, true); + + if (i <= kDifOtpCtrlStatusCodeHasCauseLast) { + bitfield_field32_t field; + field = (bitfield_field32_t){ + .mask = OTP_CTRL_ERR_CODE_0_ERR_CODE_0_MASK, + .index = OTP_CTRL_ERR_CODE_0_ERR_CODE_0_OFFSET, + }; + + ptrdiff_t address = + OTP_CTRL_ERR_CODE_0_REG_OFFSET + i * (ptrdiff_t)sizeof(uint32_t); + uint32_t error_code = mmio_region_read32(otp->base_addr, address); + + dif_otp_ctrl_error_t err; + switch (bitfield_field32_read(error_code, field)) { + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_NO_ERROR: + err = kDifOtpCtrlErrorOk; + break; + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_ERROR: + err = kDifOtpCtrlErrorMacroUnspecified; + break; + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_ECC_CORR_ERROR: + err = kDifOtpCtrlErrorMacroRecoverableRead; + break; + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_ECC_UNCORR_ERROR: + err = kDifOtpCtrlErrorMacroUnrecoverableRead; + break; + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_WRITE_BLANK_ERROR: + err = kDifOtpCtrlErrorMacroBlankCheckFailed; + break; + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_ACCESS_ERROR: + err = kDifOtpCtrlErrorLockedAccess; + break; + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_CHECK_FAIL_ERROR: + err = kDifOtpCtrlErrorBackgroundCheckFailed; + break; + case OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_FSM_STATE_ERROR: + err = kDifOtpCtrlErrorFsmBadState; + break; + default: + return kDifError; + } + status->causes[i] = err; + } + } + + return kDifOk; +} + +typedef struct partition_info { + /** + * The absolute OTP address at which this partition starts. + */ + uint32_t start_addr; + /** + * The length of this partition, in bytes, including the digest. + * + * If the partition has a digest, it is expected to be at address + * `start_addr + len - sizeof(uint64_t)`. + */ + uint32_t len; + /** + * The alignment mask for this partition. + * + * A valid address for this partition must be such that + * `addr & align_mask == 0`. + */ + uint32_t align_mask; + + /** + * Whether this is a software-managed partition with a software-managed + * digest. + */ + bool is_software; + + /** + * Whether this partition has a digest field. + */ + bool has_digest; + + /** + * Whether this partition is the lifecycle partition. + */ + bool is_lifecycle; +} partition_info_t; + +// This is generates too many lines with different formatting variants, so +// We opt to just disable formatting. +// clang-format off +static const partition_info_t kPartitions[] = { +% for part in parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() + part_name_define = part_name.as_c_define() + has_digest = part["hw_digest"] or part["sw_digest"] + is_lifecycle = part["variant"] == "LifeCycle" + is_software = part["variant"] == "Unbuffered" +%>\ + [kDifOtpCtrlPartition${part_name_camel}] = { + .start_addr = OTP_CTRL_PARAM_${part_name_define}_OFFSET, + .len = OTP_CTRL_PARAM_${part_name_define}_SIZE, + .align_mask = ${"0x7" if part in secret_parts else "0x3"}, + .is_software = ${"true" if is_software else "false"}, + .has_digest = ${"true" if has_digest else "false"}, + .is_lifecycle = ${"true" if is_lifecycle else "false"}}, +% endfor +}; +// clang-format on + +dif_result_t dif_otp_ctrl_relative_address(dif_otp_ctrl_partition_t partition, + uint32_t abs_address, + uint32_t *relative_address) { + *relative_address = 0; + + if (partition >= ARRAYSIZE(kPartitions)) { + return kDifBadArg; + } + + if ((abs_address & kPartitions[partition].align_mask) != 0) { + return kDifUnaligned; + } + + if (abs_address < kPartitions[partition].start_addr) { + return kDifOutOfRange; + } + + *relative_address = abs_address - kPartitions[partition].start_addr; + if (*relative_address >= kPartitions[partition].len) { + *relative_address = 0; + return kDifOutOfRange; + } + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_read_start(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address) { + if (otp == NULL || partition >= ARRAYSIZE(kPartitions)) { + return kDifBadArg; + } + + if ((address & kPartitions[partition].align_mask) != 0) { + return kDifUnaligned; + } + + if (address >= kPartitions[partition].len) { + return kDifOutOfRange; + } + + address += kPartitions[partition].start_addr; + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + address); + + uint32_t cmd = + bitfield_bit32_write(0, OTP_CTRL_DIRECT_ACCESS_CMD_RD_BIT, true); + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + cmd); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_read32_end(const dif_otp_ctrl_t *otp, + uint32_t *value) { + if (otp == NULL || value == NULL) { + return kDifBadArg; + } + + *value = mmio_region_read32(otp->base_addr, + OTP_CTRL_DIRECT_ACCESS_RDATA_0_REG_OFFSET); + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_read64_end(const dif_otp_ctrl_t *otp, + uint64_t *value) { + if (otp == NULL || value == NULL) { + return kDifBadArg; + } + + *value = mmio_region_read32(otp->base_addr, + OTP_CTRL_DIRECT_ACCESS_RDATA_1_REG_OFFSET); + *value <<= 32; + *value |= mmio_region_read32(otp->base_addr, + OTP_CTRL_DIRECT_ACCESS_RDATA_0_REG_OFFSET); + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_program32(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address, uint32_t value) { + if (otp == NULL || partition >= ARRAYSIZE(kPartitions)) { + return kDifBadArg; + } + + // Ensure that we are writing to a 32-bit-access partition by checking that + // the alignment mask is 0b11. + // + // Note furthermore that the LC partition is *not* writeable, so we eject + // here. + if (kPartitions[partition].align_mask != 0x3 || + kPartitions[partition].is_lifecycle) { + return kDifError; + } + + if ((address & kPartitions[partition].align_mask) != 0) { + return kDifUnaligned; + } + + // NOTE: The bounds check is tightened here, since we disallow writing the + // digest directly. If the partition does not have a digest, no tightening is + // needed. + size_t digest_size = kPartitions[partition].has_digest * sizeof(uint64_t); + if (address >= kPartitions[partition].len - digest_size) { + return kDifOutOfRange; + } + + address += kPartitions[partition].start_addr; + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + address); + + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET, + value); + + uint32_t cmd = + bitfield_bit32_write(0, OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT, true); + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + cmd); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_program64(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address, uint64_t value) { + if (otp == NULL || partition >= ARRAYSIZE(kPartitions)) { + return kDifBadArg; + } + + // Ensure that we are writing to a 64-bit-access partition by checking that + // the alignment mask is 0b111. + if (kPartitions[partition].align_mask != 0x7) { + return kDifError; + } + + if ((address & kPartitions[partition].align_mask) != 0) { + return kDifUnaligned; + } + + // NOTE: The bounds check is tightened here, since we disallow writing the + // digest directly. + size_t digest_size = sizeof(uint64_t); + if (address >= kPartitions[partition].len - digest_size) { + return kDifOutOfRange; + } + + address += kPartitions[partition].start_addr; + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + address); + + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET, + value & UINT32_MAX); + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_WDATA_1_REG_OFFSET, + value >> 32); + + uint32_t cmd = + bitfield_bit32_write(0, OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT, true); + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + cmd); + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_dai_digest(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint64_t digest) { + if (otp == NULL || partition >= ARRAYSIZE(kPartitions)) { + return kDifBadArg; + } + + // Not all partitions have a digest. + if (!kPartitions[partition].has_digest) { + return kDifError; + } + + // For software partitions, the digest must be nonzero; for all other + // partitions it must be zero. + bool is_sw = kPartitions[partition].is_software; + if (is_sw == (digest == 0)) { + return kDifBadArg; + } + + uint32_t address = kPartitions[partition].start_addr; + if (is_sw) { + address += kPartitions[partition].len - sizeof(digest); + } + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + address); + + if (digest != 0) { + mmio_region_write32(otp->base_addr, + OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET, + digest & 0xffffffff); + mmio_region_write32(otp->base_addr, + OTP_CTRL_DIRECT_ACCESS_WDATA_1_REG_OFFSET, + digest >> 32); + } + + bitfield_bit32_index_t cmd_bit = is_sw + ? OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT + : OTP_CTRL_DIRECT_ACCESS_CMD_DIGEST_BIT; + uint32_t cmd = bitfield_bit32_write(0, cmd_bit, true); + mmio_region_write32(otp->base_addr, OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + cmd); + + return kDifOk; +} + +static bool get_digest_regs(dif_otp_ctrl_partition_t partition, ptrdiff_t *reg0, + ptrdiff_t *reg1) { + switch (partition) { +% for part in digest_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_define = part_name.as_c_define() +%>\ + case kDifOtpCtrlPartition${part_name.as_camel_case()}: + *reg0 = OTP_CTRL_${part_name_define}_DIGEST_0_REG_OFFSET; + *reg1 = OTP_CTRL_${part_name_define}_DIGEST_1_REG_OFFSET; + break; +% endfor + default: + return false; + } + + return true; +} + +dif_result_t dif_otp_ctrl_is_digest_computed(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + bool *is_computed) { + if (otp == NULL || is_computed == NULL) { + return kDifBadArg; + } + + ptrdiff_t reg0, reg1; + if (!get_digest_regs(partition, ®0, ®1)) { + return kDifBadArg; + } + + uint64_t value = mmio_region_read32(otp->base_addr, reg1); + value <<= 32; + value |= mmio_region_read32(otp->base_addr, reg0); + + *is_computed = value != 0; + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_get_digest(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint64_t *digest) { + if (otp == NULL || digest == NULL) { + return kDifBadArg; + } + + ptrdiff_t reg0, reg1; + if (!get_digest_regs(partition, ®0, ®1)) { + return kDifBadArg; + } + + uint64_t value = mmio_region_read32(otp->base_addr, reg1); + value <<= 32; + value |= mmio_region_read32(otp->base_addr, reg0); + + if (value == 0) { + return kDifError; + } + *digest = value; + + return kDifOk; +} + +dif_result_t dif_otp_ctrl_read_blocking(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address, uint32_t *buf, + size_t len) { + if (otp == NULL || partition >= ARRAYSIZE(kPartitions) || buf == NULL) { + return kDifBadArg; + } + + if (!kPartitions[partition].is_software) { + return kDifError; + } + + if ((address & kPartitions[partition].align_mask) != 0) { + return kDifUnaligned; + } + + if (address + len >= kPartitions[partition].len) { + return kDifOutOfRange; + } + + uint32_t reg_offset = OTP_CTRL_SW_CFG_WINDOW_REG_OFFSET + + kPartitions[partition].start_addr + address; + mmio_region_memcpy_from_mmio32(otp->base_addr, reg_offset, buf, + len * sizeof(uint32_t)); + return kDifOk; +} diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet b/hw/top_earlgrey/ip_autogen/otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet new file mode 100644 index 00000000000000..fbab47b27d488d --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/data/dif_otp_ctrl.h.tpl.not_yet @@ -0,0 +1,614 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +<% +from topgen.lib import Name + +parts = otp_mmap["partitions"] +digest_parts = [part for part in parts if + part["hw_digest"] == "true" or part["sw_digest"] == "true"] +read_locked_csr_parts = [part for part in parts if part["read_lock"] == "CSR"] +secret_parts = [part for part in parts if part["secret"] == "true"] +%>\ +#ifndef OPENTITAN_SW_DEVICE_LIB_DIF_DIF_OTP_CTRL_H_ +#define OPENTITAN_SW_DEVICE_LIB_DIF_DIF_OTP_CTRL_H_ + +/** + * @file + * @brief + * OTP Controller Device Interface Functions + */ + +#include + +#include "sw/device/lib/base/macros.h" +#include "sw/device/lib/base/mmio.h" +#include "sw/device/lib/dif/dif_base.h" + +#include "sw/device/lib/dif/autogen/dif_otp_ctrl_autogen.h" + +// Header Extern Guard (so header can be used from C and C++) +#ifdef __cplusplus +extern "C" { +#endif // __cplusplus + +/** + * A partition within OTP memory. + */ +typedef enum dif_otp_ctrl_partition { +% for part in parts: +<% + part_name = Name.from_snake_case(part["name"]) + short_desc = part["desc"].split(".")[0].strip().replace("\n", " ") + long_desc_lines = part["desc"].split(".", 1)[1].strip().splitlines() + long_desc = "\n".join([" *" + (" " if line else "") + line for + line in long_desc_lines]) +%>\ + /** + * ${short_desc}. + * + % if long_desc: +${long_desc} + %endif + */ + kDifOtpCtrlPartition${part_name.as_camel_case()}, +% endfor +} dif_otp_ctrl_partition_t; + +/** + * Runtime configuration for OTP. + * + * This struct describes runtime information for one-time configuration of the + * hardware. + */ +typedef struct dif_otp_ctrl_config { + /** + * The timeout for an integrity or consistency check to succeed, in cycles. + * + * 100'000 is recommended as a minimum safe value. + */ + uint32_t check_timeout; + /** + * A mask for the pseudo-random integrity check period. + * + * The value of this mask limits the period of the integrity check; when the + * pseudo-random period is computed, this mask is applied to limit it. For + * example, a value of 0x3'ffff would correspond to a maximum period of about + * 2.8s at 24MHz. + * + * A value of zero disables the check. + */ + uint32_t integrity_period_mask; + /** + * A mask for the pseudo-random consistency check period. + * + * The value of this mask limits the period of the consistency check; when the + * pseudo-random period is computed, this mask is applied to limit it. For + * example, a value of 0x3ff'ffff would correspond to a maximum period of + * about 716s at 24MHz. + * + * A value of zero disables the check. + */ + uint32_t consistency_period_mask; +} dif_otp_ctrl_config_t; + +/** + * A hardware-level status code. + */ +typedef enum dif_otp_ctrl_status_code { + // NOTE: This enum's API *requires* that all "error"-like codes (that is, + // those which have associated cause registers) be a prefix of the enum + // values. + // + // Note furthermore that these enum variants are intended as bit indices, so + // their values should not be randomized. +% for part in parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() +%>\ + /** + * Indicates an error occurred in the `${part_name_camel}` partition. + */ + kDifOtpCtrlStatusCode${part_name_camel}Error${" = 0" if loop.first else ""}, +% endfor + /** + * Indicates an error occurred in the direct access interface. + */ + kDifOtpCtrlStatusCodeDaiError, + /** + * Indicates an error occurred in the lifecycle interface. + */ + kDifOtpCtrlStatusCodeLciError, + /** + * This is not a status code; rather, it represents the last error code which + * has a corresponding "cause" register. + * + * See `dif_otp_ctrl_status_t` for information on how to use this. + */ + kDifOtpCtrlStatusCodeHasCauseLast = kDifOtpCtrlStatusCodeLciError, + /** + * Indicates that an integrity or consistency check has timed out. + * + * This error is unrecoverable. + */ + kDifOtpCtrlStatusCodeTimeoutError, + /** + * Indicates that the LFSR that generates pseudo-random integrity and + * consistency checks is in a bad state. + * + * This error is unrecoverable. + */ + kDifOtpCtrlStatusCodeLfsrError, + /** + * Indicates that the scrambling hardware is in a bad state. + * + * This error is unrecoverable. + */ + kDifOtpCtrlStatusCodeScramblingError, + /** + * Indicates that the key derivation hardware is in a bad state. + * + * This error is unrecoverable. + */ + kDifOtpCtrlStatusCodeKdfError, + /** + * Indicates a bus integrity error. + * + * This error will raise an alert. + */ + kDifOtpCtrlStatusCodeBusIntegError, + /** + * Indicates that the direct access interface is idle. + */ + kDifOtpCtrlStatusCodeDaiIdle, + /** + * Indicates that an integrity or consistency check is currently pending. + */ + kDifOtpCtrlStatusCodeCheckPending, +} dif_otp_ctrl_status_code_t; + +/** + * A hardware-level error code, associated with a particular error defined in + * `dif_otp_ctrl_status_t`. + */ +typedef enum dif_otp_ctrl_error { + /** + * Indicates no error. + */ + kDifOtpCtrlErrorOk, + /** + * Indicates that an OTP macro command was invalid or did not + * complete successfully. + * + * This error indicates non-recoverable hardware malfunction. + */ + kDifOtpCtrlErrorMacroUnspecified, + /** + * Indicates a recoverable error during a read operation. + * + * A followup read should work as expected. + */ + kDifOtpCtrlErrorMacroRecoverableRead, + /** + * Indicates an unrecoverable error during a read operation. + * + * This error indicates non-recoverable hardware malfunction. + */ + kDifOtpCtrlErrorMacroUnrecoverableRead, + /** + * Indicates that the blank write check failed during a write operation. + */ + kDifOtpCtrlErrorMacroBlankCheckFailed, + /** + * Indicates a locked memory region was accessed. + */ + kDifOtpCtrlErrorLockedAccess, + /** + * Indicates a parity, integrity or consistency check failed in the buffer + * registers. + * + * This error indicates non-recoverable hardware malfunction. + */ + kDifOtpCtrlErrorBackgroundCheckFailed, + /** + * Indicates that the FSM of the controller is in a bad state or that the + * controller's FSM has been moved into its terminal state due to escalation + * via the alert subsystem. + * + * This error indicates that the device has been glitched by an attacker. + */ + kDifOtpCtrlErrorFsmBadState, +} dif_otp_ctrl_error_t; + +/** + * The overall status of the OTP controller. + * + * See `dif_otp_ctrl_get_status()`. + */ +typedef struct dif_otp_ctrl_status { + /** + * Currently active statuses, given as a bit vector. To check whether a + * particular status code was returned, write + * + * bool has_code = (status.codes >> kMyStatusCode) & 1; + * + * Note that it is possible to quickly check that the controller is idle and + * error-free by writing + * + * bool is_ok = status.codes == (1 << kDifOtpStatusCodeDaiIdle); + */ + uint32_t codes; + /** + * A list of root causes for each error status code. + * + * If the error status code `error` is present in `codes`, and + * `error <= kDifOtpCtrlStatusCodeHasCauseLast`, then `causes[error]` + * will contain its root cause. + */ + dif_otp_ctrl_error_t causes[kDifOtpCtrlStatusCodeHasCauseLast + 1]; +} dif_otp_ctrl_status_t; + +/** + * Configures OTP with runtime information. + * + * This function should need to be called at most once for the lifetime of + * `otp`. + * + * @param otp An OTP handle. + * @param config Runtime configuration parameters. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_configure(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_config_t config); + +/** + * Runs an integrity check on the OTP hardware. + * + * This function can be used to trigger an integrity check independent of the + * pseudo-random hardware-generated checks. + * + * @param otp An OTP handle. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_check_integrity(const dif_otp_ctrl_t *otp); + +/** + * Runs a consistency check on the OTP hardware. + * + * This function can be used to trigger a consistency check independent of the + * pseudo-random hardware-generated checks. + * + * @param otp An OTP handle. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_check_consistency(const dif_otp_ctrl_t *otp); + +/** + * Locks out access to the direct access interface registers. + * + * This function is idempotent: calling it while functionality is locked will + * have no effect and return `kDifOk`. + * + * @param otp An OTP handle. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_lock(const dif_otp_ctrl_t *otp); + +/** + * Checks whether access to the direct access interface is locked. + * + * Note that besides locking the DAI out until the next reset using the + * dif_otp_ctrl_dai_lock function, the DAI is also temporarily locked by the + * HW itself when it is busy processing a DAI command. In such a case, the + * kDifOtpCtrlStatusCodeDaiIdle status bit will be set to 0 as well. + * + * @param otp An OTP handle. + * @param[out] is_locked Out-param for the locked state. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_is_locked(const dif_otp_ctrl_t *otp, + bool *is_locked); + +/** + * Locks out `dif_otp_ctrl_configure()` function. + * + * This function is idempotent: calling it while functionality is locked will + * have no effect and return `kDifOk`. + * + * @param otp An OTP handle. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_lock_config(const dif_otp_ctrl_t *otp); + +/** + * Checks whether `dif_otp_ctrl_configure()` function is locked-out. + * + * @param otp An OTP handle. + * @param[out] is_locked Out-param for the locked state. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_config_is_locked(const dif_otp_ctrl_t *otp, + bool *is_locked); + +/** + * Locks out `dif_otp_ctrl_check_*()` functions. + * + * This function is idempotent: calling it while functionality is locked will + * have no effect and return `kDifOk`. + * + * @param otp An OTP handle. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_lock_check_trigger(const dif_otp_ctrl_t *otp); + +/** + * Checks whether the `dif_otp_ctrl_check_*()` functions are locked-out. + * + * @param otp An OTP handle. + * @param[out] is_locked Out-param for the locked state. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_check_trigger_is_locked(const dif_otp_ctrl_t *otp, + bool *is_locked); + +/** + * Locks out reads to a SW partition. + * + * This function should only be called on SW partitions; doing otherwise will + * return an error. + * + * Note that this is distinct from the write-locking performed by calling + * `dif_otp_ctrl_dai_digest()`. In particular, the effects of this function will + * not persist past a system reset. + * + * This function is idempotent: calling it while functionality is locked will + * have no effect and return `kDifOk`. + * + * @param otp An OTP handle. + * @param partition The SW partition to lock. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_lock_reading(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition); + +/** + * Checks whether reads to a SW partition are locked out. + * + * This function should only be called on SW partitions; doing otherwise will + * return an error. + * + * @param otp An OTP handle. + * @param partition the SW partition to check for locking. + * @param[out] is_locked Out-param for the locked state. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_reading_is_locked(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + bool *is_locked); + +/** + * Gets the current status of the OTP controller. + * + * @param otp An OTP handle. + * @param[out] status Out-param for the controller's status. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_get_status(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_status_t *status); + +/** + * Calculates a `relative_address` with respect to a `partition` start + * address. + * + * @param partition The partition to use to calculate the reference start + * address. + * @param abs_address Input address relative to the OTP memory start address. + * @param[out] relative_address The result relative address with respect to the + * `partition` start address. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_relative_address(dif_otp_ctrl_partition_t partition, + uint32_t abs_address, + uint32_t *relative_address); + +/** + * Schedules a read on the Direct Access Interface. + * + * Reads are performed relative to a partition; `address` should be given + * relative to the start of `partition`. An error is returned for out-of-bounds + * access. + * + * Furthermore, `address` must be well-aligned: it must be four-byte aligned for + * normal partitions and eight-byte-aligned for secret partitions. An error is + * returned for unaligned access. + * + * @param otp An OTP handle. + * @param partition The partition to read from. + * @param address A partition-relative address to read from. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_read_start(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address); + +/** + * Gets the result of a completed 32-bit read operation on the Direct Access + * Interface. + * + * Whether this function or its 64-bit variant should be called is dependent on + * the most recent partition read from. + * + * @param otp An OTP handle. + * @param[out] value Out-param for the read value. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_read32_end(const dif_otp_ctrl_t *otp, + uint32_t *value); + +/** + * Gets the result of a completed 64-bit read operation on the Direct Access + * Interface. + * + * Whether this function or its 32-bit variant should be called is dependent on + * the most recent partition read from. + * + * @param otp An OTP handle. + * @param[out] value Out-param for the read value. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_read64_end(const dif_otp_ctrl_t *otp, + uint64_t *value); + +/** + * Schedules a 32-bit write on the Direct Access Interface. + * + * Writes are performed relative to a partition; `address` should be given + * relative to the start of `partition`. An error is returned for out-of-bounds + * access. + * + * Furthermore, `address` must be four-byte-aligned, and `partition` must not be + * a secret partition. An error is returned if neither condition is met. + * + * Note that this function cannot be used to program the digest at the end of a + * `SW` partition; `dif_otp_ctrl_dai_digest()` must be used instead. + * + * @param otp An OTP handle. + * @param partition The partition to program. + * @param address A partition-relative address to program. + * @param value The value to program into the OTP. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_program32(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address, uint32_t value); + +/** + * Schedules a 64-bit write on the Direct Access Interface. + * + * Writes are performed relative to a partition; `address` should be given + * relative to the start of `partition`. An error is returned for out-of-bounds + * access. + * + * Furthermore, `address` must be eight-byte-aligned, and `partition` must be + * a secret partition. An error is returned if neither condition is met. + * + * @param otp An OTP handle. + * @param partition The partition to program. + * @param address A partition-relative address to program. + * @param value The value to program into the OTP. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_program64(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address, uint64_t value); + +/** + * Schedules a hardware digest operation on the Direct Access Interface. + * + * **This operation will also lock writes for the given partition.** + * + * If `partition` is a SW partition, `digest` must be non-zero; if it is a + * partition with a hardware-managed digest, `digest` *must* be zero (since the + * digest will be generated by the hardware). An error is returned if either + * precondition is not met. + * + * This function does not work with the lifecycle state partition, and will + * return an error in that case. + * + * @param otp An OTP handle. + * @param partition The partition to digest and lock. + * @param digest The digest to program (for SW partitions). + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_dai_digest(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint64_t digest); + +/** + * Checks if the digest value for the given partition has been computed. Once a + * digest has been computed for a partition, the partition is write-locked + * (additionally, read-locked if the partition is secret). + * + * The lifecycle partition does not have a digest, and checking if this region + * has a computed digest will return an error. + * + * @param otp An OTP handle. + * @param partition The partition to check the digest of. + * @param[out] is_computed Indicates if the digest has been computed. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_is_digest_computed(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + bool *is_computed); + +/** + * Gets the buffered digest value for the given partition. + * + * Note that this value is only updated when the device is reset; if the digest + * has not been computed yet, or has been computed but not since device reset, + * this function will return an error. + * + * The lifecycle partition does not have a digest and will result in an error + * being returned. + * + * @param otp An OTP handle. + * @param partition The partition to get a digest for. + * @param[out] digest Out-param for the digest. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_get_digest(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint64_t *digest); + +/** + * Performs a memory-mapped read of the given partition, if it supports them. + * + * In particular, this function will read `len` words, starting at `address`, + * relative to the start of `partition`. + * + * The same caveats for `dif_otp_ctrl_dai_read_start()` apply to `address`; in + * addition, `address + len` must also be in-range and must not overflow. + * + * This function will block until the read completes, unlike Direct Access + * Interface functions. + * + * @param otp An OTP handle. + * @param partition The partition to read from. + * @param address A partition-relative address to read from. + * @param[out] buf A buffer of words to write read values to. + * @param len The number of words to read. + * @return The result of the operation. + */ +OT_WARN_UNUSED_RESULT +dif_result_t dif_otp_ctrl_read_blocking(const dif_otp_ctrl_t *otp, + dif_otp_ctrl_partition_t partition, + uint32_t address, uint32_t *buf, + size_t len); + +#ifdef __cplusplus +} // extern "C" +#endif // __cplusplus + +#endif // OPENTITAN_SW_DEVICE_LIB_DIF_DIF_OTP_CTRL_H_ diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl.not_yet b/hw/top_earlgrey/ip_autogen/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl.not_yet new file mode 100644 index 00000000000000..2063fd5971ebc5 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/data/dif_otp_ctrl_unittest.cc.tpl.not_yet @@ -0,0 +1,789 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +<% +from topgen.lib import Name + +parts = otp_mmap["partitions"] +digest_parts = [part for part in parts if + part["hw_digest"] or part["sw_digest"]] +hw_digest_parts = [part for part in parts if part["hw_digest"]] +sw_digest_parts = [part for part in parts if part["sw_digest"]] +read_locked_csr_parts = [part for part in parts if part["read_lock"] == "CSR"] +secret_parts = [part for part in parts if part["secret"]] +%>\ +#include "sw/device/lib/dif/dif_otp_ctrl.h" + +#include +#include +#include + +#include "gtest/gtest.h" +#include "sw/device/lib/base/mmio.h" +#include "sw/device/lib/base/mock_mmio.h" +#include "sw/device/lib/dif/dif_test_base.h" + +#include "otp_ctrl_regs.h" // Generated. + +namespace dif_otp_ctrl_unittest { +namespace { +using ::mock_mmio::LeInt; +using ::mock_mmio::MmioTest; +using ::mock_mmio::MockDevice; +using ::testing::Each; +using ::testing::ElementsAre; + +class OtpTest : public testing::Test, public MmioTest { + protected: + dif_otp_ctrl_t otp_ = {.base_addr = dev().region()}; +}; + +class DaiRegwenTest : public OtpTest {}; + +TEST_F(DaiRegwenTest, LockDai) { + EXPECT_WRITE32( + OTP_CTRL_DIRECT_ACCESS_REGWEN_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_REGWEN_DIRECT_ACCESS_REGWEN_BIT, false}}); + EXPECT_DIF_OK(dif_otp_ctrl_dai_lock(&otp_)); +} + +TEST_F(DaiRegwenTest, IsDaiLocked) { + bool flag; + + EXPECT_READ32( + OTP_CTRL_DIRECT_ACCESS_REGWEN_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_REGWEN_DIRECT_ACCESS_REGWEN_BIT, true}}); + EXPECT_DIF_OK(dif_otp_ctrl_dai_is_locked(&otp_, &flag)); + EXPECT_FALSE(flag); + + EXPECT_READ32( + OTP_CTRL_DIRECT_ACCESS_REGWEN_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_REGWEN_DIRECT_ACCESS_REGWEN_BIT, false}}); + EXPECT_DIF_OK(dif_otp_ctrl_dai_is_locked(&otp_, &flag)); + EXPECT_TRUE(flag); +} + +TEST_F(DaiRegwenTest, NullArgs) { + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_lock(nullptr)); + + bool flag; + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_is_locked(nullptr, &flag)); + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_is_locked(&otp_, nullptr)); +} + +class ConfigTest : public OtpTest {}; + +TEST_F(ConfigTest, Basic) { + dif_otp_ctrl_config_t config = { + .check_timeout = 100'000, + .integrity_period_mask = 0x3'ffff, + .consistency_period_mask = 0x3ff'ffff, + }; + + EXPECT_READ32(OTP_CTRL_CHECK_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT, true}}); + + EXPECT_WRITE32(OTP_CTRL_CHECK_TIMEOUT_REG_OFFSET, config.check_timeout); + EXPECT_WRITE32(OTP_CTRL_INTEGRITY_CHECK_PERIOD_REG_OFFSET, + config.integrity_period_mask); + EXPECT_WRITE32(OTP_CTRL_CONSISTENCY_CHECK_PERIOD_REG_OFFSET, + config.consistency_period_mask); + + EXPECT_DIF_OK(dif_otp_ctrl_configure(&otp_, config)); +} + +TEST_F(ConfigTest, Locked) { + EXPECT_READ32(OTP_CTRL_CHECK_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT, false}}); + + EXPECT_EQ(dif_otp_ctrl_configure(&otp_, {}), kDifLocked); +} + +TEST_F(ConfigTest, IsConfigLocked) { + bool flag; + + EXPECT_READ32(OTP_CTRL_CHECK_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT, true}}); + EXPECT_DIF_OK(dif_otp_ctrl_config_is_locked(&otp_, &flag)); + EXPECT_FALSE(flag); + + EXPECT_READ32(OTP_CTRL_CHECK_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT, false}}); + EXPECT_DIF_OK(dif_otp_ctrl_config_is_locked(&otp_, &flag)); + EXPECT_TRUE(flag); +} + +TEST_F(ConfigTest, LockConfig) { + EXPECT_WRITE32(OTP_CTRL_CHECK_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_REGWEN_CHECK_REGWEN_BIT, false}}); + EXPECT_DIF_OK(dif_otp_ctrl_lock_config(&otp_)); +} + +TEST_F(ConfigTest, NullArgs) { + EXPECT_DIF_BADARG(dif_otp_ctrl_configure(nullptr, {})); + + bool flag; + EXPECT_DIF_BADARG(dif_otp_ctrl_config_is_locked(nullptr, &flag)); + EXPECT_DIF_BADARG(dif_otp_ctrl_config_is_locked(&otp_, nullptr)); + + EXPECT_DIF_BADARG(dif_otp_ctrl_lock_config(nullptr)); +} + +class CheckTest : public OtpTest {}; + +TEST_F(CheckTest, Integrity) { + EXPECT_READ32( + OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT, true}}); + EXPECT_WRITE32(OTP_CTRL_CHECK_TRIGGER_REG_OFFSET, + {{OTP_CTRL_CHECK_TRIGGER_INTEGRITY_BIT, true}}); + + EXPECT_DIF_OK(dif_otp_ctrl_check_integrity(&otp_)); +} + +TEST_F(CheckTest, Consistency) { + EXPECT_READ32( + OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT, true}}); + EXPECT_WRITE32(OTP_CTRL_CHECK_TRIGGER_REG_OFFSET, + {{OTP_CTRL_CHECK_TRIGGER_CONSISTENCY_BIT, true}}); + + EXPECT_DIF_OK(dif_otp_ctrl_check_consistency(&otp_)); +} + +TEST_F(CheckTest, LockTrigger) { + EXPECT_WRITE32( + OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT, false}}); + EXPECT_DIF_OK(dif_otp_ctrl_lock_check_trigger(&otp_)); +} + +TEST_F(CheckTest, Locked) { + EXPECT_READ32( + OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT, false}}); + EXPECT_EQ(dif_otp_ctrl_check_integrity(&otp_), kDifLocked); + + EXPECT_READ32( + OTP_CTRL_CHECK_TRIGGER_REGWEN_REG_OFFSET, + {{OTP_CTRL_CHECK_TRIGGER_REGWEN_CHECK_TRIGGER_REGWEN_BIT, false}}); + EXPECT_EQ(dif_otp_ctrl_check_consistency(&otp_), kDifLocked); +} + +TEST_F(CheckTest, NullArgs) { + EXPECT_DIF_BADARG(dif_otp_ctrl_check_integrity(nullptr)); + EXPECT_DIF_BADARG(dif_otp_ctrl_check_consistency(nullptr)); +} + +class ReadLockTest : public OtpTest {}; + +// Too many formatting variants in template code, so disabling clang-format. +// clang-format off +TEST_F(ReadLockTest, IsLocked) { + bool flag; + +% for part in read_locked_csr_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() + part_name_define = part_name.as_c_define() +%>\ + EXPECT_READ32( + OTP_CTRL_${part_name_define}_READ_LOCK_REG_OFFSET, + {{OTP_CTRL_${part_name_define}_READ_LOCK_${part_name_define}_READ_LOCK_BIT, + true}}); + EXPECT_DIF_OK(dif_otp_ctrl_reading_is_locked( + &otp_, kDifOtpCtrlPartition${part_name_camel}, &flag)); + EXPECT_FALSE(flag); + + EXPECT_READ32( + OTP_CTRL_${part_name_define}_READ_LOCK_REG_OFFSET, + {{OTP_CTRL_${part_name_define}_READ_LOCK_${part_name_define}_READ_LOCK_BIT, + false}}); + EXPECT_DIF_OK(dif_otp_ctrl_reading_is_locked( + &otp_, kDifOtpCtrlPartition${part_name_camel}, &flag)); + EXPECT_TRUE(flag); + % if not loop.last: + + %endif +% endfor +} + +TEST_F(ReadLockTest, Lock) { +% for part in read_locked_csr_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() + part_name_define = part_name.as_c_define() +%>\ + EXPECT_WRITE32( + OTP_CTRL_${part_name_define}_READ_LOCK_REG_OFFSET, + {{OTP_CTRL_${part_name_define}_READ_LOCK_${part_name_define}_READ_LOCK_BIT, + false}}); + EXPECT_DIF_OK(dif_otp_ctrl_lock_reading( + &otp_, kDifOtpCtrlPartition${part_name_camel})); + % if not loop.last: + + %endif +% endfor +} + +TEST_F(ReadLockTest, NotLockablePartitions) { + bool flag; +% for part in [p for p in parts if p not in read_locked_csr_parts]: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() +%>\ + EXPECT_DIF_BADARG( + dif_otp_ctrl_lock_reading(&otp_, kDifOtpCtrlPartition${part_name_camel})); + EXPECT_DIF_BADARG(dif_otp_ctrl_reading_is_locked( + &otp_, kDifOtpCtrlPartition${part_name_camel}, &flag)); + % if not loop.last: + + %endif +% endfor +} +// clang-format on + +TEST_F(ReadLockTest, NullArgs) { + bool flag; +% for part in read_locked_csr_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() + lock_reading_line = f"dif_otp_ctrl_lock_reading(nullptr, kDifOtpCtrlPartition{part_name_camel}));" +%>\ + EXPECT_DIF_BADARG(dif_otp_ctrl_reading_is_locked( + nullptr, kDifOtpCtrlPartition${part_name_camel}, &flag)); + EXPECT_DIF_BADARG(dif_otp_ctrl_reading_is_locked( + &otp_, kDifOtpCtrlPartition${part_name_camel}, nullptr)); + % if len(lock_reading_line) > 80 - 6: + EXPECT_DIF_BADARG(dif_otp_ctrl_lock_reading( + nullptr, kDifOtpCtrlPartition${part_name_camel})); + % else: + EXPECT_DIF_BADARG( + ${lock_reading_line} + % endif + % if not loop.last: + + %endif +% endfor +} + +class StatusTest : public OtpTest {}; + +TEST_F(StatusTest, Idle) { + dif_otp_ctrl_status_t status; + + EXPECT_READ32(OTP_CTRL_STATUS_REG_OFFSET, + {{OTP_CTRL_STATUS_DAI_IDLE_BIT, true}}); + EXPECT_DIF_OK(dif_otp_ctrl_get_status(&otp_, &status)); + + EXPECT_EQ(status.codes, 1 << kDifOtpCtrlStatusCodeDaiIdle); + EXPECT_THAT(status.causes, Each(kDifOtpCtrlErrorOk)); +} + +TEST_F(StatusTest, Errors) { + dif_otp_ctrl_status_t status; + + EXPECT_READ32(OTP_CTRL_STATUS_REG_OFFSET, + { + {OTP_CTRL_STATUS_DAI_IDLE_BIT, true}, + {OTP_CTRL_STATUS_HW_CFG0_ERROR_BIT, true}, + {OTP_CTRL_STATUS_LCI_ERROR_BIT, true}, + }); + +<% + hw_cfg0_error_index = [i for i, p in enumerate(parts) + if p["name"] == "HW_CFG0"][0] + lci_error_index = len(parts) + 1 +%>\ + EXPECT_READ32(OTP_CTRL_ERR_CODE_${hw_cfg0_error_index}_REG_OFFSET, + {{OTP_CTRL_ERR_CODE_0_ERR_CODE_0_OFFSET, + OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_ECC_CORR_ERROR}}); + EXPECT_READ32(OTP_CTRL_ERR_CODE_${lci_error_index}_REG_OFFSET, + {{OTP_CTRL_ERR_CODE_0_ERR_CODE_0_OFFSET, + OTP_CTRL_ERR_CODE_0_ERR_CODE_0_VALUE_MACRO_ERROR}}); + + EXPECT_DIF_OK(dif_otp_ctrl_get_status(&otp_, &status)); + EXPECT_EQ(status.codes, (1 << kDifOtpCtrlStatusCodeDaiIdle) | + (1 << kDifOtpCtrlStatusCodeHwCfg0Error) | + (1 << kDifOtpCtrlStatusCodeLciError)); + EXPECT_EQ(status.causes[kDifOtpCtrlStatusCodeHwCfg0Error], + kDifOtpCtrlErrorMacroRecoverableRead); + EXPECT_EQ(status.causes[kDifOtpCtrlStatusCodeLciError], + kDifOtpCtrlErrorMacroUnspecified); +} + +TEST_F(StatusTest, NullArgs) { + dif_otp_ctrl_status_t status; + + EXPECT_DIF_BADARG(dif_otp_ctrl_get_status(nullptr, &status)); + EXPECT_DIF_BADARG(dif_otp_ctrl_get_status(&otp_, nullptr)); +} + +struct RelativeAddressParams { + std::string name; + dif_otp_ctrl_partition_t partition; + uint32_t abs_address; + dif_result_t expected_result; + uint32_t expected_relative_address; +}; + +class RelativeAddress + : public OtpTest, + public testing::WithParamInterface {}; + +TEST_P(RelativeAddress, RelativeAddress) { + uint32_t got_relative_address; + dif_result_t got_result = dif_otp_ctrl_relative_address( + GetParam().partition, GetParam().abs_address, &got_relative_address); + EXPECT_EQ(got_result, GetParam().expected_result); + EXPECT_EQ(got_relative_address, GetParam().expected_relative_address); +} + +INSTANTIATE_TEST_SUITE_P( + AllPartitions, RelativeAddress, + testing::Values( +% for part in parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() + part_name_define = part_name.as_c_define() + step = 8 if part["secret"] else 4 +%>\ + RelativeAddressParams{ + "${part_name_camel}Okay", + kDifOtpCtrlPartition${part_name_camel}, + OTP_CTRL_PARAM_${part_name_define}_OFFSET + ${step}, + kDifOk, + ${step}, + }, + RelativeAddressParams{ + "${part_name_camel}Unaligned", + kDifOtpCtrlPartition${part_name_camel}, + OTP_CTRL_PARAM_${part_name_define}_OFFSET + 1, + kDifUnaligned, + 0, + }, +<% + ## Exclude first partition to avoid a negative offset. +%>\ + % if not loop.first: + RelativeAddressParams{ + "${part_name_camel}OutOfRangeBeforeStart", + kDifOtpCtrlPartition${part_name_camel}, + OTP_CTRL_PARAM_${part_name_define}_OFFSET - ${step}, + kDifOutOfRange, + 0, + }, + % endif + RelativeAddressParams{ + "${part_name_camel}OutOfRangePastEnd", + kDifOtpCtrlPartition${part_name_camel}, + % if len(f"OTP_CTRL_PARAM_{part_name_define}_OFFSET + OTP_CTRL_PARAM_{part_name_define}_SIZE,") <= 80 - 12: + OTP_CTRL_PARAM_${part_name_define}_OFFSET + OTP_CTRL_PARAM_${part_name_define}_SIZE, + % else: + OTP_CTRL_PARAM_${part_name_define}_OFFSET + + OTP_CTRL_PARAM_${part_name_define}_SIZE, + % endif + kDifOutOfRange, + 0, + }${")," if loop.last else ","} +% endfor + [](const testing::TestParamInfo &info) { + return info.param.name; + }); + +class DaiReadTest : public OtpTest {}; + +TEST_F(DaiReadTest, Read32) { + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + OTP_CTRL_PARAM_MANUF_STATE_OFFSET); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_CMD_RD_BIT, true}}); + + EXPECT_DIF_OK(dif_otp_ctrl_dai_read_start(&otp_, kDifOtpCtrlPartitionHwCfg0, + /*address=*/0x20)); + + EXPECT_READ32(OTP_CTRL_DIRECT_ACCESS_RDATA_0_REG_OFFSET, 0x12345678); + + uint32_t val; + EXPECT_DIF_OK(dif_otp_ctrl_dai_read32_end(&otp_, &val)); + EXPECT_EQ(val, 0x12345678); +} + +TEST_F(DaiReadTest, Read64) { + uint64_t val; +% for part in secret_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() + part_name_define = part_name.as_c_define() +%>\ + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + OTP_CTRL_PARAM_${part_name_define}_OFFSET + 0x8); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_CMD_RD_BIT, true}}); + + EXPECT_DIF_OK(dif_otp_ctrl_dai_read_start(&otp_, kDifOtpCtrlPartition${part_name_camel}, + /*address=*/0x8)); + + EXPECT_READ32(OTP_CTRL_DIRECT_ACCESS_RDATA_1_REG_OFFSET, 0x12345678); + EXPECT_READ32(OTP_CTRL_DIRECT_ACCESS_RDATA_0_REG_OFFSET, 0x90abcdef); + + EXPECT_DIF_OK(dif_otp_ctrl_dai_read64_end(&otp_, &val)); + EXPECT_EQ(val, 0x1234567890abcdef); + % if not loop.last: + + % endif +% endfor +} + +TEST_F(DaiReadTest, Unaligned) { + EXPECT_EQ(dif_otp_ctrl_dai_read_start(&otp_, kDifOtpCtrlPartitionHwCfg0, + /*address=*/0b01), + kDifUnaligned); + EXPECT_EQ(dif_otp_ctrl_dai_read_start(&otp_, kDifOtpCtrlPartitionSecret2, + /*address=*/0b100), + kDifUnaligned); +} + +TEST_F(DaiReadTest, OutOfRange) { + EXPECT_EQ(dif_otp_ctrl_dai_read_start(&otp_, kDifOtpCtrlPartitionHwCfg0, + /*address=*/0x100), + kDifOutOfRange); +} + +TEST_F(DaiReadTest, NullArgs) { + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_read_start(nullptr, + kDifOtpCtrlPartitionHwCfg0, + /*address=*/0x0)); + + uint32_t val32; + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_read32_end(nullptr, &val32)); + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_read32_end(&otp_, nullptr)); + + uint64_t val64; + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_read64_end(nullptr, &val64)); + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_read64_end(&otp_, nullptr)); +} + +class DaiProgramTest : public OtpTest {}; + +TEST_F(DaiProgramTest, Program32) { + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + OTP_CTRL_PARAM_MANUF_STATE_OFFSET); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET, 0x12345678); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT, true}}); + + EXPECT_DIF_OK(dif_otp_ctrl_dai_program32(&otp_, kDifOtpCtrlPartitionHwCfg0, + /*address=*/0x20, + /*value=*/0x12345678)); +} + +TEST_F(DaiProgramTest, Program64) { + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + OTP_CTRL_PARAM_SECRET2_OFFSET + 0x8); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET, 0x90abcdef); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_WDATA_1_REG_OFFSET, 0x12345678); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT, true}}); + + EXPECT_DIF_OK(dif_otp_ctrl_dai_program64(&otp_, kDifOtpCtrlPartitionSecret2, + /*address=*/0x8, + /*value=*/0x1234567890abcdef)); +} + +TEST_F(DaiProgramTest, BadPartition) { + EXPECT_EQ(dif_otp_ctrl_dai_program32(&otp_, kDifOtpCtrlPartitionSecret1, + /*address=*/0x0, /*value=*/42), + kDifError); + EXPECT_EQ(dif_otp_ctrl_dai_program64(&otp_, kDifOtpCtrlPartitionHwCfg0, + /*address=*/0x0, /*value=*/42), + kDifError); + + // LC is never writeable. + EXPECT_EQ(dif_otp_ctrl_dai_program32(&otp_, kDifOtpCtrlPartitionLifeCycle, + /*address=*/0x0, /*value=*/42), + kDifError); +} + +TEST_F(DaiProgramTest, Unaligned) { + EXPECT_EQ(dif_otp_ctrl_dai_program32(&otp_, kDifOtpCtrlPartitionHwCfg0, + /*address=*/0b01, /*value=*/42), + kDifUnaligned); + EXPECT_EQ(dif_otp_ctrl_dai_program64(&otp_, kDifOtpCtrlPartitionSecret2, + /*address=*/0b100, /*value=*/42), + kDifUnaligned); +} + +TEST_F(DaiProgramTest, OutOfRange) { + // Check that we can't write a digest directly. + EXPECT_EQ(dif_otp_ctrl_dai_program32( + &otp_, kDifOtpCtrlPartitionCreatorSwCfg, + /*address=*/OTP_CTRL_PARAM_CREATOR_SW_CFG_DIGEST_OFFSET, + /*value=*/42), + kDifOutOfRange); + + // Same digest check for 64-bit. + EXPECT_EQ(dif_otp_ctrl_dai_program64( + &otp_, kDifOtpCtrlPartitionSecret2, + /*address=*/OTP_CTRL_PARAM_SECRET2_DIGEST_OFFSET, /*value=*/42), + kDifOutOfRange); +} + +TEST_F(DaiProgramTest, NullArgs) { + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_program32(nullptr, + kDifOtpCtrlPartitionHwCfg0, + /*address=*/0x0, /*value=*/42)); + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_program64(nullptr, + kDifOtpCtrlPartitionSecret0, + /*address=*/0x0, /*value=*/42)); +} + +class DaiDigestTest : public OtpTest {}; + +TEST_F(DaiDigestTest, DigestSw) { +% for part in sw_digest_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_define = part_name.as_c_define() + part_name_camel = part_name.as_camel_case() + dai_digest_line = ("EXPECT_DIF_OK(dif_otp_ctrl_dai_digest(&otp_, " + + f"kDifOtpCtrlPartition{part_name_camel},") +%>\ + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + OTP_CTRL_PARAM_${part_name.as_c_define()}_DIGEST_OFFSET); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_WDATA_0_REG_OFFSET, 0x00abcdef); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_WDATA_1_REG_OFFSET, 0xabcdef00); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_CMD_WR_BIT, true}}); + + % if len(dai_digest_line) > 80 - 2: +<% dai_digest_line = (f"kDifOtpCtrlPartition{part_name_camel},") %>\ + % if len(dai_digest_line) > 80 - 40: + EXPECT_DIF_OK( + dif_otp_ctrl_dai_digest(&otp_, kDifOtpCtrlPartition${part_name_camel}, + /*digest=*/0xabcdef0000abcdef)); + % else: + EXPECT_DIF_OK(dif_otp_ctrl_dai_digest(&otp_, + kDifOtpCtrlPartition${part_name_camel}, + /*digest=*/0xabcdef0000abcdef)); + % endif + % else: + ${dai_digest_line} + /*digest=*/0xabcdef0000abcdef)); + % endif + % if not loop.last: + + % endif +% endfor +} + +TEST_F(DaiDigestTest, DigestHw) { +% for part in hw_digest_parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_define = part_name.as_c_define() + part_name_camel = part_name.as_camel_case() + dai_digest_line = ("EXPECT_DIF_OK(dif_otp_ctrl_dai_digest(&otp_, " + + f"kDifOtpCtrlPartition{part_name_camel},") +%>\ + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_ADDRESS_REG_OFFSET, + OTP_CTRL_PARAM_${part_name_define}_OFFSET); + EXPECT_WRITE32(OTP_CTRL_DIRECT_ACCESS_CMD_REG_OFFSET, + {{OTP_CTRL_DIRECT_ACCESS_CMD_DIGEST_BIT, true}}); + + EXPECT_DIF_OK(dif_otp_ctrl_dai_digest(&otp_, kDifOtpCtrlPartition${part_name_camel}, + /*digest=*/0)); + % if not loop.last: + + % endif +% endfor +} + +TEST_F(DaiDigestTest, BadPartition) { + EXPECT_EQ(dif_otp_ctrl_dai_digest(&otp_, kDifOtpCtrlPartitionLifeCycle, + /*digest=*/0), + kDifError); +} + +TEST_F(DaiDigestTest, BadDigest) { + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_digest(&otp_, kDifOtpCtrlPartitionHwCfg0, + /*digest=*/0xabcdef0000abcdef)); + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_digest(&otp_, + kDifOtpCtrlPartitionCreatorSwCfg, + /*digest=*/0)); +} + +TEST_F(DaiDigestTest, NullArgs) { + EXPECT_DIF_BADARG(dif_otp_ctrl_dai_digest(nullptr, + kDifOtpCtrlPartitionCreatorSwCfg, + /*digest=*/0xabcdef0000abcdef)); +} + +class IsDigestComputed : public OtpTest {}; + +TEST_F(IsDigestComputed, NullArgs) { + bool is_computed; + EXPECT_DIF_BADARG(dif_otp_ctrl_is_digest_computed( + nullptr, kDifOtpCtrlPartitionSecret2, &is_computed)); + EXPECT_DIF_BADARG(dif_otp_ctrl_is_digest_computed( + &otp_, kDifOtpCtrlPartitionSecret2, nullptr)); + EXPECT_DIF_BADARG(dif_otp_ctrl_is_digest_computed( + nullptr, kDifOtpCtrlPartitionSecret2, nullptr)); +} + +TEST_F(IsDigestComputed, BadPartition) { + bool is_computed; + EXPECT_DIF_BADARG(dif_otp_ctrl_is_digest_computed( + &otp_, kDifOtpCtrlPartitionLifeCycle, &is_computed)); +} + +TEST_F(IsDigestComputed, Success) { + bool is_computed; + + EXPECT_READ32(OTP_CTRL_SECRET2_DIGEST_1_REG_OFFSET, 0x98abcdef); + EXPECT_READ32(OTP_CTRL_SECRET2_DIGEST_0_REG_OFFSET, 0xabcdef01); + EXPECT_DIF_OK(dif_otp_ctrl_is_digest_computed( + &otp_, kDifOtpCtrlPartitionSecret2, &is_computed)); + EXPECT_TRUE(is_computed); + + EXPECT_READ32(OTP_CTRL_SECRET2_DIGEST_1_REG_OFFSET, 0); + EXPECT_READ32(OTP_CTRL_SECRET2_DIGEST_0_REG_OFFSET, 0); + EXPECT_DIF_OK(dif_otp_ctrl_is_digest_computed( + &otp_, kDifOtpCtrlPartitionSecret2, &is_computed)); + EXPECT_FALSE(is_computed); +} + +struct DigestParams { + dif_otp_ctrl_partition_t partition; + bool has_digest; + ptrdiff_t reg0, reg1; +}; + +class GetDigest : public OtpTest, + public testing::WithParamInterface {}; + +TEST_P(GetDigest, GetDigest) { + if (!GetParam().has_digest) { + uint64_t digest; + EXPECT_DIF_BADARG( + dif_otp_ctrl_get_digest(&otp_, GetParam().partition, &digest)); + return; + } + + EXPECT_READ32(GetParam().reg1, 0xabcdef99); + EXPECT_READ32(GetParam().reg0, 0x99abcdef); + + uint64_t digest; + EXPECT_DIF_OK(dif_otp_ctrl_get_digest(&otp_, GetParam().partition, &digest)); + EXPECT_EQ(digest, 0xabcdef9999abcdef); +} + +TEST_P(GetDigest, BadDigest) { + if (!GetParam().has_digest) { + return; + } + + EXPECT_READ32(GetParam().reg1, 0x0); + EXPECT_READ32(GetParam().reg0, 0x0); + + uint64_t digest; + EXPECT_EQ(dif_otp_ctrl_get_digest(&otp_, GetParam().partition, &digest), + kDifError); +} + +TEST_P(GetDigest, NullArgs) { + uint64_t digest; + EXPECT_DIF_BADARG( + dif_otp_ctrl_get_digest(nullptr, GetParam().partition, &digest)); + EXPECT_DIF_BADARG( + dif_otp_ctrl_get_digest(&otp_, GetParam().partition, nullptr)); +} + +// This depends on the maximum length of partition names, which will +// be changing, so turn formatting off. +// clang-format off +INSTANTIATE_TEST_SUITE_P( + AllDigests, GetDigest, + testing::Values( +% for part in parts: +<% + part_name = Name.from_snake_case(part["name"]) + part_name_camel = part_name.as_camel_case() + part_name_define = part_name.as_c_define() +%>\ + % if part in digest_parts: + DigestParams{ + kDifOtpCtrlPartition${part_name_camel}, + true, + OTP_CTRL_${part_name_define}_DIGEST_0_REG_OFFSET, + OTP_CTRL_${part_name_define}_DIGEST_1_REG_OFFSET, + }${"" if loop.last else ","} + % else: + DigestParams{ + kDifOtpCtrlPartition${part_name_camel}, + false, + 0, + 0, + }${"));" if loop.last else ","} + % endif +% endfor +// clang-format on + +class BlockingIoTest : public OtpTest { + protected: + static constexpr size_t kWords = 4; +}; + +TEST_F(BlockingIoTest, Read) { + for (size_t i = 0; i < kWords; ++i) { + auto offset = + OTP_CTRL_PARAM_OWNER_SW_CFG_OFFSET + 0x10 + i * sizeof(uint32_t); + EXPECT_READ32(OTP_CTRL_SW_CFG_WINDOW_REG_OFFSET + offset, i + 1); + } + + std::vector buf(kWords); + EXPECT_DIF_OK(dif_otp_ctrl_read_blocking( + &otp_, kDifOtpCtrlPartitionOwnerSwCfg, 0x10, buf.data(), buf.size())); + EXPECT_THAT(buf, ElementsAre(1, 2, 3, 4)); +} + +TEST_F(BlockingIoTest, BadPartition) { + std::vector buf(kWords); + EXPECT_EQ(dif_otp_ctrl_read_blocking(&otp_, kDifOtpCtrlPartitionHwCfg0, 0x10, + buf.data(), buf.size()), + kDifError); +} + +TEST_F(BlockingIoTest, Unaligned) { + std::vector buf(kWords); + EXPECT_EQ(dif_otp_ctrl_read_blocking(&otp_, kDifOtpCtrlPartitionOwnerSwCfg, + 0x11, buf.data(), buf.size()), + kDifUnaligned); +} + +TEST_F(BlockingIoTest, OutOfRange) { + std::vector buf(0x2f0); + EXPECT_EQ(dif_otp_ctrl_read_blocking(&otp_, kDifOtpCtrlPartitionOwnerSwCfg, + 0x300, buf.data(), buf.size()), + kDifOutOfRange); + EXPECT_EQ(dif_otp_ctrl_read_blocking(&otp_, kDifOtpCtrlPartitionOwnerSwCfg, + 0x10, buf.data(), 0x330), + kDifOutOfRange); +} + +TEST_F(BlockingIoTest, NullArgs) { + std::vector buf(kWords); + EXPECT_DIF_BADARG(dif_otp_ctrl_read_blocking( + nullptr, kDifOtpCtrlPartitionOwnerSwCfg, 0x10, buf.data(), buf.size())); + EXPECT_DIF_BADARG(dif_otp_ctrl_read_blocking( + &otp_, kDifOtpCtrlPartitionOwnerSwCfg, 0x10, nullptr, buf.size())); +} + +} // namespace +} // namespace dif_otp_ctrl_unittest diff --git a/hw/ip/otp_ctrl/data/otp_ctrl.hjson b/hw/top_earlgrey/ip_autogen/otp_ctrl/data/otp_ctrl.hjson similarity index 99% rename from hw/ip/otp_ctrl/data/otp_ctrl.hjson rename to hw/top_earlgrey/ip_autogen/otp_ctrl/data/otp_ctrl.hjson index 44844724b52a81..8339633c1e9ef4 100644 --- a/hw/ip/otp_ctrl/data/otp_ctrl.hjson +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/data/otp_ctrl.hjson @@ -4,9 +4,6 @@ // // HJSON with partition metadata. // -// DO NOT EDIT THIS FILE DIRECTLY. -// It has been generated with ./util/design/gen-otp-mmap.py - { name: "otp_ctrl", human_name: "One-Time Programmable Memory Controller", @@ -114,19 +111,19 @@ // Random netlist constants { name: "RndCnstLfsrSeed", desc: "Compile-time random bits for initial LFSR seed", - type: "otp_ctrl_pkg::lfsr_seed_t" + type: "otp_ctrl_top_specific_pkg::lfsr_seed_t" randcount: "40", randtype: "data", // randomize randcount databits } { name: "RndCnstLfsrPerm", desc: "Compile-time random permutation for LFSR output", - type: "otp_ctrl_pkg::lfsr_perm_t" + type: "otp_ctrl_top_specific_pkg::lfsr_perm_t" randcount: "40", randtype: "perm", // random permutation for randcount elements } { name: "RndCnstScrmblKeyInit", desc: "Compile-time random permutation for scrambling key/nonce register reset value", - type: "otp_ctrl_pkg::scrmbl_key_init_t" + type: "otp_ctrl_top_specific_pkg::scrmbl_key_init_t" randcount: "256", randtype: "data", // random permutation for randcount elements } diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/data/otp_ctrl_sec_cm_testplan.hjson b/hw/top_earlgrey/ip_autogen/otp_ctrl/data/otp_ctrl_sec_cm_testplan.hjson new file mode 100644 index 00000000000000..650a36067a49ca --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/data/otp_ctrl_sec_cm_testplan.hjson @@ -0,0 +1,303 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// Security countermeasures testplan extracted from the IP Hjson using reggen. +// +// This testplan is auto-generated only the first time it is created. This is +// because this testplan needs to be hand-editable. It is possible that these +// testpoints can go out of date if the spec is updated with new +// countermeasures. When `reggen` is invoked when this testplan already exists, +// It checks if the list of testpoints is up-to-date and enforces the user to +// make further manual updates. +// +// These countermeasures and their descriptions can be found here: +// .../otp_ctrl/data/otp_ctrl.hjson +// +// It is possible that the testing of some of these countermeasures may already +// be covered as a testpoint in a different testplan. This duplication is ok - +// the test would have likely already been developed. We simply map those tests +// to the testpoints below using the `tests` key. +// +// Please ensure that this testplan is imported in: +// .../otp_ctrl/data/otp_ctrl_testplan.hjson +{ + testpoints: [ + { + name: sec_cm_bus_integrity + desc: "Verify the countermeasure(s) BUS.INTEGRITY." + stage: V2S + tests: ["otp_ctrl_tl_intg_err"] + } + { + name: sec_cm_secret_mem_scramble + desc: "Verify the countermeasure(s) SECRET.MEM.SCRAMBLE." + stage: V2S + tests: ["otp_ctrl_smoke"] + } + { + name: sec_cm_part_mem_digest + desc: "Verify the countermeasure(s) PART.MEM.DIGEST." + stage: V2S + tests: ["otp_ctrl_smoke"] + } + { + name: sec_cm_dai_fsm_sparse + desc: "Verify the countermeasure(s) DAI.FSM.SPARSE." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_kdi_fsm_sparse + desc: "Verify the countermeasure(s) KDI.FSM.SPARSE." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_lci_fsm_sparse + desc: "Verify the countermeasure(s) LCI.FSM.SPARSE." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_part_fsm_sparse + desc: "Verify the countermeasure(s) PART.FSM.SPARSE." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_scrmbl_fsm_sparse + desc: "Verify the countermeasure(s) SCRMBL.FSM.SPARSE." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_timer_fsm_sparse + desc: "Verify the countermeasure(s) TIMER.FSM.SPARSE." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_dai_ctr_redun + desc: "Verify the countermeasure(s) DAI.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_kdi_seed_ctr_redun + desc: "Verify the countermeasure(s) KDI_SEED.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_kdi_entropy_ctr_redun + desc: "Verify the countermeasure(s) KDI_ENTROPY.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_lci_ctr_redun + desc: "Verify the countermeasure(s) LCI.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_part_ctr_redun + desc: "Verify the countermeasure(s) PART.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_scrmbl_ctr_redun + desc: "Verify the countermeasure(s) SCRMBL.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_timer_integ_ctr_redun + desc: "Verify the countermeasure(s) TIMER_INTEG.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_timer_cnsty_ctr_redun + desc: "Verify the countermeasure(s) TIMER_CNSTY.CTR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_timer_lfsr_redun + desc: "Verify the countermeasure(s) TIMER.LFSR.REDUN." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_dai_fsm_local_esc + desc: "Verify the countermeasure(s) DAI.FSM.LOCAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc", "otp_ctrl_sec_cm"] + } + { + name: sec_cm_lci_fsm_local_esc + desc: "Verify the countermeasure(s) LCI.FSM.LOCAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc"] + } + { + name: sec_cm_kdi_fsm_local_esc + desc: "Verify the countermeasure(s) KDI.FSM.LOCAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc"] + } + { + name: sec_cm_part_fsm_local_esc + desc: "Verify the countermeasure(s) PART.FSM.LOCAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc", "otp_ctrl_macro_errs"] + } + { + name: sec_cm_scrmbl_fsm_local_esc + desc: "Verify the countermeasure(s) SCRMBL.FSM.LOCAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc"] + } + { + name: sec_cm_timer_fsm_local_esc + desc: "Verify the countermeasure(s) TIMER.FSM.LOCAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc", "otp_ctrl_sec_cm"] + } + { + name: sec_cm_dai_fsm_global_esc + desc: "Verify the countermeasure(s) DAI.FSM.GLOBAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc", "otp_ctrl_sec_cm"] + } + { + name: sec_cm_lci_fsm_global_esc + desc: "Verify the countermeasure(s) LCI.FSM.GLOBAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc"] + } + { + name: sec_cm_kdi_fsm_global_esc + desc: "Verify the countermeasure(s) KDI.FSM.GLOBAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc"] + } + { + name: sec_cm_part_fsm_global_esc + desc: "Verify the countermeasure(s) PART.FSM.GLOBAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc", "otp_ctrl_macro_errs"] + } + { + name: sec_cm_scrmbl_fsm_global_esc + desc: "Verify the countermeasure(s) SCRMBL.FSM.GLOBAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc"] + } + { + name: sec_cm_timer_fsm_global_esc + desc: "Verify the countermeasure(s) TIMER.FSM.GLOBAL_ESC." + stage: V2S + tests: ["otp_ctrl_parallel_lc_esc", "otp_ctrl_sec_cm"] + } + { + name: sec_cm_part_data_reg_integrity + desc: "Verify the countermeasure(s) PART.DATA_REG.INTEGRITY." + stage: V2S + tests: ["otp_ctrl_init_fail"] + } + { + name: sec_cm_part_data_reg_bkgn_chk + desc: "Verify the countermeasure(s) PART.DATA_REG.BKGN_CHK." + stage: V2S + tests: ["otp_ctrl_check_fail"] + } + { + name: sec_cm_part_mem_regren + desc: "Verify the countermeasure(s) PART.MEM.REGREN." + stage: V2S + tests: ["otp_ctrl_dai_lock"] + } + { + name: sec_cm_part_mem_sw_unreadable + desc: "Verify the countermeasure(s) PART.MEM.SW_UNREADABLE." + stage: V2S + tests: ["otp_ctrl_dai_lock"] + } + { + name: sec_cm_part_mem_sw_unwritable + desc: "Verify the countermeasure(s) PART.MEM.SW_UNWRITABLE." + stage: V2S + tests: ["otp_ctrl_dai_lock"] + } + { + name: sec_cm_lc_part_mem_sw_noaccess + desc: "Verify the countermeasure(s) LC_PART.MEM.SW_NOACCESS." + stage: V2S + tests: ["otp_ctrl_dai_lock"] + } + { + name: sec_cm_access_ctrl_mubi + desc: "Verify the countermeasure(s) ACCESS.CTRL.MUBI." + stage: V2S + tests: ["otp_ctrl_dai_lock"] + } + { + name: sec_cm_token_valid_ctrl_mubi + desc: "Verify the countermeasure(s) TOKEN_VALID.CTRL.MUBI." + stage: V2S + tests: ["otp_ctrl_smoke"] + } + { + name: sec_cm_lc_ctrl_intersig_mubi + desc: "Verify the countermeasure(s) LC_CTRL.INTERSIG.MUBI." + stage: V2S + tests: ["otp_ctrl_dai_lock"] + } + { + name: sec_cm_test_bus_lc_gated + desc: "Verify the countermeasure(s) TEST.BUS.LC_GATED." + stage: V2S + tests: ["otp_ctrl_smoke"] + } + { + name: sec_cm_test_tl_lc_gate_fsm_sparse + desc: "Verify the countermeasure(s) TEST_TL_LC_GATE.FSM.SPARSE." + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: sec_cm_direct_access_config_regwen + desc: "Verify the countermeasure(s) DIRECT_ACCESS.CONFIG.REGWEN." + stage: V2S + tests: ["otp_ctrl_regwen"] + } + { + name: sec_cm_check_trigger_config_regwen + desc: "Verify the countermeasure(s) CHECK_TRIGGER.CONFIG.REGWEN." + stage: V2S + tests: ["otp_ctrl_smoke"] + } + { + name: sec_cm_check_config_regwen + desc: "Verify the countermeasure(s) CHECK.CONFIG.REGWEN." + stage: V2S + tests: ["otp_ctrl_smoke"] + } + { + name: sec_cm_macro_mem_integrity + desc: "Verify the countermeasure(s) MACRO.MEM.INTEGRITY." + stage: V2S + tests: ["otp_ctrl_macro_errs"] + } + { + name: sec_cm_macro_mem_cm + desc: "Verify the countermeasure(s) MACRO.MEM.CM." + stage: V2S + tests: ["N/A"] + } + ] +} diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/data/otp_ctrl_testplan.hjson b/hw/top_earlgrey/ip_autogen/otp_ctrl/data/otp_ctrl_testplan.hjson new file mode 100644 index 00000000000000..95b6081b7da897 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/data/otp_ctrl_testplan.hjson @@ -0,0 +1,404 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +{ + name: "otp_ctrl" + import_testplans: ["hw/dv/tools/dvsim/testplans/csr_testplan.hjson", + "hw/dv/tools/dvsim/testplans/mem_testplan.hjson", + "hw/dv/tools/dvsim/testplans/intr_test_testplan.hjson", + "hw/dv/tools/dvsim/testplans/alert_test_testplan.hjson", + "hw/dv/tools/dvsim/testplans/tl_device_access_types_testplan.hjson", + "hw/dv/tools/dvsim/testplans/sec_cm_count_testplan.hjson", + "hw/dv/tools/dvsim/testplans/sec_cm_fsm_testplan.hjson", + "hw/dv/tools/dvsim/testplans/stress_all_with_reset_testplan.hjson", + "otp_ctrl_sec_cm_testplan.hjson"] + testpoints: [ + { + name: wake_up + desc: ''' + Wake_up test walks through otp_ctrl's power-on initialization, read, program, and + digest functionalities. + + - drive pwrmgr's request pin to trigger OTP initialization after reset, check status + after OTP initialization + - write all-ones to a random address within OTP partition 0, wait until this operation + completes + - read out the random selected write address, check if the readout value is all-ones + - trigger a digest calculation for a Software partition, check if the OtpError + interrupt is set + - trigger a digest calculation for a non-software partition, expect operation completes + without the OtpError interrupt + - read out secrets through the hardware interfaces + ''' + stage: V1 + tests: ["otp_ctrl_wake_up"] + } + { + name: smoke + desc: ''' + OTP_CTRL smoke test provisions and locks partitions. + + - drive pwrmgr's request pin to trigger OTP initialization after reset, check status + after OTP initialization + - randomly read out keys pertaining to `key_manager`, `flash`, `sram`, `otbn` + - randomly issue LC program request + - write random values to random addresses within each OTP partition + - read out the random selected write addresses, check if the readout values are expected + - during read and write operations, check if direct_access_regwen is correctly set by HW + - perform a system-level reset and check corresponding CSRs are set correctly + - lock all partitions except life_cycle by triggering digest calculations + - read back and verify the digest + - perform a system-level reset to verify the corresponding CSRs exposing the digests + have been populated + + **Checks**: + - Assertion checks to ensure vendor specific I/Os: `otp_vendor_test_status_o`, + `otp_vendor_test_ctrl_i`, `cio_test_o`, and `cio_test_en_o` are connected currently + with `lc_dft_en_i` On and Off. + ''' + stage: V1 + tests: ["otp_ctrl_smoke"] + } + { + name: dai_access_partition_walk + desc: ''' + Similar to UVM's memory walk test, this test ensures every address in each partition + can be accessed successfully via DAI and TLUL interfacs according to its access policy. + ''' + stage: V2 + tests: ["otp_ctrl_partition_walk"] + } + { + name: init_fail + desc: ''' + Based on OTP_CTRL smoke test, this test creates OTP_CTRL's initialization failure: + - write and read OTP memory via DAI interface + - randomly issue DAI digest command to lock HW partitions + - keep writing to OTP memory via DAI interface without asserting reset + - if digests are not locked, backdoor inject ECC correctable or uncorrectable errors + - issue reset and power initialization + - if the injected errors are all correctable errors, disable the `lc_bypass_chk_en` + after LC program request to create an LC partition check failure + + If fatal error is triggered, this test will check: + - OTP initialization failure triggers fatal alert + - `status`, `intr_state`, `err_code` CSRs reflect correct fatal error + + If OTP initialization finished without any fatal error, this test will check: + - OTP initialization finishes with power init output goes to 1 + - `status`, `intr_state`, `err_code` CSRs reflect ECC correctable error + ''' + stage: V2 + tests: ["otp_ctrl_init_fail"] + } + { + name: partition_check + desc: ''' + Randomly program the partition check related CSRs including: + - `check_timeout` + - `integrity_check_period` + - `consistency_check_period` + - `check_trigger` + + Create a failure scenario by randomly picking one of these three methods: + - inject ECC errors into the OTP macro via backdoor + - set the `check_timeout` CSR with a very small value + - write to a random OTP partition after digest is issued but before reset is asserted + + **Checks**: + - the corresponding alerts are triggered + - the error_code register is set correctly + Note that due to limited simulation time, for background checks, this test only write + random value that is less than 20 to the check period. + ''' + stage: V2 + tests: ["otp_ctrl_check_fail", "otp_ctrl_background_chks"] + } + { + name: regwen_during_otp_init + desc: ''' + The `direct_access_regwen` is a RO register which controls the write-enable of other + reigsters. It is not verified by the common CSR tests. HW sets it to 0 when the DAI + interface is busy. + + Stimulus and checks: + - randomly read `direct_access_regwen` and verify that it returns 0 during OTP + initialization + - verify that the writes to the registers controlled by it do not go through during OTP + initialization + ''' + stage: V2 + tests: ["otp_ctrl_regwen"] + } + { + name: partition_lock + desc: ''' + This test will cover two methods of locking read and write: digest calculation and CSR + write. After locking the partitions, issue read or program sequences and check if the + operations are locked correctly, and check if the `AccessError` is set. + ''' + stage: V2 + tests: ["otp_ctrl_dai_lock"] + } + { + name: interface_key_check + desc: ''' + OTP_CTRL will generate keys to `flash`, `sram`, and `otbn` upon their requests. + Based on the DAI access sequence, this test will run key requests sequence in + parallel, and check if correct keys are generated. + ''' + stage: V2 + tests: ["otp_ctrl_parallel_key_req"] + } + { + name: lc_interactions + desc: ''' + Verify the procotols between OTP_CTRL and LC_CTRL. Based on the DAI access sequence, + run the following sequences in parallel: + + - request a LC state transition via the programming interface + - enable the `lc_escalation_en` signal + + **Checks**: + - if the LC program request has `AccessError`, check the LC program response sets + the `error` bit to 1 + - if `lc_escalation_en` is enabled, verify that alert is triggered and OTP_CTRL entered + terminal state + ''' + stage: V2 + tests: ["otp_ctrl_parallel_lc_req", "otp_ctrl_parallel_lc_esc"] + } + { name: otp_dai_errors + desc: ''' + Based on the otp_dai_lock test, this test will randomly run the following OTP errors: + - DAI interface writes non-blank OTP address + - DAI interface accesses LC partition + - DAI interface writes HW digests + - DAI interface writes non-empty memory + + **Checks**: + - `err_code` and `status` CSRs + - `otp_error` interrupt + ''' + stage: V2 + tests: ["otp_ctrl_dai_errs"] + } + { name: otp_macro_errors + desc: ''' + Randomly run the following OTP errors: + - MacroError + - MacroEccCorrError + - MacroEccUncorrError + + **Checks**: + - `err_code` and `status` CSRs + - `otp_error` interrupt + - if the error is unrecoverable, verify that alert is triggered and OTP_CTRL entered + terminal state + ''' + stage: V2 + tests: ["otp_ctrl_macro_errs"] + } + { + name: test_access + desc: ''' + This test checks if the test access to OTP macro is connected correctly. + + **Stimulus and Checks**: + - Write and check read results from the prim_tl_i/o. + - Ensure no error or alert occurs from DUT. + ''' + stage: V2 + tests: ["otp_ctrl_test_access"] + } + { + name: stress_all + desc: ''' + - combine above sequences in one test to run sequentially, except csr sequence + - randomly add reset between each sequence + ''' + stage: V2 + tests: ["{name}_stress_all"] + } + { + name: sec_cm_additional_check + desc: ''' + Verify the outcome of injecting faults to security countermeasures. + + Stimulus: + As mentioned in `prim_count_check`, `prim_fsm_check` and `prim_double_lfsr_check`. + + Checks: + - Check the value of status register according to where the fault is injected. + - Check OTP_CTRL is locked after the fatal fault injection by trying to access OTP_CTRL + via dai, kdi, and lci interfaces. + ''' + stage: V2S + tests: ["otp_ctrl_sec_cm"] + } + { + name: otp_ctrl_low_freq_read + desc: ''' + This test checks if OTP's read operation can operate successfully in a low clock + frequency before the clock is calibrated. + + **Stimulus and Checks**: + - Configure OTP_CTRL's clock to 6MHz low frequency. + - Backdoor write OTP memory. + - Use DAI access to read each memory address and compare if the value is correct. + - If DAI address is in a SW partition, read and check again via TLUL interface. + ''' + stage: V3 + tests: ["otp_ctrl_low_freq_read"] + } + ] + + covergroups: [ + { + name: power_on_cg + desc: '''Covers the following conditions when OTP_CTRL finishes power-on initialization: + - whether `lc_escalation_en` is On + - whether any partition (except life cycle partition) is locked + ''' + } + { + name: flash_req_cg + desc: '''Covers whether secret1 partition is locked during `flash` data or address + request.''' + } + { + name: sram_req_cg + desc: '''Covers whether secret1 partition is locked during all `srams` key request.''' + } + { + name: otbn_req_cg + desc: '''Covers whether secret1 partition is locked during `otbn` key request.''' + } + { + name: lc_prog_cg + desc: '''Covers whether the error bit is set during LC program request.''' + } + { + name: keymgr_o_cg + desc: '''Covers the following conditions when scoreboard checks `keymgr_o` value: + - whether secret2 partition is locked + - whether `lc_seed_hw_rd_en_i` is On + ''' + } + { + name: req_dai_access_after_alert_cg + desc: '''Covers if sequence issued various DAI requests after any fatal alert is + triggered.''' + } + { + name: issue_checks_after_alert_cg + desc: '''Covers if sequence issued various OTP_CTRL's background checks after any fatal alert + is triggered.''' + } + { + name: csr_rd_after_alert_cg + desc: '''Covers if the following CSRs are being read and the value is checked in scoreboard + after any fatal alert is triggered: + - unbuffered partitions' digest CSRs + - HW partition's digest CSRs + - secrets partitions' digest CSRs + - direct_access read data CSRs + - status CSR + - error_code CSR + ''' + } + { + name: dai_err_code_cg + desc: '''Covers all applicable error codes in DAI, and cross each error code with all + 7 partitions.''' + } + { + name: lci_err_code_cg + desc: '''Covers all applicable error codes in LCI.''' + } + { + name: unbuf_err_code_cg + desc: '''This is an array of covergroups to cover all applicable error codes in three + unbuffered partitions.''' + } + { + name: buf_err_code_cg + desc: '''This is an array of covergroups to cover all applicable error codes in five + buffered partitions.''' + } + { + name: unbuf_access_lock_cg_wrap_cg + desc: '''This is an array of covergroups to cover lock conditions below in three + unbuffered partitions: + - the partition is write-locked + - the partition is read-locked + - the current operation type + Then cross the three coverpoints.''' + } + { + name: dai_access_secret2_cg + desc: '''Covers whether `lc_creator_seed_sw_rw_en` is On during any DAI accesses.''' + } + { + name: status_csr_cg + desc: '''Covers the value of every bit in `status` CSR.''' + } + // The following covergroups are implemented in `otp_ctrl_cov_if.sv`. + { + name: lc_esc_en_condition_cg + desc: '''Covers the following conditions when `lc_escalation_en` is On: + - whether any key requests is in progress + - whether LC program reqeust is in progress + - whether DAI interface is busy + ''' + } + { + name: flash_data_req_condition_cg + desc: '''Covers the following conditions when `lc_escalation_en` is On: + - whether any key requests is in progress + - whether DAI interface is busy + - whether lc_esc_en is On + ''' + } + { + name: flash_addr_req_condition_cg + desc: '''Covers the following conditions when `lc_escalation_en` is On: + - whether any key requests is in progress + - whether DAI interface is busy + - whether lc_esc_en is On + ''' + } + { + name: sram_0_req_condition_cg + desc: '''Covers the following conditions when `lc_escalation_en` is On: + - whether any key requests is in progress + - whether DAI interface is busy + - whether lc_esc_en is On + ''' + } + { + name: sram_1_req_condition_cg + desc: '''Covers the following conditions when `lc_escalation_en` is On: + - whether any key requests is in progress + - whether DAI interface is busy + - whether lc_esc_en is On + ''' + } + { + name: otbn_req_condition_cg + desc: '''Covers the following conditions when `lc_escalation_en` is On: + - whether any key requests is in progress + - whether DAI interface is busy + - whether lc_esc_en is On + ''' + } + { + name: lc_prog_req_condition_cg + desc: '''Covers the following conditions when `lc_escalation_en` is On: + - whether any key requests is in progress + - whether DAI interface is busy + - whether lc_esc_en is On + ''' + } + ] +} diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/data/top_earlgrey_otp_ctrl.ipconfig.hjson b/hw/top_earlgrey/ip_autogen/otp_ctrl/data/top_earlgrey_otp_ctrl.ipconfig.hjson new file mode 100644 index 00000000000000..fac0eb85b68688 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/data/top_earlgrey_otp_ctrl.ipconfig.hjson @@ -0,0 +1,1990 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +{ + instance_name: top_earlgrey_otp_ctrl + param_values: + { + otp_mmap: + { + seed: 36021179872380457113239299468132194022238108125576166239904535336103582949069 + otp: + { + width: 2 + depth: 1024 + size: 2048 + addr_width: 10 + byte_addr_width: 11 + } + scrambling: + { + key_size: 16 + iv_size: 8 + cnst_size: 16 + keys: + [ + { + name: Secret0Key + value: 79261096542507317550041524418378318810 + } + { + name: Secret1Key + value: 318985785134653313354004803705056942690 + } + { + name: Secret2Key + value: 177669530988656006400383198120896388291 + } + ] + digests: + [ + { + name: CnstyDigest + iv_value: 13739798386390731029 + cnst_value: 19387815248521063059786024249941633354 + } + { + name: FlashDataKey + iv_value: 16161367748560767875 + cnst_value: 52429663447164110622287741421444995907 + } + { + name: FlashAddrKey + iv_value: 825413095277504501 + cnst_value: 284497036978760360808556468979666055207 + } + { + name: SramDataKey + iv_value: 17981827341411119748 + cnst_value: 98543724195391010298071380840458392475 + } + ] + } + partitions: + [ + { + name: VENDOR_TEST + variant: Unbuffered + size: 64 + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: false + bkout_type: false + items: + [ + { + name: SCRATCH + size: 56 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 0 + } + { + name: VENDOR_TEST_DIGEST + size: 8 + offset: 56 + ismubi: false + isdigest: true + inv_default: 5280432312436422266 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Vendor test partition. + This is reserved for manufacturing smoke checks. The OTP wrapper + control logic inside prim_otp is allowed to read/write to this + region. ECC uncorrectable errors seen on the functional prim_otp + interface will not lead to an alert for this partition. + Instead, such errors will be reported as correctable ECC errors. + ''' + absorb: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 0 + } + { + name: CREATOR_SW_CFG + variant: Unbuffered + absorb: true + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: CREATOR_SW_CFG_AST_CFG + size: 156 + desc: + ''' + AST configuration data. These values get blindly copied to + the AST CSRs during ROM execution. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 64 + } + { + name: CREATOR_SW_CFG_AST_INIT_EN + size: 4 + desc: + ''' + Controls whether or not the CREATOR_SW_CFG_AST_CFG values + get copied to the AST CSRs during ROM execution. + A value of kMultiBitBool4True enables copying; all other + values disable. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 220 + } + { + name: CREATOR_SW_CFG_ROM_EXT_SKU + size: 4 + desc: SKU identifier metadata. Unused by SiliconCreator software. + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 224 + } + { + name: CREATOR_SW_CFG_SIGVERIFY_SPX_EN + size: 4 + desc: + ''' + Controls whether or not SPHINCS+ signature verification will + be executed when the ROM attempts to boot the ROM_EXT. + A value of kSigverifySpxDisabledOtp disables SPHINCS+ + signature verification, while all other values enable it. + Note, SPHINCS+ signature verification is always disabled in + TEST_UNLOCKED* LC states. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 228 + } + { + name: CREATOR_SW_CFG_FLASH_DATA_DEFAULT_CFG + size: 4 + desc: + ''' + The default scrambling, ECC, and high endurance + configuration settings for flash data pages set during ROM + execution. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 232 + } + { + name: CREATOR_SW_CFG_FLASH_INFO_BOOT_DATA_CFG + size: 4 + desc: + ''' + The scrambling, ECC, and high endurance configuration + settings for the boot data flash info pages set during ROM + execution. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 236 + } + { + name: CREATOR_SW_CFG_FLASH_HW_INFO_CFG_OVERRIDE + size: 4 + desc: + ''' + Scrambling and ECC configuration overrides (set by the ROM) + for the Creator and OwnerSeed flash info pages (pages 1 and + 2 in partition 0) that are hardwired to the keygmr. By + default, scrambling and ECC on these flash info pages must + be enabled in order to successfully crank the keygmr (in the + ROM_EXT). However, values of kMultiBitBool4True for each + subfield in this field will disable this requirement. See + the HW_INFO_CFG_OVERRIDE flash_ctrl CSR for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 240 + } + { + name: CREATOR_SW_CFG_RNG_EN + size: 4 + desc: + ''' + Whether or not to enable use of hardware generated entropy + (from the entropy complex via EDN) in the `rnd_uint32` + function. A value of kHardenedBoolTrue enables the use of + hardware generated entropy, while all other values disable. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 244 + } + { + name: CREATOR_SW_CFG_JITTER_EN + size: 4 + desc: + ''' + Whether or not to enable clock jitter. A value of + kMultiBitBool4False disables, while all other values enable. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 248 + } + { + name: CREATOR_SW_CFG_RET_RAM_RESET_MASK + size: 4 + desc: + ''' + Reset reason mask used to initialize (by overwriting with + random data) retention SRAM during ROM execution. A value of + 0 only initializes retention SRAM on power-on-resets.See + rstmgr RESET_INFO CSR documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 252 + } + { + name: CREATOR_SW_CFG_MANUF_STATE + size: 4 + desc: + ''' + Manufacturing state binding field. For use by + SiliconCreators or SiliconOwners to bind ROM_EXT images to a + specific device or set of devices. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 256 + } + { + name: CREATOR_SW_CFG_ROM_EXEC_EN + size: 4 + desc: + ''' + Whether or not to enable execution of ROM. A value of 0 + disables, while all other values enable. This enables + provisioning flows to attach JTAG connections and halt the + CPU before the device has been fully provisioned. All SKUs + should set this field to a non-zero value. Provisioning + flows shall take care to program this field at the + appropriate time during. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 260 + } + { + name: CREATOR_SW_CFG_CPUCTRL + size: 4 + desc: + ''' + Value to write to the Ibex CPUCTRL CSR during ROM execution. + This field controls settings such as ICACHE enablement. See + Ibex documentation for more information. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 264 + } + { + name: CREATOR_SW_CFG_MIN_SEC_VER_ROM_EXT + size: 4 + desc: + ''' + Value of the min_security_version_rom_ext field of the + default boot data. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 268 + } + { + name: CREATOR_SW_CFG_MIN_SEC_VER_BL0 + size: 4 + desc: + ''' + Value of the min_security_version_bl0 field of the + default boot data. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 272 + } + { + name: CREATOR_SW_CFG_DEFAULT_BOOT_DATA_IN_PROD_EN + size: 4 + desc: + ''' + Whether or not to enable the default boot data in PROD and + PROD_END life cycle states. A value of kHardenedBoolTrue + enables, all other values disable. If left disabled, + provisioning flows are required to setup boot data pages + prior to enabling ROM execution. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 276 + } + { + name: CREATOR_SW_CFG_RMA_SPIN_EN + size: 4 + desc: + ''' + Whether or not to enable a busy-wait delay loop in the ROM, + when a specific SW strapping configuration is applied during + boot, to provide time to trigger an RMA lifecycle transition + over JTAG. A value of kHardenedBoolTrue enables, all other + values disable. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 280 + } + { + name: CREATOR_SW_CFG_RMA_SPIN_CYCLES + size: 4 + desc: + ''' + The number of Ibex clock cycles to spin for when waiting for + an RMA transition. Used in combination with the + CREATOR_SW_CFG_RMA_SPIN_EN field. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 284 + } + { + name: CREATOR_SW_CFG_RNG_REPCNT_THRESHOLDS + size: 4 + desc: + ''' + The repetition count health test thresholds to enable + entropy_src with during ROM execution. + This must be configured if CREATOR_SW_CFG_RNG_EN is true. + See entropy_src documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 288 + } + { + name: CREATOR_SW_CFG_RNG_REPCNTS_THRESHOLDS + size: 4 + desc: + ''' + The repetition count symbol health test thresholds to enable + entropy_src with during ROM execution. + This must be configured if CREATOR_SW_CFG_RNG_EN is true. + See entropy_src documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 292 + } + { + name: CREATOR_SW_CFG_RNG_ADAPTP_HI_THRESHOLDS + size: 4 + desc: + ''' + The adaptive proportion health test high thresholds to + enable entropy_src with during ROM execution. + This must be configured if CREATOR_SW_CFG_RNG_EN is true. + See entropy_src documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 296 + } + { + name: CREATOR_SW_CFG_RNG_ADAPTP_LO_THRESHOLDS + size: 4 + desc: + ''' + The adaptive proportion health test low thresholds to enable + entropy_src with during ROM execution. + This must be configured if CREATOR_SW_CFG_RNG_EN is true. + See entropy_src documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 300 + } + { + name: CREATOR_SW_CFG_RNG_BUCKET_THRESHOLDS + size: 4 + desc: + ''' + The bucket health test thresholds to enable entropy_src with + during ROM execution. + This must be configured if CREATOR_SW_CFG_RNG_EN is true. + See entropy_src documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 304 + } + { + name: CREATOR_SW_CFG_RNG_MARKOV_HI_THRESHOLDS + size: 4 + desc: + ''' + The Markov health test high thresholds to enable entropy_src + with during ROM execution. + This must be configured if CREATOR_SW_CFG_RNG_EN is true. + See entropy_src documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 308 + } + { + name: CREATOR_SW_CFG_RNG_MARKOV_LO_THRESHOLDS + size: 4 + desc: + ''' + The Markov health test low thresholds to enable entropy_src + with during ROM execution. + This must be configured if CREATOR_SW_CFG_RNG_EN is true. + See entropy_src documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 312 + } + { + name: CREATOR_SW_CFG_RNG_EXTHT_HI_THRESHOLDS + size: 4 + desc: + ''' + The external health test high thresholds to enable the + entropy_src with during ROM execution. + This must be configured if CREATOR_SW_CFG_RNG_EN is true. + See entropy_src documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 316 + } + { + name: CREATOR_SW_CFG_RNG_EXTHT_LO_THRESHOLDS + size: 4 + desc: + ''' + The external health test low thresholds to enable the + entropy_src with during ROM execution. + This must be configured if CREATOR_SW_CFG_RNG_EN is true. + See entropy_src documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 320 + } + { + name: CREATOR_SW_CFG_RNG_ALERT_THRESHOLD + size: 4 + desc: + ''' + The alert threshold to configure the entropy_src with during + ROM execution. + This must be configured if CREATOR_SW_CFG_RNG_EN is true. + See entropy_src documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 324 + } + { + name: CREATOR_SW_CFG_RNG_HEALTH_CONFIG_DIGEST + size: 4 + desc: + ''' + A CRC32 digest of all entropy_src health test threshold + configuration fields above. + This must be configured if CREATOR_SW_CFG_RNG_EN is true. + See entropy_src documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 328 + } + { + name: CREATOR_SW_CFG_SRAM_KEY_RENEW_EN + size: 4 + desc: + ''' + Whether or not the ROM should request SRAM to be rescrambled + with a new key on every boot. kHardenedBoolFalse disables, + while all other values enable. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 332 + } + { + name: CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_EN + size: 4 + desc: + ''' + Enablement of the ROM_EXT immutable code section. + A value of kHardenedBoolTrue enables the feature. + All other values disable it. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 336 + } + { + name: CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_START_OFFSET + size: 4 + desc: + ''' + Relative offset from the start of the ROM_EXT slot to + find the immutable code section at. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 340 + } + { + name: CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_LENGTH + size: 4 + desc: Length (in bytes) of the immutable code section. + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 344 + } + { + name: CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_SHA256_HASH + size: 32 + desc: SHA256 hash of the immutable ROM_EXT section. + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 348 + } + { + name: CREATOR_SW_CFG_RESERVED + size: 32 + desc: + ''' + Unused bits in the CREATOR_SW_CFG OTP partition. These can + be claimed by software as needed. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 380 + } + { + name: CREATOR_SW_CFG_DIGEST + size: 8 + offset: 424 + ismubi: false + isdigest: true + inv_default: 234369395612291906 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Software configuration partition. + This is for device-specific calibration data, e.g, clock, LDO, RNG, + and configuration settings set by the ROM. + ''' + iskeymgr_creator: false + iskeymgr_owner: false + size: 368 + offset: 64 + } + { + name: OWNER_SW_CFG + variant: Unbuffered + absorb: true + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: OWNER_SW_CFG_ROM_ERROR_REPORTING + size: 4 + desc: + ''' + The shutdown error reporting verbosity used by the ROM. + Should be configured to one of several + `shutdown_error_redact_t` values. See + `sw/device/silicon_creator/lib/shutdown.h` for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 432 + } + { + name: OWNER_SW_CFG_ROM_BOOTSTRAP_DIS + size: 4 + desc: + ''' + Whether or not to disable ROM bootstrap mechanism. A value + of kHardenedBoolTrue disable bootstrap mechanism in the ROM, + while all other values enable it. Note, the provisioning + flow should take care when to program this field if it is + used by a SKU, as there is no way to get firmware into flash + in a PROD LC state if a valid ROM_EXT does not already exist + in flash once this value is configured to true. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 436 + } + { + name: OWNER_SW_CFG_ROM_ALERT_CLASS_EN + size: 4 + desc: + ''' + A four byte packed field, where each byte controls whether + or not the ROM enables each alert class (A through D) of the + alert_handler. The byte-sized subfields are arranged from D + to A, MSB to LSB. Each byte should be set to an + `alert_enable_t` value accordingly. See the alert_handler + documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 440 + } + { + name: OWNER_SW_CFG_ROM_ALERT_ESCALATION + size: 4 + desc: + ''' + A four byte packed field, where each byte controls the + escalation configuration for each alert class (A through D) + of the alert_handler configured by the ROM. The byte-sized + subfields are arranged from D to A, MSB to LSB. Each byte + should be set to an `alert_escalate_t` value accordingly. + See the alert_handler documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 444 + } + { + name: OWNER_SW_CFG_ROM_ALERT_CLASSIFICATION + size: 320 + desc: + ''' + The alert classifications (A through D) for each alert + source of the alert_handler to be configured by the ROM. The + field consists of a contiguous 320-byte block, or 80 32-bit + words. The four bytes in each word encode the configuration + of a single alert source across four lifecycle states, in + order from LSB to MSB: PROD, PROD_END, DEV, and RMA. Each + byte should be set to an `alert_class_t` value accordingly. + The order of the 80 32-bit words, from LSB to MSB can be + found in the EARLGREY_ALERTS list in `rules/const.bzl`. + See the alert_handler documentation for more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 448 + } + { + name: OWNER_SW_CFG_ROM_LOCAL_ALERT_CLASSIFICATION + size: 64 + desc: + ''' + Same as the `OWNER_SW_CFG_ROM_ALERT_CLASSIFICATION` field, + except these configuration correspond to the local alert + sources found in the `EARLGREY_LOC_ALERTS list in + `rules/const.bzl`. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 768 + } + { + name: OWNER_SW_CFG_ROM_ALERT_ACCUM_THRESH + size: 16 + desc: + ''' + The alert accumulation threshold values for each alert class + (A through D) of the alert_handler to be configured by the + ROM. This field consists of four 32-bit words encoding the + accumulation thresholds for each alert class A through D + arranged LSW to MSW. See the alert_handler documentation for + more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 832 + } + { + name: OWNER_SW_CFG_ROM_ALERT_TIMEOUT_CYCLES + size: 16 + desc: + ''' + Same as the `OWNER_SW_CFG_ROM_ALERT_ACCUM_THRESH` field, + except each value corresponds to the interrupt timeout + configuration of an alert class. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 848 + } + { + name: OWNER_SW_CFG_ROM_ALERT_PHASE_CYCLES + size: 64 + desc: + ''' + The alert escalation phase durations, measured in clock + cycles, the ROM will configure the four alert phases for + each alert class of the alert_handler. This field consists + of a contiguous 64-byte block, or an array of four 128-bit + fields. Each 128-bit subfield encodes four 32-bit words that + contain the alert phase cycle count configurations for alert + escalation phases 0 to 3, from LSW to MSW. Each 128-bit + subfield is contains all cofigurations for a single alert + class, arranged from class A to D, from LS to MS. For + example, the cycle durations of each escalation phase in + this field should be configured as such, from LSB to MSB: + ......... + . See the alert_handler documentation for + more details. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 864 + } + { + name: OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD + size: 4 + desc: + ''' + The expected CRC32 digest over all of the alert_handler + configurations set up by the ROM for a device operating in + the PROD LC state. The ROM reads this field and checks it + against a digest it computes over the alert_handler + configuration it programmed. This field is expected to be + automatically computed by the `otp_alert_digest()` Bazel + rule. See the `alert_config_crc32()` function in the + SiliconCreator alert_handler driver for more details on what + configurations are included in this digest. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 928 + } + { + name: OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD_END + size: 4 + desc: + ''' + Same as the `OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD` field, + except the expected digest is for chips operating in the + PROD_END LC state. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 932 + } + { + name: OWNER_SW_CFG_ROM_ALERT_DIGEST_DEV + size: 4 + desc: + ''' + Same as the `OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD` field, + except the expected digest is for chips operating in the + DEV LC state. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 936 + } + { + name: OWNER_SW_CFG_ROM_ALERT_DIGEST_RMA + size: 4 + desc: + ''' + Same as the `OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD` field, + except the expected digest is for chips operating in the + RMA LC state. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 940 + } + { + name: OWNER_SW_CFG_ROM_WATCHDOG_BITE_THRESHOLD_CYCLES + size: 4 + desc: Watchdog timer bite threshold (in cycles) configured by the ROM. + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 944 + } + { + name: OWNER_SW_CFG_ROM_KEYMGR_OTP_MEAS_EN + size: 4 + desc: + ''' + Whether or not to configure the attestation SW binding CSRs + of the keymgr with the value in ROM_EXT manifest or the + measurement of the OTP CreatorSwCfg, OwnerSwCfg, and + secure boot key integrity digest. A value of + kHardenedBoolTrue uses the ROM computed OTP measurements, + while all other values trigger the use of the binding values + included in the ROM_EXT manifest. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 948 + } + { + name: OWNER_SW_CFG_MANUF_STATE + size: 4 + desc: + ''' + Manufacturing state binding field. For use by + SiliconCreators or SiliconOwners to bind BL0 images to a + specific device or set of devices. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 952 + } + { + name: OWNER_SW_CFG_ROM_RSTMGR_INFO_EN + size: 4 + desc: + ''' + A two byte packed word that indicates the expected rstmgr + alert and CPU info dump enable states, configured in the + rstmgr's ALERT_INFO_CTRL and CPU_INFO_CTRL CSRs respectively. + The expected format of this fields is + {0,0,kHardenedBool*,kHardenedBool*}, read MSB to LSB, where + the left most kHardenedBool* entry indicates the expected + enablement state of the ALERT_INFO_CTRL, and the right most + indicates the enablement state of the CPU_INFO_CTRL. Since + the ROM expects both to be disabled upon handing over + execution control to the ROM_EXT, this entire OTP field + **should be left unprovisioned, or set to all 0**. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 956 + } + { + name: OWNER_SW_CFG_ROM_EXT_BOOTSTRAP_EN + size: 4 + desc: Unused. Set to 0. + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 960 + } + { + name: OWNER_SW_CFG_ROM_SENSOR_CTRL_ALERT_CFG + size: 12 + desc: + ''' + Alert configuration values for the sensor_ctrl block + that will be configured by the ROM. This field is 12 bytes + long, where each byte contains two 4-bit packed subfields, + encoding two four bit boolean values (kMultiBitBool4False or + kMultiBitBool4True), as follows (read MSB to LSB): + {fatality, enablement}. For example, the byte 0x69 would + encode the alert is: 1) recoverable, and 2) disabled. + + Currently, there are only 11 alerts in sensor_ctrl to + configure, thus only the least significant bytes in this + field are used. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 964 + } + { + name: OWNER_SW_CFG_ROM_SRAM_READBACK_EN + size: 4 + desc: + ''' + This field encodes the enablements of the readback security + features for the main and retention SRAMs to be configured + by the ROM. This field is four bytes, but the LSB contains + two 4-bit packed kMultiBitBool4* values that indicate the + enablement of the feature for the retention SRAM and + main SRAM, from MSB to LSB respectively. See the READBACK + CSR of the sram_ctrl for more details on this feature. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 976 + } + { + name: OWNER_SW_CFG_ROM_PRESERVE_RESET_REASON_EN + size: 4 + desc: + ''' + Whether or not the ROM should preserve the reset reasons CSR + state in the rstmgr, or clear it. A value of + kHardenedBoolTrue preserves the CSR, while other values + trigger the ROM to clear the CSR after copying the reason to + the retention SRAM. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 980 + } + { + name: OWNER_SW_CFG_ROM_RESET_REASON_CHECK_VALUE + size: 4 + desc: + ''' + Two packed 16-bit values that indicate whether the ROM + should perform a validation check on the the reset reasons + during boot. The validation check is a hardening mechanism + that checks the reset reasons for consistency at two + different points in time. Both packed values should be the + same. Values of kHardenedBoolFalse will instruct the ROM to + skip the reset reasons validation check, while all other + values will instruct the ROM to perform the check. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 984 + } + { + name: OWNER_SW_CFG_ROM_BANNER_EN + size: 4 + desc: + ''' + Whether or not the ROM should print a banner message to the + console UART during boot. A value of kHardenedBoolFalse + disables the message printing, while all other values enable + it. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 988 + } + { + name: OWNER_SW_CFG_ROM_FLASH_ECC_EXC_HANDLER_EN + size: 4 + desc: + ''' + Whether or not the ROM should use the flash ECC exception + handler during execution. A value of kHardenedBoolTrue + allows the ROM to use an the exception handler that recovers + gracefully and continues the boot process if a flash ECC + error is encountered during verification of a specific + ROM_EXT slot. This enables the ROM to attempt booting the + next ROM_EXT slot if the first slot attempted has been + corrupted. All other values trigger default ROM exception + handling, which is all exceptions trigger a chip shutdown + and reset. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 992 + } + { + name: OWNER_SW_CFG_RESERVED + size: 128 + desc: + ''' + Unused bits in the OWNER_SW_CFG OTP partition. These can be + claimed by software as needed. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 996 + } + { + name: OWNER_SW_CFG_DIGEST + size: 8 + offset: 1136 + ismubi: false + isdigest: true + inv_default: 16327599381873551525 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Software configuration partition. + This contains data that changes software behavior in the ROM, for + example enabling defensive features in ROM or selecting failure + modes if verification fails. + ''' + iskeymgr_creator: false + iskeymgr_owner: false + size: 712 + offset: 432 + } + { + name: ROT_CREATOR_AUTH_CODESIGN + variant: Unbuffered + absorb: false + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE0 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1144 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY0 + size: 64 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1148 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE1 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1212 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY1 + size: 64 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1216 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE2 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1280 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY2 + size: 64 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1284 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE3 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1348 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY3 + size: 64 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1352 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE0 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1416 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_SPX_KEY0 + size: 32 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1420 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG0 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1452 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE1 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1456 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_SPX_KEY1 + size: 32 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1460 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG1 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1492 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE2 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1496 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_SPX_KEY2 + size: 32 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1500 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG2 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1532 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE3 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1536 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_SPX_KEY3 + size: 32 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1540 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG3 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1572 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_BLOCK_SHA2_256_HASH + size: 32 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1576 + } + { + name: ROT_CREATOR_AUTH_CODESIGN_DIGEST + size: 8 + offset: 1608 + ismubi: false + isdigest: true + inv_default: 1581155821231578393 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + This OTP partition is used to store four P-256 keys + and four Sphincs+ keys. The partition requires 464 + bytes of software visible storage. The partition is + locked at manufacturing time to protect against + malicious write attempts. + ''' + iskeymgr_creator: false + iskeymgr_owner: false + size: 472 + offset: 1144 + } + { + name: ROT_CREATOR_AUTH_STATE + variant: Unbuffered + absorb: false + secret: false + sw_digest: true + hw_digest: false + write_lock: Digest + read_lock: CSR + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: ROT_CREATOR_AUTH_STATE_ECDSA_KEY0 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1616 + } + { + name: ROT_CREATOR_AUTH_STATE_ECDSA_KEY1 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1620 + } + { + name: ROT_CREATOR_AUTH_STATE_ECDSA_KEY2 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1624 + } + { + name: ROT_CREATOR_AUTH_STATE_ECDSA_KEY3 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1628 + } + { + name: ROT_CREATOR_AUTH_STATE_SPX_KEY0 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1632 + } + { + name: ROT_CREATOR_AUTH_STATE_SPX_KEY1 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1636 + } + { + name: ROT_CREATOR_AUTH_STATE_SPX_KEY2 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1640 + } + { + name: ROT_CREATOR_AUTH_STATE_SPX_KEY3 + size: 4 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + inv_default: 0 + offset: 1644 + } + { + name: ROT_CREATOR_AUTH_STATE_DIGEST + size: 8 + offset: 1648 + ismubi: false + isdigest: true + inv_default: 2324999962355908533 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + This OTP partition is used to capture the state of + each key slot. Each key can be in one of the + following states: BLANK, ENABLED, DISABLED. The + encoded values are such that transitions between + BLANK -> ENABLED -> DISABLED are possible without + causing ECC errors (this is a mechanism similar to + how we manage life cycle state transitions). The + partition is left unlocked to allow STATE updates in + the field. The ROM_EXT is required to lock access to + the OTP Direct Access Interface to prevent DoS + attacks from malicious code executing on Silicon + Owner partitions. DAI write locking is available in + EarlGrey. + ''' + iskeymgr_creator: false + iskeymgr_owner: false + size: 40 + offset: 1616 + } + { + name: HW_CFG0 + variant: Buffered + secret: false + sw_digest: false + hw_digest: true + write_lock: Digest + read_lock: None + key_sel: NoKey + integrity: true + bkout_type: true + items: + [ + { + name: DEVICE_ID + size: 32 + inv_default: 45106338074792724304821242469256559359553488864622296623233734261107367140478 + desc: + ''' + Unique device identifier that is always exposed through the + lifecycle JTAG tap. See OpenTitan documentation on Device + Identifiers. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 1656 + } + { + name: MANUF_STATE + size: 32 + inv_default: 100965650985106319336595787448038909280859230372109746770436794677173954816422 + desc: + ''' + Field to capture manufacturing status. Currently unused. Set + to 0. + ''' + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 1688 + } + { + name: HW_CFG0_DIGEST + size: 8 + offset: 1720 + ismubi: false + isdigest: true + inv_default: 17905165971232536359 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Hardware configuration 0 partition. + This contains a device identifier and manufacturing state. + ''' + absorb: false + iskeymgr_creator: false + iskeymgr_owner: false + size: 72 + offset: 1656 + } + { + name: HW_CFG1 + variant: Buffered + secret: false + sw_digest: false + hw_digest: true + write_lock: Digest + read_lock: None + key_sel: NoKey + integrity: true + bkout_type: true + items: + [ + { + name: EN_SRAM_IFETCH + size: 1 + ismubi: true + inv_default: 105 + desc: + ''' + Enablement of execute from SRAM switch in the sram_ctrl (see + EXEC CSR). A kMultiBitBool8True value enables, while all + other values disable. + ''' + isdigest: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 1728 + } + { + name: EN_CSRNG_SW_APP_READ + size: 1 + ismubi: true + inv_default: 105 + desc: + ''' + Enablement of CSRNG software application interface. A + kMultiBitBool8True value enables, while all other values + disable. Enablement is required to extract output from + CSRNG via software. + ''' + isdigest: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 1729 + } + { + name: DIS_RV_DM_LATE_DEBUG + size: 1 + ismubi: true + inv_default: 105 + desc: + ''' + Disablement of RV_DM late debug feature (see rv_dm + documentation). A kMultiBitBool8True disables the late debug + feature and renders the rv_dm fully ungated in DEV lifecycle + states. All other values gate rv_dm reachability based on + the value of LATE_DEBUG_ENABLE CSR in the rv_dm block. + ''' + isdigest: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 1730 + } + { + name: HW_CFG1_DIGEST + size: 8 + offset: 1736 + ismubi: false + isdigest: true + inv_default: 13543634046774891762 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Hardware configuration 1 partition. + This contains several hardware feature switches. + ''' + absorb: false + iskeymgr_creator: false + iskeymgr_owner: false + size: 16 + offset: 1728 + } + { + name: SECRET0 + variant: Buffered + secret: true + sw_digest: false + hw_digest: true + write_lock: Digest + read_lock: Digest + key_sel: Secret0Key + integrity: true + bkout_type: false + items: + [ + { + name: TEST_UNLOCK_TOKEN + inv_default: 241483977691757977923367841809339206658 + size: 16 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 1744 + } + { + name: TEST_EXIT_TOKEN + inv_default: 150353733052794565317978902093985221175 + size: 16 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 1760 + } + { + name: SECRET0_DIGEST + size: 8 + offset: 1776 + ismubi: false + isdigest: true + inv_default: 13698041678974836896 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Secret partition 0. + This contains TEST lifecycle unlock tokens. + ''' + absorb: false + iskeymgr_creator: false + iskeymgr_owner: false + size: 40 + offset: 1744 + } + { + name: SECRET1 + variant: Buffered + secret: true + sw_digest: false + hw_digest: true + write_lock: Digest + read_lock: Digest + key_sel: Secret1Key + integrity: true + bkout_type: false + items: + [ + { + name: FLASH_ADDR_KEY_SEED + inv_default: 99410155969389534278404680059562745406179175357375916509714598543499986721841 + size: 32 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 1784 + } + { + name: FLASH_DATA_KEY_SEED + inv_default: 7391079586881058628634247621478375370202976073106929415694865999470879218122 + size: 32 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 1816 + } + { + name: SRAM_DATA_KEY_SEED + inv_default: 297822815773166572715338675789917702294 + size: 16 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 1848 + } + { + name: SECRET1_DIGEST + size: 8 + offset: 1864 + ismubi: false + isdigest: true + inv_default: 14153060542995762600 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Secret partition 1. + This contains SRAM and flash scrambling keys. + ''' + absorb: false + iskeymgr_creator: false + iskeymgr_owner: false + size: 88 + offset: 1784 + } + { + name: SECRET2 + variant: Buffered + secret: true + sw_digest: false + hw_digest: true + write_lock: Digest + read_lock: Digest + key_sel: Secret2Key + integrity: true + bkout_type: false + items: + [ + { + name: RMA_TOKEN + inv_default: 197791420282250691661260177080212266122 + size: 16 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 1872 + } + { + name: CREATOR_ROOT_KEY_SHARE0 + inv_default: 94411066183681302546622452158435937172067990320926238136744201607471434110567 + size: 32 + iskeymgr_creator: true + isdigest: false + ismubi: false + iskeymgr_owner: false + offset: 1888 + } + { + name: CREATOR_ROOT_KEY_SHARE1 + inv_default: 97043349937164487093045210023379701147105192659006254486994315071370236037145 + size: 32 + iskeymgr_creator: true + isdigest: false + ismubi: false + iskeymgr_owner: false + offset: 1920 + } + { + name: SECRET2_DIGEST + size: 8 + offset: 1952 + ismubi: false + isdigest: true + inv_default: 10140889213198706984 + iskeymgr_creator: false + iskeymgr_owner: false + } + ] + desc: + ''' + Secret partition 2. + This contains RMA unlock token, creator root key, and creator seed. + ''' + absorb: false + iskeymgr_creator: true + iskeymgr_owner: false + size: 88 + offset: 1872 + } + { + name: LIFE_CYCLE + variant: LifeCycle + secret: false + sw_digest: false + hw_digest: false + write_lock: None + read_lock: None + key_sel: NoKey + integrity: true + bkout_type: false + items: + [ + { + name: LC_TRANSITION_CNT + inv_default: 24752457638992396544754265753960069142084552974894085834358048399991475854089517042067373962426813673346557748670398 + size: 48 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 1960 + } + { + name: LC_STATE + inv_default: 1232459460109293830193445129566839459062644876142083683633340923439827006874097651236687620186710 + size: 40 + isdigest: false + ismubi: false + iskeymgr_creator: false + iskeymgr_owner: false + offset: 2008 + } + ] + desc: + ''' + Lifecycle partition. + This contains lifecycle transition count and state. This partition + cannot be locked since the life cycle state needs to advance to RMA + in-field. Note that while this partition is not marked secret, it + is not readable nor writeable via the DAI. Only the LC controller + can access this partition, and even via the LC controller it is not + possible to read the raw manufacturing life cycle state in encoded + form, since that encoding is considered a netlist secret. The LC + controller only exposes a decoded version of this state. + ''' + absorb: false + iskeymgr_creator: false + iskeymgr_owner: false + size: 88 + offset: 1960 + } + ] + } + topname: earlgrey + } +} diff --git a/hw/ip/otp_ctrl/defs.bzl b/hw/top_earlgrey/ip_autogen/otp_ctrl/defs.bzl similarity index 78% rename from hw/ip/otp_ctrl/defs.bzl rename to hw/top_earlgrey/ip_autogen/otp_ctrl/defs.bzl index 0f76f2989d30a7..4ec7f83d574aa2 100644 --- a/hw/ip/otp_ctrl/defs.bzl +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/defs.bzl @@ -5,5 +5,5 @@ load("//rules/opentitan:hw.bzl", "opentitan_ip") OTP_CTRL = opentitan_ip( name = "otp_ctrl", - hjson = "//hw/ip/otp_ctrl/data:otp_ctrl.hjson", + hjson = "//hw/top_earlgrey/ip_autogen/otp_ctrl:data/otp_ctrl.hjson", ) diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/checklist.md b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/checklist.md new file mode 100644 index 00000000000000..117db7dd8d60bd --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/checklist.md @@ -0,0 +1,271 @@ +# OTP_CTRL Checklist + + +This checklist is for [Hardware Stage](../../../../../doc/project_governance/development_stages.md) transitions for the [OTP_CTRL peripheral.](../README.md) +All checklist items refer to the content in the [Checklist.](../../../../../doc/project_governance/checklist/README.md) + +## Design Checklist + +### D1 + +Type | Item | Resolution | Note/Collaterals +--------------|--------------------------------|-------------|------------------ +Documentation | [SPEC_COMPLETE][] | Done | [OTP_CTRL Design Spec](../README.md) +Documentation | [CSR_DEFINED][] | Done | +RTL | [CLKRST_CONNECTED][] | Done | +RTL | [IP_TOP][] | Done | +RTL | [IP_INSTANTIABLE][] | Done | +RTL | [PHYSICAL_MACROS_DEFINED_80][] | Done | +RTL | [FUNC_IMPLEMENTED][] | Done | +RTL | [ASSERT_KNOWN_ADDED][] | Done | +Code Quality | [LINT_SETUP][] | Done | + +[SPEC_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#spec_complete +[CSR_DEFINED]: ../../../../../doc/project_governance/checklist/README.md#csr_defined +[CLKRST_CONNECTED]: ../../../../../doc/project_governance/checklist/README.md#clkrst_connected +[IP_TOP]: ../../../../../doc/project_governance/checklist/README.md#ip_top +[IP_INSTANTIABLE]: ../../../../../doc/project_governance/checklist/README.md#ip_instantiable +[PHYSICAL_MACROS_DEFINED_80]: ../../../../../doc/project_governance/checklist/README.md#mem_instanced_80 +[FUNC_IMPLEMENTED]: ../../../../../doc/project_governance/checklist/README.md#func_implemented +[ASSERT_KNOWN_ADDED]: ../../../../../doc/project_governance/checklist/README.md#assert_known_added +[LINT_SETUP]: ../../../../../doc/project_governance/checklist/README.md#lint_setup + +### D2 + +Type | Item | Resolution | Note/Collaterals +--------------|---------------------------|-------------|------------------ +Documentation | [NEW_FEATURES][] | Done | +Documentation | [BLOCK_DIAGRAM][] | Done | +Documentation | [DOC_INTERFACE][] | Done | +Documentation | [DOC_INTEGRATION_GUIDE][] | Waived | This checklist item has been added retrospectively. +Documentation | [MISSING_FUNC][] | Done | +Documentation | [FEATURE_FROZEN][] | Done | +RTL | [FEATURE_COMPLETE][] | Done | +RTL | [PORT_FROZEN][] | Done | +RTL | [ARCHITECTURE_FROZEN][] | Done | +RTL | [REVIEW_TODO][] | Done | +RTL | [STYLE_X][] | Done | +RTL | [CDC_SYNCMACRO][] | N/A | +Code Quality | [LINT_PASS][] | Done | +Code Quality | [CDC_SETUP][] | Waived | No block-level flow available - waived to top-level signoff. +Code Quality | [RDC_SETUP][] | Waived | No block-level flow available - waived to top-level signoff. +Code Quality | [AREA_CHECK][] | Done | +Code Quality | [TIMING_CHECK][] | Done | +Security | [SEC_CM_DOCUMENTED][] | Done | + +[NEW_FEATURES]: ../../../../../doc/project_governance/checklist/README.md#new_features +[BLOCK_DIAGRAM]: ../../../../../doc/project_governance/checklist/README.md#block_diagram +[DOC_INTERFACE]: ../../../../../doc/project_governance/checklist/README.md#doc_interface +[DOC_INTEGRATION_GUIDE]: ../../../../../doc/project_governance/checklist/README.md#doc_integration_guide +[MISSING_FUNC]: ../../../../../doc/project_governance/checklist/README.md#missing_func +[FEATURE_FROZEN]: ../../../../../doc/project_governance/checklist/README.md#feature_frozen +[FEATURE_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#feature_complete +[PORT_FROZEN]: ../../../../../doc/project_governance/checklist/README.md#port_frozen +[ARCHITECTURE_FROZEN]: ../../../../../doc/project_governance/checklist/README.md#architecture_frozen +[REVIEW_TODO]: ../../../../../doc/project_governance/checklist/README.md#review_todo +[STYLE_X]: ../../../../../doc/project_governance/checklist/README.md#style_x +[CDC_SYNCMACRO]: ../../../../../doc/project_governance/checklist/README.md#cdc_syncmacro +[LINT_PASS]: ../../../../../doc/project_governance/checklist/README.md#lint_pass +[CDC_SETUP]: ../../../../../doc/project_governance/checklist/README.md#cdc_setup +[RDC_SETUP]: ../../../../../doc/project_governance/checklist/README.md#rdc_setup +[AREA_CHECK]: ../../../../../doc/project_governance/checklist/README.md#area_check +[TIMING_CHECK]: ../../../../../doc/project_governance/checklist/README.md#timing_check +[SEC_CM_DOCUMENTED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_documented + +### D2S + + Type | Item | Resolution | Note/Collaterals +--------------|------------------------------|-------------|------------------ +Security | [SEC_CM_ASSETS_LISTED][] | Done | +Security | [SEC_CM_IMPLEMENTED][] | Done | +Security | [SEC_CM_RND_CNST][] | Done | +Security | [SEC_CM_NON_RESET_FLOPS][] | Done | +Security | [SEC_CM_SHADOW_REGS][] | Done | +Security | [SEC_CM_RTL_REVIEWED][] | Done | +Security | [SEC_CM_COUNCIL_REVIEWED][] | Done | + +[SEC_CM_ASSETS_LISTED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_assets_listed +[SEC_CM_IMPLEMENTED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_implemented +[SEC_CM_RND_CNST]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_rnd_cnst +[SEC_CM_NON_RESET_FLOPS]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_non_reset_flops +[SEC_CM_SHADOW_REGS]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_shadow_regs +[SEC_CM_RTL_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_rtl_reviewed +[SEC_CM_COUNCIL_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_council_reviewed + +### D3 + + Type | Item | Resolution | Note/Collaterals +--------------|-------------------------|-------------|------------------ +Documentation | [NEW_FEATURES_D3][] | Done | +RTL | [TODO_COMPLETE][] | Done | +Code Quality | [LINT_COMPLETE][] | Done | Waiver files approved by TC on 2024-08-08. +Code Quality | [CDC_COMPLETE][] | Waived | No block-level flow available - waived to top-level signoff. +Code Quality | [RDC_COMPLETE][] | Waived | No block-level flow available - waived to top-level signoff. +Review | [REVIEW_RTL][] | Done | +Review | [REVIEW_DELETED_FF][] | Waived | No block-level flow available - waived to top-level signoff. +Review | [REVIEW_SW_CHANGE][] | Done | +Review | [REVIEW_SW_ERRATA][] | Done | +Review | Reviewer(s) | Done | adk@ vogelpi@ +Review | Signoff date | Done | 2024-08-08 + +[NEW_FEATURES_D3]: ../../../../../doc/project_governance/checklist/README.md#new_features_d3 +[TODO_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#todo_complete +[LINT_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#lint_complete +[CDC_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#cdc_complete +[RDC_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#rdc_complete +[REVIEW_RTL]: ../../../../../doc/project_governance/checklist/README.md#review_rtl +[REVIEW_DELETED_FF]: ../../../../../doc/project_governance/checklist/README.md#review_deleted_ff +[REVIEW_SW_CHANGE]: ../../../../../doc/project_governance/checklist/README.md#review_sw_change +[REVIEW_SW_ERRATA]: ../../../../../doc/project_governance/checklist/README.md#review_sw_errata + +## Verification Checklist + +### V1 + + Type | Item | Resolution | Note/Collaterals +--------------|---------------------------------------|-------------|------------------ +Documentation | [DV_DOC_DRAFT_COMPLETED][] | Done | [OTP_CTRL DV document](../dv/README.md) +Documentation | [TESTPLAN_COMPLETED][] | Done | [OTP_CTRL Testplan](../dv/README.md#testplan) +Testbench | [TB_TOP_CREATED][] | Done | +Testbench | [PRELIMINARY_ASSERTION_CHECKS_ADDED][]| Done | +Testbench | [SIM_TB_ENV_CREATED][] | Done | +Testbench | [SIM_RAL_MODEL_GEN_AUTOMATED][] | Done | +Testbench | [CSR_CHECK_GEN_AUTOMATED][] | Done | +Testbench | [TB_GEN_AUTOMATED][] | Done | +Tests | [SIM_SMOKE_TEST_PASSING][] | Done | +Tests | [SIM_CSR_MEM_TEST_SUITE_PASSING][] | Done | +Tests | [FPV_MAIN_ASSERTIONS_PROVEN][] | N/A | +Tool Setup | [SIM_ALT_TOOL_SETUP][] | Done | +Regression | [SIM_SMOKE_REGRESSION_SETUP][] | Done | +Regression | [SIM_NIGHTLY_REGRESSION_SETUP][] | Done | +Regression | [FPV_REGRESSION_SETUP][] | N/A | +Coverage | [SIM_COVERAGE_MODEL_ADDED][] | Done | +Code Quality | [TB_LINT_SETUP][] | Done | +Integration | [PRE_VERIFIED_SUB_MODULES_V1][] | N/A | Exception for IP modules +Review | [DESIGN_SPEC_REVIEWED][] | Done | +Review | [TESTPLAN_REVIEWED][] | Done | +Review | [STD_TEST_CATEGORIES_PLANNED][] | Done | Exception (Security, Power, Debug) +Review | [V2_CHECKLIST_SCOPED][] | Done | + +[DV_DOC_DRAFT_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#dv_doc_draft_completed +[TESTPLAN_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#testplan_completed +[TB_TOP_CREATED]: ../../../../../doc/project_governance/checklist/README.md#tb_top_created +[PRELIMINARY_ASSERTION_CHECKS_ADDED]: ../../../../../doc/project_governance/checklist/README.md#preliminary_assertion_checks_added +[SIM_TB_ENV_CREATED]: ../../../../../doc/project_governance/checklist/README.md#sim_tb_env_created +[SIM_RAL_MODEL_GEN_AUTOMATED]: ../../../../../doc/project_governance/checklist/README.md#sim_ral_model_gen_automated +[CSR_CHECK_GEN_AUTOMATED]: ../../../../../doc/project_governance/checklist/README.md#csr_check_gen_automated +[TB_GEN_AUTOMATED]: ../../../../../doc/project_governance/checklist/README.md#tb_gen_automated +[SIM_SMOKE_TEST_PASSING]: ../../../../../doc/project_governance/checklist/README.md#sim_smoke_test_passing +[SIM_CSR_MEM_TEST_SUITE_PASSING]: ../../../../../doc/project_governance/checklist/README.md#sim_csr_mem_test_suite_passing +[FPV_MAIN_ASSERTIONS_PROVEN]: ../../../../../doc/project_governance/checklist/README.md#fpv_main_assertions_proven +[SIM_ALT_TOOL_SETUP]: ../../../../../doc/project_governance/checklist/README.md#sim_alt_tool_setup +[SIM_SMOKE_REGRESSION_SETUP]: ../../../../../doc/project_governance/checklist/README.md#sim_smoke_regression_setup +[SIM_NIGHTLY_REGRESSION_SETUP]: ../../../../../doc/project_governance/checklist/README.md#sim_nightly_regression_setup +[FPV_REGRESSION_SETUP]: ../../../../../doc/project_governance/checklist/README.md#fpv_regression_setup +[SIM_COVERAGE_MODEL_ADDED]: ../../../../../doc/project_governance/checklist/README.md#sim_coverage_model_added +[PRE_VERIFIED_SUB_MODULES_V1]: ../../../../../doc/project_governance/checklist/README.md#pre_verified_sub_modules_v1 +[TB_LINT_SETUP]: ../../../../../doc/project_governance/checklist/README.md#tb_lint_setup +[DESIGN_SPEC_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#design_spec_reviewed +[TESTPLAN_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#testplan_reviewed +[STD_TEST_CATEGORIES_PLANNED]: ../../../../../doc/project_governance/checklist/README.md#std_test_categories_planned +[V2_CHECKLIST_SCOPED]: ../../../../../doc/project_governance/checklist/README.md#v2_checklist_scoped + +### V2 + + Type | Item | Resolution | Note/Collaterals +--------------|-----------------------------------------|-------------|------------------ +Documentation | [DESIGN_DELTAS_CAPTURED_V2][] | Done | +Documentation | [DV_DOC_COMPLETED][] | Done | +Testbench | [FUNCTIONAL_COVERAGE_IMPLEMENTED][] | Done | +Testbench | [ALL_INTERFACES_EXERCISED][] | Done | `prim_tl_o/i` has a simple prim_tl_agent support, but need to be replaced with auto-generated tl_agent once reggen tool is optimized. +Testbench | [ALL_ASSERTION_CHECKS_ADDED][] | Done | +Testbench | [SIM_TB_ENV_COMPLETED][] | Done | +Tests | [SIM_ALL_TESTS_PASSING][] | Done | +Tests | [FPV_ALL_ASSERTIONS_WRITTEN][] | N/A | +Tests | [FPV_ALL_ASSUMPTIONS_REVIEWED][] | N/A | +Tests | [SIM_FW_SIMULATED][] | N/A | +Regression | [SIM_NIGHTLY_REGRESSION_V2][] | Done | +Coverage | [SIM_CODE_COVERAGE_V2][] | Done | +Coverage | [SIM_FUNCTIONAL_COVERAGE_V2][] | Done | +Coverage | [FPV_CODE_COVERAGE_V2][] | N/A | +Coverage | [FPV_COI_COVERAGE_V2][] | N/A | +Integration | [PRE_VERIFIED_SUB_MODULES_V2][] | Done | Waived `prim_alert_sender` and `prim_lfsr` +Issues | [NO_HIGH_PRIORITY_ISSUES_PENDING][] | Done | +Issues | [ALL_LOW_PRIORITY_ISSUES_ROOT_CAUSED][] | Done | +Review | [DV_DOC_TESTPLAN_REVIEWED][] | Done | Reviewed on 05/24/2021 +Review | [V3_CHECKLIST_SCOPED][] | Done | + +[DESIGN_DELTAS_CAPTURED_V2]: ../../../../../doc/project_governance/checklist/README.md#design_deltas_captured_v2 +[DV_DOC_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#dv_doc_completed +[FUNCTIONAL_COVERAGE_IMPLEMENTED]: ../../../../../doc/project_governance/checklist/README.md#functional_coverage_implemented +[ALL_INTERFACES_EXERCISED]: ../../../../../doc/project_governance/checklist/README.md#all_interfaces_exercised +[ALL_ASSERTION_CHECKS_ADDED]: ../../../../../doc/project_governance/checklist/README.md#all_assertion_checks_added +[SIM_TB_ENV_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#sim_tb_env_completed +[SIM_ALL_TESTS_PASSING]: ../../../../../doc/project_governance/checklist/README.md#sim_all_tests_passing +[FPV_ALL_ASSERTIONS_WRITTEN]: ../../../../../doc/project_governance/checklist/README.md#fpv_all_assertions_written +[FPV_ALL_ASSUMPTIONS_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#fpv_all_assumptions_reviewed +[SIM_FW_SIMULATED]: ../../../../../doc/project_governance/checklist/README.md#sim_fw_simulated +[SIM_NIGHTLY_REGRESSION_V2]: ../../../../../doc/project_governance/checklist/README.md#sim_nightly_regression_v2 +[SIM_CODE_COVERAGE_V2]: ../../../../../doc/project_governance/checklist/README.md#sim_code_coverage_v2 +[SIM_FUNCTIONAL_COVERAGE_V2]: ../../../../../doc/project_governance/checklist/README.md#sim_functional_coverage_v2 +[FPV_CODE_COVERAGE_V2]: ../../../../../doc/project_governance/checklist/README.md#fpv_code_coverage_v2 +[FPV_COI_COVERAGE_V2]: ../../../../../doc/project_governance/checklist/README.md#fpv_coi_coverage_v2 +[PRE_VERIFIED_SUB_MODULES_V2]: ../../../../../doc/project_governance/checklist/README.md#pre_verified_sub_modules_v2 +[NO_HIGH_PRIORITY_ISSUES_PENDING]: ../../../../../doc/project_governance/checklist/README.md#no_high_priority_issues_pending +[ALL_LOW_PRIORITY_ISSUES_ROOT_CAUSED]:../../../../../doc/project_governance/checklist/README.md#all_low_priority_issues_root_caused +[DV_DOC_TESTPLAN_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#dv_doc_testplan_reviewed +[V3_CHECKLIST_SCOPED]: ../../../../../doc/project_governance/checklist/README.md#v3_checklist_scoped + +### V2S + + Type | Item | Resolution | Note/Collaterals +--------------|-----------------------------------------|-------------|------------------ +Documentation | [SEC_CM_TESTPLAN_COMPLETED][] | Done | +Tests | [FPV_SEC_CM_VERIFIED][] | Done | +Tests | [SIM_SEC_CM_VERIFIED][] | Done | +Coverage | [SIM_COVERAGE_REVIEWED][] | Done | +Review | [SEC_CM_DV_REVIEWED][] | Done | + +[SEC_CM_TESTPLAN_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_testplan_completed +[FPV_SEC_CM_VERIFIED]: ../../../../../doc/project_governance/checklist/README.md#fpv_sec_cm_verified +[SIM_SEC_CM_VERIFIED]: ../../../../../doc/project_governance/checklist/README.md#sim_sec_cm_verified +[SIM_COVERAGE_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#sim_coverage_reviewed +[SEC_CM_DV_REVIEWED]: ../../../../../doc/project_governance/checklist/README.md#sec_cm_dv_reviewed + +### V3 + + Type | Item | Resolution | Note/Collaterals +--------------|-----------------------------------|-------------|------------------ +Documentation | [DESIGN_DELTAS_CAPTURED_V3][] | Not Started | +Tests | [X_PROP_ANALYSIS_COMPLETED][] | Not Started | +Tests | [FPV_ASSERTIONS_PROVEN_AT_V3][] | Not Started | +Regression | [SIM_NIGHTLY_REGRESSION_AT_V3][] | Not Started | +Coverage | [SIM_CODE_COVERAGE_AT_100][] | Not Started | +Coverage | [SIM_FUNCTIONAL_COVERAGE_AT_100][]| Not Started | +Coverage | [FPV_CODE_COVERAGE_AT_100][] | Not Started | +Coverage | [FPV_COI_COVERAGE_AT_100][] | Not Started | +Code Quality | [ALL_TODOS_RESOLVED][] | Not Started | +Code Quality | [NO_TOOL_WARNINGS_THROWN][] | Not Started | +Code Quality | [TB_LINT_COMPLETE][] | Not Started | +Integration | [PRE_VERIFIED_SUB_MODULES_V3][] | Not Started | +Issues | [NO_ISSUES_PENDING][] | Not Started | +Review | Reviewer(s) | Not Started | +Review | Signoff date | Not Started | + +[DESIGN_DELTAS_CAPTURED_V3]: ../../../../../doc/project_governance/checklist/README.md#design_deltas_captured_v3 +[X_PROP_ANALYSIS_COMPLETED]: ../../../../../doc/project_governance/checklist/README.md#x_prop_analysis_completed +[FPV_ASSERTIONS_PROVEN_AT_V3]: ../../../../../doc/project_governance/checklist/README.md#fpv_assertions_proven_at_v3 +[SIM_NIGHTLY_REGRESSION_AT_V3]: ../../../../../doc/project_governance/checklist/README.md#sim_nightly_regression_at_v3 +[SIM_CODE_COVERAGE_AT_100]: ../../../../../doc/project_governance/checklist/README.md#sim_code_coverage_at_100 +[SIM_FUNCTIONAL_COVERAGE_AT_100]:../../../../../doc/project_governance/checklist/README.md#sim_functional_coverage_at_100 +[FPV_CODE_COVERAGE_AT_100]: ../../../../../doc/project_governance/checklist/README.md#fpv_code_coverage_at_100 +[FPV_COI_COVERAGE_AT_100]: ../../../../../doc/project_governance/checklist/README.md#fpv_coi_coverage_at_100 +[ALL_TODOS_RESOLVED]: ../../../../../doc/project_governance/checklist/README.md#all_todos_resolved +[NO_TOOL_WARNINGS_THROWN]: ../../../../../doc/project_governance/checklist/README.md#no_tool_warnings_thrown +[TB_LINT_COMPLETE]: ../../../../../doc/project_governance/checklist/README.md#tb_lint_complete +[PRE_VERIFIED_SUB_MODULES_V3]: ../../../../../doc/project_governance/checklist/README.md#pre_verified_sub_modules_v3 +[NO_ISSUES_PENDING]: ../../../../../doc/project_governance/checklist/README.md#no_issues_pending diff --git a/hw/ip/otp_ctrl/doc/interfaces.md b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/interfaces.md similarity index 98% rename from hw/ip/otp_ctrl/doc/interfaces.md rename to hw/top_earlgrey/ip_autogen/otp_ctrl/doc/interfaces.md index 188b26c539ed23..6faed3afb80bc9 100644 --- a/hw/ip/otp_ctrl/doc/interfaces.md +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/interfaces.md @@ -13,11 +13,9 @@ Parameter | Default (Max) | Top Earlgrey | Description `RndCnstKey` | (see RTL) | (see RTL) | Random scrambling keys for secret partitions, to be used in the [scrambling datapath](#scrambling-datapath). `RndCnstDigestConst` | (see RTL) | (see RTL) | Random digest finalization constants, to be used in the [scrambling datapath](#scrambling-datapath). `RndCnstDigestIV` | (see RTL) | (see RTL) | Random digest initialization vectors, to be used in the [scrambling datapath](#scrambling-datapath). -`RndCnstRawUnlockToken` | (see RTL) | (see RTL) | Global RAW unlock token to be used for the first life cycle transition. See also [conditional life cycle transitions](../../lc_ctrl/README.md#conditional-transitions). +`RndCnstRawUnlockToken` | (see RTL) | (see RTL) | Global RAW unlock token to be used for the first life cycle transition. See also [conditional life cycle transitions](../../../../ip/lc_ctrl/README.md#conditional-transitions). -## Signals - - + Referring to the [Comportable guideline for peripheral device functionality](https://opentitan.org/book/doc/contributing/hw/comportability), the module **`otp_ctrl`** has the following hardware interfaces defined - Primary Clock: **`clk_i`** - Other Clocks: **`clk_edn_i`** @@ -136,7 +134,7 @@ The OTP controller contains various interfaces that connect to other comportable The entropy request interface that talks to EDN in order to fetch fresh entropy for ephemeral SRAM scrambling key derivation and the LFSR counters for background checks. It is comprised of the `otp_edn_o` and `otp_edn_i` signals and follows a req / ack protocol. -See also [EDN documentation](../../edn/README.md). +See also [EDN documentation](../../../../ip/edn/README.md). ### Power Manager Interface @@ -149,13 +147,13 @@ The idle indication signal `pwr_otp_o.otp_idle` indicates whether there is an on Since the power manager may run in a different clock domain, the `pwr_otp_i.otp_init` signal is synchronized within the OTP controller. The power manager is responsible for synchronizing the `pwr_otp_o.otp_done` and `pwr_otp_o.otp_idle` signals. -See also [power manager documentation](../../../top_earlgrey/ip_autogen/pwrmgr/README.md). +See also [power manager documentation](../../pwrmgr/README.md). ### Life Cycle Interfaces The interface to the life cycle controller can be split into three functional sub-interfaces (vendor test, state output, state transitions), and these are explained in more detail below. Note that the OTP and life cycle controllers are supposed to be in the same clock domain, hence no additional signal synchronization is required. -See also [life cycle controller documentation](../../lc_ctrl/README.md) for more details. +See also [life cycle controller documentation](../../../../ip/lc_ctrl/README.md) for more details. #### Vendor Test Signals @@ -218,7 +216,7 @@ The request must remain asserted until the life cycle controller has responded. An error is fatal and indicates that the OTP programming operation has failed. Note that the new state must not clear any bits that have already been programmed to OTP - i.e., the new state must be incrementally programmable on top of the previous state. -There are hence some implications on the life cycle encoding due to the ECC employed, see [life cycle state encoding](../../lc_ctrl/README.md#life-cycle-manufacturing-state-encodings) for details. +There are hence some implications on the life cycle encoding due to the ECC employed, see [life cycle state encoding](../../../../ip/lc_ctrl/README.md#life-cycle-manufacturing-state-encodings) for details. Note that the behavior of the `lc_otp_program_i.otp_test_ctrl` signal is vendor-specific, and hence the signal is set to `x` in the timing diagram above. The purpose of this signal is to control vendor-specific test mechanisms, and its value will only be forwarded to the OTP macro in RAW, TEST_* and RMA states. diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_behavioral_model.svg b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_behavioral_model.svg new file mode 100644 index 00000000000000..a5dab4a4bfc212 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_behavioral_model.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_blockdiag.svg b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_blockdiag.svg new file mode 100644 index 00000000000000..f2169883ce4b2b --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_blockdiag.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg new file mode 100644 index 00000000000000..d06037468e3c21 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_buf_part_fsm.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_dai_fsm.svg b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_dai_fsm.svg new file mode 100644 index 00000000000000..f84a80cc0f0f0b --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_dai_fsm.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_digest_mechanism.svg b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_digest_mechanism.svg new file mode 100644 index 00000000000000..dac2323afce333 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_digest_mechanism.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_digests.md b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_digests.md new file mode 100644 index 00000000000000..ddb4029750b9d6 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_digests.md @@ -0,0 +1,17 @@ + + +| Digest Name | Affected Partition | Calculated by HW | +|:---------------------------------------------------------------------------:|:-------------------------:|:------------------:| +| [VENDOR_TEST_DIGEST](#Reg_vendor_test_digest_0) | VENDOR_TEST | no | +| [CREATOR_SW_CFG_DIGEST](#Reg_creator_sw_cfg_digest_0) | CREATOR_SW_CFG | no | +| [OWNER_SW_CFG_DIGEST](#Reg_owner_sw_cfg_digest_0) | OWNER_SW_CFG | no | +| [ROT_CREATOR_AUTH_CODESIGN_DIGEST](#Reg_rot_creator_auth_codesign_digest_0) | ROT_CREATOR_AUTH_CODESIGN | no | +| [ROT_CREATOR_AUTH_STATE_DIGEST](#Reg_rot_creator_auth_state_digest_0) | ROT_CREATOR_AUTH_STATE | no | +| [HW_CFG0_DIGEST](#Reg_hw_cfg0_digest_0) | HW_CFG0 | yes | +| [HW_CFG1_DIGEST](#Reg_hw_cfg1_digest_0) | HW_CFG1 | yes | +| [SECRET0_DIGEST](#Reg_secret0_digest_0) | SECRET0 | yes | +| [SECRET1_DIGEST](#Reg_secret1_digest_0) | SECRET1 | yes | +| [SECRET2_DIGEST](#Reg_secret2_digest_0) | SECRET2 | yes | diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_field_descriptions.md b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_field_descriptions.md new file mode 100644 index 00000000000000..700506ae7c1b75 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_field_descriptions.md @@ -0,0 +1,101 @@ + + +| Partition | Item | Size [B] | Description | +|:-------------------------:|:-----------------------------------------------:|:----------:|:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| +| CREATOR_SW_CFG | CREATOR_SW_CFG_AST_CFG | 156 | AST configuration data. These values get blindly copied to the AST CSRs during ROM execution. | +| | CREATOR_SW_CFG_AST_INIT_EN | 4 | Controls whether or not the CREATOR_SW_CFG_AST_CFG values get copied to the AST CSRs during ROM execution. A value of kMultiBitBool4True enables copying; all other values disable. | +| | CREATOR_SW_CFG_ROM_EXT_SKU | 4 | SKU identifier metadata. Unused by SiliconCreator software. | +| | CREATOR_SW_CFG_SIGVERIFY_SPX_EN | 4 | Controls whether or not SPHINCS+ signature verification will be executed when the ROM attempts to boot the ROM_EXT. A value of kSigverifySpxDisabledOtp disables SPHINCS+ signature verification, while all other values enable it. Note, SPHINCS+ signature verification is always disabled in TEST_UNLOCKED* LC states. | +| | CREATOR_SW_CFG_FLASH_DATA_DEFAULT_CFG | 4 | The default scrambling, ECC, and high endurance configuration settings for flash data pages set during ROM execution. | +| | CREATOR_SW_CFG_FLASH_INFO_BOOT_DATA_CFG | 4 | The scrambling, ECC, and high endurance configuration settings for the boot data flash info pages set during ROM execution. | +| | CREATOR_SW_CFG_FLASH_HW_INFO_CFG_OVERRIDE | 4 | Scrambling and ECC configuration overrides (set by the ROM) for the Creator and OwnerSeed flash info pages (pages 1 and 2 in partition 0) that are hardwired to the keygmr. By default, scrambling and ECC on these flash info pages must be enabled in order to successfully crank the keygmr (in the ROM_EXT). However, values of kMultiBitBool4True for each subfield in this field will disable this requirement. See the HW_INFO_CFG_OVERRIDE flash_ctrl CSR for more details. | +| | CREATOR_SW_CFG_RNG_EN | 4 | Whether or not to enable use of hardware generated entropy (from the entropy complex via EDN) in the `rnd_uint32` function. A value of kHardenedBoolTrue enables the use of hardware generated entropy, while all other values disable. | +| | CREATOR_SW_CFG_JITTER_EN | 4 | Whether or not to enable clock jitter. A value of kMultiBitBool4False disables, while all other values enable. | +| | CREATOR_SW_CFG_RET_RAM_RESET_MASK | 4 | Reset reason mask used to initialize (by overwriting with random data) retention SRAM during ROM execution. A value of 0 only initializes retention SRAM on power-on-resets.See rstmgr RESET_INFO CSR documentation for more details. | +| | CREATOR_SW_CFG_MANUF_STATE | 4 | Manufacturing state binding field. For use by SiliconCreators or SiliconOwners to bind ROM_EXT images to a specific device or set of devices. | +| | CREATOR_SW_CFG_ROM_EXEC_EN | 4 | Whether or not to enable execution of ROM. A value of 0 disables, while all other values enable. This enables provisioning flows to attach JTAG connections and halt the CPU before the device has been fully provisioned. All SKUs should set this field to a non-zero value. Provisioning flows shall take care to program this field at the appropriate time during. | +| | CREATOR_SW_CFG_CPUCTRL | 4 | Value to write to the Ibex CPUCTRL CSR during ROM execution. This field controls settings such as ICACHE enablement. See Ibex documentation for more information. | +| | CREATOR_SW_CFG_MIN_SEC_VER_ROM_EXT | 4 | Value of the min_security_version_rom_ext field of the default boot data. | +| | CREATOR_SW_CFG_MIN_SEC_VER_BL0 | 4 | Value of the min_security_version_bl0 field of the default boot data. | +| | CREATOR_SW_CFG_DEFAULT_BOOT_DATA_IN_PROD_EN | 4 | Whether or not to enable the default boot data in PROD and PROD_END life cycle states. A value of kHardenedBoolTrue enables, all other values disable. If left disabled, provisioning flows are required to setup boot data pages prior to enabling ROM execution. | +| | CREATOR_SW_CFG_RMA_SPIN_EN | 4 | Whether or not to enable a busy-wait delay loop in the ROM, when a specific SW strapping configuration is applied during boot, to provide time to trigger an RMA lifecycle transition over JTAG. A value of kHardenedBoolTrue enables, all other values disable. | +| | CREATOR_SW_CFG_RMA_SPIN_CYCLES | 4 | The number of Ibex clock cycles to spin for when waiting for an RMA transition. Used in combination with the CREATOR_SW_CFG_RMA_SPIN_EN field. | +| | CREATOR_SW_CFG_RNG_REPCNT_THRESHOLDS | 4 | The repetition count health test thresholds to enable entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_REPCNTS_THRESHOLDS | 4 | The repetition count symbol health test thresholds to enable entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_ADAPTP_HI_THRESHOLDS | 4 | The adaptive proportion health test high thresholds to enable entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_ADAPTP_LO_THRESHOLDS | 4 | The adaptive proportion health test low thresholds to enable entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_BUCKET_THRESHOLDS | 4 | The bucket health test thresholds to enable entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_MARKOV_HI_THRESHOLDS | 4 | The Markov health test high thresholds to enable entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_MARKOV_LO_THRESHOLDS | 4 | The Markov health test low thresholds to enable entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_EXTHT_HI_THRESHOLDS | 4 | The external health test high thresholds to enable the entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_EXTHT_LO_THRESHOLDS | 4 | The external health test low thresholds to enable the entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_ALERT_THRESHOLD | 4 | The alert threshold to configure the entropy_src with during ROM execution. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_RNG_HEALTH_CONFIG_DIGEST | 4 | A CRC32 digest of all entropy_src health test threshold configuration fields above. This must be configured if CREATOR_SW_CFG_RNG_EN is true. See entropy_src documentation for more details. | +| | CREATOR_SW_CFG_SRAM_KEY_RENEW_EN | 4 | Whether or not the ROM should request SRAM to be rescrambled with a new key on every boot. kHardenedBoolFalse disables, while all other values enable. | +| | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_EN | 4 | Enablement of the ROM_EXT immutable code section. A value of kHardenedBoolTrue enables the feature. All other values disable it. | +| | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_START_OFFSET | 4 | Relative offset from the start of the ROM_EXT slot to find the immutable code section at. | +| | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_LENGTH | 4 | Length (in bytes) of the immutable code section. | +| | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_SHA256_HASH | 32 | SHA256 hash of the immutable ROM_EXT section. | +| | CREATOR_SW_CFG_RESERVED | 32 | Unused bits in the CREATOR_SW_CFG OTP partition. These can be claimed by software as needed. | +| OWNER_SW_CFG | OWNER_SW_CFG_ROM_ERROR_REPORTING | 4 | The shutdown error reporting verbosity used by the ROM. Should be configured to one of several `shutdown_error_redact_t` values. See `sw/device/silicon_creator/lib/shutdown.h` for more details. | +| | OWNER_SW_CFG_ROM_BOOTSTRAP_DIS | 4 | Whether or not to disable ROM bootstrap mechanism. A value of kHardenedBoolTrue disable bootstrap mechanism in the ROM, while all other values enable it. Note, the provisioning flow should take care when to program this field if it is used by a SKU, as there is no way to get firmware into flash in a PROD LC state if a valid ROM_EXT does not already exist in flash once this value is configured to true. | +| | OWNER_SW_CFG_ROM_ALERT_CLASS_EN | 4 | A four byte packed field, where each byte controls whether or not the ROM enables each alert class (A through D) of the alert_handler. The byte-sized subfields are arranged from D to A, MSB to LSB. Each byte should be set to an `alert_enable_t` value accordingly. See the alert_handler documentation for more details. | +| | OWNER_SW_CFG_ROM_ALERT_ESCALATION | 4 | A four byte packed field, where each byte controls the escalation configuration for each alert class (A through D) of the alert_handler configured by the ROM. The byte-sized subfields are arranged from D to A, MSB to LSB. Each byte should be set to an `alert_escalate_t` value accordingly. See the alert_handler documentation for more details. | +| | OWNER_SW_CFG_ROM_ALERT_CLASSIFICATION | 320 | The alert classifications (A through D) for each alert source of the alert_handler to be configured by the ROM. The field consists of a contiguous 320-byte block, or 80 32-bit words. The four bytes in each word encode the configuration of a single alert source across four lifecycle states, in order from LSB to MSB: PROD, PROD_END, DEV, and RMA. Each byte should be set to an `alert_class_t` value accordingly. The order of the 80 32-bit words, from LSB to MSB can be found in the EARLGREY_ALERTS list in `rules/const.bzl`. See the alert_handler documentation for more details. | +| | OWNER_SW_CFG_ROM_LOCAL_ALERT_CLASSIFICATION | 64 | Same as the `OWNER_SW_CFG_ROM_ALERT_CLASSIFICATION` field, except these configuration correspond to the local alert sources found in the `EARLGREY_LOC_ALERTS list in `rules/const.bzl`. | +| | OWNER_SW_CFG_ROM_ALERT_ACCUM_THRESH | 16 | The alert accumulation threshold values for each alert class (A through D) of the alert_handler to be configured by the ROM. This field consists of four 32-bit words encoding the accumulation thresholds for each alert class A through D arranged LSW to MSW. See the alert_handler documentation for more details. | +| | OWNER_SW_CFG_ROM_ALERT_TIMEOUT_CYCLES | 16 | Same as the `OWNER_SW_CFG_ROM_ALERT_ACCUM_THRESH` field, except each value corresponds to the interrupt timeout configuration of an alert class. | +| | OWNER_SW_CFG_ROM_ALERT_PHASE_CYCLES | 64 | The alert escalation phase durations, measured in clock cycles, the ROM will configure the four alert phases for each alert class of the alert_handler. This field consists of a contiguous 64-byte block, or an array of four 128-bit fields. Each 128-bit subfield encodes four 32-bit words that contain the alert phase cycle count configurations for alert escalation phases 0 to 3, from LSW to MSW. Each 128-bit subfield is contains all cofigurations for a single alert class, arranged from class A to D, from LS to MS. For example, the cycle durations of each escalation phase in this field should be configured as such, from LSB to MSB: ......... . See the alert_handler documentation for more details. | +| | OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD | 4 | The expected CRC32 digest over all of the alert_handler configurations set up by the ROM for a device operating in the PROD LC state. The ROM reads this field and checks it against a digest it computes over the alert_handler configuration it programmed. This field is expected to be automatically computed by the `otp_alert_digest()` Bazel rule. See the `alert_config_crc32()` function in the SiliconCreator alert_handler driver for more details on what configurations are included in this digest. | +| | OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD_END | 4 | Same as the `OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD` field, except the expected digest is for chips operating in the PROD_END LC state. | +| | OWNER_SW_CFG_ROM_ALERT_DIGEST_DEV | 4 | Same as the `OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD` field, except the expected digest is for chips operating in the DEV LC state. | +| | OWNER_SW_CFG_ROM_ALERT_DIGEST_RMA | 4 | Same as the `OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD` field, except the expected digest is for chips operating in the RMA LC state. | +| | OWNER_SW_CFG_ROM_WATCHDOG_BITE_THRESHOLD_CYCLES | 4 | Watchdog timer bite threshold (in cycles) configured by the ROM. | +| | OWNER_SW_CFG_ROM_KEYMGR_OTP_MEAS_EN | 4 | Whether or not to configure the attestation SW binding CSRs of the keymgr with the value in ROM_EXT manifest or the measurement of the OTP CreatorSwCfg, OwnerSwCfg, and secure boot key integrity digest. A value of kHardenedBoolTrue uses the ROM computed OTP measurements, while all other values trigger the use of the binding values included in the ROM_EXT manifest. | +| | OWNER_SW_CFG_MANUF_STATE | 4 | Manufacturing state binding field. For use by SiliconCreators or SiliconOwners to bind BL0 images to a specific device or set of devices. | +| | OWNER_SW_CFG_ROM_RSTMGR_INFO_EN | 4 | A two byte packed word that indicates the expected rstmgr alert and CPU info dump enable states, configured in the rstmgr's ALERT_INFO_CTRL and CPU_INFO_CTRL CSRs respectively. The expected format of this fields is {0,0,kHardenedBool*,kHardenedBool*}, read MSB to LSB, where the left most kHardenedBool* entry indicates the expected enablement state of the ALERT_INFO_CTRL, and the right most indicates the enablement state of the CPU_INFO_CTRL. Since the ROM expects both to be disabled upon handing over execution control to the ROM_EXT, this entire OTP field **should be left unprovisioned, or set to all 0**. | +| | OWNER_SW_CFG_ROM_EXT_BOOTSTRAP_EN | 4 | Unused. Set to 0. | +| | OWNER_SW_CFG_ROM_SENSOR_CTRL_ALERT_CFG | 12 | Alert configuration values for the sensor_ctrl block that will be configured by the ROM. This field is 12 bytes long, where each byte contains two 4-bit packed subfields, encoding two four bit boolean values (kMultiBitBool4False or kMultiBitBool4True), as follows (read MSB to LSB): {fatality, enablement}. For example, the byte 0x69 would encode the alert is: 1) recoverable, and 2) disabled. Currently, there are only 11 alerts in sensor_ctrl to configure, thus only the least significant bytes in this field are used. | +| | OWNER_SW_CFG_ROM_SRAM_READBACK_EN | 4 | This field encodes the enablements of the readback security features for the main and retention SRAMs to be configured by the ROM. This field is four bytes, but the LSB contains two 4-bit packed kMultiBitBool4* values that indicate the enablement of the feature for the retention SRAM and main SRAM, from MSB to LSB respectively. See the READBACK CSR of the sram_ctrl for more details on this feature. | +| | OWNER_SW_CFG_ROM_PRESERVE_RESET_REASON_EN | 4 | Whether or not the ROM should preserve the reset reasons CSR state in the rstmgr, or clear it. A value of kHardenedBoolTrue preserves the CSR, while other values trigger the ROM to clear the CSR after copying the reason to the retention SRAM. | +| | OWNER_SW_CFG_ROM_RESET_REASON_CHECK_VALUE | 4 | Two packed 16-bit values that indicate whether the ROM should perform a validation check on the the reset reasons during boot. The validation check is a hardening mechanism that checks the reset reasons for consistency at two different points in time. Both packed values should be the same. Values of kHardenedBoolFalse will instruct the ROM to skip the reset reasons validation check, while all other values will instruct the ROM to perform the check. | +| | OWNER_SW_CFG_ROM_BANNER_EN | 4 | Whether or not the ROM should print a banner message to the console UART during boot. A value of kHardenedBoolFalse disables the message printing, while all other values enable it. | +| | OWNER_SW_CFG_ROM_FLASH_ECC_EXC_HANDLER_EN | 4 | Whether or not the ROM should use the flash ECC exception handler during execution. A value of kHardenedBoolTrue allows the ROM to use an the exception handler that recovers gracefully and continues the boot process if a flash ECC error is encountered during verification of a specific ROM_EXT slot. This enables the ROM to attempt booting the next ROM_EXT slot if the first slot attempted has been corrupted. All other values trigger default ROM exception handling, which is all exceptions trigger a chip shutdown and reset. | +| | OWNER_SW_CFG_RESERVED | 128 | Unused bits in the OWNER_SW_CFG OTP partition. These can be claimed by software as needed. | +| ROT_CREATOR_AUTH_CODESIGN | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE0 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY0 | 64 | | +| | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE1 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY1 | 64 | | +| | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE2 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY2 | 64 | | +| | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE3 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY3 | 64 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE0 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY0 | 32 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG0 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE1 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY1 | 32 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG1 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE2 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY2 | 32 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG2 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE3 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY3 | 32 | | +| | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG3 | 4 | | +| | ROT_CREATOR_AUTH_CODESIGN_BLOCK_SHA2_256_HASH | 32 | | +| ROT_CREATOR_AUTH_STATE | ROT_CREATOR_AUTH_STATE_ECDSA_KEY0 | 4 | | +| | ROT_CREATOR_AUTH_STATE_ECDSA_KEY1 | 4 | | +| | ROT_CREATOR_AUTH_STATE_ECDSA_KEY2 | 4 | | +| | ROT_CREATOR_AUTH_STATE_ECDSA_KEY3 | 4 | | +| | ROT_CREATOR_AUTH_STATE_SPX_KEY0 | 4 | | +| | ROT_CREATOR_AUTH_STATE_SPX_KEY1 | 4 | | +| | ROT_CREATOR_AUTH_STATE_SPX_KEY2 | 4 | | +| | ROT_CREATOR_AUTH_STATE_SPX_KEY3 | 4 | | +| HW_CFG0 | DEVICE_ID | 32 | Unique device identifier that is always exposed through the lifecycle JTAG tap. See OpenTitan documentation on Device Identifiers. | +| | MANUF_STATE | 32 | Field to capture manufacturing status. Currently unused. Set to 0. | +| HW_CFG1 | EN_SRAM_IFETCH | 1 | Enablement of execute from SRAM switch in the sram_ctrl (see EXEC CSR). A kMultiBitBool8True value enables, while all other values disable. | +| | EN_CSRNG_SW_APP_READ | 1 | Enablement of CSRNG software application interface. A kMultiBitBool8True value enables, while all other values disable. Enablement is required to extract output from CSRNG via software. | +| | DIS_RV_DM_LATE_DEBUG | 1 | Disablement of RV_DM late debug feature (see rv_dm documentation). A kMultiBitBool8True disables the late debug feature and renders the rv_dm fully ungated in DEV lifecycle states. All other values gate rv_dm reachability based on the value of LATE_DEBUG_ENABLE CSR in the rv_dm block. | diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_kdi_fsm.svg b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_kdi_fsm.svg new file mode 100644 index 00000000000000..0f91a212c440d3 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_kdi_fsm.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_key_req_ack.svg b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_key_req_ack.svg new file mode 100644 index 00000000000000..c5cd58b5c1578b --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_key_req_ack.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_lci_fsm.svg b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_lci_fsm.svg new file mode 100644 index 00000000000000..63459e87af287b --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_lci_fsm.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_mmap.md b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_mmap.md new file mode 100644 index 00000000000000..ee030099dbeeea --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_mmap.md @@ -0,0 +1,122 @@ + + +| Index | Partition | Size [B] | Access Granule | Item | Byte Address | Size [B] | +|:-------:|:-------------------------:|:----------:|:----------------:|:---------------------------------------------------------------------------:|:--------------:|:----------:| +| 0 | VENDOR_TEST | 64 | 32bit | SCRATCH | 0x000 | 56 | +| | | | 64bit | [VENDOR_TEST_DIGEST](#Reg_vendor_test_digest_0) | 0x038 | 8 | +| 1 | CREATOR_SW_CFG | 368 | 32bit | CREATOR_SW_CFG_AST_CFG | 0x040 | 156 | +| | | | 32bit | CREATOR_SW_CFG_AST_INIT_EN | 0x0DC | 4 | +| | | | 32bit | CREATOR_SW_CFG_ROM_EXT_SKU | 0x0E0 | 4 | +| | | | 32bit | CREATOR_SW_CFG_SIGVERIFY_SPX_EN | 0x0E4 | 4 | +| | | | 32bit | CREATOR_SW_CFG_FLASH_DATA_DEFAULT_CFG | 0x0E8 | 4 | +| | | | 32bit | CREATOR_SW_CFG_FLASH_INFO_BOOT_DATA_CFG | 0x0EC | 4 | +| | | | 32bit | CREATOR_SW_CFG_FLASH_HW_INFO_CFG_OVERRIDE | 0x0F0 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_EN | 0x0F4 | 4 | +| | | | 32bit | CREATOR_SW_CFG_JITTER_EN | 0x0F8 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RET_RAM_RESET_MASK | 0x0FC | 4 | +| | | | 32bit | CREATOR_SW_CFG_MANUF_STATE | 0x100 | 4 | +| | | | 32bit | CREATOR_SW_CFG_ROM_EXEC_EN | 0x104 | 4 | +| | | | 32bit | CREATOR_SW_CFG_CPUCTRL | 0x108 | 4 | +| | | | 32bit | CREATOR_SW_CFG_MIN_SEC_VER_ROM_EXT | 0x10C | 4 | +| | | | 32bit | CREATOR_SW_CFG_MIN_SEC_VER_BL0 | 0x110 | 4 | +| | | | 32bit | CREATOR_SW_CFG_DEFAULT_BOOT_DATA_IN_PROD_EN | 0x114 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RMA_SPIN_EN | 0x118 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RMA_SPIN_CYCLES | 0x11C | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_REPCNT_THRESHOLDS | 0x120 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_REPCNTS_THRESHOLDS | 0x124 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_ADAPTP_HI_THRESHOLDS | 0x128 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_ADAPTP_LO_THRESHOLDS | 0x12C | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_BUCKET_THRESHOLDS | 0x130 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_MARKOV_HI_THRESHOLDS | 0x134 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_MARKOV_LO_THRESHOLDS | 0x138 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_EXTHT_HI_THRESHOLDS | 0x13C | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_EXTHT_LO_THRESHOLDS | 0x140 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_ALERT_THRESHOLD | 0x144 | 4 | +| | | | 32bit | CREATOR_SW_CFG_RNG_HEALTH_CONFIG_DIGEST | 0x148 | 4 | +| | | | 32bit | CREATOR_SW_CFG_SRAM_KEY_RENEW_EN | 0x14C | 4 | +| | | | 32bit | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_EN | 0x150 | 4 | +| | | | 32bit | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_START_OFFSET | 0x154 | 4 | +| | | | 32bit | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_LENGTH | 0x158 | 4 | +| | | | 32bit | CREATOR_SW_CFG_IMMUTABLE_ROM_EXT_SHA256_HASH | 0x15C | 32 | +| | | | 32bit | CREATOR_SW_CFG_RESERVED | 0x17C | 32 | +| | | | 64bit | [CREATOR_SW_CFG_DIGEST](#Reg_creator_sw_cfg_digest_0) | 0x1A8 | 8 | +| 2 | OWNER_SW_CFG | 712 | 32bit | OWNER_SW_CFG_ROM_ERROR_REPORTING | 0x1B0 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_BOOTSTRAP_DIS | 0x1B4 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_CLASS_EN | 0x1B8 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_ESCALATION | 0x1BC | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_CLASSIFICATION | 0x1C0 | 320 | +| | | | 32bit | OWNER_SW_CFG_ROM_LOCAL_ALERT_CLASSIFICATION | 0x300 | 64 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_ACCUM_THRESH | 0x340 | 16 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_TIMEOUT_CYCLES | 0x350 | 16 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_PHASE_CYCLES | 0x360 | 64 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD | 0x3A0 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_DIGEST_PROD_END | 0x3A4 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_DIGEST_DEV | 0x3A8 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_ALERT_DIGEST_RMA | 0x3AC | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_WATCHDOG_BITE_THRESHOLD_CYCLES | 0x3B0 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_KEYMGR_OTP_MEAS_EN | 0x3B4 | 4 | +| | | | 32bit | OWNER_SW_CFG_MANUF_STATE | 0x3B8 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_RSTMGR_INFO_EN | 0x3BC | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_EXT_BOOTSTRAP_EN | 0x3C0 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_SENSOR_CTRL_ALERT_CFG | 0x3C4 | 12 | +| | | | 32bit | OWNER_SW_CFG_ROM_SRAM_READBACK_EN | 0x3D0 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_PRESERVE_RESET_REASON_EN | 0x3D4 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_RESET_REASON_CHECK_VALUE | 0x3D8 | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_BANNER_EN | 0x3DC | 4 | +| | | | 32bit | OWNER_SW_CFG_ROM_FLASH_ECC_EXC_HANDLER_EN | 0x3E0 | 4 | +| | | | 32bit | OWNER_SW_CFG_RESERVED | 0x3E4 | 128 | +| | | | 64bit | [OWNER_SW_CFG_DIGEST](#Reg_owner_sw_cfg_digest_0) | 0x470 | 8 | +| 3 | ROT_CREATOR_AUTH_CODESIGN | 472 | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE0 | 0x478 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY0 | 0x47C | 64 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE1 | 0x4BC | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY1 | 0x4C0 | 64 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE2 | 0x500 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY2 | 0x504 | 64 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY_TYPE3 | 0x544 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_ECDSA_KEY3 | 0x548 | 64 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE0 | 0x588 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY0 | 0x58C | 32 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG0 | 0x5AC | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE1 | 0x5B0 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY1 | 0x5B4 | 32 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG1 | 0x5D4 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE2 | 0x5D8 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY2 | 0x5DC | 32 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG2 | 0x5FC | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_TYPE3 | 0x600 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY3 | 0x604 | 32 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_SPX_KEY_CONFIG3 | 0x624 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_CODESIGN_BLOCK_SHA2_256_HASH | 0x628 | 32 | +| | | | 64bit | [ROT_CREATOR_AUTH_CODESIGN_DIGEST](#Reg_rot_creator_auth_codesign_digest_0) | 0x648 | 8 | +| 4 | ROT_CREATOR_AUTH_STATE | 40 | 32bit | ROT_CREATOR_AUTH_STATE_ECDSA_KEY0 | 0x650 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_STATE_ECDSA_KEY1 | 0x654 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_STATE_ECDSA_KEY2 | 0x658 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_STATE_ECDSA_KEY3 | 0x65C | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_STATE_SPX_KEY0 | 0x660 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_STATE_SPX_KEY1 | 0x664 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_STATE_SPX_KEY2 | 0x668 | 4 | +| | | | 32bit | ROT_CREATOR_AUTH_STATE_SPX_KEY3 | 0x66C | 4 | +| | | | 64bit | [ROT_CREATOR_AUTH_STATE_DIGEST](#Reg_rot_creator_auth_state_digest_0) | 0x670 | 8 | +| 5 | HW_CFG0 | 72 | 32bit | DEVICE_ID | 0x678 | 32 | +| | | | 32bit | MANUF_STATE | 0x698 | 32 | +| | | | 64bit | [HW_CFG0_DIGEST](#Reg_hw_cfg0_digest_0) | 0x6B8 | 8 | +| 6 | HW_CFG1 | 16 | 32bit | EN_SRAM_IFETCH | 0x6C0 | 1 | +| | | | 32bit | EN_CSRNG_SW_APP_READ | 0x6C1 | 1 | +| | | | 32bit | DIS_RV_DM_LATE_DEBUG | 0x6C2 | 1 | +| | | | 64bit | [HW_CFG1_DIGEST](#Reg_hw_cfg1_digest_0) | 0x6C8 | 8 | +| 7 | SECRET0 | 40 | 64bit | TEST_UNLOCK_TOKEN | 0x6D0 | 16 | +| | | | 64bit | TEST_EXIT_TOKEN | 0x6E0 | 16 | +| | | | 64bit | [SECRET0_DIGEST](#Reg_secret0_digest_0) | 0x6F0 | 8 | +| 8 | SECRET1 | 88 | 64bit | FLASH_ADDR_KEY_SEED | 0x6F8 | 32 | +| | | | 64bit | FLASH_DATA_KEY_SEED | 0x718 | 32 | +| | | | 64bit | SRAM_DATA_KEY_SEED | 0x738 | 16 | +| | | | 64bit | [SECRET1_DIGEST](#Reg_secret1_digest_0) | 0x748 | 8 | +| 9 | SECRET2 | 88 | 64bit | RMA_TOKEN | 0x750 | 16 | +| | | | 64bit | CREATOR_ROOT_KEY_SHARE0 | 0x760 | 32 | +| | | | 64bit | CREATOR_ROOT_KEY_SHARE1 | 0x780 | 32 | +| | | | 64bit | [SECRET2_DIGEST](#Reg_secret2_digest_0) | 0x7A0 | 8 | +| 10 | LIFE_CYCLE | 88 | 32bit | LC_TRANSITION_CNT | 0x7A8 | 48 | +| | | | 32bit | LC_STATE | 0x7D8 | 40 | diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_overview.svg b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_overview.svg new file mode 100644 index 00000000000000..ef2adaf1f99f82 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_overview.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_partitions.md b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_partitions.md new file mode 100644 index 00000000000000..e19aad14717b80 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_partitions.md @@ -0,0 +1,57 @@ + + +| Partition | Secret | Buffered | Integrity | WR Lockable | RD Lockable | Description | +|:-------------------------:|:--------:|:----------:|:-----------:|:-------------:|:-------------:|:--------------------------------------------------------------------| +| VENDOR_TEST | no | no | no | yes (Digest) | yes (CSR) | Vendor test partition. | +| | | | | | | This is reserved for manufacturing smoke checks. The OTP wrapper | +| | | | | | | control logic inside prim_otp is allowed to read/write to this | +| | | | | | | region. ECC uncorrectable errors seen on the functional prim_otp | +| | | | | | | interface will not lead to an alert for this partition. | +| | | | | | | Instead, such errors will be reported as correctable ECC errors. | +| CREATOR_SW_CFG | no | no | yes | yes (Digest) | yes (CSR) | Software configuration partition. | +| | | | | | | This is for device-specific calibration data, e.g, clock, LDO, RNG, | +| | | | | | | and configuration settings set by the ROM. | +| OWNER_SW_CFG | no | no | yes | yes (Digest) | yes (CSR) | Software configuration partition. | +| | | | | | | This contains data that changes software behavior in the ROM, for | +| | | | | | | example enabling defensive features in ROM or selecting failure | +| | | | | | | modes if verification fails. | +| ROT_CREATOR_AUTH_CODESIGN | no | no | yes | yes (Digest) | yes (CSR) | This OTP partition is used to store four P-256 keys | +| | | | | | | and four Sphincs+ keys. The partition requires 464 | +| | | | | | | bytes of software visible storage. The partition is | +| | | | | | | locked at manufacturing time to protect against | +| | | | | | | malicious write attempts. | +| ROT_CREATOR_AUTH_STATE | no | no | yes | yes (Digest) | yes (CSR) | This OTP partition is used to capture the state of | +| | | | | | | each key slot. Each key can be in one of the | +| | | | | | | following states: BLANK, ENABLED, DISABLED. The | +| | | | | | | encoded values are such that transitions between | +| | | | | | | BLANK -> ENABLED -> DISABLED are possible without | +| | | | | | | causing ECC errors (this is a mechanism similar to | +| | | | | | | how we manage life cycle state transitions). The | +| | | | | | | partition is left unlocked to allow STATE updates in | +| | | | | | | the field. The ROM_EXT is required to lock access to | +| | | | | | | the OTP Direct Access Interface to prevent DoS | +| | | | | | | attacks from malicious code executing on Silicon | +| | | | | | | Owner partitions. DAI write locking is available in | +| | | | | | | EarlGrey. | +| HW_CFG0 | no | yes | yes | yes (Digest) | no | Hardware configuration 0 partition. | +| | | | | | | This contains a device identifier and manufacturing state. | +| HW_CFG1 | no | yes | yes | yes (Digest) | no | Hardware configuration 1 partition. | +| | | | | | | This contains several hardware feature switches. | +| SECRET0 | yes | yes | yes | yes (Digest) | yes (Digest) | Secret partition 0. | +| | | | | | | This contains TEST lifecycle unlock tokens. | +| SECRET1 | yes | yes | yes | yes (Digest) | yes (Digest) | Secret partition 1. | +| | | | | | | This contains SRAM and flash scrambling keys. | +| SECRET2 | yes | yes | yes | yes (Digest) | yes (Digest) | Secret partition 2. | +| | | | | | | This contains RMA unlock token, creator root key, and creator seed. | +| LIFE_CYCLE | no | yes | yes | no | no | Lifecycle partition. | +| | | | | | | This contains lifecycle transition count and state. This partition | +| | | | | | | cannot be locked since the life cycle state needs to advance to RMA | +| | | | | | | in-field. Note that while this partition is not marked secret, it | +| | | | | | | is not readable nor writeable via the DAI. Only the LC controller | +| | | | | | | can access this partition, and even via the LC controller it is not | +| | | | | | | possible to read the raw manufacturing life cycle state in encoded | +| | | | | | | form, since that encoding is considered a netlist secret. The LC | +| | | | | | | controller only exposes a decoded version of this state. | diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_prim_otp.svg b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_prim_otp.svg new file mode 100644 index 00000000000000..df6b31f078ed95 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_prim_otp.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg new file mode 100644 index 00000000000000..cd1a7ca1d70528 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/otp_ctrl_unbuf_part_fsm.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/programmers_guide.md b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/programmers_guide.md new file mode 100644 index 00000000000000..a3d00735f7668c --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/programmers_guide.md @@ -0,0 +1,261 @@ +# Programmer's Guide + +During provisioning and manufacturing, SW interacts with the OTP controller mostly through the Direct Access Interface (DAI), which is described below. +Afterwards during production, SW is expected to perform only read accesses via the exposed CSRs and CSR windows, since all write access to the partitions has been locked down. + +The following sections provide some general guidance, followed by an explanation of the DAI and a detailed OTP memory map. +Typical programming sequences are explained at the end of the Programmer's guide. + +## General Guidance + +### Initialization + +The OTP controller initializes automatically upon power-up and is fully operational by the time the processor boots. +The only initialization steps that SW should perform are: + +1. Check that the OTP controller has successfully initialized by reading [`STATUS`](registers.md#status). I.e., make sure that none of the ERROR bits are set, and that the DAI is idle ([`STATUS.DAI_IDLE`](registers.md#status)). +2. Set up the periodic background checks: + - Choose whether to enable periodic [background checks](#partition-checks) by programming nonzero mask values to [`INTEGRITY_CHECK_PERIOD`](registers.md#integrity_check_period) and [`CONSISTENCY_CHECK_PERIOD`](registers.md#consistency_check_period). + - Choose whether such checks shall be subject to a timeout by programming a nonzero timeout cycle count to [`CHECK_TIMEOUT`](registers.md#check_timeout). + - It is recommended to lock down the background check registers via [`CHECK_REGWEN`](registers.md#check_regwen), once the background checks have been set up. + +If needed, one-off integrity and consistency checks can be triggered via [`CHECK_TRIGGER`](registers.md#check_trigger). +If this functionality is not needed, it is recommended to lock down the trigger register via [`CHECK_TRIGGER_REGWEN`](registers.md#check_trigger_regwen). + +Later on during the boot process, SW may also choose to block read access to the SW managed partitions via the associated partition lock registers, e.g. [`CREATOR_SW_CFG_READ_LOCK`](registers.md#creator_sw_cfg_read_lock) or [`OWNER_SW_CFG_READ_LOCK`](registers.md#owner_sw_cfg_read_lock). + +### Reset Considerations + +It is important to note that values in OTP **can be corrupted** if a reset occurs during a programming operation. +This should be of minor concern for SW, however, since all partitions except for the LIFE_CYCLE partition are being provisioned in secure and controlled environments, and not in the field. +The LIFE_CYCLE partition is the only partition that is modified in the field - but that partition is entirely owned by the life cycle controller and not by SW. + +### Programming Already Programmed Regions + +OTP words cannot be programmed twice, and doing so may damage the memory array. +Hence the OTP controller performs a blank check and returns an error if a write operation is issued to an already programmed location. + +### Potential Side-Effects on Flash via Life Cycle + +It should be noted that the locked status of the partition holding the creator root key (i.e., the value of the [`SECRET2_DIGEST_0`](registers.md#secret2_digest)) determines the ID_STATUS of the device, which in turn determines SW accessibility of creator seed material in flash and OTP. +That means that creator-seed-related collateral needs to be provisioned to Flash **before** the OTP digest lockdown mechanism is triggered, since otherwise accessibility to the corresponding flash region is lost. +See the [life cycle controller documentation](../../../../ip/lc_ctrl/README.md#id-state-of-the-device) for more details. + +## Direct Access Interface + +OTP has to be programmed via the Direct Access Interface, which is comprised of the following CSRs: + +CSR Name | Description +-------------------------------------|------------------------------------ +[`DIRECT_ACCESS_WDATA_0`](registers.md#direct_access_wdata) | Low 32bit word to be written. +[`DIRECT_ACCESS_WDATA_1`](registers.md#direct_access_wdata) | High 32bit word to be written. +[`DIRECT_ACCESS_RDATA_0`](registers.md#direct_access_rdata) | Low 32bit word that has been read. +[`DIRECT_ACCESS_RDATA_1`](registers.md#direct_access_rdata) | High 32bit word that has been read. +[`DIRECT_ACCESS_ADDRESS`](registers.md#direct_access_address) | byte address for the access. +[`DIRECT_ACCESS_CMD`](registers.md#direct_access_cmd) | Command register to trigger a read or a write access. +[`DIRECT_ACCESS_REGWEN`](registers.md#direct_access_regwen) | Write protection register for DAI. + +See further below for a detailed [Memory Map](#direct-access-memory-map) of the address space accessible via the DAI. + +### Readout Sequence + +A typical readout sequence looks as follows: + +1. Check whether the DAI is idle by reading the [`STATUS`](registers.md#status) register. +2. Write the byte address for the access to [`DIRECT_ACCESS_ADDRESS`](registers.md#direct_access_address). +Note that the address is aligned with the granule, meaning that either 2 or 3 LSBs of the address are ignored, depending on whether the access granule is 32 or 64bit. +3. Trigger a read command by writing 0x1 to [`DIRECT_ACCESS_CMD`](registers.md#direct_access_cmd). +4. Poll the [`STATUS`](registers.md#status) until the DAI state goes back to idle. +Alternatively, the `otp_operation_done` interrupt can be enabled up to notify the processor once an access has completed. +5. If the status register flags a DAI error, additional handling is required (see [Section on Error handling](#error-handling)). +6. If the region accessed has a 32bit access granule, the 32bit chunk of read data can be read from [`DIRECT_ACCESS_RDATA_0`](registers.md#direct_access_rdata). +If the region accessed has a 64bit access granule, the 64bit chunk of read data can be read from the [`DIRECT_ACCESS_RDATA_0`](registers.md#direct_access_rdata) and [`DIRECT_ACCESS_RDATA_1`](registers.md#direct_access_rdata) registers. +7. Go back to 1. and repeat until all data has been read. + +The hardware will set [`DIRECT_ACCESS_REGWEN`](registers.md#direct_access_regwen) to 0x0 while an operation is pending in order to temporarily lock write access to the CSRs registers. + +### Programming Sequence + +A typical programming sequence looks as follows: + +1. Check whether the DAI is idle by reading the [`STATUS`](registers.md#status) register. +2. If the region to be accessed has a 32bit access granule, place a 32bit chunk of data into [`DIRECT_ACCESS_WDATA_0`](registers.md#direct_access_wdata). +If the region to be accessed has a 64bit access granule, both the [`DIRECT_ACCESS_WDATA_0`](registers.md#direct_access_wdata) and [`DIRECT_ACCESS_WDATA_1`](registers.md#direct_access_wdata) registers have to be used. +3. Write the byte address for the access to [`DIRECT_ACCESS_ADDRESS`](registers.md#direct_access_address). +Note that the address is aligned with the granule, meaning that either 2 or 3 LSBs of the address are ignored, depending on whether the access granule is 32 or 64bit. +4. Trigger a write command by writing 0x2 to [`DIRECT_ACCESS_CMD`](registers.md#direct_access_cmd). +5. Poll the [`STATUS`](registers.md#status) until the DAI state goes back to idle. +Alternatively, the `otp_operation_done` interrupt can be enabled up to notify the processor once an access has completed. +6. If the status register flags a DAI error, additional handling is required (see [Section on Error handling](#error-handling)). +7. Go back to 1. and repeat until all data has been written. + +The hardware will set [`DIRECT_ACCESS_REGWEN`](registers.md#direct_access_regwen) to 0x0 while an operation is pending in order to temporarily lock write access to the CSRs registers. + +Note that SW is responsible for keeping track of already programmed OTP word locations during the provisioning phase. +**It is imperative that SW does not write the same word location twice**, since this can lead to ECC inconsistencies, thereby potentially rendering the device useless. + +### Digest Calculation Sequence + +The hardware digest computation for the hardware and secret partitions can be triggered as follows: + +1. Check whether the DAI is idle by reading the [`STATUS`](registers.md#status) register. +3. Write the partition base address to [`DIRECT_ACCESS_ADDRESS`](registers.md#direct_access_address). +4. Trigger a digest calculation command by writing 0x4 to [`DIRECT_ACCESS_CMD`](registers.md#direct_access_cmd). +5. Poll the [`STATUS`](registers.md#status) until the DAI state goes back to idle. +Alternatively, the `otp_operation_done` interrupt can be enabled up to notify the processor once an access has completed. +6. If the status register flags a DAI error, additional handling is required (see [Section on Error handling](#error-handling)). + +The hardware will set [`DIRECT_ACCESS_REGWEN`](registers.md#direct_access_regwen) to 0x0 while an operation is pending in order to temporarily lock write access to the CSRs registers. + +It should also be noted that the effect of locking a partition via the digest only takes effect **after** the next system reset. +To prevent integrity check failures SW must therefore ensure that no more programming operations are issued to the affected partition after initiating the digest calculation sequence. + +### Software Integrity Handling + +As opposed to buffered partitions, the digest and integrity handling of unbuffered partitions is entirely up to software. +The only hardware-assisted feature in unbuffered partitions is the digest lock, which locks write access to an unbuffered partition once a nonzero value has been programmed to the 64bit digest location. + +In a similar vein, it should be noted that the system-wide bus-integrity metadata does not travel alongside the data end-to-end in the OTP controller (i.e., the bus-integrity metadata bits are not stored into the OTP memory array). +This means that data written to and read from the OTP macro is not protected by the bus integrity feature at all stages. +In case of buffered partitions this does not pose a concern since data integrity in these partitions is checked via the hardware assisted digest mechanism. +In case of unbuffered partitions however, the data integrity checking is entirely up to software. +I.e., if data is read from an unbuffered partition (either through the DAI or CSR windows), software should perform an integrity check on that data. + +## Error Handling + +The agents that can access the OTP macro (DAI, LCI, buffered/unbuffered partitions) expose detailed error codes that can be used to root cause any failure. +The error codes are defined in the table below, and the corresponding `otp_err_e` enum type can be found in the `otp_ctrl_pkg`. +The table also lists which error codes are supported by which agent. + +Errors that are not "recoverable" are severe errors that move the corresponding partition or DAI/LCI FSM into a terminal error state, where no more commands can be accepted (a system reset is required to restore functionality in that case). +Errors that are "recoverable" are less severe and do not cause the FSM to jump into a terminal error state. + +Note that error codes that originate in the physical OTP macro are prefixed with `Macro*`. + +Error Code | Enum Name | Recoverable | DAI | LCI | Unbuf | Buf | Description +-----------|------------------------|-------------|-----|-----|-------|-------|------------- +0x0 | `NoError` | - | x | x | x | x | No error has occurred. +0x1 | `MacroError` | no | x | x | x | x | Returned if the OTP macro command did not complete successfully due to a macro malfunction. +0x2 | `MacroEccCorrError` | yes | x | - | x | x | A correctable ECC error has occurred during a read operation in the OTP macro. +0x3 | `MacroEccUncorrError` | no | x | - | x* | x | An uncorrectable ECC error has occurred during a read operation in the OTP macro. Note (*): This error is collapsed into `MacroEccCorrError` if the partition is a vendor test partition. It then becomes a recoverable error. +0x4 | `MacroWriteBlankError` | yes / no* | x | x | - | - | This error is returned if a write operation attempted to clear an already programmed bit location. Note (*): This error is recoverable if encountered in the DAI, but unrecoverable if encountered in the LCI. +0x5 | `AccessError` | yes | x | - | x | - | An access error has occurred (e.g. write to write-locked region, or read to a read-locked region). +0x6 | `CheckFailError` | no | - | - | x | x | An unrecoverable ECC, integrity or consistency error has been detected. +0x7 | `FsmStateError` | no | x | x | x | x | The FSM has been glitched into an invalid state, or escalation has been triggered and the FSM has been moved into a terminal error state. + +All non-zero error codes listed above trigger an `otp_error` interrupt. +In addition, all unrecoverable OTP `Macro*` errors (codes 0x1, 0x3) trigger a `fatal_macro_error` alert, while all remaining unrecoverable errors trigger a `fatal_check_error` alert. + +If software receives an `otp_error` interrupt, but all error codes read back as 0x0 (`NoError`), this should be treated as a fatal error condition, and the system should be shut down as soon as possible. + +Note that the `MacroWriteBlankError` will only be generated if the write attempt over already written data fails within the OTP macro after applying any means supported within it to enable a write on existing data, e.g., a bit-reversal option. +Also note that while this error is marked as a recoverable error, the affected OTP word may be in an inconsistent state after this error has been returned. +This can cause several issues when the word is accessed again (either as part of a regular read operation, as part of the readout at boot, or as part of a background check). +It is important that SW ensures that each word is only written once, since this can render the device useless. + +## Direct Access Memory Map + +The table below provides a detailed overview of the items stored in the OTP partitions. +Some of the items that are buffered in registers is readable via memory mapped CSRs, and these CSRs are linked in the table below. +Items that are not linked can only be accessed via the direct programming interface (if the partition is not locked via the corresponding digest). +It should be noted that CREATOR_SW_CFG and OWNER_SW_CFG are accessible through a memory mapped window, and content of these partitions is not buffered. +Hence, a read access to those windows will take in the order of 10-20 cycles until the read returns. + +Sizes below are specified in multiples of 32bit words. + +{{#include otp_ctrl_mmap.md}} + +Note that since the content in the SECRET* partitions are scrambled using a 64bit PRESENT cipher, read and write access through the DAI needs to occur at a 64bit granularity. +Also, all digests (no matter whether they are SW or HW digests) have an access granule of 64bit. + +The table below lists digests locations, and the corresponding locked partitions. + +{{#include otp_ctrl_digests.md}} + +Write access to the affected partition will be locked if the digest has a nonzero value. + +For the software partition digests, it is entirely up to software to decide on the digest algorithm to be used. +Hardware will determine the lock condition only based on whether a non-zero value is present at that location or not. + +For the hardware partitions, hardware calculates this digest and uses it for [background verification](#partition-checks). +Digest calculation can be triggered via the DAI. + +Finally, it should be noted that the RMA_TOKEN and CREATOR_ROOT_KEY_SHARE0 / CREATOR_ROOT_KEY_SHARE1 items can only be programmed when the device is in the DEV, PROD, PROD_END and RMA stages. +Please consult the [life cycle controller documentation](../../../../ip/lc_ctrl/README.md) documentation for more information. + +## OTP Field Descriptions + +The table below describes what each field in the OTP partitions is used for. + +{{#include otp_ctrl_field_descriptions.md}} + +## Examples + +### Provisioning Items + +The following represents a typical provisioning sequence for items in all partitions (except for the LIFE_CYCLE partition, which is not software-programmable): + +1. [Program](#programming-sequence) the item in 32bit or 64bit chunks via the DAI. +2. [Read back](#readout-sequence) and verify the item via the DAI. +3. If the item is exposed via CSRs or a CSR window, perform a full-system reset and verify whether those fields are correctly populated. + +Note that any unrecoverable errors during the programming steps, or mismatches during the readback and verification steps indicate that the device might be malfunctioning (possibly due to fabrication defects) and hence the device may have to be scrapped. +This is however rare and should not happen after fabrication testing. + +### Locking Partitions + +Once a partition has been fully populated, write access to that partition has to be permanently locked. +For the HW_CFG* and SECRET* partitions, this can be achieved as follows: + +1. [Trigger](#digest-calculation-sequence) a digest calculation via the DAI. +2. [Read back](#readout-sequence) and verify the digest location via the DAI. +3. Perform a full-system reset and verify that the corresponding CSRs exposing the 64bit digest have been populated ([`HW_CFG_DIGEST_0`](registers.md#hw_cfg_digest), [`SECRET0_DIGEST_0`](registers.md#secret0_digest), [`SECRET1_DIGEST_0`](registers.md#secret1_digest) or [`SECRET2_DIGEST_0`](registers.md#secret2_digest)). + +It should be noted that locking only takes effect after a system reset since the affected partitions first have to re-sense the digest values. +Hence, it is critical that SW ensures that no more data is written to the partition to be locked after triggering the hardware digest calculation. +Otherwise, the device will likely be rendered inoperable as this can lead to permanent digest mismatch errors after system reboot. + +For the [`CREATOR_SW_CFG`](registers.md#creator_sw_cfg) and [`OWNER_SW_CFG`](registers.md#owner_sw_cfg) partitions, the process is similar, but computation and programming of the digest is entirely up to software: + +1. Compute a 64bit digest over the relevant parts of the partition, and [program](#programming-sequence) that value to [`CREATOR_SW_CFG_DIGEST_0`](registers.md#creator_sw_cfg_digest) or [`OWNER_SW_CFG_DIGEST_0`](registers.md#owner_sw_cfg_digest) via the DAI. Note that digest accesses through the DAI have an access granule of 64bit. +2. [Read back](#readout-sequence) and verify the digest location via the DAI. +3. Perform a full-system reset and verify that the corresponding digest CSRs [`CREATOR_SW_CFG_DIGEST_0`](registers.md#creator_sw_cfg_digest) or [`OWNER_SW_CFG_DIGEST_0`](registers.md#owner_sw_cfg_digest) have been populated with the correct 64bit value. + +Note that any unrecoverable errors during the programming steps, or mismatches during the read-back and verification steps indicate that the device might be malfunctioning (possibly due to fabrication defects) and hence the device may have to be scrapped. +This is however rare and should not happen after fabrication testing. + +## Device Interface Functions (DIFs) + +- [Device Interface Functions](../../../../../sw/device/lib/dif/dif_otp_ctrl.h) + +# Additional Notes + +## OTP IP Assumptions + +It is assumed the OTP IP employed in production has reasonable physical defense characteristics. +Specifically which defensive features will likely be use case dependent, but at a minimum they should have the properties below. +Note some properties are worded with "SHALL" and others with "SHOULD". +"SHALL" refers to features that must be present, while "SHOULD" refers to features that are ideal, but optional. + +- The contents shall not be observable via optical microscopy (for example anti-fuse technology). +- The IP lifetime shall not be limited by the amount of read cycles performed. +- If the IP contains field programmability (internal charge pumps and LDOs), there shall be mechanisms in place to selectively disable this function based on device context. +- If the IP contains redundant columns, rows, pages or banks for yield improvement, it shall provide a mechanism to lock down arbitrary manipulation of page / bank swapping during run-time. +- The IP shall be clear on what bits must be manipulated by the user, what bits are automatically manipulated by hardware (for example ECC or redundancy) and what areas the user can influence. +- The IP shall be compatible, through the use of a proprietary wrapper or shim, with an open-source friendly IO interface. +- The IP should functionally support the programming of already programmed bits without information leakage. +- The IP should offer SCA resistance: + - For example, the content may be stored differentially. + - For example, the sensing exhibits similar power signatures no matter if the stored bit is 0 or 1. +- The IP interface shall be memory-like if beyond a certain size. +- When a particular location is read, a fixed width output is returned; similar when a particular location is programmed, a fixed width input is supplied. +- The IP does not output all stored bits in parallel. +- The contents should be electrically hidden. For example, it should be difficult for an attacker to energize the fuse array and observe how the charge leaks. +- The IP should route critical nets at lower metal levels to avoid probing. +- The IP should contain native detectors for fault injection attacks. +- The IP should contain mechanisms to guard against interrupted programming - either through malicious intent or unexpected power loss and glitched address lines. +- The IP should contain mechanisms for error corrections (single bit errors). + - For example ECC or redundant bits voting / or-ing. + - As error correction mechanisms are technology dependent, that information should not be exposed to the open-source controller, instead the controller should simply receive information on whether a read / program was successful. +- The IP should have self-test functionality to assess the health of the storage and analog structures. +- The IP may contain native PUF-like functionality. diff --git a/hw/ip/otp_ctrl/doc/registers.md b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/registers.md similarity index 99% rename from hw/ip/otp_ctrl/doc/registers.md rename to hw/top_earlgrey/ip_autogen/otp_ctrl/doc/registers.md index 435a5ddb2f2a21..517922748a11d1 100644 --- a/hw/ip/otp_ctrl/doc/registers.md +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/registers.md @@ -1,6 +1,6 @@ # Registers - + ## Summary of the **`core`** interface's registers | Name | Offset | Length | Description | diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/theory_of_operation.md b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/theory_of_operation.md new file mode 100644 index 00000000000000..e3c5de4040a111 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/doc/theory_of_operation.md @@ -0,0 +1,514 @@ +# Theory of Operation + +Conceptually speaking, the OTP functionality is at a high level split into "front-end" and "back-end". +The "front-end" contains the logical partitions that feed the hardware and software consumer interfaces of the system. +The "back-end" represents the programming interface used by hardware and software components to stage the upcoming values. +The diagram below illustrates this behavioral model. + +![OTP Controller Block Diagram](otp_ctrl_behavioral_model.svg) + +Note that the front-end contains both buffered and unbuffered partitions. +Buffered partitions are sensed once per power cycle and their contents are stored in registers, whereas unbuffered partitions are read on-demand. +The former are typically partitions that contain data like hardware configuration bits, key material and the life cycle state that need to be always available to the hardware, whereas the latter are large partitions that are accessed infrequently, such as the software configurations. +Values that are programmed into a buffered partition via the programming interface (coupled with read verification) are merely "staged", and do not take effect until the next power cycle. + +The sections below describe the operation of various pieces of the OTP controller and how it supports the described functionality. + +## Logical Partitions + +The OTP is logically separated into partitions that represent different functions. +This means the isolation is virtual and maintained by the OTP controller instead of the underlying OTP IP. + +Within each logical partition, there are specific enforceable properties + +- Confidentiality via secret partitions + - This controls whether a particular partition contains secret data. + - If secret, a partition is not readable by software once locked, and is scrambled in storage. +- Read lockability + - This controls whether a particular partition disables software readability for later stage software. + - Some partitions can be locked statically (by computing and storing an associated digest in OTP), others can be read locked at runtime via CSRs. +- Write lockability + - This controls whether a partition is locked and prevented from future updates. + - A locked partition is stored alongside a digest to be used later for integrity verification. +- Integrity Verification + - Once a partition is write-locked by calculating and writing a non-zero [digest](#locking-a-partition) to it, it can undergo periodic verification (time-scale configurable by software). +This verification takes two forms, partition integrity checks, and storage consistency checks. + +Since the OTP is memory-like in nature (it only outputs a certain number of bits per address location), some of the logical partitions are buffered in registers for instantaneous and parallel access by hardware. +This is a critical point, since after power-up, these particular OTP contents are stored in flip flops and sourced to the system. +I.e., buffered partitions are **NOT** directly sourced from the OTP macro itself. +Thus the security of both volatile (OTP controller) and non-volatile (OTP IP) storage becomes important. + +### Partition Listing and Description + +The OTP controller for OpenTitan contains the seven logical partitions shown below. + +{{#include otp_ctrl_partitions.md}} + +Generally speaking, the production life cycle of a device is split into 5 stages "Manufacturing" -> "Calibration and Testing" -> "Provisioning" -> "Mission" -> "RMA". +OTP values are usually programmed during "Calibration and Testing", "Provisioning" and "RMA" stages, as explained below. +A detailed listing of all the items and the corresponding memory map can be found in the [Programmer's Guide](programmers_guide.md)) further below. + +### Calibration and Test + +During this stage, the device is tested for functionality and calibrated to ensure uniformity. +The calibration can focus on a number of things, but usually is centered around adjusting clock, voltage and timing sources to remove process variation. +These calibration values are programmed into the CREATOR_SW_CFG partition, as they are non-secret values meant to be read out by software and programmed into respective peripherals. + +Early on during this stage, the various tokens are also programmed into the secret partitions and harvested by the silicon creator. + +### Provisioning + +During this stage, the device is provisioned with the final firmware and a "unique" seed or identity. +The secret partitions are populated with root secrets and keys that are critical to establishing the device identity. + +As part of injecting the final firmware, the stock-keeping-unit-specific hardware and software configurations are also programmed. + +### Life Cycle Partition + +The life cycle partition is active throughout all stages and hence it is the **ONLY** partition that cannot be locked. +After the device finishes provisioning and goes into production, it must retain the ability to transition back to RMA in case of unexpected failures. + +In order to support this transition, the [life cycle state](../../../../ip/lc_ctrl/README.md) and counters must always be update-able. + +## Locking a Partition + +Write access to a partition can be permanently locked when software determines it will no longer make any updates to that partition. +To lock, an integrity constant is calculated and programmed alongside the other data of that partition. +The size of that integrity constant depends on the partition size granule, and is either 32bit or 64bit (see also [Direct Access Memory Map](#direct-access-memory-map)). + +Once the "integrity digest" is non-zero, no further updates are allowed. +If the partition is secret, software is in addition no longer able to read its contents (see [Secret Partition description](#secret-vs-nonsecret-partitions)). + +Note however, in all partitions, the digest itself is **ALWAYS** readable. +This gives software an opportunity to confirm that the locking operation has proceeded correctly, and if not, scrap the part immediately. + +Calculation of the integrity digest depends on whether the partition requires periodic background verification. + +### Vendor Test Partition + +The vendor test partition is intended to be used for OTP programming smoke checks during the manufacturing flow. +The silicon creator may implement these checks inside the proprietary version of the `prim_otp` wrapper. +This partition behaves like any other SW partition, with the exception that ECC uncorrectable errors will not lead to fatal errors / alerts as they do in all other partitions. +This is due to the nature of the OTP programming smoke checks, which may leave certain OTP words in a state inconsistent with the ECC polynomial employed upon OTP readout. + +### Software Configuration Partitions + +The software configuration partitions are used as non-volatile storage for flags, configuration and calibration data. +As such, the contents of this partition are usually consumed once as part of code execution, or moved to another storage compartment somewhere in the design. +For example, the clock calibration values and the LDO calibration values are programmed to the analog sensor top (AST) at startup. + +As such, it is not necessary to check periodically at the OTP source. +Instead, software can simply check as part of secure boot and take other measures when these values are programmed into peripherals. + +For this partition it is thus the responsibility of software to calculate the integrity digest and program it into the OTP. +It is also reasonable to shadow (parts of) this partition in main memory, and there is not an immediate impact from OTP contents to hardware. + +### Hardware Configuration and Secret Partitions + +The hardware and secret partitions directly affect downstream hardware. +The contents must go through periodic integrity checks and therefore the stored digest is calculated by hardware when software provides the intent to lock (as opposed to the software partitions where the digest has to be calculated by software). + +### Life Cycle Partition + +The life cycle partition cannot be locked and will therefore not contain a stored digest. +Note however that only the life cycle controller has access to this partition, i.e., the Direct Access Interface (DAI) cannot read nor write from/to the life cycle partition. + +## Secret vs Non-Secret Partitions + +Non-secret OTP partitions hold data that can be public; or data that has no impact on security. +For example, the current value of lock bits or clock calibration values. +These values are stored in OTP as plaintext. + +Secret partitions contain data that are critical to security, for example FLASH scrambling keys, device root secret and unlock tokens. +These values are stored scrambled in OTP, and are descrambled upon read. +The currently employed cipher is PRESENT, as it lends itself well to iterative decomposition, and it is a proven lightweight block cipher (see also [PRESENT Scrambling Primitive](../../../../ip/prim/doc/prim_present.md). +The usage of a block cipher however implies that the secret partitions can only be written in 64bit chunks. + +Further, the contents of a particular secret partition are not readable by software once locked (other than the digest which must be always readable); while non-secret partitions are always readable unless read accessibility is explicitly removed by software. + +Unfortunately, secret partitions must utilize a global netlist key for the scrambling operation, as there is no other non-volatile storage to store a unique key. + + +## Partition Checks + +### Integrity + +Once the appropriate partitions have been locked, the hardware integrity checker employs two integrity checks to verify the content of the volatile buffer registers: + +1. All buffered partitions have additional ECC protection (8bit ECC for each 64bit block) that is concurrently monitored. +2. The digest of the partition is recomputed at semi-random intervals and compared to the digest stored alongside the partition. + +The purpose of this check is NOT to check between the storage flops and the OTP, but whether the buffer register contents remain consistent with the calculated digest. +This verification is primarily concerned with whether the storage flops have experienced fault attacks. +This check applies to only the HW_CFG* and SECRET* partitions. +If a failure is encountered, the OTP controller will send out a `fatal_check_error` alert and reset all of its hardware outputs to their defaults. + +### Storage Consistency + +This verification ensures the value stored in the buffer registers remain consistent with those in the OTP. +This process re-reads the OTP at semi-random intervals and confirms the value read is the same as the value stored. +Note, given there are integrity checks in parallel, it is not necessary for some partitions to check ALL read contents for consistency. +If there is an integrity digest, only the digest needs to be read; otherwise, all values must be read. + + +This check applies to LIFE_CYCLE, HW_CFG* and SECRET* partitions. +If a failure is encountered, the OTP controller will send out a `fatal_check_error` alert and reset all of its hardware outputs to their defaults. + +Note that checks applied to life cycle could cause a failure if life cycle is updated, because life cycle is the only partition that may contain live updates. +The controller hence detects this condition based on the `lc_check_byp_en_i` signal coming from the life cycle controller, and pauses background checks on this partition in order to prevent false positives. + +### Secret Partition Integrity Checks + +Since the secret partitions are stored scrambled, this also implies the integrity digest is calculated over the scrambled form. +In order to balance the amount of buffer registers needed, only the decrypted form of the secret partitions is held in buffer registers. +Hardware calculates the digest by re-scrambling the data before passing it through the digest. + + +## Power-up and Sense + +The OTP controller partition storage must output a specified safe default (it is not always 0 like a blank OTP) upon reset release. +This default output must remain until the OTP controller completes all checks. + +The OTP controller reads from the OTP IP. +If the reads pass OTP IP internal checks (for example ECC or redundancy), the partition storage is updated; however the output is still held at the default state via an output mux. +After all read is complete, the OTP controller performs integrity checks on the HW_CFG* and SECRET* partitions. +If a partition fails the integrity checks at this point it would signal an initialization error in the status CSR and abort further initialization. + +After all integrity checks are complete, the OTP controller releases the output gating and marks outputs as valid. +However, any partition marked with "error" continues to hold its output in the default state. + +Once the above steps are complete, the partition storage in buffered registers is not updated again (except for updates to the life cycle partition through the life cycle interface). +I.e., values programmed to OTP via the programming interface will not be visible in buffered registers until after the next power cycle. + +At this point, outputs of the partition storage are NOT expected to change unless a periodic check suddenly fails. +When this failure occurs, all outputs are reverted to their default state, and an alert is immediately triggered to the alert handler. +For timing purposes, OTP outputs can be treated as semi-static, as this error event should be rare and exceptional. + + +## Partition Defaults + +Partition defaults are context specific. +For example, a hardware configuration item that locks down specific access should default to "no access". +This ensures that a glitch attack on the OTP cannot easily revert the design to an insecure state. + +This hence suggests that when an OTP is all 0's and all 1's, it should, whenever possible, reflect an invalid or inert state in the encoding space of the affected item. +This also implies the reset state of consuming agents (for example key manager and life cycle), should default to invalid / inert state as well. + + +## Program and Read Ports + +As shown previously, the OTP is split into a front and back end. +The back-end interface is primarily used to update OTP contents, and read back for debug and verification purposes. +Despite being a separate functional access port from the logical partitions, the program and read ports are subjected to the same access controls. + +When a partition is write-locked, programming accesses are disallowed. +If the partition is secret, read accesses by the back-end interface are also disallowed (except for the digest which must always be readable). +Software can also disable any read accesses to the software configuration partitions via CSR settings to prevent later stage software from reading any content. + +The exception to the above is the life cycle partition. +The life cycle controller interface also acts as a "back-end" interface that always has programming access to ensure life cycle state can be advanced. + +Note, the program and read ports can conflict with ongoing background storage checks, and the OTP controller arbitrates between these two sides. +An in-progress operation will always be completed. +Afterwards, or when two requests arrive at the same time, the priority is life cycle > programming interface > on-demand read accesses via CSR windows > background checks. + + +## Programming the OTP + +The OTP controller has two programming paths: + +1. a functional programming path through software (the program port), +2. Life cycle programming path through hardware. + +The functional interface is used to update all partitions except for life cycle. +As mentioned previously, any updates made during the current power cycle are **NOT** reflected in the buffered partitions until the next reboot. + +The life cycle interface is used to update the life cycle state and transition counter only. +The commands are issued from the [life cycle controller](../../../../ip/lc_ctrl/README.md), and similarly, successful or failed indications are also sent back to the life cycle controller. +Similar to the functional interface, the life cycle controller allows only one update per power cycle, and after a requested transition reverts to an inert state until reboot. + +For more details on how the software programs the OTP, please refer to the [Programmer's Guide](programmers_guide.md)) further below. + + +## Design Details + +### Block Diagram + +The following is a high-level block diagram that illustrates everything that has been discussed. + +![OTP Controller Block Diagram](otp_ctrl_blockdiag.svg) + +Each of the partitions P0-P7 has its [own controller FSM](#partition-implementations) that interacts with the OTP wrapper and the [scrambling datapath](#scrambling-datapath) to fulfill its tasks. +The partitions expose the address ranges and access control information to the Direct Access Interface (DAI) in order to block accesses that go to locked address ranges. +Further, the only two blocks that have (conditional) write access to the OTP are the DAI and the Life Cycle Interface (LCI) blocks. +The partitions can only issue read transactions to the OTP macro. +Note that the access ranges of the DAI and the LCI are mutually exclusive. +I.e., the DAI cannot read from nor write to the life cycle partition. +The LCI cannot read the OTP, but is allowed to write to the life cycle partition. + +The CSR node on the left side of this diagram connects to the DAI, the OTP partitions (P0-P7) and the OTP wrapper through a gated TL-UL interface. +All connections from the partitions to the CSR node are read-only, and typically only carry a subset of the information available. +E.g., the secret partitions only expose their digest value via the CSRs. + +The Key Derivation Interface (KDI) on the bottom right side interacts with the scrambling datapath, the EDN and the partition holding the scrambling root keys in order to derive static and ephemeral scrambling keys for FLASH and SRAM scrambling. + +The test access gate shown at the top of the block diagram is governed by the life cycle qualification signal `dft_en_i`, which is only enabled during the TEST_UNLOCKED* life cycle states. +Otherwise, test access via this TL-UL window is locked down. + +In addition to the blocks mentioned so far, the OTP controller also contains an LFSR timer that creates pseudo-randomly distributed partition check requests, and provides pseudo random data at high bandwidth in the event of a secure erase request due to chip-wide alert escalation. +For security reasons, the LFSR is periodically reseeded with entropy coming from EDN. + +### Data Allocation and Packing +#### Software View + +The effective word width of an OTP IP typically depends on a couple of factors, including the redundancy scheme employed. +For this the design at hand, it is assumed that this native OTP word-width is 16bit. +For software convenience, however, these details are abstracted and the open-source OTP controller exposes the OTP storage as a linear address space of 32bit words, which is aligned with the machine word size of the Ibex processor. +Since the OTP IP employs a redundancy mechanism similar to ECC, this implies however that write operations take place at a granularity of 32bit blocks for non-secret and 64bit blocks for secret partitions (due to the scrambling). +Hence, software is responsible to appropriately pack and program items, since each 32bit location can only be programmed once. + +#### Life Cycle View + +Since the life cycle partition is the only partition that needs live updates in-field, proper care must be taken to properly encode data in this partition such that incremental updates are possible. +The life cycle state is hence encoded such that incremental updates to the state are always carried out at the granularity of a 16bit word. +Further, the life cycle transition counter is encoded such that each stroke consumes a full 16bit word for the same reason. + +See [life cycle controller documentation](../../../../ip/lc_ctrl/README.md) for more details on the life cycle encoding. + +### Partition Controllers + +In RTL, we distinguish between buffered and unbuffered partition modules. +These are parameterized, such that we can assemble the array of OTP partitions with these two modules only. +The corresponding controller FSMs are explained in more detail below. + +#### Unbuffered Partition + +![Unbuffered Partition FSM](otp_ctrl_unbuf_part_fsm.svg) + +As shown above, the unbuffered partition module has a relatively simple controller FSM that only reads out the digest value of the partition upon initialization, and then basically waits for TL-UL read transactions to its corresponding window in the CSR space. + +Write access through the DAI will be locked in case the digest is set to a non-zero value. +Also, read access through the DAI and the CSR window can be locked at runtime via a CSR. +Read transactions through the CSR window will error out if they are out of bounds, or if read access is locked. + +Note that unrecoverable [OTP errors](#generalized-open-source-interface), ECC failures in the digest register or external escalation via `lc_escalate_en` will move the partition controller into a terminal error state. + +#### Buffered Partition + +![Buffered Partition FSM](otp_ctrl_buf_part_fsm.svg) + +The controller FSM of the buffered partition module is more complex than the unbuffered counterpart, since it has to account for scrambling and digest calculation. + +Upon initialization, the controller reads out the whole partition and descrambles it on the fly if needed. + +Then, right after the initial readout, the partition controller jumps into the first integrity check, which behaves somewhat differently, depending on whether the partition is digest protected (or not) and/or scrambled (or not). +If the partition is not digest protected, or if the digest has not yet been computed, the check completes right away, and the buffered values are released for hardware broadcast. +Otherwise, the partition contents in the buffer registers are re-scrambled if needed, and a digest is computed on the fly. +If the computed digest matches with the one that has been read out before, the buffered registers are released for hardware broadcast. +Otherwise, the buffered values are gated to their default, and an alert is triggered through the error handling logic. + +After initialization, the integrity check (as described above) and the consistency check can be triggered by the LFSR timer mechanism on a periodic basis. + +The consistency check behaves differently, depending on whether the partition is digest protected or not. +If it is, the consistency check will read out the digest stored in OTP and compare it with the value stored in the buffer register. +Otherwise, if no digest is available, the controller will read out the whole partition from OTP, and compare it to the contents stored in the buffer registers. +In case of a mismatch, the buffered values are gated to their default, and an alert is triggered through the error handling logic. + +Note that in case of unrecoverable OTP errors or ECC failures in the buffer registers, the partition controller FSM is moved into a terminal error state, which locks down all access through DAI and clamps the values that are broadcast in hardware to their defaults. + +External escalation via the `lc_escalate_en` signal will move the partition controller FSM into the terminal error state as well. +See [life cycle controller documentation](../../../../ip/lc_ctrl/README.md) for more details. + +### Direct Access Interface Control + +![Direct Access Interface FSM](otp_ctrl_dai_fsm.svg) + +Upon reset release, the DAI controller first sends an initialization command to the OTP macro. +Once the OTP macro becomes operational, an initialization request is sent to all partition controllers, which will read out and initialize the corresponding buffer registers. +The DAI then becomes operational once all partitions have initialized, and supports read, write and digest calculation commands (see [here](#direct-access-interface) for more information about how to interact with the DAI through the CSRs). + +Read and write commands transfer either 32bit or 64bit of data from the OTP to the corresponding CSR and vice versa. The access size is determined automatically, depending on whether the partition is scrambled or not. Also, (de)scrambling is performed transparently, depending on whether the partition is scrambled or not. + +Digest calculation commands read out the complete contents of a particular partition, compute a digest and write that digest value to the predefined location at the end of the partition. + +Note that any unrecoverable OTP error will move the DAI into a terminal error state, where all access through the DAI will be locked. +Also, the DAI consumes the read and write access information provided by the partition controller, and if a certain read or write access is not permitted, a recoverable error will be flagged in the status / error CSRs. + +### Life Cycle Interface Control + +![Life Cycle Interface FSM](otp_ctrl_lci_fsm.svg) + +Upon reset release the LCI FSM waits until the OTP controller has initialized and the LCI gets enabled. +Once it is in the idle state, life cycle state updates can be initiated via the life cycle interface as [described here](#state-transitions). +The LCI controller takes the life cycle state to be programmed and writes all 16bit words to OTP. +In case of unrecoverable OTP errors, the FSM signals an error to the life cycle controller and moves into a terminal error state. + +### Key Derivation Interface + +![Key Derivation Interface FSM](otp_ctrl_kdi_fsm.svg) + +Upon reset release the KDI FSM waits until the OTP controller has initialized and the KDI gets enabled. +Once it is in the idle state, key derivation can be requested via the [flash](#interface-to-flash-scrambler) and [sram](#interface-to-sram-and-otbn-scramblers) interfaces. +Based on which interface makes the request, the KDI controller will evaluate a variant of the PRESENT digest mechanism as described in more detail below. + +### Scrambling Datapath + +![OTP Digest Mechanism](otp_ctrl_digest_mechanism.svg) + +The scrambling datapath is built around an iterative implementation of the [PRESENT lightweight cipher](../../../../ip/prim/doc/prim_present.md) that performs one round per cycle. +The datapath contains some additional multiplexing circuitry to enable the DAI, KDI and partition controllers to evaluate different functions with the same datapath. +The algorithmic steps of these functions are explained in more detail below. + +#### Scrambling + +As illustrated in subfigure a) in the diagram above, the standard 128bit-key PRESENT configuration with 31 rounds is used for scrambling operations. +The key used for scrambling is a global netlist constant chosen by the silicon creator, and all secret partitions are encrypted using the their own distinct netlist constant. +Note that the amount of data that is being scrambled is small (160byte = 20 x 64bit blocks) and the scrambled data remains constant. +Hence, no additional masking or diversification scheme is applied since only a very limited amount of information can be gathered by observing the scrambling operation via side-channels. + +#### Digest Calculation + +The integrity digests used in the [partition checks](#partition-checks) are computed using a custom [Merkle-Damgard](https://en.wikipedia.org/wiki/Merkle%E2%80%93Damg%C3%A5rd_construction) scheme, where the employed one-way compression function F is constructed by using PRESENT in a [Davies-Meyer arrangement](https://en.wikipedia.org/wiki/One-way_compression_function#Davies%E2%80%93Meyer). +This is illustrated in subfigure b). + +At the beginning of the digest calculation the 64bit state is initialized with an initialization vector (IV). +Then, the data to be digested is split into 128bit chunks, each of which is used as a 128bit key input for updating the 64bit state with the compression function F. +Chunks that are not aligned with 128bit are padded with zero, and the finalization operation consists of another 31-round encryption pass with a finalization constant. +Note that both the IV as well as the finalization constant are global netlist constants chosen by the silicon creator. + +#### Scrambling Key Derivation + +The key derivation functions for ephemeral SRAM and static FLASH scrambling keys employ a similar construction as the digest calculation function. +In particular, the keys are derived by repeatedly reducing a (partially random) block of data into a 64bit block, as illustrated in subfigures c) and d). + +For ephemeral SRAM scrambling keys, the data block is composed of the 128bit SRAM_DATA_KEY_SEED stored in OTP, as well as 128bit of fresh entropy fetched from the EDN. +This process is repeated twice in order to produce a 128bit key. + +For static FLASH scrambling keys, the data block is composed of a 128bit part of either the FLASH_DATA_KEY_SEED or the FLASH_ADDR_KEY_SEED stored in OTP. +These key seeds are 256bit in size, allowing to use a unique chunk of 128bit of key seed data to derive a 64bit halve of a particular scrambling key. + +Note that the IV and finalization constants are distinct for SRAM and FLASH data and FLASH address scrambling keys. +These constants are chosen by the silicon creator prior to the tapeout. + +### Access Arbitration + +Access to the OTP wrapper and the scrambling datapath are both round-robin arbitrated, where the former arbitration occurs at cycle level (i.e., individual OTP memory accesses), and the latter occurs at the level of complete transactions (i.e., full digest or encryption). +Arbitration at transaction level is implemented similarly to cycle-based arbitration, with the difference that the grant signals remain asserted until the requestor deasserts the request (thereby releasing the arbiter, which acts as a mutex in this case). +This is behavior illustrated in the example below. + +```wavejson +{signal: [ + {name: 'clk_i', wave: 'p............'}, + {name: 'part_scrmbl_mtx_req[0]', wave: '01....0.1....'}, + {name: 'part_scrmbl_mtx_req[1]', wave: '0.1......0...'}, + {name: 'part_scrmbl_mtx_req[2]', wave: '0.1........0.'}, + {}, + {name: 'part_scrmbl_mtx_gnt[0]', wave: '01....0....1.'}, + {name: 'part_scrmbl_mtx_gnt[1]', wave: '0.....1..0...'}, + {name: 'part_scrmbl_mtx_gnt[2]', wave: '0........1.0.'}, +]} +``` + +### Primitive Wrapper and FPGA Emulation + +![OTP Wrapper Block Diagram](otp_ctrl_prim_otp.svg) + +The OTP IP is wrapped up in a primitive wrapper that exposes a TL-UL interface for testing purposes, and a generalized open-source interface for functional operation (described below). +Any OTP redundancy mechanism like per-word ECC is assumed to be handled inside the wrapper, which means that the word width exposed as part of the generalized interface is the effective word width. + +Note that the register space exposed via the TL-UL test interface, as well as DFT and power-related signals are dependent on the underlying proprietary OTP IP. +They are therefore not further described in this document. + +#### Generalized Open-source Interface + +The generalized open-source interface uses a couple of parameters (defaults set for Earlgrey configuration). + +Parameter | Default | Top Earlgrey | Description +---------------|---------|---------------|--------------- +`Width` | 16 | 16 | Native OTP word width. +`Depth` | 1024 | 1024 | Depth of OTP macro. +`CmdWidth` | 7 | 7 | Width of the OTP command. +`ErrWidth` | 3 | 3 | Width of error code output signal. +`PwrSeqWidth` | 2 | 2 | Width of power sequencing signals to/from AST. +`SizeWidth` | 2 | 2 | Width of the size field. +`IfWidth` | 2^`SizeWidth` * `Width` | 2^`SizeWidth` * `Width` | Data interface width. + +The generalized open-source interface is a simple command interface with a ready / valid handshake that makes it possible to introduce back pressure if the OTP macro is not able to accept a command due to an ongoing operation. + +In order to facilitate the scrambling and digest operations, the data width has been sized such that data blocks up to the PRESENT block size (64bit) can be transferred across the generalized interface. The actual size of a transfer is determined via the size_i field. Transfer sizes are specified in multiples of the native OTP block size, as listed below. + +Value of `size_i` | #Native OTP Words | Bit Slice +------------------|-------------------|------------ +2'b00 | 1 | `{word0} = data[15:0]` +2'b01 | 2 | `{word1, word0} = data[31:0]` +2'b10 | 3 | `{word2, word1, word0} = data[47:0]` +2'b11 | 4 | `{word3, word2, word1, word0} = data[63:0]` + +Responses are returned in-order via an unidirectional response interface (i.e., without back pressure capability). +Downstream logic must be able to sink the response in any case. +The response optionally carries read data, depending on whether the operation that took place was a read or not. +Also, an error signal returns a non-zero error code in case an error occurred while carrying out the OTP command. + +The signals pertaining to the generalized open-source interface are listed below. + +Signal | Direction | Type | Description +------------------------|------------------|-----------------------------|--------------- +`fatal_alert_o` | `output` | `logic` | Fatal alert output from the primitive. This is connected to a separate alert channel in the instantiating IP. The instantiating IP latches the alert indication and continuously outputs alert events until reset. +`recov_alert_o` | `output` | `logic` | Recoverable alert output from the primitive. This is connected to a separate alert channel in the instantiating IP. Should only be pulsed high for each alert occurrence. The instantiating IP then sends out a single alert event for each pulse. +`ready_o` | `output` | `logic` | Ready signal for the command handshake. +`valid_i` | `input` | `logic` | Valid signal for the command handshake. +`size_i` | `input` | `logic [SizeWidth-1:0]` | Number of native OTP words to transfer, minus one: `2'b00 = 1 native word` ... `2'b11 = 4 native words`. +`cmd_i` | `input` | `logic [CmdWidth-1:0]` | OTP command: `7'b1000101 = read`, `7'b0110111 = write`, `7'b1111001 = read raw`, `7'b1100010 = write raw`, `7'b0101100 = initialize` +`addr_i` | `input` | `logic [$clog2(Depth)-1:0]` | OTP word address. +`wdata_i` | `input` | `logic [IfWidth-1:0]` | Write data for write commands. +`valid_o` | `output` | `logic` | Valid signal for command response. +`rdata_o` | `output` | `logic [IfWidth-1:0]` | Read data from read commands. +`err_o` | `output` | `logic [ErrWidth-1:0]` | Error code. + +The `write raw` and `read raw` command instructs the `prim_otp` wrapper to store / read the data in raw format without generating nor checking integrity information. +That means that the wrapper must return the raw, uncorrected data and no integrity errors. + +The `prim_otp` wrapper implements the `Macro*` error codes (0x0 - 0x4) defined in [OTP error handling](#error-handling). + +The timing diagram below illustrates the timing of a command. +Note that both read and write commands return a response, and each command is independent of the previously issued commands. +The latency from accepting a command to returning a response depends on the underlying OTP IP and is typically larger than 10 cycles. +The returned values depend on the command type and whether an error occurred or not. + +```wavejson +{ + signal: [ + { name: 'clk_i', wave: 'p.............' }, + { name: 'ready_o', wave: '0..10|.10.|...' , node: '...a...c'}, + { name: 'valid_i', wave: '01..0|1.0.|...' }, + { name: 'size_i', wave: '03..0|3.0.|...' }, + { name: 'cmd_i', wave: '04..0|4.0.|...' }, + { name: 'wdata_i', wave: '05..0|5.0.|...' }, + { name: 'valid_o', wave: '0....|..10|.10' , node: '........b...d'}, + { name: 'rdata_o', wave: '0....|..50|.50' }, + { name: 'err_o', wave: '0....|..40|.40' }, + ], + edge: [ + 'a~>b', + 'c~>d', + ], + head: { + text: 'Timing of an OTP command.', + }, + foot: { + text: "Cmd's are accepted in cycles 3/7, and the corresponding responses return in cycles 8/12.", + tick: 0, + } +} +``` + +Note that the open source OTP controller allows up to two outstanding OTP commands, meaning that it is permissible to acknowledge an incoming command and start working on it while the results of the last command are still in the process of being output (e.g., due to an output register stage). + +#### Generic Simulation and FPGA Emulation Model + +For open-source simulation and FPGA emulation, a synthesizable and generic OTP wrapper module is provided (`prim_generic_otp`). +This is automatically selected in the OpenTitan build flow via the technology primitive mechanism if no proprietary OTP IP is available for a specific technology. +The OTP storage in `prim_generic_otp` is emulated using a standard RAM primitive `prim_generic_ram_1p`. +While this storage element is volatile, the primitive is constructed such that the contents are not wiped upon a system-wide reset. +I.e., only a power-cycle wipes the RAM primitive, thereby enabling limited emulation of the OTP function and life cycle transitions also on an FPGA device. diff --git a/hw/ip/otp_ctrl/dv/README.md b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/README.md similarity index 76% rename from hw/ip/otp_ctrl/dv/README.md rename to hw/top_earlgrey/ip_autogen/otp_ctrl/dv/README.md index 23b6ec64751dfe..9be5f533377d4f 100644 --- a/hw/ip/otp_ctrl/dv/README.md +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/README.md @@ -8,32 +8,32 @@ * Verify TileLink device protocol compliance with an SVA based testbench ## Current status -* [Design & verification stage](../../../README.md) - * [HW development stages](../../../../doc/project_governance/development_stages.md) -* [Simulation results](https://reports.opentitan.org/hw/ip/otp_ctrl/dv/latest/report.html) +* [Design & verification stage](../../../../README.md) + * [HW development stages](../../../../../doc/project_governance/development_stages.md) +* [Simulation results](https://reports.opentitan.org/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/latest/report.html) ## Design features For detailed information on OTP_CTRL design features, please see the [OTP_CTRL HW IP technical specification](../README.md). ## Testbench architecture -OTP_CTRL testbench has been constructed based on the [CIP testbench architecture](../../../dv/sv/cip_lib/README.md). +OTP_CTRL testbench has been constructed based on the [CIP testbench architecture](../../../../dv/sv/cip_lib/README.md). ### Block diagram ![Block diagram](./doc/tb.svg) ### Top level testbench -Top level testbench is located at `hw/ip/otp_ctrl/dv/tb.sv`. It instantiates the OTP_CTRL DUT module `hw/ip/otp_ctrl/rtl/otp_ctrl.sv`. +Top level testbench is located at `hw/top_earlgrey/ip_autogen/otp_ctrl/dv/tb.sv`. It instantiates the OTP_CTRL DUT module `hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl.sv`. In addition, it instantiates the following interfaces, connects them to the DUT and sets their handle into `uvm_config_db`: -* [Clock and reset interface](../../../dv/sv/common_ifs/README.md) -* [TileLink host interface](../../../dv/sv/tl_agent/README.md) +* [Clock and reset interface](../../../../dv/sv/common_ifs/README.md) +* [TileLink host interface](../../../../dv/sv/tl_agent/README.md) * OTP_CTRL IOs -* Interrupts ([`pins_if`](../../../dv/sv/common_ifs/README.md)) -* Alerts ([`alert_esc_if`](../../../dv/sv/alert_esc_agent/README.md)) +* Interrupts ([`pins_if`](../../../../dv/sv/common_ifs/README.md)) +* Alerts ([`alert_esc_if`](../../../../dv/sv/alert_esc_agent/README.md)) ### Common DV utility components The following utilities provide generic helper tasks and functions to perform activities that are common across the project: -* [dv_utils_pkg](../../../dv/sv/dv_utils/README.md) -* [csr_utils_pkg](../../../dv/sv/csr_utils/README.md) +* [dv_utils_pkg](../../../../dv/sv/dv_utils/README.md) +* [csr_utils_pkg](../../../../dv/sv/csr_utils/README.md) ### Global types & methods All common types and methods defined at the package level can be found in @@ -56,10 +56,10 @@ All common types and methods defined at the package level can be found in ``` ### TL_agent -OTP_CTRL testbench instantiates (already handled in CIP base env) [tl_agent](../../../dv/sv/tl_agent/README.md), which provides the ability to drive and independently monitor random traffic via TL host interface into OTP_CTRL device. +OTP_CTRL testbench instantiates (already handled in CIP base env) [tl_agent](../../../../dv/sv/tl_agent/README.md), which provides the ability to drive and independently monitor random traffic via TL host interface into OTP_CTRL device. ### Alert_agents -OTP_CTRL testbench instantiates (already handled in CIP base env) two [alert_agents](../../../dv/sv/alert_esc_agent/README.md): +OTP_CTRL testbench instantiates (already handled in CIP base env) two [alert_agents](../../../../dv/sv/alert_esc_agent/README.md): fatal_check_alert and fatal_macro_alert. The alert_agents provide the ability to drive and independently monitor alert handshakes via alert interfaces in OTP_CTRL device. @@ -68,20 +68,20 @@ OTP_CTRL design has specific inputs and outputs to communicate with other IPs in This interface is created to initialize, use simple task to drive, and use assertions to monitor these signals. ### Memory backdoor interface -OTP_CTRL testbench binds design's non-volatile OTP memory with a [`mem_bkdr_util`](../../../dv/sv/mem_bkdr_util/README.md), which supports read, write, and injection of ECC errors to design's OTP memory. +OTP_CTRL testbench binds design's non-volatile OTP memory with a [`mem_bkdr_util`](../../../../dv/sv/mem_bkdr_util/README.md), which supports read, write, and injection of ECC errors to design's OTP memory. ### UVM RAL model -The OTP_CTRL RAL model is created with the [`ralgen`](../../../dv/tools/ralgen/README.md) FuseSoC generator script automatically when the simulation is at the build stage. +The OTP_CTRL RAL model is created with the [`ralgen`](../../../../dv/tools/ralgen/README.md) FuseSoC generator script automatically when the simulation is at the build stage. -It can be created manually by invoking [`regtool`](../../../../util/reggen/doc/setup_and_use.md). +It can be created manually by invoking [`regtool`](../../../../../util/reggen/doc/setup_and_use.md). ### Reference models -The OTP_CTRL's utilizes [PRESENT](../../prim/doc/prim_present.md as the cipher to scramble and protect secrets. +The OTP_CTRL's utilizes [PRESENT](../../../../ip/prim/doc/prim_present.md as the cipher to scramble and protect secrets. Thus OTP_CTRL's scoreboard adopted PRESENT's C reference model, located under `hw/ip/prim/dv/prim_present/` folder, for encryption and decryption purpose. ### Stimulus strategy #### Test sequences -All test sequences reside in `hw/ip/otp_ctrl/dv/env/seq_lib`. +All test sequences reside in `hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib`. The `otp_ctrl_base_vseq` virtual sequence is extended from `cip_base_vseq` and serves as a starting point. All test sequences are extended from `otp_ctrl_base_vseq`. It provides commonly used handles, variables, functions and tasks that the test sequences can simple use / call. @@ -130,16 +130,16 @@ If a reset or lc_escalation_en is issued during an OTP_CTRL write operation, sco To avoid mismatches, scoreboard utilizes flags `dai_wr_ip` and `dai_digest_ip` to track otp write operations, and issue a backdoor read if the write operation is interrupted. #### Assertions -* TLUL assertions: The `tb/otp_ctrl_bind.sv` binds the `tlul_assert` [assertions](../../tlul/doc/TlulProtocolChecker.md) to the IP to ensure TileLink interface protocol compliance. +* TLUL assertions: The `tb/otp_ctrl_bind.sv` binds the `tlul_assert` [assertions](../../../../ip/tlul/doc/TlulProtocolChecker.md) to the IP to ensure TileLink interface protocol compliance. * Unknown checks on DUT outputs: The RTL has assertions to ensure all outputs are initialized to known values after coming out of reset. * OTP_CTRL_IF assertions: This interface has assertions to ensure certain OTP_CTRL's outputs (such as: otp_broadcast_o, keymgr_key_o) are stable after OTP initialization. ## Building and running tests -We are using our in-house developed [regression tool](../../../../util/dvsim/README.md) for building and running our tests and regressions. +We are using our in-house developed [regression tool](../../../../../util/dvsim/README.md) for building and running our tests and regressions. Please take a look at the link for detailed information on the usage, capabilities, features and known issues. Here's how to run a smoke test: ```console -$ $REPO_TOP/util/dvsim/dvsim.py $REPO_TOP/hw/ip/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson -i otp_ctrl_smoke +$ $REPO_TOP/util/dvsim/dvsim.py $REPO_TOP/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/otp_ctrl_sim_cfg.hjson -i otp_ctrl_smoke ``` ## Testplan diff --git a/hw/ip/otp_ctrl/dv/cov/otp_ctrl_cov.core b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov.core similarity index 70% rename from hw/ip/otp_ctrl/dv/cov/otp_ctrl_cov.core rename to hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov.core index 361da46371a068..d79dccd6302c14 100644 --- a/hw/ip/otp_ctrl/dv/cov/otp_ctrl_cov.core +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov.core @@ -2,12 +2,16 @@ CAPI=2: # Copyright lowRISC contributors (OpenTitan project). # Licensed under the Apache License, Version 2.0, see LICENSE for details. # SPDX-License-Identifier: Apache-2.0 -name: "lowrisc:dv:otp_ctrl_cov" +name: lowrisc:opentitan:top_earlgrey_otp_ctrl_cov description: "OTP_CTRL functional coverage and bind files" +virtual: + - lowrisc:ip_interfaces:otp_ctrl_cov + filesets: files_rtl: depend: - - lowrisc:ip:otp_ctrl + - lowrisc:ip_interfaces:pwrmgr_pkg + - lowrisc:opentitan:top_earlgrey_otp_ctrl_top_specific_pkg files_dv: depend: diff --git a/hw/ip/otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv similarity index 97% rename from hw/ip/otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv rename to hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv index 308ee9822e3fd4..5e2caa6456f4a0 100644 --- a/hw/ip/otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_bind.sv @@ -4,9 +4,6 @@ // // Binds OTP_CTRL functional coverage interaface to the top level OTP_CTRL module. // -// DO NOT EDIT THIS FILE DIRECTLY. -// It has been generated with ./util/design/gen-otp-mmap.py - `define PART_MUBI_COV(__part_name, __index) \ bind otp_ctrl cip_mubi_cov_if #(.Width(8)) ``__part_name``_read_lock_mubi_cov_if ( \ .rst_ni (rst_ni), \ diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_fsm_unr_excl.el b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_fsm_unr_excl.el new file mode 100644 index 00000000000000..1c3412b6954185 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_fsm_unr_excl.el @@ -0,0 +1,199 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// The current UNR flow marks some transition to reset states as unreachable. +// This file manually removed the reset transition states. +//================================================== +// This file contains the Excluded objects +// Generated By User: chencindy +// Format Version: 2 +// Date: Fri Jan 6 11:14:02 2023 +// ExclMode: default +//================================================== +CHECKSUM: "2063559012 2582220255" +INSTANCE: tb.dut.gen_partitions[3].gen_buffered.u_part_buf +Fsm state_q "441128463" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +Transition IntegDigSt->IntegDigFinSt "1890->1765" +Fsm state_q "441128463" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +State InitDescrWaitSt "2472" +Fsm state_q "441128463" +State IntegScrSt "3418" +Fsm state_q "441128463" +State IntegScrWaitSt "2207" +Fsm state_q "441128463" +State InitDescrSt "3204" +Fsm error_q "2410907799" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2410907799" +Transition CheckFailError->FsmStateError "6->7" +CHECKSUM: "2063559012 2582220255" +INSTANCE: tb.dut.gen_partitions[4].gen_buffered.u_part_buf +Fsm state_q "441128463" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +Transition IntegDigClrSt->IntegDigSt "2625->1890" +Fsm state_q "441128463" +Transition InitWaitSt->InitSt "945->3367" +Fsm state_q "441128463" +State IntegDigPadSt "855" +Fsm error_q "2410907799" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2410907799" +Transition CheckFailError->FsmStateError "6->7" +CHECKSUM: "2063559012 2582220255" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf +Fsm state_q "441128463" +Transition InitWaitSt->InitSt "945->3367" +Fsm state_q "441128463" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +Transition IntegDigClrSt->IntegDigSt "2625->1890" +Fsm state_q "441128463" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +State IntegDigPadSt "855" +Fsm error_q "2410907799" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2410907799" +Transition CheckFailError->FsmStateError "6->7" +CHECKSUM: "2063559012 2582220255" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf +Fsm state_q "441128463" +Transition InitWaitSt->InitSt "945->3367" +Fsm state_q "441128463" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +Transition IntegDigClrSt->IntegDigSt "2625->1890" +Fsm state_q "441128463" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +State IntegDigPadSt "855" +Fsm error_q "2410907799" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2410907799" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +Transition CheckFailError->FsmStateError "6->7" +CHECKSUM: "2063559012 2582220255" +INSTANCE: tb.dut.gen_partitions[7].gen_lifecycle.u_part_buf +Fsm state_q "441128463" +Transition IdleSt->IntegDigClrSt "1357->2625" +Fsm state_q "441128463" +State InitDescrWaitSt "2472" +Fsm state_q "441128463" +State IntegDigFinSt "1765" +Fsm state_q "441128463" +State IntegDigPadSt "855" +Fsm state_q "441128463" +State IntegDigSt "1890" +Fsm state_q "441128463" +State IntegDigWaitSt "2290" +Fsm state_q "441128463" +State IntegScrSt "3418" +Fsm state_q "441128463" +State IntegScrWaitSt "2207" +Fsm state_q "441128463" +State InitDescrSt "3204" +Fsm error_q "2410907799" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2410907799" +Transition CheckFailError->FsmStateError "6->7" +CHECKSUM: "2940612991 27342893" +INSTANCE: tb.dut.gen_partitions[0].gen_unbuffered.u_part_unbuf +Fsm error_q "2210720134" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "2210720134" +Transition CheckFailError->AccessError "6->5" +Fsm error_q "2210720134" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2210720134" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2210720134" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2210720134" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2210720134" +Transition MacroEccCorrError->AccessError "2->5" +Fsm error_q "2210720134" +Transition AccessError->CheckFailError "5->6" +CHECKSUM: "2940612991 27342893" +INSTANCE: tb.dut.gen_partitions[1].gen_unbuffered.u_part_unbuf +Fsm error_q "2210720134" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "2210720134" +Transition CheckFailError->AccessError "6->5" +Fsm error_q "2210720134" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2210720134" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2210720134" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2210720134" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2210720134" +Transition MacroEccCorrError->AccessError "2->5" +Fsm error_q "2210720134" +Transition AccessError->CheckFailError "5->6" +CHECKSUM: "2940612991 27342893" +INSTANCE: tb.dut.gen_partitions[2].gen_unbuffered.u_part_unbuf +Fsm error_q "2210720134" +Transition CheckFailError->AccessError "6->5" +Fsm error_q "2210720134" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2210720134" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2210720134" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2210720134" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2210720134" +Transition MacroEccCorrError->AccessError "2->5" +Fsm error_q "2210720134" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "2210720134" +Transition AccessError->CheckFailError "5->6" +CHECKSUM: "4205406832 4258846959" +INSTANCE: tb.dut.u_otp_ctrl_dai +Fsm error_q "1085514286" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "1085514286" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "1085514286" +Transition AccessError->MacroEccCorrError "5->2" +CHECKSUM: "761735614 2379312231" +INSTANCE: tb.dut.u_otp_ctrl_kdi +Fsm state_q "2979668442" +Transition DigWaitSt->FinishSt "913->760" +Fsm state_q "2979668442" +Transition DigWaitSt->DigLoadSt "913->183" diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_if.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_if.sv new file mode 100644 index 00000000000000..106178b88066d2 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_if.sv @@ -0,0 +1,113 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Implements functional coverage for OTP_CTRL + +interface otp_ctrl_cov_if + import otp_ctrl_reg_pkg::*; + ( + input pwrmgr_pkg::pwr_otp_rsp_t pwr_otp_o, + input otp_ctrl_pkg::lc_otp_program_req_t lc_otp_program_i, + input bit [3:0] lc_escalate_en_i, + input otp_ctrl_pkg::flash_otp_key_req_t flash_otp_key_i, + input otp_ctrl_pkg::sram_otp_key_req_t [NumSramKeyReqSlots-1:0] sram_otp_key_i, + input otp_ctrl_pkg::otbn_otp_key_req_t otbn_otp_key_i + ); + + import uvm_pkg::*; + import dv_utils_pkg::*; + `include "dv_fcov_macros.svh" + + covergroup lc_esc_en_condition_cg @(lc_escalate_en_i == lc_ctrl_pkg::On); + lc_esc_during_flash_data_req: coverpoint flash_otp_key_i.data_req; + lc_esc_during_flash_addr_req: coverpoint flash_otp_key_i.addr_req; + lc_esc_during_sram_0_req: coverpoint sram_otp_key_i[0].req; + lc_esc_during_sram_1_req: coverpoint sram_otp_key_i[1].req; + lc_esc_during_otbn_req: coverpoint otbn_otp_key_i.req; + lc_esc_during_otp_idle: coverpoint pwr_otp_o.otp_idle; + lc_esc_during_lc_otp_prog_req: coverpoint lc_otp_program_i.req; + endgroup + + covergroup flash_data_req_condition_cg @(flash_otp_key_i.data_req); + flash_data_req_during_lc_esc: coverpoint lc_escalate_en_i { + bins lc_esc_on = {lc_ctrl_pkg::On}; + bins lc_esc_off = {[0 : lc_ctrl_pkg::On-1], [lc_ctrl_pkg::On+1 : '1]}; + } + flash_data_req_during_flash_addr_req: coverpoint flash_otp_key_i.addr_req; + flash_data_req_during_sram_0_req: coverpoint sram_otp_key_i[0].req; + flash_data_req_during_sram_1_req: coverpoint sram_otp_key_i[1].req; + flash_data_req_during_otbn_req: coverpoint otbn_otp_key_i.req; + flash_data_req_during_otp_idle: coverpoint pwr_otp_o.otp_idle; + endgroup + + covergroup flash_addr_req_condition_cg @(flash_otp_key_i.addr_req); + flash_addr_req_during_lc_esc: coverpoint lc_escalate_en_i { + bins lc_esc_on = {lc_ctrl_pkg::On}; + bins lc_esc_off = {[0 : lc_ctrl_pkg::On-1], [lc_ctrl_pkg::On+1 : '1]}; + } + flash_addr_req_during_flash_data_req: coverpoint flash_otp_key_i.data_req; + flash_addr_req_during_sram_0_req: coverpoint sram_otp_key_i[0].req; + flash_addr_req_during_sram_1_req: coverpoint sram_otp_key_i[1].req; + flash_addr_req_during_otbn_req: coverpoint otbn_otp_key_i.req; + flash_addr_req_during_otp_idle: coverpoint pwr_otp_o.otp_idle; + endgroup + + covergroup sram_0_req_condition_cg @(sram_otp_key_i[0].req); + sram_0_req_during_lc_esc: coverpoint lc_escalate_en_i { + bins lc_esc_on = {lc_ctrl_pkg::On}; + bins lc_esc_off = {[0 : lc_ctrl_pkg::On-1], [lc_ctrl_pkg::On+1 : '1]}; + } + sram_0_req_during_flash_addr_req: coverpoint flash_otp_key_i.addr_req; + sram_0_req_during_flash_data_req: coverpoint flash_otp_key_i.data_req; + sram_0_req_during_sram_1_req: coverpoint sram_otp_key_i[1].req; + sram_0_req_during_otbn_req: coverpoint otbn_otp_key_i.req; + sram_0_req_during_otp_idle: coverpoint pwr_otp_o.otp_idle; + endgroup + + covergroup sram_1_req_condition_cg @(sram_otp_key_i[1].req); + sram_1_req_during_lc_esc: coverpoint lc_escalate_en_i { + bins lc_esc_on = {lc_ctrl_pkg::On}; + bins lc_esc_off = {[0 : lc_ctrl_pkg::On-1], [lc_ctrl_pkg::On+1 : '1]}; + } + sram_1_req_during_flash_addr_req: coverpoint flash_otp_key_i.addr_req; + sram_1_req_during_flash_data_req: coverpoint flash_otp_key_i.data_req; + sram_1_req_during_sram_0_req: coverpoint sram_otp_key_i[0].req; + sram_1_req_during_otbn_req: coverpoint otbn_otp_key_i.req; + sram_1_req_during_otp_idle: coverpoint pwr_otp_o.otp_idle; + endgroup + + covergroup otbn_req_condition_cg @(otbn_otp_key_i.req); + otbn_req_during_lc_esc: coverpoint lc_escalate_en_i { + bins lc_esc_on = {lc_ctrl_pkg::On}; + bins lc_esc_off = {[0 : lc_ctrl_pkg::On-1], [lc_ctrl_pkg::On+1 : '1]}; + } + otbn_req_during_flash_addr_req: coverpoint flash_otp_key_i.addr_req; + otbn_req_during_flash_data_req: coverpoint flash_otp_key_i.data_req; + otbn_req_during_sram_0_req: coverpoint sram_otp_key_i[0].req; + otbn_req_during_sram_1_req: coverpoint sram_otp_key_i[1].req; + otbn_req_during_otp_idle: coverpoint pwr_otp_o.otp_idle; + endgroup + + covergroup lc_prog_req_condition_cg @(lc_otp_program_i.req); + lc_prog_req_during_lc_esc: coverpoint lc_escalate_en_i { + bins lc_esc_on = {lc_ctrl_pkg::On}; + bins lc_esc_off = {[0 : lc_ctrl_pkg::On-1], [lc_ctrl_pkg::On+1 : '1]}; + } + lc_prog_req_during_flash_addr_req: coverpoint flash_otp_key_i.addr_req; + lc_prog_req_during_flash_data_req: coverpoint flash_otp_key_i.data_req; + lc_prog_req_during_sram_0_req: coverpoint sram_otp_key_i[0].req; + lc_prog_req_during_sram_1_req: coverpoint sram_otp_key_i[1].req; + lc_prog_req_during_otbn_req: coverpoint otbn_otp_key_i.req; + lc_prog_req_during_otp_idle: coverpoint pwr_otp_o.otp_idle; + endgroup + + `DV_FCOV_INSTANTIATE_CG(lc_esc_en_condition_cg) + `DV_FCOV_INSTANTIATE_CG(flash_data_req_condition_cg) + `DV_FCOV_INSTANTIATE_CG(flash_addr_req_condition_cg) + `DV_FCOV_INSTANTIATE_CG(sram_0_req_condition_cg) + `DV_FCOV_INSTANTIATE_CG(sram_1_req_condition_cg) + `DV_FCOV_INSTANTIATE_CG(otbn_req_condition_cg) + `DV_FCOV_INSTANTIATE_CG(lc_prog_req_condition_cg) + +endinterface diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el new file mode 100644 index 00000000000000..99dfe4f72ff2c9 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cov_unr_excl.el @@ -0,0 +1,6133 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Generated UNR file from Synopsys UNR tool with security modules being black-boxed. +//================================================== +// This file contains the Excluded objects +// Generated By User: miguelosorio +// Format Version: 2 +// Date: Fri Aug 30 23:42:43 2024 +// ExclMode: default +//================================================== +CHECKSUM: "2868806991 3096942133" +INSTANCE: tb.dut.gen_alert_tx[4].u_prim_alert_sender +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 alert_ack_o "logic alert_ack_o" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 alert_ack_o "logic alert_ack_o" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 alert_state_o "logic alert_state_o" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 alert_state_o "logic alert_state_o" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[0].gen_unbuffered.u_part_unbuf.gen_ecc_reg.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[1].gen_unbuffered.u_part_unbuf.gen_ecc_reg.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[2].gen_unbuffered.u_part_unbuf.gen_ecc_reg.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[3].gen_unbuffered.u_part_unbuf.gen_ecc_reg.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[4].gen_unbuffered.u_part_unbuf.gen_ecc_reg.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[4].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[1].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[2].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[3].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[5].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[6].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[7].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[8].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[1].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[1].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[2].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[3].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[4].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[1].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[2].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[3].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[4].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[5].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[6].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[7].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[8].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[9].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[10].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[1].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[2].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[3].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[4].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[5].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[6].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[7].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[8].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[9].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[10].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[1].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[2].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[3].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[4].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[5].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[6].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[7].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[8].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[9].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "1436819047 249500095" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf.u_otp_ctrl_ecc_reg.gen_ecc_dec[10].u_prim_secded_inv_72_64_dec +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [0] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [1] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [2] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [3] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [4] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [5] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [6] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 syndrome_o [7] "logic syndrome_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [0] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 err_o [1] "logic err_o[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 err_o [1] "logic err_o[1:0]" +CHECKSUM: "3215070453 3446030929" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_reqfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Block 19 "4019242409" "wptr_wrap_cnt_q <= (wptr_wrap_cnt_q + {{(WrapPtrW - 1) {1'b0}}, 1'b1});" +ANNOTATION: "VC_COV_UNR" +Block 28 "1113085816" "rptr_wrap_cnt_q <= (rptr_wrap_cnt_q + {{(WrapPtrW - 1) {1'b0}}, 1'b1});" +CHECKSUM: "3215070453 3446030929" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_sramreqfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Block 19 "4019242409" "wptr_wrap_cnt_q <= (wptr_wrap_cnt_q + {{(WrapPtrW - 1) {1'b0}}, 1'b1});" +ANNOTATION: "VC_COV_UNR" +Block 28 "1113085816" "rptr_wrap_cnt_q <= (rptr_wrap_cnt_q + {{(WrapPtrW - 1) {1'b0}}, 1'b1});" +CHECKSUM: "3215070453 3446030929" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_rspfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Block 19 "4019242409" "wptr_wrap_cnt_q <= (wptr_wrap_cnt_q + {{(WrapPtrW - 1) {1'b0}}, 1'b1});" +ANNOTATION: "VC_COV_UNR" +Block 28 "1113085816" "rptr_wrap_cnt_q <= (rptr_wrap_cnt_q + {{(WrapPtrW - 1) {1'b0}}, 1'b1});" +CHECKSUM: "3171246264 1537087436" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic +ANNOTATION: "VC_COV_UNR" +Block 24 "3494210324" ";" +CHECKSUM: "3665351474 3190968676" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Block 22 "662519215" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 23 "617613755" "state_d = InitDescrWaitSt;" +ANNOTATION: "VC_COV_UNR" +Block 25 "2942482414" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 26 "3475289755" "state_d = InitSt;" +ANNOTATION: "VC_COV_UNR" +Block 68 "746922678" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 69 "369318088" "state_d = IntegScrWaitSt;" +ANNOTATION: "VC_COV_UNR" +Block 71 "1545788457" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 72 "2540127236" "state_d = IntegDigSt;" +ANNOTATION: "VC_COV_UNR" +Block 78 "4176855572" "state_d = IntegDigPadSt;" +ANNOTATION: "VC_COV_UNR" +Block 86 "1288945484" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 87 "984470440" "state_d = IntegDigFinSt;" +ANNOTATION: "VC_COV_UNR" +Block 105 "1192338528" "state_d = ErrorSt;" +ANNOTATION: "VC_COV_UNR" +Block 106 "2515805717" "error_d = CheckFailError;" +CHECKSUM: "3665351474 328877407" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Block 22 "662519215" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 23 "617613755" "state_d = InitDescrWaitSt;" +ANNOTATION: "VC_COV_UNR" +Block 25 "2942482414" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 26 "3475289755" "state_d = InitSt;" +ANNOTATION: "VC_COV_UNR" +Block 68 "746922678" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 69 "369318088" "state_d = IntegScrWaitSt;" +ANNOTATION: "VC_COV_UNR" +Block 71 "1545788457" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 72 "2540127236" "state_d = IntegDigSt;" +ANNOTATION: "VC_COV_UNR" +Block 77 "3141955456" "scrmbl_cmd_o = Digest;" +ANNOTATION: "VC_COV_UNR" +Block 105 "1192338528" "state_d = ErrorSt;" +ANNOTATION: "VC_COV_UNR" +Block 106 "2515805717" "error_d = CheckFailError;" +CHECKSUM: "3665351474 4034042967" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Block 78 "4176855572" "state_d = IntegDigPadSt;" +ANNOTATION: "VC_COV_UNR" +Block 86 "1288945484" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 87 "984470440" "state_d = IntegDigFinSt;" +ANNOTATION: "VC_COV_UNR" +Block 105 "1192338528" "state_d = ErrorSt;" +ANNOTATION: "VC_COV_UNR" +Block 106 "2515805717" "error_d = CheckFailError;" +CHECKSUM: "3665351474 2385925532" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Block 78 "4176855572" "state_d = IntegDigPadSt;" +ANNOTATION: "VC_COV_UNR" +Block 86 "1288945484" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 87 "984470440" "state_d = IntegDigFinSt;" +ANNOTATION: "VC_COV_UNR" +Block 105 "1192338528" "state_d = ErrorSt;" +ANNOTATION: "VC_COV_UNR" +Block 106 "2515805717" "error_d = CheckFailError;" +CHECKSUM: "3665351474 2380508828" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Block 78 "4176855572" "state_d = IntegDigPadSt;" +ANNOTATION: "VC_COV_UNR" +Block 86 "1288945484" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 87 "984470440" "state_d = IntegDigFinSt;" +ANNOTATION: "VC_COV_UNR" +Block 105 "1192338528" "state_d = ErrorSt;" +ANNOTATION: "VC_COV_UNR" +Block 106 "2515805717" "error_d = CheckFailError;" +CHECKSUM: "3665351474 3973220603" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf +ANNOTATION: "VC_COV_UNR" +Block 22 "662519215" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 25 "2942482414" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 68 "746922678" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 71 "1545788457" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 74 "736744825" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 86 "1288945484" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 89 "967932616" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 92 "3253244488" "scrmbl_mtx_req_o = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 105 "1192338528" "state_d = ErrorSt;" +ANNOTATION: "VC_COV_UNR" +Block 106 "2515805717" "error_d = CheckFailError;" +CHECKSUM: "3882079776 1965827338" +INSTANCE: tb.dut.u_otp_ctrl_scrmbl +ANNOTATION: "VC_COV_UNR" +Block 28 "3494210324" ";" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field3 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field4 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field5 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field6 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field7 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field8 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field2 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field3 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field4 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field5 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field0 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field1 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field2 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "4255502330 223073768" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field3 +ANNOTATION: "VC_COV_UNR" +Block 4 "1824183207" "q <= wr_data;" +CHECKSUM: "3162909804 919553166" +INSTANCE: tb.dut.gen_partitions[0].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Block 17 "814628507" "error_d = MacroEccCorrError;" +ANNOTATION: "VC_COV_UNR" +Block 32 "3415759745" "error_d = MacroEccCorrError;" +CHECKSUM: "3162909804 2229302740" +INSTANCE: tb.dut.gen_partitions[4].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Block 44 "1192338528" "state_d = ErrorSt;" +ANNOTATION: "VC_COV_UNR" +Block 45 "2515805717" "error_d = CheckFailError;" +CHECKSUM: "662936270 450589591" +INSTANCE: tb.dut.u_tlul_adapter_sram +ANNOTATION: "VC_COV_UNR" +Block 22 "3478134645" "d_valid = 1'b1;" +CHECKSUM: "3436844037 1915699783" +INSTANCE: tb.dut +ANNOTATION: "VC_COV_UNR" +Block 15 "1636116421" "tlul_oob_err_d = 1'b1;" +ANNOTATION: "VC_COV_UNR" +Block 32 "3434355351" "part_access_pre[k] = {2 {MuBi8True}};" +CHECKSUM: "1158524476 3747170265" +INSTANCE: tb.dut.u_otp_ctrl_kdi +ANNOTATION: "VC_COV_UNR" +Block 88 "3038555774" "state_d = DigLoadSt;" +CHECKSUM: "1611327958 113940473" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top +ANNOTATION: "VC_COV_UNR" +Condition 20 "3585319611" "(reg_we && ((!addrmiss))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 22 "1026062099" "(addrmiss | wr_err | intg_err) 1 -1" (4 "100") +CHECKSUM: "3171246264 2882888745" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic +ANNOTATION: "VC_COV_UNR" +Condition 1 "2104830463" "(cmd_i == Init) 1 -1" (1 "0") +CHECKSUM: "3162909804 3458814989" +INSTANCE: tb.dut.gen_partitions[1].gen_unbuffered.u_part_unbuf +Fsm state_q "4141872371" +ANNOTATION: "VC_COV_UNR" +Transition ResetSt->IdleSt "694->745" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->CheckFailError "5->6" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->AccessError "6->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition MacroEccCorrError->AccessError "2->5" +CHECKSUM: "3162909804 3458814989" +INSTANCE: tb.dut.gen_partitions[2].gen_unbuffered.u_part_unbuf +Fsm state_q "4141872371" +ANNOTATION: "VC_COV_UNR" +Transition ResetSt->IdleSt "694->745" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->CheckFailError "5->6" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->AccessError "6->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition MacroEccCorrError->AccessError "2->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition MacroEccCorrError->CheckFailError "2->6" +CHECKSUM: "3162909804 3458814989" +INSTANCE: tb.dut.gen_partitions[3].gen_unbuffered.u_part_unbuf +Fsm state_q "4141872371" +ANNOTATION: "VC_COV_UNR" +Transition ResetSt->IdleSt "694->745" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->CheckFailError "5->6" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->AccessError "6->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition MacroEccCorrError->AccessError "2->5" +CHECKSUM: "3162909804 3458814989" +INSTANCE: tb.dut.gen_partitions[4].gen_unbuffered.u_part_unbuf +Fsm state_q "4141872371" +ANNOTATION: "VC_COV_UNR" +Transition ResetSt->IdleSt "694->745" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +State CheckFailError "6" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition MacroEccCorrError->AccessError "2->5" +CHECKSUM: "3436844037 2267239089" +INSTANCE: tb.dut +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_ast_pwr_seq_o.pwr_seq [0] "logic otp_ast_pwr_seq_o.pwr_seq[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_ast_pwr_seq_o.pwr_seq [0] "logic otp_ast_pwr_seq_o.pwr_seq[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_ast_pwr_seq_o.pwr_seq [1] "logic otp_ast_pwr_seq_o.pwr_seq[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_ast_pwr_seq_o.pwr_seq [1] "logic otp_ast_pwr_seq_o.pwr_seq[1:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [0] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [0] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [1] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [1] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [2] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [2] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [3] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [3] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [4] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [4] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [5] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [5] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [6] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [6] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [7] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [7] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [8] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [8] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [9] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [9] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [10] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [10] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [11] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [11] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [12] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [12] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [13] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [13] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [14] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [14] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [15] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [15] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [16] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [16] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [17] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [17] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [18] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [18] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [19] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [19] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [20] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [20] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [21] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [21] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [22] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [22] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [23] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [23] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [24] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [24] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [25] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [25] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [26] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [26] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [27] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [27] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [28] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [28] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [29] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [29] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [30] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [30] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 lc_otp_vendor_test_o.status [31] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 lc_otp_vendor_test_o.status [31] "logic lc_otp_vendor_test_o.status[31:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed_valid "logic otp_keymgr_key_o.owner_seed_valid" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed_valid "logic otp_keymgr_key_o.owner_seed_valid" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [0] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [0] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [1] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [1] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [2] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [2] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [3] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [3] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [4] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [4] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [5] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [5] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [6] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [6] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [7] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [7] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [8] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [8] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [9] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [9] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [10] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [10] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [11] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [11] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [12] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [12] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [13] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [13] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [14] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [14] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [15] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [15] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [16] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [16] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [17] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [17] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [18] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [18] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [19] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [19] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [20] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [20] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [21] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [21] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [22] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [22] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [23] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [23] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [24] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [24] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [25] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [25] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [26] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [26] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [27] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [27] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [28] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [28] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [29] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [29] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [30] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [30] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [31] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [31] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [32] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [32] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [33] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [33] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [34] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [34] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [35] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [35] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [36] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [36] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [37] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [37] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [38] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [38] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [39] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [39] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [40] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [40] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [41] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [41] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [42] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [42] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [43] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [43] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [44] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [44] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [45] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [45] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [46] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [46] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [47] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [47] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [48] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [48] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [49] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [49] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [50] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [50] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [51] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [51] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [52] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [52] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [53] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [53] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [54] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [54] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [55] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [55] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [56] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [56] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [57] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [57] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [58] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [58] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [59] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [59] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [60] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [60] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [61] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [61] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [62] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [62] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [63] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [63] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [64] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [64] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [65] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [65] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [66] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [66] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [67] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [67] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [68] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [68] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [69] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [69] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [70] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [70] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [71] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [71] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [72] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [72] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [73] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [73] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [74] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [74] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [75] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [75] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [76] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [76] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [77] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [77] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [78] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [78] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [79] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [79] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [80] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [80] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [81] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [81] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [82] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [82] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [83] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [83] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [84] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [84] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [85] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [85] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [86] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [86] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [87] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [87] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [88] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [88] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [89] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [89] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [90] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [90] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [91] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [91] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [92] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [92] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [93] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [93] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [94] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [94] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [95] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [95] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [96] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [96] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [97] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [97] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [98] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [98] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [99] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [99] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [100] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [100] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [101] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [101] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [102] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [102] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [103] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [103] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [104] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [104] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [105] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [105] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [106] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [106] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [107] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [107] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [108] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [108] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [109] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [109] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [110] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [110] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [111] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [111] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [112] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [112] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [113] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [113] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [114] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [114] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [115] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [115] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [116] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [116] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [117] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [117] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [118] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [118] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [119] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [119] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [120] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [120] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [121] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [121] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [122] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [122] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [123] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [123] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [124] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [124] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [125] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [125] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [126] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [126] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [127] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [127] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [128] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [128] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [129] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [129] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [130] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [130] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [131] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [131] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [132] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [132] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [133] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [133] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [134] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [134] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [135] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [135] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [136] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [136] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [137] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [137] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [138] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [138] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [139] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [139] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [140] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [140] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [141] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [141] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [142] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [142] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [143] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [143] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [144] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [144] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [145] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [145] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [146] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [146] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [147] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [147] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [148] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [148] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [149] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [149] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [150] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [150] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [151] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [151] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [152] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [152] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [153] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [153] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [154] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [154] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [155] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [155] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [156] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [156] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [157] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [157] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [158] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [158] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [159] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [159] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [160] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [160] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [161] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [161] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [162] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [162] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [163] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [163] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [164] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [164] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [165] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [165] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [166] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [166] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [167] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [167] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [168] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [168] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [169] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [169] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [170] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [170] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [171] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [171] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [172] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [172] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [173] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [173] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [174] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [174] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [175] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [175] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [176] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [176] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [177] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [177] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [178] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [178] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [179] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [179] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [180] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [180] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [181] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [181] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [182] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [182] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [183] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [183] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [184] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [184] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [185] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [185] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [186] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [186] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [187] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [187] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [188] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [188] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [189] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [189] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [190] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [190] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [191] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [191] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [192] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [192] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [193] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [193] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [194] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [194] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [195] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [195] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [196] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [196] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [197] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [197] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [198] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [198] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [199] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [199] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [200] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [200] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [201] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [201] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [202] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [202] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [203] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [203] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [204] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [204] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [205] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [205] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [206] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [206] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [207] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [207] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [208] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [208] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [209] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [209] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [210] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [210] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [211] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [211] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [212] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [212] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [213] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [213] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [214] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [214] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [215] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [215] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [216] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [216] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [217] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [217] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [218] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [218] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [219] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [219] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [220] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [220] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [221] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [221] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [222] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [222] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [223] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [223] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [224] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [224] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [225] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [225] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [226] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [226] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [227] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [227] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [228] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [228] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [229] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [229] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [230] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [230] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [231] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [231] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [232] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [232] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [233] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [233] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [234] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [234] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [235] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [235] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [236] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [236] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [237] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [237] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [238] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [238] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [239] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [239] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [240] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [240] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [241] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [241] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [242] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [242] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [243] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [243] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [244] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [244] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [245] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [245] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [246] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [246] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [247] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [247] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [248] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [248] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [249] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [249] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [250] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [250] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [251] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [251] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [252] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [252] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [253] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [253] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [254] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [254] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.owner_seed [255] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.owner_seed [255] "logic otp_keymgr_key_o.owner_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed_valid "logic otp_keymgr_key_o.creator_seed_valid" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed_valid "logic otp_keymgr_key_o.creator_seed_valid" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [0] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [0] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [1] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [1] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [2] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [2] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [3] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [3] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [4] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [4] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [5] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [5] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [6] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [6] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [7] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [7] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [8] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [8] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [9] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [9] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [10] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [10] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [11] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [11] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [12] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [12] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [13] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [13] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [14] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [14] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [15] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [15] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [16] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [16] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [17] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [17] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [18] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [18] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [19] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [19] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [20] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [20] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [21] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [21] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [22] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [22] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [23] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [23] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [24] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [24] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [25] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [25] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [26] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [26] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [27] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [27] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [28] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [28] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [29] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [29] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [30] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [30] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [31] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [31] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [32] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [32] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [33] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [33] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [34] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [34] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [35] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [35] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [36] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [36] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [37] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [37] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [38] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [38] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [39] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [39] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [40] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [40] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [41] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [41] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [42] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [42] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [43] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [43] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [44] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [44] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [45] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [45] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [46] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [46] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [47] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [47] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [48] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [48] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [49] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [49] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [50] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [50] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [51] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [51] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [52] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [52] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [53] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [53] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [54] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [54] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [55] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [55] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [56] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [56] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [57] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [57] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [58] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [58] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [59] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [59] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [60] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [60] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [61] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [61] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [62] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [62] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [63] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [63] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [64] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [64] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [65] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [65] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [66] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [66] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [67] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [67] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [68] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [68] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [69] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [69] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [70] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [70] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [71] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [71] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [72] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [72] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [73] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [73] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [74] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [74] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [75] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [75] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [76] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [76] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [77] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [77] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [78] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [78] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [79] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [79] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [80] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [80] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [81] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [81] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [82] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [82] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [83] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [83] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [84] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [84] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [85] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [85] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [86] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [86] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [87] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [87] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [88] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [88] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [89] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [89] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [90] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [90] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [91] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [91] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [92] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [92] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [93] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [93] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [94] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [94] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [95] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [95] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [96] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [96] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [97] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [97] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [98] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [98] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [99] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [99] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [100] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [100] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [101] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [101] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [102] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [102] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [103] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [103] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [104] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [104] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [105] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [105] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [106] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [106] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [107] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [107] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [108] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [108] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [109] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [109] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [110] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [110] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [111] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [111] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [112] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [112] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [113] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [113] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [114] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [114] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [115] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [115] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [116] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [116] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [117] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [117] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [118] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [118] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [119] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [119] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [120] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [120] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [121] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [121] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [122] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [122] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [123] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [123] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [124] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [124] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [125] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [125] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [126] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [126] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [127] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [127] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [128] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [128] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [129] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [129] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [130] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [130] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [131] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [131] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [132] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [132] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [133] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [133] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [134] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [134] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [135] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [135] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [136] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [136] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [137] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [137] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [138] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [138] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [139] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [139] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [140] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [140] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [141] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [141] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [142] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [142] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [143] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [143] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [144] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [144] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [145] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [145] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [146] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [146] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [147] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [147] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [148] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [148] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [149] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [149] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [150] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [150] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [151] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [151] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [152] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [152] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [153] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [153] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [154] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [154] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [155] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [155] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [156] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [156] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [157] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [157] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [158] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [158] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [159] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [159] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [160] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [160] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [161] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [161] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [162] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [162] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [163] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [163] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [164] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [164] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [165] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [165] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [166] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [166] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [167] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [167] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [168] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [168] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [169] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [169] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [170] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [170] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [171] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [171] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [172] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [172] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [173] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [173] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [174] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [174] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [175] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [175] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [176] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [176] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [177] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [177] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [178] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [178] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [179] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [179] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [180] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [180] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [181] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [181] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [182] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [182] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [183] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [183] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [184] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [184] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [185] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [185] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [186] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [186] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [187] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [187] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [188] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [188] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [189] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [189] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [190] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [190] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [191] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [191] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [192] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [192] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [193] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [193] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [194] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [194] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [195] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [195] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [196] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [196] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [197] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [197] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [198] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [198] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [199] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [199] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [200] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [200] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [201] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [201] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [202] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [202] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [203] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [203] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [204] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [204] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [205] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [205] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [206] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [206] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [207] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [207] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [208] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [208] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [209] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [209] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [210] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [210] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [211] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [211] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [212] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [212] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [213] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [213] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [214] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [214] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [215] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [215] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [216] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [216] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [217] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [217] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [218] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [218] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [219] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [219] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [220] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [220] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [221] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [221] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [222] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [222] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [223] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [223] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [224] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [224] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [225] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [225] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [226] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [226] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [227] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [227] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [228] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [228] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [229] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [229] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [230] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [230] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [231] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [231] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [232] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [232] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [233] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [233] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [234] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [234] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [235] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [235] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [236] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [236] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [237] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [237] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [238] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [238] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [239] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [239] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [240] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [240] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [241] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [241] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [242] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [242] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [243] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [243] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [244] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [244] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [245] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [245] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [246] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [246] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [247] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [247] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [248] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [248] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [249] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [249] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [250] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [250] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [251] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [251] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [252] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [252] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [253] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [253] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [254] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [254] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 otp_keymgr_key_o.creator_seed [255] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 otp_keymgr_key_o.creator_seed [255] "logic otp_keymgr_key_o.creator_seed[255:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [0] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [0] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [1] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [1] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [2] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [2] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [3] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [3] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [4] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [4] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [5] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [5] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [6] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [6] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 0to1 cio_test_o [7] "logic cio_test_o[7:0]" +ANNOTATION: "VC_COV_UNR" +Toggle 1to0 cio_test_o [7] "logic cio_test_o[7:0]" +CHECKSUM: "3665351474 2582220255" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State InitDescrSt "3204" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State InitDescrWaitSt "2472" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigPadSt "855" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegScrSt "3418" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegScrWaitSt "2207" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +CHECKSUM: "3665351474 2582220255" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State InitDescrSt "3204" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State InitDescrWaitSt "2472" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegScrSt "3418" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegScrWaitSt "2207" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigSt->IntegDigFinSt "1890->1765" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +CHECKSUM: "3665351474 2582220255" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigPadSt "855" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition InitWaitSt->InitSt "945->3367" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IntegDigSt "2625->1890" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +CHECKSUM: "3665351474 2582220255" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigPadSt "855" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition InitWaitSt->InitSt "945->3367" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IntegDigSt "2625->1890" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +CHECKSUM: "3665351474 2582220255" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigPadSt "855" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition CnstyReadWaitSt->CnstyReadSt "2684->107" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition InitWaitSt->InitSt "945->3367" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IdleSt "2625->1357" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IntegDigClrSt->IntegDigSt "2625->1890" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +CHECKSUM: "3665351474 2582220255" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State InitDescrSt "3204" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State InitDescrWaitSt "2472" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigFinSt "1765" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigPadSt "855" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigSt "1890" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegDigWaitSt "2290" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegScrSt "3418" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +State IntegScrWaitSt "2207" +Fsm state_q "441128463" +ANNOTATION: "VC_COV_UNR" +Transition IdleSt->IntegDigClrSt "1357->2625" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->MacroEccCorrError "6->2" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +Fsm error_q "2410907799" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +CHECKSUM: "3162909804 3458814989" +INSTANCE: tb.dut.gen_partitions[0].gen_unbuffered.u_part_unbuf +Fsm state_q "4141872371" +ANNOTATION: "VC_COV_UNR" +Transition ResetSt->IdleSt "694->745" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +State MacroEccCorrError "2" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->AccessError "6->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition CheckFailError->FsmStateError "6->7" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "2210720134" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->CheckFailError "7->6" +CHECKSUM: "903559179 4258846959" +INSTANCE: tb.dut.u_otp_ctrl_dai +Fsm error_q "1085514286" +ANNOTATION: "VC_COV_UNR" +Transition AccessError->MacroEccCorrError "5->2" +Fsm error_q "1085514286" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->AccessError "7->5" +Fsm error_q "1085514286" +ANNOTATION: "VC_COV_UNR" +Transition FsmStateError->MacroEccCorrError "7->2" +CHECKSUM: "1158524476 2379312231" +INSTANCE: tb.dut.u_otp_ctrl_kdi +Fsm state_q "2979668442" +ANNOTATION: "VC_COV_UNR" +Transition DigWaitSt->DigLoadSt "913->183" +CHECKSUM: "165375753 2072593586" +INSTANCE: tb.dut.u_otp_ctrl_lfsr_timer +ANNOTATION: "VC_COV_UNR" +Condition 15 "1557524712" "(edn_req_o & edn_ack_i) 1 -1" (1 "01") +CHECKSUM: "3436844037 2360113432" +INSTANCE: tb.dut +ANNOTATION: "VC_COV_UNR" +Condition 1 "83486244" "(tlul_part_sel_oh != '0) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 2 "1218027842" "(((|part_tlul_gnt)) | tlul_oob_err_q) 1 -1" (2 "01") +ANNOTATION: "VC_COV_UNR" +Condition 3 "3291433605" "(((|part_tlul_rvalid)) | tlul_oob_err_q) 1 -1" (2 "01") +ANNOTATION: "VC_COV_UNR" +Condition 4 "3779635843" "(fatal_bus_integ_error_q | ((|intg_error))) 1 -1" (3 "10") +ANNOTATION: "VC_COV_UNR" +Condition 5 "1876397257" "(part_error[k] == MacroError) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 10 "2334459673" "(otp_rvalid & otp_fifo_valid) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 13 "1319238213" "(reg2hw.direct_access_cmd.digest.qe | reg2hw.direct_access_cmd.wr.qe | reg2hw.direct_access_cmd.rd.qe) 1 -1" (2 "001") +ANNOTATION: "VC_COV_UNR" +Condition 13 "1319238213" "(reg2hw.direct_access_cmd.digest.qe | reg2hw.direct_access_cmd.wr.qe | reg2hw.direct_access_cmd.rd.qe) 1 -1" (3 "010") +ANNOTATION: "VC_COV_UNR" +Condition 13 "1319238213" "(reg2hw.direct_access_cmd.digest.qe | reg2hw.direct_access_cmd.wr.qe | reg2hw.direct_access_cmd.rd.qe) 1 -1" (4 "100") +ANNOTATION: "vcs_gen_start:k=10:vcs_gen_end:VC_COV_UNR" +Condition 42 "1479216946" "(({tlul_addr, 2'b0} >= 11'b11110101000) & ({1'b0, {tlul_addr, 2'b0}} < gen_part_sel[10].PartEnd)) 1 -1" (2 "10") +CHECKSUM: "1746381268 1271541636" +INSTANCE: tb.dut.u_reg_core.u_socket +ANNOTATION: "VC_COV_UNR" +Condition 3 "118253128" "(tl_t_o.a_valid & tl_t_i.a_ready) 1 -1" (1 "01") +CHECKSUM: "74367784 3785313510" +INSTANCE: tb.dut.u_reg_core.u_reg_if +ANNOTATION: "VC_COV_UNR" +Condition 18 "3340270436" "(addr_align_err | malformed_meta_err | tl_err | instr_error | intg_error) 1 -1" (5 "01000") +CHECKSUM: "74367784 3785313510" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_reg_if +ANNOTATION: "VC_COV_UNR" +Condition 18 "3340270436" "(addr_align_err | malformed_meta_err | tl_err | instr_error | intg_error) 1 -1" (5 "01000") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field3 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field4 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field5 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field6 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field7 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field8 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field2 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field4 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field5 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field2 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3274445021" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field3 +ANNOTATION: "VC_COV_UNR" +Condition 1 "2397158838" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3858770513" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field3 +ANNOTATION: "VC_COV_UNR" +Condition 1 "1301967206" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3858770513" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field1 +ANNOTATION: "VC_COV_UNR" +Condition 1 "1301967206" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "4255502330 3201188367" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field0 +ANNOTATION: "VC_COV_UNR" +Condition 1 "1807203824" "(wr_en ? wr_data : qs) 1 -1" (2 "1") +CHECKSUM: "2099741489 1445279304" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field0.wr_en_data_arb +ANNOTATION: "VC_COV_UNR" +Condition 1 "505266581" "(we | de) 1 -1" (2 "01") +CHECKSUM: "2099741489 1445279304" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field1.wr_en_data_arb +ANNOTATION: "VC_COV_UNR" +Condition 1 "505266581" "(we | de) 1 -1" (2 "01") +CHECKSUM: "2099741489 3636044484" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field2.wr_en_data_arb +ANNOTATION: "VC_COV_UNR" +Condition 1 "505266581" "(we | de) 1 -1" (2 "01") +ANNOTATION: "VC_COV_UNR" +Condition 2 "2306794614" "((de ? d : q) & (we ? ((~wd)) : '1)) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 2 "2306794614" "((de ? d : q) & (we ? ((~wd)) : '1)) 1 -1" (3 "11") +ANNOTATION: "VC_COV_UNR" +Condition 3 "2289961458" "(de ? d : q) 1 -1" (2 "1") +CHECKSUM: "2099741489 1283100255" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field6.wr_en_data_arb +ANNOTATION: "VC_COV_UNR" +Condition 1 "505266581" "(we | de) 1 -1" (2 "01") +CHECKSUM: "2099741489 1077956591" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field0.wr_en_data_arb +ANNOTATION: "VC_COV_UNR" +Condition 1 "505266581" "(we | de) 1 -1" (2 "01") +CHECKSUM: "2099741489 4164822555" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field1.wr_en_data_arb +ANNOTATION: "VC_COV_UNR" +Condition 1 "505266581" "(we | de) 1 -1" (2 "01") +CHECKSUM: "662936270 2081652359" +INSTANCE: tb.dut.u_tlul_adapter_sram +ANNOTATION: "VC_COV_UNR" +Condition 2 "3455933711" "(reqfifo_rdata.op == OpRead) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 8 "3066913128" "(intg_error | rsp_fifo_error | sramreqfifo_error | reqfifo_error | intg_error_q) 1 -1" (2 "00001") +ANNOTATION: "VC_COV_UNR" +Condition 17 "3469950311" "(wr_attr_error | wr_vld_error | rd_vld_error | instr_error | tlul_error | intg_error) 1 -1" (7 "100000") +ANNOTATION: "VC_COV_UNR" +Condition 20 "709191362" "(req_o & gnt_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 21 "3623514242" "(d_valid & reqfifo_rvalid & rspfifo_rvalid & (reqfifo_rdata.op == OpRead)) 1 -1" (1 "0111") +ANNOTATION: "VC_COV_UNR" +Condition 21 "3623514242" "(d_valid & reqfifo_rvalid & rspfifo_rvalid & (reqfifo_rdata.op == OpRead)) 1 -1" (2 "1011") +ANNOTATION: "VC_COV_UNR" +Condition 21 "3623514242" "(d_valid & reqfifo_rvalid & rspfifo_rvalid & (reqfifo_rdata.op == OpRead)) 1 -1" (4 "1110") +ANNOTATION: "VC_COV_UNR" +Condition 25 "2807788926" "((vld_rd_rsp && reqfifo_rdata.error) ? error_blanking_integ : (vld_rd_rsp ? rspfifo_rdata.data_intg : prim_secded_pkg::SecdedInv3932ZeroEcc)) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 26 "561780173" "(vld_rd_rsp && reqfifo_rdata.error) 1 -1" (3 "11") +ANNOTATION: "VC_COV_UNR" +Condition 34 "201396280" "(d_valid && d_error) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 35 "3680494467" "((gnt_i | missed_err_gnt_q) & reqfifo_wready & sramreqfifo_wready & sramreqaddrfifo_wready) 1 -1" (3 "1101") +ANNOTATION: "VC_COV_UNR" +Condition 37 "2164803938" "(tl_i_int.a_valid & reqfifo_wready & ((~error_internal))) 1 -1" (1 "011") +ANNOTATION: "VC_COV_UNR" +Condition 42 "2041272341" "(sram_ack & ((~we_o))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 43 "721931741" "(rvalid_i & reqfifo_rvalid) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:i=0:vcs_gen_end:VC_COV_UNR" +Condition 48 "3548937587" "(((|wmask_intg)) & ((|wdata_intg))) 1 -1" (1 "01") +CHECKSUM: "2974379282 2951929728" +INSTANCE: tb.dut.u_part_sel_idx +ANNOTATION: "vcs_gen_start:level=0,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 1 "2750612666" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[0].gen_level[0].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[0].gen_level[0].C1]) 1 -1" (1 "00") +CHECKSUM: "2974379282 2951929728" +INSTANCE: tb.dut.u_otp_ctrl_dai.u_part_sel_idx +ANNOTATION: "vcs_gen_start:level=0,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 1 "2750612666" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[0].gen_level[0].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[0].gen_level[0].C1]) 1 -1" (1 "00") +CHECKSUM: "2032872600 2493710885" +INSTANCE: tb.dut.u_edn_arb +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 10 "635964333" "(req_i[0] & gen_normal_case.prio_mask_q[0]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 11 "2124571033" "(req_i[0] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[1].gen_level[0].Pa] & ready_i) 1 -1" (1 "011") +CHECKSUM: "2032872600 3068670743" +INSTANCE: tb.dut.u_otp_arb +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 118 "3667925887" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 119 "2125455247" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 119 "2125455247" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 120 "2371949082" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[6].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[6].C1]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 121 "4111290463" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 124 "2230658126" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[6].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel))) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 125 "1884579875" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[6].Pa] & gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 137 "1731868698" "(req_i[0] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[0].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 142 "1060032545" "(req_i[1] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[1].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 147 "1299459822" "(req_i[2] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[2].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 152 "358663893" "(req_i[3] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[3].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 157 "2252512415" "(req_i[4] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[4].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 162 "3730207908" "(req_i[5] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[5].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 167 "2886503019" "(req_i[6] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[6].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=7:vcs_gen_end:VC_COV_UNR" +Condition 172 "4095238736" "(req_i[7] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[7].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=8:vcs_gen_end:VC_COV_UNR" +Condition 177 "296455364" "(req_i[8] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[8].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=9:vcs_gen_end:VC_COV_UNR" +Condition 182 "1237279999" "(req_i[9] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[9].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=10:vcs_gen_end:VC_COV_UNR" +Condition 187 "565898016" "(req_i[10] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[10].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=11:vcs_gen_end:VC_COV_UNR" +Condition 192 "2218826282" "(req_i[11] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[11].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=12:vcs_gen_end:VC_COV_UNR" +Condition 197 "4217824807" "(req_i[12] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[12].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 201 "1343278836" "(req_i[13] & gen_normal_case.prio_mask_q[13]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 201 "1343278836" "(req_i[13] & gen_normal_case.prio_mask_q[13]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 202 "1587320621" "(req_i[13] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 202 "1587320621" "(req_i[13] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ready_i) 1 -1" (2 "101") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 202 "1587320621" "(req_i[13] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 202 "1587320621" "(req_i[13] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ready_i) 1 -1" (4 "111") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 204 "3359315662" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ((~ready_i)))) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 204 "3359315662" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 205 "87484194" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ((~ready_i))) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 205 "87484194" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ((~ready_i))) 1 -1" (3 "11") +CHECKSUM: "2032872600 835765284" +INSTANCE: tb.dut.u_scrmbl_mtx +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 10 "1993460578" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[1].gen_level[0].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C1])) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 10 "1993460578" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[1].gen_level[0].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 11 "1567077924" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 12 "3158492049" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[1].gen_level[0].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[1].gen_level[0].C1]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 13 "96040801" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[1].gen_level[0].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 14 "1193252123" "(gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel ? gen_normal_case.idx_tree[gen_normal_case.gen_tree[1].gen_level[0].C1] : gen_normal_case.idx_tree[gen_normal_case.gen_tree[1].gen_level[0].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 15 "445016888" "(gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel ? gen_normal_case.data_tree[gen_normal_case.gen_tree[1].gen_level[0].C1] : gen_normal_case.data_tree[gen_normal_case.gen_tree[1].gen_level[0].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 16 "1314162940" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[1].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel))) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 16 "1314162940" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[1].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 17 "2175331913" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[1].gen_level[0].Pa] & gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 18 "2134997267" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[1].gen_level[0].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[1].gen_level[0].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 28 "2070917326" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[2].gen_level[0].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C1])) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 28 "2070917326" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[2].gen_level[0].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 29 "3264610632" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 29 "3264610632" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C1]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 30 "1496330476" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[2].gen_level[0].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[2].gen_level[0].C1]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 30 "1496330476" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[2].gen_level[0].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[2].gen_level[0].C1]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 31 "475390859" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 31 "475390859" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[2].gen_level[0].C0]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 32 "3851361741" "(gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel ? gen_normal_case.idx_tree[gen_normal_case.gen_tree[2].gen_level[0].C1] : gen_normal_case.idx_tree[gen_normal_case.gen_tree[2].gen_level[0].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 33 "834897871" "(gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel ? gen_normal_case.data_tree[gen_normal_case.gen_tree[2].gen_level[0].C1] : gen_normal_case.data_tree[gen_normal_case.gen_tree[2].gen_level[0].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 34 "827710109" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[2].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel))) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 34 "827710109" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[2].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel))) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 34 "827710109" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[2].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 35 "1296065718" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[2].gen_level[0].Pa] & gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 35 "1296065718" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[2].gen_level[0].Pa] & gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 36 "2587069038" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[2].gen_level[0].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[2].gen_level[0].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=2,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 45 "1389557192" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[2].gen_level[1].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[2].gen_level[1].C0]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 64 "210767427" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[0].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C1])) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 64 "210767427" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[0].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 65 "433071849" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 65 "433071849" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C1]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 66 "676485814" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[0].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[0].C1]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 66 "676485814" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[0].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[0].C1]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 67 "2859603461" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 67 "2859603461" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[0].C0]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 68 "499725081" "(gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel ? gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[0].C1] : gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[0].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 69 "2136469291" "(gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel ? gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[0].C1] : gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[0].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 70 "3457930055" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel))) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 70 "3457930055" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel))) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 70 "3457930055" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[0].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 71 "1202012" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[0].Pa] & gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 71 "1202012" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[0].Pa] & gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 72 "3948011572" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[3].gen_level[0].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[0].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 73 "2058222911" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[1].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C1])) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 73 "2058222911" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[1].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 74 "3215638766" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 74 "3215638766" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C1]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 75 "3769836816" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[1].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[1].C1]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 75 "3769836816" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[1].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[1].C1]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 76 "2034837615" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 76 "2034837615" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[1].C0]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 77 "2239186227" "(gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel ? gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[1].C1] : gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[1].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 78 "2004432906" "(gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel ? gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[1].C1] : gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[1].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 79 "1357880086" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel))) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 79 "1357880086" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel))) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 79 "1357880086" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 80 "3927989974" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] & gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 80 "3927989974" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] & gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 81 "598777746" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 81 "598777746" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].C0]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 82 "3219580251" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[2].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C1])) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 82 "3219580251" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[2].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 83 "26276459" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 84 "1842145640" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[2].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[2].C1]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 85 "147158886" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[2].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 86 "2038251222" "(gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel ? gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[2].C1] : gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[2].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 87 "1520563742" "(gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel ? gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[2].C1] : gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[2].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 88 "2599996672" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[2].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel))) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 88 "2599996672" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[2].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 89 "193340754" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[2].Pa] & gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 90 "2932823018" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[3].gen_level[2].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[2].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 90 "2932823018" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[3].gen_level[2].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[2].C0]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 109 "535495451" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[5].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C1])) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 109 "535495451" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[5].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 110 "3231051018" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 111 "2023010" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[5].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[5].C1]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 112 "2223050582" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[5].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 113 "1661998521" "(gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel ? gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[5].C1] : gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[5].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 114 "328876404" "(gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel ? gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[5].C1] : gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[5].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 115 "1323463256" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[5].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel))) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 115 "1323463256" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[5].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 116 "2448561063" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[5].Pa] & gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 117 "3273483488" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[3].gen_level[5].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[5].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 118 "3667925887" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1])) 1 -1" (1 "00") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 118 "3667925887" "(((~gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) | (((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1])) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 119 "2125455247" "(((~gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0])) & gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 120 "2371949082" "(gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[6].C0] | gen_normal_case.req_tree[gen_normal_case.gen_tree[3].gen_level[6].C1]) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 121 "4111290463" "(gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C1] | gen_normal_case.prio_tree[gen_normal_case.gen_tree[3].gen_level[6].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 122 "2669309532" "(gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel ? gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[6].C1] : gen_normal_case.idx_tree[gen_normal_case.gen_tree[3].gen_level[6].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 123 "1044549472" "(gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel ? gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[6].C1] : gen_normal_case.data_tree[gen_normal_case.gen_tree[3].gen_level[6].C0]) 1 -1" (1 "0") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 124 "2230658126" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[6].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel))) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 124 "2230658126" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[6].Pa] & ((~gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 125 "1884579875" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[6].Pa] & gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 126 "1315088536" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[3].gen_level[6].Pa] | gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[6].C0]) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 136 "2430171309" "(req_i[0] & gen_normal_case.prio_mask_q[0]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 136 "2430171309" "(req_i[0] & gen_normal_case.prio_mask_q[0]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 136 "2430171309" "(req_i[0] & gen_normal_case.prio_mask_q[0]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 137 "1731868698" "(req_i[0] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[0].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 139 "4029129651" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[0].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[0].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 140 "2870412309" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[0].Pa] & ((~ready_i))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 141 "2424159075" "(req_i[1] & gen_normal_case.prio_mask_q[1]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 141 "2424159075" "(req_i[1] & gen_normal_case.prio_mask_q[1]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 141 "2424159075" "(req_i[1] & gen_normal_case.prio_mask_q[1]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 142 "1060032545" "(req_i[1] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[1].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 144 "2044991706" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[1].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[1].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 144 "2044991706" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[1].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[1].Pa] & ((~ready_i)))) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 145 "3094700199" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[1].Pa] & ((~ready_i))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 146 "2442198833" "(req_i[2] & gen_normal_case.prio_mask_q[2]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 146 "2442198833" "(req_i[2] & gen_normal_case.prio_mask_q[2]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 146 "2442198833" "(req_i[2] & gen_normal_case.prio_mask_q[2]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 147 "1299459822" "(req_i[2] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[2].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 149 "303405805" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[2].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[2].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 149 "303405805" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[2].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[2].Pa] & ((~ready_i)))) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 150 "3088640414" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[2].Pa] & ((~ready_i))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 151 "2436119807" "(req_i[3] & gen_normal_case.prio_mask_q[3]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 151 "2436119807" "(req_i[3] & gen_normal_case.prio_mask_q[3]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 151 "2436119807" "(req_i[3] & gen_normal_case.prio_mask_q[3]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 152 "358663893" "(req_i[3] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[3].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 154 "2614501764" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[3].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[3].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 154 "2614501764" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[3].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[3].Pa] & ((~ready_i)))) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 155 "2876935468" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[3].Pa] & ((~ready_i))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 156 "1150101804" "(req_i[4] & gen_normal_case.prio_mask_q[4]) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 156 "1150101804" "(req_i[4] & gen_normal_case.prio_mask_q[4]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 156 "1150101804" "(req_i[4] & gen_normal_case.prio_mask_q[4]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 157 "2252512415" "(req_i[4] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[4].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 159 "2077403342" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[4].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[4].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 159 "2077403342" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[4].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[4].Pa] & ((~ready_i)))) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 160 "362208751" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[4].Pa] & ((~ready_i))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 164 "4061205415" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[5].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[5].Pa] & ((~ready_i)))) 1 -1" (3 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=10:vcs_gen_end:VC_COV_UNR" +Condition 186 "1702269588" "(req_i[10] & gen_normal_case.prio_mask_q[10]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=10:vcs_gen_end:VC_COV_UNR" +Condition 186 "1702269588" "(req_i[10] & gen_normal_case.prio_mask_q[10]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=10:vcs_gen_end:VC_COV_UNR" +Condition 187 "565898016" "(req_i[10] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[10].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=10:vcs_gen_end:VC_COV_UNR" +Condition 189 "2771416427" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[10].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[10].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=10:vcs_gen_end:VC_COV_UNR" +Condition 190 "2876058957" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[10].Pa] & ((~ready_i))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=12:vcs_gen_end:VC_COV_UNR" +Condition 196 "3119114945" "(req_i[12] & gen_normal_case.prio_mask_q[12]) 1 -1" (2 "10") +ANNOTATION: "vcs_gen_start:level=4,offset=12:vcs_gen_end:VC_COV_UNR" +Condition 196 "3119114945" "(req_i[12] & gen_normal_case.prio_mask_q[12]) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=12:vcs_gen_end:VC_COV_UNR" +Condition 197 "4217824807" "(req_i[12] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[12].Pa] & ready_i) 1 -1" (3 "110") +ANNOTATION: "vcs_gen_start:level=4,offset=12:vcs_gen_end:VC_COV_UNR" +Condition 199 "2102103111" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[12].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[12].Pa] & ((~ready_i)))) 1 -1" (2 "01") +ANNOTATION: "vcs_gen_start:level=4,offset=12:vcs_gen_end:VC_COV_UNR" +Condition 200 "3779810229" "(gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[12].Pa] & ((~ready_i))) 1 -1" (3 "11") +ANNOTATION: "vcs_gen_start:level=4,offset=13:vcs_gen_end:VC_COV_UNR" +Condition 204 "3359315662" "(gen_normal_case.mask_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] | (gen_normal_case.sel_tree[gen_normal_case.gen_tree[4].gen_level[13].Pa] & ((~ready_i)))) 1 -1" (1 "00") +CHECKSUM: "2032872600 3109464092" +INSTANCE: tb.dut.u_otp_ctrl_kdi.u_req_arb +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Condition 65 "2951635521" "(req_i[0] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[0].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Condition 70 "4160391802" "(req_i[1] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[1].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Condition 75 "2241885365" "(req_i[2] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[2].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=3,offset=3:vcs_gen_end:VC_COV_UNR" +Condition 80 "3719601294" "(req_i[3] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[3].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=3,offset=4:vcs_gen_end:VC_COV_UNR" +Condition 85 "1318504132" "(req_i[4] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[4].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Condition 90 "377630463" "(req_i[5] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[5].Pa] & ready_i) 1 -1" (1 "011") +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Condition 95 "1691930672" "(req_i[6] & gen_normal_case.sel_tree[gen_normal_case.gen_tree[3].gen_level[6].Pa] & ready_i) 1 -1" (1 "011") +CHECKSUM: "3655552781 2309055648" +INSTANCE: tb.dut.u_prim_edn_req.u_prim_packer_fifo +ANNOTATION: "VC_COV_UNR" +Condition 12 "2853235687" "((depth_q == FullDepth) && ((!clr_q))) 1 -1" (2 "10") +CHECKSUM: "4224194069 639524789" +INSTANCE: tb.dut.u_tlul_lc_gate +ANNOTATION: "VC_COV_UNR" +Condition 1 "1380914983" "(a_ack && ((!d_ack))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 2 "2824798557" "(d_ack && ((!a_ack))) 1 -1" (2 "10") +CHECKSUM: "903559179 1044941866" +INSTANCE: tb.dut.u_otp_ctrl_dai +ANNOTATION: "VC_COV_UNR" +Condition 6 "3686409107" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 8 "4163455672" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 9 "1539831361" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "vcs_gen_start:k=10:vcs_gen_end:VC_COV_UNR" +Condition 33 "138415549" "((dai_addr_i >= 11'b11110101000) & ({1'b0, dai_addr_i} < gen_part_sel[10].PartEndInt[otp_ctrl_reg_pkg::OtpByteAddrWidth:0])) 1 -1" (2 "10") +CHECKSUM: "1158524476 2909360515" +INSTANCE: tb.dut.u_otp_ctrl_kdi +ANNOTATION: "VC_COV_UNR" +Condition 2 "1539831361" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +CHECKSUM: "3162909804 4223786199" +INSTANCE: tb.dut.gen_partitions[3].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Condition 14 "2502713177" "(({tlul_addr_q, 2'b0} >= 11'b10001111000) && ({1'b0, tlul_addr_q, 2'b0} < PartEnd)) 1 -1" (2 "10") +CHECKSUM: "3162909804 475089886" +INSTANCE: tb.dut.gen_partitions[0].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Condition 1 "1099175909" "(otp_err != NoError) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 2 "2883220586" "(otp_err != NoError) 1 -1" (2 "1") +CHECKSUM: "3162909804 1722272287" +INSTANCE: tb.dut.gen_partitions[1].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Condition 14 "478960819" "(({tlul_addr_q, 2'b0} >= 11'b00001000000) && ({1'b0, tlul_addr_q, 2'b0} < PartEnd)) 1 -1" (2 "10") +CHECKSUM: "3162909804 1775057944" +INSTANCE: tb.dut.gen_partitions[2].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Condition 14 "673734063" "(({tlul_addr_q, 2'b0} >= 11'b00110110000) && ({1'b0, tlul_addr_q, 2'b0} < PartEnd)) 1 -1" (2 "10") +CHECKSUM: "3162909804 4078376581" +INSTANCE: tb.dut.gen_partitions[4].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Condition 4 "3809681040" "(state_q != ErrorSt) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 4 "3809681040" "(state_q != ErrorSt) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 14 "677249997" "(({tlul_addr_q, 2'b0} >= 11'b11001010000) && ({1'b0, tlul_addr_q, 2'b0} < PartEnd)) 1 -1" (2 "10") +CHECKSUM: "3665351474 720525075" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (3 "11") +ANNOTATION: "VC_COV_UNR" +Condition 10 "4291765346" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (2 "1") +CHECKSUM: "3665351474 384791011" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 9 "1580211052" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (2 "1") +CHECKSUM: "3665351474 599055118" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (3 "11") +ANNOTATION: "VC_COV_UNR" +Condition 10 "4291765346" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (2 "1") +CHECKSUM: "3665351474 698462587" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 9 "1580211052" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (2 "1") +CHECKSUM: "3665351474 2302263073" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 18 "624370688" "((base_sel == DigOffset) ? DigestOffset : 11'b11110101000) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 19 "4038180897" "(base_sel == DigOffset) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 20 "705391888" "((data_sel == ScrmblData) ? scrmbl_data_i : otp_rdata_i) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 21 "1499701630" "(data_sel == ScrmblData) 1 -1" (2 "1") +CHECKSUM: "3665351474 691864715" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Condition 3 "1999344180" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 9 "1580211052" "(scrmbl_mtx_gnt_i && scrmbl_ready_i) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 16 "3809681040" "(state_q != ErrorSt) 1 -1" (2 "1") +CHECKSUM: "7115036 2825631531" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_sramreqfifo +ANNOTATION: "VC_COV_UNR" +Condition 2 "1709501387" "(((~gen_normal_fifo.empty)) & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "101") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (1 "011") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +CHECKSUM: "7115036 2432857915" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_rspfifo +ANNOTATION: "VC_COV_UNR" +Condition 2 "1709501387" "(((~gen_normal_fifo.empty)) & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "101") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (1 "011") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +ANNOTATION: "VC_COV_UNR" +Condition 6 "4208363759" "(gen_normal_fifo.fifo_empty && wvalid_i) 1 -1" (1 "01") +CHECKSUM: "7115036 3923796707" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_reqfifo +ANNOTATION: "VC_COV_UNR" +Condition 2 "1709501387" "(((~gen_normal_fifo.empty)) & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "101") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (1 "011") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +CHECKSUM: "7115036 2279662283" +INSTANCE: tb.dut.u_otp_rsp_fifo +ANNOTATION: "VC_COV_UNR" +Condition 1 "2400173860" "(((~full_o)) & ((~gen_normal_fifo.under_rst))) 1 -1" (1 "01") +ANNOTATION: "VC_COV_UNR" +Condition 2 "1709501387" "(((~gen_normal_fifo.empty)) & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (2 "101") +ANNOTATION: "VC_COV_UNR" +Condition 3 "786039886" "(wvalid_i & wready_o & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (1 "011") +ANNOTATION: "VC_COV_UNR" +Condition 4 "1324655787" "(rvalid_o & rready_i & ((~gen_normal_fifo.under_rst))) 1 -1" (3 "110") +CHECKSUM: "3215070453 33318353" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_reqfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Condition 1 "2532211833" "(incr_wptr_i & (wptr_o == 1'((Depth - 1)))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "2597027294" "(incr_rptr_i & (rptr_o == 1'((Depth - 1)))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 8 "1599734576" "((wptr_wrap_msb == rptr_wrap_msb) ? ((1'(wptr_o) - 1'(rptr_o))) : (((1'(Depth) - 1'(rptr_o)) + 1'(wptr_o)))) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 9 "446195871" "(wptr_wrap_msb == rptr_wrap_msb) 1 -1" (1 "0") +CHECKSUM: "3215070453 33318353" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_sramreqfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Condition 1 "2532211833" "(incr_wptr_i & (wptr_o == 1'((Depth - 1)))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "2597027294" "(incr_rptr_i & (rptr_o == 1'((Depth - 1)))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 8 "1599734576" "((wptr_wrap_msb == rptr_wrap_msb) ? ((1'(wptr_o) - 1'(rptr_o))) : (((1'(Depth) - 1'(rptr_o)) + 1'(wptr_o)))) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 9 "446195871" "(wptr_wrap_msb == rptr_wrap_msb) 1 -1" (1 "0") +CHECKSUM: "3215070453 33318353" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_rspfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Condition 1 "2532211833" "(incr_wptr_i & (wptr_o == 1'((Depth - 1)))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 3 "2597027294" "(incr_rptr_i & (rptr_o == 1'((Depth - 1)))) 1 -1" (2 "10") +ANNOTATION: "VC_COV_UNR" +Condition 8 "1599734576" "((wptr_wrap_msb == rptr_wrap_msb) ? ((1'(wptr_o) - 1'(rptr_o))) : (((1'(Depth) - 1'(rptr_o)) + 1'(wptr_o)))) 1 -1" (1 "0") +ANNOTATION: "VC_COV_UNR" +Condition 9 "446195871" "(wptr_wrap_msb == rptr_wrap_msb) 1 -1" (1 "0") +CHECKSUM: "3215070453 563859410" +INSTANCE: tb.dut.u_otp_rsp_fifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Condition 5 "3619927060" "(wptr_wrap_cnt_q == (rptr_wrap_cnt_q ^ {1'b1, {(WrapPtrW - 1) {1'b0}}})) 1 -1" (2 "1") +ANNOTATION: "VC_COV_UNR" +Condition 7 "4040012966" "(full_o ? (2'(Depth)) : ((wptr_wrap_msb == rptr_wrap_msb) ? ((2'(wptr_o) - 2'(rptr_o))) : (((2'(Depth) - 2'(rptr_o)) + 2'(wptr_o))))) 1 -1" (2 "1") +CHECKSUM: "3171246264 3336016746" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic +ANNOTATION: "VC_COV_UNR" +Branch 3 "1554177250" "state_q" (1) "state_q ResetSt ,1,0,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1554177250" "state_q" (8) "state_q IdleSt ,-,-,1,default,-,-,-,-,-,-,-,-" +CHECKSUM: "3436844037 1401458059" +INSTANCE: tb.dut +ANNOTATION: "VC_COV_UNR" +Branch 4 "1264131593" "tlul_req" (1) "tlul_req 1,0" +CHECKSUM: "1746381268 3161287359" +INSTANCE: tb.dut.u_reg_core.u_socket +ANNOTATION: "VC_COV_UNR" +Branch 4 "3202860295" "(!rst_ni)" (2) "(!rst_ni) 0,1,0,-" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field3 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field4 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field5 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field6 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field7 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr3_field8 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field2 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field3 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field4 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr5_field5 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field0 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field1 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field2 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "4255502330 3554514034" +INSTANCE: tb.dut.u_otp.gen_generic.u_impl_generic.u_reg_top.u_csr7_field3 +ANNOTATION: "VC_COV_UNR" +Branch 0 "3759852512" "wr_en" (0) "wr_en 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "1017474648" "(!rst_ni)" (1) "(!rst_ni) 0,1" +CHECKSUM: "662936270 1924774061" +INSTANCE: tb.dut.u_tlul_adapter_sram +ANNOTATION: "VC_COV_UNR" +Branch 2 "1058271942" "(vld_rd_rsp && reqfifo_rdata.error)" (0) "(vld_rd_rsp && reqfifo_rdata.error) 1,-" +ANNOTATION: "VC_COV_UNR" +Branch 6 "744749108" "reqfifo_rvalid" (2) "reqfifo_rvalid 1,0,0" +CHECKSUM: "2032872600 3832429488" +INSTANCE: tb.dut.u_scrmbl_mtx +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Branch 2 "1747167515" "gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel" (1) "gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=1,offset=0:vcs_gen_end:VC_COV_UNR" +Branch 3 "1747167515" "gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel" (1) "gen_normal_case.gen_tree[1].gen_level[0].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Branch 6 "3123337246" "gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel" (1) "gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=2,offset=0:vcs_gen_end:VC_COV_UNR" +Branch 7 "3123337246" "gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel" (1) "gen_normal_case.gen_tree[2].gen_level[0].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Branch 14 "840384514" "gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=0:vcs_gen_end:VC_COV_UNR" +Branch 15 "840384514" "gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[0].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Branch 16 "2813219087" "gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=1:vcs_gen_end:VC_COV_UNR" +Branch 17 "2813219087" "gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[1].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Branch 18 "2452890474" "gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=2:vcs_gen_end:VC_COV_UNR" +Branch 19 "2452890474" "gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[2].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Branch 24 "3811150440" "gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=5:vcs_gen_end:VC_COV_UNR" +Branch 25 "3811150440" "gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[5].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Branch 26 "3602118669" "gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel 0" +ANNOTATION: "vcs_gen_start:level=3,offset=6:vcs_gen_end:VC_COV_UNR" +Branch 27 "3602118669" "gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel" (1) "gen_normal_case.gen_tree[3].gen_level[6].gen_nodes.sel 0" +CHECKSUM: "4224194069 3219254590" +INSTANCE: tb.dut.u_tlul_lc_gate +ANNOTATION: "VC_COV_UNR" +Branch 2 "1850090820" "state_q" (8) "state_q StFlush ,-,-,-,0,0,-,-" +CHECKSUM: "3882079776 3692779052" +INSTANCE: tb.dut.u_otp_ctrl_scrmbl +ANNOTATION: "VC_COV_UNR" +Branch 5 "2137472258" "state_q" (8) "state_q IdleSt ,1,default,-,-,-,-,-" +CHECKSUM: "903559179 3978479804" +INSTANCE: tb.dut.u_otp_ctrl_dai +ANNOTATION: "VC_COV_UNR" +Branch 2 "2060689171" "state_q" (1) "state_q ResetSt ,1,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +CHECKSUM: "1158524476 2213598664" +INSTANCE: tb.dut.u_otp_ctrl_kdi +ANNOTATION: "VC_COV_UNR" +Branch 4 "853326673" "state_q" (20) "state_q DigWaitSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,1,-,-,-" +CHECKSUM: "3162909804 3977884699" +INSTANCE: tb.dut.gen_partitions[0].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Branch 5 "490981166" "state_q" (4) "state_q InitSt ,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "490981166" "state_q" (5) "state_q InitWaitSt ,-,-,-,1,1,1,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "490981166" "state_q" (14) "state_q ReadWaitSt ,-,-,-,-,-,-,-,-,-,1,1,1,-,-,-" +CHECKSUM: "3162909804 1847756134" +INSTANCE: tb.dut.gen_partitions[4].gen_unbuffered.u_part_unbuf +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (0) "ecc_err 1,1" +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (1) "ecc_err 1,0" +CHECKSUM: "3665351474 3964686460" +INSTANCE: tb.dut.gen_partitions[7].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (20) "state_q CnstyReadSt ,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (35) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (37) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (43) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,0,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (47) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,-,0,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (49) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (50) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (0) "ecc_err 1,1" +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (1) "ecc_err 1,0" +CHECKSUM: "3665351474 3964686460" +INSTANCE: tb.dut.gen_partitions[8].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (20) "state_q CnstyReadSt ,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (35) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (37) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (43) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,0,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (47) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,-,0,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (49) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (50) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (0) "ecc_err 1,1" +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (1) "ecc_err 1,0" +CHECKSUM: "3665351474 3964686460" +INSTANCE: tb.dut.gen_partitions[9].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (20) "state_q CnstyReadSt ,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (35) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (37) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (43) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,0,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (47) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,-,0,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (49) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "3673468110" "state_q" (50) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (0) "ecc_err 1,1" +ANNOTATION: "VC_COV_UNR" +Branch 6 "375157548" "ecc_err" (1) "ecc_err 1,0" +CHECKSUM: "3665351474 2334161493" +INSTANCE: tb.dut.gen_partitions[5].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (11) "state_q InitDescrSt ,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (12) "state_q InitDescrSt ,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (13) "state_q InitDescrWaitSt ,-,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (14) "state_q InitDescrWaitSt ,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (20) "state_q CnstyReadSt ,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (33) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (37) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (38) "state_q IntegScrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (39) "state_q IntegScrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (40) "state_q IntegScrWaitSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (41) "state_q IntegScrWaitSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (43) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,0,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (49) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (50) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "375157548" "ecc_err" (0) "ecc_err 1,1" +ANNOTATION: "VC_COV_UNR" +Branch 5 "375157548" "ecc_err" (1) "ecc_err 1,0" +CHECKSUM: "3665351474 2334161493" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (11) "state_q InitDescrSt ,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (12) "state_q InitDescrSt ,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (13) "state_q InitDescrWaitSt ,-,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (14) "state_q InitDescrWaitSt ,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (20) "state_q CnstyReadSt ,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (33) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (37) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (38) "state_q IntegScrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (39) "state_q IntegScrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (40) "state_q IntegScrWaitSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (41) "state_q IntegScrWaitSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (42) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,1,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "344890278" "state_q" (45) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,0,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 5 "375157548" "ecc_err" (0) "ecc_err 1,1" +ANNOTATION: "VC_COV_UNR" +Branch 5 "375157548" "ecc_err" (1) "ecc_err 1,0" +CHECKSUM: "3665351474 2810977924" +INSTANCE: tb.dut.gen_partitions[10].gen_lifecycle.u_part_buf +ANNOTATION: "VC_COV_UNR" +Branch 0 "2541341865" "(base_sel == DigOffset)" (0) "(base_sel == DigOffset) 1" +ANNOTATION: "VC_COV_UNR" +Branch 1 "341865418" "(data_sel == ScrmblData)" (0) "(data_sel == ScrmblData) 1" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (12) "state_q InitDescrSt ,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (14) "state_q InitDescrWaitSt ,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (33) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,1,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (35) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (37) "state_q IntegDigClrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (39) "state_q IntegScrSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (41) "state_q IntegScrWaitSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (45) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,0,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (47) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,1,0,-,-,0,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (48) "state_q IntegDigSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (50) "state_q IntegDigPadSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (52) "state_q IntegDigFinSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 3 "1949926999" "state_q" (56) "state_q IntegDigWaitSt ,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,-,0,-,-,-" +ANNOTATION: "VC_COV_UNR" +Branch 4 "375157548" "ecc_err" (0) "ecc_err 1,1" +ANNOTATION: "VC_COV_UNR" +Branch 4 "375157548" "ecc_err" (1) "ecc_err 1,0" +CHECKSUM: "3818998033 3877782530" +INSTANCE: tb.dut.gen_partitions[6].gen_buffered.u_part_buf.u_otp_ctrl_ecc_reg +ANNOTATION: "VC_COV_UNR" +Branch 1 "2154824802" "(32'(addr_i) < Depth)" (2) "(32'(addr_i) < Depth) 0,-" +CHECKSUM: "3215070453 1827096802" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_reqfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Branch 0 "721764659" "full_o" (2) "full_o 0,0" +ANNOTATION: "VC_COV_UNR" +Branch 1 "2417346495" "(!rst_ni)" (3) "(!rst_ni) 0,0,0,1" +ANNOTATION: "VC_COV_UNR" +Branch 2 "456961687" "(!rst_ni)" (3) "(!rst_ni) 0,0,0,1" +CHECKSUM: "3215070453 1827096802" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_sramreqfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Branch 0 "721764659" "full_o" (2) "full_o 0,0" +ANNOTATION: "VC_COV_UNR" +Branch 1 "2417346495" "(!rst_ni)" (3) "(!rst_ni) 0,0,0,1" +ANNOTATION: "VC_COV_UNR" +Branch 2 "456961687" "(!rst_ni)" (3) "(!rst_ni) 0,0,0,1" +CHECKSUM: "3215070453 1827096802" +INSTANCE: tb.dut.u_tlul_adapter_sram.u_rspfifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Branch 0 "721764659" "full_o" (2) "full_o 0,0" +ANNOTATION: "VC_COV_UNR" +Branch 1 "2417346495" "(!rst_ni)" (3) "(!rst_ni) 0,0,0,1" +ANNOTATION: "VC_COV_UNR" +Branch 2 "456961687" "(!rst_ni)" (3) "(!rst_ni) 0,0,0,1" +CHECKSUM: "3215070453 1827096802" +INSTANCE: tb.dut.u_otp_rsp_fifo.gen_normal_fifo.u_fifo_cnt +ANNOTATION: "VC_COV_UNR" +Branch 0 "721764659" "full_o" (0) "full_o 1,-" diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cover.cfg b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cover.cfg new file mode 100644 index 00000000000000..d69a1932762301 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/cov/otp_ctrl_cover.cfg @@ -0,0 +1,14 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + + +// The modules below are preverified in FPV testbench. +// There are many conditional coverage and hard to them all. +-moduletree prim_secded_inv_72_64_dec +-moduletree prim_secded_inv_72_64_enc + +begin tgl + +module prim_secded_inv_72_64_dec + +module prim_secded_inv_72_64_enc +end diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/doc/tb.svg b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/doc/tb.svg new file mode 100644 index 00000000000000..a910d806170370 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/doc/tb.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_ast_inputs_cfg.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_ast_inputs_cfg.sv new file mode 100644 index 00000000000000..4b723d7905ff2c --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_ast_inputs_cfg.sv @@ -0,0 +1,35 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// +// Configuration values for DUT input signals +// +// +// This class randomizes values for DUT signal inputs +// and sets constraints on these values. +// +// This class will be instantiated inside otp_ctrl_env_cfg object, and will connect +// to it's otp_ctrl_vif signals and drive them each reset event +// +// The constraints can be hardened and softened as needed in +// closed-source environment. +// In order to override these constraints, please inherit this class +// and set a type override in the closed source environment + +class otp_ctrl_ast_inputs_cfg extends uvm_object; + `uvm_object_utils(otp_ctrl_ast_inputs_cfg); + `uvm_object_new + + // Group: Variables + rand otp_ast_rsp_t otp_ast_pwr_seq_h; + rand logic [otp_ctrl_pkg::OtpTestCtrlWidth-1:0] otp_vendor_test_ctrl; + rand prim_mubi_pkg::mubi4_t scanmode; + rand logic scan_en, scan_rst_n; + + // Group: Constraints + constraint dut_values_c { + otp_vendor_test_ctrl == 32'h0; + } + +endclass: otp_ctrl_ast_inputs_cfg diff --git a/hw/ip/otp_ctrl/dv/env/otp_ctrl_env.core b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.core similarity index 96% rename from hw/ip/otp_ctrl/dv/env/otp_ctrl_env.core rename to hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.core index 4b35e4c2118ab1..7e4e044ac275b4 100644 --- a/hw/ip/otp_ctrl/dv/env/otp_ctrl_env.core +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.core @@ -2,8 +2,11 @@ CAPI=2: # Copyright lowRISC contributors (OpenTitan project). # Licensed under the Apache License, Version 2.0, see LICENSE for details. # SPDX-License-Identifier: Apache-2.0 -name: "lowrisc:dv:otp_ctrl_env:0.1" +name: lowrisc:opentitan:top_earlgrey_otp_ctrl_env:0.1 description: "OTP_CTRL DV UVM environment" +virtual: + - lowrisc:ip_interfaces:otp_ctrl_env + filesets: files_dv: depend: diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.sv new file mode 100644 index 00000000000000..454d14df149ee4 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env.sv @@ -0,0 +1,106 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +class otp_ctrl_env #( + type CFG_T = otp_ctrl_env_cfg, + type COV_T = otp_ctrl_env_cov, + type VIRTUAL_SEQUENCER_T = otp_ctrl_virtual_sequencer, + type SCOREBOARD_T = otp_ctrl_scoreboard + ) + extends cip_base_env #( + .CFG_T (CFG_T), + .COV_T (COV_T), + .VIRTUAL_SEQUENCER_T(VIRTUAL_SEQUENCER_T), + .SCOREBOARD_T (SCOREBOARD_T) + ); + `uvm_component_param_utils(otp_ctrl_env #(CFG_T, COV_T, VIRTUAL_SEQUENCER_T, SCOREBOARD_T)) + + `uvm_component_new + + push_pull_agent#(.DeviceDataWidth(SRAM_DATA_SIZE)) m_sram_pull_agent[NumSramKeyReqSlots]; + push_pull_agent#(.DeviceDataWidth(OTBN_DATA_SIZE)) m_otbn_pull_agent; + push_pull_agent#(.DeviceDataWidth(FLASH_DATA_SIZE)) m_flash_addr_pull_agent; + push_pull_agent#(.DeviceDataWidth(FLASH_DATA_SIZE)) m_flash_data_pull_agent; + push_pull_agent#(.DeviceDataWidth(1), .HostDataWidth(LC_PROG_DATA_SIZE)) m_lc_prog_pull_agent; + + function void build_phase(uvm_phase phase); + super.build_phase(phase); + + // build sram-otp pull agent + for (int i = 0; i < NumSramKeyReqSlots; i++) begin + string sram_agent_name = $sformatf("m_sram_pull_agent[%0d]", i); + m_sram_pull_agent[i] = push_pull_agent#(.DeviceDataWidth(SRAM_DATA_SIZE))::type_id::create( + sram_agent_name, this); + uvm_config_db#(push_pull_agent_cfg#(.DeviceDataWidth(SRAM_DATA_SIZE)))::set(this, + $sformatf("%0s*", sram_agent_name), "cfg", cfg.m_sram_pull_agent_cfg[i]); + end + + // build otbn-otp pull agent + m_otbn_pull_agent = push_pull_agent#(.DeviceDataWidth(OTBN_DATA_SIZE))::type_id::create( + "m_otbn_pull_agent", this); + uvm_config_db#(push_pull_agent_cfg#(.DeviceDataWidth(OTBN_DATA_SIZE)))::set( + this, "m_otbn_pull_agent", "cfg", cfg.m_otbn_pull_agent_cfg); + + // build flash-otp pull agent + m_flash_addr_pull_agent = push_pull_agent#(.DeviceDataWidth(FLASH_DATA_SIZE))::type_id::create( + "m_flash_addr_pull_agent", this); + uvm_config_db#(push_pull_agent_cfg#(.DeviceDataWidth(FLASH_DATA_SIZE)))::set( + this, "m_flash_addr_pull_agent", "cfg", cfg.m_flash_addr_pull_agent_cfg); + m_flash_data_pull_agent = push_pull_agent#(.DeviceDataWidth(FLASH_DATA_SIZE))::type_id::create( + "m_flash_data_pull_agent", this); + uvm_config_db#(push_pull_agent_cfg#(.DeviceDataWidth(FLASH_DATA_SIZE)))::set( + this, "m_flash_data_pull_agent", "cfg", cfg.m_flash_data_pull_agent_cfg); + + // build lc-otp program pull agent + m_lc_prog_pull_agent = push_pull_agent#(.HostDataWidth(LC_PROG_DATA_SIZE), .DeviceDataWidth(1)) + ::type_id::create("m_lc_prog_pull_agent", this); + uvm_config_db#(push_pull_agent_cfg#(.HostDataWidth(LC_PROG_DATA_SIZE), .DeviceDataWidth(1))):: + set(this, "m_lc_prog_pull_agent", "cfg", cfg.m_lc_prog_pull_agent_cfg); + + // config mem virtual interface + if (!uvm_config_db#(mem_bkdr_util)::get(this, "", "mem_bkdr_util", cfg.mem_bkdr_util_h)) begin + `uvm_fatal(`gfn, "failed to get mem_bkdr_util from uvm_config_db") + end + + // config otp_ctrl output data virtual interface + if (!uvm_config_db#(otp_ctrl_vif)::get(this, "", "otp_ctrl_vif", cfg.otp_ctrl_vif)) begin + `uvm_fatal(`gfn, "failed to get otp_ctrl_vif from uvm_config_db") + end + + // Check if `NumPart` constant is assigned to the correct value. + `DV_CHECK(NumPart == (LifeCycleIdx + 1)) + + endfunction + + function void connect_phase(uvm_phase phase); + super.connect_phase(phase); + + // connect SRAM sequencer and analysis ports + for (int i = 0; i < NumSramKeyReqSlots; i++) begin + virtual_sequencer.sram_pull_sequencer_h[i] = m_sram_pull_agent[i].sequencer; + if (cfg.en_scb) begin + m_sram_pull_agent[i].monitor.analysis_port.connect( + scoreboard.sram_fifos[i].analysis_export); + end + end + + virtual_sequencer.otbn_pull_sequencer_h = m_otbn_pull_agent.sequencer; + virtual_sequencer.flash_addr_pull_sequencer_h = m_flash_addr_pull_agent.sequencer; + virtual_sequencer.flash_data_pull_sequencer_h = m_flash_data_pull_agent.sequencer; + virtual_sequencer.lc_prog_pull_sequencer_h = m_lc_prog_pull_agent.sequencer; + + if (cfg.en_scb) begin + m_otbn_pull_agent.monitor.analysis_port.connect(scoreboard.otbn_fifo.analysis_export); + m_flash_addr_pull_agent.monitor.analysis_port.connect( + scoreboard.flash_addr_fifo.analysis_export); + m_flash_data_pull_agent.monitor.analysis_port.connect( + scoreboard.flash_data_fifo.analysis_export); + m_lc_prog_pull_agent.monitor.analysis_port.connect(scoreboard.lc_prog_fifo.analysis_export); + end + + // connect the DUT cfg instance to the handle in the otp_ctrl_vif + this.cfg.otp_ctrl_vif.dut_cfg = this.cfg.dut_cfg; + endfunction + +endclass diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cfg.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cfg.sv new file mode 100644 index 00000000000000..52c67e8ff2d209 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cfg.sv @@ -0,0 +1,106 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +`define OTP_CLK_CONSTRAINT(FREQ_) \ + FREQ_ dist { \ + 6 :/ 2, \ + [24:25] :/ 2, \ + [26:47] :/ 1, \ + [48:50] :/ 2, \ + [51:95] :/ 1, \ + 96 :/ 1, \ + [97:99] :/ 1, \ + 100 :/ 1 \ + }; + +class otp_ctrl_env_cfg extends cip_base_env_cfg #(.RAL_T(otp_ctrl_core_reg_block)); + + // ext component cfgs + rand push_pull_agent_cfg#(.DeviceDataWidth(SRAM_DATA_SIZE)) + m_sram_pull_agent_cfg[NumSramKeyReqSlots]; + rand push_pull_agent_cfg#(.DeviceDataWidth(OTBN_DATA_SIZE)) m_otbn_pull_agent_cfg; + rand push_pull_agent_cfg#(.DeviceDataWidth(FLASH_DATA_SIZE)) m_flash_data_pull_agent_cfg; + rand push_pull_agent_cfg#(.DeviceDataWidth(FLASH_DATA_SIZE)) m_flash_addr_pull_agent_cfg; + rand push_pull_agent_cfg#(.DeviceDataWidth(1), .HostDataWidth(LC_PROG_DATA_SIZE)) + m_lc_prog_pull_agent_cfg; + + // Memory backdoor util instance for OTP. + mem_bkdr_util mem_bkdr_util_h; + + // ext interfaces + otp_ctrl_vif otp_ctrl_vif; + virtual clk_rst_if clk_rst_vif_otp_ctrl_prim_reg_block; + + bit backdoor_clear_mem; + + // Check ECC errors + otp_ecc_err_e ecc_chk_err [NumPart] = '{default:OtpNoEccErr}; + + // values for otp_ctrl_if signals connected to DUT + rand otp_ctrl_ast_inputs_cfg dut_cfg; + + // Introduce this flag to avoid close source conflict. + bit create_prim_tl_agent = 1; + + `uvm_object_utils_begin(otp_ctrl_env_cfg) + `uvm_object_utils_end + + `uvm_object_new + + constraint clk_freq_mhz_c { + `OTP_CLK_CONSTRAINT(clk_freq_mhz) + foreach (clk_freqs_mhz[i]) { + `OTP_CLK_CONSTRAINT(clk_freqs_mhz[i]) + } + } + + virtual function void initialize(bit [31:0] csr_base_addr = '1); + string prim_ral_name = "otp_ctrl_prim_reg_block"; + ral_model_names.push_back(prim_ral_name); + clk_freqs_mhz[prim_ral_name] = clk_freq_mhz; + + list_of_alerts = otp_ctrl_env_pkg::LIST_OF_ALERTS; + num_edn = 1; + tl_intg_alert_name = "fatal_bus_integ_error"; + sec_cm_alert_name = "fatal_check_error"; + + super.initialize(csr_base_addr); + + // create push_pull agent config obj + for (int i = 0; i < NumSramKeyReqSlots; i++) begin + string cfg_name = $sformatf("sram_pull_agent_cfg[%0d]", i); + m_sram_pull_agent_cfg[i] = push_pull_agent_cfg#(.DeviceDataWidth(SRAM_DATA_SIZE))::type_id + ::create(cfg_name); + m_sram_pull_agent_cfg[i].agent_type = PullAgent; + end + + m_otbn_pull_agent_cfg = push_pull_agent_cfg#(.DeviceDataWidth(OTBN_DATA_SIZE))::type_id + ::create("m_otbn_pull_agent_cfg"); + m_otbn_pull_agent_cfg.agent_type = PullAgent; + + m_flash_data_pull_agent_cfg = push_pull_agent_cfg#(.DeviceDataWidth(FLASH_DATA_SIZE))::type_id + ::create("m_flash_data_pull_agent_cfg"); + m_flash_data_pull_agent_cfg.agent_type = PullAgent; + m_flash_addr_pull_agent_cfg = push_pull_agent_cfg#(.DeviceDataWidth(FLASH_DATA_SIZE))::type_id + ::create("m_flash_addr_pull_agent_cfg"); + m_flash_addr_pull_agent_cfg.agent_type = PullAgent; + + m_lc_prog_pull_agent_cfg = push_pull_agent_cfg#(.HostDataWidth(LC_PROG_DATA_SIZE), + .DeviceDataWidth(1))::type_id::create("m_lc_prog_pull_agent_cfg"); + m_lc_prog_pull_agent_cfg.agent_type = PullAgent; + + // set num_interrupts & num_alerts + num_interrupts = ral.intr_state.get_n_used_bits(); + + // only support 1 outstanding TL items in tlul_adapter + m_tl_agent_cfg.max_outstanding_req = 1; + m_tl_agent_cfgs["otp_ctrl_prim_reg_block"].max_outstanding_req = 1; + + // create the inputs cfg instance + dut_cfg = otp_ctrl_ast_inputs_cfg::type_id::create("dut_cfg"); + endfunction + +endclass + +`undef OTP_CLK_CONSTRAINT diff --git a/hw/ip/otp_ctrl/dv/env/otp_ctrl_env_cov.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cov.sv similarity index 99% rename from hw/ip/otp_ctrl/dv/env/otp_ctrl_env_cov.sv rename to hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cov.sv index 10e7d90168fea3..1de5e44d9b949c 100644 --- a/hw/ip/otp_ctrl/dv/env/otp_ctrl_env_cov.sv +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_cov.sv @@ -1,9 +1,6 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -// DO NOT EDIT THIS FILE DIRECTLY. -// It has been generated with ./util/design/gen-otp-mmap.py - /** * Covergoups that are dependent on run-time parameters that may be available * only in build_phase can be defined here diff --git a/hw/ip/otp_ctrl/dv/env/otp_ctrl_env_pkg.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_pkg.sv similarity index 98% rename from hw/ip/otp_ctrl/dv/env/otp_ctrl_env_pkg.sv rename to hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_pkg.sv index 79026dbc95a433..2a4f9f3cbafcde 100644 --- a/hw/ip/otp_ctrl/dv/env/otp_ctrl_env_pkg.sv +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_env_pkg.sv @@ -1,9 +1,6 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -// DO NOT EDIT THIS FILE DIRECTLY. -// It has been generated with ./util/design/gen-otp-mmap.py - package otp_ctrl_env_pkg; // dep packages import uvm_pkg::*; @@ -20,6 +17,7 @@ package otp_ctrl_env_pkg; import otp_ctrl_reg_pkg::*; import otp_ctrl_pkg::*; import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; import lc_ctrl_pkg::*; import lc_ctrl_state_pkg::*; import lc_ctrl_dv_utils_pkg::*; diff --git a/hw/ip/otp_ctrl/dv/env/otp_ctrl_if.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_if.sv similarity index 99% rename from hw/ip/otp_ctrl/dv/env/otp_ctrl_if.sv rename to hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_if.sv index a2ae621cdbf538..885877aef2eb3b 100644 --- a/hw/ip/otp_ctrl/dv/env/otp_ctrl_if.sv +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_if.sv @@ -1,9 +1,6 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -// DO NOT EDIT THIS FILE DIRECTLY. -// It has been generated with ./util/design/gen-otp-mmap.py - // This interface collect the broadcast output data from OTP, // and drive input requests coming into OTP. `define ECC_REG_PATH gen_ecc_reg.u_otp_ctrl_ecc_reg.gen_ecc_dec[0].u_prim_secded_inv_72_64_dec diff --git a/hw/ip/otp_ctrl/dv/env/otp_ctrl_scoreboard.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_scoreboard.sv similarity index 99% rename from hw/ip/otp_ctrl/dv/env/otp_ctrl_scoreboard.sv rename to hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_scoreboard.sv index 81dac97e9f5b4c..7c01de130dbd61 100644 --- a/hw/ip/otp_ctrl/dv/env/otp_ctrl_scoreboard.sv +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_scoreboard.sv @@ -1,9 +1,6 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -// DO NOT EDIT THIS FILE DIRECTLY. -// It has been generated with ./util/design/gen-otp-mmap.py - class otp_ctrl_scoreboard #(type CFG_T = otp_ctrl_env_cfg) extends cip_base_scoreboard #( .CFG_T(CFG_T), diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_virtual_sequencer.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_virtual_sequencer.sv new file mode 100644 index 00000000000000..4780a806599403 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/otp_ctrl_virtual_sequencer.sv @@ -0,0 +1,19 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +class otp_ctrl_virtual_sequencer extends cip_base_virtual_sequencer #( + .CFG_T(otp_ctrl_env_cfg), + .COV_T(otp_ctrl_env_cov) + ); + `uvm_component_utils(otp_ctrl_virtual_sequencer) + + `uvm_component_new + + push_pull_sequencer#(.DeviceDataWidth(SRAM_DATA_SIZE)) sram_pull_sequencer_h[NumSramKeyReqSlots]; + push_pull_sequencer#(.DeviceDataWidth(OTBN_DATA_SIZE)) otbn_pull_sequencer_h; + push_pull_sequencer#(.DeviceDataWidth(FLASH_DATA_SIZE)) flash_data_pull_sequencer_h; + push_pull_sequencer#(.DeviceDataWidth(FLASH_DATA_SIZE)) flash_addr_pull_sequencer_h; + push_pull_sequencer#(.DeviceDataWidth(1), .HostDataWidth(LC_PROG_DATA_SIZE)) + lc_prog_pull_sequencer_h; +endclass diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_background_chks_vseq.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_background_chks_vseq.sv new file mode 100644 index 00000000000000..d316e1a12a5218 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_background_chks_vseq.sv @@ -0,0 +1,73 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// This simple sequence checks if the background check can be triggered once the period is set. + +class otp_ctrl_background_chks_vseq extends otp_ctrl_dai_lock_vseq; + `uvm_object_utils(otp_ctrl_background_chks_vseq) + + `uvm_object_new + + rand bit [1:0] trigger_chks; + rand uint check_period; + + constraint regwens_c {check_regwen_val == 1;} + + // At least one check will be triggered + constraint check_triggers_c {trigger_chks > 0;} + + constraint check_period_c { + check_period < 20; + check_period > 0; + } + + task body(); + int check_wait_cycles; + super.body(); + + // For stress_all_with_rand_reset test, if previous lc_esc_en is not cleared, then skip the + // background check. + if (cfg.otp_ctrl_vif.lc_esc_on == 0) begin + + // Write background check + if (trigger_chks[0]) csr_wr(ral.integrity_check_period, check_period); + if (trigger_chks[1]) csr_wr(ral.consistency_check_period, check_period); + `uvm_info(`gfn, $sformatf("trigger background check %0h", trigger_chks), UVM_LOW) + + cfg.en_scb = 0; + // According to spec, check period will append an 'hFF from the LSF. Add 10 cycle buffers for + // register updates. + check_wait_cycles = (check_period + 1) << 8 + 10; + + // Wait for first check done + repeat($countones(trigger_chks)) begin + csr_spinwait(.ptr(ral.status.check_pending), .exp_data(1), + .timeout_ns(cfg.clk_rst_vif.clk_period_ps / 1000 * check_wait_cycles)); + + csr_spinwait(.ptr(ral.status.check_pending), .exp_data(0)); + end + + // Configure timeout settings to trigger check error + csr_wr(ral.check_timeout, $urandom_range(1, 5)); + `uvm_info(`gfn, "trigger check timeout error", UVM_LOW) + + // Wait for fatal alert + `DV_SPINWAIT_EXIT( + wait(cfg.m_alert_agent_cfgs["fatal_check_error"].vif.alert_tx_final.alert_p);, + cfg.clk_rst_vif.wait_clks(check_wait_cycles);, + $sformatf("Timeout waiting for alert %0s", "fatal_check_error")) + check_fatal_alert_nonblocking("fatal_check_error"); + + cfg.clk_rst_vif.wait_clks($urandom_range(50, 1000)); + csr_rd_check(.ptr(ral.status.timeout_error), .compare_value(1)); + end + endtask + + // Enable scoreboard is done in stress_all sequence and `apply_resets_concurrently` task to + // avoid otp_ctrl_scoreboard reporting failures when reset has not been issued. + virtual task post_start(); + expect_fatal_alerts = 1; + super.post_start(); + endtask +endclass diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_base_vseq.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_base_vseq.sv similarity index 98% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_base_vseq.sv rename to hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_base_vseq.sv index 4f7744fe621f9f..6ce2d572cc3e9b 100644 --- a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_base_vseq.sv +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_base_vseq.sv @@ -1,9 +1,6 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -// DO NOT EDIT THIS FILE DIRECTLY. -// It has been generated with ./util/design/gen-otp-mmap.py - class otp_ctrl_base_vseq extends cip_base_vseq #( .RAL_T (otp_ctrl_core_reg_block), .CFG_T (otp_ctrl_env_cfg), @@ -166,7 +163,7 @@ class otp_ctrl_base_vseq extends cip_base_vseq #( if (is_secret(addr) || is_sw_digest(addr)) csr_wr(ral.direct_access_wdata[1], wdata1); do_otp_wr = 1; - csr_wr(ral.direct_access_cmd, int'(otp_ctrl_pkg::DaiWrite)); + csr_wr(ral.direct_access_cmd, int'(otp_ctrl_top_specific_pkg::DaiWrite)); `uvm_info(`gfn, $sformatf("DAI write, address %0h, data0 %0h data1 %0h, is_secret = %0b", addr, wdata0, wdata1, is_secret(addr)), UVM_DEBUG) @@ -194,7 +191,7 @@ class otp_ctrl_base_vseq extends cip_base_vseq #( csr_wr(ral.direct_access_address, addr); do_otp_rd = 1; - csr_wr(ral.direct_access_cmd, int'(otp_ctrl_pkg::DaiRead)); + csr_wr(ral.direct_access_cmd, int'(otp_ctrl_top_specific_pkg::DaiRead)); if (cfg.zero_delays && is_valid_dai_op && cfg.otp_ctrl_vif.lc_escalate_en_i == lc_ctrl_pkg::Off) begin @@ -225,7 +222,7 @@ class otp_ctrl_base_vseq extends cip_base_vseq #( virtual task cal_digest(int part_idx); bit [TL_DW-1:0] val; csr_wr(ral.direct_access_address, PART_BASE_ADDRS[part_idx]); - csr_wr(ral.direct_access_cmd, otp_ctrl_pkg::DaiDigest); + csr_wr(ral.direct_access_cmd, otp_ctrl_top_specific_pkg::DaiDigest); if (cfg.zero_delays && is_valid_dai_op && cfg.otp_ctrl_vif.lc_escalate_en_i == lc_ctrl_pkg::Off) begin diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_callback_vseq.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_callback_vseq.sv new file mode 100644 index 00000000000000..155e6f5e3615ca --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_callback_vseq.sv @@ -0,0 +1,22 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// A sequence hook to attach to otp_ctrl_base_vseq. +class otp_ctrl_callback_vseq extends cip_base_vseq #( + .RAL_T (otp_ctrl_core_reg_block), + .CFG_T (otp_ctrl_env_cfg), + .COV_T (otp_ctrl_env_cov), + .VIRTUAL_SEQUENCER_T (otp_ctrl_virtual_sequencer) + ); + `uvm_object_utils(otp_ctrl_callback_vseq) + `uvm_object_new + + virtual task dut_init_callback(); + // Do nothing but can be overridden in closed source environment. + endtask + + virtual task post_otp_pwr_init(); + // Do nothing but can be overridden in closed source environment. + endtask : post_otp_pwr_init +endclass : otp_ctrl_callback_vseq diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_check_fail_vseq.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_check_fail_vseq.sv new file mode 100644 index 00000000000000..061ebe96a30a08 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_check_fail_vseq.sv @@ -0,0 +1,32 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// This sequence creates the following check failure scenarios: +// 1. Check timeout +// 2. Correctable ECC check error +// 3. Uncorrectable ECC error +class otp_ctrl_check_fail_vseq extends otp_ctrl_dai_lock_vseq; + `uvm_object_utils(otp_ctrl_check_fail_vseq) + + `uvm_object_new + + constraint ecc_otp_err_c { + ecc_otp_err inside {OtpEccCorrErr, OtpNoEccErr}; + } + + constraint ecc_chk_err_c { + ecc_chk_err dist {OtpNoEccErr :/ 1, + OtpEccCorrErr :/ 1, + OtpEccUncorrErr :/1 }; + } + + // 50% chance of having a check timeout + // Because of the regwen, even though we constrain the timeout value, it might not apply to the + // DUT. + constraint check_timeout_val_c { + check_timeout_val dist {[1 : CHK_TIMEOUT_CYC] :/ 1, + [100_000 :'1] :/ 1}; + } + +endclass diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_common_vseq.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_common_vseq.sv new file mode 100644 index 00000000000000..58f419c77c0ad5 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_common_vseq.sv @@ -0,0 +1,232 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +class otp_ctrl_common_vseq extends otp_ctrl_base_vseq; + `uvm_object_utils(otp_ctrl_common_vseq) + + rand bit [TL_DW-1:0] dai_addr, wdata0, wdata1; + rand port_drive_condition_e reset_drive_cond; + + string prim_otp_alert_name = "fatal_prim_otp_alert"; + string integ_err_alert_name = "fatal_bus_integ_error"; + + // This flag is used to identify if the sec_cm or tl_intg_err uses prim_otp_tl_i/o. + protected bit is_prim_otp; + + constraint dai_addr_c { + dai_addr dist { + [0 : (PartInfo[LifeCycleIdx].offset - 1)] :/ 1, + [PartInfo[LifeCycleIdx].offset : {OTP_ADDR_WIDTH{1'b1}}] :/ 1}; + } + + constraint reset_drive_cond_c { + reset_drive_cond dist { + DriveRandomly :/ 7, + DuringOTPDaiBusy :/ 1, + DuringOTPRead :/ 1, + DuringOTPDaiDigest :/ 1 + }; + } + + constraint num_trans_c { + num_trans inside {[1:2]}; + } + `uvm_object_new + + virtual task dut_init(string reset_kind = "HARD"); + super.dut_init(reset_kind); + // drive dft_en pins to access the test_access memory + cfg.otp_ctrl_vif.drive_lc_dft_en(lc_ctrl_pkg::On); + // once turn on lc_dft_en regiser, will need some time to update the state register + // two clock cycles for lc_async mode, one clock cycle for driving dft_en, one more clock cycle + // so there is no racing condition. + if (cfg.en_dv_cdc) cfg.clk_rst_vif.wait_clks(5); + else cfg.clk_rst_vif.wait_clks(3); + endtask + + virtual task body(); + if (common_seq_type == "sec_cm_fi") begin + // OTP_CTRL has many sec_cm items, so too many iterations of this test will consume too much + // simulation time and eventually causes timeout. So we reduce to 10 iterations. + run_sec_cm_fi_vseq(10); + end else begin + run_common_vseq_wrapper(num_trans); + end + endtask : body + + virtual task apply_resets_concurrently(int reset_duration_ps = 0); + // For stress_all_with_rand_reset test only - backdoor clear OTP memory, + // and re-initialize OTP_ctrl after reset. + if (common_seq_type == "stress_all_with_rand_reset") begin + cfg.otp_ctrl_vif.release_part_access_mubi(); + cfg.otp_ctrl_vif.drive_lc_escalate_en(lc_ctrl_pkg::Off); + // Set dft_en to On to allow the csr_test to check all registers' default value after reset. + cfg.otp_ctrl_vif.drive_lc_dft_en(lc_ctrl_pkg::On); + otp_ctrl_init(); + otp_pwr_init(); + super.apply_resets_concurrently(reset_duration_ps); + cfg.en_scb = 1; + end else begin + super.apply_resets_concurrently(reset_duration_ps); + end + clear_seq_flags(); + endtask + + virtual task wait_to_issue_reset(uint reset_delay_bound = 10_000_000); + `DV_CHECK_MEMBER_RANDOMIZE_FATAL(reset_drive_cond) + case (reset_drive_cond) + DriveRandomly: begin + super.wait_to_issue_reset(reset_delay_bound); + end + DuringOTPDaiBusy: begin + `DV_SPINWAIT_EXIT( + wait (do_otp_wr); + cfg.clk_rst_vif.wait_clks($urandom_range(1, 70));, + super.wait_to_issue_reset(reset_delay_bound);) + #($urandom_range(0, cfg.clk_rst_vif.clk_period_ps) * 1ps); + end + DuringOTPDaiDigest: begin + `DV_SPINWAIT_EXIT( + wait (do_digest_cal); + cfg.clk_rst_vif.wait_clks($urandom_range(1, 350));, + super.wait_to_issue_reset(reset_delay_bound);) + #($urandom_range(0, cfg.clk_rst_vif.clk_period_ps) * 1ps); + end + DuringOTPRead: begin + `DV_SPINWAIT_EXIT( + wait (do_otp_rd); + cfg.clk_rst_vif.wait_clks($urandom_range(1, 10));, + super.wait_to_issue_reset(reset_delay_bound);) + #($urandom_range(0, cfg.clk_rst_vif.clk_period_ps) * 1ps); + end + default: `uvm_fatal(`gfn, $sformatf("Unsupported reset_drive_cond %0d", reset_drive_cond)) + endcase + endtask : wait_to_issue_reset + + // This task overrides the check for `prim_onehot_check` and `tl_intg_error`. + // Alerts coming from the `prim_otp` module will only bypass OTP_CTRL, it won't affect the + // OTP_CTRL and will fire its own alerts. + virtual task check_tl_intg_error_response(); + if (is_prim_otp) begin + repeat ($urandom_range(5, 20)) begin + wait_alert_trigger(prim_otp_alert_name, .wait_complete(1)); + end + end else begin + super.check_tl_intg_error_response(); + end + endtask + + virtual task check_sec_cm_alert(string sec_type_name, string alert_name); + `uvm_info(`gfn, $sformatf("expected fatal alert is triggered for %s", + sec_type_name), UVM_LOW) + + // This is a fatal alert and design keeps sending it until reset is issued. + // Check alerts are triggered for a few times + repeat (5) begin + wait_alert_trigger(alert_name, .wait_complete(1)); + end + endtask + + // In tl_intg_err test, override this task to set is_prim_otp flag. + virtual task run_tl_intg_err_vseq_sub(string ral_name); + if (ral_name == "otp_ctrl_prim_reg_block") is_prim_otp = 1; + else is_prim_otp = 0; + super.run_tl_intg_err_vseq_sub(ral_name); + endtask + + virtual task check_sec_cm_fi_resp(sec_cm_base_if_proxy if_proxy); + bit [TL_DW-1:0] exp_status_val, rdata0, rdata1; + string prim_otp_alert_name = "fatal_prim_otp_alert"; + string integ_err_alert_name = "fatal_bus_integ_error"; + + // Alerts coming from the `prim_otp` module will only bypass OTP_CTRL, it won't affect the + // OTP_CTRL and will fire its own alerts. + if (is_prim_otp) begin + check_sec_cm_alert(if_proxy.sec_cm_type.name, prim_otp_alert_name); + + // Alerts coming from the `u_tlul_lc_gate` module will only trigger bus_integrity alerts, and + // bus_integrity related status. + // This error won't local escalate to OTP partitions. + end else if (!uvm_re_match("*.u_tlul_lc_gate*", if_proxy.path)) begin + check_sec_cm_alert(if_proxy.sec_cm_type.name, integ_err_alert_name); + + exp_status_val[OtpBusIntegErrorIdx] = 1; + exp_status_val[OtpDaiIdleIdx] = 1; + + // All other errors triggers normal fatal alerts, and will locally escalate to other + // partitions. + end else begin + super.check_sec_cm_fi_resp(if_proxy); + + // Set expected status error val. + for (int i = 0; i <= OtpLciErrIdx; i++) exp_status_val[i] = 1; + if (!uvm_re_match("*.u_otp_ctrl_lfsr_timer*", if_proxy.path)) begin + exp_status_val[OtpLfsrFsmErrIdx] = 1; + end else if (!uvm_re_match("*u_otp_ctrl_kdi*", if_proxy.path)) begin + exp_status_val[OtpDerivKeyFsmErrIdx] = 1; + end else if (!uvm_re_match("*u_otp_ctrl_scrmbl*", if_proxy.path)) begin + exp_status_val[OtpScramblingFsmErrIdx] = 1; + end + + csr_rd_check(.ptr(ral.status), .compare_value(exp_status_val), .err_msg( + $sformatf("cm_fi status failed at injection %0s", if_proxy.sec_cm_type.name))); + + // Check OTP is locked after fault error. + `DV_CHECK_RANDOMIZE_FATAL(this) + is_valid_dai_op = 0; + + // Access OTP via DAI interface. + dai_wr(dai_addr, wdata0, wdata1); + dai_rd(dai_addr, rdata0, rdata1); + `DV_CHECK_EQ(rdata0, 0) + `DV_CHECK_EQ(rdata1, 0) + if (is_sw_part(dai_addr)) begin + uvm_reg_addr_t tlul_addr = cfg.ral.get_addr_from_offset(get_sw_window_offset(dai_addr)); + tl_access(.addr(tlul_addr), .write(0), .data(rdata0), .blocking(1), .check_rsp(1), + .exp_err_rsp(1), .exp_data('1)); + end + cal_hw_digests(); + write_sw_digests(); + + // Access OTP via app interface. + if ($urandom_range(0, 1)) req_otbn_key(0); + if ($urandom_range(0, 1)) req_flash_addr_key(0); + if ($urandom_range(0, 1)) req_flash_data_key(0); + if ($urandom_range(0, 1)) req_all_sram_keys(0); + cfg.clk_rst_vif.wait_clks($urandom_range(10, 20)); + + csr_rd_check(.ptr(ral.status), .compare_value(exp_status_val), + .err_msg("status failure after OTP fatal fault error")); + end + endtask : check_sec_cm_fi_resp + + virtual function void sec_cm_fi_ctrl_svas(sec_cm_base_if_proxy if_proxy, bit enable); + case (if_proxy.sec_cm_type) + SecCmPrimCount: begin + if (!enable) begin + $assertoff(0, "tb.dut.gen_partitions[3].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $assertoff(0, "tb.dut.gen_partitions[4].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $assertoff(0, "tb.dut.gen_partitions[5].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $assertoff(0, "tb.dut.gen_partitions[6].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $assertoff(0, "tb.dut.gen_partitions[7].gen_lifecycle.u_part_buf.ScrmblDataKnown_A"); + end else begin + $asserton(0, "tb.dut.gen_partitions[3].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $asserton(0, "tb.dut.gen_partitions[4].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $asserton(0, "tb.dut.gen_partitions[5].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $asserton(0, "tb.dut.gen_partitions[6].gen_buffered.u_part_buf.ScrmblDataKnown_A"); + $asserton(0, "tb.dut.gen_partitions[7].gen_lifecycle.u_part_buf.ScrmblDataKnown_A"); + end + end + SecCmPrimSparseFsmFlop, SecCmPrimDoubleLfsr, SecCmPrimOnehot: begin + // No assertion error. + end + default: `uvm_fatal(`gfn, $sformatf("unexpected sec_cm_type %s", if_proxy.sec_cm_type.name)) + endcase + + // Set the flag to store if the error injection is on prim_tlul_if or core_tlul_if. + if (!uvm_re_match("*.u_otp.*", if_proxy.path)) is_prim_otp = 1; + else is_prim_otp = 0; + endfunction: sec_cm_fi_ctrl_svas + +endclass diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv new file mode 100644 index 00000000000000..c8ea43b32a6bcb --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_errs_vseq.sv @@ -0,0 +1,51 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// otp_ctrl_dai_errs_vseq is developed to randomly read/write to any address within OTP: +// - A writeblank error will be triggered if write to a non-empty address +// - An access error will be triggered if write to lc partition via DAI interface, or if DAI write +// to digest addrs for non-sw partitions +class otp_ctrl_dai_errs_vseq extends otp_ctrl_dai_lock_vseq; + `uvm_object_utils(otp_ctrl_dai_errs_vseq) + + bit[31:0] exp_status; + `uvm_object_new + + // Only run one transition to avoid dut_init in the sequence. Because write-blank-error can cause + // otp_init failure. + constraint num_trans_c { + num_trans == 1; + num_dai_op inside {[100:500]}; + } + + constraint regwens_c { + check_trigger_regwen_val == 0; + } + + constraint rd_check_after_wr_c { + rand_wr == rand_rd; + } + + function void pre_randomize(); + this.dai_wr_blank_addr_c.constraint_mode(0); + this.no_access_err_c.constraint_mode(0); + this.dai_wr_digests_c.constraint_mode(0); + write_unused_addr = 0; + endfunction + + task body(); + do_apply_reset = 0; + if (do_lc_trans && !cfg.otp_ctrl_vif.alert_reqs) begin + req_lc_transition(do_lc_trans, lc_prog_blocking); + end + super.body(); + endtask + + virtual task post_start(); + expect_fatal_alerts = 1; + do_apply_reset = 1; + do_otp_ctrl_init = 1; + super.post_start(); + endtask +endclass diff --git a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv similarity index 97% rename from hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv rename to hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv index c02559aded48e0..783b130e0a7dd6 100644 --- a/hw/ip/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_dai_lock_vseq.sv @@ -1,9 +1,6 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -// DO NOT EDIT THIS FILE DIRECTLY. -// It has been generated with ./util/design/gen-otp-mmap.py - // otp_ctrl_dai_lock_vseq is developed to read/write lock DAI interface by partitions, and request // read/write access to check if correct status and error code is triggered diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv new file mode 100644 index 00000000000000..6a5a2526f4ed41 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/dv/env/seq_lib/otp_ctrl_init_fail_vseq.sv @@ -0,0 +1,235 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// Otp_ctrl_init_fail_vseq is developed to check if otp_ctrl reacts correctly if initialization +// failed. +// Note that scoreboard is disabled in this test and all checks are done within this sequence. +// This test writes and reads to OTP_memory via DAI interface, then triggers digest calculations. +// Afterwards instead of issuing reset, this sequence continues to write to DAI interface. +// If any of the hardware partition is updated, then in next power cycle, the initialization will +// fail. +// If no check failure, we will random inject ECC errors to create init macro errors. +// We will also trigger sw partition ECC reg failure by forcing the ECC reg error output. +// +// This sequence will check the following items if OTP init failed with fatal error: +// - Otp_initialization failure triggers fatal alert +// - Status register reflect the correct error +// - Otp_ctrl's power init output stays 0 +// This sequence will check the following items if OTP init failed with correctable error: +// - Otp_initialtion passed with power init output changes to 1 +// - Otp status and interrupt reflect the correct error message + +class otp_ctrl_init_fail_vseq extends otp_ctrl_smoke_vseq; + `uvm_object_utils(otp_ctrl_init_fail_vseq) + + `uvm_object_new + + rand uint num_to_lock_digests; + bit [NumPart-1:0] init_chk_err; + bit part_locked; + + // If num_to_lock_digests is larger than num_dai_op, that means there won't be OTP init check + // error, so this sequence will trigger ECC error instead. + // We set 25% possibility that OTP init check fails due to writing OTP after digest is locked. + constraint lock_digest_c {num_to_lock_digests < num_dai_op * 4;} + constraint num_iterations_c {num_dai_op inside {[20:100]};} + constraint ecc_otp_err_c { + $countones(ecc_otp_err) dist {OtpNoEccErr :/ 2, + OtpEccCorrErr :/ 2, + OtpEccUncorrErr :/ 1}; + } + + virtual task pre_start(); + super.pre_start(); + num_to_lock_digests.rand_mode(0); + endtask + + task body(); + bit [TL_DW-1:0] exp_status; + `uvm_info(`gfn, $sformatf("Number of dai operation is %0d, number to lock digest is %0d", + num_dai_op, num_to_lock_digests), UVM_MEDIUM) + + for (uint i = 0; i <= num_dai_op; i++) begin + bit [TL_DW-1:0] tlul_val; + if (cfg.stop_transaction_generators()) return; + + `DV_CHECK_RANDOMIZE_FATAL(this) + `uvm_info(`gfn, $sformatf("starting dai access seq %0d/%0d with addr %0h in partition %0d", + i, num_dai_op, dai_addr, part_idx), UVM_MEDIUM) + + if (i > num_to_lock_digests && PartInfo[part_idx].hw_digest && + !used_dai_addrs.exists(dai_addr)) begin + init_chk_err[part_idx] = 1; + end + + // OTP write via DAI + dai_wr(dai_addr, wdata0, wdata1); + + // OTP read via DAI, check data in scb + dai_rd(dai_addr, wdata0, wdata1); + + // If write sw partitions, check tlul window + if (is_sw_part(dai_addr)) begin + uvm_reg_addr_t tlul_addr = cfg.ral.get_addr_from_offset(get_sw_window_offset(dai_addr)); + tl_access(.addr(tlul_addr), .write(0), .data(tlul_val), .blocking(1), .check_rsp(0)); + end + + if (i == num_to_lock_digests) begin + cal_hw_digests('1); + part_locked = 1; + end + + csr_rd(ral.status, tlul_val); + end + + do_otp_ctrl_init = 0; + do_otp_pwr_init = 0; + cfg.en_scb = 0; + dut_init(); + + if (init_chk_err) begin + `uvm_info(`gfn, $sformatf("OTP_init check failure with init error = %0h", init_chk_err), + UVM_LOW) + foreach(init_chk_err[i]) begin + if (cfg.stop_transaction_generators()) break; + if (init_chk_err[i]) exp_status |= 1'b1 << i; + end + + check_otp_fatal_err("fatal_check_error", exp_status); + + // If not check error, force ECC correctable and uncorrectable error + end else begin + bit is_fatal, is_correctable; + bit [TL_DW-1:0] addr; + + for (int i = 0; i < NumPart; i++) begin + if (cfg.stop_transaction_generators()) return; + `DV_CHECK_RANDOMIZE_FATAL(this); + + if (PartInfo[i].sw_digest) begin + // During OTP init, SW partitions only read digest value + addr = PART_OTP_DIGEST_ADDRS[i] << 2; + end else begin + // During OTP init, non SW partitions read all value + addr = $urandom_range(PartInfo[i].offset, PartInfo[i].offset + PartInfo[i].size - 1); + end + + void'(backdoor_inject_ecc_err(addr, ecc_otp_err)); + // VendorTest partition's ECC error is not fatal. + if (!is_fatal && ecc_otp_err == OtpEccUncorrErr && part_has_integrity(i)) begin + is_fatal = 1; + end else if (!is_correctable && ecc_otp_err == OtpEccCorrErr && part_has_integrity(i)) begin + is_correctable = 1; + end + if (ecc_otp_err != OtpNoEccErr && part_has_integrity(i)) exp_status[i] = 1; + end + + if (is_fatal) begin + // ECC uncorrectable error. + `uvm_info(`gfn, "OTP_init macro ECC uncorrectable failure", UVM_LOW) + check_otp_fatal_err("fatal_macro_error", exp_status); + end else if ($urandom_range(0, 1)) begin + + // Randomly force ECC reg in sw partitions to create a check failure. + // Totaly three sw partitions, and each bit indexes a partition. + bit [NumPartUnbuf-1:0] sw_check_fail = $urandom_range(1, (1'b1<= PartInfo\[0\]\.offset.*} \ + -comment "This invalid compare is due to the first partition offset being zero." + +waive -rules {INVALID_COMPARE} -location {otp_ctrl.sv} -regexp {.*tlul_addr.* >= PartInfo\[0\]\.offset.*} \ + -comment "This invalid compare is due to the first partition offset being zero." + +waive -rules {LOOP_VAR_OP FOR_LOOP_BOUNDS} -location {prim_cipher_pkg.sv} -regexp {Loop.*round_cnt.*constant.*} \ + -comment "This function needs to iterate over the key schedule function in order to derive the decryption key." + +waive -rules {ASSIGN_SIGN} -location {otp_ctrl_part_buf.sv} -regexp {Unsigned target 'step_i' assigned signed value 'CntWidth'\(1\)'} \ + -comment "This is fine as long as CntWidth is greater equal 2 which is checked using an SVA." diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vbl b/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vbl new file mode 100644 index 00000000000000..709cf2bb305769 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vbl @@ -0,0 +1,5 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# verible waiver file for OTP controller generic package diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vlt b/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vlt new file mode 100644 index 00000000000000..fefa0f75baad1d --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.vlt @@ -0,0 +1,5 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// waiver file for OTP controller generic package diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.waiver b/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.waiver new file mode 100644 index 00000000000000..e0846938574c63 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_pkg.waiver @@ -0,0 +1,5 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# ascentlint waiver file for OTP controller generic package diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vbl b/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vbl new file mode 100644 index 00000000000000..dadab89d18ef1e --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vbl @@ -0,0 +1,8 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# waiver file for OTP controller top-specific package + +# waive long line violations in generated code +waive --rule=line-length --location="otp_ctrl_part_pkg.sv" diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vlt b/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vlt new file mode 100644 index 00000000000000..06603a4760cf74 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.vlt @@ -0,0 +1,5 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// waiver file for OTP controller top-specific package diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.waiver b/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.waiver new file mode 100644 index 00000000000000..69259ce4b16170 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/lint/otp_ctrl_top_specific_pkg.waiver @@ -0,0 +1,9 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# waiver file for OTP controller top-specific package + +# waive line length violations in generated code +waive -rules LINE_LENGTH -location {otp_ctrl_part_pkg.sv} \ + -comment "Waive line length violations in generated code." diff --git a/hw/ip/otp_ctrl/otp_ctrl.core b/hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl.core similarity index 93% rename from hw/ip/otp_ctrl/otp_ctrl.core rename to hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl.core index 302558501b0502..a7951dd4ccccbe 100644 --- a/hw/ip/otp_ctrl/otp_ctrl.core +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl.core @@ -2,13 +2,16 @@ CAPI=2: # Copyright lowRISC contributors (OpenTitan project). # Licensed under the Apache License, Version 2.0, see LICENSE for details. # SPDX-License-Identifier: Apache-2.0 -name: "lowrisc:ip:otp_ctrl:1.0" +name: lowrisc:opentitan:top_earlgrey_otp_ctrl:1.0 description: "OTP Controller" +virtual: + - lowrisc:ip_interfaces:otp_ctrl filesets: files_rtl: depend: - lowrisc:ip:otp_ctrl_pkg + - lowrisc:opentitan:top_earlgrey_otp_ctrl_top_specific_pkg - lowrisc:ip:tlul - lowrisc:prim:all - lowrisc:prim:ram_1p diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl_pkg.core b/hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl_pkg.core new file mode 100644 index 00000000000000..6698fbdb8541fa --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl_pkg.core @@ -0,0 +1,67 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: lowrisc:ip:otp_ctrl_pkg:1.0 +description: "OTP Controller Package" +filesets: + files_rtl: + depend: + - lowrisc:ip:lc_ctrl_pkg + + files: + - rtl/otp_ctrl_pkg.sv + file_type: systemVerilogSource + + files_verilator_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_pkg.vlt + file_type: vlt + + files_ascentlint_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_pkg.waiver + file_type: waiver + + files_veriblelint_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_pkg.vbl + file_type: veribleLintWaiver + +parameters: + SYNTHESIS: + datatype: bool + paramtype: vlogdefine + + +targets: + default: &default_target + filesets: + - tool_verilator ? (files_verilator_waiver) + - tool_ascentlint ? (files_ascentlint_waiver) + - tool_veriblelint ? (files_veriblelint_waiver) + - files_rtl + toplevel: lc_ctrl + + lint: + <<: *default_target + default_tool: verilator + parameters: + - SYNTHESIS=true + tools: + verilator: + mode: lint-only + verilator_options: + - "-Wall" diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl_prim_reg_top.core b/hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl_prim_reg_top.core new file mode 100644 index 00000000000000..817b3e74434c60 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl_prim_reg_top.core @@ -0,0 +1,29 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: "lowrisc:ip:otp_ctrl_prim_reg_top:1.0" +description: "Generic register top for the OTP wrapper" +filesets: + files_rtl: + depend: + # otp_ctrl_prim_reg_top.sv should depend on generic items for now. + # This may change and will require reworking the flow to generate + # the otp prim. + - lowrisc:ip_interfaces:otp_ctrl_top_specific_pkg + files: + - rtl/otp_ctrl_prim_reg_top.sv + file_type: systemVerilogSource + + +parameters: + SYNTHESIS: + datatype: bool + paramtype: vlogdefine + + +targets: + default: &default_target + filesets: + - files_rtl + toplevel: lc_ctrl diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl_top_specific_pkg.core b/hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl_top_specific_pkg.core new file mode 100644 index 00000000000000..f212eaab353e7f --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/otp_ctrl_top_specific_pkg.core @@ -0,0 +1,75 @@ +CAPI=2: +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +name: lowrisc:opentitan:top_earlgrey_otp_ctrl_top_specific_pkg:1.0 +description: "OTP Controller Top Specific Packages" +virtual: + - lowrisc:ip_interfaces:otp_ctrl_top_specific_pkg + +filesets: + files_rtl: + depend: + - lowrisc:tlul:headers + - lowrisc:ip:lc_ctrl_pkg + - lowrisc:ip:otp_ctrl_pkg + - lowrisc:prim:mubi + + files: + - rtl/otp_ctrl_reg_pkg.sv + - rtl/otp_ctrl_top_specific_pkg.sv + - rtl/otp_ctrl_part_pkg.sv + file_type: systemVerilogSource + + files_verilator_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_top_specific_pkg.vlt + file_type: vlt + + files_ascentlint_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_top_specific_pkg.waiver + file_type: waiver + + files_veriblelint_waiver: + depend: + # common waivers + - lowrisc:lint:common + - lowrisc:lint:comportable + files: + - lint/otp_ctrl_top_specific_pkg.vbl + file_type: veribleLintWaiver + +parameters: + SYNTHESIS: + datatype: bool + paramtype: vlogdefine + + +targets: + default: &default_target + filesets: + - tool_verilator ? (files_verilator_waiver) + - tool_ascentlint ? (files_ascentlint_waiver) + - tool_veriblelint ? (files_veriblelint_waiver) + - files_rtl + toplevel: lc_ctrl + + lint: + <<: *default_target + default_tool: verilator + parameters: + - SYNTHESIS=true + tools: + verilator: + mode: lint-only + verilator_options: + - "-Wall" diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl.sv new file mode 100644 index 00000000000000..6d5647a989a41b --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl.sv @@ -0,0 +1,1572 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// OTP Controller top. +// + +`include "prim_assert.sv" + +module otp_ctrl + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; +#( + // Enable asynchronous transitions on alerts. + parameter logic [NumAlerts-1:0] AlertAsyncOn = {NumAlerts{1'b1}}, + // Compile time random constants, to be overriden by topgen. + parameter lfsr_seed_t RndCnstLfsrSeed = RndCnstLfsrSeedDefault, + parameter lfsr_perm_t RndCnstLfsrPerm = RndCnstLfsrPermDefault, + parameter scrmbl_key_init_t RndCnstScrmblKeyInit = RndCnstScrmblKeyInitDefault, + // Hexfile file to initialize the OTP macro. + // Note that the hexdump needs to account for ECC. + parameter MemInitFile = "" +) ( + // OTP clock + input clk_i, + input rst_ni, + // EDN clock and interface + logic clk_edn_i, + logic rst_edn_ni, + output edn_pkg::edn_req_t edn_o, + input edn_pkg::edn_rsp_t edn_i, + // Bus Interface + input tlul_pkg::tl_h2d_t core_tl_i, + output tlul_pkg::tl_d2h_t core_tl_o, + input tlul_pkg::tl_h2d_t prim_tl_i, + output tlul_pkg::tl_d2h_t prim_tl_o, + // Interrupt Requests + output logic intr_otp_operation_done_o, + output logic intr_otp_error_o, + // Alerts + input prim_alert_pkg::alert_rx_t [NumAlerts-1:0] alert_rx_i, + output prim_alert_pkg::alert_tx_t [NumAlerts-1:0] alert_tx_o, + // Observability to AST + input ast_pkg::ast_obs_ctrl_t obs_ctrl_i, + output logic [7:0] otp_obs_o, + // Macro-specific power sequencing signals to/from AST. + output otp_ast_req_t otp_ast_pwr_seq_o, + input otp_ast_rsp_t otp_ast_pwr_seq_h_i, + // Power manager interface (inputs are synced to OTP clock domain) + input pwrmgr_pkg::pwr_otp_req_t pwr_otp_i, + output pwrmgr_pkg::pwr_otp_rsp_t pwr_otp_o, + // Macro-specific test registers going to lifecycle TAP + input lc_otp_vendor_test_req_t lc_otp_vendor_test_i, + output lc_otp_vendor_test_rsp_t lc_otp_vendor_test_o, + // Lifecycle transition command interface + input lc_otp_program_req_t lc_otp_program_i, + output lc_otp_program_rsp_t lc_otp_program_o, + // Lifecycle broadcast inputs + // SEC_CM: LC_CTRL.INTERSIG.MUBI + input lc_ctrl_pkg::lc_tx_t lc_creator_seed_sw_rw_en_i, + input lc_ctrl_pkg::lc_tx_t lc_owner_seed_sw_rw_en_i, + input lc_ctrl_pkg::lc_tx_t lc_seed_hw_rd_en_i, + input lc_ctrl_pkg::lc_tx_t lc_dft_en_i, + input lc_ctrl_pkg::lc_tx_t lc_escalate_en_i, + input lc_ctrl_pkg::lc_tx_t lc_check_byp_en_i, + // OTP broadcast outputs + // SEC_CM: TOKEN_VALID.CTRL.MUBI + output otp_lc_data_t otp_lc_data_o, + output otp_keymgr_key_t otp_keymgr_key_o, + // Scrambling key requests + input flash_otp_key_req_t flash_otp_key_i, + output flash_otp_key_rsp_t flash_otp_key_o, + input sram_otp_key_req_t [NumSramKeyReqSlots-1:0] sram_otp_key_i, + output sram_otp_key_rsp_t [NumSramKeyReqSlots-1:0] sram_otp_key_o, + input otbn_otp_key_req_t otbn_otp_key_i, + output otbn_otp_key_rsp_t otbn_otp_key_o, + // Hardware config bits + output otp_broadcast_t otp_broadcast_o, + // External voltage for OTP + inout wire otp_ext_voltage_h_io, + // Scan + input scan_en_i, + input scan_rst_ni, + input prim_mubi_pkg::mubi4_t scanmode_i, + // Test-related GPIO output + output logic [OtpTestVectWidth-1:0] cio_test_o, + output logic [OtpTestVectWidth-1:0] cio_test_en_o +); + + import prim_mubi_pkg::*; + import prim_util_pkg::vbits; + + //////////////////////// + // Integration Checks // + //////////////////////// + + // This ensures that we can transfer scrambler data blocks in and out of OTP atomically. + `ASSERT_INIT(OtpIfWidth_A, OtpIfWidth == ScrmblBlockWidth) + + // These error codes need to be identical. + `ASSERT_INIT(ErrorCodeWidth_A, OtpErrWidth == prim_otp_pkg::ErrWidth) + `ASSERT_INIT(OtpErrorCode0_A, int'(NoError) == int'(prim_otp_pkg::NoError)) + `ASSERT_INIT(OtpErrorCode1_A, int'(MacroError) == int'(prim_otp_pkg::MacroError)) + `ASSERT_INIT(OtpErrorCode2_A, int'(MacroEccCorrError) == int'(prim_otp_pkg::MacroEccCorrError)) + `ASSERT_INIT(OtpErrorCode3_A, + int'(MacroEccUncorrError) == int'(prim_otp_pkg::MacroEccUncorrError)) + `ASSERT_INIT(OtpErrorCode4_A, + int'(MacroWriteBlankError) == int'(prim_otp_pkg::MacroWriteBlankError)) + + ///////////// + // Regfile // + ///////////// + + // We have one CSR node, one functional TL-UL window and a gate module for that window + logic [2:0] intg_error; + + tlul_pkg::tl_h2d_t tl_win_h2d; + tlul_pkg::tl_d2h_t tl_win_d2h; + + otp_ctrl_reg_pkg::otp_ctrl_core_reg2hw_t reg2hw; + otp_ctrl_reg_pkg::otp_ctrl_core_hw2reg_t hw2reg; + + // SEC_CM: DIRECT_ACCESS.CONFIG.REGWEN, CHECK_TRIGGER.CONFIG.REGWEN, CHECK.CONFIG.REGWEN + otp_ctrl_core_reg_top u_reg_core ( + .clk_i, + .rst_ni, + .tl_i ( core_tl_i ), + .tl_o ( core_tl_o ), + .tl_win_o ( tl_win_h2d ), + .tl_win_i ( tl_win_d2h ), + .reg2hw ( reg2hw ), + .hw2reg ( hw2reg ), + // SEC_CM: BUS.INTEGRITY + .intg_err_o( intg_error[0] ) + ); + + /////////////////////////////////////// + // Life Cycle Signal Synchronization // + /////////////////////////////////////// + + lc_ctrl_pkg::lc_tx_t lc_creator_seed_sw_rw_en, lc_owner_seed_sw_rw_en, + lc_seed_hw_rd_en, lc_check_byp_en; + lc_ctrl_pkg::lc_tx_t [2:0] lc_dft_en; + // NumAgents + lfsr timer and scrambling datapath. + lc_ctrl_pkg::lc_tx_t [NumAgentsIdx+1:0] lc_escalate_en, lc_escalate_en_synced; + // Single wire for gating assertions in arbitration and CDC primitives. + logic lc_escalate_en_any; + + prim_lc_sync #( + .NumCopies(NumAgentsIdx+2) + ) u_prim_lc_sync_escalate_en ( + .clk_i, + .rst_ni, + .lc_en_i(lc_escalate_en_i), + .lc_en_o(lc_escalate_en_synced) + ); + + prim_lc_sync #( + .NumCopies(1) + ) u_prim_lc_sync_creator_seed_sw_rw_en ( + .clk_i, + .rst_ni, + .lc_en_i(lc_creator_seed_sw_rw_en_i), + .lc_en_o({lc_creator_seed_sw_rw_en}) + ); + + prim_lc_sync #( + .NumCopies(1) + ) u_prim_lc_sync_owner_seed_sw_rw_en ( + .clk_i, + .rst_ni, + .lc_en_i(lc_owner_seed_sw_rw_en_i), + .lc_en_o({lc_owner_seed_sw_rw_en}) + ); + + prim_lc_sync #( + .NumCopies(1) + ) u_prim_lc_sync_seed_hw_rd_en ( + .clk_i, + .rst_ni, + .lc_en_i(lc_seed_hw_rd_en_i), + .lc_en_o({lc_seed_hw_rd_en}) + ); + + prim_lc_sync #( + .NumCopies(3) + ) u_prim_lc_sync_dft_en ( + .clk_i, + .rst_ni, + .lc_en_i(lc_dft_en_i), + .lc_en_o(lc_dft_en) + ); + + prim_lc_sync #( + .NumCopies(1) + ) u_prim_lc_sync_check_byp_en ( + .clk_i, + .rst_ni, + .lc_en_i(lc_check_byp_en_i), + .lc_en_o({lc_check_byp_en}) + ); + + ///////////////////////////////////// + // TL-UL SW partition select logic // + ///////////////////////////////////// + + // The SW partitions share the same TL-UL adapter. + logic tlul_req, tlul_gnt, tlul_rvalid; + logic [SwWindowAddrWidth-1:0] tlul_addr; + logic [1:0] tlul_rerror; + logic [31:0] tlul_rdata; + + import prim_mubi_pkg::MuBi4False; + tlul_adapter_sram #( + .SramAw ( SwWindowAddrWidth ), + .SramDw ( 32 ), + .Outstanding ( 1 ), + .ByteAccess ( 0 ), + .ErrOnWrite ( 1 ) // No write accesses allowed here. + ) u_tlul_adapter_sram ( + .clk_i, + .rst_ni, + .en_ifetch_i ( MuBi4False ), + .tl_i ( tl_win_h2d ), + .tl_o ( tl_win_d2h ), + .req_o ( tlul_req ), + .gnt_i ( tlul_gnt ), + .we_o ( ), // unused + .addr_o ( tlul_addr ), + .wdata_o ( ), // unused + .wmask_o ( ), // unused + // SEC_CM: BUS.INTEGRITY + .intg_error_o ( intg_error[1] ), + .user_rsvd_o ( ), + .rdata_i ( tlul_rdata ), + .rvalid_i ( tlul_rvalid ), + .rerror_i ( tlul_rerror ), + .req_type_o ( ), + .compound_txn_in_progress_o ( ), + .readback_en_i ( MuBi4False ), + .readback_error_o ( ), + .wr_collision_i ( 1'b0 ), + .write_pending_i ( 1'b0 ) + ); + + logic [NumPart-1:0] tlul_part_sel_oh; + for (genvar k = 0; k < NumPart; k++) begin : gen_part_sel + localparam logic [OtpByteAddrWidth:0] PartEnd = (OtpByteAddrWidth+1)'(PartInfo[k].offset) + + (OtpByteAddrWidth+1)'(PartInfo[k].size); + if (PartInfo[k].offset == 0) begin : gen_zero_offset + assign tlul_part_sel_oh[k] = ({1'b0, {tlul_addr, 2'b00}} < PartEnd); + end else begin : gen_nonzero_offset + assign tlul_part_sel_oh[k] = ({tlul_addr, 2'b00} >= PartInfo[k].offset) & + ({1'b0, {tlul_addr, 2'b00}} < PartEnd); + end + end + + `ASSERT(PartSelMustBeOnehot_A, $onehot0(tlul_part_sel_oh)) + + logic [NumPartWidth-1:0] tlul_part_idx; + prim_arbiter_fixed #( + .N(NumPart), + .EnDataPort(0) + ) u_part_sel_idx ( + .clk_i, + .rst_ni, + .req_i ( tlul_part_sel_oh ), + .data_i ( '{default: '0} ), + .gnt_o ( ), // unused + .idx_o ( tlul_part_idx ), + .valid_o ( ), // unused + .data_o ( ), // unused + .ready_i ( 1'b0 ) + ); + + logic tlul_oob_err_d, tlul_oob_err_q; + logic [NumPart-1:0] part_tlul_req, part_tlul_gnt, part_tlul_rvalid; + logic [SwWindowAddrWidth-1:0] part_tlul_addr; + logic [NumPart-1:0][1:0] part_tlul_rerror; + logic [NumPart-1:0][31:0] part_tlul_rdata; + + always_comb begin : p_tlul_assign + // Send request to the correct partition. + part_tlul_addr = tlul_addr; + part_tlul_req = '0; + tlul_oob_err_d = 1'b0; + if (tlul_req) begin + if (tlul_part_sel_oh != '0) begin + part_tlul_req[tlul_part_idx] = 1'b1; + end else begin + // Error out in the next cycle if address was out of bounds. + tlul_oob_err_d = 1'b1; + end + end + + // aggregate TL-UL responses + tlul_gnt = |part_tlul_gnt | tlul_oob_err_q; + tlul_rvalid = |part_tlul_rvalid | tlul_oob_err_q; + tlul_rerror = '0; + tlul_rdata = '0; + for (int k = 0; k < NumPart; k++) begin + tlul_rerror |= part_tlul_rerror[k]; + tlul_rdata |= part_tlul_rdata[k]; + end + end + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_tlul_reg + if (!rst_ni) begin + tlul_oob_err_q <= 1'b0; + end else begin + tlul_oob_err_q <= tlul_oob_err_d; + end + end + + ////////////////////////////// + // Access Defaults and CSRs // + ////////////////////////////// + + // SEC_CM: ACCESS.CTRL.MUBI + part_access_t [NumPart-1:0] part_access_pre, part_access; + always_comb begin : p_access_control + // Assigns default and extracts named CSR read enables for SW_CFG partitions. + // SEC_CM: PART.MEM.REGREN + part_access_pre = named_part_access_pre(reg2hw); + + // Permanently lock DAI write and read access to the life cycle partition. + // The LC partition can only be read from and written to via the LC controller. + // SEC_CM: LC_PART.MEM.SW_NOACCESS + part_access_pre[LifeCycleIdx].write_lock = MuBi8True; + part_access_pre[LifeCycleIdx].read_lock = MuBi8True; + + // Special partitions for keymgr material only become writable when + // provisioning is enabled. + if (lc_ctrl_pkg::lc_tx_test_false_loose(lc_creator_seed_sw_rw_en)) begin + for (int k = 0; k < NumPart; k++) begin + if (PartInfo[k].iskeymgr_creator) begin + part_access_pre[k] = {2{prim_mubi_pkg::MuBi8True}}; + end + end + end + if (lc_ctrl_pkg::lc_tx_test_false_loose(lc_owner_seed_sw_rw_en)) begin + for (int k = 0; k < NumPart; k++) begin + if (PartInfo[k].iskeymgr_owner) begin + part_access_pre[k] = {2{prim_mubi_pkg::MuBi8True}}; + end + end + end + end + + // This prevents the synthesis tool from optimizing the multibit signals. + for (genvar k = 0; k < NumPart; k++) begin : gen_bufs + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_write_lock ( + .clk_i, + .rst_ni, + .mubi_i(part_access_pre[k].write_lock), + .mubi_o(part_access[k].write_lock) + ); + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_read_lock ( + .clk_i, + .rst_ni, + .mubi_i(part_access_pre[k].read_lock), + .mubi_o(part_access[k].read_lock) + ); + end + + ////////////////////// + // DAI-related CSRs // + ////////////////////// + + logic dai_idle; + logic dai_req; + dai_cmd_e dai_cmd; + logic [OtpByteAddrWidth-1:0] dai_addr; + logic [NumDaiWords-1:0][31:0] dai_wdata, dai_rdata; + logic direct_access_regwen_d, direct_access_regwen_q; + + // This is the HWEXT implementation of a RW0C regwen bit. + assign direct_access_regwen_d = (reg2hw.direct_access_regwen.qe && + !reg2hw.direct_access_regwen.q) ? 1'b0 : direct_access_regwen_q; + + // Any write to this register triggers a DAI command. + assign dai_req = reg2hw.direct_access_cmd.digest.qe | + reg2hw.direct_access_cmd.wr.qe | + reg2hw.direct_access_cmd.rd.qe; + + assign dai_cmd = dai_cmd_e'({reg2hw.direct_access_cmd.digest.q, + reg2hw.direct_access_cmd.wr.q, + reg2hw.direct_access_cmd.rd.q}); + + assign dai_addr = reg2hw.direct_access_address.q; + assign dai_wdata = reg2hw.direct_access_wdata; + + // The DAI and the LCI can initiate write transactions, which + // are critical and we must not power down if such transactions + // are pending. Hence, we signal the LCI/DAI idle state to the + // power manager. This signal is flopped here as it has to + // cross a clock boundary to the power manager. + logic dai_prog_idle, lci_prog_idle, otp_idle_d, otp_idle_q; + assign otp_idle_d = lci_prog_idle & dai_prog_idle; + assign pwr_otp_o.otp_idle = otp_idle_q; + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_idle_regwen_regs + if (!rst_ni) begin + otp_idle_q <= 1'b0; + // The regwen bit has to reset to 1 so that CSR accesses are enabled by default. + direct_access_regwen_q <= 1'b1; + end else begin + otp_idle_q <= otp_idle_d; + direct_access_regwen_q <= direct_access_regwen_d; + end + end + + ////////////////////////////////////// + // Ctrl/Status CSRs, Errors, Alerts // + ////////////////////////////////////// + + // Status and error reporting CSRs, error interrupt generation and alerts. + otp_err_e [NumPart+1:0] part_error; + logic [NumAgents-1:0] part_fsm_err; + logic [NumPart+1:0] part_errors_reduced; + logic otp_operation_done, otp_error; + logic fatal_macro_error_d, fatal_macro_error_q; + logic fatal_check_error_d, fatal_check_error_q; + logic fatal_bus_integ_error_d, fatal_bus_integ_error_q; + logic chk_pending, chk_timeout; + logic lfsr_fsm_err, scrmbl_fsm_err; + always_comb begin : p_errors_alerts + // Note: since these are all fatal alert events, we latch them and keep on sending + // alert events via the alert senders. These regs can only be cleared via a system reset. + fatal_macro_error_d = fatal_macro_error_q; + fatal_check_error_d = fatal_check_error_q; + fatal_bus_integ_error_d = fatal_bus_integ_error_q | (|intg_error); + // These are the per-partition buffered escalation inputs + lc_escalate_en = lc_escalate_en_synced; + // Need a single wire for gating assertions in arbitration and CDC primitives. + lc_escalate_en_any = 1'b0; + + // Aggregate all the macro alerts from the partitions + for (int k = 0; k < NumPart; k++) begin + // Filter for critical error codes that should not occur in the field. + fatal_macro_error_d |= part_error[k] == MacroError; + // While uncorrectable ECC errors are always reported, they do not trigger a fatal alert + // event in some partitions like the VENDOR_TEST partition. + if (PartInfo[k].integrity) begin + fatal_macro_error_d |= part_error[k] == MacroEccUncorrError; + end + end + // Aggregate all the macro alerts from the DAI/LCI + for (int k = NumPart; k < NumPart+2; k++) begin + // Filter for critical error codes that should not occur in the field. + fatal_macro_error_d |= part_error[k] inside {MacroError, MacroEccUncorrError}; + end + + // Aggregate all the remaining errors / alerts from the partitions and the DAI/LCI + for (int k = 0; k < NumPart+2; k++) begin + // Set the error bit if the error status of the corresponding partition is nonzero. + // Need to reverse the order here since the field enumeration in hw2reg.status is reversed. + part_errors_reduced[NumPart+1-k] = |part_error[k]; + // Filter for integrity and consistency check failures. + fatal_check_error_d |= part_error[k] inside {CheckFailError, FsmStateError}; + + // If a fatal alert has been observed in any of the partitions/FSMs, + // we locally trigger escalation within OTP, which moves all FSMs + // to a terminal error state. + if (fatal_macro_error_q || fatal_check_error_q) begin + lc_escalate_en[k] = lc_ctrl_pkg::On; + end + if (lc_ctrl_pkg::lc_tx_test_true_strict(lc_escalate_en[k])) begin + lc_escalate_en_any = 1'b1; + end + end + + // Errors from other non-partition FSMs. + fatal_check_error_d |= chk_timeout | + lfsr_fsm_err | + scrmbl_fsm_err | + (|part_fsm_err); + end + + // If we got an error, we trigger an interrupt. + logic [$bits(part_errors_reduced)+4-1:0] interrupt_triggers_d, interrupt_triggers_q; + + // This makes sure that interrupts are not sticky. + assign interrupt_triggers_d = { + part_errors_reduced, + chk_timeout, + lfsr_fsm_err, + scrmbl_fsm_err, + |part_fsm_err + }; + + assign otp_error = |(interrupt_triggers_d & ~interrupt_triggers_q); + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_alert_regs + if (!rst_ni) begin + fatal_macro_error_q <= '0; + fatal_check_error_q <= '0; + fatal_bus_integ_error_q <= '0; + interrupt_triggers_q <= '0; + end else begin + fatal_macro_error_q <= fatal_macro_error_d; + fatal_check_error_q <= fatal_check_error_d; + fatal_bus_integ_error_q <= fatal_bus_integ_error_d; + interrupt_triggers_q <= interrupt_triggers_d; + end + end + + // CSR assignments are done in one combo process so that we can use + // the parameterized digest_assign task below without multiple driver issues. + logic unused_part_digest; + logic [NumPart-1:0][ScrmblBlockWidth-1:0] part_digest; + logic intr_state_otp_operation_done_d, intr_state_otp_operation_done_de; + logic intr_state_otp_error_d, intr_state_otp_error_de; + always_comb begin : p_csr_assign + // Not all partition digests are consumed, and assigning them to an unused_* signal in the + // function below does not seem to work for some linters. + unused_part_digest = ^part_digest; + // Assign named CSRs (like digests). + hw2reg = named_reg_assign(part_digest); + // DAI related CSRs + hw2reg.direct_access_rdata = dai_rdata; + // ANDing this state with dai_idle write-protects all DAI regs during pending operations. + hw2reg.direct_access_regwen.d = direct_access_regwen_q & dai_idle; + // Assign these to the status register. + hw2reg.status = {part_errors_reduced, + chk_timeout, + lfsr_fsm_err, + scrmbl_fsm_err, + part_fsm_err[KdiIdx], + fatal_bus_integ_error_q, + dai_idle, + chk_pending}; + // Error code registers. + hw2reg.err_code = part_error; + // Interrupt signals + hw2reg.intr_state.otp_operation_done.de = intr_state_otp_operation_done_de; + hw2reg.intr_state.otp_operation_done.d = intr_state_otp_operation_done_d; + hw2reg.intr_state.otp_error.de = intr_state_otp_error_de; + hw2reg.intr_state.otp_error.d = intr_state_otp_error_d; +end + + + ////////////////////////////////// + // Interrupts and Alert Senders // + ////////////////////////////////// + + prim_intr_hw #( + .Width(1) + ) u_intr_operation_done ( + .clk_i, + .rst_ni, + .event_intr_i ( otp_operation_done ), + .reg2hw_intr_enable_q_i ( reg2hw.intr_enable.otp_operation_done.q ), + .reg2hw_intr_test_q_i ( reg2hw.intr_test.otp_operation_done.q ), + .reg2hw_intr_test_qe_i ( reg2hw.intr_test.otp_operation_done.qe ), + .reg2hw_intr_state_q_i ( reg2hw.intr_state.otp_operation_done.q ), + .hw2reg_intr_state_de_o ( intr_state_otp_operation_done_de ), + .hw2reg_intr_state_d_o ( intr_state_otp_operation_done_d ), + .intr_o ( intr_otp_operation_done_o ) + ); + + prim_intr_hw #( + .Width(1) + ) u_intr_error ( + .clk_i, + .rst_ni, + .event_intr_i ( otp_error ), + .reg2hw_intr_enable_q_i ( reg2hw.intr_enable.otp_error.q ), + .reg2hw_intr_test_q_i ( reg2hw.intr_test.otp_error.q ), + .reg2hw_intr_test_qe_i ( reg2hw.intr_test.otp_error.qe ), + .reg2hw_intr_state_q_i ( reg2hw.intr_state.otp_error.q ), + .hw2reg_intr_state_de_o ( intr_state_otp_error_de ), + .hw2reg_intr_state_d_o ( intr_state_otp_error_d ), + .intr_o ( intr_otp_error_o ) + ); + + logic [NumAlerts-1:0] alerts; + logic [NumAlerts-1:0] alert_test; + logic fatal_prim_otp_alert, recov_prim_otp_alert; + + assign alerts = { + recov_prim_otp_alert, + fatal_prim_otp_alert, + fatal_bus_integ_error_q, + fatal_check_error_q, + fatal_macro_error_q + }; + + assign alert_test = { + reg2hw.alert_test.recov_prim_otp_alert.q & + reg2hw.alert_test.recov_prim_otp_alert.qe, + reg2hw.alert_test.fatal_prim_otp_alert.q & + reg2hw.alert_test.fatal_prim_otp_alert.qe, + reg2hw.alert_test.fatal_bus_integ_error.q & + reg2hw.alert_test.fatal_bus_integ_error.qe, + reg2hw.alert_test.fatal_check_error.q & + reg2hw.alert_test.fatal_check_error.qe, + reg2hw.alert_test.fatal_macro_error.q & + reg2hw.alert_test.fatal_macro_error.qe + }; + + localparam logic [NumAlerts-1:0] AlertIsFatal = { + 1'b0, // recov_prim_otp_alert + 1'b1, // fatal_prim_otp_alert + 1'b1, // fatal_bus_integ_error_q + 1'b1, // fatal_check_error_q + 1'b1 // fatal_macro_error_q + }; + + for (genvar k = 0; k < NumAlerts; k++) begin : gen_alert_tx + prim_alert_sender #( + .AsyncOn(AlertAsyncOn[k]), + .IsFatal(AlertIsFatal[k]) + ) u_prim_alert_sender ( + .clk_i, + .rst_ni, + .alert_test_i ( alert_test[k] ), + .alert_req_i ( alerts[k] ), + .alert_ack_o ( ), + .alert_state_o ( ), + .alert_rx_i ( alert_rx_i[k] ), + .alert_tx_o ( alert_tx_o[k] ) + ); + end + + //////////////////////////////// + // LFSR Timer and CSR mapping // + //////////////////////////////// + + logic integ_chk_trig, cnsty_chk_trig; + logic [NumPart-1:0] integ_chk_req, integ_chk_ack; + logic [NumPart-1:0] cnsty_chk_req, cnsty_chk_ack; + logic lfsr_edn_req, lfsr_edn_ack; + logic [EdnDataWidth-1:0] edn_data; + + assign integ_chk_trig = reg2hw.check_trigger.integrity.q & + reg2hw.check_trigger.integrity.qe; + assign cnsty_chk_trig = reg2hw.check_trigger.consistency.q & + reg2hw.check_trigger.consistency.qe; + + // SEC_CM: PART.DATA_REG.BKGN_CHK + otp_ctrl_lfsr_timer #( + .RndCnstLfsrSeed(RndCnstLfsrSeed), + .RndCnstLfsrPerm(RndCnstLfsrPerm) + ) u_otp_ctrl_lfsr_timer ( + .clk_i, + .rst_ni, + .edn_req_o ( lfsr_edn_req ), + .edn_ack_i ( lfsr_edn_ack ), + .edn_data_i ( edn_data ), + // We can enable the timer once OTP has initialized. + // Note that this is only the initial release that gets + // the timer FSM into an operational state. + // Whether or not the timers / background checks are + // activated depends on the CSR configuration (by default + // they are switched off). + .timer_en_i ( pwr_otp_o.otp_done ), + // This idle signal is the same that is output to the power + // manager, and indicates whether there is an ongoing OTP programming + // operation. It is used to pause the consistency check timeout + // counter in order to prevent spurious timeouts (OTP programming + // operations are very slow compared to readout operations and can + // hence interfere with the timeout mechanism). + .otp_prog_busy_i ( ~otp_idle_d ), + .integ_chk_trig_i ( integ_chk_trig ), + .cnsty_chk_trig_i ( cnsty_chk_trig ), + .chk_pending_o ( chk_pending ), + .timeout_i ( reg2hw.check_timeout.q ), + .integ_period_msk_i ( reg2hw.integrity_check_period.q ), + .cnsty_period_msk_i ( reg2hw.consistency_check_period.q ), + .integ_chk_req_o ( integ_chk_req ), + .cnsty_chk_req_o ( cnsty_chk_req ), + .integ_chk_ack_i ( integ_chk_ack ), + .cnsty_chk_ack_i ( cnsty_chk_ack ), + .escalate_en_i ( lc_escalate_en[NumAgents] ), + .chk_timeout_o ( chk_timeout ), + .fsm_err_o ( lfsr_fsm_err ) + ); + + /////////////////////////////////////// + // EDN Arbitration, Request and Sync // + /////////////////////////////////////// + + // Both the key derivation and LFSR reseeding are low bandwidth, + // hence they can share the same EDN interface. + logic edn_req, edn_ack; + logic key_edn_req, key_edn_ack; + prim_arbiter_tree #( + .N(2), + .EnDataPort(0) + ) u_edn_arb ( + .clk_i, + .rst_ni, + .req_chk_i ( ~lc_escalate_en_any ), + .req_i ( {lfsr_edn_req, key_edn_req} ), + .data_i ( '{default: '0} ), + .gnt_o ( {lfsr_edn_ack, key_edn_ack} ), + .idx_o ( ), // unused + .valid_o ( edn_req ), + .data_o ( ), // unused + .ready_i ( edn_ack ) + ); + + // This synchronizes the data coming from EDN and stacks the + // 32bit EDN words to achieve an internal entropy width of 64bit. + prim_edn_req #( + .OutWidth(EdnDataWidth) + ) u_prim_edn_req ( + .clk_i, + .rst_ni, + .req_chk_i ( ~lc_escalate_en_any ), + .req_i ( edn_req ), + .ack_o ( edn_ack ), + .data_o ( edn_data ), + .fips_o ( ), // unused + .err_o ( ), // unused + .clk_edn_i, + .rst_edn_ni, + .edn_o, + .edn_i + ); + + /////////////////////////////// + // OTP Macro and Arbitration // + /////////////////////////////// + + typedef struct packed { + prim_otp_pkg::cmd_e cmd; + logic [OtpSizeWidth-1:0] size; // Number of native words to write. + logic [OtpIfWidth-1:0] wdata; + logic [OtpAddrWidth-1:0] addr; // Halfword address. + } otp_bundle_t; + + logic [NumAgents-1:0] part_otp_arb_req, part_otp_arb_gnt; + otp_bundle_t part_otp_arb_bundle [NumAgents]; + logic otp_arb_valid, otp_arb_ready; + logic otp_prim_valid, otp_prim_ready; + logic otp_rsp_fifo_valid, otp_rsp_fifo_ready; + logic [vbits(NumAgents)-1:0] otp_arb_idx; + otp_bundle_t otp_arb_bundle; + + // The OTP interface is arbitrated on a per-cycle basis, meaning that back-to-back + // transactions can be completely independent. + prim_arbiter_tree #( + .N(NumAgents), + .DW($bits(otp_bundle_t)) + ) u_otp_arb ( + .clk_i, + .rst_ni, + .req_chk_i ( ~lc_escalate_en_any ), + .req_i ( part_otp_arb_req ), + .data_i ( part_otp_arb_bundle ), + .gnt_o ( part_otp_arb_gnt ), + .idx_o ( otp_arb_idx ), + .valid_o ( otp_arb_valid ), + .data_o ( otp_arb_bundle ), + .ready_i ( otp_arb_ready ) + ); + + // Don't issue more transactions than what the rsp_fifo can keep track of. + assign otp_arb_ready = otp_prim_ready & otp_rsp_fifo_ready; + assign otp_prim_valid = otp_arb_valid & otp_rsp_fifo_ready; + assign otp_rsp_fifo_valid = otp_prim_ready & otp_prim_valid; + + prim_otp_pkg::err_e part_otp_err; + logic [OtpIfWidth-1:0] part_otp_rdata; + logic otp_rvalid; + tlul_pkg::tl_h2d_t prim_tl_h2d_gated; + tlul_pkg::tl_d2h_t prim_tl_d2h_gated; + + // Life cycle qualification of TL-UL test interface. + // SEC_CM: TEST.BUS.LC_GATED + // SEC_CM: TEST_TL_LC_GATE.FSM.SPARSE + tlul_lc_gate #( + .NumGatesPerDirection(2) + ) u_tlul_lc_gate ( + .clk_i, + .rst_ni, + .tl_h2d_i(prim_tl_i), + .tl_d2h_o(prim_tl_o), + .tl_h2d_o(prim_tl_h2d_gated), + .tl_d2h_i(prim_tl_d2h_gated), + .lc_en_i (lc_dft_en[0]), + .flush_req_i('0), + .flush_ack_o(), + .resp_pending_o(), + .err_o (intg_error[2]) + ); + + // Test-related GPIOs. + // SEC_CM: TEST.BUS.LC_GATED + logic [OtpTestVectWidth-1:0] otp_test_vect; + assign cio_test_o = (lc_ctrl_pkg::lc_tx_test_true_strict(lc_dft_en[1])) ? + otp_test_vect : '0; + assign cio_test_en_o = (lc_ctrl_pkg::lc_tx_test_true_strict(lc_dft_en[2])) ? + {OtpTestVectWidth{1'b1}} : '0; + + // SEC_CM: MACRO.MEM.CM, MACRO.MEM.INTEGRITY + prim_otp #( + .Width ( OtpWidth ), + .Depth ( OtpDepth ), + .SizeWidth ( OtpSizeWidth ), + .PwrSeqWidth ( OtpPwrSeqWidth ), + .TestCtrlWidth ( OtpTestCtrlWidth ), + .TestStatusWidth ( OtpTestStatusWidth ), + .TestVectWidth ( OtpTestVectWidth ), + .MemInitFile ( MemInitFile ), + .VendorTestOffset ( VendorTestOffset ), + .VendorTestSize ( VendorTestSize ) + ) u_otp ( + .clk_i, + .rst_ni, + // Observability controls to/from AST + .obs_ctrl_i, + .otp_obs_o, + // Power sequencing signals to/from AST + .pwr_seq_o ( otp_ast_pwr_seq_o.pwr_seq ), + .pwr_seq_h_i ( otp_ast_pwr_seq_h_i.pwr_seq_h ), + .ext_voltage_io ( otp_ext_voltage_h_io ), + // Test interface + .test_ctrl_i ( lc_otp_vendor_test_i.ctrl ), + .test_status_o ( lc_otp_vendor_test_o.status ), + .test_vect_o ( otp_test_vect ), + .test_tl_i ( prim_tl_h2d_gated ), + .test_tl_o ( prim_tl_d2h_gated ), + // Other DFT signals + .scan_en_i, + .scan_rst_ni, + .scanmode_i, + // Alerts + .fatal_alert_o ( fatal_prim_otp_alert ), + .recov_alert_o ( recov_prim_otp_alert ), + // Read / Write command interface + .ready_o ( otp_prim_ready ), + .valid_i ( otp_prim_valid ), + .cmd_i ( otp_arb_bundle.cmd ), + .size_i ( otp_arb_bundle.size ), + .addr_i ( otp_arb_bundle.addr ), + .wdata_i ( otp_arb_bundle.wdata ), + // Read data out + .valid_o ( otp_rvalid ), + .rdata_o ( part_otp_rdata ), + .err_o ( part_otp_err ) + ); + + logic otp_fifo_valid; + logic [vbits(NumAgents)-1:0] otp_part_idx; + logic [NumAgents-1:0] part_otp_rvalid; + + // We can have up to two OTP commands in flight, hence we size this to be 2 deep. + // The partitions can unconditionally sink requested data. + prim_fifo_sync #( + .Width(vbits(NumAgents)), + .Depth(2) + ) u_otp_rsp_fifo ( + .clk_i, + .rst_ni, + .clr_i ( 1'b0 ), + .wvalid_i ( otp_rsp_fifo_valid ), + .wready_o ( otp_rsp_fifo_ready ), + .wdata_i ( otp_arb_idx ), + .rvalid_o ( otp_fifo_valid ), + .rready_i ( otp_rvalid ), + .rdata_o ( otp_part_idx ), + .depth_o ( ), + .full_o ( ), + .err_o ( ) + ); + + // Steer response back to the partition where this request originated. + always_comb begin : p_rvalid + part_otp_rvalid = '0; + part_otp_rvalid[otp_part_idx] = otp_rvalid & otp_fifo_valid; + end + + // Note that this must be true by construction. + `ASSERT(OtpRespFifoUnderflow_A, otp_rvalid |-> otp_fifo_valid) + + ///////////////////////////////////////// + // Scrambling Datapath and Arbitration // + ///////////////////////////////////////// + + // Note: as opposed to the OTP arbitration above, we do not perform cycle-wise arbitration, but + // transaction-wise arbitration. This is implemented using a RR arbiter that acts as a mutex. + // I.e., each agent (e.g. the DAI or a partition) can request a lock on the mutex. Once granted, + // the partition can keep the lock as long as needed for the transaction to complete. The + // partition must yield its lock by deasserting the request signal for the arbiter to proceed. + // Since this scheme does not have built-in preemtion, it must be ensured that the agents + // eventually release their locks for this to be fair. + // + // See also https://docs.opentitan.org/hw/ip/otp_ctrl/index.html#otp-controller-overview for + // details. + typedef struct packed { + otp_scrmbl_cmd_e cmd; + digest_mode_e mode; + logic [ConstSelWidth-1:0] sel; + logic [ScrmblBlockWidth-1:0] data; + logic valid; + } scrmbl_bundle_t; + + logic [NumAgents-1:0] part_scrmbl_mtx_req, part_scrmbl_mtx_gnt; + scrmbl_bundle_t part_scrmbl_req_bundle [NumAgents]; + scrmbl_bundle_t scrmbl_req_bundle; + logic [vbits(NumAgents)-1:0] scrmbl_mtx_idx; + logic scrmbl_mtx_valid; + + // Note that arbiter decisions do not change when backpressured. + // Hence, the idx_o signal is guaranteed to remain stable until ack'ed. + prim_arbiter_tree #( + .N(NumAgents), + .DW($bits(scrmbl_bundle_t)) + ) u_scrmbl_mtx ( + .clk_i, + .rst_ni, + .req_chk_i ( 1'b0 ), // REQ is allowed to go low again without ACK even + // during normal operation. + .req_i ( part_scrmbl_mtx_req ), + .data_i ( part_scrmbl_req_bundle ), + .gnt_o ( ), + .idx_o ( scrmbl_mtx_idx ), + .valid_o ( scrmbl_mtx_valid ), + .data_o ( scrmbl_req_bundle ), + .ready_i ( 1'b0 ) + ); + + // Since the ready_i signal of the arbiter is statically set to 1'b0 above, we are always in a + // "backpressure" situation, where the RR arbiter will automatically advance the internal RR state + // to give the current winner max priority in subsequent cycles in order to keep the decision + // stable. Rearbitration occurs once the winning agent deasserts its request. + always_comb begin : p_mutex + part_scrmbl_mtx_gnt = '0; + part_scrmbl_mtx_gnt[scrmbl_mtx_idx] = scrmbl_mtx_valid; + end + + logic [ScrmblBlockWidth-1:0] part_scrmbl_rsp_data; + logic scrmbl_arb_req_ready, scrmbl_arb_rsp_valid; + logic [NumAgents-1:0] part_scrmbl_req_ready, part_scrmbl_rsp_valid; + + // SEC_CM: SECRET.MEM.SCRAMBLE + // SEC_CM: PART.MEM.DIGEST + otp_ctrl_scrmbl u_otp_ctrl_scrmbl ( + .clk_i, + .rst_ni, + .cmd_i ( scrmbl_req_bundle.cmd ), + .mode_i ( scrmbl_req_bundle.mode ), + .sel_i ( scrmbl_req_bundle.sel ), + .data_i ( scrmbl_req_bundle.data ), + .valid_i ( scrmbl_req_bundle.valid ), + .ready_o ( scrmbl_arb_req_ready ), + .data_o ( part_scrmbl_rsp_data ), + .valid_o ( scrmbl_arb_rsp_valid ), + .escalate_en_i ( lc_escalate_en[NumAgents+1] ), + .fsm_err_o ( scrmbl_fsm_err ) + ); + + // steer back responses + always_comb begin : p_scmrbl_resp + part_scrmbl_req_ready = '0; + part_scrmbl_rsp_valid = '0; + part_scrmbl_req_ready[scrmbl_mtx_idx] = scrmbl_arb_req_ready; + part_scrmbl_rsp_valid[scrmbl_mtx_idx] = scrmbl_arb_rsp_valid; + end + + ///////////////////////////// + // Direct Access Interface // + ///////////////////////////// + + logic part_init_req; + logic [NumPart-1:0] part_init_done; + part_access_t [NumPart-1:0] part_access_dai; + + // The init request comes from the power manager, which lives in the AON clock domain. + logic pwr_otp_req_synced; + prim_flop_2sync #( + .Width(1) + ) u_otp_init_sync ( + .clk_i, + .rst_ni, + .d_i ( pwr_otp_i.otp_init ), + .q_o ( pwr_otp_req_synced ) + ); + + // Register this signal as it has to cross a clock boundary. + logic pwr_otp_rsp_d, pwr_otp_rsp_q; + assign pwr_otp_o.otp_done = pwr_otp_rsp_q; + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_init_reg + if (!rst_ni) begin + pwr_otp_rsp_q <= 1'b0; + end else begin + pwr_otp_rsp_q <= pwr_otp_rsp_d; + end + end + + otp_ctrl_dai u_otp_ctrl_dai ( + .clk_i, + .rst_ni, + .init_req_i ( pwr_otp_req_synced ), + .init_done_o ( pwr_otp_rsp_d ), + .part_init_req_o ( part_init_req ), + .part_init_done_i ( part_init_done ), + .escalate_en_i ( lc_escalate_en[DaiIdx] ), + .error_o ( part_error[DaiIdx] ), + .fsm_err_o ( part_fsm_err[DaiIdx] ), + .part_access_i ( part_access_dai ), + .dai_addr_i ( dai_addr ), + .dai_cmd_i ( dai_cmd ), + .dai_req_i ( dai_req ), + .dai_wdata_i ( dai_wdata ), + .dai_idle_o ( dai_idle ), + .dai_prog_idle_o ( dai_prog_idle ), + .dai_cmd_done_o ( otp_operation_done ), + .dai_rdata_o ( dai_rdata ), + .otp_req_o ( part_otp_arb_req[DaiIdx] ), + .otp_cmd_o ( part_otp_arb_bundle[DaiIdx].cmd ), + .otp_size_o ( part_otp_arb_bundle[DaiIdx].size ), + .otp_wdata_o ( part_otp_arb_bundle[DaiIdx].wdata ), + .otp_addr_o ( part_otp_arb_bundle[DaiIdx].addr ), + .otp_gnt_i ( part_otp_arb_gnt[DaiIdx] ), + .otp_rvalid_i ( part_otp_rvalid[DaiIdx] ), + .otp_rdata_i ( part_otp_rdata ), + .otp_err_i ( part_otp_err ), + .scrmbl_mtx_req_o ( part_scrmbl_mtx_req[DaiIdx] ), + .scrmbl_mtx_gnt_i ( part_scrmbl_mtx_gnt[DaiIdx] ), + .scrmbl_cmd_o ( part_scrmbl_req_bundle[DaiIdx].cmd ), + .scrmbl_mode_o ( part_scrmbl_req_bundle[DaiIdx].mode ), + .scrmbl_sel_o ( part_scrmbl_req_bundle[DaiIdx].sel ), + .scrmbl_data_o ( part_scrmbl_req_bundle[DaiIdx].data ), + .scrmbl_valid_o ( part_scrmbl_req_bundle[DaiIdx].valid ), + .scrmbl_ready_i ( part_scrmbl_req_ready[DaiIdx] ), + .scrmbl_valid_i ( part_scrmbl_rsp_valid[DaiIdx] ), + .scrmbl_data_i ( part_scrmbl_rsp_data ) + ); + + //////////////////////////////////// + // Lifecycle Transition Interface // + //////////////////////////////////// + + logic [PartInfo[LifeCycleIdx].size-1:0][7:0] lc_otp_program_data; + assign lc_otp_program_data[LcStateOffset-LifeCycleOffset +: LcStateSize] = + lc_otp_program_i.state; + assign lc_otp_program_data[LcTransitionCntOffset-LifeCycleOffset +: LcTransitionCntSize] = + lc_otp_program_i.count; + + otp_ctrl_lci #( + .Info(PartInfo[LifeCycleIdx]) + ) u_otp_ctrl_lci ( + .clk_i, + .rst_ni, + .lci_en_i ( pwr_otp_o.otp_done ), + .escalate_en_i ( lc_escalate_en[LciIdx] ), + .error_o ( part_error[LciIdx] ), + .fsm_err_o ( part_fsm_err[LciIdx] ), + .lci_prog_idle_o ( lci_prog_idle ), + .lc_req_i ( lc_otp_program_i.req ), + .lc_data_i ( lc_otp_program_data ), + .lc_ack_o ( lc_otp_program_o.ack ), + .lc_err_o ( lc_otp_program_o.err ), + .otp_req_o ( part_otp_arb_req[LciIdx] ), + .otp_cmd_o ( part_otp_arb_bundle[LciIdx].cmd ), + .otp_size_o ( part_otp_arb_bundle[LciIdx].size ), + .otp_wdata_o ( part_otp_arb_bundle[LciIdx].wdata ), + .otp_addr_o ( part_otp_arb_bundle[LciIdx].addr ), + .otp_gnt_i ( part_otp_arb_gnt[LciIdx] ), + .otp_rvalid_i ( part_otp_rvalid[LciIdx] ), + .otp_rdata_i ( part_otp_rdata ), + .otp_err_i ( part_otp_err ) + ); + + // Tie off unused connections. + assign part_scrmbl_mtx_req[LciIdx] = '0; + assign part_scrmbl_req_bundle[LciIdx] = '0; + + // This stops lint from complaining about unused signals. + logic unused_lci_scrmbl_sigs; + assign unused_lci_scrmbl_sigs = ^{part_scrmbl_mtx_gnt[LciIdx], + part_scrmbl_req_ready[LciIdx], + part_scrmbl_rsp_valid[LciIdx]}; + + //////////////////////////////////// + // Key Derivation Interface (KDI) // + //////////////////////////////////// + + logic scrmbl_key_seed_valid; + logic [SramKeySeedWidth-1:0] sram_data_key_seed; + logic [FlashKeySeedWidth-1:0] flash_data_key_seed, flash_addr_key_seed; + + otp_ctrl_kdi #( + .RndCnstScrmblKeyInit(RndCnstScrmblKeyInit) + ) u_otp_ctrl_kdi ( + .clk_i, + .rst_ni, + .kdi_en_i ( pwr_otp_o.otp_done ), + .escalate_en_i ( lc_escalate_en[KdiIdx] ), + .fsm_err_o ( part_fsm_err[KdiIdx] ), + .scrmbl_key_seed_valid_i ( scrmbl_key_seed_valid ), + .flash_data_key_seed_i ( flash_data_key_seed ), + .flash_addr_key_seed_i ( flash_addr_key_seed ), + .sram_data_key_seed_i ( sram_data_key_seed ), + .edn_req_o ( key_edn_req ), + .edn_ack_i ( key_edn_ack ), + .edn_data_i ( edn_data ), + .flash_otp_key_i, + .flash_otp_key_o, + .sram_otp_key_i, + .sram_otp_key_o, + .otbn_otp_key_i, + .otbn_otp_key_o, + .scrmbl_mtx_req_o ( part_scrmbl_mtx_req[KdiIdx] ), + .scrmbl_mtx_gnt_i ( part_scrmbl_mtx_gnt[KdiIdx] ), + .scrmbl_cmd_o ( part_scrmbl_req_bundle[KdiIdx].cmd ), + .scrmbl_mode_o ( part_scrmbl_req_bundle[KdiIdx].mode ), + .scrmbl_sel_o ( part_scrmbl_req_bundle[KdiIdx].sel ), + .scrmbl_data_o ( part_scrmbl_req_bundle[KdiIdx].data ), + .scrmbl_valid_o ( part_scrmbl_req_bundle[KdiIdx].valid ), + .scrmbl_ready_i ( part_scrmbl_req_ready[KdiIdx] ), + .scrmbl_valid_i ( part_scrmbl_rsp_valid[KdiIdx] ), + .scrmbl_data_i ( part_scrmbl_rsp_data ) + ); + + // Tie off OTP bus access, since this is not needed. + assign part_otp_arb_req[KdiIdx] = 1'b0; + assign part_otp_arb_bundle[KdiIdx] = '0; + + // This stops lint from complaining about unused signals. + logic unused_kdi_otp_sigs; + assign unused_kdi_otp_sigs = ^{part_otp_arb_gnt[KdiIdx], + part_otp_rvalid[KdiIdx]}; + + ///////////////////////// + // Partition Instances // + ///////////////////////// + + logic [$bits(PartInvDefault)/8-1:0][7:0] part_buf_data; + + for (genvar k = 0; k < NumPart; k ++) begin : gen_partitions + //////////////////////////////////////////////////////////////////////////////////////////////// + if (PartInfo[k].variant == Unbuffered) begin : gen_unbuffered + otp_ctrl_part_unbuf #( + .Info(PartInfo[k]) + ) u_part_unbuf ( + .clk_i, + .rst_ni, + .init_req_i ( part_init_req ), + .init_done_o ( part_init_done[k] ), + .escalate_en_i ( lc_escalate_en[k] ), + .error_o ( part_error[k] ), + .fsm_err_o ( part_fsm_err[k] ), + .access_i ( part_access[k] ), + .access_o ( part_access_dai[k] ), + .digest_o ( part_digest[k] ), + .tlul_req_i ( part_tlul_req[k] ), + .tlul_gnt_o ( part_tlul_gnt[k] ), + .tlul_addr_i ( part_tlul_addr ), + .tlul_rerror_o ( part_tlul_rerror[k] ), + .tlul_rvalid_o ( part_tlul_rvalid[k] ), + .tlul_rdata_o ( part_tlul_rdata[k] ), + .otp_req_o ( part_otp_arb_req[k] ), + .otp_cmd_o ( part_otp_arb_bundle[k].cmd ), + .otp_size_o ( part_otp_arb_bundle[k].size ), + .otp_wdata_o ( part_otp_arb_bundle[k].wdata ), + .otp_addr_o ( part_otp_arb_bundle[k].addr ), + .otp_gnt_i ( part_otp_arb_gnt[k] ), + .otp_rvalid_i ( part_otp_rvalid[k] ), + .otp_rdata_i ( part_otp_rdata ), + .otp_err_i ( part_otp_err ) + ); + + // Tie off unused connections. + assign part_scrmbl_mtx_req[k] = '0; + assign part_scrmbl_req_bundle[k] = '0; + // These checks do not exist in this partition type, + // so we always acknowledge the request. + assign integ_chk_ack[k] = 1'b1; + assign cnsty_chk_ack[k] = 1'b1; + + // No buffered data to expose. + assign part_buf_data[PartInfo[k].offset +: PartInfo[k].size] = '0; + + // This stops lint from complaining about unused signals. + logic unused_part_scrmbl_sigs; + assign unused_part_scrmbl_sigs = ^{part_scrmbl_mtx_gnt[k], + part_scrmbl_req_ready[k], + part_scrmbl_rsp_valid[k], + integ_chk_req[k], + cnsty_chk_req[k]}; + + // Alert assertion for sparse FSM. + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlPartUnbufFsmCheck_A, + u_part_unbuf.u_state_regs, alert_tx_o[1]) + //////////////////////////////////////////////////////////////////////////////////////////////// + end else if (PartInfo[k].variant == Buffered) begin : gen_buffered + otp_ctrl_part_buf #( + .Info(PartInfo[k]), + .DataDefault(PartInvDefault[PartInfo[k].offset*8 +: PartInfo[k].size*8]) + ) u_part_buf ( + .clk_i, + .rst_ni, + .init_req_i ( part_init_req ), + .init_done_o ( part_init_done[k] ), + .integ_chk_req_i ( integ_chk_req[k] ), + .integ_chk_ack_o ( integ_chk_ack[k] ), + .cnsty_chk_req_i ( cnsty_chk_req[k] ), + .cnsty_chk_ack_o ( cnsty_chk_ack[k] ), + .escalate_en_i ( lc_escalate_en[k] ), + // Only supported by life cycle partition (see further below). + .check_byp_en_i ( lc_ctrl_pkg::Off ), + .error_o ( part_error[k] ), + .fsm_err_o ( part_fsm_err[k] ), + .access_i ( part_access[k] ), + .access_o ( part_access_dai[k] ), + .digest_o ( part_digest[k] ), + .data_o ( part_buf_data[PartInfo[k].offset +: PartInfo[k].size] ), + .otp_req_o ( part_otp_arb_req[k] ), + .otp_cmd_o ( part_otp_arb_bundle[k].cmd ), + .otp_size_o ( part_otp_arb_bundle[k].size ), + .otp_wdata_o ( part_otp_arb_bundle[k].wdata ), + .otp_addr_o ( part_otp_arb_bundle[k].addr ), + .otp_gnt_i ( part_otp_arb_gnt[k] ), + .otp_rvalid_i ( part_otp_rvalid[k] ), + .otp_rdata_i ( part_otp_rdata ), + .otp_err_i ( part_otp_err ), + .scrmbl_mtx_req_o ( part_scrmbl_mtx_req[k] ), + .scrmbl_mtx_gnt_i ( part_scrmbl_mtx_gnt[k] ), + .scrmbl_cmd_o ( part_scrmbl_req_bundle[k].cmd ), + .scrmbl_mode_o ( part_scrmbl_req_bundle[k].mode ), + .scrmbl_sel_o ( part_scrmbl_req_bundle[k].sel ), + .scrmbl_data_o ( part_scrmbl_req_bundle[k].data ), + .scrmbl_valid_o ( part_scrmbl_req_bundle[k].valid ), + .scrmbl_ready_i ( part_scrmbl_req_ready[k] ), + .scrmbl_valid_i ( part_scrmbl_rsp_valid[k] ), + .scrmbl_data_i ( part_scrmbl_rsp_data ) + ); + + // Buffered partitions are not accessible via the TL-UL window. + logic unused_part_tlul_sigs; + assign unused_part_tlul_sigs = ^part_tlul_req[k]; + assign part_tlul_gnt[k] = 1'b0; + assign part_tlul_rerror[k] = '0; + assign part_tlul_rvalid[k] = 1'b0; + assign part_tlul_rdata[k] = '0; + + // Alert assertion for sparse FSM. + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlPartBufFsmCheck_A, + u_part_buf.u_state_regs, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntPartBufCheck_A, + u_part_buf.u_prim_count, alert_tx_o[1]) + //////////////////////////////////////////////////////////////////////////////////////////////// + end else if (PartInfo[k].variant == LifeCycle) begin : gen_lifecycle + otp_ctrl_part_buf #( + .Info(PartInfo[k]), + .DataDefault(PartInvDefault[PartInfo[k].offset*8 +: PartInfo[k].size*8]) + ) u_part_buf ( + .clk_i, + .rst_ni, + .init_req_i ( part_init_req ), + .init_done_o ( part_init_done[k] ), + .integ_chk_req_i ( integ_chk_req[k] ), + .integ_chk_ack_o ( integ_chk_ack[k] ), + .cnsty_chk_req_i ( cnsty_chk_req[k] ), + .cnsty_chk_ack_o ( cnsty_chk_ack[k] ), + .escalate_en_i ( lc_escalate_en[k] ), + // This is only supported by the life cycle partition. We need to prevent this partition + // from escalating once the life cycle state in memory is being updated (and hence not + // consistent with the values in the buffer regs anymore). + .check_byp_en_i ( lc_check_byp_en ), + .error_o ( part_error[k] ), + .fsm_err_o ( part_fsm_err[k] ), + .access_i ( part_access[k] ), + .access_o ( part_access_dai[k] ), + .digest_o ( part_digest[k] ), + .data_o ( part_buf_data[PartInfo[k].offset +: PartInfo[k].size] ), + .otp_req_o ( part_otp_arb_req[k] ), + .otp_cmd_o ( part_otp_arb_bundle[k].cmd ), + .otp_size_o ( part_otp_arb_bundle[k].size ), + .otp_wdata_o ( part_otp_arb_bundle[k].wdata ), + .otp_addr_o ( part_otp_arb_bundle[k].addr ), + .otp_gnt_i ( part_otp_arb_gnt[k] ), + .otp_rvalid_i ( part_otp_rvalid[k] ), + .otp_rdata_i ( part_otp_rdata ), + .otp_err_i ( part_otp_err ), + // The LC partition does not need any scrambling features. + .scrmbl_mtx_req_o ( ), + .scrmbl_mtx_gnt_i ( 1'b0 ), + .scrmbl_cmd_o ( ), + .scrmbl_mode_o ( ), + .scrmbl_sel_o ( ), + .scrmbl_data_o ( ), + .scrmbl_valid_o ( ), + .scrmbl_ready_i ( 1'b0 ), + .scrmbl_valid_i ( 1'b0 ), + .scrmbl_data_i ( '0 ) + ); + + // Buffered partitions are not accessible via the TL-UL window. + logic unused_part_tlul_sigs; + assign unused_part_tlul_sigs = ^part_tlul_req[k]; + assign part_tlul_gnt[k] = 1'b0; + assign part_tlul_rerror[k] = '0; + assign part_tlul_rvalid[k] = 1'b0; + assign part_tlul_rdata[k] = '0; + + // Tie off unused connections. + assign part_scrmbl_mtx_req[k] = '0; + assign part_scrmbl_req_bundle[k] = '0; + + // This stops lint from complaining about unused signals. + logic unused_part_scrmbl_sigs; + assign unused_part_scrmbl_sigs = ^{part_scrmbl_mtx_gnt[k], + part_scrmbl_req_ready[k], + part_scrmbl_rsp_valid[k]}; + // Alert assertion for sparse FSM. + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlPartLcFsmCheck_A, + u_part_buf.u_state_regs, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntPartLcCheck_A, + u_part_buf.u_prim_count, alert_tx_o[1]) + //////////////////////////////////////////////////////////////////////////////////////////////// + end else begin : gen_invalid + // This is invalid and should break elaboration + assert_static_in_generate_invalid assert_static_in_generate_invalid(); + end + //////////////////////////////////////////////////////////////////////////////////////////////// + end + + ////////////////////////////////// + // Buffered Data Output Mapping // + ////////////////////////////////// + + // Output complete hardware config partition. + // Actual mapping to other IPs is done via the intersignal topgen feature, + // selection of fields can be done using the otp_hw_cfg_t struct fields. + otp_broadcast_t otp_broadcast; + assign otp_broadcast = named_broadcast_assign(part_init_done, part_buf_data); + + // Make sure the broadcast valid is flopped before sending it out. + lc_ctrl_pkg::lc_tx_t otp_broadcast_valid_q; + prim_lc_sender u_prim_lc_sender_otp_broadcast_valid ( + .clk_i, + .rst_ni, + .lc_en_i(otp_broadcast.valid), + .lc_en_o(otp_broadcast_valid_q) + ); + + always_comb begin : p_otp_broadcast_valid + otp_broadcast_o = otp_broadcast; + otp_broadcast_o.valid = otp_broadcast_valid_q; + end + + // Root keys and seeds. + // This uses a generated function to assign all collateral that is marked with "iskeymgr" in + // the memory map. Note that in this case the type is static and represents a superset of all + // options so that we can maintain a stable interface with keymgr (otherwise keymgr will have + // to be templated as well. Unused key material will be tied off to '0. The keymgr has to be + // parameterized accordingly (via SV parameters) to consume the correct key material. + // + // The key material valid signals are set to true if the corresponding digest is nonzero and the + // partition is initialized. On top of that, the entire output is gated by lc_seed_hw_rd_en. + otp_keymgr_key_t otp_keymgr_key; + assign otp_keymgr_key = named_keymgr_key_assign(part_digest, + part_buf_data, + lc_seed_hw_rd_en); + + // Note regarding these breakouts: named_keymgr_key_assign will tie off unused key material / + // valid signals to '0. This is the case for instance in system configurations that keep the seed + // material in the flash instead of OTP. + logic creator_root_key_share0_valid_d, creator_root_key_share0_valid_q; + logic creator_root_key_share1_valid_d, creator_root_key_share1_valid_q; + logic creator_seed_valid_d, creator_seed_valid_q; + logic owner_seed_valid_d, owner_seed_valid_q; + prim_flop #( + .Width(4) + ) u_keygmr_key_valid ( + .clk_i, + .rst_ni, + .d_i ({creator_root_key_share0_valid_d, + creator_root_key_share1_valid_d, + creator_seed_valid_d, + owner_seed_valid_d}), + .q_o ({creator_root_key_share0_valid_q, + creator_root_key_share1_valid_q, + creator_seed_valid_q, + owner_seed_valid_q}) + ); + + always_comb begin : p_otp_keymgr_key_valid + // Valid reg inputs + creator_root_key_share0_valid_d = otp_keymgr_key.creator_root_key_share0_valid; + creator_root_key_share1_valid_d = otp_keymgr_key.creator_root_key_share1_valid; + creator_seed_valid_d = otp_keymgr_key.creator_seed_valid; + owner_seed_valid_d = otp_keymgr_key.owner_seed_valid; + // Output to keymgr + otp_keymgr_key_o = otp_keymgr_key; + otp_keymgr_key_o.creator_root_key_share0_valid = creator_root_key_share0_valid_q; + otp_keymgr_key_o.creator_root_key_share1_valid = creator_root_key_share1_valid_q; + otp_keymgr_key_o.creator_seed_valid = creator_seed_valid_q; + otp_keymgr_key_o.owner_seed_valid = owner_seed_valid_q; + end + + // Check that the lc_seed_hw_rd_en remains stable, once the key material is valid. + `ASSERT(LcSeedHwRdEnStable0_A, + $rose(creator_root_key_share0_valid_q) |=> $stable(lc_seed_hw_rd_en) [*1:$], + clk_i, !rst_ni || lc_ctrl_pkg::lc_tx_test_true_loose(lc_escalate_en_i) // Disable if escalating + ) + `ASSERT(LcSeedHwRdEnStable1_A, + $rose(creator_root_key_share1_valid_q) |=> $stable(lc_seed_hw_rd_en) [*1:$], + clk_i, !rst_ni || lc_ctrl_pkg::lc_tx_test_true_loose(lc_escalate_en_i) // Disable if escalating + ) + `ASSERT(LcSeedHwRdEnStable2_A, + $rose(creator_seed_valid_q) |=> $stable(lc_seed_hw_rd_en) [*1:$], + clk_i, !rst_ni || lc_ctrl_pkg::lc_tx_test_true_loose(lc_escalate_en_i) // Disable if escalating + ) + `ASSERT(LcSeedHwRdEnStable3_A, + $rose(owner_seed_valid_q) |=> $stable(lc_seed_hw_rd_en) [*1:$], + clk_i, !rst_ni || lc_ctrl_pkg::lc_tx_test_true_loose(lc_escalate_en_i) // Disable if escalating + ) + + // Scrambling Keys + assign scrmbl_key_seed_valid = part_digest[Secret1Idx] != '0; + assign sram_data_key_seed = part_buf_data[SramDataKeySeedOffset +: + SramDataKeySeedSize]; + assign flash_data_key_seed = part_buf_data[FlashDataKeySeedOffset +: + FlashDataKeySeedSize]; + assign flash_addr_key_seed = part_buf_data[FlashAddrKeySeedOffset +: + FlashAddrKeySeedSize]; + + // Test unlock and exit tokens and RMA token + assign otp_lc_data_o.test_exit_token = part_buf_data[TestExitTokenOffset +: + TestExitTokenSize]; + assign otp_lc_data_o.test_unlock_token = part_buf_data[TestUnlockTokenOffset +: + TestUnlockTokenSize]; + assign otp_lc_data_o.rma_token = part_buf_data[RmaTokenOffset +: + RmaTokenSize]; + + lc_ctrl_pkg::lc_tx_t test_tokens_valid, rma_token_valid, secrets_valid; + // The test tokens have been provisioned. + assign test_tokens_valid = (part_digest[Secret0Idx] != '0) ? lc_ctrl_pkg::On : lc_ctrl_pkg::Off; + // The rma token has been provisioned. + assign rma_token_valid = (part_digest[Secret2Idx] != '0) ? lc_ctrl_pkg::On : lc_ctrl_pkg::Off; + // The device is personalized if the root key has been provisioned and locked. + assign secrets_valid = (part_digest[Secret2Idx] != '0) ? lc_ctrl_pkg::On : lc_ctrl_pkg::Off; + + // Buffer these constants in order to ensure that synthesis does not try to optimize the encoding. + // SEC_CM: TOKEN_VALID.CTRL.MUBI + prim_lc_sender #( + .AsyncOn(0) + ) u_prim_lc_sender_test_tokens_valid ( + .clk_i, + .rst_ni, + .lc_en_i(test_tokens_valid), + .lc_en_o(otp_lc_data_o.test_tokens_valid) + ); + + prim_lc_sender #( + .AsyncOn(0) + ) u_prim_lc_sender_rma_token_valid ( + .clk_i, + .rst_ni, + .lc_en_i(rma_token_valid), + .lc_en_o(otp_lc_data_o.rma_token_valid) + ); + + prim_lc_sender #( + .AsyncOn(0) + ) u_prim_lc_sender_secrets_valid ( + .clk_i, + .rst_ni, + .lc_en_i(secrets_valid), + .lc_en_o(otp_lc_data_o.secrets_valid) + ); + + // Lifecycle state + assign otp_lc_data_o.state = lc_ctrl_state_pkg::lc_state_e'(part_buf_data[LcStateOffset +: + LcStateSize]); + assign otp_lc_data_o.count = lc_ctrl_state_pkg::lc_cnt_e'(part_buf_data[LcTransitionCntOffset +: + LcTransitionCntSize]); + + // Assert life cycle state valid signal only when all partitions have initialized. + assign otp_lc_data_o.valid = &part_init_done; + // Signal whether there are any errors in the life cycle partition (both correctable and + // uncorrectable ones). This bit is made available via the JTAG TAP, which is useful for + // production testing in RAW life cycle state where the OTP regs are not accessible. + assign otp_lc_data_o.error = |part_error[LifeCycleIdx]; + + // Not all bits of part_buf_data are used here. + logic unused_buf_data; + assign unused_buf_data = ^part_buf_data; + + //////////////// + // Assertions // + //////////////// + + `ASSERT_INIT(CreatorRootKeyShare0Size_A, KeyMgrKeyWidth == CreatorRootKeyShare0Size * 8) + `ASSERT_INIT(CreatorRootKeyShare1Size_A, KeyMgrKeyWidth == CreatorRootKeyShare1Size * 8) + `ASSERT_INIT(FlashDataKeySeedSize_A, FlashKeySeedWidth == FlashDataKeySeedSize * 8) + `ASSERT_INIT(FlashAddrKeySeedSize_A, FlashKeySeedWidth == FlashAddrKeySeedSize * 8) + `ASSERT_INIT(SramDataKeySeedSize_A, SramKeySeedWidth == SramDataKeySeedSize * 8) + + `ASSERT_INIT(RmaTokenSize_A, lc_ctrl_state_pkg::LcTokenWidth == RmaTokenSize * 8) + `ASSERT_INIT(TestUnlockTokenSize_A, lc_ctrl_state_pkg::LcTokenWidth == TestUnlockTokenSize * 8) + `ASSERT_INIT(TestExitTokenSize_A, lc_ctrl_state_pkg::LcTokenWidth == TestExitTokenSize * 8) + `ASSERT_INIT(LcStateSize_A, lc_ctrl_state_pkg::LcStateWidth == LcStateSize * 8) + `ASSERT_INIT(LcTransitionCntSize_A, lc_ctrl_state_pkg::LcCountWidth == LcTransitionCntSize * 8) + + `ASSERT_KNOWN(OtpAstPwrSeqKnown_A, otp_ast_pwr_seq_o) + `ASSERT_KNOWN(CoreTlOutKnown_A, core_tl_o) + `ASSERT_KNOWN(PrimTlOutKnown_A, prim_tl_o) + `ASSERT_KNOWN(IntrOtpOperationDoneKnown_A, intr_otp_operation_done_o) + `ASSERT_KNOWN(IntrOtpErrorKnown_A, intr_otp_error_o) + `ASSERT_KNOWN(AlertTxKnown_A, alert_tx_o) + `ASSERT_KNOWN(PwrOtpInitRspKnown_A, pwr_otp_o) + `ASSERT_KNOWN(LcOtpProgramRspKnown_A, lc_otp_program_o) + `ASSERT_KNOWN(OtpLcDataKnown_A, otp_lc_data_o) + `ASSERT_KNOWN(OtpKeymgrKeyKnown_A, otp_keymgr_key_o) + `ASSERT_KNOWN(FlashOtpKeyRspKnown_A, flash_otp_key_o) + `ASSERT_KNOWN(OtpSramKeyKnown_A, sram_otp_key_o) + `ASSERT_KNOWN(OtpOtgnKeyKnown_A, otbn_otp_key_o) + `ASSERT_KNOWN(OtpBroadcastKnown_A, otp_broadcast_o) + + // Alert assertions for sparse FSMs. + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlDaiFsmCheck_A, + u_otp_ctrl_dai.u_state_regs, alert_tx_o[1]) + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlKdiFsmCheck_A, + u_otp_ctrl_kdi.u_state_regs, alert_tx_o[1]) + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlLciFsmCheck_A, + u_otp_ctrl_lci.u_state_regs, alert_tx_o[1]) + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlLfsrTimerFsmCheck_A, + u_otp_ctrl_lfsr_timer.u_state_regs, alert_tx_o[1]) + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(CtrlScrambleFsmCheck_A, + u_otp_ctrl_scrmbl.u_state_regs, alert_tx_o[1]) + + // Alert assertions for redundant counters. + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntIntegCheck_A, + u_otp_ctrl_lfsr_timer.u_prim_count_integ, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntCnstyCheck_A, + u_otp_ctrl_lfsr_timer.u_prim_count_cnsty, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntDaiCheck_A, + u_otp_ctrl_dai.u_prim_count, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntKdiSeedCheck_A, + u_otp_ctrl_kdi.u_prim_count_seed, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntKdiEntropyCheck_A, + u_otp_ctrl_kdi.u_prim_count_entropy, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntLciCheck_A, + u_otp_ctrl_lci.u_prim_count, alert_tx_o[1]) + `ASSERT_PRIM_COUNT_ERROR_TRIGGER_ALERT(CntScrmblCheck_A, + u_otp_ctrl_scrmbl.u_prim_count, alert_tx_o[1]) + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(TlLcGateFsm_A, + u_tlul_lc_gate.u_state_regs, alert_tx_o[2]) + + // Alert assertions for double LFSR. + `ASSERT_PRIM_DOUBLE_LFSR_ERROR_TRIGGER_ALERT(DoubleLfsrCheck_A, + u_otp_ctrl_lfsr_timer.u_prim_double_lfsr, alert_tx_o[1]) + + // Alert assertions for reg_we onehot check + `ASSERT_PRIM_REG_WE_ONEHOT_ERROR_TRIGGER_ALERT(RegWeOnehotCheck_A, u_reg_core, alert_tx_o[2]) + + // Assertions for countermeasures inside prim_otp + `ifndef PRIM_DEFAULT_IMPL + `define PRIM_DEFAULT_IMPL prim_pkg::ImplGeneric + `endif + if (`PRIM_DEFAULT_IMPL == prim_pkg::ImplGeneric) begin : gen_reg_we_assert_generic + `ASSERT_PRIM_FSM_ERROR_TRIGGER_ALERT(PrimFsmCheck_A, + u_otp.gen_generic.u_impl_generic.u_state_regs, alert_tx_o[3]) + `ASSERT_PRIM_REG_WE_ONEHOT_ERROR_TRIGGER_ALERT(PrimRegWeOnehotCheck_A, + u_otp.gen_generic.u_impl_generic.u_reg_top, alert_tx_o[3]) + end +endmodule : otp_ctrl diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl_core_reg_top.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_core_reg_top.sv similarity index 100% rename from hw/ip/otp_ctrl/rtl/otp_ctrl_core_reg_top.sv rename to hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_core_reg_top.sv diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_dai.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_dai.sv new file mode 100644 index 00000000000000..95240cfa9d165b --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_dai.sv @@ -0,0 +1,859 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Direct access interface for OTP controller. +// + +`include "prim_flop_macros.sv" + +module otp_ctrl_dai + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; +( + input clk_i, + input rst_ni, + // Init reqest from power manager + input init_req_i, + output logic init_done_o, + // Init request going to partitions + output logic part_init_req_o, + input [NumPart-1:0] part_init_done_i, + // Escalation input. This moves the FSM into a terminal state and locks down + // the DAI. + input lc_ctrl_pkg::lc_tx_t escalate_en_i, + // Output error state of DAI, to be consumed by OTP error/alert logic. + // Note that most errors are not recoverable and move the DAI FSM into + // a terminal error state. + output otp_err_e error_o, + // This error signal is pulsed high if the FSM has been glitched into an invalid state. + // Although it is somewhat redundant with the error code in error_o above, it is + // meant to cover cases where we already latched an error code while the FSM is + // glitched into an invalid state (since in that case, the error code will not be + // overridden with the FSM error code so that the original error code is still + // discoverable). + output logic fsm_err_o, + // Access/lock status from partitions + // SEC_CM: ACCESS.CTRL.MUBI + input part_access_t [NumPart-1:0] part_access_i, + // CSR interface + input [OtpByteAddrWidth-1:0] dai_addr_i, + input dai_cmd_e dai_cmd_i, + input logic dai_req_i, + input [NumDaiWords-1:0][31:0] dai_wdata_i, + output logic dai_idle_o, // wired to the status CSRs + output logic dai_prog_idle_o, // wired to lfsr timer and pwrmgr + output logic dai_cmd_done_o, // this is used to raise an IRQ + output logic [NumDaiWords-1:0][31:0] dai_rdata_o, + // OTP interface + output logic otp_req_o, + output prim_otp_pkg::cmd_e otp_cmd_o, + output logic [OtpSizeWidth-1:0] otp_size_o, + output logic [OtpIfWidth-1:0] otp_wdata_o, + output logic [OtpAddrWidth-1:0] otp_addr_o, + input otp_gnt_i, + input otp_rvalid_i, + input [ScrmblBlockWidth-1:0] otp_rdata_i, + input prim_otp_pkg::err_e otp_err_i, + // Scrambling mutex request + output logic scrmbl_mtx_req_o, + input scrmbl_mtx_gnt_i, + // Scrambling datapath interface + output otp_scrmbl_cmd_e scrmbl_cmd_o, + output digest_mode_e scrmbl_mode_o, + output logic [ConstSelWidth-1:0] scrmbl_sel_o, + output logic [ScrmblBlockWidth-1:0] scrmbl_data_o, + output logic scrmbl_valid_o, + input logic scrmbl_ready_i, + input logic scrmbl_valid_i, + input logic [ScrmblBlockWidth-1:0] scrmbl_data_i +); + + //////////////////////// + // Integration Checks // + //////////////////////// + + import prim_mubi_pkg::*; + import prim_util_pkg::vbits; + + localparam int CntWidth = OtpByteAddrWidth - $clog2(ScrmblBlockWidth/8); + + // Integration checks for parameters. + `ASSERT_INIT(CheckNativeOtpWidth0_A, (ScrmblBlockWidth % OtpWidth) == 0) + `ASSERT_INIT(CheckNativeOtpWidth1_A, (32 % OtpWidth) == 0) + + ///////////////////// + // DAI Control FSM // + ///////////////////// + + // SEC_CM: DAI.FSM.SPARSE + // Encoding generated with: + // $ ./util/design/sparse-fsm-encode.py -d 5 -m 20 -n 12 \ + // -s 3011551511 --language=sv + // + // Hamming distance histogram: + // + // 0: -- + // 1: -- + // 2: -- + // 3: -- + // 4: -- + // 5: |||||||||||||||| (31.05%) + // 6: |||||||||||||||||||| (36.84%) + // 7: |||||||| (15.26%) + // 8: |||| (8.95%) + // 9: || (5.26%) + // 10: (1.58%) + // 11: (1.05%) + // 12: -- + // + // Minimum Hamming distance: 5 + // Maximum Hamming distance: 11 + // Minimum Hamming weight: 2 + // Maximum Hamming weight: 9 + // + localparam int StateWidth = 12; + typedef enum logic [StateWidth-1:0] { + ResetSt = 12'b101111010100, + InitOtpSt = 12'b110000110010, + InitPartSt = 12'b000111111001, + IdleSt = 12'b111010000011, + ErrorSt = 12'b100010001110, + ReadSt = 12'b100101100110, + ReadWaitSt = 12'b001100000000, + DescrSt = 12'b011000101111, + DescrWaitSt = 12'b110101011111, + WriteSt = 12'b110111001000, + WriteWaitSt = 12'b111001111100, + ScrSt = 12'b000000010101, + ScrWaitSt = 12'b010110110100, + DigClrSt = 12'b001111001111, + DigReadSt = 12'b001001110011, + DigReadWaitSt = 12'b101110111010, + DigSt = 12'b011111100010, + DigPadSt = 12'b011010011000, + DigFinSt = 12'b110011100101, + DigWaitSt = 12'b100000101001 + } state_e; + + typedef enum logic [1:0] { + OtpData = 2'b00, + DaiData = 2'b01, + ScrmblData = 2'b10 + } data_sel_e; + + + typedef enum logic { + PartOffset = 1'b0, + DaiOffset = 1'b1 + } addr_sel_e; + + state_e state_d, state_q; + logic [CntWidth-1:0] cnt; + logic cnt_en, cnt_clr, cnt_err; + otp_err_e error_d, error_q; + logic data_en, data_clr; + data_sel_e data_sel; + addr_sel_e base_sel_d, base_sel_q; + logic [ScrmblBlockWidth-1:0] data_q; + logic [NumPartWidth-1:0] part_idx; + logic [NumPart-1:0][OtpAddrWidth-1:0] digest_addr_lut; + logic part_sel_valid; + + // Depending on the partition configuration, the wrapper is instructed to ignore integrity + // calculations and checks. To be on the safe side, the partition filters error responses at this + // point and does not report any integrity errors if integrity is disabled. + otp_err_e otp_err; + always_comb begin + otp_err = otp_err_e'(otp_err_i); + if (!PartInfo[part_idx].integrity && + otp_err_e'(otp_err_i) inside {MacroEccCorrError, MacroEccUncorrError}) begin + otp_err = NoError; + end + end + + // Output partition error state. + assign error_o = error_q; + // Working register is connected to data outputs. + assign otp_wdata_o = data_q; + assign scrmbl_data_o = data_q; + // Only expose this working register in IdleSt. + // The FSM below makes sure to clear this register + // after digest and write ops. + assign dai_rdata_o = (state_q == IdleSt) ? data_q : '0; + + always_comb begin : p_fsm + state_d = state_q; + + // Init signals + init_done_o = 1'b1; + part_init_req_o = 1'b0; + + // DAI signals + dai_idle_o = 1'b0; + dai_prog_idle_o = 1'b1; + dai_cmd_done_o = 1'b0; + + // OTP signals + otp_req_o = 1'b0; + otp_cmd_o = prim_otp_pkg::Init; + + // Scrambling mutex + scrmbl_mtx_req_o = 1'b0; + + // Scrambling datapath + scrmbl_cmd_o = LoadShadow; + scrmbl_sel_o = CnstyDigest; + scrmbl_mode_o = StandardMode; + scrmbl_valid_o = 1'b0; + + // Counter + cnt_en = 1'b0; + cnt_clr = 1'b0; + base_sel_d = base_sel_q; + + // Temporary data register + data_en = 1'b0; + data_clr = 1'b0; + data_sel = OtpData; + + // Error Register + error_d = error_q; + fsm_err_o = 1'b0; + + unique case (state_q) + /////////////////////////////////////////////////////////////////// + // We get here after reset and wait until the power manager + // requests OTP initialization. If initialization is requested, + // an init command is written to the OTP macro, and we move on + // to the InitOtpSt waiting state. + ResetSt: begin + init_done_o = 1'b0; + dai_prog_idle_o = 1'b0; + data_clr = 1'b1; + if (init_req_i) begin + otp_req_o = 1'b1; + if (otp_gnt_i) begin + state_d = InitOtpSt; + end + end + end + /////////////////////////////////////////////////////////////////// + // We wait here unitl the OTP macro has initialized without + // error. If an error occurred during this stage, we latch that + // error and move into a terminal error state. + InitOtpSt: begin + init_done_o = 1'b0; + dai_prog_idle_o = 1'b0; + if (otp_rvalid_i) begin + if ((!(otp_err inside {NoError, MacroEccCorrError}))) begin + state_d = ErrorSt; + error_d = otp_err; + end else begin + state_d = InitPartSt; + end + end + end + /////////////////////////////////////////////////////////////////// + // Since the OTP macro is now functional, we can send out an + // initialization request to all partitions and wait until they + // all have initialized. + InitPartSt: begin + init_done_o = 1'b0; + dai_prog_idle_o = 1'b0; + part_init_req_o = 1'b1; + if (part_init_done_i == {NumPart{1'b1}}) begin + state_d = IdleSt; + end + end + /////////////////////////////////////////////////////////////////// + // Idle state where we wait for incoming commands. + // Invalid commands trigger a CmdInvErr, which is recoverable. + IdleSt: begin + dai_idle_o = 1'b1; + if (dai_req_i) begin + // This clears previous (recoverable) and reset the counter. + error_d = NoError; + cnt_clr = 1'b1; + unique case (dai_cmd_i) + DaiRead: begin + state_d = ReadSt; + // Clear the temporary data register. + data_clr = 1'b1; + base_sel_d = DaiOffset; + end + DaiWrite: begin + data_sel = DaiData; + // Fetch data block. + data_en = 1'b1; + base_sel_d = DaiOffset; + // If this partition is scrambled, directly go to write scrambling first. + if (PartInfo[part_idx].secret) begin + state_d = ScrSt; + end else begin + state_d = WriteSt; + end + end + DaiDigest: begin + state_d = DigClrSt; + scrmbl_mtx_req_o = 1'b1; + base_sel_d = PartOffset; + end + default: ; // Ignore invalid commands + endcase // dai_cmd_i + end // dai_req_i + end + /////////////////////////////////////////////////////////////////// + // Each time we request a block of data from OTP, we re-check + // whether read access has been locked for this partition. If + // that is the case, we immediately bail out. Otherwise, we + // request a block of data from OTP. + ReadSt: begin + if (part_sel_valid && (mubi8_test_false_strict(part_access_i[part_idx].read_lock) || + // HW digests always remain readable. + PartInfo[part_idx].hw_digest && otp_addr_o == + digest_addr_lut[part_idx])) begin + otp_req_o = 1'b1; + // Depending on the partition configuration, + // the wrapper is instructed to ignore integrity errors. + if (PartInfo[part_idx].integrity) begin + otp_cmd_o = prim_otp_pkg::Read; + end else begin + otp_cmd_o = prim_otp_pkg::ReadRaw; + end + if (otp_gnt_i) begin + state_d = ReadWaitSt; + end + end else begin + state_d = IdleSt; + error_d = AccessError; // Signal this error, but do not go into terminal error state. + dai_cmd_done_o = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response and write to readout register. Check + // whether descrambling is required or not. In case an OTP + // transaction fails, latch the OTP error code, and jump to + // terminal error state. + ReadWaitSt: begin + // Continuously check read access and bail out if this is not consistent. + if (part_sel_valid && (mubi8_test_false_strict(part_access_i[part_idx].read_lock) || + // HW digests always remain readable. + PartInfo[part_idx].hw_digest && otp_addr_o == + digest_addr_lut[part_idx])) begin + if (otp_rvalid_i) begin + // Check OTP return code. + if (otp_err inside {NoError, MacroEccCorrError}) begin + data_en = 1'b1; + // We do not need to descramble the digest values. + if (PartInfo[part_idx].secret && otp_addr_o != digest_addr_lut[part_idx]) begin + state_d = DescrSt; + end else begin + state_d = IdleSt; + dai_cmd_done_o = 1'b1; + end + // At this point the only error that we could have gotten are correctable ECC errors. + if (otp_err != NoError) begin + error_d = MacroEccCorrError; + end + end else begin + state_d = ErrorSt; + error_d = otp_err; + end + end + // At this point, this check MUST succeed - otherwise this means that + // there was a tampering attempt. Hence we go into a terminal error state + // when this check fails. + end else begin + state_d = ErrorSt; + error_d = FsmStateError; + end + end + /////////////////////////////////////////////////////////////////// + // Descrambling state. This first acquires the scrambling + // datapath mutex. Note that once the mutex is acquired, we have + // exclusive access to the scrambling datapath until we release + // the mutex by deasserting scrmbl_mtx_req_o. + // SEC_CM: SECRET.MEM.SCRAMBLE + DescrSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = Decrypt; + scrmbl_sel_o = PartInfo[part_idx].key_sel; + if (scrmbl_mtx_gnt_i && scrmbl_ready_i) begin + state_d = DescrWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for the descrambled data to return. Note that we release + // the mutex lock upon leaving this state. + // SEC_CM: SECRET.MEM.SCRAMBLE + DescrWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_sel_o = PartInfo[part_idx].key_sel; + data_sel = ScrmblData; + if (scrmbl_valid_i) begin + state_d = IdleSt; + data_en = 1'b1; + dai_cmd_done_o = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // First, check whether write accesses are allowed to this + // partition, and error out otherwise. Note that for buffered + // partitions, we do not allow DAI writes to the digest offset. + // Unbuffered partitions have SW managed digests, hence that + // check is not needed in that case. The LC partition is + // permanently write locked and can hence not be written via the DAI. + WriteSt: begin + dai_prog_idle_o = 1'b0; + if (part_sel_valid && mubi8_test_false_strict(part_access_i[part_idx].write_lock) && + // If this is a HW digest write to a buffered partition. + ((PartInfo[part_idx].variant == Buffered && PartInfo[part_idx].hw_digest && + base_sel_q == PartOffset && otp_addr_o == digest_addr_lut[part_idx]) || + // If this is a non HW digest write to a buffered partition. + (PartInfo[part_idx].variant == Buffered && PartInfo[part_idx].hw_digest && + base_sel_q == DaiOffset && otp_addr_o < digest_addr_lut[part_idx]) || + // If this is a write to an unbuffered partition + (PartInfo[part_idx].variant != Buffered && base_sel_q == DaiOffset))) begin + otp_req_o = 1'b1; + // Depending on the partition configuration, + // the wrapper is instructed to ignore integrity errors. + if (PartInfo[part_idx].integrity) begin + otp_cmd_o = prim_otp_pkg::Write; + end else begin + otp_cmd_o = prim_otp_pkg::WriteRaw; + end + if (otp_gnt_i) begin + state_d = WriteWaitSt; + end + end else begin + // Clear working register state. + data_clr = 1'b1; + state_d = IdleSt; + error_d = AccessError; // Signal this error, but do not go into terminal error state. + dai_cmd_done_o = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response, and then go back to idle. In case an + // OTP transaction fails, latch the OTP error code, and jump to + // terminal error state. + WriteWaitSt: begin + dai_prog_idle_o = 1'b0; + // Continuously check write access and bail out if this is not consistent. + if (part_sel_valid && mubi8_test_false_strict(part_access_i[part_idx].write_lock) && + // If this is a HW digest write to a buffered partition. + ((PartInfo[part_idx].variant == Buffered && PartInfo[part_idx].hw_digest && + base_sel_q == PartOffset && otp_addr_o == digest_addr_lut[part_idx]) || + // If this is a non HW digest write to a buffered partition. + (PartInfo[part_idx].variant == Buffered && PartInfo[part_idx].hw_digest && + base_sel_q == DaiOffset && otp_addr_o < digest_addr_lut[part_idx]) || + // If this is a write to an unbuffered partition + (PartInfo[part_idx].variant != Buffered && base_sel_q == DaiOffset))) begin + + if (otp_rvalid_i) begin + // Check OTP return code. Note that non-blank errors are recoverable. + if ((!(otp_err inside {NoError, MacroWriteBlankError}))) begin + state_d = ErrorSt; + error_d = otp_err; + end else begin + // Clear working register state. + data_clr = 1'b1; + state_d = IdleSt; + dai_cmd_done_o = 1'b1; + // Signal non-blank state, but do not go to terminal error state. + if (otp_err == MacroWriteBlankError) begin + error_d = otp_err; + end + end + end + // At this point, this check MUST succeed - otherwise this means that + // there was a tampering attempt. Hence we go into a terminal error state + // when this check fails. + end else begin + state_d = ErrorSt; + error_d = FsmStateError; + end + end + /////////////////////////////////////////////////////////////////// + // Scrambling state. This first acquires the scrambling + // datapath mutex. Note that once the mutex is acquired, we have + // exclusive access to the scrambling datapath until we release + // the mutex by deasserting scrmbl_mtx_req_o. + // SEC_CM: SECRET.MEM.SCRAMBLE + ScrSt: begin + scrmbl_mtx_req_o = 1'b1; + // Check write access and bail out if this is not consistent. + if (part_sel_valid && mubi8_test_false_strict(part_access_i[part_idx].write_lock) && + // If this is a non HW digest write to a buffered partition. + (PartInfo[part_idx].variant == Buffered && PartInfo[part_idx].secret && + PartInfo[part_idx].hw_digest && base_sel_q == DaiOffset && + otp_addr_o < digest_addr_lut[part_idx])) begin + + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = Encrypt; + scrmbl_sel_o = PartInfo[part_idx].key_sel; + if (scrmbl_mtx_gnt_i && scrmbl_ready_i) begin + state_d = ScrWaitSt; + end + end else begin + state_d = IdleSt; + error_d = AccessError; // Signal this error, but do not go into terminal error state. + dai_cmd_done_o = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for the scrambled data to return. Note that we release + // the mutex lock upon leaving this state. + // SEC_CM: SECRET.MEM.SCRAMBLE + ScrWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + // Continously check write access and bail out if this is not consistent. + if (part_sel_valid && mubi8_test_false_strict(part_access_i[part_idx].write_lock) && + // If this is a non HW digest write to a buffered partition. + (PartInfo[part_idx].variant == Buffered && PartInfo[part_idx].secret && + PartInfo[part_idx].hw_digest && base_sel_q == DaiOffset && + otp_addr_o < digest_addr_lut[part_idx])) begin + data_sel = ScrmblData; + if (scrmbl_valid_i) begin + state_d = WriteSt; + data_en = 1'b1; + end + // At this point, this check MUST succeed - otherwise this means that + // there was a tampering attempt. Hence we go into a terminal error state + // when this check fails. + end else begin + state_d = ErrorSt; + error_d = FsmStateError; + end + end + /////////////////////////////////////////////////////////////////// + // First, acquire the mutex for the digest and clear the digest state. + // SEC_CM: PART.MEM.DIGEST + DigClrSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + // Need to reset the digest state and set digest mode to "standard". + scrmbl_cmd_o = DigestInit; + if (scrmbl_mtx_gnt_i && scrmbl_ready_i) begin + state_d = DigReadSt; + end + end + /////////////////////////////////////////////////////////////////// + // This requests a 64bit block to be pushed into the digest datapath. + // We also check here whether the partition has been write locked. + // SEC_CM: PART.MEM.DIGEST + DigReadSt: begin + scrmbl_mtx_req_o = 1'b1; + if (part_sel_valid && + mubi8_test_false_strict(part_access_i[part_idx].read_lock) && + mubi8_test_false_strict(part_access_i[part_idx].write_lock)) begin + otp_req_o = 1'b1; + // Depending on the partition configuration, + // the wrapper is instructed to ignore integrity errors. + if (PartInfo[part_idx].integrity) begin + otp_cmd_o = prim_otp_pkg::Read; + end else begin + otp_cmd_o = prim_otp_pkg::ReadRaw; + end + if (otp_gnt_i) begin + state_d = DigReadWaitSt; + end + end else begin + state_d = IdleSt; + error_d = AccessError; // Signal this error, but do not go into terminal error state. + dai_cmd_done_o = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response and write to readout register. Check + // whether descrambling is required or not. In case an OTP + // transaction fails, latch the OTP error code, and jump to + // terminal error state. + // SEC_CM: PART.MEM.DIGEST + DigReadWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + if (otp_rvalid_i) begin + cnt_en = 1'b1; + // Check OTP return code. + if ((!(otp_err inside {NoError, MacroEccCorrError}))) begin + state_d = ErrorSt; + error_d = otp_err; + end else begin + data_en = 1'b1; + state_d = DigSt; + // Signal soft ECC errors, but do not go into terminal error state. + if (otp_err == MacroEccCorrError) begin + error_d = otp_err; + end + end + end + end + /////////////////////////////////////////////////////////////////// + // Push the word read into the scrambling datapath. The last + // block is repeated in case the number blocks in this partition + // is odd. + // SEC_CM: PART.MEM.DIGEST + DigSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + // No need to digest the digest value itself + if (otp_addr_o == digest_addr_lut[part_idx]) begin + // Trigger digest round in case this is the second block in a row. + if (!cnt[0]) begin + scrmbl_cmd_o = Digest; + if (scrmbl_ready_i) begin + state_d = DigFinSt; + end + // Otherwise, just load low word and go to padding state. + end else if (scrmbl_ready_i) begin + state_d = DigPadSt; + end + end else begin + // Trigger digest round in case this is the second block in a row. + if (!cnt[0]) begin + scrmbl_cmd_o = Digest; + end + // Go back and fetch more data blocks. + if (scrmbl_ready_i) begin + state_d = DigReadSt; + end + end + end + /////////////////////////////////////////////////////////////////// + // Padding state, just repeat the last block and go to digest + // finalization. + // SEC_CM: PART.MEM.DIGEST + DigPadSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = Digest; + if (scrmbl_ready_i) begin + state_d = DigFinSt; + end + end + /////////////////////////////////////////////////////////////////// + // Trigger digest finalization and go wait for the result. + // SEC_CM: PART.MEM.DIGEST + DigFinSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = DigestFinalize; + if (scrmbl_ready_i) begin + state_d = DigWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for the digest to return, and write the result to OTP. + // Note that the write address will be correct in this state, + // since the counter has been stepped to the correct address as + // part of the readout sequence, and the correct size for this + // access has been loaded before. + // SEC_CM: PART.MEM.DIGEST + DigWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + data_sel = ScrmblData; + if (scrmbl_valid_i) begin + state_d = WriteSt; + data_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Terminal Error State. This locks access to the DAI. Make sure + // an FsmStateError error code is assigned here, in case no error code has + // been assigned yet. + ErrorSt: begin + if (error_q == NoError) begin + error_d = FsmStateError; + end + end + /////////////////////////////////////////////////////////////////// + // We should never get here. If we do (e.g. via a malicious + // glitch), error out immediately. + default: begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + endcase // state_q + + // Unconditionally jump into the terminal error state in case of escalation. + // SEC_CM: DAI.FSM.LOCAL_ESC, DAI.FSM.GLOBAL_ESC + if (lc_ctrl_pkg::lc_tx_test_true_loose(escalate_en_i) || cnt_err) begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + if (state_q != ErrorSt) begin + error_d = FsmStateError; + end + end + end + + //////////////////////////// + // Partition Select Logic // + //////////////////////////// + + // This checks which partition the address belongs to by comparing + // the incoming address to the partition address ranges. The onehot + // bitvector generated by the parallel comparisons is fed into a + // binary tree that determines the partition index with O(log(N)) delay. + + logic [NumPart-1:0] part_sel_oh; + for (genvar k = 0; k < NumPart; k++) begin : gen_part_sel + localparam int unsigned PartEndInt = 32'(PartInfo[k].offset) + 32'(PartInfo[k].size); + localparam int unsigned DigestOffsetInt = PartEndInt - ScrmblBlockWidth / 8; + localparam int unsigned DigestAddrLutInt = DigestOffsetInt >> OtpAddrShift; + + // PartEnd has an extra bit to cope with the case where offset + size overflows. However, we + // arrange the address map to make sure that PartEndInt is at most 1 << OtpByteAddrWidth. Check + // that here. + `ASSERT_INIT(PartEndMax_A, PartEndInt <= (1 << OtpByteAddrWidth)) + + // The shift right by OtpAddrShift drops exactly the bottom bits that are needed to convert + // between OtpAddrWidth and OtpByteAddrWidth, so we know that we can slice safely here. + localparam bit [OtpAddrWidth-1:0] DigestAddrLut = DigestAddrLutInt[OtpAddrWidth-1:0]; + + if (PartInfo[k].offset == 0) begin : gen_zero_offset + assign part_sel_oh[k] = ({1'b0, dai_addr_i} < PartEndInt[OtpByteAddrWidth:0]); + + end else begin : gen_nonzero_offset + assign part_sel_oh[k] = (dai_addr_i >= PartInfo[k].offset) & + ({1'b0, dai_addr_i} < PartEndInt[OtpByteAddrWidth:0]); + end + assign digest_addr_lut[k] = DigestAddrLut; + end + + `ASSERT(ScrmblBlockWidthGe8_A, ScrmblBlockWidth >= 8) + `ASSERT(PartSelMustBeOnehot_A, $onehot0(part_sel_oh)) + + prim_arbiter_fixed #( + .N(NumPart), + .EnDataPort(0) + ) u_part_sel_idx ( + .clk_i, + .rst_ni, + .req_i ( part_sel_oh ), + .data_i ( '{default: '0} ), + .gnt_o ( ), // unused + .idx_o ( part_idx ), + .valid_o ( part_sel_valid ), // used for detecting OOB addresses + .data_o ( ), // unused + .ready_i ( 1'b0 ) + ); + + ///////////////////////////////////// + // Address Calculations for Digest // + ///////////////////////////////////// + + // Depending on whether this is a 32bit or 64bit partition, we cut off the lower address bits. + // Access sizes are either 64bit or 32bit, depending on what region the access goes to. + logic [OtpByteAddrWidth-1:0] addr_base; + always_comb begin : p_size_sel + otp_size_o = OtpSizeWidth'(unsigned'(32 / OtpWidth - 1)); + addr_base = {dai_addr_i[OtpByteAddrWidth-1:2], 2'h0}; + + // 64bit transaction for scrambled partitions. + if (PartInfo[part_idx].secret) begin + otp_size_o = OtpSizeWidth'(unsigned'(ScrmblBlockWidth / OtpWidth - 1)); + addr_base = {dai_addr_i[OtpByteAddrWidth-1:3], 3'h0}; + // 64bit transaction if computing a digest. + end else if (PartInfo[part_idx].hw_digest && (base_sel_q == PartOffset)) begin + otp_size_o = OtpSizeWidth'(unsigned'(ScrmblBlockWidth / OtpWidth - 1)); + addr_base = PartInfo[part_idx].offset; + // 64bit transaction if the DAI address points to the partition's digest offset. + end else if ((PartInfo[part_idx].hw_digest || PartInfo[part_idx].sw_digest) && + (base_sel_q == DaiOffset) && + ({dai_addr_i[OtpByteAddrWidth-1:3], 2'b0} == digest_addr_lut[part_idx])) begin + otp_size_o = OtpSizeWidth'(unsigned'(ScrmblBlockWidth / OtpWidth - 1)); + addr_base = {dai_addr_i[OtpByteAddrWidth-1:3], 3'h0}; + end + end + + // Address counter - this is only used for computing a digest, hence the increment is + // fixed to 8 byte. + // SEC_CM: DAI.CTR.REDUN + prim_count #( + .Width(CntWidth) + ) u_prim_count ( + .clk_i, + .rst_ni, + .clr_i(cnt_clr), + .set_i(1'b0), + .set_cnt_i('0), + .incr_en_i(cnt_en), + .decr_en_i(1'b0), + .step_i(CntWidth'(1)), + .commit_i(1'b1), + .cnt_o(cnt), + .cnt_after_commit_o(), + .err_o(cnt_err) + ); + + // Note that OTP works on halfword (16bit) addresses, hence need to + // shift the addresses appropriately. + logic [OtpByteAddrWidth-1:0] addr_calc; + assign addr_calc = {cnt, {$clog2(ScrmblBlockWidth/8){1'b0}}} + addr_base; + assign otp_addr_o = OtpAddrWidth'(addr_calc >> OtpAddrShift); + + /////////////// + // Registers // + /////////////// + + `PRIM_FLOP_SPARSE_FSM(u_state_regs, state_d, state_q, state_e, ResetSt) + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + error_q <= NoError; + data_q <= '0; + base_sel_q <= DaiOffset; + end else begin + error_q <= error_d; + base_sel_q <= base_sel_d; + + // Working register + if (data_clr) begin + data_q <= '0; + end else if (data_en) begin + if (data_sel == ScrmblData) begin + data_q <= scrmbl_data_i; + end else if (data_sel == DaiData) begin + data_q <= dai_wdata_i; + end else begin + data_q <= otp_rdata_i; + end + end + end + end + + //////////////// + // Assertions // + //////////////// + + // Known assertions + `ASSERT_KNOWN(InitDoneKnown_A, init_done_o) + `ASSERT_KNOWN(PartInitReqKnown_A, part_init_req_o) + `ASSERT_KNOWN(ErrorKnown_A, error_o) + `ASSERT_KNOWN(DaiIdleKnown_A, dai_idle_o) + `ASSERT_KNOWN(DaiRdataKnown_A, dai_rdata_o) + `ASSERT_KNOWN(OtpReqKnown_A, otp_req_o) + `ASSERT_KNOWN(OtpCmdKnown_A, otp_cmd_o) + `ASSERT_KNOWN(OtpSizeKnown_A, otp_size_o) + `ASSERT_KNOWN(OtpWdataKnown_A, otp_wdata_o) + `ASSERT_KNOWN(OtpAddrKnown_A, otp_addr_o) + `ASSERT_KNOWN(ScrmblMtxReqKnown_A, scrmbl_mtx_req_o) + `ASSERT_KNOWN(ScrmblCmdKnown_A, scrmbl_cmd_o) + `ASSERT_KNOWN(ScrmblModeKnown_A, scrmbl_mode_o) + `ASSERT_KNOWN(ScrmblSelKnown_A, scrmbl_sel_o) + `ASSERT_KNOWN(ScrmblDataKnown_A, scrmbl_data_o) + `ASSERT_KNOWN(ScrmblValidKnown_A, scrmbl_valid_o) + + // OTP error response + `ASSERT(OtpErrorState_A, + state_q inside {InitOtpSt, ReadWaitSt, WriteWaitSt, DigReadWaitSt} && otp_rvalid_i && + !(otp_err inside {NoError, MacroEccCorrError, MacroWriteBlankError}) + |=> + state_q == ErrorSt && error_o == $past(otp_err)) + +endmodule : otp_ctrl_dai diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_ecc_reg.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_ecc_reg.sv new file mode 100644 index 00000000000000..2199a7cc6024d0 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_ecc_reg.sv @@ -0,0 +1,105 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Register file for buffered OTP partitions. ECC is used to detect up +// to two simultaneous errors within each 64bit word. + +`include "prim_assert.sv" + +module otp_ctrl_ecc_reg #( + parameter int Width = 64, // bit + parameter int Depth = 128, + localparam int Aw = prim_util_pkg::vbits(Depth) // derived parameter +) ( + input logic clk_i, + input logic rst_ni, + + input logic wren_i, + input logic [Aw-1:0] addr_i, + input logic [Width-1:0] wdata_i, + output logic [Width-1:0] rdata_o, + + // Concurrent output of the register state. + output logic [Depth-1:0][Width-1:0] data_o, + // Concurrent ECC check error is flagged via this signal. + output logic ecc_err_o +); + + // Integration checks for parameters. + `ASSERT_INIT(WidthMustBe64bit_A, Width == 64) + + localparam int EccWidth = 8; + + logic [Depth-1:0][Width-1:0] data_d, data_q; + logic [Depth-1:0][EccWidth-1:0] ecc_d, ecc_q; + logic [Width+EccWidth-1:0] ecc_enc; + + // Only one encoder is needed. + prim_secded_inv_72_64_enc u_prim_secded_inv_72_64_enc ( + .data_i(wdata_i), + .data_o(ecc_enc) + ); + + if (Depth == 1) begin : gen_one_word_only + always_comb begin : p_write + data_o = data_q; + data_d = data_q; + ecc_d = ecc_q; + + rdata_o = '0; + if (32'(addr_i) < Depth) begin + rdata_o = data_q[0]; + if (wren_i) begin + {ecc_d[0], data_d[0]} = ecc_enc; + end + end + end + end else begin : gen_multiple_words + always_comb begin : p_write + data_o = data_q; + data_d = data_q; + ecc_d = ecc_q; + + rdata_o = '0; + if (32'(addr_i) < Depth) begin + rdata_o = data_q[addr_i]; + if (wren_i) begin + {ecc_d[addr_i], data_d[addr_i]} = ecc_enc; + end + end + end + end + + // Concurrent ECC checks. + logic [Depth-1:0][1:0] err; + for (genvar k = 0; k < Depth; k++) begin : gen_ecc_dec + prim_secded_inv_72_64_dec u_prim_secded_inv_72_64_dec ( + .data_i({ecc_q[k], data_q[k]}), + // We only rely on the error detection mechanism, + // and not on error correction. + .data_o(), + .syndrome_o(), + .err_o(err[k]) + ); + end + + assign ecc_err_o = |err; + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + ecc_q <= {Depth{prim_secded_pkg::SecdedInv7264ZeroEcc}}; + data_q <= '0; + end else begin + ecc_q <= ecc_d; + data_q <= data_d; + end + end + + `ASSERT_KNOWN(EccKnown_A, ecc_q) + `ASSERT_KNOWN(DataKnown_A, data_q) + `ASSERT_KNOWN(RDataOutKnown_A, rdata_o) + `ASSERT_KNOWN(DataOutKnown_A, data_o) + `ASSERT_KNOWN(EccErrKnown_A, ecc_err_o) + +endmodule : otp_ctrl_ecc_reg diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_kdi.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_kdi.sv new file mode 100644 index 00000000000000..3b5d6b454f950f --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_kdi.sv @@ -0,0 +1,604 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Scrambling key derivation module for OTP. +// + +`include "prim_flop_macros.sv" + +module otp_ctrl_kdi + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; +#( + parameter scrmbl_key_init_t RndCnstScrmblKeyInit = RndCnstScrmblKeyInitDefault +) ( + input clk_i, + input rst_ni, + // Pulse to enable this module after OTP partitions have + // been initialized. + input kdi_en_i, + // Escalation input. This moves the FSM into a terminal state. + input lc_ctrl_pkg::lc_tx_t escalate_en_i, + // FSM is in error state + output logic fsm_err_o, + // Key seed inputs from OTP + input logic scrmbl_key_seed_valid_i, + input logic [FlashKeySeedWidth-1:0] flash_data_key_seed_i, + input logic [FlashKeySeedWidth-1:0] flash_addr_key_seed_i, + input logic [SramKeySeedWidth-1:0] sram_data_key_seed_i, + // EDN interface for requesting entropy + output logic edn_req_o, + input edn_ack_i, + input [EdnDataWidth-1:0] edn_data_i, + // Scrambling key requests + input flash_otp_key_req_t flash_otp_key_i, + output flash_otp_key_rsp_t flash_otp_key_o, + input sram_otp_key_req_t [NumSramKeyReqSlots-1:0] sram_otp_key_i, + output sram_otp_key_rsp_t [NumSramKeyReqSlots-1:0] sram_otp_key_o, + input otbn_otp_key_req_t otbn_otp_key_i, + output otbn_otp_key_rsp_t otbn_otp_key_o, + // Scrambling mutex request + output logic scrmbl_mtx_req_o, + input scrmbl_mtx_gnt_i, + // Scrambling datapath interface + output otp_scrmbl_cmd_e scrmbl_cmd_o, + output digest_mode_e scrmbl_mode_o, + output logic [ConstSelWidth-1:0] scrmbl_sel_o, + output logic [ScrmblBlockWidth-1:0] scrmbl_data_o, + output logic scrmbl_valid_o, + input logic scrmbl_ready_i, + input logic scrmbl_valid_i, + input logic [ScrmblBlockWidth-1:0] scrmbl_data_i +); + + import prim_util_pkg::vbits; + + //////////////////////// + // Integration Checks // + //////////////////////// + + // 2xFlash, OTBN + SRAM slots + localparam int NumReq = 3 + NumSramKeyReqSlots; + // Make sure key sizes in the system are multiples of 64bit and not larger than 256bit. + `ASSERT_INIT(KeyNonceSize0_A, (FlashKeySeedWidth <= 256) && ((FlashKeySeedWidth % 64) == 0)) + `ASSERT_INIT(KeyNonceSize1_A, (SramKeySeedWidth <= 256) && ((SramKeySeedWidth % 64) == 0)) + `ASSERT_INIT(KeyNonceSize2_A, (FlashKeyWidth <= 256) && ((FlashKeyWidth % 64) == 0)) + `ASSERT_INIT(KeyNonceSize3_A, (SramKeyWidth <= 256) && ((SramKeyWidth % 64) == 0)) + `ASSERT_INIT(KeyNonceSize4_A, (SramNonceWidth <= 256) && ((SramNonceWidth % 64) == 0)) + `ASSERT_INIT(KeyNonceSize5_A, (OtbnKeyWidth <= 256) && ((OtbnKeyWidth % 64) == 0)) + `ASSERT_INIT(KeyNonceSize6_A, (OtbnNonceWidth <= 256) && ((OtbnNonceWidth % 64) == 0)) + + // Make sure EDN interface has compatible width. + `ASSERT_INIT(EntropyWidthDividesDigestBlockWidth_A, (ScrmblKeyWidth % EdnDataWidth) == 0) + + // Currently the assumption is that the SRAM nonce is the widest. + `ASSERT_INIT(NonceWidth_A, NumNonceChunks * ScrmblBlockWidth == SramNonceWidth) + + /////////////////////////////////// + // Input Mapping and Arbitration // + /////////////////////////////////// + + // The key derivation and token hashing functions are aligned such that 2 x 128bit key + // seeds / token blocks are processed in two subsequent steps using the digest primitive. + // This effectively compresses these blocks down into 2 x 64bit blocks, thereby creating + // one 128bit key or token output. + // + // The same FSM is shared among the different flavors of key derivation and token + // hashing functions, and the following configuration options are available: + // + // 1) ingest an additional 128bit entropy block after ingesting a 128bit key seed. + // 2) keep digest state after producing the first 64bit block instead of reverting to the IV. + // 3) netlist constant index. + // 4) fetch additional entropy for the nonce output. + // 5) whether or not the key seed is valid. if not, it will be defaulted to '0. + // 6) 256bit key seed / token input. + // + // The configuration options are set further below, depending on the request type. + + typedef struct packed { + logic ingest_entropy; // 1) + logic chained_digest; // 2) + digest_sel_e digest_sel; // 3) + logic fetch_nonce; // 4) + logic [1:0] nonce_size; // 4) + logic seed_valid; // 5) + logic [3:0][ScrmblBlockWidth-1:0] seed; // 6) + } req_bundle_t; + + logic [NumReq-1:0] req, gnt; + req_bundle_t req_bundles [NumReq]; + + assign req[0] = flash_otp_key_i.data_req; + assign req[1] = flash_otp_key_i.addr_req; + assign req[2] = otbn_otp_key_i.req; + + assign flash_otp_key_o.data_ack = gnt[0]; + assign flash_otp_key_o.addr_ack = gnt[1]; + assign otbn_otp_key_o.ack = gnt[2]; + + // anchored seeds + logic [FlashKeySeedWidth-1:0] flash_data_key_seed; + logic [FlashKeySeedWidth-1:0] flash_addr_key_seed; + logic [SramKeySeedWidth-1:0] sram_data_key_seed; + + prim_sec_anchor_buf #( + .Width(FlashKeySeedWidth) + ) u_flash_data_key_anchor ( + .in_i(flash_data_key_seed_i), + .out_o(flash_data_key_seed) + ); + + prim_sec_anchor_buf #( + .Width(FlashKeySeedWidth) + ) u_flash_addr_key_anchor ( + .in_i(flash_addr_key_seed_i), + .out_o(flash_addr_key_seed) + ); + + prim_sec_anchor_buf #( + .Width(SramKeySeedWidth) + ) u_sram_data_key_anchor ( + .in_i(sram_data_key_seed_i), + .out_o(sram_data_key_seed) + ); + + // Flash data key + assign req_bundles[0] = '{ingest_entropy: 1'b0, // no random entropy added + chained_digest: 1'b0, // revert to netlist IV between blocks + digest_sel: FlashDataKey, + fetch_nonce: 1'b1, + nonce_size: 2'(FlashKeyWidth/EdnDataWidth-1), + seed_valid: scrmbl_key_seed_valid_i, + seed: flash_data_key_seed}; // 2x128bit + // Flash addr key + assign req_bundles[1] = '{ingest_entropy: 1'b0, // no random entropy added + chained_digest: 1'b0, // revert to netlist IV between blocks + digest_sel: FlashAddrKey, + fetch_nonce: 1'b1, + nonce_size: '0, + seed_valid: scrmbl_key_seed_valid_i, + seed: flash_addr_key_seed}; // 2x128bit + // OTBN key + assign req_bundles[2] = '{ingest_entropy: 1'b1, // ingest random data + chained_digest: 1'b0, // revert to netlist IV between blocks + digest_sel: SramDataKey, + fetch_nonce: 1'b1, // fetch nonce + nonce_size: 2'(OtbnNonceWidth/EdnDataWidth-1), + seed_valid: scrmbl_key_seed_valid_i, + seed: {sram_data_key_seed, // reuse same seed + sram_data_key_seed}}; + + // SRAM keys + for (genvar k = 3; k < NumReq; k++) begin : gen_req_assign + assign req[k] = sram_otp_key_i[k-3].req; + assign sram_otp_key_o[k-3].ack = gnt[k]; + assign req_bundles[k] = '{ingest_entropy: 1'b1, // ingest random data + chained_digest: 1'b0, // revert to netlist IV between blocks + digest_sel: SramDataKey, + fetch_nonce: 1'b1, // fetch nonce + nonce_size: 2'(SramNonceWidth/EdnDataWidth-1), + seed_valid: scrmbl_key_seed_valid_i, + seed: {sram_data_key_seed, // reuse same seed + sram_data_key_seed}}; + end + + // This arbitrates among incoming key derivation requests on a + // round robin basis to prevent deadlock. + logic req_valid, req_ready; + req_bundle_t req_bundle; + + prim_arbiter_tree #( + .N(NumReq), + .DW($bits(req_bundle_t))) + u_req_arb ( + .clk_i, + .rst_ni, + .req_chk_i ( 1'b1 ), + .req_i ( req ), + .data_i ( req_bundles ), + .gnt_o ( gnt ), + .idx_o ( ), + .valid_o ( req_valid ), + .data_o ( req_bundle ), + .ready_i ( req_ready ) + ); + + ////////////////////////////// + // Temporary Regs and Muxes // + ////////////////////////////// + + localparam int CntWidth = 2; + logic seed_cnt_clr, seed_cnt_en, entropy_cnt_clr, entropy_cnt_en, seed_cnt_err, entropy_cnt_err; + logic [CntWidth-1:0] seed_cnt, entropy_cnt; + + // SEC_CM: KDI_SEED.CTR.REDUN + prim_count #( + .Width(CntWidth) + ) u_prim_count_seed ( + .clk_i, + .rst_ni, + .clr_i(seed_cnt_clr), + .set_i(1'b0), + .set_cnt_i('0), + .incr_en_i(seed_cnt_en), + .decr_en_i(1'b0), + .step_i(CntWidth'(1)), + .commit_i(1'b1), + .cnt_o(seed_cnt), + .cnt_after_commit_o(), + .err_o(seed_cnt_err) + ); + + // SEC_CM: KDI_ENTROPY.CTR.REDUN + prim_count #( + .Width(CntWidth) + ) u_prim_count_entropy ( + .clk_i, + .rst_ni, + .clr_i(entropy_cnt_clr), + .set_i(1'b0), + .set_cnt_i('0), + .incr_en_i(entropy_cnt_en), + .decr_en_i(1'b0), + .step_i(CntWidth'(1)), + .commit_i(1'b1), + .cnt_o(entropy_cnt), + .cnt_after_commit_o(), + .err_o(entropy_cnt_err) + ); + + logic seed_valid_reg_en; + logic key_reg_en, nonce_reg_en; + logic seed_valid_d, seed_valid_q; + logic [ScrmblKeyWidth/ScrmblBlockWidth-1:0][ScrmblBlockWidth-1:0] key_out_d, key_out_q; + logic [NumNonceChunks-1:0][ScrmblBlockWidth-1:0] nonce_out_d, nonce_out_q; + + always_comb begin : p_outregs + key_out_d = key_out_q; + nonce_out_d = nonce_out_q; + seed_valid_d = seed_valid_q; + if (key_reg_en) begin + key_out_d[seed_cnt[1]] = scrmbl_data_i; + end + if (nonce_reg_en) begin + nonce_out_d[entropy_cnt[$clog2(NumNonceChunks)-1:0]] = edn_data_i; + end + if (seed_valid_reg_en) begin + seed_valid_d = req_bundle.seed_valid; + end + end + + // Connect keys/nonce outputs to output regs. + prim_sec_anchor_flop #( + .Width(ScrmblKeyWidth), + .ResetValue(RndCnstScrmblKeyInit.key) + ) u_key_out_anchor ( + .clk_i, + .rst_ni, + .d_i(key_out_d), + .q_o(key_out_q) + ); + + assign otbn_otp_key_o.key = key_out_q; + assign otbn_otp_key_o.nonce = nonce_out_q[OtbnNonceSel-1:0]; + assign otbn_otp_key_o.seed_valid = seed_valid_q; + + assign flash_otp_key_o.key = key_out_q; + assign flash_otp_key_o.rand_key = nonce_out_q[FlashNonceSel-1:0]; + assign flash_otp_key_o.seed_valid = seed_valid_q; + + for (genvar k = 0; k < NumSramKeyReqSlots; k++) begin : gen_out_assign + assign sram_otp_key_o[k].key = key_out_q; + assign sram_otp_key_o[k].nonce = nonce_out_q[SramNonceSel-1:0]; + assign sram_otp_key_o[k].seed_valid = seed_valid_q; + end + + typedef enum logic { + SeedData, + EntropyData + } data_sel_e; + + // Select correct 64bit block. + data_sel_e data_sel; + assign scrmbl_data_o = (data_sel == EntropyData) ? nonce_out_q[entropy_cnt[0]] : + // Gate seed value to '0 if invalid. + (req_bundle.seed_valid) ? req_bundle.seed[seed_cnt] : '0; + + ///////////////// + // Control FSM // + ///////////////// + + // SEC_CM: KDI.FSM.SPARSE + // Encoding generated with: + // $ ./util/design/sparse-fsm-encode.py -d 5 -m 11 -n 10 \ + // -s 2544133835 --language=sv + // + // Hamming distance histogram: + // + // 0: -- + // 1: -- + // 2: -- + // 3: -- + // 4: -- + // 5: |||||||||||||||||||| (54.55%) + // 6: |||||||||||||||| (45.45%) + // 7: -- + // 8: -- + // 9: -- + // 10: -- + // + // Minimum Hamming distance: 5 + // Maximum Hamming distance: 6 + // Minimum Hamming weight: 3 + // Maximum Hamming weight: 9 + // + localparam int StateWidth = 10; + typedef enum logic [StateWidth-1:0] { + ResetSt = 10'b0101100001, + IdleSt = 10'b0001011011, + DigClrSt = 10'b1101010110, + DigLoadSt = 10'b0010110111, + FetchEntropySt = 10'b1000001101, + DigEntropySt = 10'b0100111100, + DigFinSt = 10'b1000100010, + DigWaitSt = 10'b1110010001, + FetchNonceSt = 10'b0011000100, + FinishSt = 10'b1011111000, + ErrorSt = 10'b1111101111 + } state_e; + + state_e state_d, state_q; + logic edn_req_d, edn_req_q; + assign edn_req_o = edn_req_q; + + always_comb begin : p_fsm + state_d = state_q; + + // FSM Error output + fsm_err_o = 1'b0; + + // Counters + seed_cnt_en = 1'b0; + seed_cnt_clr = 1'b0; + entropy_cnt_en = 1'b0; + entropy_cnt_clr = 1'b0; + + // EDN 128bit block fetch request. + // This keeps the request alive until it has + // been acked to adhere to the req/ack protocol + // even in cases where the FSM jumps into + // an error state while waiting for a request. + edn_req_d = edn_req_q & ~edn_ack_i; + + // Data selection and temp registers + data_sel = SeedData; + key_reg_en = 1'b0; + nonce_reg_en = 1'b0; + seed_valid_reg_en = 1'b0; + + // Scrambling datapath + scrmbl_mtx_req_o = 1'b0; + scrmbl_sel_o = req_bundle.digest_sel; + scrmbl_cmd_o = LoadShadow; + scrmbl_mode_o = StandardMode; + + scrmbl_valid_o = 1'b0; + + // Request acknowledgement + req_ready = 1'b0; + + unique case (state_q) + /////////////////////////////////////////////////////////////////// + // State right after reset. Wait here until KDI gets enabled. + ResetSt: begin + if (kdi_en_i) begin + state_d = IdleSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for a request, then go and acquire the mutex. + IdleSt: begin + if (req_valid) begin + state_d = DigClrSt; + seed_cnt_clr = 1'b1; + entropy_cnt_clr = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // First, acquire the mutex for the digest and clear the digest state. + DigClrSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + // Need to reset the digest state and set digest mode to "standard". + scrmbl_cmd_o = DigestInit; + if (scrmbl_mtx_gnt_i && scrmbl_ready_i) begin + state_d = DigLoadSt; + end + end + /////////////////////////////////////////////////////////////////// + // Load two 64bit blocks of the seed, and trigger digest calculation. + DigLoadSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + // Trigger digest round in case this is the second block in a row. + if (seed_cnt[0]) begin + scrmbl_cmd_o = Digest; + if (scrmbl_ready_i) begin + // Go and ingest a block of entropy if required. + if (req_bundle.ingest_entropy) begin + state_d = FetchEntropySt; + // Otherwise go to digest finalization state. + end else begin + state_d = DigFinSt; + end + end + // Just load first 64bit block and stay here. + end else if (scrmbl_ready_i) begin + seed_cnt_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Fetch random data to ingest for key derivation. + FetchEntropySt: begin + scrmbl_mtx_req_o = 1'b1; + edn_req_d = 1'b1; + if (edn_ack_i) begin + nonce_reg_en = 1'b1; + // Finished, go and acknowledge this request. + if (entropy_cnt == 2'h1) begin + state_d = DigEntropySt; + entropy_cnt_clr = 1'b1; + // Keep on requesting entropy. + end else begin + entropy_cnt_en = 1'b1; + end + end + end + /////////////////////////////////////////////////////////////////// + // Load two 64bit blocks of entropy data. + DigEntropySt: begin + scrmbl_mtx_req_o = 1'b1; + data_sel = EntropyData; + scrmbl_valid_o = 1'b1; + // Trigger digest round in case this is the second block in a row, + // and go to digest finalization. + if (entropy_cnt[0]) begin + scrmbl_cmd_o = Digest; + if (scrmbl_ready_i) begin + state_d = DigFinSt; + entropy_cnt_clr = 1'b1; + end + // Just load first 64bit block and stay here. + end else if (scrmbl_ready_i) begin + entropy_cnt_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Trigger digest finalization and go wait for the result. + DigFinSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = DigestFinalize; + if (scrmbl_ready_i) begin + state_d = DigWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for the digest to return, and write the result to the key + // output register. Go back and process the second part of the + // input seed if needed. + DigWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + if (scrmbl_valid_i) begin + key_reg_en = 1'b1; + // Not finished yet, need to go back and produce second 64bit block. + if (seed_cnt == 2'h1) begin + seed_cnt_en = 1'b1; + // In this case the previous digest state is kept, + // which leads to a chained digest. + if (req_bundle.chained_digest) begin + state_d = DigLoadSt; + // In this case we revert the digest state to the netlist IV. + end else begin + state_d = DigClrSt; + end + // This was the second 64bit output block. + end else begin + seed_cnt_clr = 1'b1; + // Make sure we output the status of the key seed in OTP. + seed_valid_reg_en = 1'b1; + // Check whether we need to fetch additional nonce data. + if (req_bundle.fetch_nonce) begin + state_d = FetchNonceSt; + end else begin + // Finished, go and acknowledge this request. + state_d = FinishSt; + end + end + end + end + /////////////////////////////////////////////////////////////////// + // Fetch additional nonce data. Note that the mutex is released in + // this state. + FetchNonceSt: begin + edn_req_d = 1'b1; + if (edn_ack_i) begin + nonce_reg_en = 1'b1; + // Finished, go and acknowledge this request. + if (entropy_cnt == req_bundle.nonce_size) begin + state_d = FinishSt; + entropy_cnt_clr = 1'b1; + // Keep on requesting entropy. + end else begin + entropy_cnt_en = 1'b1; + end + end + end + /////////////////////////////////////////////////////////////////// + // Acknowledge request and go back to IdleSt. + FinishSt: begin + state_d = IdleSt; + req_ready = 1'b1; + end + /////////////////////////////////////////////////////////////////// + // Terminal error state. This raises an alert. + ErrorSt: begin + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + // This should never happen, hence we directly jump into the + // error state, where an alert will be triggered. + default: begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + endcase // state_q + + // Unconditionally jump into the terminal error state in case of escalation. + // SEC_CM: KDI.FSM.LOCAL_ESC, KDI.FSM.GLOBAL_ESC + if (lc_ctrl_pkg::lc_tx_test_true_loose(escalate_en_i) || + seed_cnt_err || entropy_cnt_err) begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + end + + /////////////// + // Registers // + /////////////// + + `PRIM_FLOP_SPARSE_FSM(u_state_regs, state_d, state_q, state_e, ResetSt) + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + nonce_out_q <= RndCnstScrmblKeyInit.nonce; + seed_valid_q <= 1'b0; + edn_req_q <= 1'b0; + end else begin + nonce_out_q <= nonce_out_d; + seed_valid_q <= seed_valid_d; + edn_req_q <= edn_req_d; + end + end + + //////////////// + // Assertions // + //////////////// + + `ASSERT_KNOWN(FsmErrKnown_A, fsm_err_o) + `ASSERT_KNOWN(EdnReqKnown_A, edn_req_o) + `ASSERT_KNOWN(FlashOtpKeyRspKnown_A, flash_otp_key_o) + `ASSERT_KNOWN(SramOtpKeyRspKnown_A, sram_otp_key_o) + `ASSERT_KNOWN(OtbnOtpKeyRspKnown_A, otbn_otp_key_o) + `ASSERT_KNOWN(ScrmblMtxReqKnown_A, scrmbl_mtx_req_o) + `ASSERT_KNOWN(ScrmblCmdKnown_A, scrmbl_cmd_o) + `ASSERT_KNOWN(ScrmblModeKnown_A, scrmbl_mode_o) + `ASSERT_KNOWN(ScrmblSelKnown_A, scrmbl_sel_o) + `ASSERT_KNOWN(ScrmblDataKnown_A, scrmbl_data_o) + `ASSERT_KNOWN(ScrmblValidKnown_A, scrmbl_valid_o) + +endmodule : otp_ctrl_kdi diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_lci.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_lci.sv new file mode 100644 index 00000000000000..b2f98d41be9d06 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_lci.sv @@ -0,0 +1,299 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Life cycle interface for performing life cycle transitions in OTP. +// + +`include "prim_flop_macros.sv" + +module otp_ctrl_lci + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; +#( + // Lifecycle partition information + parameter part_info_t Info = PartInfoDefault +) ( + input clk_i, + input rst_ni, + input lci_en_i, + // Escalation input. This moves the FSM into a terminal state and locks down + // the partition. + input lc_ctrl_pkg::lc_tx_t escalate_en_i, + // Life cycle transition request. In order to perform a state transition, + // the LC controller signals the new count and state. The OTP wrapper then + // only programs bits that have not been programmed before. + // Note that a transition request will fail if the request attempts to + // clear already programmed bits within OTP. + input lc_req_i, + input logic [Info.size*8-1:0] lc_data_i, + output logic lc_ack_o, + output logic lc_err_o, + // Output error state of partition, to be consumed by OTP error/alert logic. + // Note that most errors are not recoverable and move the partition FSM into + // a terminal error state. + output otp_err_e error_o, + // This error signal is pulsed high if the FSM has been glitched into an invalid state. + // Although it is somewhat redundant with the error code in error_o above, it is + // meant to cover cases where we already latched an error code while the FSM is + // glitched into an invalid state (since in that case, the error code will not be + // overridden with the FSM error code so that the original error code is still + // discoverable). + output logic fsm_err_o, + output logic lci_prog_idle_o, + // OTP interface + output logic otp_req_o, + output prim_otp_pkg::cmd_e otp_cmd_o, + output logic [OtpSizeWidth-1:0] otp_size_o, + output logic [OtpIfWidth-1:0] otp_wdata_o, + output logic [OtpAddrWidth-1:0] otp_addr_o, + input otp_gnt_i, + input otp_rvalid_i, + input [ScrmblBlockWidth-1:0] otp_rdata_i, + input prim_otp_pkg::err_e otp_err_i +); + + //////////////////////// + // Integration Checks // + //////////////////////// + + import prim_util_pkg::vbits; + + localparam int NumLcOtpWords = int'(Info.size) >> OtpAddrShift; + localparam int CntWidth = vbits(NumLcOtpWords); + + localparam int unsigned LastLcOtpWordInt = NumLcOtpWords - 1; + localparam bit [CntWidth-1:0] LastLcOtpWord = LastLcOtpWordInt[CntWidth-1:0]; + + // This is required, since each native OTP word can only be programmed once. + `ASSERT_INIT(LcValueMustBeWiderThanNativeOtpWidth_A, lc_ctrl_state_pkg::LcValueWidth >= OtpWidth) + + //////////////////// + // Controller FSM // + //////////////////// + + // SEC_CM: LCI.FSM.SPARSE + // Encoding generated with: + // $ ./util/design/sparse-fsm-encode.py -d 5 -m 5 -n 9 \ + // -s 558234734 --language=sv + // + // Hamming distance histogram: + // + // 0: -- + // 1: -- + // 2: -- + // 3: -- + // 4: -- + // 5: |||||||||||||||||||| (60.00%) + // 6: ||||||||||||| (40.00%) + // 7: -- + // 8: -- + // 9: -- + // + // Minimum Hamming distance: 5 + // Maximum Hamming distance: 6 + // Minimum Hamming weight: 1 + // Maximum Hamming weight: 7 + // + localparam int StateWidth = 9; + typedef enum logic [StateWidth-1:0] { + ResetSt = 9'b000101011, + IdleSt = 9'b110011110, + WriteSt = 9'b101010001, + WriteWaitSt = 9'b010000000, + ErrorSt = 9'b011111101 + } state_e; + + state_e state_d, state_q; + logic cnt_clr, cnt_en, cnt_err; + logic [CntWidth-1:0] cnt; + otp_err_e error_d, error_q; + + // Output LCI errors + assign error_o = error_q; + + always_comb begin : p_fsm + state_d = state_q; + + // Counter + cnt_en = 1'b0; + cnt_clr = 1'b0; + + // Idle status + lci_prog_idle_o = 1'b1; + + // OTP signals + otp_req_o = 1'b0; + otp_cmd_o = prim_otp_pkg::Read; + + // Response to LC controller + lc_err_o = 1'b0; + lc_ack_o = 1'b0; + + // Error Register + error_d = error_q; + fsm_err_o = 1'b0; + + unique case (state_q) + /////////////////////////////////////////////////////////////////// + // State right after reset. Wait here until LCI gets enabled. + ResetSt: begin + lci_prog_idle_o = 1'b0; + if (lci_en_i) begin + state_d = IdleSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for a request from the life cycle controller + IdleSt: begin + if (lc_req_i) begin + state_d = WriteSt; + cnt_clr = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Loop through the lifecycle sate and burn in all words. + // If the write data contains a 0 bit in a position where a bit has already been + // programmed to 1 before, the OTP errors out. + WriteSt: begin + otp_req_o = 1'b1; + otp_cmd_o = prim_otp_pkg::Write; + lci_prog_idle_o = 1'b0; + if (otp_gnt_i) begin + state_d = WriteWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response, and check whether there are more words to burn in. + // In case an OTP transaction fails, latch the OTP error code, and jump to + // terminal error state. + WriteWaitSt: begin + lci_prog_idle_o = 1'b0; + if (otp_rvalid_i) begin + // Check OTP return code. + // Note that if errors occur, we aggregate the error code + // but still attempt to program all remaining words. + // This is done to ensure that a life cycle state with + // ECC correctable errors in some words can still be scrapped. + if (otp_err_e'(otp_err_i) != NoError) begin + error_d = otp_err_e'(otp_err_i); + end + + // Check whether we programmed all OTP words. + // If yes, we are done and can go back to idle. + if (cnt == LastLcOtpWord) begin + state_d = IdleSt; + lc_ack_o = 1'b1; + // If in any of the words a programming error has occurred, + // we signal that accordingly and go to the error state. + if (error_d != NoError) begin + lc_err_o = 1'b1; + state_d = ErrorSt; + end + // Otherwise we increase the OTP word counter. + end else begin + state_d = WriteSt; + cnt_en = 1'b1; + end + end + end + /////////////////////////////////////////////////////////////////// + // Terminal Error State. This locks access to the partition. + // Make sure the partition signals an error state if no error + // code has been latched so far, and lock the buffer regs down. + ErrorSt: begin + if (error_q == NoError) begin + error_d = FsmStateError; + end + end + /////////////////////////////////////////////////////////////////// + // We should never get here. If we do (e.g. via a malicious + // glitch), error out immediately. + default: begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + endcase // state_q + + // Unconditionally jump into the terminal error state in case of escalation. + // SEC_CM: LCI.FSM.LOCAL_ESC, LCI.FSM.GLOBAL_ESC + if (lc_ctrl_pkg::lc_tx_test_true_loose(escalate_en_i) || cnt_err) begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + if (error_q == NoError) begin + error_d = FsmStateError; + end + end + + end + + ////////////////////////////// + // Counter and address calc // + ////////////////////////////// + + // Native OTP word counter + // SEC_CM: LCI.CTR.REDUN + prim_count #( + .Width(CntWidth) + ) u_prim_count ( + .clk_i, + .rst_ni, + .clr_i(cnt_clr), + .set_i(1'b0), + .set_cnt_i('0), + .incr_en_i(cnt_en), + .decr_en_i(1'b0), + .step_i(CntWidth'(1)), + .commit_i(1'b1), + .cnt_o(cnt), + .cnt_after_commit_o(), + .err_o(cnt_err) + ); + + // The output address is "offset + count", but we have to convert Info.offset from a byte address + // to a halfword (16-bit) address by discarding the bottom OtpAddrShift bits. We also make the + // zero-extension of cnt explicit (to avoid width mismatch warnings). + assign otp_addr_o = Info.offset[OtpByteAddrWidth-1:OtpAddrShift] + OtpAddrWidth'(cnt); + + // Always transfer 16bit blocks. + assign otp_size_o = '0; + + logic [NumLcOtpWords-1:0][OtpWidth-1:0] data; + assign data = lc_data_i; + assign otp_wdata_o = (otp_req_o) ? OtpIfWidth'(data[cnt]) : '0; + + logic unused_rdata; + assign unused_rdata = ^otp_rdata_i; + + /////////////// + // Registers // + /////////////// + + `PRIM_FLOP_SPARSE_FSM(u_state_regs, state_d, state_q, state_e, ResetSt) + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + error_q <= NoError; + end else begin + error_q <= error_d; + end + end + + //////////////// + // Assertions // + //////////////// + + `ASSERT_KNOWN(LcAckKnown_A, lc_ack_o) + `ASSERT_KNOWN(LcErrKnown_A, lc_err_o) + `ASSERT_KNOWN(ErrorKnown_A, error_o) + `ASSERT_KNOWN(LciIdleKnown_A, lci_prog_idle_o) + `ASSERT_KNOWN(OtpReqKnown_A, otp_req_o) + `ASSERT_KNOWN(OtpCmdKnown_A, otp_cmd_o) + `ASSERT_KNOWN(OtpSizeKnown_A, otp_size_o) + `ASSERT_KNOWN(OtpWdataKnown_A, otp_wdata_o) + `ASSERT_KNOWN(OtpAddrKnown_A, otp_addr_o) + +endmodule : otp_ctrl_lci diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv new file mode 100644 index 00000000000000..775c8586cfcaf4 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_lfsr_timer.sv @@ -0,0 +1,398 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// This module implements the LFSR timer for triggering periodic consistency and integrity checks in +// OTP. In particular, this module contains two 40bit counters (one for the consistency and one +// for the integrity checks) and a 40bit LFSR to draw pseudo random wait counts. +// +// The integ_period_msk_i and cnsty_period_msk_i mask signals are used to mask off the LFSR outputs +// and hence determine the maximum wait count that can be drawn. If these values are set to +// zero, the corresponding timer is disabled. +// +// Once a particular check timer has expired, the module will send out a check request to all +// partitions and wait for an acknowledgment. If a particular partition encounters an integrity or +// consistency mismatch, this will be directly reported via the error and alert logic. +// +// In order to guard against wedged partition controllers or arbitration lock ups due to tampering +// attempts, this check timer module also supports a 32bit timeout that can optionally be +// programmed. If a particular check times out, chk_timeout_o will be asserted, which will raise +// an alert via the error logic. +// +// The EntropyWidth LSBs of the LFSR are reseeded with fresh entropy from EDN once +// LfsrUsageThreshold values have been drawn from the LFSR. +// +// It is also possible to trigger one-off checks via integ_chk_trig_i and cnsty_chk_trig_i. +// This can be useful if SW chooses to leave the periodic checks disabled. +// + +`include "prim_flop_macros.sv" + +module otp_ctrl_lfsr_timer + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_top_specific_pkg::*; +#( + // Compile time random constants, to be overriden by topgen. + parameter lfsr_seed_t RndCnstLfsrSeed = RndCnstLfsrSeedDefault, + parameter lfsr_perm_t RndCnstLfsrPerm = RndCnstLfsrPermDefault +) ( + input clk_i, + input rst_ni, + output logic edn_req_o, // request to EDN + input edn_ack_i, // ack from EDN + input [EdnDataWidth-1:0] edn_data_i, // from EDN + input timer_en_i, // enable timer + input otp_prog_busy_i, // indicates whether prog ops are in progress + input integ_chk_trig_i, // one-off trigger for integrity check + input cnsty_chk_trig_i, // one-off trigger for consistency check + output logic chk_pending_o, // indicates whether there are pending checks + input [31:0] timeout_i, // check timeout + input [31:0] integ_period_msk_i, // maximum integrity check mask + input [31:0] cnsty_period_msk_i, // maximum consistency check mask + output logic [NumPart-1:0] integ_chk_req_o, // request to all partitions + output logic [NumPart-1:0] cnsty_chk_req_o, // request to all partitions + input [NumPart-1:0] integ_chk_ack_i, // response from partitions + input [NumPart-1:0] cnsty_chk_ack_i, // response from partitions + input lc_ctrl_pkg::lc_tx_t escalate_en_i, // escalation input, moves FSM into ErrorSt + output logic chk_timeout_o, // a check has timed out + output logic fsm_err_o // the FSM has reached an invalid state +); + + //////////////////// + // Reseed counter // + //////////////////// + + // Count how many times the LFSR has been used to generate a value. + // Once we've reached the limit, we request new entropy from EDN to reseed + // the LFSR. Note that this is not a blocking operation for the timer below. + // I.e., the timer is allowed to continue its operation, and may draw more + // values, even if the EDN reseed request is still in progress. + logic reseed_en, lfsr_en; + logic [$clog2(LfsrUsageThreshold+1)-1:0] reseed_cnt_d, reseed_cnt_q; + assign reseed_cnt_d = (reseed_en) ? '0 : + (edn_req_o) ? reseed_cnt_q : + (lfsr_en) ? reseed_cnt_q + 1'b1 : + reseed_cnt_q; + + assign edn_req_o = (reseed_cnt_q >= LfsrUsageThreshold); + assign reseed_en = edn_req_o & edn_ack_i; + + /////////////////////////// + // Tandem LFSR Instances // + /////////////////////////// + + logic lfsr_err; + logic [LfsrWidth-1:0] entropy; + logic [LfsrWidth-1:0] lfsr_state; + assign entropy = (reseed_en) ? edn_data_i[LfsrWidth-1:0] : '0; + + // We employ two redundant LFSRs to guard against FI attacks. + // If any of the two is glitched and the two LFSR states do not agree, + // the FSM below is moved into a terminal error state. + // SEC_CM: TIMER.LFSR.REDUN + prim_double_lfsr #( + .LfsrDw ( LfsrWidth ), + .EntropyDw ( LfsrWidth ), + .StateOutDw ( LfsrWidth ), + .DefaultSeed ( RndCnstLfsrSeed ), + .StatePermEn ( 1'b1 ), + .StatePerm ( RndCnstLfsrPerm ), + .ExtSeedSVA ( 1'b0 ) + ) u_prim_double_lfsr ( + .clk_i, + .rst_ni, + .seed_en_i ( 1'b0 ), + .seed_i ( '0 ), + .lfsr_en_i ( reseed_en || lfsr_en ), + .entropy_i ( entropy ), + .state_o ( lfsr_state ), + .err_o ( lfsr_err ) + ); + + // Not all entropy bits are used. + logic unused_seed; + assign unused_seed = ^edn_data_i; + + `ASSERT_INIT(EdnIsWideEnough_A, EdnDataWidth >= LfsrWidth) + + ////////////////////////////// + // Tandem Counter Instances // + ////////////////////////////// + + // We employ redundant counters to guard against FI attacks. + // If any of them is glitched and the redundant counter states do not agree, + // the FSM below is moved into a terminal error state. + logic [LfsrWidth-1:0] integ_cnt, cnsty_cnt, integ_cnt_set_val, cnsty_cnt_set_val; + logic [LfsrWidth-1:0] integ_mask, cnsty_mask; + logic integ_set_period, integ_set_timeout, integ_cnt_zero; + logic cnsty_set_period, cnsty_set_timeout, cnsty_cnt_zero; + logic integ_cnt_set, cnsty_cnt_set, integ_cnt_err, cnsty_cnt_err; + logic timeout_zero, integ_msk_zero, cnsty_msk_zero, cnsty_cnt_pause; + + assign timeout_zero = (timeout_i == '0); + assign integ_msk_zero = (integ_period_msk_i == '0); + assign cnsty_msk_zero = (cnsty_period_msk_i == '0); + assign integ_cnt_zero = (integ_cnt == '0); + assign cnsty_cnt_zero = (cnsty_cnt == '0); + + assign integ_cnt_set = integ_set_period || integ_set_timeout; + assign cnsty_cnt_set = cnsty_set_period || cnsty_set_timeout; + + assign integ_mask = {integ_period_msk_i, {LfsrWidth-32{1'b1}}}; + assign cnsty_mask = {cnsty_period_msk_i, {LfsrWidth-32{1'b1}}}; + assign integ_cnt_set_val = (integ_set_period) ? (lfsr_state & integ_mask) : LfsrWidth'(timeout_i); + assign cnsty_cnt_set_val = (cnsty_set_period) ? (lfsr_state & cnsty_mask) : LfsrWidth'(timeout_i); + + // SEC_CM: TIMER_INTEG.CTR.REDUN + prim_count #( + .Width(LfsrWidth) + ) u_prim_count_integ ( + .clk_i, + .rst_ni, + .clr_i(1'b0), + .set_i(integ_cnt_set), + .set_cnt_i(integ_cnt_set_val), + .incr_en_i(1'b0), + .decr_en_i(!integ_cnt_zero), + .step_i(LfsrWidth'(1)), + .commit_i(1'b1), + .cnt_o(integ_cnt), + .cnt_after_commit_o(), + .err_o(integ_cnt_err) + ); + + // SEC_CM: TIMER_CNSTY.CTR.REDUN + prim_count #( + .Width(LfsrWidth) + ) u_prim_count_cnsty ( + .clk_i, + .rst_ni, + .clr_i(1'b0), + .set_i(cnsty_cnt_set), + .set_cnt_i(cnsty_cnt_set_val), + .incr_en_i(1'b0), + .decr_en_i(!cnsty_cnt_zero && !cnsty_cnt_pause), + .step_i(LfsrWidth'(1)), + .commit_i(1'b1), + .cnt_o(cnsty_cnt), + .cnt_after_commit_o(), + .err_o(cnsty_cnt_err) + ); + + ///////////////////// + // Request signals // + ///////////////////// + + logic set_all_integ_reqs, set_all_cnsty_reqs; + logic [NumPart-1:0] integ_chk_req_d, integ_chk_req_q; + logic [NumPart-1:0] cnsty_chk_req_d, cnsty_chk_req_q; + assign integ_chk_req_o = integ_chk_req_q; + assign cnsty_chk_req_o = cnsty_chk_req_q; + assign integ_chk_req_d = (set_all_integ_reqs) ? {NumPart{1'b1}} : + integ_chk_req_q & ~integ_chk_ack_i; + assign cnsty_chk_req_d = (set_all_cnsty_reqs) ? {NumPart{1'b1}} : + cnsty_chk_req_q & ~cnsty_chk_ack_i; + + + // external triggers + logic clr_integ_chk_trig, clr_cnsty_chk_trig; + logic integ_chk_trig_d, integ_chk_trig_q; + logic cnsty_chk_trig_d, cnsty_chk_trig_q; + assign integ_chk_trig_d = (integ_chk_trig_q & ~clr_integ_chk_trig) | integ_chk_trig_i; + assign cnsty_chk_trig_d = (cnsty_chk_trig_q & ~clr_cnsty_chk_trig) | cnsty_chk_trig_i; + + //////////////////////////// + // Ping and Timeout Logic // + //////////////////////////// + + // SEC_CM: TIMER.FSM.SPARSE + // Encoding generated with: + // $ ./util/design/sparse-fsm-encode.py -d 5 -m 5 -n 9 \ + // -s 628816752 --language=sv + // + // Hamming distance histogram: + // + // 0: -- + // 1: -- + // 2: -- + // 3: -- + // 4: -- + // 5: |||||||||||||||||||| (60.00%) + // 6: ||||||||||||| (40.00%) + // 7: -- + // 8: -- + // 9: -- + // + // Minimum Hamming distance: 5 + // Maximum Hamming distance: 6 + // Minimum Hamming weight: 4 + // Maximum Hamming weight: 6 + // + localparam int StateWidth = 9; + typedef enum logic [StateWidth-1:0] { + ResetSt = 9'b100100101, + IdleSt = 9'b001101110, + IntegWaitSt = 9'b010110011, + CnstyWaitSt = 9'b111010110, + ErrorSt = 9'b001011001 + } state_e; + + state_e state_d, state_q; + logic chk_timeout_d, chk_timeout_q; + + assign chk_timeout_o = chk_timeout_q; + + always_comb begin : p_fsm + state_d = state_q; + + // LFSR and counter signals + lfsr_en = 1'b0; + integ_set_period = 1'b0; + cnsty_set_period = 1'b0; + integ_set_timeout = 1'b0; + cnsty_set_timeout = 1'b0; + cnsty_cnt_pause = 1'b0; + + // Requests going to partitions. + set_all_integ_reqs = '0; + set_all_cnsty_reqs = '0; + + // Status signals going to CSRs and error logic. + chk_timeout_d = chk_timeout_q; + chk_pending_o = cnsty_chk_trig_q || integ_chk_trig_q; + fsm_err_o = 1'b0; + + // Clear signals for external triggers + clr_integ_chk_trig = 1'b0; + clr_cnsty_chk_trig = 1'b0; + + unique case (state_q) + /////////////////////////////////////////////////////////////////// + // Wait until enabled. We never return to this state + // once enabled! + ResetSt: begin + if (timer_en_i) begin + state_d = IdleSt; + lfsr_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Wait here until one of the two timers expires (if enabled) or if + // a check is triggered externally. + IdleSt: begin + if ((!integ_msk_zero && integ_cnt_zero) || integ_chk_trig_q) begin + state_d = IntegWaitSt; + integ_set_timeout = 1'b1; + set_all_integ_reqs = 1'b1; + clr_integ_chk_trig = integ_chk_trig_q; + end else if ((!cnsty_msk_zero && cnsty_cnt_zero) || cnsty_chk_trig_q) begin + state_d = CnstyWaitSt; + cnsty_set_timeout = 1'b1; + set_all_cnsty_reqs = 1'b1; + clr_cnsty_chk_trig = cnsty_chk_trig_q; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for all the partitions to respond and go back to idle. + // If the timeout is enabled, bail out into terminal error state + // if the timeout counter expires (this will raise an alert). + IntegWaitSt: begin + chk_pending_o = 1'b1; + if (!timeout_zero && integ_cnt_zero) begin + state_d = ErrorSt; + chk_timeout_d = 1'b1; + end else if (integ_chk_req_q == '0) begin + state_d = IdleSt; + // This draws the next wait period. + integ_set_period = 1'b1; + lfsr_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for all the partitions to respond and go back to idle. + // If the timeout is enabled, bail out into terminal error state + // if the timeout counter expires (this will raise an alert). + CnstyWaitSt: begin + chk_pending_o = 1'b1; + // Note that consistency checks go back and read from OTP. Hence, + // life cycle transitions and DAI programming operations + // may interfere with these checks and cause them to take longer + // than typically expected. Therefore, the timeout counter is stopped + // during programming operations. + cnsty_cnt_pause = otp_prog_busy_i; + if (!timeout_zero && cnsty_cnt_zero) begin + state_d = ErrorSt; + chk_timeout_d = 1'b1; + end else if (cnsty_chk_req_q == '0) begin + state_d = IdleSt; + // This draws the next wait period. + cnsty_set_period = 1'b1; + lfsr_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Terminal error state. This raises an alert. + ErrorSt: begin + // Continuously clear pending checks. + clr_integ_chk_trig = 1'b1; + clr_cnsty_chk_trig = 1'b1; + if (!chk_timeout_q) begin + fsm_err_o = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // This should never happen, hence we directly jump into the + // error state, where an alert will be triggered. + default: begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + endcase // state_q + + // Unconditionally jump into the terminal error state in case of escalation, + // or if the two LFSR or counter states do not agree. + // SEC_CM: TIMER.FSM.LOCAL_ESC, TIMER.FSM.GLOBAL_ESC + if (lfsr_err || integ_cnt_err || cnsty_cnt_err || + lc_ctrl_pkg::lc_tx_test_true_loose(escalate_en_i)) begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + end + + /////////////// + // Registers // + /////////////// + + `PRIM_FLOP_SPARSE_FSM(u_state_regs, state_d, state_q, state_e, ResetSt) + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + integ_chk_req_q <= '0; + cnsty_chk_req_q <= '0; + chk_timeout_q <= 1'b0; + reseed_cnt_q <= '0; + integ_chk_trig_q <= 1'b0; + cnsty_chk_trig_q <= 1'b0; + end else begin + integ_chk_req_q <= integ_chk_req_d; + cnsty_chk_req_q <= cnsty_chk_req_d; + chk_timeout_q <= chk_timeout_d; + reseed_cnt_q <= reseed_cnt_d; + integ_chk_trig_q <= integ_chk_trig_d; + cnsty_chk_trig_q <= cnsty_chk_trig_d; + end + end + + //////////////// + // Assertions // + //////////////// + + `ASSERT_KNOWN(EdnReqKnown_A, edn_req_o) + `ASSERT_KNOWN(ChkPendingKnown_A, chk_pending_o) + `ASSERT_KNOWN(IntegChkReqKnown_A, integ_chk_req_o) + `ASSERT_KNOWN(CnstyChkReqKnown_A, cnsty_chk_req_o) + `ASSERT_KNOWN(ChkTimeoutKnown_A, chk_timeout_o) + +endmodule : otp_ctrl_lfsr_timer diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_buf.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_buf.sv new file mode 100644 index 00000000000000..c4da5a89302840 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_buf.sv @@ -0,0 +1,822 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Buffered partition for OTP controller. +// + +`include "prim_flop_macros.sv" + +module otp_ctrl_part_buf + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; +#( + // Partition information. + parameter part_info_t Info = PartInfoDefault, + parameter logic [Info.size*8-1:0] DataDefault = '0 +) ( + input clk_i, + input rst_ni, + // Pulse to start partition initialisation (required once per power cycle). + input init_req_i, + output logic init_done_o, + // Integrity check requests + input integ_chk_req_i, + output logic integ_chk_ack_o, + // Consistency check requests + input cnsty_chk_req_i, + output logic cnsty_chk_ack_o, + // Escalation input. This moves the FSM into a terminal state and locks down + // the partition. + input lc_ctrl_pkg::lc_tx_t escalate_en_i, + // Check bypass enable. This bypasses integrity and consistency checks and + // acknowledges all incoming check requests (only used by life cycle). + input lc_ctrl_pkg::lc_tx_t check_byp_en_i, + // Output error state of partition, to be consumed by OTP error/alert logic. + // Note that most errors are not recoverable and move the partition FSM into + // a terminal error state. + output otp_err_e error_o, + // This error signal is pulsed high if the FSM has been glitched into an invalid state. + // Although it is somewhat redundant with the error code in error_o above, it is + // meant to cover cases where we already latched an error code while the FSM is + // glitched into an invalid state (since in that case, the error code will not be + // overridden with the FSM error code so that the original error code is still + // discoverable). + output logic fsm_err_o, + // Access/lock status + // SEC_CM: ACCESS.CTRL.MUBI + input part_access_t access_i, // runtime lock from CSRs + output part_access_t access_o, + // Buffered 64bit digest output. + output logic [ScrmblBlockWidth-1:0] digest_o, + output logic [Info.size*8-1:0] data_o, + // OTP interface + output logic otp_req_o, + output prim_otp_pkg::cmd_e otp_cmd_o, + output logic [OtpSizeWidth-1:0] otp_size_o, + output logic [OtpIfWidth-1:0] otp_wdata_o, + output logic [OtpAddrWidth-1:0] otp_addr_o, + input otp_gnt_i, + input otp_rvalid_i, + input [ScrmblBlockWidth-1:0] otp_rdata_i, + input prim_otp_pkg::err_e otp_err_i, + // Scrambling mutex request + output logic scrmbl_mtx_req_o, + input scrmbl_mtx_gnt_i, + // Scrambling datapath interface + output otp_scrmbl_cmd_e scrmbl_cmd_o, + output digest_mode_e scrmbl_mode_o, + output logic [ConstSelWidth-1:0] scrmbl_sel_o, + output logic [ScrmblBlockWidth-1:0] scrmbl_data_o, + output logic scrmbl_valid_o, + input logic scrmbl_ready_i, + input logic scrmbl_valid_i, + input logic [ScrmblBlockWidth-1:0] scrmbl_data_i +); + + //////////////////////// + // Integration Checks // + //////////////////////// + + import prim_mubi_pkg::*; + import prim_util_pkg::vbits; + + localparam int unsigned DigestOffsetInt = (int'(Info.offset) + + int'(Info.size) - ScrmblBlockWidth/8); + localparam int NumScrmblBlocks = int'(Info.size) / (ScrmblBlockWidth/8); + localparam int CntWidth = vbits(NumScrmblBlocks); + + localparam bit [OtpByteAddrWidth-1:0] DigestOffset = DigestOffsetInt[OtpByteAddrWidth-1:0]; + + localparam int unsigned LastScrmblBlockInt = NumScrmblBlocks - 1; + localparam int unsigned PenultimateScrmblBlockInt = NumScrmblBlocks - 2; + localparam bit [CntWidth-1:0] LastScrmblBlock = LastScrmblBlockInt[CntWidth-1:0]; + localparam bit [CntWidth-1:0] PenultimateScrmblBlock = PenultimateScrmblBlockInt[CntWidth-1:0]; + + // Integration checks for parameters. + `ASSERT_INIT(OffsetMustBeBlockAligned_A, (Info.offset % (ScrmblBlockWidth/8)) == 0) + `ASSERT_INIT(SizeMustBeBlockAligned_A, (Info.size % (ScrmblBlockWidth/8)) == 0) + `ASSERT_INIT(DigestOffsetMustBeRepresentable_A, DigestOffsetInt == int'(DigestOffset)) + `ASSERT(ScrambledImpliesDigest_A, Info.secret |-> Info.hw_digest) + `ASSERT(WriteLockImpliesDigest_A, Info.read_lock |-> Info.hw_digest) + `ASSERT(ReadLockImpliesDigest_A, Info.write_lock |-> Info.hw_digest) + + // This feature is only supposed to be used with partitions that are not scrambled + // and that do not have a digest. + `ASSERT(BypassEnable0_A, Info.secret |-> lc_ctrl_pkg::lc_tx_test_false_strict(check_byp_en_i)) + `ASSERT(BypassEnable1_A, Info.hw_digest |-> lc_ctrl_pkg::lc_tx_test_false_strict(check_byp_en_i)) + + /////////////////////// + // OTP Partition FSM // + /////////////////////// + + // SEC_CM: PART.FSM.SPARSE + // Encoding generated with: + // $ ./util/design/sparse-fsm-encode.py -d 5 -m 16 -n 12 \ + // -s 3370657881 --language=sv + // + // Hamming distance histogram: + // + // 0: -- + // 1: -- + // 2: -- + // 3: -- + // 4: -- + // 5: |||||||||||||| (28.33%) + // 6: |||||||||||||||||||| (38.33%) + // 7: |||||||||| (19.17%) + // 8: ||| (5.83%) + // 9: || (4.17%) + // 10: | (2.50%) + // 11: (0.83%) + // 12: (0.83%) + // + // Minimum Hamming distance: 5 + // Maximum Hamming distance: 12 + // Minimum Hamming weight: 4 + // Maximum Hamming weight: 8 + // + localparam int StateWidth = 12; + typedef enum logic [StateWidth-1:0] { + ResetSt = 12'b011000001110, + InitSt = 12'b110100100111, + InitWaitSt = 12'b001110110001, + InitDescrSt = 12'b110010000100, + InitDescrWaitSt = 12'b100110101000, + IdleSt = 12'b010101001101, + IntegScrSt = 12'b110101011010, + IntegScrWaitSt = 12'b100010011111, + IntegDigClrSt = 12'b101001000001, + IntegDigSt = 12'b011101100010, + IntegDigPadSt = 12'b001101010111, + IntegDigFinSt = 12'b011011100101, + IntegDigWaitSt = 12'b100011110010, + CnstyReadSt = 12'b000001101011, + CnstyReadWaitSt = 12'b101001111100, + ErrorSt = 12'b010110111110 + } state_e; + + typedef enum logic { + ScrmblData, + OtpData + } data_sel_e; + + typedef enum logic { + PartOffset, + DigOffset + } base_sel_e; + + state_e state_d, state_q; + otp_err_e error_d, error_q; + data_sel_e data_sel; + base_sel_e base_sel; + mubi8_t dout_locked_d, dout_locked_q; + logic [CntWidth-1:0] cnt; + logic cnt_en, cnt_clr, cnt_err; + logic ecc_err; + logic buffer_reg_en; + logic [ScrmblBlockWidth-1:0] data_mux; + + // Output partition error state. + assign error_o = error_q; + + // This partition cannot do any write accesses, hence we tie this + // constantly off. + assign otp_wdata_o = '0; + // Depending on the partition configuration, the wrapper is instructed to ignore integrity + // calculations and checks. To be on the safe side, the partition filters error responses at this + // point and does not report any integrity errors if integrity is disabled. + otp_err_e otp_err; + if (Info.integrity) begin : gen_integrity + assign otp_cmd_o = prim_otp_pkg::Read; + assign otp_err = otp_err_e'(otp_err_i); + end else begin : gen_no_integrity + assign otp_cmd_o = prim_otp_pkg::ReadRaw; + always_comb begin + if (otp_err_e'(otp_err_i) inside {MacroEccCorrError, MacroEccUncorrError}) begin + otp_err = NoError; + end else begin + otp_err = otp_err_e'(otp_err_i); + end + end + end + + always_comb begin : p_fsm + state_d = state_q; + + // Redundantly encoded lock signal for buffer regs. + dout_locked_d = dout_locked_q; + + // OTP signals + otp_req_o = 1'b0; + + // Scrambling mutex + scrmbl_mtx_req_o = 1'b0; + + // Scrambling datapath + scrmbl_cmd_o = LoadShadow; + scrmbl_sel_o = CnstyDigest; + scrmbl_mode_o = StandardMode; + scrmbl_valid_o = 1'b0; + + // Counter + cnt_en = 1'b0; + cnt_clr = 1'b0; + base_sel = PartOffset; + + // Buffer register + buffer_reg_en = 1'b0; + data_sel = OtpData; + + // Error Register + error_d = error_q; + fsm_err_o = 1'b0; + + // Integrity/Consistency check responses + cnsty_chk_ack_o = 1'b0; + integ_chk_ack_o = 1'b0; + + unique case (state_q) + /////////////////////////////////////////////////////////////////// + // State right after reset. Wait here until we get a an + // initialization request. + ResetSt: begin + if (init_req_i) begin + state_d = InitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Initialization reads out the digest only in unbuffered + // partitions. Wait here until the OTP request has been granted. + // And then wait until the OTP word comes back. + InitSt: begin + otp_req_o = 1'b1; + if (otp_gnt_i) begin + state_d = InitWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response and write to buffer register, then go to + // descrambling state. In case an OTP transaction fails, latch the + // OTP error code and jump to a + // terminal error state. + InitWaitSt: begin + if (otp_rvalid_i) begin + buffer_reg_en = 1'b1; + if (otp_err inside {NoError, MacroEccCorrError}) begin + // Once we've read and descrambled the whole partition, we can go to integrity + // verification. Note that the last block is the digest value, which does not + // have to be descrambled. + if (cnt == LastScrmblBlock) begin + state_d = IntegDigClrSt; + // Only need to descramble if this is a scrambled partition. + // Otherwise, we can just go back to InitSt and read the next block. + end else if (Info.secret) begin + state_d = InitDescrSt; + end else begin + state_d = InitSt; + cnt_en = 1'b1; + end + // At this point the only error that we could have gotten are correctable ECC errors. + if (otp_err != NoError) begin + error_d = MacroEccCorrError; + end + end else begin + state_d = ErrorSt; + error_d = otp_err; + end + end + end + /////////////////////////////////////////////////////////////////// + // Descrambling state. This first acquires the scrambling + // datapath mutex. Note that once the mutex is acquired, we have + // exclusive access to the scrambling datapath until we release + // the mutex by deasserting scrmbl_mtx_req_o. + // SEC_CM: SECRET.MEM.SCRAMBLE + InitDescrSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = Decrypt; + scrmbl_sel_o = Info.key_sel; + if (scrmbl_mtx_gnt_i && scrmbl_ready_i) begin + state_d = InitDescrWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for the descrambled data to return. Note that we release + // the mutex lock upon leaving this state. + // SEC_CM: SECRET.MEM.SCRAMBLE + InitDescrWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_sel_o = Info.key_sel; + data_sel = ScrmblData; + if (scrmbl_valid_i) begin + state_d = InitSt; + buffer_reg_en = 1'b1; + cnt_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Idle state. We basically wait for integrity and consistency check + // triggers in this state. + IdleSt: begin + if (integ_chk_req_i) begin + if (Info.hw_digest) begin + state_d = IntegDigClrSt; + // In case there is nothing to check we can just + // acknowledge the request right away, without going to the + // integrity check. + end else begin + integ_chk_ack_o = 1'b1; + end + end else if (cnsty_chk_req_i) begin + state_d = CnstyReadSt; + cnt_clr = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Read the digest. Wait here until the OTP request has been granted. + // And then wait until the OTP word comes back. + // SEC_CM: PART.DATA_REG.BKGN_CHK + CnstyReadSt: begin + otp_req_o = 1'b1; + // In case this partition has a hardware digest, we only have to read + // and compare the digest value. In that case we select the digest offset here. + // Otherwise we have to read and compare the whole partition, in which case we + // select the partition offset, which is the default assignment of base_sel. + if (Info.hw_digest) begin + base_sel = DigOffset; + end + if (otp_gnt_i) begin + state_d = CnstyReadWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response and compare the digest. In case there is + // a mismatch, lock down the partition and go into the terminal error + // state. In case an OTP transaction fails, latch the OTP error code + // and jump to a terminal error state. + // SEC_CM: PART.DATA_REG.BKGN_CHK + CnstyReadWaitSt: begin + if (otp_rvalid_i) begin + if (otp_err inside {NoError, MacroEccCorrError}) begin + // Check whether we need to compare the digest or the full partition + // contents here. + if (Info.hw_digest) begin + // Note that we ignore this check if the digest is still blank. + if (digest_o == data_mux || digest_o == '0) begin + state_d = IdleSt; + cnsty_chk_ack_o = 1'b1; + // Error out and lock the partition if this check fails. + end else begin + state_d = ErrorSt; + error_d = CheckFailError; + // The check has finished and found an error. + cnsty_chk_ack_o = 1'b1; + end + end else begin + // Check whether the read data corresponds with the data buffered in regs. + // Note that this particular check can be bypassed in case a transition is ongoing. + if (scrmbl_data_o == data_mux || + lc_ctrl_pkg::lc_tx_test_true_strict(check_byp_en_i)) begin + // Can go back to idle and acknowledge the + // request if this is the last block. + if (cnt == LastScrmblBlock) begin + state_d = IdleSt; + cnsty_chk_ack_o = 1'b1; + // Need to go back and read out more blocks. + end else begin + state_d = CnstyReadSt; + cnt_en = 1'b1; + end + end else begin + state_d = ErrorSt; + error_d = CheckFailError; + // The check has finished and found an error. + cnsty_chk_ack_o = 1'b1; + end + end + // At this point the only error that we could have gotten are correctable ECC errors. + if (otp_err != NoError) begin + error_d = MacroEccCorrError; + end + end else begin + state_d = ErrorSt; + error_d = otp_err; + // The check has finished and found an error. + cnsty_chk_ack_o = 1'b1; + end + end + end + /////////////////////////////////////////////////////////////////// + // First, acquire the mutex for the digest and clear the digest state. + // SEC_CM: PART.DATA_REG.BKGN_CHK + IntegDigClrSt: begin + // Check whether this partition requires checking at all. + if (Info.hw_digest) begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + cnt_clr = 1'b1; + // Need to reset the digest state and set it to chained + // mode if this partition is scrambled. + scrmbl_cmd_o = DigestInit; + if (Info.secret) begin + scrmbl_mode_o = ChainedMode; + if (scrmbl_mtx_gnt_i && scrmbl_ready_i) begin + state_d = IntegScrSt; + end + // If this partition is not scrambled, we can just directly + // jump to the digest state. + end else begin + scrmbl_mode_o = StandardMode; + if (scrmbl_mtx_gnt_i && scrmbl_ready_i) begin + state_d = IntegDigSt; + end + end + // Otherwise, if this partition is not digest protected, + // we can just go to idle, since there is nothing to check. + // Note that we do not come back to this state in case there is no + // digest, and hence it is safe to unlock the buffer regs at this point. + // This is the only way the buffer regs can get unlocked. + end else begin + state_d = IdleSt; + if (mubi8_test_true_strict(dout_locked_q)) begin + dout_locked_d = MuBi8False; + end + end + end + /////////////////////////////////////////////////////////////////// + // Scramble buffered data (which is held in plaintext form). + // This moves the previous scrambling result into the shadow reg + // for later use. + // SEC_CM: PART.DATA_REG.BKGN_CHK + IntegScrSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = Encrypt; + scrmbl_sel_o = Info.key_sel; + if (scrmbl_ready_i) begin + state_d = IntegScrWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for the scrambled data to return. + // SEC_CM: PART.DATA_REG.BKGN_CHK + IntegScrWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_sel_o = Info.key_sel; + if (scrmbl_valid_i) begin + state_d = IntegDigSt; + end + end + /////////////////////////////////////////////////////////////////// + // Push the word read into the scrambling datapath. The last + // block is repeated in case the number blocks in this partition + // is odd. + // SEC_CM: PART.MEM.DIGEST + // SEC_CM: PART.DATA_REG.BKGN_CHK + IntegDigSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + if (scrmbl_ready_i) begin + cnt_en = 1'b1; + // No need to digest the digest value itself + if (cnt == PenultimateScrmblBlock) begin + // Note that the digest operates on 128bit blocks since the data is fed in via the + // PRESENT key input. Therefore, we only trigger a digest update on every second + // 64bit block that is pushed into the scrambling datapath. + if (cnt[0]) begin + scrmbl_cmd_o = Digest; + state_d = IntegDigFinSt; + end else begin + state_d = IntegDigPadSt; + cnt_en = 1'b0; + end + end else begin + // Trigger digest round in case this is the second block in a row. + if (cnt[0]) begin + scrmbl_cmd_o = Digest; + end + // Go back and scramble the next data block if this is + // a scrambled partition. Otherwise just stay here. + if (Info.secret) begin + state_d = IntegScrSt; + end + end + end + end + /////////////////////////////////////////////////////////////////// + // Padding state. When we get here, we've copied the last encryption + // result into the shadow register such that we've effectively + // repeated the last block twice in order to pad the data to 128bit. + // SEC_CM: PART.MEM.DIGEST + // SEC_CM: PART.DATA_REG.BKGN_CHK + IntegDigPadSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = Digest; + if (scrmbl_ready_i) begin + state_d = IntegDigFinSt; + end + end + /////////////////////////////////////////////////////////////////// + // Trigger digest finalization and go wait for the result. + // SEC_CM: PART.MEM.DIGEST + // SEC_CM: PART.DATA_REG.BKGN_CHK + IntegDigFinSt: begin + scrmbl_mtx_req_o = 1'b1; + scrmbl_valid_o = 1'b1; + scrmbl_cmd_o = DigestFinalize; + if (scrmbl_ready_i) begin + state_d = IntegDigWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for the digest to return, and double check whether the digest + // matches. If yes, unlock the partition. Otherwise, go into the terminal + // error state, where the partition will be locked down. + // SEC_CM: PART.MEM.DIGEST + // SEC_CM: PART.DATA_REG.BKGN_CHK + IntegDigWaitSt: begin + scrmbl_mtx_req_o = 1'b1; + data_sel = ScrmblData; + if (scrmbl_valid_i) begin + // This is the only way the buffer regs can get unlocked. + // Note that we ignore this check if the digest is still blank. + if (digest_o == data_mux || digest_o == '0) begin + state_d = IdleSt; + // If the partition is still locked, this is the first integrity check after + // initialization. This is the only way the buffer regs can get unlocked. + if (mubi8_test_true_strict(dout_locked_q)) begin + dout_locked_d = MuBi8False; + // Otherwise, this integrity check has requested by the LFSR timer, and we have + // to acknowledge its completion. + end else begin + integ_chk_ack_o = 1'b1; + end + // Error out and lock the partition if this check fails. + end else begin + state_d = ErrorSt; + error_d = CheckFailError; + // The check has finished and found an error. + integ_chk_ack_o = 1'b1; + end + end + end + /////////////////////////////////////////////////////////////////// + // Terminal Error State. This locks access to the partition. + // Make sure the partition signals an error state if no error + // code has been latched so far, and lock the buffer regs down. + ErrorSt: begin + dout_locked_d = MuBi8True; + if (error_q == NoError) begin + error_d = FsmStateError; + end + // If we are in error state, we cannot execute the checks anymore. + // Hence the acknowledgements are returned immediately. + cnsty_chk_ack_o = 1'b1; + integ_chk_ack_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + // We should never get here. If we do (e.g. via a malicious + // glitch), error out immediately. + default: begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + endcase // state_q + + + // Unconditionally jump into the terminal error state in case of + // an ECC error or escalation, and lock access to the partition down. + // SEC_CM: PART.FSM.LOCAL_ESC + if (ecc_err) begin + state_d = ErrorSt; + if (state_q != ErrorSt) begin + error_d = CheckFailError; + end + end + // SEC_CM: PART.FSM.LOCAL_ESC, PART.FSM.GLOBAL_ESC + if (lc_ctrl_pkg::lc_tx_test_true_loose(escalate_en_i) || cnt_err) begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + if (state_q != ErrorSt) begin + error_d = FsmStateError; + end + end + end + + //////////////////////////// + // Address Calc and Muxes // + //////////////////////////// + + // Address counter - this is only used for computing a digest, hence the increment is + // fixed to 8 byte. + // SEC_CM: PART.CTR.REDUN + prim_count #( + .Width(CntWidth) + ) u_prim_count ( + .clk_i, + .rst_ni, + .clr_i(cnt_clr), + .set_i(1'b0), + .set_cnt_i('0), + .incr_en_i(cnt_en), + .decr_en_i(1'b0), + .step_i(CntWidth'(1)), + .commit_i(1'b1), + .cnt_o(cnt), + .cnt_after_commit_o(), + .err_o(cnt_err) + ); + + logic [OtpByteAddrWidth-1:0] addr_base; + assign addr_base = (base_sel == DigOffset) ? DigestOffset : Info.offset; + + // Note that OTP works on halfword (16bit) addresses, hence need to + // shift the addresses appropriately. + logic [OtpByteAddrWidth-1:0] addr_calc; + assign addr_calc = OtpByteAddrWidth'({cnt, {$clog2(ScrmblBlockWidth/8){1'b0}}}) + addr_base; + assign otp_addr_o = addr_calc[OtpByteAddrWidth-1:OtpAddrShift]; + + if (OtpAddrShift > 0) begin : gen_unused + logic unused_bits; + assign unused_bits = ^addr_calc[OtpAddrShift-1:0]; + end + + // Always transfer 64bit blocks. + assign otp_size_o = OtpSizeWidth'(unsigned'(ScrmblBlockWidth / OtpWidth) - 1); + + assign data_mux = (data_sel == ScrmblData) ? scrmbl_data_i : otp_rdata_i; + + ///////////////// + // Buffer Regs // + ///////////////// + + // SEC_CM: PART.DATA_REG.INTEGRITY + logic [Info.size*8-1:0] data; + otp_ctrl_ecc_reg #( + .Width ( ScrmblBlockWidth ), + .Depth ( NumScrmblBlocks ) + ) u_otp_ctrl_ecc_reg ( + .clk_i, + .rst_ni, + .wren_i ( buffer_reg_en ), + .addr_i ( cnt ), + .wdata_i ( data_mux ), + .rdata_o ( scrmbl_data_o ), + .data_o ( data ), + .ecc_err_o ( ecc_err ) + ); + + // We have successfully initialized the partition once it has been unlocked. + assign init_done_o = mubi8_test_false_strict(dout_locked_q); + // Hardware output gating. + // Note that this is decoupled from the DAI access rules further below. + assign data_o = (init_done_o) ? data : DataDefault; + // The digest does not have to be gated. + assign digest_o = data[$high(data_o) -: ScrmblBlockWidth]; + + //////////////////////// + // DAI Access Control // + //////////////////////// + + // Aggregate all possible DAI write /readlocks. The partition is also locked when uninitialized. + // Note that the locks are redundantly encoded values. + part_access_t access_pre; + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_write_lock_pre ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(dout_locked_q, access_i.write_lock)), + .mubi_o(access_pre.write_lock) + ); + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_read_lock_pre ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(dout_locked_q, access_i.read_lock)), + .mubi_o(access_pre.read_lock) + ); + + // SEC_CM: PART.MEM.SW_UNWRITABLE + if (Info.write_lock) begin : gen_digest_write_lock + mubi8_t digest_locked; + assign digest_locked = (digest_o != '0) ? MuBi8True : MuBi8False; + + // This prevents the synthesis tool from optimizing the multibit signal. + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_write_lock ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(access_pre.write_lock, digest_locked)), + .mubi_o(access_o.write_lock) + ); + + `ASSERT(DigestWriteLocksPartition_A, digest_o |-> mubi8_test_true_loose(access_o.write_lock)) + end else begin : gen_no_digest_write_lock + assign access_o.write_lock = access_pre.write_lock; + end + + // SEC_CM: PART.MEM.SW_UNREADABLE + if (Info.read_lock) begin : gen_digest_read_lock + mubi8_t digest_locked; + assign digest_locked = (digest_o != '0) ? MuBi8True : MuBi8False; + + // This prevents the synthesis tool from optimizing the multibit signal. + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_read_lock ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(access_pre.read_lock, digest_locked)), + .mubi_o(access_o.read_lock) + ); + + `ASSERT(DigestReadLocksPartition_A, digest_o |-> mubi8_test_true_loose(access_o.read_lock)) + end else begin : gen_no_digest_read_lock + assign access_o.read_lock = access_pre.read_lock; + end + + /////////////// + // Registers // + /////////////// + + `PRIM_FLOP_SPARSE_FSM(u_state_regs, state_d, state_q, state_e, ResetSt) + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + error_q <= NoError; + // data output is locked by default + dout_locked_q <= MuBi8True; + end else begin + error_q <= error_d; + dout_locked_q <= dout_locked_d; + end + end + + //////////////// + // Assertions // + //////////////// + + // Known assertions + `ASSERT_KNOWN(InitDoneKnown_A, init_done_o) + `ASSERT_KNOWN(IntegChkAckKnown_A, integ_chk_ack_o) + `ASSERT_KNOWN(CnstyChkAckKnown_A, cnsty_chk_ack_o) + `ASSERT_KNOWN(ErrorKnown_A, error_o) + `ASSERT_KNOWN(AccessKnown_A, access_o) + `ASSERT_KNOWN(DigestKnown_A, digest_o) + `ASSERT_KNOWN(DataKnown_A, data_o) + `ASSERT_KNOWN(OtpReqKnown_A, otp_req_o) + `ASSERT_KNOWN(OtpCmdKnown_A, otp_cmd_o) + `ASSERT_KNOWN(OtpSizeKnown_A, otp_size_o) + `ASSERT_KNOWN(OtpWdataKnown_A, otp_wdata_o) + `ASSERT_KNOWN(OtpAddrKnown_A, otp_addr_o) + `ASSERT_KNOWN(ScrmblMtxReqKnown_A, scrmbl_mtx_req_o) + `ASSERT_KNOWN(ScrmblCmdKnown_A, scrmbl_cmd_o) + `ASSERT_KNOWN(ScrmblModeKnown_A, scrmbl_mode_o) + `ASSERT_KNOWN(ScrmblSelKnown_A, scrmbl_sel_o) + `ASSERT_KNOWN(ScrmblDataKnown_A, scrmbl_data_o) + `ASSERT_KNOWN(ScrmblValidKnown_A, scrmbl_valid_o) + + // Uninitialized partitions should always be locked, no matter what. + `ASSERT(InitWriteLocksPartition_A, + mubi8_test_true_loose(dout_locked_q) + |-> + mubi8_test_true_loose(access_o.write_lock)) + `ASSERT(InitReadLocksPartition_A, + mubi8_test_true_loose(dout_locked_q) + |-> + mubi8_test_true_loose(access_o.read_lock)) + // Incoming Lock propagation + `ASSERT(WriteLockPropagation_A, + mubi8_test_true_loose(access_i.write_lock) + |-> + mubi8_test_true_loose(access_o.write_lock)) + `ASSERT(ReadLockPropagation_A, + mubi8_test_true_loose(access_i.read_lock) + |-> + mubi8_test_true_loose(access_o.read_lock)) + // ECC error in buffer regs + `ASSERT(EccErrorState_A, + ecc_err + |=> + state_q == ErrorSt) + // OTP error response + `ASSERT(OtpErrorState_A, + state_q inside {InitWaitSt, CnstyReadWaitSt} && otp_rvalid_i && + !(otp_err inside {NoError, MacroEccCorrError}) && !ecc_err + |=> + state_q == ErrorSt && error_o == $past(otp_err)) + + // The partition size must be greater than one scrambling block for the address calculation + // and muxing to work correctly. + `ASSERT_INIT(OtpPartBufSize_A, Info.size > (ScrmblBlockWidth/8)) + +endmodule : otp_ctrl_part_buf diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl_part_pkg.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_pkg.sv similarity index 99% rename from hw/ip/otp_ctrl/rtl/otp_ctrl_part_pkg.sv rename to hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_pkg.sv index ae1e92cef8678d..0197fa02f93933 100644 --- a/hw/ip/otp_ctrl/rtl/otp_ctrl_part_pkg.sv +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_pkg.sv @@ -4,14 +4,12 @@ // // Package partition metadata. // -// DO NOT EDIT THIS FILE DIRECTLY. -// It has been generated with ./util/design/gen-otp-mmap.py - package otp_ctrl_part_pkg; import prim_util_pkg::vbits; import otp_ctrl_reg_pkg::*; import otp_ctrl_pkg::*; + import otp_ctrl_top_specific_pkg::*; //////////////////////////////////// // Scrambling Constants and Types // diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv new file mode 100644 index 00000000000000..09eb0fe0d73fa5 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_part_unbuf.sv @@ -0,0 +1,532 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Unbuffered partition for OTP controller. +// + +`include "prim_flop_macros.sv" + +module otp_ctrl_part_unbuf + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + import otp_ctrl_part_pkg::*; + import otp_ctrl_top_specific_pkg::*; +#( + // Partition information. + parameter part_info_t Info = PartInfoDefault +) ( + input clk_i, + input rst_ni, + // Pulse to start partition initialisation (required once per power cycle). + input init_req_i, + output logic init_done_o, + // Escalation input. This moves the FSM into a terminal state and locks down + // the partition. + input lc_ctrl_pkg::lc_tx_t escalate_en_i, + // Output error state of partition, to be consumed by OTP error/alert logic. + // Note that most errors are not recoverable and move the partition FSM into + // a terminal error state. + output otp_err_e error_o, + // This error signal is pulsed high if the FSM has been glitched into an invalid state. + // Although it is somewhat redundant with the error code in error_o above, it is + // meant to cover cases where we already latched an error code while the FSM is + // glitched into an invalid state (since in that case, the error code will not be + // overridden with the FSM error code so that the original error code is still + // discoverable). + output logic fsm_err_o, + // Access/lock status + // SEC_CM: ACCESS.CTRL.MUBI + input part_access_t access_i, // runtime lock from CSRs + output part_access_t access_o, + // Buffered 64bit digest output. + output logic [ScrmblBlockWidth-1:0] digest_o, + // Interface to TL-UL adapter + input logic tlul_req_i, + output logic tlul_gnt_o, + input [SwWindowAddrWidth-1:0] tlul_addr_i, + output logic [1:0] tlul_rerror_o, + output logic tlul_rvalid_o, + output logic [31:0] tlul_rdata_o, + // OTP interface + output logic otp_req_o, + output prim_otp_pkg::cmd_e otp_cmd_o, + output logic [OtpSizeWidth-1:0] otp_size_o, + output logic [OtpIfWidth-1:0] otp_wdata_o, + output logic [OtpAddrWidth-1:0] otp_addr_o, + input otp_gnt_i, + input otp_rvalid_i, + input [ScrmblBlockWidth-1:0] otp_rdata_i, + input prim_otp_pkg::err_e otp_err_i +); + + //////////////////////// + // Integration Checks // + //////////////////////// + + import prim_mubi_pkg::*; + import prim_util_pkg::vbits; + + localparam logic [OtpByteAddrWidth:0] PartEnd = (OtpByteAddrWidth+1)'(Info.offset) + + (OtpByteAddrWidth+1)'(Info.size); + localparam int unsigned DigestOffsetInt = int'(PartEnd) - ScrmblBlockWidth/8; + + localparam bit [OtpByteAddrWidth-1:0] DigestOffset = DigestOffsetInt[OtpByteAddrWidth-1:0]; + + // Integration checks for parameters. + `ASSERT_INIT(OffsetMustBeBlockAligned_A, (Info.offset % (ScrmblBlockWidth/8)) == 0) + `ASSERT_INIT(SizeMustBeBlockAligned_A, (Info.size % (ScrmblBlockWidth/8)) == 0) + `ASSERT_INIT(DigestOffsetMustBeRepresentable_A, DigestOffsetInt == int'(DigestOffset)) + + /////////////////////// + // OTP Partition FSM // + /////////////////////// + + // SEC_CM: PART.FSM.SPARSE + // Encoding generated with: + // $ ./util/design/sparse-fsm-encode.py -d 5 -m 7 -n 10 \ + // -s 4247417884 --language=sv + // + // Hamming distance histogram: + // + // 0: -- + // 1: -- + // 2: -- + // 3: -- + // 4: -- + // 5: |||||||||||||||||||| (52.38%) + // 6: |||||||||||| (33.33%) + // 7: | (4.76%) + // 8: ||| (9.52%) + // 9: -- + // 10: -- + // + // Minimum Hamming distance: 5 + // Maximum Hamming distance: 8 + // Minimum Hamming weight: 3 + // Maximum Hamming weight: 9 + // + localparam int StateWidth = 10; + typedef enum logic [StateWidth-1:0] { + ResetSt = 10'b1010110110, + InitSt = 10'b0100010011, + InitWaitSt = 10'b0001011000, + IdleSt = 10'b1011101001, + ReadSt = 10'b0101101110, + ReadWaitSt = 10'b0110100101, + ErrorSt = 10'b1111011111 + } state_e; + + typedef enum logic { + DigestAddrSel = 1'b0, + DataAddrSel = 1'b1 + } addr_sel_e; + + state_e state_d, state_q; + addr_sel_e otp_addr_sel; + otp_err_e error_d, error_q; + + logic digest_reg_en; + logic ecc_err; + + logic tlul_addr_in_range; + logic [SwWindowAddrWidth-1:0] tlul_addr_d, tlul_addr_q; + + // This is only used to return bus errors when the FSM is in ErrorSt. + logic pending_tlul_error_d, pending_tlul_error_q; + + // Output partition error state. + assign error_o = error_q; + + // This partition cannot do any write accesses, hence we tie this + // constantly off. + assign otp_wdata_o = '0; + // Depending on the partition configuration, the wrapper is instructed to ignore integrity + // calculations and checks. To be on the safe side, the partition filters error responses at this + // point and does not report any integrity errors if integrity is disabled. + otp_err_e otp_err; + if (Info.integrity) begin : gen_integrity + assign otp_cmd_o = prim_otp_pkg::Read; + assign otp_err = otp_err_e'(otp_err_i); + end else begin : gen_no_integrity + assign otp_cmd_o = prim_otp_pkg::ReadRaw; + always_comb begin + if (otp_err_e'(otp_err_i) inside {MacroEccCorrError, MacroEccUncorrError}) begin + otp_err = NoError; + end else begin + otp_err = otp_err_e'(otp_err_i); + end + end + end + + `ASSERT_KNOWN(FsmStateKnown_A, state_q) + always_comb begin : p_fsm + // Default assignments + state_d = state_q; + + // Response to init request + init_done_o = 1'b0; + + // OTP signals + otp_req_o = 1'b0; + otp_addr_sel = DigestAddrSel; + + // TL-UL signals + tlul_gnt_o = 1'b0; + tlul_rvalid_o = 1'b0; + tlul_rerror_o = '0; + + // Enable for buffered digest register + digest_reg_en = 1'b0; + + // Error Register + error_d = error_q; + pending_tlul_error_d = 1'b0; + fsm_err_o = 1'b0; + + unique case (state_q) + /////////////////////////////////////////////////////////////////// + // State right after reset. Wait here until we get a an + // initialization request. + ResetSt: begin + if (init_req_i) begin + // If the partition does not have a digest, no initialization is necessary. + if (Info.sw_digest) begin + state_d = InitSt; + end else begin + state_d = IdleSt; + end + end + end + /////////////////////////////////////////////////////////////////// + // Initialization reads out the digest only in unbuffered + // partitions. Wait here until the OTP request has been granted. + // And then wait until the OTP word comes back. + InitSt: begin + otp_req_o = 1'b1; + if (otp_gnt_i) begin + state_d = InitWaitSt; + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response and write to digest buffer register. In + // case an OTP transaction fails, latch the OTP error code and + // jump to a terminal error state. + InitWaitSt: begin + if (otp_rvalid_i) begin + digest_reg_en = 1'b1; + if (otp_err inside {NoError, MacroEccCorrError}) begin + state_d = IdleSt; + // At this point the only error that we could have gotten are correctable ECC errors. + if (otp_err != NoError) begin + error_d = MacroEccCorrError; + end + end else begin + state_d = ErrorSt; + error_d = otp_err; + end + end + end + /////////////////////////////////////////////////////////////////// + // Wait for TL-UL requests coming in. + // Then latch address and go to readout state. + IdleSt: begin + init_done_o = 1'b1; + if (tlul_req_i) begin + error_d = NoError; // clear recoverable soft errors. + state_d = ReadSt; + tlul_gnt_o = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // If the address is out of bounds, or if the partition is + // locked, signal back a bus error. Note that such an error does + // not cause the partition to go into error state. Otherwise if + // these checks pass, an OTP word is requested. + ReadSt: begin + init_done_o = 1'b1; + // Double check the address range. + if (tlul_addr_in_range && mubi8_test_false_strict(access_o.read_lock)) begin + otp_req_o = 1'b1; + otp_addr_sel = DataAddrSel; + if (otp_gnt_i) begin + state_d = ReadWaitSt; + end + end else begin + state_d = IdleSt; + error_d = AccessError; // Signal this error, but do not go into terminal error state. + tlul_rvalid_o = 1'b1; + tlul_rerror_o = 2'b11; // This causes the TL-UL adapter to return a bus error. + end + end + /////////////////////////////////////////////////////////////////// + // Wait for OTP response and release the TL-UL response. In + // case an OTP transaction fails, latch the OTP error code, + // signal a TL-Ul bus error and jump to a terminal error state. + ReadWaitSt: begin + init_done_o = 1'b1; + if (otp_rvalid_i) begin + tlul_rvalid_o = 1'b1; + if (otp_err inside {NoError, MacroEccCorrError}) begin + state_d = IdleSt; + // At this point the only error that we could have gotten are correctable ECC errors. + if (otp_err != NoError) begin + error_d = MacroEccCorrError; + end + end else begin + state_d = ErrorSt; + error_d = otp_err; + // This causes the TL-UL adapter to return a bus error. + tlul_rerror_o = 2'b11; + end + end + end + /////////////////////////////////////////////////////////////////// + // Terminal Error State. This locks access to the partition. + // Make sure the partition signals an error state if no error + // code has been latched so far. + ErrorSt: begin + if (error_q == NoError) begin + error_d = FsmStateError; + end + + // Return bus errors if there are pending TL-UL requests. + if (pending_tlul_error_q) begin + tlul_rerror_o = 2'b11; + tlul_rvalid_o = 1'b1; + end else if (tlul_req_i) begin + tlul_gnt_o = 1'b1; + pending_tlul_error_d = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // We should never get here. If we do (e.g. via a malicious + // glitch), error out immediately. + default: begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + endcase // state_q + + // Unconditionally jump into the terminal error state in case of + // an ECC error or escalation, and lock access to the partition down. + // SEC_CM: PART.FSM.LOCAL_ESC + if (ecc_err) begin + state_d = ErrorSt; + if (state_q != ErrorSt) begin + error_d = CheckFailError; + end + end + // SEC_CM: PART.FSM.GLOBAL_ESC + if (lc_ctrl_pkg::lc_tx_test_true_loose(escalate_en_i)) begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + if (state_q != ErrorSt) begin + error_d = FsmStateError; + end + end + end + + /////////////////////////////////// + // Signals to/from TL-UL Adapter // + /////////////////////////////////// + + assign tlul_addr_d = tlul_addr_i; + // Do not forward data in case of an error. + assign tlul_rdata_o = (tlul_rvalid_o && tlul_rerror_o == '0) ? otp_rdata_i[31:0] : '0; + + if (Info.offset == 0) begin : gen_zero_offset + assign tlul_addr_in_range = {1'b0, tlul_addr_q, 2'b00} < PartEnd; + + end else begin : gen_nonzero_offset + assign tlul_addr_in_range = {tlul_addr_q, 2'b00} >= Info.offset && + {1'b0, tlul_addr_q, 2'b00} < PartEnd; + end + + // Note that OTP works on halfword (16bit) addresses, hence need to + // shift the addresses appropriately. + logic [OtpByteAddrWidth-1:0] addr_calc; + assign addr_calc = (otp_addr_sel == DigestAddrSel) ? DigestOffset : {tlul_addr_q, 2'b00}; + assign otp_addr_o = addr_calc[OtpByteAddrWidth-1:OtpAddrShift]; + + if (OtpAddrShift > 0) begin : gen_unused + logic unused_bits; + assign unused_bits = ^addr_calc[OtpAddrShift-1:0]; + end + + // Request 32bit except in case of the digest. + assign otp_size_o = (otp_addr_sel == DigestAddrSel) ? + OtpSizeWidth'(unsigned'(ScrmblBlockWidth / OtpWidth - 1)) : + OtpSizeWidth'(unsigned'(32 / OtpWidth - 1)); + + //////////////// + // Digest Reg // + //////////////// + + if (Info.sw_digest) begin : gen_ecc_reg + // SEC_CM: PART.DATA_REG.INTEGRITY + otp_ctrl_ecc_reg #( + .Width ( ScrmblBlockWidth ), + .Depth ( 1 ) + ) u_otp_ctrl_ecc_reg ( + .clk_i, + .rst_ni, + .wren_i ( digest_reg_en ), + .addr_i ( '0 ), + .wdata_i ( otp_rdata_i ), + .rdata_o ( ), + .data_o ( digest_o ), + .ecc_err_o ( ecc_err ) + ); + end else begin : gen_no_ecc_reg + logic unused_digest_reg_en; + logic unused_rdata; + assign unused_digest_reg_en = digest_reg_en; + assign unused_rdata = ^otp_rdata_i[32 +: 32]; // Upper word is not connected in this case. + assign digest_o = '0; + assign ecc_err = 1'b0; + end + + //////////////////////// + // DAI Access Control // + //////////////////////// + + mubi8_t init_locked; + assign init_locked = (~init_done_o) ? MuBi8True : MuBi8False; + + // Aggregate all possible DAI write locks. The partition is also locked when uninitialized. + // Note that the locks are redundantly encoded values. + part_access_t access_pre; + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_write_lock_pre ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(init_locked, access_i.write_lock)), + .mubi_o(access_pre.write_lock) + ); + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_read_lock_pre ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(init_locked, access_i.read_lock)), + .mubi_o(access_pre.read_lock) + ); + + // SEC_CM: PART.MEM.SW_UNWRITABLE + if (Info.write_lock) begin : gen_digest_write_lock + mubi8_t digest_locked; + assign digest_locked = (digest_o != '0) ? MuBi8True : MuBi8False; + + // This prevents the synthesis tool from optimizing the multibit signal. + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_write_lock ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(access_pre.write_lock, digest_locked)), + .mubi_o(access_o.write_lock) + ); + + `ASSERT(DigestWriteLocksPartition_A, digest_o |-> mubi8_test_true_loose(access_o.write_lock)) + end else begin : gen_no_digest_write_lock + assign access_o.write_lock = access_pre.write_lock; + end + + // SEC_CM: PART.MEM.SW_UNREADABLE + if (Info.read_lock) begin : gen_digest_read_lock + mubi8_t digest_locked; + assign digest_locked = (digest_o != '0) ? MuBi8True : MuBi8False; + + // This prevents the synthesis tool from optimizing the multibit signal. + prim_mubi8_sender #( + .AsyncOn(0) + ) u_prim_mubi8_sender_read_lock ( + .clk_i, + .rst_ni, + .mubi_i(mubi8_and_lo(access_pre.read_lock, digest_locked)), + .mubi_o(access_o.read_lock) + ); + + `ASSERT(DigestReadLocksPartition_A, digest_o |-> mubi8_test_true_loose(access_o.read_lock)) + end else begin : gen_no_digest_read_lock + assign access_o.read_lock = access_pre.read_lock; + end + + /////////////// + // Registers // + /////////////// + + `PRIM_FLOP_SPARSE_FSM(u_state_regs, state_d, state_q, state_e, ResetSt) + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + error_q <= NoError; + tlul_addr_q <= '0; + pending_tlul_error_q <= 1'b0; + end else begin + error_q <= error_d; + pending_tlul_error_q <= pending_tlul_error_d; + if (tlul_gnt_o) begin + tlul_addr_q <= tlul_addr_d; + end + end + end + + //////////////// + // Assertions // + //////////////// + + // Known assertions + `ASSERT_KNOWN(InitDoneKnown_A, init_done_o) + `ASSERT_KNOWN(ErrorKnown_A, error_o) + `ASSERT_KNOWN(AccessKnown_A, access_o) + `ASSERT_KNOWN(DigestKnown_A, digest_o) + `ASSERT_KNOWN(TlulGntKnown_A, tlul_gnt_o) + `ASSERT_KNOWN(TlulRerrorKnown_A, tlul_rerror_o) + `ASSERT_KNOWN(TlulRvalidKnown_A, tlul_rvalid_o) + `ASSERT_KNOWN(TlulRdataKnown_A, tlul_rdata_o) + `ASSERT_KNOWN(OtpReqKnown_A, otp_req_o) + `ASSERT_KNOWN(OtpCmdKnown_A, otp_cmd_o) + `ASSERT_KNOWN(OtpSizeKnown_A, otp_size_o) + `ASSERT_KNOWN(OtpWdataKnown_A, otp_wdata_o) + `ASSERT_KNOWN(OtpAddrKnown_A, otp_addr_o) + + // Uninitialized partitions should always be locked, no matter what. + `ASSERT(InitWriteLocksPartition_A, + ~init_done_o + |-> + mubi8_test_true_loose(access_o.write_lock)) + `ASSERT(InitReadLocksPartition_A, + ~init_done_o + |-> + mubi8_test_true_loose(access_o.read_lock)) + // Incoming Lock propagation + `ASSERT(WriteLockPropagation_A, + mubi8_test_true_loose(access_i.write_lock) + |-> + mubi8_test_true_loose(access_o.write_lock)) + `ASSERT(ReadLockPropagation_A, + mubi8_test_true_loose(access_i.read_lock) + |-> + mubi8_test_true_loose(access_o.read_lock)) + // If the partition is read locked, the TL-UL access must error out + `ASSERT(TlulReadOnReadLock_A, + tlul_req_i && tlul_gnt_o ##1 mubi8_test_true_loose(access_o.read_lock) + |-> + tlul_rerror_o > '0 && tlul_rvalid_o) + // ECC error in buffer regs. + `ASSERT(EccErrorState_A, + ecc_err + |=> + state_q == ErrorSt) + // OTP error response + `ASSERT(OtpErrorState_A, + state_q inside {InitWaitSt, ReadWaitSt} && otp_rvalid_i && + !(otp_err inside {NoError, MacroEccCorrError}) && !ecc_err + |=> + state_q == ErrorSt && error_o == $past(otp_err)) + +endmodule : otp_ctrl_part_unbuf diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_pkg.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_pkg.sv new file mode 100644 index 00000000000000..0450b103b07741 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_pkg.sv @@ -0,0 +1,232 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// This package can be imported by generic IPs: +// - It does not import otp_ctrl_reg_pkg, which is generated and top-specific. + +package otp_ctrl_pkg; + + //////////////////////// + // General Parameters // + //////////////////////// + + // Number of vendor-specific test CSR bits coming from and going to + // the life cycle TAP registers. + parameter int OtpTestCtrlWidth = 32; + parameter int OtpTestStatusWidth = 32; + parameter int OtpTestVectWidth = 8; + + parameter int DeviceIdWidth = 256; + typedef logic [DeviceIdWidth-1:0] otp_device_id_t; + + parameter int ManufStateWidth = 256; + typedef logic [ManufStateWidth-1:0] otp_manuf_state_t; + + // OTP-macro specific + parameter int OtpPwrSeqWidth = 2; + + ///////////////////////////////// + // Typedefs for OTP Scrambling // + ///////////////////////////////// + + parameter int ScrmblKeyWidth = 128; + parameter int ScrmblBlockWidth = 64; + + /////////////////////////////// + // Typedefs for LC Interface // + /////////////////////////////// + + // The tokens below are all hash post-images + typedef struct packed { + logic valid; + logic error; + // Use lc_state_t and lc_cnt_t here as very wide enumerations ( > 64 bits ) + // are not supported for virtual interfaces by Excelium yet + // https://github.com/lowRISC/opentitan/issues/8884 (Cadence issue: cds_46570160) + // The enumeration types lc_state_e and lc_cnt_e are still ok in other circumstances + lc_ctrl_state_pkg::lc_state_t state; + lc_ctrl_state_pkg::lc_cnt_t count; + // This is set to "On" if the partition containing the + // root secrets have been locked. In that case, the device + // is considered "personalized". + lc_ctrl_pkg::lc_tx_t secrets_valid; + // This is set to "On" if the partition containing the + // test tokens has been locked. + lc_ctrl_pkg::lc_tx_t test_tokens_valid; + lc_ctrl_state_pkg::lc_token_t test_unlock_token; + lc_ctrl_state_pkg::lc_token_t test_exit_token; + // This is set to "On" if the partition containing the + // rma token has been locked. + lc_ctrl_pkg::lc_tx_t rma_token_valid; + lc_ctrl_state_pkg::lc_token_t rma_token; + } otp_lc_data_t; + + // Default for dangling connection. + // Note that we put the life cycle into + // TEST_UNLOCKED0 by default such that top levels without + // the OTP controller can still function. + parameter otp_lc_data_t OTP_LC_DATA_DEFAULT = '{ + valid: 1'b1, + error: 1'b0, + state: lc_ctrl_state_pkg::LcStTestUnlocked0, + count: lc_ctrl_state_pkg::LcCnt1, + secrets_valid: lc_ctrl_pkg::Off, + test_tokens_valid: lc_ctrl_pkg::Off, + test_unlock_token: '0, + test_exit_token: '0, + rma_token_valid: lc_ctrl_pkg::Off, + rma_token: '0 + }; + + typedef struct packed { + logic req; + lc_ctrl_state_pkg::lc_state_e state; + lc_ctrl_state_pkg::lc_cnt_e count; + } lc_otp_program_req_t; + + typedef struct packed { + logic err; + logic ack; + } lc_otp_program_rsp_t; + + typedef struct packed { + logic [OtpTestCtrlWidth-1:0] ctrl; + } lc_otp_vendor_test_req_t; + + typedef struct packed { + logic [OtpTestStatusWidth-1:0] status; + } lc_otp_vendor_test_rsp_t; + + //////////////////////////////// + // Typedefs for Key Broadcast // + //////////////////////////////// + + parameter int FlashKeySeedWidth = 256; + parameter int SramKeySeedWidth = 128; + parameter int KeyMgrKeyWidth = 256; + parameter int FlashKeyWidth = 128; + parameter int SramKeyWidth = 128; + parameter int SramNonceWidth = 128; + parameter int OtbnKeyWidth = 128; + parameter int OtbnNonceWidth = 64; + + typedef logic [SramKeyWidth-1:0] sram_key_t; + typedef logic [SramNonceWidth-1:0] sram_nonce_t; + typedef logic [OtbnKeyWidth-1:0] otbn_key_t; + typedef logic [OtbnNonceWidth-1:0] otbn_nonce_t; + + localparam int OtbnNonceSel = OtbnNonceWidth / ScrmblBlockWidth; + localparam int FlashNonceSel = FlashKeyWidth / ScrmblBlockWidth; + localparam int SramNonceSel = SramNonceWidth / ScrmblBlockWidth; + + // Get maximum nonce width + localparam int NumNonceChunks = + (OtbnNonceWidth > FlashKeyWidth) ? + ((OtbnNonceWidth > SramNonceSel) ? OtbnNonceSel : SramNonceSel) : + ((FlashKeyWidth > SramNonceSel) ? FlashNonceSel : SramNonceSel); + + typedef struct packed { + logic [KeyMgrKeyWidth-1:0] creator_root_key_share0; + logic creator_root_key_share0_valid; + logic [KeyMgrKeyWidth-1:0] creator_root_key_share1; + logic creator_root_key_share1_valid; + logic [KeyMgrKeyWidth-1:0] creator_seed; + logic creator_seed_valid; + logic [KeyMgrKeyWidth-1:0] owner_seed; + logic owner_seed_valid; + } otp_keymgr_key_t; + + parameter otp_keymgr_key_t OTP_KEYMGR_KEY_DEFAULT = '{ + creator_root_key_share0: 256'hefb7ea7ee90093cf4affd9aaa2d6c0ec446cfdf5f2d5a0bfd7e2d93edc63a102, + creator_root_key_share0_valid: 1'b1, + creator_root_key_share1: 256'h56d24a00181de99e0f690b447a8dde2a1ffb8bc306707107aa6e2410f15cfc37, + creator_root_key_share1_valid: 1'b1, + creator_seed: 256'hc7c50b38655cc87f821e5b07fed85d2c07e222a9e00bef308b3eccba0ba406fa, + creator_seed_valid: 1'b1, + owner_seed: 256'hf5052c0f14782d8b066be9f49c0b2000d3643ff3723ea7db972f69cd3e2e3e68, + owner_seed_valid: 1'b1 + }; + + typedef struct packed { + logic data_req; // Requests static key for data scrambling. + logic addr_req; // Requests static key for address scrambling. + } flash_otp_key_req_t; + + typedef struct packed { + logic req; // Requests ephemeral scrambling key and nonce. + } sram_otp_key_req_t; + + typedef struct packed { + logic req; // Requests ephemeral scrambling key and nonce. + } otbn_otp_key_req_t; + + typedef struct packed { + logic data_ack; // Ack for data key. + logic addr_ack; // Ack for address key. + logic [FlashKeyWidth-1:0] key; // 128bit static scrambling key. + logic [FlashKeyWidth-1:0] rand_key; + logic seed_valid; // Set to 1 if the key seed has been provisioned and is + // valid. + } flash_otp_key_rsp_t; + + // Default for dangling connection + parameter flash_otp_key_rsp_t FLASH_OTP_KEY_RSP_DEFAULT = '{ + data_ack: 1'b1, + addr_ack: 1'b1, + key: '0, + rand_key: '0, + seed_valid: 1'b1 + }; + + typedef struct packed { + logic ack; // Ack for key. + sram_key_t key; // 128bit ephemeral scrambling key. + sram_nonce_t nonce; // 128bit nonce. + logic seed_valid; // Set to 1 if the key seed has been provisioned and is valid. + } sram_otp_key_rsp_t; + + // Default for dangling connection + parameter sram_otp_key_rsp_t SRAM_OTP_KEY_RSP_DEFAULT = '{ + ack: 1'b1, + key: '0, + nonce: '0, + seed_valid: 1'b1 + }; + + typedef struct packed { + logic ack; // Ack for key. + otbn_key_t key; // 128bit ephemeral scrambling key. + otbn_nonce_t nonce; // 256bit nonce. + logic seed_valid; // Set to 1 if the key seed has been provisioned and is valid. + } otbn_otp_key_rsp_t; + + //////////////////////////////// + // Power/Reset Ctrl Interface // + //////////////////////////////// + + typedef struct packed { + logic init; + } pwr_otp_init_req_t; + + typedef struct packed { + logic done; + } pwr_otp_init_rsp_t; + + typedef struct packed { + logic idle; + } otp_pwr_state_t; + + /////////////////// + // AST Interface // + /////////////////// + + typedef struct packed { + logic [OtpPwrSeqWidth-1:0] pwr_seq; + } otp_ast_req_t; + + typedef struct packed { + logic [OtpPwrSeqWidth-1:0] pwr_seq_h; + } otp_ast_rsp_t; + +endpackage : otp_ctrl_pkg diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_prim_reg_top.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_prim_reg_top.sv new file mode 100644 index 00000000000000..fb30ea5203126a --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_prim_reg_top.sv @@ -0,0 +1,1467 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// +// Register Top module auto-generated by `reggen` + +`include "prim_assert.sv" + +module otp_ctrl_prim_reg_top ( + input clk_i, + input rst_ni, + input tlul_pkg::tl_h2d_t tl_i, + output tlul_pkg::tl_d2h_t tl_o, + // To HW + output otp_ctrl_reg_pkg::otp_ctrl_prim_reg2hw_t reg2hw, // Write + input otp_ctrl_reg_pkg::otp_ctrl_prim_hw2reg_t hw2reg, // Read + + // Integrity check errors + output logic intg_err_o +); + + import otp_ctrl_reg_pkg::* ; + + localparam int AW = 5; + localparam int DW = 32; + localparam int DBW = DW/8; // Byte Width + + // register signals + logic reg_we; + logic reg_re; + logic [AW-1:0] reg_addr; + logic [DW-1:0] reg_wdata; + logic [DBW-1:0] reg_be; + logic [DW-1:0] reg_rdata; + logic reg_error; + + logic addrmiss, wr_err; + + logic [DW-1:0] reg_rdata_next; + logic reg_busy; + + tlul_pkg::tl_h2d_t tl_reg_h2d; + tlul_pkg::tl_d2h_t tl_reg_d2h; + + + // incoming payload check + logic intg_err; + tlul_cmd_intg_chk u_chk ( + .tl_i(tl_i), + .err_o(intg_err) + ); + + // also check for spurious write enables + logic reg_we_err; + logic [7:0] reg_we_check; + prim_reg_we_check #( + .OneHotWidth(8) + ) u_prim_reg_we_check ( + .clk_i(clk_i), + .rst_ni(rst_ni), + .oh_i (reg_we_check), + .en_i (reg_we && !addrmiss), + .err_o (reg_we_err) + ); + + logic err_q; + always_ff @(posedge clk_i or negedge rst_ni) begin + if (!rst_ni) begin + err_q <= '0; + end else if (intg_err || reg_we_err) begin + err_q <= 1'b1; + end + end + + // integrity error output is permanent and should be used for alert generation + // register errors are transactional + assign intg_err_o = err_q | intg_err | reg_we_err; + + // outgoing integrity generation + tlul_pkg::tl_d2h_t tl_o_pre; + tlul_rsp_intg_gen #( + .EnableRspIntgGen(1), + .EnableDataIntgGen(1) + ) u_rsp_intg_gen ( + .tl_i(tl_o_pre), + .tl_o(tl_o) + ); + + assign tl_reg_h2d = tl_i; + assign tl_o_pre = tl_reg_d2h; + + tlul_adapter_reg #( + .RegAw(AW), + .RegDw(DW), + .EnableDataIntgGen(0) + ) u_reg_if ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + .tl_i (tl_reg_h2d), + .tl_o (tl_reg_d2h), + + .en_ifetch_i(prim_mubi_pkg::MuBi4False), + .intg_error_o(), + + .we_o (reg_we), + .re_o (reg_re), + .addr_o (reg_addr), + .wdata_o (reg_wdata), + .be_o (reg_be), + .busy_i (reg_busy), + .rdata_i (reg_rdata), + .error_i (reg_error) + ); + + // cdc oversampling signals + + assign reg_rdata = reg_rdata_next ; + assign reg_error = addrmiss | wr_err | intg_err; + + // Define SW related signals + // Format: __{wd|we|qs} + // or _{wd|we|qs} if field == 1 or 0 + logic csr0_we; + logic csr0_field0_qs; + logic csr0_field0_wd; + logic csr0_field1_qs; + logic csr0_field1_wd; + logic csr0_field2_qs; + logic csr0_field2_wd; + logic [9:0] csr0_field3_qs; + logic [9:0] csr0_field3_wd; + logic [10:0] csr0_field4_qs; + logic [10:0] csr0_field4_wd; + logic csr1_we; + logic [6:0] csr1_field0_qs; + logic [6:0] csr1_field0_wd; + logic csr1_field1_qs; + logic csr1_field1_wd; + logic [6:0] csr1_field2_qs; + logic [6:0] csr1_field2_wd; + logic csr1_field3_qs; + logic csr1_field3_wd; + logic [15:0] csr1_field4_qs; + logic [15:0] csr1_field4_wd; + logic csr2_we; + logic csr2_qs; + logic csr2_wd; + logic csr3_we; + logic [2:0] csr3_field0_qs; + logic [2:0] csr3_field0_wd; + logic [9:0] csr3_field1_qs; + logic [9:0] csr3_field1_wd; + logic csr3_field2_qs; + logic csr3_field2_wd; + logic csr3_field3_qs; + logic csr3_field4_qs; + logic csr3_field5_qs; + logic csr3_field6_qs; + logic csr3_field7_qs; + logic csr3_field8_qs; + logic csr4_we; + logic [9:0] csr4_field0_qs; + logic [9:0] csr4_field0_wd; + logic csr4_field1_qs; + logic csr4_field1_wd; + logic csr4_field2_qs; + logic csr4_field2_wd; + logic csr4_field3_qs; + logic csr4_field3_wd; + logic csr5_we; + logic [5:0] csr5_field0_qs; + logic [5:0] csr5_field0_wd; + logic [1:0] csr5_field1_qs; + logic [1:0] csr5_field1_wd; + logic csr5_field2_qs; + logic [2:0] csr5_field3_qs; + logic csr5_field4_qs; + logic csr5_field5_qs; + logic [15:0] csr5_field6_qs; + logic [15:0] csr5_field6_wd; + logic csr6_we; + logic [9:0] csr6_field0_qs; + logic [9:0] csr6_field0_wd; + logic csr6_field1_qs; + logic csr6_field1_wd; + logic csr6_field2_qs; + logic csr6_field2_wd; + logic [15:0] csr6_field3_qs; + logic [15:0] csr6_field3_wd; + logic [5:0] csr7_field0_qs; + logic [2:0] csr7_field1_qs; + logic csr7_field2_qs; + logic csr7_field3_qs; + + // Register instances + // R[csr0]: V(False) + // F[field0]: 0:0 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr0_field0 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr0_we), + .wd (csr0_field0_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr0.field0.q), + .ds (), + + // to register interface (read) + .qs (csr0_field0_qs) + ); + + // F[field1]: 1:1 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr0_field1 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr0_we), + .wd (csr0_field1_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr0.field1.q), + .ds (), + + // to register interface (read) + .qs (csr0_field1_qs) + ); + + // F[field2]: 2:2 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr0_field2 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr0_we), + .wd (csr0_field2_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr0.field2.q), + .ds (), + + // to register interface (read) + .qs (csr0_field2_qs) + ); + + // F[field3]: 13:4 + prim_subreg #( + .DW (10), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (10'h0), + .Mubi (1'b0) + ) u_csr0_field3 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr0_we), + .wd (csr0_field3_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr0.field3.q), + .ds (), + + // to register interface (read) + .qs (csr0_field3_qs) + ); + + // F[field4]: 26:16 + prim_subreg #( + .DW (11), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (11'h0), + .Mubi (1'b0) + ) u_csr0_field4 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr0_we), + .wd (csr0_field4_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr0.field4.q), + .ds (), + + // to register interface (read) + .qs (csr0_field4_qs) + ); + + + // R[csr1]: V(False) + // F[field0]: 6:0 + prim_subreg #( + .DW (7), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (7'h0), + .Mubi (1'b0) + ) u_csr1_field0 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr1_we), + .wd (csr1_field0_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr1.field0.q), + .ds (), + + // to register interface (read) + .qs (csr1_field0_qs) + ); + + // F[field1]: 7:7 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr1_field1 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr1_we), + .wd (csr1_field1_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr1.field1.q), + .ds (), + + // to register interface (read) + .qs (csr1_field1_qs) + ); + + // F[field2]: 14:8 + prim_subreg #( + .DW (7), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (7'h0), + .Mubi (1'b0) + ) u_csr1_field2 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr1_we), + .wd (csr1_field2_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr1.field2.q), + .ds (), + + // to register interface (read) + .qs (csr1_field2_qs) + ); + + // F[field3]: 15:15 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr1_field3 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr1_we), + .wd (csr1_field3_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr1.field3.q), + .ds (), + + // to register interface (read) + .qs (csr1_field3_qs) + ); + + // F[field4]: 31:16 + prim_subreg #( + .DW (16), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (16'h0), + .Mubi (1'b0) + ) u_csr1_field4 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr1_we), + .wd (csr1_field4_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr1.field4.q), + .ds (), + + // to register interface (read) + .qs (csr1_field4_qs) + ); + + + // R[csr2]: V(False) + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr2 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr2_we), + .wd (csr2_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr2.q), + .ds (), + + // to register interface (read) + .qs (csr2_qs) + ); + + + // R[csr3]: V(False) + // F[field0]: 2:0 + prim_subreg #( + .DW (3), + .SwAccess(prim_subreg_pkg::SwAccessW1C), + .RESVAL (3'h0), + .Mubi (1'b0) + ) u_csr3_field0 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr3_we), + .wd (csr3_field0_wd), + + // from internal hardware + .de (hw2reg.csr3.field0.de), + .d (hw2reg.csr3.field0.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr3.field0.q), + .ds (), + + // to register interface (read) + .qs (csr3_field0_qs) + ); + + // F[field1]: 13:4 + prim_subreg #( + .DW (10), + .SwAccess(prim_subreg_pkg::SwAccessW1C), + .RESVAL (10'h0), + .Mubi (1'b0) + ) u_csr3_field1 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr3_we), + .wd (csr3_field1_wd), + + // from internal hardware + .de (hw2reg.csr3.field1.de), + .d (hw2reg.csr3.field1.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr3.field1.q), + .ds (), + + // to register interface (read) + .qs (csr3_field1_qs) + ); + + // F[field2]: 16:16 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessW1C), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr3_field2 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr3_we), + .wd (csr3_field2_wd), + + // from internal hardware + .de (hw2reg.csr3.field2.de), + .d (hw2reg.csr3.field2.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr3.field2.q), + .ds (), + + // to register interface (read) + .qs (csr3_field2_qs) + ); + + // F[field3]: 17:17 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRO), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr3_field3 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (1'b0), + .wd ('0), + + // from internal hardware + .de (hw2reg.csr3.field3.de), + .d (hw2reg.csr3.field3.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr3.field3.q), + .ds (), + + // to register interface (read) + .qs (csr3_field3_qs) + ); + + // F[field4]: 18:18 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRO), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr3_field4 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (1'b0), + .wd ('0), + + // from internal hardware + .de (hw2reg.csr3.field4.de), + .d (hw2reg.csr3.field4.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr3.field4.q), + .ds (), + + // to register interface (read) + .qs (csr3_field4_qs) + ); + + // F[field5]: 19:19 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRO), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr3_field5 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (1'b0), + .wd ('0), + + // from internal hardware + .de (hw2reg.csr3.field5.de), + .d (hw2reg.csr3.field5.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr3.field5.q), + .ds (), + + // to register interface (read) + .qs (csr3_field5_qs) + ); + + // F[field6]: 20:20 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRO), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr3_field6 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (1'b0), + .wd ('0), + + // from internal hardware + .de (hw2reg.csr3.field6.de), + .d (hw2reg.csr3.field6.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr3.field6.q), + .ds (), + + // to register interface (read) + .qs (csr3_field6_qs) + ); + + // F[field7]: 21:21 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRO), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr3_field7 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (1'b0), + .wd ('0), + + // from internal hardware + .de (hw2reg.csr3.field7.de), + .d (hw2reg.csr3.field7.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr3.field7.q), + .ds (), + + // to register interface (read) + .qs (csr3_field7_qs) + ); + + // F[field8]: 22:22 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRO), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr3_field8 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (1'b0), + .wd ('0), + + // from internal hardware + .de (hw2reg.csr3.field8.de), + .d (hw2reg.csr3.field8.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr3.field8.q), + .ds (), + + // to register interface (read) + .qs (csr3_field8_qs) + ); + + + // R[csr4]: V(False) + // F[field0]: 9:0 + prim_subreg #( + .DW (10), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (10'h0), + .Mubi (1'b0) + ) u_csr4_field0 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr4_we), + .wd (csr4_field0_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr4.field0.q), + .ds (), + + // to register interface (read) + .qs (csr4_field0_qs) + ); + + // F[field1]: 12:12 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr4_field1 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr4_we), + .wd (csr4_field1_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr4.field1.q), + .ds (), + + // to register interface (read) + .qs (csr4_field1_qs) + ); + + // F[field2]: 13:13 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr4_field2 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr4_we), + .wd (csr4_field2_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr4.field2.q), + .ds (), + + // to register interface (read) + .qs (csr4_field2_qs) + ); + + // F[field3]: 14:14 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr4_field3 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr4_we), + .wd (csr4_field3_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr4.field3.q), + .ds (), + + // to register interface (read) + .qs (csr4_field3_qs) + ); + + + // R[csr5]: V(False) + // F[field0]: 5:0 + prim_subreg #( + .DW (6), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (6'h0), + .Mubi (1'b0) + ) u_csr5_field0 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr5_we), + .wd (csr5_field0_wd), + + // from internal hardware + .de (hw2reg.csr5.field0.de), + .d (hw2reg.csr5.field0.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr5.field0.q), + .ds (), + + // to register interface (read) + .qs (csr5_field0_qs) + ); + + // F[field1]: 7:6 + prim_subreg #( + .DW (2), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (2'h0), + .Mubi (1'b0) + ) u_csr5_field1 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr5_we), + .wd (csr5_field1_wd), + + // from internal hardware + .de (hw2reg.csr5.field1.de), + .d (hw2reg.csr5.field1.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr5.field1.q), + .ds (), + + // to register interface (read) + .qs (csr5_field1_qs) + ); + + // F[field2]: 8:8 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRO), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr5_field2 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (1'b0), + .wd ('0), + + // from internal hardware + .de (hw2reg.csr5.field2.de), + .d (hw2reg.csr5.field2.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr5.field2.q), + .ds (), + + // to register interface (read) + .qs (csr5_field2_qs) + ); + + // F[field3]: 11:9 + prim_subreg #( + .DW (3), + .SwAccess(prim_subreg_pkg::SwAccessRO), + .RESVAL (3'h0), + .Mubi (1'b0) + ) u_csr5_field3 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (1'b0), + .wd ('0), + + // from internal hardware + .de (hw2reg.csr5.field3.de), + .d (hw2reg.csr5.field3.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr5.field3.q), + .ds (), + + // to register interface (read) + .qs (csr5_field3_qs) + ); + + // F[field4]: 12:12 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRO), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr5_field4 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (1'b0), + .wd ('0), + + // from internal hardware + .de (hw2reg.csr5.field4.de), + .d (hw2reg.csr5.field4.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr5.field4.q), + .ds (), + + // to register interface (read) + .qs (csr5_field4_qs) + ); + + // F[field5]: 13:13 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRO), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr5_field5 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (1'b0), + .wd ('0), + + // from internal hardware + .de (hw2reg.csr5.field5.de), + .d (hw2reg.csr5.field5.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr5.field5.q), + .ds (), + + // to register interface (read) + .qs (csr5_field5_qs) + ); + + // F[field6]: 31:16 + prim_subreg #( + .DW (16), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (16'h0), + .Mubi (1'b0) + ) u_csr5_field6 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr5_we), + .wd (csr5_field6_wd), + + // from internal hardware + .de (hw2reg.csr5.field6.de), + .d (hw2reg.csr5.field6.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr5.field6.q), + .ds (), + + // to register interface (read) + .qs (csr5_field6_qs) + ); + + + // R[csr6]: V(False) + // F[field0]: 9:0 + prim_subreg #( + .DW (10), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (10'h0), + .Mubi (1'b0) + ) u_csr6_field0 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr6_we), + .wd (csr6_field0_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr6.field0.q), + .ds (), + + // to register interface (read) + .qs (csr6_field0_qs) + ); + + // F[field1]: 11:11 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr6_field1 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr6_we), + .wd (csr6_field1_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr6.field1.q), + .ds (), + + // to register interface (read) + .qs (csr6_field1_qs) + ); + + // F[field2]: 12:12 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr6_field2 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr6_we), + .wd (csr6_field2_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr6.field2.q), + .ds (), + + // to register interface (read) + .qs (csr6_field2_qs) + ); + + // F[field3]: 31:16 + prim_subreg #( + .DW (16), + .SwAccess(prim_subreg_pkg::SwAccessRW), + .RESVAL (16'h0), + .Mubi (1'b0) + ) u_csr6_field3 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (csr6_we), + .wd (csr6_field3_wd), + + // from internal hardware + .de (1'b0), + .d ('0), + + // to internal hardware + .qe (), + .q (reg2hw.csr6.field3.q), + .ds (), + + // to register interface (read) + .qs (csr6_field3_qs) + ); + + + // R[csr7]: V(False) + // F[field0]: 5:0 + prim_subreg #( + .DW (6), + .SwAccess(prim_subreg_pkg::SwAccessRO), + .RESVAL (6'h0), + .Mubi (1'b0) + ) u_csr7_field0 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (1'b0), + .wd ('0), + + // from internal hardware + .de (hw2reg.csr7.field0.de), + .d (hw2reg.csr7.field0.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr7.field0.q), + .ds (), + + // to register interface (read) + .qs (csr7_field0_qs) + ); + + // F[field1]: 10:8 + prim_subreg #( + .DW (3), + .SwAccess(prim_subreg_pkg::SwAccessRO), + .RESVAL (3'h0), + .Mubi (1'b0) + ) u_csr7_field1 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (1'b0), + .wd ('0), + + // from internal hardware + .de (hw2reg.csr7.field1.de), + .d (hw2reg.csr7.field1.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr7.field1.q), + .ds (), + + // to register interface (read) + .qs (csr7_field1_qs) + ); + + // F[field2]: 14:14 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRO), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr7_field2 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (1'b0), + .wd ('0), + + // from internal hardware + .de (hw2reg.csr7.field2.de), + .d (hw2reg.csr7.field2.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr7.field2.q), + .ds (), + + // to register interface (read) + .qs (csr7_field2_qs) + ); + + // F[field3]: 15:15 + prim_subreg #( + .DW (1), + .SwAccess(prim_subreg_pkg::SwAccessRO), + .RESVAL (1'h0), + .Mubi (1'b0) + ) u_csr7_field3 ( + .clk_i (clk_i), + .rst_ni (rst_ni), + + // from register interface + .we (1'b0), + .wd ('0), + + // from internal hardware + .de (hw2reg.csr7.field3.de), + .d (hw2reg.csr7.field3.d), + + // to internal hardware + .qe (), + .q (reg2hw.csr7.field3.q), + .ds (), + + // to register interface (read) + .qs (csr7_field3_qs) + ); + + + + logic [7:0] addr_hit; + always_comb begin + addr_hit = '0; + addr_hit[0] = (reg_addr == OTP_CTRL_CSR0_OFFSET); + addr_hit[1] = (reg_addr == OTP_CTRL_CSR1_OFFSET); + addr_hit[2] = (reg_addr == OTP_CTRL_CSR2_OFFSET); + addr_hit[3] = (reg_addr == OTP_CTRL_CSR3_OFFSET); + addr_hit[4] = (reg_addr == OTP_CTRL_CSR4_OFFSET); + addr_hit[5] = (reg_addr == OTP_CTRL_CSR5_OFFSET); + addr_hit[6] = (reg_addr == OTP_CTRL_CSR6_OFFSET); + addr_hit[7] = (reg_addr == OTP_CTRL_CSR7_OFFSET); + end + + assign addrmiss = (reg_re || reg_we) ? ~|addr_hit : 1'b0 ; + + // Check sub-word write is permitted + always_comb begin + wr_err = (reg_we & + ((addr_hit[0] & (|(OTP_CTRL_PRIM_PERMIT[0] & ~reg_be))) | + (addr_hit[1] & (|(OTP_CTRL_PRIM_PERMIT[1] & ~reg_be))) | + (addr_hit[2] & (|(OTP_CTRL_PRIM_PERMIT[2] & ~reg_be))) | + (addr_hit[3] & (|(OTP_CTRL_PRIM_PERMIT[3] & ~reg_be))) | + (addr_hit[4] & (|(OTP_CTRL_PRIM_PERMIT[4] & ~reg_be))) | + (addr_hit[5] & (|(OTP_CTRL_PRIM_PERMIT[5] & ~reg_be))) | + (addr_hit[6] & (|(OTP_CTRL_PRIM_PERMIT[6] & ~reg_be))) | + (addr_hit[7] & (|(OTP_CTRL_PRIM_PERMIT[7] & ~reg_be))))); + end + + // Generate write-enables + assign csr0_we = addr_hit[0] & reg_we & !reg_error; + + assign csr0_field0_wd = reg_wdata[0]; + + assign csr0_field1_wd = reg_wdata[1]; + + assign csr0_field2_wd = reg_wdata[2]; + + assign csr0_field3_wd = reg_wdata[13:4]; + + assign csr0_field4_wd = reg_wdata[26:16]; + assign csr1_we = addr_hit[1] & reg_we & !reg_error; + + assign csr1_field0_wd = reg_wdata[6:0]; + + assign csr1_field1_wd = reg_wdata[7]; + + assign csr1_field2_wd = reg_wdata[14:8]; + + assign csr1_field3_wd = reg_wdata[15]; + + assign csr1_field4_wd = reg_wdata[31:16]; + assign csr2_we = addr_hit[2] & reg_we & !reg_error; + + assign csr2_wd = reg_wdata[0]; + assign csr3_we = addr_hit[3] & reg_we & !reg_error; + + assign csr3_field0_wd = reg_wdata[2:0]; + + assign csr3_field1_wd = reg_wdata[13:4]; + + assign csr3_field2_wd = reg_wdata[16]; + assign csr4_we = addr_hit[4] & reg_we & !reg_error; + + assign csr4_field0_wd = reg_wdata[9:0]; + + assign csr4_field1_wd = reg_wdata[12]; + + assign csr4_field2_wd = reg_wdata[13]; + + assign csr4_field3_wd = reg_wdata[14]; + assign csr5_we = addr_hit[5] & reg_we & !reg_error; + + assign csr5_field0_wd = reg_wdata[5:0]; + + assign csr5_field1_wd = reg_wdata[7:6]; + + assign csr5_field6_wd = reg_wdata[31:16]; + assign csr6_we = addr_hit[6] & reg_we & !reg_error; + + assign csr6_field0_wd = reg_wdata[9:0]; + + assign csr6_field1_wd = reg_wdata[11]; + + assign csr6_field2_wd = reg_wdata[12]; + + assign csr6_field3_wd = reg_wdata[31:16]; + + // Assign write-enables to checker logic vector. + always_comb begin + reg_we_check = '0; + reg_we_check[0] = csr0_we; + reg_we_check[1] = csr1_we; + reg_we_check[2] = csr2_we; + reg_we_check[3] = csr3_we; + reg_we_check[4] = csr4_we; + reg_we_check[5] = csr5_we; + reg_we_check[6] = csr6_we; + reg_we_check[7] = 1'b0; + end + + // Read data return + always_comb begin + reg_rdata_next = '0; + unique case (1'b1) + addr_hit[0]: begin + reg_rdata_next[0] = csr0_field0_qs; + reg_rdata_next[1] = csr0_field1_qs; + reg_rdata_next[2] = csr0_field2_qs; + reg_rdata_next[13:4] = csr0_field3_qs; + reg_rdata_next[26:16] = csr0_field4_qs; + end + + addr_hit[1]: begin + reg_rdata_next[6:0] = csr1_field0_qs; + reg_rdata_next[7] = csr1_field1_qs; + reg_rdata_next[14:8] = csr1_field2_qs; + reg_rdata_next[15] = csr1_field3_qs; + reg_rdata_next[31:16] = csr1_field4_qs; + end + + addr_hit[2]: begin + reg_rdata_next[0] = csr2_qs; + end + + addr_hit[3]: begin + reg_rdata_next[2:0] = csr3_field0_qs; + reg_rdata_next[13:4] = csr3_field1_qs; + reg_rdata_next[16] = csr3_field2_qs; + reg_rdata_next[17] = csr3_field3_qs; + reg_rdata_next[18] = csr3_field4_qs; + reg_rdata_next[19] = csr3_field5_qs; + reg_rdata_next[20] = csr3_field6_qs; + reg_rdata_next[21] = csr3_field7_qs; + reg_rdata_next[22] = csr3_field8_qs; + end + + addr_hit[4]: begin + reg_rdata_next[9:0] = csr4_field0_qs; + reg_rdata_next[12] = csr4_field1_qs; + reg_rdata_next[13] = csr4_field2_qs; + reg_rdata_next[14] = csr4_field3_qs; + end + + addr_hit[5]: begin + reg_rdata_next[5:0] = csr5_field0_qs; + reg_rdata_next[7:6] = csr5_field1_qs; + reg_rdata_next[8] = csr5_field2_qs; + reg_rdata_next[11:9] = csr5_field3_qs; + reg_rdata_next[12] = csr5_field4_qs; + reg_rdata_next[13] = csr5_field5_qs; + reg_rdata_next[31:16] = csr5_field6_qs; + end + + addr_hit[6]: begin + reg_rdata_next[9:0] = csr6_field0_qs; + reg_rdata_next[11] = csr6_field1_qs; + reg_rdata_next[12] = csr6_field2_qs; + reg_rdata_next[31:16] = csr6_field3_qs; + end + + addr_hit[7]: begin + reg_rdata_next[5:0] = csr7_field0_qs; + reg_rdata_next[10:8] = csr7_field1_qs; + reg_rdata_next[14] = csr7_field2_qs; + reg_rdata_next[15] = csr7_field3_qs; + end + + default: begin + reg_rdata_next = '1; + end + endcase + end + + // shadow busy + logic shadow_busy; + assign shadow_busy = 1'b0; + + // register busy + assign reg_busy = shadow_busy; + + // Unused signal tieoff + + // wdata / byte enable are not always fully used + // add a blanket unused statement to handle lint waivers + logic unused_wdata; + logic unused_be; + assign unused_wdata = ^reg_wdata; + assign unused_be = ^reg_be; + + // Assertions for Register Interface + `ASSERT_PULSE(wePulse, reg_we, clk_i, !rst_ni) + `ASSERT_PULSE(rePulse, reg_re, clk_i, !rst_ni) + + `ASSERT(reAfterRv, $rose(reg_re || reg_we) |=> tl_o_pre.d_valid, clk_i, !rst_ni) + + `ASSERT(en2addrHit, (reg_we || reg_re) |-> $onehot0(addr_hit), clk_i, !rst_ni) + + // this is formulated as an assumption such that the FPV testbenches do disprove this + // property by mistake + //`ASSUME(reqParity, tl_reg_h2d.a_valid |-> tl_reg_h2d.a_user.chk_en == tlul_pkg::CheckDis) + +endmodule diff --git a/hw/ip/otp_ctrl/rtl/otp_ctrl_reg_pkg.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_reg_pkg.sv similarity index 100% rename from hw/ip/otp_ctrl/rtl/otp_ctrl_reg_pkg.sv rename to hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_reg_pkg.sv diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_scrmbl.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_scrmbl.sv new file mode 100644 index 00000000000000..7c5296929b6f85 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_scrmbl.sv @@ -0,0 +1,512 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// This module contains the scrambling datapath for the OTP controller. It basically consists of +// two single-round PRESENT primitives (one for encryption and one for decryption mode), a counter +// with a simple FSM and four working registers, as listed below. +// +// key_state_q (128bit): working register to hold the round key (needed for the key schedule). +// +// data_state_q (64bit): working register to hold the data state in between rounds. +// +// data_shadow_q (64bit): shadow register for holding a second 64bit block of input data. This is +// used to form a 128bit data block for the digest mode, which has a block +// size of 128bit. +// +// digest_state_q (64bit): register to hold the digest state in between digest updates. Technically, +// this is not needed when the data for the digest is fed into this block +// back-to-back. However, the partition integrity checks require that it is +// possible to interleave encryption operations and digest update steps, +// hence an additional state register is needed, as otherwise the digest +// state would be lost. +// +// The scrambling datapath is arranged such that it can also be used for calculating a digest using +// the encryption primitive in a Merkle-Damgard construction. To that end, the PRESENT block cipher +// is turned into a one way function according to the Davies-Meyer scheme. Note however that this +// makes the digest block size 128bit wide, since the Merkle-Damgard construction leverages the +// cipher key input to ingest data. +// +// The scrambling datapath exposes a few simple commands and the FSM hides the complexity +// of steering the appropriate muxes and keeping track of the cipher rounds. These commands are +// briefly explained below. +// +// Decrypt: This decrypts the data block provided via data_i with the key at index sel_i. +// +// Encrypt: This encrypts the data block provided via data_i with the key at index sel_i. +// In addition, this command copies the prvious result into a shadow register before +// the first encryption round for later use in the digest (see description further below). +// This enables interleaved encrypt/digest operation needed for the integrity checks of +// the secret partitions. +// +// LoadShadow: In "StandardMode", the LoadShadow command loads the data provided via data_i into a +// shadow register that is mapped to the lower 64bit of the 128bit digest input data +// block. In "ChainedMode", this command copies the contents of the data state register +// into the shadow register. +// +// DigestInit: This ensures that the digest initialization vector (IV) is selected upon the next +// call of the Digest command. Also, mode_i can be used to set the digest mode. If +// mode_i is set to "StandardMode", the data to be digested has to be provided via +// data_i and LoadShadow. If mode_i is set to "ChainedMode", the digest input is formed +// by concatenating the results of the revious two encryption commands. +// +// Digest: In "StandardMode", this command concatenates the data input supplied via data_i with +// the shadow register in order to form a 128bit block ({data_i, data_shadow_q}). This block +// is then used to encrypt the digest state. In "ChainedMode" digest mode, the 128bit block +// to be digested is formed by concatenating {data_state_q, data_shadow_q}. If a DigestInit +// command has been executed right before calling Digest, the IV selected with sel_i is +// used to initialize the state. +// +// DigestFinalize: This command encrypts the digest state with the finalization constant selected +// by sel_i in order to form the final digest. +// +// References: +// - https://docs.opentitan.org/hw/ip/otp_ctrl/doc/theory_of_operation.html#block-diagram +// - https://docs.opentitan.org/hw/ip/prim/doc/prim_present/ +// - https://en.wikipedia.org/wiki/Merkle-Damgard_construction +// - https://en.wikipedia.org/wiki/One-way_compression_function#Davies%E2%80%93Meyer +// - https://en.wikipedia.org/wiki/PRESENT +// - http://www.lightweightcrypto.org/present/present_ches2007.pdf +// + +`include "prim_flop_macros.sv" + +module otp_ctrl_scrmbl + import otp_ctrl_pkg::*; + import otp_ctrl_top_specific_pkg::*; + import otp_ctrl_part_pkg::*; +( + input clk_i, + input rst_ni, + // input data and command + input otp_scrmbl_cmd_e cmd_i, + input digest_mode_e mode_i, + input [ConstSelWidth-1:0] sel_i, + input [ScrmblBlockWidth-1:0] data_i, + input valid_i, + output logic ready_o, + // output data + output logic [ScrmblBlockWidth-1:0] data_o, + output logic valid_o, + // escalation input and FSM error indication + input lc_ctrl_pkg::lc_tx_t escalate_en_i, + output logic fsm_err_o +); + + import prim_util_pkg::vbits; + + //////////////////////// + // Decryption Key LUT // + //////////////////////// + + // Anchor keys, constants and IVs + key_array_t rnd_cnst_key_anchor; + digest_const_array_t rnd_cnst_digest_anchor; + digest_iv_array_t rnd_cnst_digest_iv_anchor; + + for (genvar i = 0; i < NumScrmblKeys; i++) begin : gen_anchor_keys + prim_sec_anchor_buf #( + .Width(ScrmblKeyWidth) + ) u_key_anchor_buf ( + .in_i(RndCnstKey[i]), + .out_o(rnd_cnst_key_anchor[i]) + ); + end + + for (genvar i = 0; i < NumDigestSets; i++) begin : gen_anchor_digests + prim_sec_anchor_buf #( + .Width(ScrmblKeyWidth) + ) u_const_anchor_buf ( + .in_i(RndCnstDigestConst[i]), + .out_o(rnd_cnst_digest_anchor[i]) + ); + + prim_sec_anchor_buf #( + .Width(ScrmblBlockWidth) + ) u_iv_anchor_buf ( + .in_i(RndCnstDigestIV[i]), + .out_o(rnd_cnst_digest_iv_anchor[i]) + ); + end + + + // Align these arrays to power of 2's to prevent X's in the muxing operations further below. + logic [2**$clog2(NumScrmblKeys)-1:0][ScrmblKeyWidth-1:0] otp_enc_key_lut; + logic [2**$clog2(NumScrmblKeys)-1:0][ScrmblKeyWidth-1:0] otp_dec_key_lut; + logic [2**$clog2(NumDigestSets)-1:0][ScrmblKeyWidth-1:0] digest_const_lut; + logic [2**$clog2(NumDigestSets)-1:0][ScrmblBlockWidth-1:0] digest_iv_lut; + + // This pre-calculates the inverse scrambling keys at elab time. + `ASSERT_INIT(NumMaxPresentRounds_A, NumPresentRounds <= 31) + + always_comb begin : p_luts + otp_enc_key_lut = '0; + otp_dec_key_lut = '0; + digest_const_lut = '0; + digest_iv_lut = '0; + + for (int k = 0; k < NumScrmblKeys; k++) begin + localparam logic [4:0] NumRounds = 5'(unsigned'(NumPresentRounds)); + otp_enc_key_lut[k] = rnd_cnst_key_anchor[k]; + // Due to the PRESENT key schedule, we have to step the key schedule function by + // NumPresentRounds forwards to get the decryption key. + otp_dec_key_lut[k] = + prim_cipher_pkg::present_get_dec_key128(rnd_cnst_key_anchor[k], NumRounds); + end + + for (int k = 0; k < NumDigestSets; k++) begin + digest_const_lut[k] = rnd_cnst_digest_anchor[k]; + digest_iv_lut[k] = rnd_cnst_digest_iv_anchor[k]; + end + end + `ASSERT_KNOWN(EncKeyLutKnown_A, otp_enc_key_lut) + `ASSERT_KNOWN(DecKeyLutKnown_A, otp_dec_key_lut) + `ASSERT_KNOWN(DigestConstLutKnown_A, digest_const_lut) + `ASSERT_KNOWN(DigestIvLutKnown_A, digest_iv_lut) + + ////////////// + // Datapath // + ////////////// + + logic [4:0] idx_state_d, idx_state_q; + logic [ScrmblKeyWidth-1:0] key_state_d, key_state_q; + logic [ScrmblBlockWidth-1:0] data_state_d, data_state_q, data_shadow_q; + logic [ScrmblBlockWidth-1:0] digest_state_d, digest_state_q; + logic [ScrmblBlockWidth-1:0] enc_data_out, enc_data_out_xor, dec_data_out; + logic [ScrmblKeyWidth-1:0] dec_key_out, enc_key_out; + logic [4:0] dec_idx_out, enc_idx_out; + logic [ScrmblKeyWidth-1:0] otp_digest_const_mux, otp_enc_key_mux, otp_dec_key_mux; + logic [ScrmblBlockWidth-1:0] otp_digest_iv_mux; + + typedef enum logic [2:0] {SelEncDataOut, + SelDecDataOut, + SelDigestState, + SelEncDataOutXor, + SelDataInput} data_state_sel_e; + + typedef enum logic [2:0] {SelDecKeyOut, + SelEncKeyOut, + SelDecKeyInit, + SelEncKeyInit, + SelDigestConst, + SelDigestInput, + SelDigestChained} key_state_sel_e; + + logic digest_init; + data_state_sel_e data_state_sel; + key_state_sel_e key_state_sel; + logic data_state_en, data_shadow_copy, data_shadow_load, digest_state_en, key_state_en; + digest_mode_e digest_mode_d, digest_mode_q; + + assign otp_enc_key_mux = otp_enc_key_lut[ScrmblKeySelWidth'(sel_i)]; + assign otp_dec_key_mux = otp_dec_key_lut[ScrmblKeySelWidth'(sel_i)]; + assign otp_digest_const_mux = digest_const_lut[DigestSetSelWidth'(sel_i)]; + assign otp_digest_iv_mux = digest_iv_lut[DigestSetSelWidth'(sel_i)]; + + // Make sure we always select a valid key / digest constant. + `ASSERT(CheckNumEncKeys_A, key_state_sel == SelEncKeyInit |-> sel_i < NumScrmblKeys) + `ASSERT(CheckNumDecKeys_A, key_state_sel == SelDecKeyInit |-> sel_i < NumScrmblKeys) + `ASSERT(CheckNumDigest1_A, key_state_sel == SelDigestConst |-> sel_i < NumDigestSets) + + assign data_state_d = (data_state_sel == SelEncDataOut) ? enc_data_out : + (data_state_sel == SelDecDataOut) ? dec_data_out : + (data_state_sel == SelDigestState) ? digest_state_q : + (data_state_sel == SelEncDataOutXor) ? enc_data_out_xor : + data_i; + + assign key_state_d = (key_state_sel == SelDecKeyOut) ? dec_key_out : + (key_state_sel == SelEncKeyOut) ? enc_key_out : + (key_state_sel == SelDecKeyInit) ? otp_dec_key_mux : + (key_state_sel == SelEncKeyInit) ? otp_enc_key_mux : + (key_state_sel == SelDigestConst) ? otp_digest_const_mux : + (key_state_sel == SelDigestChained) ? {data_state_q, data_shadow_q} : + {data_i, data_shadow_q}; + + // Initialize the round index state with 1 in all cases, except for the decrypt operation. + assign idx_state_d = (key_state_sel == SelDecKeyOut) ? dec_idx_out : + (key_state_sel == SelEncKeyOut) ? enc_idx_out : + (key_state_sel == SelDecKeyInit) ? unsigned'(5'(NumPresentRounds)) : + 5'd1; + + // The XOR is for the Davies-Mayer one-way function construction. + assign enc_data_out_xor = enc_data_out ^ digest_state_q; + assign digest_state_d = (digest_init) ? otp_digest_iv_mux : enc_data_out_xor; + + logic valid_q; //valid_d defined below + assign data_o = (valid_q) ? data_state_q : 0; + + ///////// + // FSM // + ///////// + + // SEC_CM: SCRMBL.FSM.SPARSE + // Encoding generated with: + // $ ./util/design/sparse-fsm-encode.py -d 5 -m 5 -n 9 \ + // -s 2193087944 --language=sv + // + // Hamming distance histogram: + // + // 0: -- + // 1: -- + // 2: -- + // 3: -- + // 4: -- + // 5: |||||||||||||||||||| (60.00%) + // 6: ||||||||||||| (40.00%) + // 7: -- + // 8: -- + // 9: -- + // + // Minimum Hamming distance: 5 + // Maximum Hamming distance: 6 + // Minimum Hamming weight: 4 + // Maximum Hamming weight: 7 + // + localparam int StateWidth = 9; + typedef enum logic [StateWidth-1:0] { + IdleSt = 9'b100011001, + DecryptSt = 9'b101101111, + EncryptSt = 9'b010010111, + DigestSt = 9'b111000010, + ErrorSt = 9'b011111000 + } state_e; + + localparam int CntWidth = $clog2(NumPresentRounds+1); + localparam int unsigned LastPresentRoundInt = NumPresentRounds - 1; + localparam bit [CntWidth-1:0] LastPresentRound = LastPresentRoundInt[CntWidth-1:0]; + + state_e state_d, state_q; + logic [CntWidth-1:0] cnt; + logic cnt_clr, cnt_en, cnt_err; + logic valid_d; //valid_q defined above + + assign valid_o = valid_q; + + // SEC_CM: SCRMBL.CTR.REDUN + prim_count #( + .Width(CntWidth) + ) u_prim_count ( + .clk_i, + .rst_ni, + .clr_i(cnt_clr), + .set_i(1'b0), + .set_cnt_i('0), + .incr_en_i(cnt_en), + .decr_en_i(1'b0), + .step_i(CntWidth'(1)), + .commit_i(1'b1), + .cnt_o(cnt), + .cnt_after_commit_o(), + .err_o(cnt_err) + ); + + always_comb begin : p_fsm + state_d = state_q; + digest_mode_d = digest_mode_q; + data_state_sel = SelDataInput; + key_state_sel = SelDigestInput; + digest_init = 1'b0; + data_state_en = 1'b0; + data_shadow_copy = 1'b0; + data_shadow_load = 1'b0; + key_state_en = 1'b0; + digest_state_en = 1'b0; + cnt_en = 1'b0; + cnt_clr = 1'b0; + valid_d = 1'b0; + ready_o = 1'b0; + fsm_err_o = 1'b0; + + unique case (state_q) + /////////////////////////////////////////////////////////////////// + // Idle State: decode command and + // load working regs accordingly + IdleSt: begin + cnt_clr = 1'b1; + ready_o = 1'b1; + + if (valid_i) begin + unique case (cmd_i) + Decrypt: begin + state_d = DecryptSt; + key_state_sel = SelDecKeyInit; + data_state_en = 1'b1; + key_state_en = 1'b1; + end + Encrypt: begin + state_d = EncryptSt; + key_state_sel = SelEncKeyInit; + data_state_en = 1'b1; + key_state_en = 1'b1; + end + LoadShadow: begin + if (digest_mode_q == ChainedMode) begin + data_shadow_copy = 1'b1; + end else begin + data_shadow_load = 1'b1; + end + end + Digest: begin + state_d = DigestSt; + data_state_sel = SelDigestState; + key_state_sel = (digest_mode_q == ChainedMode) ? SelDigestChained : SelDigestInput; + data_state_en = 1'b1; + key_state_en = 1'b1; + end + DigestInit: begin + digest_mode_d = mode_i; + digest_init = 1'b1; + digest_state_en = 1'b1; + end + DigestFinalize: begin + state_d = DigestSt; + data_state_sel = SelDigestState; + key_state_sel = SelDigestConst; + data_state_en = 1'b1; + key_state_en = 1'b1; + digest_mode_d = StandardMode; + end + default: ; // ignore + endcase // cmd_i + end + end + /////////////////////////////////////////////////////////////////// + // Perform decrypt rounds. + DecryptSt: begin + data_state_sel = SelDecDataOut; + key_state_sel = SelDecKeyOut; + data_state_en = 1'b1; + key_state_en = 1'b1; + cnt_en = 1'b1; + if (cnt == LastPresentRound) begin + state_d = IdleSt; + valid_d = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Perform encrypt rounds. + EncryptSt: begin + data_state_sel = SelEncDataOut; + key_state_sel = SelEncKeyOut; + data_state_en = 1'b1; + key_state_en = 1'b1; + cnt_en = 1'b1; + if (cnt == LastPresentRound) begin + state_d = IdleSt; + valid_d = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // The digest is calculated with a Merkle-Damgard construction that + // employs the PRESENT encryption datapath. + DigestSt: begin + data_state_sel = SelEncDataOut; + key_state_sel = SelEncKeyOut; + data_state_en = 1'b1; + key_state_en = 1'b1; + cnt_en = 1'b1; + if (cnt == LastPresentRound) begin + state_d = IdleSt; + valid_d = 1'b1; + // Apply XOR for Davies-Meyer construction. + data_state_sel = SelEncDataOutXor; + // Backup digest state for next round of updates. We can't keep this state in the + // data state register as a digest may be calculated together with encryption + // operations in an interleaved way. + digest_state_en = 1'b1; + end + end + /////////////////////////////////////////////////////////////////// + // Terminal error state. This raises an alert. + ErrorSt: begin + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + // This should never happen, hence we directly jump into the + // error state, where an alert will be triggered. + default: begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + /////////////////////////////////////////////////////////////////// + endcase // state_q + + // Unconditionally jump into the terminal error state in case of escalation. + // SEC_CM: SCRMBL.FSM.LOCAL_ESC, SCRMBL.FSM.GLOBAL_ESC + if (lc_ctrl_pkg::lc_tx_test_true_loose(escalate_en_i) || cnt_err) begin + state_d = ErrorSt; + fsm_err_o = 1'b1; + end + end + + ///////////////////////////// + // PRESENT DEC/ENC Modules // + ///////////////////////////// + + prim_present #( + .KeyWidth(128), + .NumRounds(NumPresentRounds), + .NumPhysRounds(1) + ) u_prim_present_enc ( + .data_i ( data_state_q ), + .key_i ( key_state_q ), + .idx_i ( idx_state_q ), + .data_o ( enc_data_out ), + .key_o ( enc_key_out ), + .idx_o ( enc_idx_out ) + ); + + prim_present #( + .KeyWidth(128), + // We are using an iterative full-round implementation here. + .NumRounds(NumPresentRounds), + .NumPhysRounds(1), + .Decrypt(1) + ) u_prim_present_dec ( + .data_i ( data_state_q ), + .key_i ( key_state_q ), + .idx_i ( idx_state_q ), + .data_o ( dec_data_out ), + .key_o ( dec_key_out ), + .idx_o ( dec_idx_out ) + ); + + /////////////// + // Registers // + /////////////// + + `PRIM_FLOP_SPARSE_FSM(u_state_regs, state_d, state_q, state_e, IdleSt) + + always_ff @(posedge clk_i or negedge rst_ni) begin : p_regs + if (!rst_ni) begin + key_state_q <= '0; + idx_state_q <= '0; + data_state_q <= '0; + data_shadow_q <= '0; + digest_state_q <= '0; + valid_q <= 1'b0; + digest_mode_q <= StandardMode; + end else begin + valid_q <= valid_d; + digest_mode_q <= digest_mode_d; + + // enable regs + if (key_state_en) begin + key_state_q <= key_state_d; + idx_state_q <= idx_state_d; + end + if (data_state_en) begin + data_state_q <= data_state_d; + end + if (data_shadow_copy) begin + data_shadow_q <= data_state_q; + end else if (data_shadow_load) begin + data_shadow_q <= data_state_d; + end + if (digest_state_en) begin + digest_state_q <= digest_state_d; + end + end + end + +endmodule : otp_ctrl_scrmbl diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_token_const.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_token_const.sv new file mode 100644 index 00000000000000..2443cd8a11d61a --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_token_const.sv @@ -0,0 +1,65 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 + +// This module contains the hash post-image constants for the all-zero and raw unlock tokens. +// This implementation relies on constant propagation to precompute these constants from the +// random netlist constants at compile time, and hence does not contain any "real" logic. + +module otp_ctrl_token_const import otp_ctrl_pkg::*; #( + // Compile time random constants, to be overriden by topgen. + parameter digest_const_array_t RndCnstDigestConst = RndCnstDigestConstDefault, + parameter digest_iv_array_t RndCnstDigestIV = RndCnstDigestIVDefault, + parameter lc_ctrl_pkg::lc_token_t RndCnstRawUnlockToken = RndCnstRawUnlockTokenDefault +) ( + output lc_ctrl_pkg::lc_token_t all_zero_token_hashed_o, + output lc_ctrl_pkg::lc_token_t raw_unlock_token_hashed_o +); + + localparam int NumHashes = 2; + localparam int AllZeroIdx = 0; + localparam int RawUnlockIdx = 1; + + logic [NumHashes-1:0][1:0][ScrmblKeyWidth-1:0] data; + logic [NumHashes-1:0][4:0][ScrmblBlockWidth-1:0] state; + + // First digest is for the all zero token, the second is for the raw unlock token. + assign data[AllZeroIdx][0] = '0; + assign data[RawUnlockIdx][0] = RndCnstRawUnlockToken; + + // Repeat for all precomputed hashes. + for (genvar j = 0; j < NumHashes; j++) begin : gen_hashes + // Initialize all hashes with digest IV. + assign state[j][0] = RndCnstDigestIV[LcRawDigest]; + // Second data block is always the digest finalization constant. + assign data[j][1] = RndCnstDigestConst[LcRawDigest]; + + // Each hash takes four invocations, see diagram c) on + // https://docs.opentitan.org/hw/ip/otp_ctrl/doc/theory_of_operation.html#scrambling-datapath + for (genvar k = 0; k < 4; k++) begin : gen_invocations + logic [ScrmblBlockWidth-1:0] next_state; + + // This relies on constant propagation to + // statically precompute the hashed token values. + prim_present #( + .KeyWidth(128), + .NumRounds(NumPresentRounds) + ) u_prim_present_enc_0 ( + .data_i ( state[j][k] ), + .key_i ( data[j][k%2] ), + .idx_i ( 5'h1 ), + .data_o ( next_state ), + .key_o ( ), + .idx_o ( ) + ); + + // XOR in last state according to the Davies-Meyer scheme. + assign state[j][k+1] = next_state ^ state[j][k]; + end + end + + // Concatenate the two 64bit hash results to form the final digests. + assign all_zero_token_hashed_o = {state[AllZeroIdx][4], state[AllZeroIdx][2]}; + assign raw_unlock_token_hashed_o = {state[RawUnlockIdx][4], state[RawUnlockIdx][2]}; + +endmodule : otp_ctrl_token_const diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv new file mode 100644 index 00000000000000..e9c41a99d920a2 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/rtl/otp_ctrl_top_specific_pkg.sv @@ -0,0 +1,108 @@ +// Copyright lowRISC contributors (OpenTitan project). +// Licensed under the Apache License, Version 2.0, see LICENSE for details. +// SPDX-License-Identifier: Apache-2.0 +// + +// This package contains top specific and implementation items. +// +// It imports otp_ctrl_reg_pkg, which is generated from the top specific hjson file. +// The items that are generic and used by top-independent IPs are placed in otp_ctrl_pkg. +package otp_ctrl_top_specific_pkg; + + import prim_util_pkg::vbits; + import otp_ctrl_pkg::*; + import otp_ctrl_reg_pkg::*; + + //////////////////////// + // General Parameters // + //////////////////////// + + // Width of entropy input + parameter int EdnDataWidth = 64; + + parameter int NumPartWidth = vbits(NumPart); + + parameter int SwWindowAddrWidth = vbits(NumSwCfgWindowWords); + + // Background check timer LFSR width. + parameter int LfsrWidth = 40; + // The LFSR will be reseeded once LfsrUsageThreshold + // values have been drawn from it. + parameter int LfsrUsageThreshold = 16; + + // Redundantly encoded and complementary values are used to for signalling to the partition + // controller FSMs and the DAI whether a partition is locked or not. Any other value than + // "Mubi8Lo" is interpreted as "Locked" in those FSMs. + typedef struct packed { + prim_mubi_pkg::mubi8_t read_lock; + prim_mubi_pkg::mubi8_t write_lock; + } part_access_t; + + parameter int DaiCmdWidth = 3; + typedef enum logic [DaiCmdWidth-1:0] { + DaiRead = 3'b001, + DaiWrite = 3'b010, + DaiDigest = 3'b100 + } dai_cmd_e; + + ////////////////////////////////////// + // Typedefs for OTP Macro Interface // + ////////////////////////////////////// + + // OTP-macro specific + parameter int OtpWidth = 16; + parameter int OtpAddrWidth = OtpByteAddrWidth - $clog2(OtpWidth/8); + parameter int OtpDepth = 2**OtpAddrWidth; + parameter int OtpSizeWidth = 2; // Allows to transfer up to 4 native OTP words at once. + parameter int OtpErrWidth = 3; + parameter int OtpIfWidth = 2**OtpSizeWidth*OtpWidth; + // Number of Byte address bits to cut off in order to get the native OTP word address. + parameter int OtpAddrShift = OtpByteAddrWidth - OtpAddrWidth; + + typedef enum logic [OtpErrWidth-1:0] { + NoError = 3'h0, + MacroError = 3'h1, + MacroEccCorrError = 3'h2, + MacroEccUncorrError = 3'h3, + MacroWriteBlankError = 3'h4, + AccessError = 3'h5, + CheckFailError = 3'h6, + FsmStateError = 3'h7 + } otp_err_e; + + ///////////////////////////////// + // Typedefs for OTP Scrambling // + ///////////////////////////////// + + parameter int NumPresentRounds = 31; + parameter int ScrmblBlockHalfWords = ScrmblBlockWidth / OtpWidth; + + typedef enum logic [2:0] { + Decrypt, + Encrypt, + LoadShadow, + Digest, + DigestInit, + DigestFinalize + } otp_scrmbl_cmd_e; + + /////////////////////////////////////////// + // Defaults for random netlist constants // + /////////////////////////////////////////// + + // These LFSR parameters have been generated with + // $ util/design/gen-lfsr-seed.py --width 40 --seed 4247488366 + typedef logic [LfsrWidth-1:0] lfsr_seed_t; + typedef logic [LfsrWidth-1:0][$clog2(LfsrWidth)-1:0] lfsr_perm_t; + localparam lfsr_seed_t RndCnstLfsrSeedDefault = 40'h453d28ea98; + localparam lfsr_perm_t RndCnstLfsrPermDefault = + 240'h4235171482c225f79289b32181a0163a760355d3447063d16661e44c12a5; + + typedef struct packed { + sram_key_t key; + sram_nonce_t nonce; + } scrmbl_key_init_t; + localparam scrmbl_key_init_t RndCnstScrmblKeyInitDefault = + 256'hcebeb96ffe0eced795f8b2cfe23c1e519e4fa08047a6bcfb811b04f0a479006e; + +endpackage : otp_ctrl_top_specific_pkg diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/syn/constraints.sdc b/hw/top_earlgrey/ip_autogen/otp_ctrl/syn/constraints.sdc new file mode 100644 index 00000000000000..0b119da9708326 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/syn/constraints.sdc @@ -0,0 +1,50 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# Generic constraints file for simple testsynthesis flow + +# note that we do not fix hold timing in this flow +set SETUP_CLOCK_UNCERTAINTY 0.5 + +##################### +# main clock # +##################### +set MAIN_CLK_PIN clk_i +set MAIN_RST_PIN rst_ni +# set main clock to 125 MHz +set MAIN_TCK 8.0 +set_ideal_network ${MAIN_CLK_PIN} +set_ideal_network ${MAIN_RST_PIN} +set_clock_uncertainty ${SETUP_CLOCK_UNCERTAINTY} ${MAIN_CLK_PIN} + +# other timing constraint in ns +set IN_DEL 1.0 +set OUT_DEL 1.0 +set DELAY ${MAIN_TCK} + +create_clock ${MAIN_CLK_PIN} -period ${MAIN_TCK} + +# in to out +set_max_delay ${DELAY} -from [all_inputs] -to [all_outputs] +# in to reg / reg to out +set_input_delay ${IN_DEL} [remove_from_collection [all_inputs] {${MAIN_CLK_PIN}}] -clock ${MAIN_CLK_PIN} +set_output_delay ${OUT_DEL} [all_outputs] -clock ${MAIN_CLK_PIN} + +##################### +# I/O drive/load # +##################### + +# attach load and drivers to IOs to get a more realistic estimate +set_driving_cell -no_design_rule -lib_cell ${DRIVING_CELL} -pin ${DRIVING_CELL_PIN} [all_inputs] +set_load [load_of ${LOAD_CELL_LIB}/${LOAD_CELL}/${LOAD_CELL_PIN}] [all_outputs] + +# set a nonzero critical range to be able to spot the violating paths better +# in the report +set_critical_range 0.5 ${DUT} + +##################### +# Size Only Cells # +##################### + +set_size_only -all_instances [get_cells -h *u_size_only*] true diff --git a/hw/ip/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson b/hw/top_earlgrey/ip_autogen/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson similarity index 81% rename from hw/ip/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson rename to hw/top_earlgrey/ip_autogen/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson index 467e4b6914d991..30b0f88b423ea4 100644 --- a/hw/ip/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson @@ -6,7 +6,7 @@ name: otp_ctrl // Fusesoc core file used for building the file list. - fusesoc_core: lowrisc:ip:{name}:0.1 + fusesoc_core: lowrisc:opentitan:top_earlgrey_{name}:0.1 import_cfgs: [// Project wide common GTECH synthesis config file "{proj_root}/hw/syn/tools/dvsim/common_gtech_syn_cfg.hjson"] @@ -18,7 +18,7 @@ } { // Deletes black-boxed hierarchies before writing out the unmapped netlist name: post_elab_script - value: "{proj_root}/hw/ip/{name}/syn/post_elab_gtech.tcl" + value: "{proj_root}/hw/top_earlgrey/ip_autogen/{name}/syn/post_elab_gtech.tcl" } ] } diff --git a/hw/ip/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson b/hw/top_earlgrey/ip_autogen/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson similarity index 81% rename from hw/ip/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson rename to hw/top_earlgrey/ip_autogen/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson index 62f97833914264..dc4ca5ca46db2d 100644 --- a/hw/ip/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson @@ -6,7 +6,7 @@ name: otp_ctrl // Fusesoc core file used for building the file list. - fusesoc_core: lowrisc:ip:{name}:0.1 + fusesoc_core: lowrisc:opentitan:top_earlgrey_{name}:0.1 import_cfgs: [// Project wide common synthesis config file "{proj_root}/hw/syn/tools/dvsim/common_syn_cfg.hjson"] @@ -19,7 +19,7 @@ ] // Timing constraints for this module - sdc_file: "{proj_root}/hw/ip/{name}/syn/constraints.sdc" + sdc_file: "{proj_root}/hw/top_earlgrey/ip_autogen/{name}/syn/constraints.sdc" // This is not needed for this module foundry_sdc_file: "" diff --git a/hw/top_earlgrey/ip_autogen/otp_ctrl/syn/post_elab_gtech.tcl b/hw/top_earlgrey/ip_autogen/otp_ctrl/syn/post_elab_gtech.tcl new file mode 100644 index 00000000000000..abe1a8c181f27e --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/otp_ctrl/syn/post_elab_gtech.tcl @@ -0,0 +1,9 @@ +# Copyright lowRISC contributors (OpenTitan project). +# Licensed under the Apache License, Version 2.0, see LICENSE for details. +# SPDX-License-Identifier: Apache-2.0 +# +# Post elab script, used in GTECH runs to modify the unmapped netlist before +# writing it out. + +# Remove generic views of ram macros +remove_design prim_generic_ram_1p_Width* diff --git a/hw/top_earlgrey/ip_autogen/pwrmgr/doc/theory_of_operation.md b/hw/top_earlgrey/ip_autogen/pwrmgr/doc/theory_of_operation.md index 5f197a894c9e36..c46bc74cd1697f 100644 --- a/hw/top_earlgrey/ip_autogen/pwrmgr/doc/theory_of_operation.md +++ b/hw/top_earlgrey/ip_autogen/pwrmgr/doc/theory_of_operation.md @@ -69,7 +69,7 @@ Instead the system goes into a terminal non-responsive state where a user or hos The fast clock domain FSM (referred to as fast FSM from here on) resets to `Low Power` state and waits for a power-up request from the slow FSM. Once received, the fast FSM releases the life cycle reset stage (see [reset controller](../../rstmgr/README.md) for more details). -This allows the [OTP](../../../../ip/otp_ctrl/README.md) to begin sensing. +This allows the [OTP](../../otp_ctrl/README.md) to begin sensing. Once OTP sensing completes, the life cycle controller is initialized. The initialization of the life cycle controller puts the device into its allowed operating state (see [life cycle controller](../../../../ip/lc_ctrl/README.md) for more details). diff --git a/hw/top_earlgrey/lint/top_earlgrey_dv_lint_cfgs.hjson b/hw/top_earlgrey/lint/top_earlgrey_dv_lint_cfgs.hjson index c18ef11f80c21c..5269918a81930e 100644 --- a/hw/top_earlgrey/lint/top_earlgrey_dv_lint_cfgs.hjson +++ b/hw/top_earlgrey/lint/top_earlgrey_dv_lint_cfgs.hjson @@ -96,9 +96,9 @@ rel_path: "hw/ip/lc_ctrl/dv/lint/{tool}" }, { name: otp_ctrl - fusesoc_core: lowrisc:dv:otp_ctrl_sim + fusesoc_core: lowrisc:opentitan:top_earlgrey_otp_ctrl_sim import_cfgs: ["{proj_root}/hw/lint/tools/dvsim/common_lint_cfg.hjson"] - rel_path: "hw/ip/otp_ctrl/dv/lint/{tool}" + rel_path: "hw/top_earlgrey/ip_autogen/otp_ctrl/dv/lint/{tool}" }, { name: pattgen fusesoc_core: lowrisc:,dv:pattgen_sim, diff --git a/hw/top_earlgrey/lint/top_earlgrey_lint_cfgs.hjson b/hw/top_earlgrey/lint/top_earlgrey_lint_cfgs.hjson index 756393f4b0f1d0..d302596eb659e6 100644 --- a/hw/top_earlgrey/lint/top_earlgrey_lint_cfgs.hjson +++ b/hw/top_earlgrey/lint/top_earlgrey_lint_cfgs.hjson @@ -147,9 +147,9 @@ rel_path: "hw/ip/otbn/lint/{tool}" }, { name: otp_ctrl - fusesoc_core: lowrisc:ip:otp_ctrl + fusesoc_core: lowrisc:opentitan:top_earlgrey_otp_ctrl import_cfgs: ["{proj_root}/hw/lint/tools/dvsim/common_lint_cfg.hjson"] - rel_path: "hw/ip/otp_ctrl/lint/{tool}" + rel_path: "hw/top_earlgrey/ip_autogen/otp_ctrl/lint/{tool}" overrides: [ { name: design_level diff --git a/hw/top_earlgrey/rtl/autogen/top_earlgrey_rnd_cnst_pkg.sv b/hw/top_earlgrey/rtl/autogen/top_earlgrey_rnd_cnst_pkg.sv index 6671bb5c3cd460..7fdcc491dcd02b 100644 --- a/hw/top_earlgrey/rtl/autogen/top_earlgrey_rnd_cnst_pkg.sv +++ b/hw/top_earlgrey/rtl/autogen/top_earlgrey_rnd_cnst_pkg.sv @@ -17,18 +17,18 @@ package top_earlgrey_rnd_cnst_pkg; // otp_ctrl //////////////////////////////////////////// // Compile-time random bits for initial LFSR seed - parameter otp_ctrl_pkg::lfsr_seed_t RndCnstOtpCtrlLfsrSeed = { - 40'h23_D532C0AB + parameter otp_ctrl_top_specific_pkg::lfsr_seed_t RndCnstOtpCtrlLfsrSeed = { + 40'hAA_3F4C7123 }; // Compile-time random permutation for LFSR output - parameter otp_ctrl_pkg::lfsr_perm_t RndCnstOtpCtrlLfsrPerm = { - 240'h35F9_E2744982_8087A351_B19228F0_5A157307_54E0D985_07166090_139642D1 + parameter otp_ctrl_top_specific_pkg::lfsr_perm_t RndCnstOtpCtrlLfsrPerm = { + 240'h3D64_4385D009_78610B29_C6CC1671_D48E460E_40880D99_A4A50625_D565F093 }; // Compile-time random permutation for scrambling key/nonce register reset value - parameter otp_ctrl_pkg::scrmbl_key_init_t RndCnstOtpCtrlScrmblKeyInit = { - 256'h55D70063_277642B5_309A1639_90B966CD_494444C3_BCDF8087_B7FACD65_E9654CD8 + parameter otp_ctrl_top_specific_pkg::scrmbl_key_init_t RndCnstOtpCtrlScrmblKeyInit = { + 256'h86BF1F41_B783B6DB_8C644C47_23CF740F_6A0563C0_C2920F63_7247508B_AB4DC752 }; //////////////////////////////////////////// @@ -36,35 +36,35 @@ package top_earlgrey_rnd_cnst_pkg; //////////////////////////////////////////// // Diversification value used for all invalid life cycle states. parameter lc_ctrl_pkg::lc_keymgr_div_t RndCnstLcCtrlLcKeymgrDivInvalid = { - 128'h5BC66D10_208C4FB5_1B97BBF4_D12C378C + 128'h164D104B_5B0B3D8F_DDB7A0C5_3617A6A3 }; // Diversification value used for the TEST_UNLOCKED* life cycle states. parameter lc_ctrl_pkg::lc_keymgr_div_t RndCnstLcCtrlLcKeymgrDivTestUnlocked = { - 128'hCFD6A5A5_BF3032DB_51FA1EB9_2F6CE10E + 128'h1CB8138A_3BBDAAE5_52050DE2_8C64D4C1 }; // Diversification value used for the DEV life cycle state. parameter lc_ctrl_pkg::lc_keymgr_div_t RndCnstLcCtrlLcKeymgrDivDev = { - 128'h90F9C5C0_6F733DC9_11A321DD_4C0AC240 + 128'h87725A4C_748BE131_7E3E01C2_27894301 }; // Diversification value used for the PROD/PROD_END life cycle states. parameter lc_ctrl_pkg::lc_keymgr_div_t RndCnstLcCtrlLcKeymgrDivProduction = { - 128'h6D467215_DAB3F2B5_4A52E672_8678AF07 + 128'h78DD3C86_9E21D220_A3543A01_30B3BCFE }; // Diversification value used for the RMA life cycle state. parameter lc_ctrl_pkg::lc_keymgr_div_t RndCnstLcCtrlLcKeymgrDivRma = { - 128'h18068E34_4D2EAE4D_73C8FA54_DE8AA4A4 + 128'hC376D4BA_C76AC7F3_2652FF80_CC92DAEF }; // Compile-time random bits used for invalid tokens in the token mux parameter lc_ctrl_pkg::lc_token_mux_t RndCnstLcCtrlInvalidTokens = { - 256'hF258F3B2_A145FD19_50AE55D6_93E57968_117E66DC_634155E0_F8150242_862EB20E, - 256'hEA9A1F75_44716551_512A112D_ED678CE8_24A08E29_C772795A_358EC30F_1BEDD1AF, - 256'h0EC9D9A9_CF36A2B1_30D32AFE_121AF734_F628363A_AA0E93D4_5C423690_ED05BD1E, - 256'h710F22D1_095889FB_D8A7B57F_B1E11DF3_67C1C1ED_E3F35CFC_8F6E72AB_DD73A9EE + 256'h2DDFEBB9_B7267FEF_2CEF1772_418F8629_C29C75AB_A36D93FF_E068F655_FA7990D5, + 256'h897E2D8D_259F101A_8BF1E2B7_46BE84F3_0380655E_E26D62DE_E6AE95AA_FB7A44E2, + 256'h6DF9CBED_BC3D5125_27FE67E8_B2E07D55_510F7DF2_00E93851_95DE4F2F_B208A71E, + 256'h82207A34_ABD439B4_62F7FDB8_FC20E296_0FA1C637_588C08C3_8374903B_58BC3484 }; //////////////////////////////////////////// @@ -72,12 +72,12 @@ package top_earlgrey_rnd_cnst_pkg; //////////////////////////////////////////// // Compile-time random bits for initial LFSR seed parameter alert_handler_pkg::lfsr_seed_t RndCnstAlertHandlerLfsrSeed = { - 32'hFBF8B708 + 32'hDF9B0E9C }; // Compile-time random permutation for LFSR output parameter alert_handler_pkg::lfsr_perm_t RndCnstAlertHandlerLfsrPerm = { - 160'hD07AF5EC_56811520_45096BB1_32D19ECF_EA7E1AE3 + 160'h84E32BE8_825F903F_D19BB633_C7182F3A_BB54B405 }; //////////////////////////////////////////// @@ -85,22 +85,22 @@ package top_earlgrey_rnd_cnst_pkg; //////////////////////////////////////////// // Compile-time random reset value for SRAM scrambling key. parameter otp_ctrl_pkg::sram_key_t RndCnstSramCtrlRetAonSramKey = { - 128'h42692B08_9A6F99B1_A7AC0842_24ADE461 + 128'h9A0FFA95_6AEADB13_BAAA10D2_336E399E }; // Compile-time random reset value for SRAM scrambling nonce. parameter otp_ctrl_pkg::sram_nonce_t RndCnstSramCtrlRetAonSramNonce = { - 128'h2980D48E_6C6493A5_0EEDBAC6_A46B7A8A + 128'h5F1AEB58_C2A1BA65_D13A0FE3_9B01C95A }; // Compile-time random bits for initial LFSR seed parameter sram_ctrl_pkg::lfsr_seed_t RndCnstSramCtrlRetAonLfsrSeed = { - 32'h9B2C7AEC + 32'h626001CC }; // Compile-time random permutation for LFSR output parameter sram_ctrl_pkg::lfsr_perm_t RndCnstSramCtrlRetAonLfsrPerm = { - 160'h791B870B_95B872C3_00AAA2FB_686C7FD1_E284CFD2 + 160'h2E6E654D_CCE40F8A_6FA302D2_28D9F520_50DD7BF2 }; //////////////////////////////////////////// @@ -108,28 +108,28 @@ package top_earlgrey_rnd_cnst_pkg; //////////////////////////////////////////// // Compile-time random bits for default address key parameter flash_ctrl_pkg::flash_key_t RndCnstFlashCtrlAddrKey = { - 128'h11E64066_908ADFD5_918F8C56_3C92184F + 128'h029CB3E6_2CE6FDDC_BA7F6C9D_2519EA1A }; // Compile-time random bits for default data key parameter flash_ctrl_pkg::flash_key_t RndCnstFlashCtrlDataKey = { - 128'h1C76A61E_112578BF_9F469FC4_82FA14A8 + 128'hACC8E192_2AF7B82D_7479CD41_D20282EB }; // Compile-time random bits for default seeds parameter flash_ctrl_pkg::all_seeds_t RndCnstFlashCtrlAllSeeds = { - 256'hF2B75C26_5C9CB3B2_F1C44A72_87AB4C06_18822E92_A084A048_FFE9FB86_140A79D3, - 256'h8055B24B_5F3B3EA7_9BD7F203_D55EABF9_F733D9B0_CC2CA2B3_94F2526E_4430C2AF + 256'h0D61AA54_CC696D54_16AF2545_001DC3DB_CA07B204_875B3A8B_E8207D25_09C0D925, + 256'h7ADDC105_A37BE10E_82F69C5D_067AC23A_86224F25_DAB6226B_F545B7FC_56675745 }; // Compile-time random bits for initial LFSR seed parameter flash_ctrl_pkg::lfsr_seed_t RndCnstFlashCtrlLfsrSeed = { - 32'hB4F5B43B + 32'hFEEC587D }; // Compile-time random permutation for LFSR output parameter flash_ctrl_pkg::lfsr_perm_t RndCnstFlashCtrlLfsrPerm = { - 160'hDA8C1FC4_8584ABA1_E2CB43D2_DCD1C7E5_FBE98182 + 160'hA206DC5F_5621F91F_FAAC3402_F5A41D16_E6E548B9 }; //////////////////////////////////////////// @@ -137,34 +137,34 @@ package top_earlgrey_rnd_cnst_pkg; //////////////////////////////////////////// // Default seed of the PRNG used for register clearing. parameter aes_pkg::clearing_lfsr_seed_t RndCnstAesClearingLfsrSeed = { - 64'hEC27F13E_B602172C + 64'h95E25E95_D30F4592 }; // Permutation applied to the LFSR of the PRNG used for clearing. parameter aes_pkg::clearing_lfsr_perm_t RndCnstAesClearingLfsrPerm = { - 128'h352B301E_82238B40_C93BBF82_AD9C4B5F, - 256'hD798CAA9_06F1D455_52D9E17F_F105E60D_CF5B86D0_73398AA7_66BA075B_AF300953 + 128'h153FE9B4_1C646AA6_607C0D73_BA69D044, + 256'h22221256_1730E843_6F43D27B_2F87546E_4AB23F77_2D652BF3_AA30BCD2_78736EDD }; // Permutation applied to the clearing PRNG output for clearing the second share of registers. parameter aes_pkg::clearing_lfsr_perm_t RndCnstAesClearingSharePerm = { - 128'h347B55E5_DFA46903_140E8241_DB2FC0F6, - 256'h7BCB7A38_520972B0_1383C5D6_B3A5D16B_A07C69B3_C6CDDB66_262AA512_1C4FBF3A + 128'h97226882_BFBD6420_F834A7B7_7509E96F, + 256'hAC140588_DCD84D57_6B42EEB3_94CE3C58_46CD2215_02DEE2BF_C44C7C75_B1AA4FE6 }; // Default seed of the PRNG used for masking. parameter aes_pkg::masking_lfsr_seed_t RndCnstAesMaskingLfsrSeed = { - 32'h80499E6F, - 256'hCCE64CEA_FD282C0E_33FD2C07_986C2A51_1755F072_EE7A8E2F_C45E111D_F07FA067 + 32'h784A0C43, + 256'h8AB14DE7_812B74CC_981F89B9_B95175A8_F02EA42F_7883754A_0EAA875B_D05EE93E }; // Permutation applied to the output of the PRNG used for masking. parameter aes_pkg::masking_lfsr_perm_t RndCnstAesMaskingLfsrPerm = { - 256'h5227992E_22693A16_3F8B5D8D_58914A1E_3283106A_866C4C2D_760E4978_31847C29, - 256'h67377924_0A710D95_81424147_06663948_8702639F_7A144F17_8009209D_0834447B, - 256'h7D612C33_2A0C6F5B_901C007F_890B1B0F_35180357_3C151F6D_9E3E6082_774D2819, - 256'h8A74941A_1D73729A_53884E68_65114623_459B8F50_3B07923D_387E596B_266E9796, - 256'h8E4B628C_702B2101_5C2F5104_13125A9C_935F5455_645E9805_25367543_85305640 + 256'h3C314F0B_95181072_006D3B3A_12779082_73110A74_912C5B87_9D395D27_42045999, + 256'h011B3E16_5C832163_97815E94_5A498409_257C3D22_9C8A8537_07081D60_9F266F0D, + 256'h5F207080_24301E0C_23922F66_7E781945_4C9A5132_754E3567_4D93712D_4B474A89, + 256'h627A367F_69964656_8C505379_2A06543F_1A1C9B98_15611368_8E1F4333_6A38038D, + 256'h287B9E40_2B574402_41052E65_52640E0F_5588176B_7D6E8B6C_8F483414_76862958 }; //////////////////////////////////////////// @@ -172,58 +172,58 @@ package top_earlgrey_rnd_cnst_pkg; //////////////////////////////////////////// // Compile-time random data for PRNG default seed parameter kmac_pkg::lfsr_seed_t RndCnstKmacLfsrSeed = { - 32'hB666A5E2, - 256'h2D3320F8_9CAA47E5_CED5D29C_C9CDB468_773EBACE_C14DA34C_505AEDC7_0EA184C7 + 32'h534D7AC4, + 256'hBD0CA8B0_3546A981_88556C8C_BF9B24D1_D32C9BD8_6DFED26F_1FE72FFB_B4B9762D }; // Compile-time random permutation for PRNG output parameter kmac_pkg::lfsr_perm_t RndCnstKmacLfsrPerm = { - 64'h31814580_84094D29, - 256'h114AAF9E_6620F99A_DEC77C9B_A71AC27A_C3B24C36_9A2380EE_E7AD8789_9D20E92F, - 256'h178BA48C_C15C4AD2_88EE8389_C0C94D23_069974B6_781CA794_7D5580D1_56EF1F26, - 256'h87C2918B_C2BB2852_789C6C2A_B686D75A_79D9A0CD_C32DD448_A504159B_466ED4AC, - 256'hED492AC3_430B878B_B97D2D65_4F75D289_7309911D_D65A319B_345F9DCE_184636BA, - 256'h4889E16F_4B19734C_388B5906_D0C25F0F_F1C88DBC_D5C09F09_A8E90A87_1E06312C, - 256'h0D451FC1_1AA10DA1_5F4BC6C1_9FF06800_801923D2_CCA0770E_7B9ECE10_00AB1844, - 256'h55451678_BF2466F1_E609995C_90B4C542_4AEC7C2A_6C17E054_9F8C4570_497A7F23, - 256'h20CDE16B_D5812D31_081779B4_95A6AC71_A2E2C044_95384E7D_9064394B_808500FD, - 256'h5E944E99_0416C628_F2619A16_5B67B515_A66FF91A_A6659B93_5DD19CEA_F9E9432C, - 256'h4ABAFC09_B22E0681_781B27D6_A64B9D31_3078BD9D_62C401BA_5021D19E_07C30836, - 256'hA1AA46AE_B49E643B_1613C672_C5406C3E_3E3CF00C_6AD42E6B_19788B0F_8A1B0192, - 256'h1F8D1198_2E3260C3_217BC09D_58DCDDA5_C81332D9_6F0BE2D5_18631D09_1C6A8E62, - 256'h88A57D62_5CB268DA_4A1251EA_CB999E73_DA7B0D07_6216131B_6176FE58_5AD6C36D, - 256'h204EAAB5_8CE44DD3_3908EF3C_9C203801_180002BD_D829E403_8A594083_06051018, - 256'h5F99E539_7C59DA1D_34BE8D67_A11B64DA_B73CC062_EEA47404_95AECBF2_E1A2ED15, - 256'h92609270_84130447_1118D408_6016A074_B73A0488_52382B0E_00BEA459_C6558A08, - 256'h0DC6B4D8_246EEF87_34B41C1F_DA111229_88668C71_3EDC751E_CF862E63_8EADA39A, - 256'h11368A10_6DD6725B_B72D34AA_BA8960AA_08A4806B_7B91C513_E963610C_356BD5A6, - 256'h10B7E702_284AC5C2_8C1C1360_80F70ED2_8BAB9985_5721CC45_D9D38711_F28DADA0, - 256'h4FE7201C_0C5DBAD6_438F05DC_368C6964_C7B9B224_950884CA_2A88F1D4_0B4FDB15, - 256'hC31669E2_22C0FC40_55D8D293_1609489D_99188F03_EAB2970B_C0E58E8A_81F7DC29, - 256'h7EAF0B58_A5880E66_19796CE7_CBD8DE84_AD07907B_A56F5240_6F94CFDC_7108AE2F, - 256'hEB8D9FC6_DD7C3821_25B1DACE_7EA956EB_78DF7EDF_5869420D_5488E929_B32BB82C, - 256'h034F0463_1D44815B_E4299424_C1942D4C_351AE4EC_B2A80022_A20FC531_122680A6, - 256'h1265989D_5FCF814D_B56346E0_41F20F50_7B610FBE_6A789263_BF55A544_36366541, - 256'h713E0998_39B1F845_9419151E_645A4152_F4938058_FE4D87F1_08AED752_5DE7A530, - 256'h57DB799D_DD1D1CE8_EDF344C0_7C145A11_6A933C22_56EF20EC_1B6E1A2A_197395C5, - 256'hB610B8AC_29498A9B_10A5A0F0_82616E5B_A2BC711D_A92ADA45_15918DEF_3AE040B2, - 256'h16B55EAB_118F6F4C_DA766E4B_9FE95692_485547FE_B44F5C19_48F478C7_88CC8E86, - 256'h6A2B169F_62903B09_5381ADB0_5EF74C82_3A214633_9983AAAA_83615549_7245CE59, - 256'h207353CF_51326915_81A9D80C_594A99A8_7725C437_02304D6A_EBED385B_67193E8D + 64'h3F69D7B0_444F62FB, + 256'h85E1BBCC_E9A23D25_29FC2CA6_57AA713B_124DA452_1A14C7D2_97617534_41488B05, + 256'h85D2C74C_7F2C23C8_CC1179EC_9B714548_9BB894F5_C7A124A9_73566537_DC7C3CDD, + 256'hBB06B6B1_C3847142_16DF2AFD_F0211C39_B9824695_C3E8D792_1A37A8B7_87506C10, + 256'hF8399465_BD559956_913590C4_C3100C18_9C5A7A38_E4AA87E2_D56DA015_AA9037AD, + 256'h08B54D84_7EC2041C_3E73A368_02CB6B6C_D7B4450C_14C312C8_34AF80A9_66150CFA, + 256'hB0B20341_97A16E92_6E8835A5_8B53F751_203CAFAA_DC852C69_30088D78_0FB86CD3, + 256'h096EB4BD_0C8A232A_408304CE_187C0753_ADA40A9C_6012D4CE_3B174A81_6ACF1182, + 256'h17DA2080_9A60F579_559442ED_962FD8CB_971659CA_7C25CBA9_ABC548C6_4DE76106, + 256'h39AE68C0_E46E0156_AA35A923_C084A23E_90962714_3A076097_F5F1A203_2231ABB6, + 256'hC800984B_92C1A0CC_A0BC6A53_F8EB45E4_31601536_D710D475_78D87816_A1B5AE7B, + 256'h0C599846_251D34C2_2DF3B1D2_5971B971_9B1A2159_1ADE77DC_7BBA844E_0A576A77, + 256'h44914C04_E24169E8_BA9257D8_B2B0DD86_A555E2A8_142C2306_F2A6E86A_5149C680, + 256'hA0D13BC7_2403A57E_910978D0_BA98AC03_01408059_305B19A0_976400F0_71070332, + 256'h7E222F02_B8DFC740_E42A634C_509A5C78_19702B38_632B4F19_E53C8440_16FDAA43, + 256'h89C13E0A_238A9AAC_9308E593_E5D017EC_4279FF31_E789B5E2_1B6B05DE_8AC0C0B0, + 256'h9E52812B_E4663137_0E0146D7_4981BAA6_4C6E4B68_2ECCBDA7_46D13905_E87BFCA4, + 256'h5C83BBCD_884C1D19_80871B02_E2AA7F22_61087B10_AF5A11BD_B5906663_6918205B, + 256'h22757A9B_C9555237_724CB356_7C026729_470E9C1F_4C603C28_557898CD_5C26C0BC, + 256'h297F1952_CD6BC344_063DF9BB_5CC6E028_3D2D5B1A_D044C0CA_589F9CD8_D7642B86, + 256'h135822EA_A890F7A6_98ABC478_58D007CE_FAB93EA8_3000419D_B8E4F5A0_6983D436, + 256'h7A5BAD5F_E19A29B7_70C80CDE_C1A5E91D_C85D2F51_846475C1_28E1153F_D037996C, + 256'hA694B03D_CD63A24A_2D9984D8_1AB4571A_8B9646F4_3096466A_BA940569_29A22468, + 256'hF9A6588C_01C5FC9B_26223359_F846C7B6_0E6B5EEE_06606B41_25F5770F_12472288, + 256'hC88302DA_CCBE61A3_F6487D8A_A4D9D4BD_B292789C_821705E3_F1F3871C_0362D19F, + 256'h46747523_44076B51_1E67DB3B_64E830E2_C39D4824_0AF53EB8_4E81E7AD_EE325B09, + 256'hA9BE52A8_9BF1500D_873495C1_C1CC510D_611289AA_C4ED7752_142CE31C_5CA3B855, + 256'hB4C53492_99C5E19A_0C742D23_AA54ABA8_D97616D2_96911010_2401AEC6_DE644CB0, + 256'hAA15582F_0D962621_352567D4_E5BEDDCB_8A4E0FC7_A6F5BA79_67A791EF_838A36B9, + 256'hDE950009_5D4173E3_52296032_97D6A0B8_12267AC9_D6459FAB_F84C3DD2_B628F675, + 256'h508B9AE9_B912E435_A4683151_DD51BF56_3BDD1F40_6BBB157C_1E71181C_EAB625D6, + 256'hB19C8ADB_16F1B610_9C69EF14_53422258_99249C2B_E5AFA042_C14A0665_15BA4460 }; // Compile-time random data for PRNG buffer default seed parameter kmac_pkg::buffer_lfsr_seed_t RndCnstKmacBufferLfsrSeed = { - 32'hAC290425, - 256'h63AE4733_313AD269_6788A873_5B27982F_5007A24A_125F876C_9CA6DDEB_52AC92F5, - 256'h600D2A41_BC465EFE_57AA3B59_BD6B0A40_21636342_0F041ED2_9A965B68_FAEA3E59, - 256'hD4F2E2A0_B42EB9A8_C3AB227B_50766D2E_AEC89EB1_3B07E9BF_4EE42784_136D42C8 + 32'hB7480D9D, + 256'h5001B775_5DE3EF26_18E28371_DD02B1DB_04469E57_70D10B00_69E332BE_0F35A726, + 256'hB339D953_A1DD95A3_F78E1438_9414ADC1_9300E093_4269B3A3_0B61EA4D_9790DE0F, + 256'hB09D2792_D26BF956_5775A284_3FFB7D42_993C0A1C_3D4D2937_F997A3B5_1AAFEBC6 }; // Compile-time random permutation for LFSR Message output parameter kmac_pkg::msg_perm_t RndCnstKmacMsgPerm = { - 128'h73A93266_A6B6E1FB_1DCE0B54_C02F3E00, - 256'h63EB8BD5_858F5418_A040791C_3C694CD4_8C9CA20B_DE5EEF7A_6BA157FD_0939B457 + 128'h2A89790D_B63EFF5A_927C7D96_C2F0B23B, + 256'hA17CAD7E_DE662023_B24CCCE2_E804A4B6_89B48510_F749A719_C4C1DDD3_D5851C4D }; //////////////////////////////////////////// @@ -231,17 +231,17 @@ package top_earlgrey_rnd_cnst_pkg; //////////////////////////////////////////// // Default seed of the PRNG used for URND. parameter otbn_pkg::urnd_prng_seed_t RndCnstOtbnUrndPrngSeed = { - 256'hA8F072E7_7EE4859B_243F92CE_B098BE6E_65C35948_0858C396_11E658C9_7FB8D3E9 + 256'h4569B54E_4CC3468B_A78749A7_093FE9EB_381742E6_95130A4F_B5238EBB_74B4F311 }; // Compile-time random reset value for IMem/DMem scrambling key. parameter otp_ctrl_pkg::otbn_key_t RndCnstOtbnOtbnKey = { - 128'hCC0E7F8F_A8725200_C726C8F9_0EF5E2B7 + 128'h8274EA0B_356B0515_35EAC4E0_98EA131F }; // Compile-time random reset value for IMem/DMem scrambling nonce. parameter otp_ctrl_pkg::otbn_nonce_t RndCnstOtbnOtbnNonce = { - 64'h4BEEE27C_FA19EA57 + 64'h95C233E6_70A4321B }; //////////////////////////////////////////// @@ -249,73 +249,73 @@ package top_earlgrey_rnd_cnst_pkg; //////////////////////////////////////////// // Compile-time random bits for initial LFSR seed parameter keymgr_pkg::lfsr_seed_t RndCnstKeymgrLfsrSeed = { - 64'h6CA8A822_5C8E1705 + 64'hAD599D4D_F689612A }; // Compile-time random permutation for LFSR output parameter keymgr_pkg::lfsr_perm_t RndCnstKeymgrLfsrPerm = { - 128'hB931FB3B_12EFADCD_AA1008BC_15948906, - 256'h6F5D9C83_21E10C03_E5E61B0A_893E95FD_0654DDFF_CACE97A5_4CD668A8_358ED44F + 128'h961D0532_D78D8ACE_1FF82767_391F0F51, + 256'h5416EBF4_A96B95FB_20BC2E1C_472064AF_75C406EF_AD88D3DF_3903076C_A8266A86 }; // Compile-time random permutation for entropy used in share overriding parameter keymgr_pkg::rand_perm_t RndCnstKeymgrRandPerm = { - 160'hB87FDDE5_8A44756A_1C05F720_93384394_C8D7AF15 + 160'h95CA1A67_1B6B87F5_F2CFD226_A0445E21_BB53C12C }; // Compile-time random bits for revision seed parameter keymgr_pkg::seed_t RndCnstKeymgrRevisionSeed = { - 256'h20BDAF59_FE2AE209_B8325D2C_8C35FC96_0679B106_A87E59E6_AEB1B530_2D334010 + 256'h46794CFC_A2B0C6CB_FB61F1BD_7ED0A8F9_26DA506E_D059049D_C9A6E2C1_826AA5BF }; // Compile-time random bits for creator identity seed parameter keymgr_pkg::seed_t RndCnstKeymgrCreatorIdentitySeed = { - 256'h70251696_FBB4BA16_9A6CD82F_AD46A0A5_C04009BB_934C7EF7_B83B6E40_610B4309 + 256'hDF596FB9_A391C672_2B8B5CAD_051F3EDB_286E5540_01528F8E_9A5E55EB_B82A171C }; // Compile-time random bits for owner intermediate identity seed parameter keymgr_pkg::seed_t RndCnstKeymgrOwnerIntIdentitySeed = { - 256'hFCB9DC54_D4276137_F9901D09_A76AEAA1_9DE5C579_FD38BDF3_8F0C21D6_A52226B6 + 256'hF1E411AD_FE91411B_1D8EE016_1ECD6990_02EF8026_E474E857_39EF0277_414266B4 }; // Compile-time random bits for owner identity seed parameter keymgr_pkg::seed_t RndCnstKeymgrOwnerIdentitySeed = { - 256'hA4A7BDB0_5FE92161_5BF2FF98_4540F7D4_3ECE76B4_EB133637_74ED2ED4_5545E927 + 256'hF811EBD0_1E7EA4AC_F08B65A5_CF51AE52_8473DD46_61D013FB_44105783_0DD1B830 }; // Compile-time random bits for software generation seed parameter keymgr_pkg::seed_t RndCnstKeymgrSoftOutputSeed = { - 256'hF6D9E4AB_AC398D42_C745EEF6_46C1464D_CA86DAFD_7C7C71E6_058DDFD8_71C51CAC + 256'hBF365BBF_C1DEB892_1D7A6896_FD1E6013_622D6614_2EE83239_BA8E92C6_1DE956A5 }; // Compile-time random bits for hardware generation seed parameter keymgr_pkg::seed_t RndCnstKeymgrHardOutputSeed = { - 256'hBAF4410F_06DCC036_FCD16FCD_E97D1718_91105DD8_95E3A1D0_A19A16D6_DBD8B20F + 256'h2768B6D5_F1650F1B_01969710_A957DBFB_2E2443F4_FE2D0A76_0F83C131_3D3EF78F }; // Compile-time random bits for generation seed when aes destination selected parameter keymgr_pkg::seed_t RndCnstKeymgrAesSeed = { - 256'hC9E662E1_E1B4982B_3E8EFF63_890DBEAE_926FD468_A77EFDE3_DE5B4CAF_4776A247 + 256'hEA4D3D48_54921724_8330D048_F78E497E_63710E97_20AC0F46_C68E74CA_B245BE89 }; // Compile-time random bits for generation seed when kmac destination selected parameter keymgr_pkg::seed_t RndCnstKeymgrKmacSeed = { - 256'hBABA4C99_08ED16BC_5415EC16_D28C5355_1312FCED_CF2832A6_6CEACF8E_D4D5B616 + 256'h11081C18_0F034806_61762D84_BFB80313_0EA592F6_3F968D16_D49748CB_C79F398E }; // Compile-time random bits for generation seed when otbn destination selected parameter keymgr_pkg::seed_t RndCnstKeymgrOtbnSeed = { - 256'h177DD43B_56FA754E_1F53AD65_8193B563_D9BDC6D2_AEE84852_F0CC7371_ED3A6FAA + 256'h21395417_31F82F0E_5E0F2EA9_61C615FC_96398950_EDED5E99_9CBB1667_0229D5A2 }; // Compile-time random bits for generation seed when no CDI is selected parameter keymgr_pkg::seed_t RndCnstKeymgrCdi = { - 256'h516C144B_34C96DFA_BB6F6E79_208A74F3_5C79F397_C4E4C7E2_2B758184_8A90A125 + 256'h4DF595BE_24C50547_533E86EF_45DC765B_C3FC1C44_92B8CD26_4903BA93_E2074570 }; // Compile-time random bits for generation seed when no destination selected parameter keymgr_pkg::seed_t RndCnstKeymgrNoneSeed = { - 256'h4B2276BE_C9FC1A7A_5722A9AA_CA4DB84A_32E5C698_5A1A6435_118B5F5F_3FD3B931 + 256'h4FC7F5A2_63F7641C_180B43FF_715AA5A2_BD793053_46EBE883_E11EEA60_C36419A2 }; //////////////////////////////////////////// @@ -323,14 +323,14 @@ package top_earlgrey_rnd_cnst_pkg; //////////////////////////////////////////// // Compile-time random bits for csrng state group diversification value parameter csrng_pkg::cs_keymgr_div_t RndCnstCsrngCsKeymgrDivNonProduction = { - 128'h3116B671_92D856DD_742D40E0_72E417AD, - 256'h9A5F2612_04CBCC69_A2147819_B290A4BB_0264347C_0F91B0CE_C93C0129_D85B9580 + 128'h27F0CBB5_F22278F0_4ABE6941_6755D0A6, + 256'h5694CB70_2F939386_38AC23E2_29502708_C3001F75_853897B4_E8EF27D1_7C5F7FB3 }; // Compile-time random bits for csrng state group diversification value parameter csrng_pkg::cs_keymgr_div_t RndCnstCsrngCsKeymgrDivProduction = { - 128'h1DBE03F1_7E69DC26_0A79998E_FF5C710B, - 256'h67838ADF_B99A5ECA_671672CF_19BCA6AD_C1EBFAB6_11F19E72_62E0A491_970E51B2 + 128'h9AFB3660_BABA1F1B_AC202D76_E627CBB3, + 256'h0AD1BC1E_181DB312_B8D5A947_9C07C534_90303552_EA4B9FBD_E86F64DA_79331682 }; //////////////////////////////////////////// @@ -338,22 +338,22 @@ package top_earlgrey_rnd_cnst_pkg; //////////////////////////////////////////// // Compile-time random reset value for SRAM scrambling key. parameter otp_ctrl_pkg::sram_key_t RndCnstSramCtrlMainSramKey = { - 128'h8E7026D0_E5DCE9E2_65A416A8_12231CFA + 128'h785D8680_F24B3700_DB2A343F_E142FE18 }; // Compile-time random reset value for SRAM scrambling nonce. parameter otp_ctrl_pkg::sram_nonce_t RndCnstSramCtrlMainSramNonce = { - 128'h76E3D982_F8FDCEC9_AE1AF0B8_CD9E1760 + 128'hB7568E2E_1AFA5BA5_CF4337A8_977E31C7 }; // Compile-time random bits for initial LFSR seed parameter sram_ctrl_pkg::lfsr_seed_t RndCnstSramCtrlMainLfsrSeed = { - 32'h6EC98106 + 32'hB94C2BCF }; // Compile-time random permutation for LFSR output parameter sram_ctrl_pkg::lfsr_perm_t RndCnstSramCtrlMainLfsrPerm = { - 160'h85097395_A09DABEE_6DDD67D1_948B1A89_9435BE41 + 160'hACB1D684_5459BD12_28EE4FD0_59E59A85_EDBE0DE0 }; //////////////////////////////////////////// @@ -361,12 +361,12 @@ package top_earlgrey_rnd_cnst_pkg; //////////////////////////////////////////// // Fixed nonce used for address / data scrambling parameter bit [63:0] RndCnstRomCtrlScrNonce = { - 64'hFEE457DE_E82B6E06 + 64'h3F40A5E8_16A506C2 }; // Randomised constant used as a scrambling key for ROM data parameter bit [127:0] RndCnstRomCtrlScrKey = { - 128'h663C2917_39FF0E7D_644758FE_E1C58564 + 128'hAB1F4A2F_01CBE908_A34C2A20_FECDA9A5 }; //////////////////////////////////////////// @@ -374,22 +374,22 @@ package top_earlgrey_rnd_cnst_pkg; //////////////////////////////////////////// // Default seed of the PRNG used for random instructions. parameter ibex_pkg::lfsr_seed_t RndCnstRvCoreIbexLfsrSeed = { - 32'hCF346D96 + 32'hFCEBA680 }; // Permutation applied to the LFSR of the PRNG used for random instructions. parameter ibex_pkg::lfsr_perm_t RndCnstRvCoreIbexLfsrPerm = { - 160'hAE357A7A_3B2CD206_3D7F3AB3_043AC11B_7B211B84 + 160'h791C0627_955E85EC_954138F7_2A5A2834_26DC77F7 }; // Default icache scrambling key parameter logic [ibex_pkg::SCRAMBLE_KEY_W-1:0] RndCnstRvCoreIbexIbexKeyDefault = { - 128'hC652D737_2432FCC4_D3092BD6_383FDBA3 + 128'h19884812_50C2C82D_EB519D69_C7051AAB }; // Default icache scrambling nonce parameter logic [ibex_pkg::SCRAMBLE_NONCE_W-1:0] RndCnstRvCoreIbexIbexNonceDefault = { - 64'h5279FCBC_D2BD2C13 + 64'h9FC45A16_58FA9063 }; endpackage : top_earlgrey_rnd_cnst_pkg diff --git a/hw/top_earlgrey/syn/top_earlgrey_batch_syn_cfg.hjson b/hw/top_earlgrey/syn/top_earlgrey_batch_syn_cfg.hjson index 9a1ce8d611e1b6..d6cb3ebd02167f 100644 --- a/hw/top_earlgrey/syn/top_earlgrey_batch_syn_cfg.hjson +++ b/hw/top_earlgrey/syn/top_earlgrey_batch_syn_cfg.hjson @@ -18,8 +18,8 @@ "{proj_root}/hw/ip/kmac/syn/kmac_syn_cfg.hjson", "{proj_root}/hw/ip/lc_ctrl/syn/lc_ctrl_syn_cfg.hjson", "{proj_root}/hw/ip/otbn/syn/otbn_syn_cfg.hjson", - "{proj_root}/hw/ip/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson", "{proj_root}/hw/ip/rv_core_ibex/syn/rv_core_ibex_syn_cfg.hjson", + "{proj_root}/hw/top_earlgrey/ip_autogen/otp_ctrl/syn/otp_ctrl_syn_cfg.hjson", // Top-level synthesis flows. // TODO: align Verilator and ASIC versions. "{proj_root}/hw/top_earlgrey/syn/chip_earlgrey_asic_syn_cfg.hjson"] diff --git a/hw/top_earlgrey/syn/top_earlgrey_gtech_batch_syn_cfg.hjson b/hw/top_earlgrey/syn/top_earlgrey_gtech_batch_syn_cfg.hjson index 5a594fcbba2b91..ee6e8df7198701 100644 --- a/hw/top_earlgrey/syn/top_earlgrey_gtech_batch_syn_cfg.hjson +++ b/hw/top_earlgrey/syn/top_earlgrey_gtech_batch_syn_cfg.hjson @@ -19,8 +19,8 @@ "{proj_root}/hw/ip/kmac/syn/kmac_gtech_syn_cfg.hjson", "{proj_root}/hw/ip/lc_ctrl/syn/lc_ctrl_gtech_syn_cfg.hjson", "{proj_root}/hw/ip/otbn/syn/otbn_gtech_syn_cfg.hjson", - "{proj_root}/hw/ip/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson", "{proj_root}/hw/ip/rv_core_ibex/syn/rv_core_ibex_gtech_syn_cfg.hjson", + "{proj_root}/hw/top_earlgrey/ip_autogen/otp_ctrl/syn/otp_ctrl_gtech_syn_cfg.hjson", // Top-level GTECH synthesis flows. // TODO: align Verilator and ASIC versions. "{proj_root}/hw/top_earlgrey/syn/top_earlgrey_gtech_syn_cfg.hjson"] diff --git a/hw/top_earlgrey/top_earlgrey.core b/hw/top_earlgrey/top_earlgrey.core index a9d7374cb3c3d3..3d4df3b6a481b7 100644 --- a/hw/top_earlgrey/top_earlgrey.core +++ b/hw/top_earlgrey/top_earlgrey.core @@ -36,7 +36,6 @@ filesets: - lowrisc:constants:top_earlgrey_top_pkg - lowrisc:constants:top_earlgrey_jtag_id_pkg - lowrisc:constants:top_earlgrey_ibex_pmp_reset_pkg - - lowrisc:ip:otp_ctrl - lowrisc:ip:lc_ctrl - lowrisc:ip:usbdev - lowrisc:top_earlgrey:xbar_main @@ -44,10 +43,11 @@ filesets: - lowrisc:opentitan:top_earlgrey_alert_handler - lowrisc:opentitan:top_earlgrey_clkmgr - lowrisc:opentitan:top_earlgrey_flash_ctrl - - lowrisc:opentitan:top_earlgrey_rstmgr - - lowrisc:opentitan:top_earlgrey_rv_plic + - lowrisc:opentitan:top_earlgrey_otp_ctrl - lowrisc:opentitan:top_earlgrey_pinmux - lowrisc:opentitan:top_earlgrey_pwrmgr + - lowrisc:opentitan:top_earlgrey_rstmgr + - lowrisc:opentitan:top_earlgrey_rv_plic - lowrisc:ip:aon_timer - lowrisc:ip:adc_ctrl - lowrisc:ip:sysrst_ctrl diff --git a/hw/top_englishbreakfast/lint/top_englishbreakfast_lint_cfgs.hjson b/hw/top_englishbreakfast/lint/top_englishbreakfast_lint_cfgs.hjson index 28346679cabff7..4aa17e4f546637 100644 --- a/hw/top_englishbreakfast/lint/top_englishbreakfast_lint_cfgs.hjson +++ b/hw/top_englishbreakfast/lint/top_englishbreakfast_lint_cfgs.hjson @@ -30,7 +30,7 @@ ] }, { name: otp_ctrl - fusesoc_core: lowrisc:ip:otp_ctrl + fusesoc_core: lowrisc:opentitan:top_englishbreakfast_otp_ctrl import_cfgs: ["{proj_root}/hw/lint/tools/dvsim/common_lint_cfg.hjson"] rel_path: "hw/ip/otp_ctrl/lint/{tool}" overrides: [ diff --git a/hw/top_englishbreakfast/top_englishbreakfast.core b/hw/top_englishbreakfast/top_englishbreakfast.core index 164aba340cf81a..5c0050bc60edbd 100644 --- a/hw/top_englishbreakfast/top_englishbreakfast.core +++ b/hw/top_englishbreakfast/top_englishbreakfast.core @@ -30,7 +30,6 @@ filesets: - lowrisc:ip:hmac - lowrisc:ip:kmac - lowrisc:ip:keymgr - - lowrisc:ip:otp_ctrl - lowrisc:ip:lc_ctrl - lowrisc:ip:usbdev - lowrisc:top_englishbreakfast:xbar_main diff --git a/release/BUILD b/release/BUILD index 069506ef4e8d76..4919aeb0b9ee83 100644 --- a/release/BUILD +++ b/release/BUILD @@ -14,7 +14,7 @@ pkg_tar( "//hw:package", "//hw/bitstream:package", "//hw/bitstream/vivado:package", - "//hw/ip/otp_ctrl/data:package", + "//hw/top_earlgrey/data/otp:package", "//sw/device/examples/hello_world:package", "//sw/device/lib/testing/test_rom:package", "//sw/device/silicon_creator/rom:package", diff --git a/rules/lc.bzl b/rules/lc.bzl index 3f27e8381fd3bb..10e39887a10e3f 100644 --- a/rules/lc.bzl +++ b/rules/lc.bzl @@ -55,7 +55,7 @@ lc_raw_unlock_token = rule( doc = "Life-cycle state definition file in Hjson format.", ), "lc_seed": attr.label( - default = "//hw/ip/otp_ctrl/data:lc_seed", + default = "//util/design/data:lc_seed", doc = "Configuration override seed used to randomize LC netlist constants.", ), "_tool": attr.label( diff --git a/rules/opentitan/legacy.bzl b/rules/opentitan/legacy.bzl index 6458665d4bec1b..d47f25197dcc13 100644 --- a/rules/opentitan/legacy.bzl +++ b/rules/opentitan/legacy.bzl @@ -97,15 +97,15 @@ scramble_flash_vmem = rv_rule( "otp": attr.label(allow_single_file = True), "otp_mmap": attr.label( allow_single_file = True, - default = "//hw/ip/otp_ctrl/data:otp_ctrl_mmap.hjson", + default = "//hw/top_earlgrey/data/otp:otp_ctrl_mmap.hjson", doc = "OTP memory map configuration HJSON file.", ), "otp_seed": attr.label( - default = "//hw/ip/otp_ctrl/data:otp_seed", + default = "//util/design/data:otp_seed", doc = "Configuration override seed used to randomize OTP netlist constants.", ), "otp_data_perm": attr.label( - default = "//hw/ip/otp_ctrl/data:data_perm", + default = "//util/design/data:data_perm", doc = "Option to indicate OTP VMEM file bit layout.", ), "_tool": attr.label( diff --git a/rules/otp.bzl b/rules/otp.bzl index 4b1663d322afbd..8cda20de09ea2b 100644 --- a/rules/otp.bzl +++ b/rules/otp.bzl @@ -45,35 +45,35 @@ load("//rules:stamp.bzl", "stamp_attr", "stamping_enabled") def get_otp_images(): """Returns a list of (otp_name, img_target) tuples. - Each tuple corresponds to an OTP image defined in //hw/ip/otp_ctrl/data. The + Each tuple corresponds to an OTP image defined in //util/design/data. The otp_name is a short, unique suffix of the image target, e.g. "rma". The img_target is the full path of the OTP image target. """ img_targets = [ - "//hw/ip/otp_ctrl/data:img_dev", - "//hw/ip/otp_ctrl/data:img_rma", - "//hw/ip/otp_ctrl/data:img_test_locked0", - "//hw/ip/otp_ctrl/data:img_test_locked1", - "//hw/ip/otp_ctrl/data:img_test_locked2", - "//hw/ip/otp_ctrl/data:img_test_locked3", - "//hw/ip/otp_ctrl/data:img_test_locked4", - "//hw/ip/otp_ctrl/data:img_test_locked5", - "//hw/ip/otp_ctrl/data:img_test_locked6", - "//hw/ip/otp_ctrl/data:img_test_unlocked0", - "//hw/ip/otp_ctrl/data:img_test_unlocked1", - "//hw/ip/otp_ctrl/data:img_test_unlocked1_initial", - "//hw/ip/otp_ctrl/data:img_test_unlocked2", - "//hw/ip/otp_ctrl/data:img_test_unlocked3", - "//hw/ip/otp_ctrl/data:img_test_unlocked4", - "//hw/ip/otp_ctrl/data:img_test_unlocked5", - "//hw/ip/otp_ctrl/data:img_test_unlocked6", - "//hw/ip/otp_ctrl/data:img_test_unlocked7", - "//hw/ip/otp_ctrl/data:img_prod", - "//hw/ip/otp_ctrl/data:img_prod_end", - "//hw/ip/otp_ctrl/data:img_exec_disabled", - "//hw/ip/otp_ctrl/data:img_bootstrap_disabled", - "//hw/ip/otp_ctrl/data:img_raw", + "//hw/top_earlgrey/data/otp:img_dev", + "//hw/top_earlgrey/data/otp:img_rma", + "//hw/top_earlgrey/data/otp:img_test_locked0", + "//hw/top_earlgrey/data/otp:img_test_locked1", + "//hw/top_earlgrey/data/otp:img_test_locked2", + "//hw/top_earlgrey/data/otp:img_test_locked3", + "//hw/top_earlgrey/data/otp:img_test_locked4", + "//hw/top_earlgrey/data/otp:img_test_locked5", + "//hw/top_earlgrey/data/otp:img_test_locked6", + "//hw/top_earlgrey/data/otp:img_test_unlocked0", + "//hw/top_earlgrey/data/otp:img_test_unlocked1", + "//hw/top_earlgrey/data/otp:img_test_unlocked1_initial", + "//hw/top_earlgrey/data/otp:img_test_unlocked2", + "//hw/top_earlgrey/data/otp:img_test_unlocked3", + "//hw/top_earlgrey/data/otp:img_test_unlocked4", + "//hw/top_earlgrey/data/otp:img_test_unlocked5", + "//hw/top_earlgrey/data/otp:img_test_unlocked6", + "//hw/top_earlgrey/data/otp:img_test_unlocked7", + "//hw/top_earlgrey/data/otp:img_prod", + "//hw/top_earlgrey/data/otp:img_prod_end", + "//hw/top_earlgrey/data/otp:img_exec_disabled", + "//hw/top_earlgrey/data/otp:img_bootstrap_disabled", + "//hw/top_earlgrey/data/otp:img_raw", ] out = [] @@ -277,23 +277,23 @@ otp_image = rule( ), "mmap_def": attr.label( allow_single_file = True, - default = "//hw/ip/otp_ctrl/data:otp_ctrl_mmap.hjson", + default = "//hw/top_earlgrey/data/otp:otp_ctrl_mmap.hjson", doc = "OTP Controller memory map file in Hjson format.", ), "img_seed": attr.label( - default = "//hw/ip/otp_ctrl/data:img_seed", + default = "//util/design/data:img_seed", doc = "Configuration override seed used to randomize field values in an OTP image.", ), "lc_seed": attr.label( - default = "//hw/ip/otp_ctrl/data:lc_seed", + default = "//util/design/data:lc_seed", doc = "Configuration override seed used to randomize LC netlist constants.", ), "otp_seed": attr.label( - default = "//hw/ip/otp_ctrl/data:otp_seed", + default = "//util/design/data:otp_seed", doc = "Configuration override seed used to randomize OTP netlist constants.", ), "data_perm": attr.label( - default = "//hw/ip/otp_ctrl/data:data_perm", + default = "//util/design/data:data_perm", doc = "Post-processing option to trigger permuting bit positions in memfile.", ), "verbose": attr.bool( @@ -357,24 +357,24 @@ otp_image_consts = rule( ), "mmap_def": attr.label( allow_single_file = True, - default = "//hw/ip/otp_ctrl/data:otp_ctrl_mmap.hjson", + default = "//hw/top_earlgrey/data/otp:otp_ctrl_mmap.hjson", doc = "OTP Controller memory map file in Hjson format.", ), "img_seed": attr.label( - default = "//hw/ip/otp_ctrl/data:img_seed", + default = "//util/design/data:img_seed", doc = "Configuration override seed used to randomize field values in an OTP image.", ), "lc_seed": attr.label( - default = "//hw/ip/otp_ctrl/data:lc_seed", + default = "//util/design/data:lc_seed", doc = "Configuration override seed used to randomize LC netlist constants.", ), "otp_seed": attr.label( - default = "//hw/ip/otp_ctrl/data:otp_seed", + default = "//util/design/data:otp_seed", doc = "Configuration override seed used to randomize OTP netlist constants.", ), "c_template": attr.label( allow_single_file = True, - default = "//hw/ip/otp_ctrl/data:otp_ctrl_img.c.tpl", + default = "//util/design/data:otp_ctrl_img.c.tpl", doc = "OTP image header template.", ), "verbose": attr.bool( @@ -400,19 +400,19 @@ OTP_SIGVERIFY_FAKE_KEYS = [ # Additional overlays can be applied on top to further customize the OTP. # This set overlays does not include any of the SECRET[0-2] partitions. STD_OTP_OVERLAYS_WITHOUT_SECRET_PARTITIONS = OTP_SIGVERIFY_FAKE_KEYS + [ - "//hw/ip/otp_ctrl/data:otp_json_creator_sw_cfg", - "//hw/ip/otp_ctrl/data:otp_json_owner_sw_cfg", - "//hw/ip/otp_ctrl/data:otp_json_alert_digest_cfg", - "//hw/ip/otp_ctrl/data:otp_json_hw_cfg0", - "//hw/ip/otp_ctrl/data:otp_json_hw_cfg1", + "//hw/top_earlgrey/data/otp:otp_json_creator_sw_cfg", + "//hw/top_earlgrey/data/otp:otp_json_owner_sw_cfg", + "//hw/top_earlgrey/data/otp:otp_json_alert_digest_cfg", + "//hw/top_earlgrey/data/otp:otp_json_hw_cfg0", + "//hw/top_earlgrey/data/otp:otp_json_hw_cfg1", ] # This is a set of overlays to generate a generic, standard OTP image. # Additional overlays can be applied on top to further customize the OTP. STD_OTP_OVERLAYS = STD_OTP_OVERLAYS_WITHOUT_SECRET_PARTITIONS + [ - "//hw/ip/otp_ctrl/data:otp_json_secret0", - "//hw/ip/otp_ctrl/data:otp_json_secret1", - "//hw/ip/otp_ctrl/data:otp_json_secret2_unlocked", + "//hw/top_earlgrey/data/otp:otp_json_secret0", + "//hw/top_earlgrey/data/otp:otp_json_secret1", + "//hw/top_earlgrey/data/otp:otp_json_secret2_unlocked", ] def otp_hex(v): diff --git a/sw/device/lib/dif/dif_otp_ctrl.c b/sw/device/lib/dif/dif_otp_ctrl.c index e2681a99e6320d..1e546785e6d23d 100644 --- a/sw/device/lib/dif/dif_otp_ctrl.c +++ b/sw/device/lib/dif/dif_otp_ctrl.c @@ -1,9 +1,6 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -// DO NOT EDIT THIS FILE DIRECTLY. -// It has been generated with ./util/design/gen-otp-mmap.py - #include "sw/device/lib/dif/dif_otp_ctrl.h" #include diff --git a/sw/device/lib/dif/dif_otp_ctrl.h b/sw/device/lib/dif/dif_otp_ctrl.h index 51b7b2969b36eb..7894ddf3e4e78b 100644 --- a/sw/device/lib/dif/dif_otp_ctrl.h +++ b/sw/device/lib/dif/dif_otp_ctrl.h @@ -1,16 +1,13 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -// DO NOT EDIT THIS FILE DIRECTLY. -// It has been generated with ./util/design/gen-otp-mmap.py - #ifndef OPENTITAN_SW_DEVICE_LIB_DIF_DIF_OTP_CTRL_H_ #define OPENTITAN_SW_DEVICE_LIB_DIF_DIF_OTP_CTRL_H_ /** * @file - * @brief OTP Controller Device Interface - * Functions + * @brief + * OTP Controller Device Interface Functions */ #include diff --git a/sw/device/lib/dif/dif_otp_ctrl.md b/sw/device/lib/dif/dif_otp_ctrl.md index 4a2b2cdb030231..225f942aa688eb 100644 --- a/sw/device/lib/dif/dif_otp_ctrl.md +++ b/sw/device/lib/dif/dif_otp_ctrl.md @@ -1,6 +1,6 @@ # OTP Controller DIF Checklist -This checklist is for [Development Stage](../../../../doc/project_governance/development_stages.md) transitions for the [OTP Controller DIF](../../../../hw/ip/otp_ctrl/README.md). +This checklist is for [Development Stage](../../../../doc/project_governance/development_stages.md) transitions for the [OTP Controller DIF](../../../../hw/top_earlgrey/ip_autogen/otp_ctrl/README.md). All checklist items refer to the content in the [Checklist](../../../../doc/project_governance/checklist/README.md).

DIF Checklist

diff --git a/sw/device/lib/dif/dif_otp_ctrl_unittest.cc b/sw/device/lib/dif/dif_otp_ctrl_unittest.cc index e021d1d614af55..442dbd9cfb882e 100644 --- a/sw/device/lib/dif/dif_otp_ctrl_unittest.cc +++ b/sw/device/lib/dif/dif_otp_ctrl_unittest.cc @@ -1,9 +1,6 @@ // Copyright lowRISC contributors (OpenTitan project). // Licensed under the Apache License, Version 2.0, see LICENSE for details. // SPDX-License-Identifier: Apache-2.0 -// DO NOT EDIT THIS FILE DIRECTLY. -// It has been generated with ./util/design/gen-otp-mmap.py - #include "sw/device/lib/dif/dif_otp_ctrl.h" #include diff --git a/sw/device/silicon_creator/lib/drivers/BUILD b/sw/device/silicon_creator/lib/drivers/BUILD index 30379bac75a22e..2dbb98bdf80b51 100644 --- a/sw/device/silicon_creator/lib/drivers/BUILD +++ b/sw/device/silicon_creator/lib/drivers/BUILD @@ -48,8 +48,8 @@ dual_cc_library( ), deps = dual_inputs( device = [ - "//hw/top:otp_ctrl_c_regs", "//hw/top:alert_handler_c_regs", + "//hw/top:otp_ctrl_c_regs", "//hw/top_earlgrey/sw/autogen:top_earlgrey", ], host = [ diff --git a/sw/device/silicon_creator/manuf/README.md b/sw/device/silicon_creator/manuf/README.md index 253e305601f3a7..55a231e6138cb9 100644 --- a/sw/device/silicon_creator/manuf/README.md +++ b/sw/device/silicon_creator/manuf/README.md @@ -16,7 +16,7 @@ ## Manufacturing Stages The following section describes the EarlGrey manufacturing stages with respect -to the state of OTP. See `//hw/ip/otp_ctrl/data/earlgrey_skus/sival/BUILD` +to the state of OTP. See `//hw/top_earlgrey/data/otp/sival_skus/BUILD` for more details. ### `MANUF_EMPTY` diff --git a/sw/device/silicon_creator/manuf/base/BUILD b/sw/device/silicon_creator/manuf/base/BUILD index ae037f5e8b0892..c203f5cc247d8b 100644 --- a/sw/device/silicon_creator/manuf/base/BUILD +++ b/sw/device/silicon_creator/manuf/base/BUILD @@ -136,7 +136,7 @@ opentitan_test( binaries = {":sram_cp_provision": "sram_cp_provision"}, changes_otp = True, needs_jtag = True, - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_test_unlocked0_manuf_empty", + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_test_unlocked0_manuf_empty", tags = ["manuf"], test_cmd = _CP_PROVISIONING_CMD_ARGS, test_harness = _CP_PROVISIONING_HARNESS, @@ -164,7 +164,7 @@ opentitan_test( }, changes_otp = True, needs_jtag = True, - otp = "//hw/ip/otp_ctrl/data:img_raw", + otp = "//hw/top_earlgrey/data/otp:img_raw", tags = ["manuf"], test_cmd = """ --provisioning-sram-elf={sram_cp_provision} @@ -397,7 +397,7 @@ filegroup( changes_otp = True, data = config["ca_data"], needs_jtag = True, - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_test_locked0_manuf_initialized", + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_test_locked0_manuf_initialized", owner_slot_b = OWNER_SLOTS["b"], rom_ext_slot_a = SLOTS["a"], rom_ext_slot_b = SLOTS["b"], diff --git a/sw/device/silicon_creator/manuf/base/provisioning_inputs.bzl b/sw/device/silicon_creator/manuf/base/provisioning_inputs.bzl index a2d7c98dc76af3..6b43499dec6778 100644 --- a/sw/device/silicon_creator/manuf/base/provisioning_inputs.bzl +++ b/sw/device/silicon_creator/manuf/base/provisioning_inputs.bzl @@ -12,7 +12,7 @@ load( # individualization binaries that configure OTP with the constants defined in # these bazel targets. EARLGREY_OTP_CFGS = { - "sival": "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_consts", + "sival": "//hw/top_earlgrey/data/otp/sival_skus:otp_consts", } | EXT_EARLGREY_OTP_CFGS # A dictionary of SKU configurations that will be used to generate FT diff --git a/sw/device/silicon_creator/manuf/lib/BUILD b/sw/device/silicon_creator/manuf/lib/BUILD index 682bddf47ce8c1..069c8a4cb3c180 100644 --- a/sw/device/silicon_creator/manuf/lib/BUILD +++ b/sw/device/silicon_creator/manuf/lib/BUILD @@ -146,7 +146,7 @@ opentitan_test( }, fpga = fpga_params( changes_otp = True, - otp = "//hw/ip/otp_ctrl/data:img_test_unlocked1", + otp = "//hw/top_earlgrey/data/otp:img_test_unlocked1", tags = ["manuf"], ), deps = [ @@ -210,7 +210,7 @@ opentitan_test( }, fpga = fpga_params( changes_otp = True, - otp = "//hw/ip/otp_ctrl/data:img_test_unlocked1", + otp = "//hw/top_earlgrey/data/otp:img_test_unlocked1", tags = ["manuf"], test_cmd = """ --bootstrap={firmware} @@ -274,7 +274,7 @@ opentitan_test( fpga = fpga_params( changes_otp = True, needs_jtag = True, - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_dev_manuf_individualized", + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_dev_manuf_individualized", tags = [ "lc_dev", "manuf", diff --git a/sw/device/silicon_creator/manuf/tests/BUILD b/sw/device/silicon_creator/manuf/tests/BUILD index 3c9e01cf66ebdf..8bb2b635797bf6 100644 --- a/sw/device/silicon_creator/manuf/tests/BUILD +++ b/sw/device/silicon_creator/manuf/tests/BUILD @@ -80,7 +80,7 @@ _MANUF_TEST_LOCKED_KEY = None changes_otp = True, lc_state = lc_state, # will be expanded in test_cmd needs_jtag = True, - otp = "//hw/ip/otp_ctrl/data:img_{}".format(lc_state.lower()), + otp = "//hw/top_earlgrey/data/otp:img_{}".format(lc_state.lower()), tags = ["manuf"], test_cmd = ( "" if ( @@ -120,7 +120,7 @@ opentitan_test( fpga = fpga_params( changes_otp = True, needs_jtag = True, - otp = "//hw/ip/otp_ctrl/data:img_raw", + otp = "//hw/top_earlgrey/data/otp:img_raw", tags = ["manuf"], test_harness = "//sw/host/tests/manuf/manuf_cp_unlock_raw", ), @@ -142,7 +142,7 @@ opentitan_test( fpga = fpga_params( changes_otp = True, needs_jtag = True, - otp = "//hw/ip/otp_ctrl/data:img_raw", + otp = "//hw/top_earlgrey/data/otp:img_raw", tags = ["manuf"], test_harness = "//sw/host/tests/manuf/manuf_cp_volatile_unlock_raw", ), @@ -160,10 +160,10 @@ opentitan_test( [ otp_image( name = "otp_img_rom_exec_disabled_test_unlocked{}".format(i), - src = "//hw/ip/otp_ctrl/data:otp_json_test_unlocked{}".format(i), + src = "//hw/top_earlgrey/data/otp:otp_json_test_unlocked{}".format(i), overlays = [ - "//hw/ip/otp_ctrl/data:otp_json_fixed_secret0", - "//hw/ip/otp_ctrl/data:otp_json_exec_disabled", + "//hw/top_earlgrey/data/otp:otp_json_fixed_secret0", + "//hw/top_earlgrey/data/otp:otp_json_exec_disabled", ] + OTP_SIGVERIFY_FAKE_KEYS, visibility = ["//visibility:public"], ) @@ -458,7 +458,7 @@ opentitan_test( otp_image( name = "otp_img_otp_ctrl_functest", - src = "//hw/ip/otp_ctrl/data:otp_json_test_unlocked0", + src = "//hw/top_earlgrey/data/otp:otp_json_test_unlocked0", visibility = ["//visibility:private"], ) diff --git a/sw/device/silicon_creator/rom/e2e/BUILD b/sw/device/silicon_creator/rom/e2e/BUILD index c26d0c5709fb68..c69b1c45046e65 100644 --- a/sw/device/silicon_creator/rom/e2e/BUILD +++ b/sw/device/silicon_creator/rom/e2e/BUILD @@ -402,7 +402,7 @@ opentitan_test( "//hw/top_earlgrey:fpga_cw310_rom_with_fake_keys": None, }, fpga = fpga_params( - otp = "//hw/ip/otp_ctrl/data:img_dev", + otp = "//hw/top_earlgrey/data/otp:img_dev", # TODO(lowRISC/opentitan#13603): Remove this "manual" tag when the # bitstream target can fetch pre-spliced bitstream from GCP. tags = ["manual"], diff --git a/sw/device/silicon_creator/rom/e2e/boot_data_recovery/BUILD b/sw/device/silicon_creator/rom/e2e/boot_data_recovery/BUILD index e1867d78cfbf0f..12532a22d0a8f8 100644 --- a/sw/device/silicon_creator/rom/e2e/boot_data_recovery/BUILD +++ b/sw/device/silicon_creator/rom/e2e/boot_data_recovery/BUILD @@ -110,7 +110,7 @@ BOOT_DATA_RECOVERY_CASES = [ case["lc_state"], case["default_boot_data"], ), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(case["lc_state"]), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(case["lc_state"]), overlays = STD_OTP_OVERLAYS + [":boot_data_recovery_creator_sw_cfg_{}_{}".format( case["lc_state"], case["default_boot_data"], diff --git a/sw/device/silicon_creator/rom/e2e/boot_policy_bad_manifest/BUILD b/sw/device/silicon_creator/rom/e2e/boot_policy_bad_manifest/BUILD index 5776abadc0be5b..f0eb5b888c680c 100644 --- a/sw/device/silicon_creator/rom/e2e/boot_policy_bad_manifest/BUILD +++ b/sw/device/silicon_creator/rom/e2e/boot_policy_bad_manifest/BUILD @@ -206,7 +206,7 @@ BOOT_POLICY_BAD_MANIFEST_CASES = [ lc_state, sec_ver, ), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS + [":otp_json_sec_ver_{}".format(sec_ver)], visibility = ["//visibility:private"], ) diff --git a/sw/device/silicon_creator/rom/e2e/boot_policy_big_image/BUILD b/sw/device/silicon_creator/rom/e2e/boot_policy_big_image/BUILD index bd43dd46720ed7..3059b8c3db5c0b 100644 --- a/sw/device/silicon_creator/rom/e2e/boot_policy_big_image/BUILD +++ b/sw/device/silicon_creator/rom/e2e/boot_policy_big_image/BUILD @@ -39,7 +39,7 @@ BOOT_POLICY_BIG_IMAGE_CASES = [ "//hw/top_earlgrey:fpga_cw310_rom_with_fake_keys": None, }, fpga = fpga_params( - otp = "//hw/ip/otp_ctrl/data:img_{}".format(lc_state), + otp = "//hw/top_earlgrey/data/otp:img_{}".format(lc_state), tags = maybe_skip_in_ci(lc_state_val), ), local_defines = ["ARRAY_SIZE={}".format(t["array_size"])], diff --git a/sw/device/silicon_creator/rom/e2e/boot_policy_flash_ecc_error/BUILD b/sw/device/silicon_creator/rom/e2e/boot_policy_flash_ecc_error/BUILD index b301a1f11045f5..f48fdef06d1544 100644 --- a/sw/device/silicon_creator/rom/e2e/boot_policy_flash_ecc_error/BUILD +++ b/sw/device/silicon_creator/rom/e2e/boot_policy_flash_ecc_error/BUILD @@ -140,14 +140,14 @@ otp_json( otp_image( name = "otp_img_boot_policy_flash_ecc_error", - src = "//hw/ip/otp_ctrl/data:otp_json_prod", + src = "//hw/top_earlgrey/data/otp:otp_json_prod", overlays = STD_OTP_OVERLAYS + [":otp_json_flash_data_cfg_default_scr_and_ecc_enabled"], visibility = ["//visibility:private"], ) otp_image( name = "otp_img_flash_exc_handler_disabled", - src = "//hw/ip/otp_ctrl/data:otp_json_prod", + src = "//hw/top_earlgrey/data/otp:otp_json_prod", overlays = STD_OTP_OVERLAYS + [ ":otp_json_flash_data_cfg_default_scr_and_ecc_enabled", ":otp_json_flash_exc_handler_disabled", diff --git a/sw/device/silicon_creator/rom/e2e/boot_policy_newer/BUILD b/sw/device/silicon_creator/rom/e2e/boot_policy_newer/BUILD index 8ab9eb3e779ecf..bbe59b1a8fcf03 100644 --- a/sw/device/silicon_creator/rom/e2e/boot_policy_newer/BUILD +++ b/sw/device/silicon_creator/rom/e2e/boot_policy_newer/BUILD @@ -106,7 +106,7 @@ BOOT_POLICY_NEWER_CASES = [ [ otp_image( name = "otp_img_boot_policy_newer_{}".format(lc_state), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS, ) for lc_state, _ in get_lc_items() diff --git a/sw/device/silicon_creator/rom/e2e/boot_policy_rollback/BUILD b/sw/device/silicon_creator/rom/e2e/boot_policy_rollback/BUILD index 20b7612e06ac58..76105139a85289 100644 --- a/sw/device/silicon_creator/rom/e2e/boot_policy_rollback/BUILD +++ b/sw/device/silicon_creator/rom/e2e/boot_policy_rollback/BUILD @@ -79,7 +79,7 @@ otp_json( [ otp_image( name = "otp_img_boot_policy_rollback_{}".format(lc_state), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS + [":otp_json_boot_policy_rollback"], visibility = ["//visibility:private"], ) diff --git a/sw/device/silicon_creator/rom/e2e/boot_policy_valid/BUILD b/sw/device/silicon_creator/rom/e2e/boot_policy_valid/BUILD index a75fd5b2ec8550..525ac44d161eae 100644 --- a/sw/device/silicon_creator/rom/e2e/boot_policy_valid/BUILD +++ b/sw/device/silicon_creator/rom/e2e/boot_policy_valid/BUILD @@ -51,7 +51,7 @@ BOOT_POLICY_VALID_CASES = [ [ otp_image( name = "otp_img_boot_policy_valid_{}".format(lc_state), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS, ) for lc_state, _ in get_lc_items() diff --git a/sw/device/silicon_creator/rom/e2e/bootstrap/BUILD b/sw/device/silicon_creator/rom/e2e/bootstrap/BUILD index 3fe5f8613bab66..f2403e0f42806c 100644 --- a/sw/device/silicon_creator/rom/e2e/bootstrap/BUILD +++ b/sw/device/silicon_creator/rom/e2e/bootstrap/BUILD @@ -56,7 +56,7 @@ otp_json( # lifecycle state in order to test the transition from PROD to RMA. otp_image( name = "otp_img_bootstrap_rma", - src = "//hw/ip/otp_ctrl/data:otp_json_prod", + src = "//hw/top_earlgrey/data/otp:otp_json_prod", overlays = STD_OTP_OVERLAYS + [ ":otp_json_bootstrap_rma", ], @@ -103,7 +103,7 @@ opentitan_test( binaries = { "//sw/device/silicon_creator/rom/e2e:new_empty_test_slot_a": "firmware", }, - otp = "//hw/ip/otp_ctrl/data:img_bootstrap_disabled", + otp = "//hw/top_earlgrey/data/otp:img_bootstrap_disabled", test_harness = "//sw/host/tests/rom/e2e_bootstrap_disabled", ), manifest = None, diff --git a/sw/device/silicon_creator/rom/e2e/epmp_init/BUILD b/sw/device/silicon_creator/rom/e2e/epmp_init/BUILD index 887c6c3effadef..83e8eec4a075f7 100644 --- a/sw/device/silicon_creator/rom/e2e/epmp_init/BUILD +++ b/sw/device/silicon_creator/rom/e2e/epmp_init/BUILD @@ -33,7 +33,7 @@ package(default_visibility = ["//visibility:public"]) "//hw/top_earlgrey:sim_verilator": None, }, fpga = fpga_params( - otp = "//hw/ip/otp_ctrl/data:img_" + lc_state, + otp = "//hw/top_earlgrey/data/otp:img_" + lc_state, tags = maybe_skip_in_ci(lc_state_val), ), local_defines = [ @@ -46,7 +46,7 @@ package(default_visibility = ["//visibility:public"]) manifest = "//sw/device/silicon_creator/rom_ext:manifest", verilator = verilator_params( timeout = "eternal", - otp = "//hw/ip/otp_ctrl/data:img_" + lc_state, + otp = "//hw/top_earlgrey/data/otp:img_" + lc_state, rom = "//sw/device/silicon_creator/rom:mask_rom", ), deps = [ diff --git a/sw/device/silicon_creator/rom/e2e/immutable_rom_ext_section/BUILD b/sw/device/silicon_creator/rom/e2e/immutable_rom_ext_section/BUILD index 1be7e5245b9079..a9b6672b456a88 100644 --- a/sw/device/silicon_creator/rom/e2e/immutable_rom_ext_section/BUILD +++ b/sw/device/silicon_creator/rom/e2e/immutable_rom_ext_section/BUILD @@ -193,7 +193,7 @@ IMMUTABLE_PARTITION_TEST_CASES = [ s["name"], ), testonly = True, - src = "//hw/ip/otp_ctrl/data:otp_json_prod", + src = "//hw/top_earlgrey/data/otp:otp_json_prod", overlays = STD_OTP_OVERLAYS + [ ":otp_json_immutable_rom_ext_{}_{}".format( t["name"], diff --git a/sw/device/silicon_creator/rom/e2e/jtag_inject/BUILD b/sw/device/silicon_creator/rom/e2e/jtag_inject/BUILD index b2946d1e6b153f..23d425aa7e1cfc 100644 --- a/sw/device/silicon_creator/rom/e2e/jtag_inject/BUILD +++ b/sw/device/silicon_creator/rom/e2e/jtag_inject/BUILD @@ -25,8 +25,8 @@ package(default_visibility = ["//visibility:public"]) [ otp_image( name = "img_{}_exec_disabled".format(lc_state), - src = "//hw/ip/otp_ctrl/data:otp_json_" + lc_state, - overlays = STD_OTP_OVERLAYS + ["//hw/ip/otp_ctrl/data:otp_json_exec_disabled"], + src = "//hw/top_earlgrey/data/otp:otp_json_" + lc_state, + overlays = STD_OTP_OVERLAYS + ["//hw/top_earlgrey/data/otp:otp_json_exec_disabled"], visibility = ["//visibility:private"], ) for lc_state, _ in get_lc_items() diff --git a/sw/device/silicon_creator/rom/e2e/keymgr/BUILD b/sw/device/silicon_creator/rom/e2e/keymgr/BUILD index 53f592056d25e1..2422b66ddc5266 100644 --- a/sw/device/silicon_creator/rom/e2e/keymgr/BUILD +++ b/sw/device/silicon_creator/rom/e2e/keymgr/BUILD @@ -58,7 +58,7 @@ rom_e2e_keymgr_init_configs = [ [ otp_image( name = "otp_img_keymgr_{}".format(config["name"]), - src = "//hw/ip/otp_ctrl/data:otp_json_rma", + src = "//hw/top_earlgrey/data/otp:otp_json_rma", overlays = STD_OTP_OVERLAYS + [":otp_json_keymgr_{}".format(config["name"])], visibility = ["//visibility:private"], ) diff --git a/sw/device/silicon_creator/rom/e2e/presigned_images/BUILD b/sw/device/silicon_creator/rom/e2e/presigned_images/BUILD index 4d923e12689893..6c1cae4f1dc2e8 100644 --- a/sw/device/silicon_creator/rom/e2e/presigned_images/BUILD +++ b/sw/device/silicon_creator/rom/e2e/presigned_images/BUILD @@ -23,7 +23,7 @@ vmem_file( scramble_flash_vmem( name = "rom_e2e_self_hash_sim_dv_scr_vmem64", src = ":rom_e2e_self_hash_sim_dv_vmem64", - otp = "//hw/ip/otp_ctrl/data:img_test_unlocked0", + otp = "//hw/top_earlgrey/data/otp:img_test_unlocked0", ) [ diff --git a/sw/device/silicon_creator/rom/e2e/release/BUILD b/sw/device/silicon_creator/rom/e2e/release/BUILD index 3c90f1f6ef4584..35a74c4a21cb7e 100644 --- a/sw/device/silicon_creator/rom/e2e/release/BUILD +++ b/sw/device/silicon_creator/rom/e2e/release/BUILD @@ -18,7 +18,7 @@ package(default_visibility = ["//visibility:public"]) otp_image( name = "otp_img_sigverify_spx_prod", - src = "//hw/ip/otp_ctrl/data:otp_json_prod", + src = "//hw/top_earlgrey/data/otp:otp_json_prod", overlays = STD_OTP_OVERLAYS + [ "//sw/device/silicon_creator/rom/e2e/sigverify_spx:otp_json_sigverify_spx_enabled_true", ], diff --git a/sw/device/silicon_creator/rom/e2e/reset_reason/BUILD b/sw/device/silicon_creator/rom/e2e/reset_reason/BUILD index 1d2f84c29dc820..05914b239a9a23 100644 --- a/sw/device/silicon_creator/rom/e2e/reset_reason/BUILD +++ b/sw/device/silicon_creator/rom/e2e/reset_reason/BUILD @@ -39,7 +39,7 @@ otp_json( # OTP images that enable the watchdog. otp_image( name = "otp_img_reset_reason", - src = "//hw/ip/otp_ctrl/data:otp_json_rma", + src = "//hw/top_earlgrey/data/otp:otp_json_rma", overlays = STD_OTP_OVERLAYS + [":otp_json_reset_reason"], ) @@ -131,7 +131,7 @@ RESET_CORRUPTION_SCENARIOS = [ [ otp_image( name = "otp_img_reset_reason_{}".format(t["name"]), - src = "//hw/ip/otp_ctrl/data:otp_json_rma", + src = "//hw/top_earlgrey/data/otp:otp_json_rma", overlays = STD_OTP_OVERLAYS + [ ":otp_json_reset_reason_{}".format(t["name"]), ], diff --git a/sw/device/silicon_creator/rom/e2e/retention_ram/BUILD b/sw/device/silicon_creator/rom/e2e/retention_ram/BUILD index b5ffb729b3d3a5..76d245f89753b1 100644 --- a/sw/device/silicon_creator/rom/e2e/retention_ram/BUILD +++ b/sw/device/silicon_creator/rom/e2e/retention_ram/BUILD @@ -41,7 +41,7 @@ otp_json( otp_image( name = "otp_img_reset_ret_ram", - src = "//hw/ip/otp_ctrl/data:otp_json_rma", + src = "//hw/top_earlgrey/data/otp:otp_json_rma", overlays = STD_OTP_OVERLAYS + [":otp_json_reset_ret_ram_overlay"], visibility = ["//visibility:private"], ) diff --git a/sw/device/silicon_creator/rom/e2e/rom_e2e_bootstrap_entry/BUILD b/sw/device/silicon_creator/rom/e2e/rom_e2e_bootstrap_entry/BUILD index 0f89cce2d33499..4af2aef407cc31 100644 --- a/sw/device/silicon_creator/rom/e2e/rom_e2e_bootstrap_entry/BUILD +++ b/sw/device/silicon_creator/rom/e2e/rom_e2e_bootstrap_entry/BUILD @@ -27,7 +27,7 @@ package(default_visibility = ["//visibility:public"]) [ otp_image( name = "otp_img_e2e_bootstrap_entry_{}".format(lc_state), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS, ) for lc_state, _ in get_lc_items() diff --git a/sw/device/silicon_creator/rom/e2e/rom_ext_upgrade_interrupt/BUILD b/sw/device/silicon_creator/rom/e2e/rom_ext_upgrade_interrupt/BUILD index c85c7a233348ac..c56773eec8f7d2 100644 --- a/sw/device/silicon_creator/rom/e2e/rom_ext_upgrade_interrupt/BUILD +++ b/sw/device/silicon_creator/rom/e2e/rom_ext_upgrade_interrupt/BUILD @@ -51,7 +51,7 @@ otp_json( [ otp_image( name = "otp_img_rom_ext_upgrade_interrupt_{}".format(lc_state), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS + [":otp_json_rom_ext_upgrade_interrupt"], visibility = ["//visibility:private"], ) diff --git a/sw/device/silicon_creator/rom/e2e/shutdown_alert/BUILD b/sw/device/silicon_creator/rom/e2e/shutdown_alert/BUILD index 4789b064be5f3b..bddad99fc3068d 100644 --- a/sw/device/silicon_creator/rom/e2e/shutdown_alert/BUILD +++ b/sw/device/silicon_creator/rom/e2e/shutdown_alert/BUILD @@ -59,7 +59,7 @@ ALERT_LC_STATES = get_lc_items( "//hw/top_earlgrey:fpga_cw310_rom_with_fake_keys": None, }, fpga = fpga_params( - otp = "//hw/ip/otp_ctrl/data:img_{}".format(lc_state), + otp = "//hw/top_earlgrey/data/otp:img_{}".format(lc_state), tags = maybe_skip_in_ci(lc_state_val), ), local_defines = [ @@ -156,7 +156,7 @@ otp_alert_digest( [ otp_image( name = "otp_img_shutdown_alert_{}".format(lc_state), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS + [ ":shutdown_alert_owner_sw_cfg", ":shutdown_alert_digest_cfg", diff --git a/sw/device/silicon_creator/rom/e2e/shutdown_output/BUILD b/sw/device/silicon_creator/rom/e2e/shutdown_output/BUILD index e11ff80051f575..13db37171ed0b6 100644 --- a/sw/device/silicon_creator/rom/e2e/shutdown_output/BUILD +++ b/sw/device/silicon_creator/rom/e2e/shutdown_output/BUILD @@ -42,7 +42,7 @@ package(default_visibility = ["//visibility:public"]) [ otp_image( name = "otp_img_shutdown_output_{}".format(lc_state), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS, ) for lc_state, _ in get_lc_items() diff --git a/sw/device/silicon_creator/rom/e2e/shutdown_redact/BUILD b/sw/device/silicon_creator/rom/e2e/shutdown_redact/BUILD index 46fea51132fe5d..0dad5bb3c6d394 100644 --- a/sw/device/silicon_creator/rom/e2e/shutdown_redact/BUILD +++ b/sw/device/silicon_creator/rom/e2e/shutdown_redact/BUILD @@ -67,7 +67,7 @@ REDACT.update({"INVALID": 0x0}) lc_state, redact.lower(), ), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS + [":otp_json_{}_overlay".format(redact.lower())], visibility = ["//visibility:private"], ) diff --git a/sw/device/silicon_creator/rom/e2e/shutdown_watchdog/BUILD b/sw/device/silicon_creator/rom/e2e/shutdown_watchdog/BUILD index afb8cf2e367d95..3bdd977604a6a0 100644 --- a/sw/device/silicon_creator/rom/e2e/shutdown_watchdog/BUILD +++ b/sw/device/silicon_creator/rom/e2e/shutdown_watchdog/BUILD @@ -82,7 +82,7 @@ SHUTDOWN_WATCHDOG_CASES = [ t["lc_state"], t["bite_threshold"], ), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(t["lc_state"]), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(t["lc_state"]), overlays = STD_OTP_OVERLAYS + [ ":otp_json_shutdown_watchdog_{}_{}".format( t["lc_state"], diff --git a/sw/device/silicon_creator/rom/e2e/sigverify_always/BUILD b/sw/device/silicon_creator/rom/e2e/sigverify_always/BUILD index a44af4cf94ee07..9f26b6cd7bdc24 100644 --- a/sw/device/silicon_creator/rom/e2e/sigverify_always/BUILD +++ b/sw/device/silicon_creator/rom/e2e/sigverify_always/BUILD @@ -39,7 +39,7 @@ package(default_visibility = ["//visibility:public"]) [otp_image( name = "otp_img_sigverify_always_{}".format(lc_state), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS, ) for lc_state, _ in get_lc_items()] diff --git a/sw/device/silicon_creator/rom/e2e/sigverify_key_type/BUILD b/sw/device/silicon_creator/rom/e2e/sigverify_key_type/BUILD index 0860fcbd3d66fa..10e17a51ee43bf 100644 --- a/sw/device/silicon_creator/rom/e2e/sigverify_key_type/BUILD +++ b/sw/device/silicon_creator/rom/e2e/sigverify_key_type/BUILD @@ -125,7 +125,7 @@ otp_json( [ otp_image( name = "otp_img_sigverify_key_type_{}".format(lc_state), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS + [ ":otp_json_enable_spx", ], diff --git a/sw/device/silicon_creator/rom/e2e/sigverify_key_validity/BUILD b/sw/device/silicon_creator/rom/e2e/sigverify_key_validity/BUILD index 3eacfa4f567d88..1670a4c5b34166 100644 --- a/sw/device/silicon_creator/rom/e2e/sigverify_key_validity/BUILD +++ b/sw/device/silicon_creator/rom/e2e/sigverify_key_validity/BUILD @@ -197,7 +197,7 @@ KEY_VALIDITY_TESTS = [ test["name"], lc_state, ), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS + [ ":otp_json_sigverify_key_validity_{}".format(test["name"]), ], diff --git a/sw/device/silicon_creator/rom/e2e/sigverify_spx/BUILD b/sw/device/silicon_creator/rom/e2e/sigverify_spx/BUILD index 8cd8530b4faaed..cfcc6f3b93807f 100644 --- a/sw/device/silicon_creator/rom/e2e/sigverify_spx/BUILD +++ b/sw/device/silicon_creator/rom/e2e/sigverify_spx/BUILD @@ -136,7 +136,7 @@ opentitan_binary( lc_state, t["name"], ), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS + [ ":otp_json_sigverify_spx_{}".format(t["name"]), ], diff --git a/sw/device/silicon_creator/rom/e2e/sigverify_usage_constraints/BUILD b/sw/device/silicon_creator/rom/e2e/sigverify_usage_constraints/BUILD index 2faf80c74c7cc7..6dffdc4b1a992b 100644 --- a/sw/device/silicon_creator/rom/e2e/sigverify_usage_constraints/BUILD +++ b/sw/device/silicon_creator/rom/e2e/sigverify_usage_constraints/BUILD @@ -79,7 +79,7 @@ otp_json( [ otp_image( name = "otp_img_sigverify_usage_constraints_{}".format(lc_state), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS + [":otp_json_set_usage_constraint_params_overlay"], visibility = ["//visibility:private"], ) diff --git a/sw/device/silicon_creator/rom/e2e/sram/BUILD b/sw/device/silicon_creator/rom/e2e/sram/BUILD index ad5409d30364cd..2c4b552a70fcb1 100644 --- a/sw/device/silicon_creator/rom/e2e/sram/BUILD +++ b/sw/device/silicon_creator/rom/e2e/sram/BUILD @@ -38,7 +38,7 @@ otp_json( # OTP images that enable the watchdog. otp_image( name = "otp_img_readback_enable", - src = "//hw/ip/otp_ctrl/data:otp_json_rma", + src = "//hw/top_earlgrey/data/otp:otp_json_rma", overlays = STD_OTP_OVERLAYS + [":otp_json_readback_enable"], ) diff --git a/sw/device/silicon_creator/rom/e2e/watchdog/BUILD b/sw/device/silicon_creator/rom/e2e/watchdog/BUILD index 27828d70f02b93..5eb8f6a52f3f50 100644 --- a/sw/device/silicon_creator/rom/e2e/watchdog/BUILD +++ b/sw/device/silicon_creator/rom/e2e/watchdog/BUILD @@ -50,7 +50,7 @@ otp_json( [ otp_image( name = "otp_img_watchdog_enable_{}".format(lc_state), - src = "//hw/ip/otp_ctrl/data:otp_json_{}".format(lc_state), + src = "//hw/top_earlgrey/data/otp:otp_json_{}".format(lc_state), overlays = STD_OTP_OVERLAYS + [":otp_json_watchdog_enable"], ) for lc_state, _ in get_lc_items() @@ -109,7 +109,7 @@ WATCHDOG_TEST_CASES = { ] WATCHDOG_OTP = { - "disable": "//hw/ip/otp_ctrl/data:img_{}", + "disable": "//hw/top_earlgrey/data/otp:img_{}", "enable": ":otp_img_watchdog_enable_{}", } diff --git a/sw/device/silicon_creator/rom_ext/e2e/BUILD b/sw/device/silicon_creator/rom_ext/e2e/BUILD index 1ffb09d8d3612c..7c06de521cf845 100644 --- a/sw/device/silicon_creator/rom_ext/e2e/BUILD +++ b/sw/device/silicon_creator/rom_ext/e2e/BUILD @@ -31,7 +31,7 @@ otp_json( otp_image( name = "otp_img_secret2_locked_rma", - src = "//hw/ip/otp_ctrl/data:otp_json_rma", + src = "//hw/top_earlgrey/data/otp:otp_json_rma", overlays = STD_OTP_OVERLAYS + [ ":otp_json_secret2_locked", ], diff --git a/sw/device/silicon_creator/rom_ext/e2e/verified_boot/BUILD b/sw/device/silicon_creator/rom_ext/e2e/verified_boot/BUILD index 52e8c80a4fb71f..a25d750ab8c4aa 100644 --- a/sw/device/silicon_creator/rom_ext/e2e/verified_boot/BUILD +++ b/sw/device/silicon_creator/rom_ext/e2e/verified_boot/BUILD @@ -55,7 +55,7 @@ filegroup( otp_image( name = "otp_img_with_{}_imm_romext_enabled".format(name), testonly = True, - src = "//hw/ip/otp_ctrl/data:otp_json_prod", + src = "//hw/top_earlgrey/data/otp:otp_json_prod", overlays = STD_OTP_OVERLAYS + [ "//sw/device/silicon_creator/rom_ext/e2e:otp_json_secret2_locked", ":otp_json_with_{}_imm_romext_enabled".format(name), diff --git a/sw/device/tests/BUILD b/sw/device/tests/BUILD index c8f43366c39392..f96bacd3b663d5 100644 --- a/sw/device/tests/BUILD +++ b/sw/device/tests/BUILD @@ -800,7 +800,7 @@ opentitan_test( {"//hw/top_earlgrey:fpga_cw310_sival": None}, ), fpga = fpga_params( - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_dev_manuf_personalized", + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_dev_manuf_personalized", ), verilator = verilator_params(tags = ["broken"]), deps = [ @@ -817,7 +817,7 @@ opentitan_test( "//hw/top_earlgrey:sim_verilator": None, }, fpga = fpga_params( - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_dev_manuf_personalized", + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_dev_manuf_personalized", # TODO(lowrisc/opentitan#19620): fpga doesn't support lowering main clk frequency tags = ["broken"], ), @@ -1767,7 +1767,7 @@ _FLASH_CTRL_INFO_ACCESS_LC_STATES = get_lc_items( "//hw/top_earlgrey:fpga_cw310_sival": None, }, fpga = fpga_params( - otp = "//hw/ip/otp_ctrl/data:img_{}".format(lc_state), + otp = "//hw/top_earlgrey/data/otp:img_{}".format(lc_state), ), deps = [ "//hw/top_earlgrey/sw/autogen:top_earlgrey", @@ -1798,13 +1798,13 @@ test_suite( name = "flash_ctrl_info_access_lc_{}_personalized".format(lc_state), srcs = ["flash_ctrl_info_access_lc.c"], dv = dv_params( - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_{}_manuf_personalized".format(lc_state), + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_{}_manuf_personalized".format(lc_state), ), exec_env = { "//hw/top_earlgrey:fpga_cw310_sival": None, }, fpga = fpga_params( - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_{}_manuf_personalized".format(lc_state), + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_{}_manuf_personalized".format(lc_state), ), deps = [ "//hw/top_earlgrey/sw/autogen:top_earlgrey", @@ -1951,7 +1951,7 @@ opentitan_test( fpga = fpga_params( changes_otp = True, needs_jtag = True, - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_dev_manuf_personalized", + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_dev_manuf_personalized", tags = [ "broken", "lc_dev", @@ -2268,7 +2268,7 @@ otp_json( otp_image( name = "otp_ctrl_descrambling_otp_image", - src = "//hw/ip/otp_ctrl/data:otp_json_rma", + src = "//hw/top_earlgrey/data/otp:otp_json_rma", overlays = STD_OTP_OVERLAYS + [":otp_ctrl_descrambling_otp_json"], visibility = ["//visibility:private"], ) @@ -5328,7 +5328,7 @@ otp_json( otp_image( name = "power_virus_systemtest_otp_img_rma", - src = "//hw/ip/otp_ctrl/data:otp_json_rma", + src = "//hw/top_earlgrey/data/otp:otp_json_rma", overlays = STD_OTP_OVERLAYS + [":power_virus_systemtest_otp_overlay"], visibility = ["//visibility:private"], ) @@ -5351,7 +5351,7 @@ opentitan_test( }, ), fpga = fpga_params( - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_prod_manuf_personalized", + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_prod_manuf_personalized", test_cmd = """ --bootstrap="{firmware}" """, @@ -5476,7 +5476,7 @@ otp_json( otp_image( name = "flash_scrambling_smoketest_otp_img_rma", - src = "//hw/ip/otp_ctrl/data:otp_json_rma", + src = "//hw/top_earlgrey/data/otp:otp_json_rma", overlays = STD_OTP_OVERLAYS + [":flash_scrambling_smoketest_otp_overlay"], visibility = ["//visibility:private"], ) @@ -5496,12 +5496,12 @@ opentitan_test( ) _SIVAL_OTP_IMAGE = { - "test_unlocked1": "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_test_unlocked1_manuf_individualized", - "test_locked0": "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_test_locked0_manuf_initialized", - "dev": "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_dev_manuf_personalized", - "prod": "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_prod_manuf_personalized", - "prod_end": "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_prod_end_manuf_personalized", - "rma": "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_rma_manuf_personalized", + "test_unlocked1": "//hw/top_earlgrey/data/otp/sival_skus:otp_img_test_unlocked1_manuf_individualized", + "test_locked0": "//hw/top_earlgrey/data/otp/sival_skus:otp_img_test_locked0_manuf_initialized", + "dev": "//hw/top_earlgrey/data/otp/sival_skus:otp_img_dev_manuf_personalized", + "prod": "//hw/top_earlgrey/data/otp/sival_skus:otp_img_prod_manuf_personalized", + "prod_end": "//hw/top_earlgrey/data/otp/sival_skus:otp_img_prod_end_manuf_personalized", + "rma": "//hw/top_earlgrey/data/otp/sival_skus:otp_img_rma_manuf_personalized", } _RV_DM_JTAG_LC_STATES = get_lc_items( @@ -5527,7 +5527,7 @@ _RV_DM_TEST_CONFIGURATIONS = [ "name": "dev_rv_dm_delayed_enabled", "lc_state": "dev", "rv_dm_delayed_en": "--rv-dm-delayed-enable", - "otp": "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_dev_manuf_personalized_enable_rv_dm_late_debug_enable", + "otp": "//hw/top_earlgrey/data/otp/sival_skus:otp_img_dev_manuf_personalized_enable_rv_dm_late_debug_enable", }, { "name": "rma", @@ -6098,7 +6098,7 @@ opentitan_test( }, fpga = fpga_params( needs_jtag = True, - otp = "//hw/ip/otp_ctrl/data:img_test_unlocked0", + otp = "//hw/top_earlgrey/data/otp:img_test_unlocked0", # TODO(#22823): Remove "broken" tag once the test is fixed. tags = ["broken"], test_cmd = """ @@ -7135,7 +7135,7 @@ opentitan_test( fpga = fpga_params( timeout = "moderate", needs_jtag = True, - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_rma_manuf_personalized", + otp = "//hw/top_earlgrey/data/otp/sival_skus:otp_img_rma_manuf_personalized", test_cmd = " ".join([ "--bootstrap=\"{firmware}\"", "--firmware-elf=\"{firmware:elf}\"", diff --git a/sw/device/tests/README.md b/sw/device/tests/README.md index b9039c1736f31e..9d4c87d06b9d57 100644 --- a/sw/device/tests/README.md +++ b/sw/device/tests/README.md @@ -45,6 +45,6 @@ For instructions on how to write a chip-level test, refer to the [on-device test ## Read More * [Build & Test Rules](../../../rules/opentitan/README.md) -* [On-Device Test Framework (OTTF)](../lib//testing/test_framework/README.md) -* [OTP Build and Test Infrastructure](../../../hw/ip/otp_ctrl/data/README.md) +* [On-Device Test Framework (OTTF)](../lib/testing/test_framework/README.md) +* [OTP Preload Image Generator](../../../util/design/README.md#otp_preload_image_generator) * [FPGA Bitstreams](../../../hw/bitstream/README.md) diff --git a/sw/device/tests/doc/sival/README.md b/sw/device/tests/doc/sival/README.md index 0dfe80d14a86d1..06a9b10cc700bd 100644 --- a/sw/device/tests/doc/sival/README.md +++ b/sw/device/tests/doc/sival/README.md @@ -218,7 +218,8 @@ bazel test --define DISABLE_VERILATOR_BUILD=true \ * [Build & Test Rules](../../../../../rules/opentitan/README.md) * [FPGA Bitstreams](../../../../../hw/bitstream/README.md) * [On-Device Test Framework](../../../lib/testing/test_framework/README.md) -* [OTP Build and Test Infrastructure](../../../../../hw/ip/otp_ctrl/data/README.md) +* [OTP Preload Image Generator](../../../../../util/design/README.md#otp_preload_image_generator) +.md) * [ROM\_EXT for Silicon Validation](../../../silicon_creator/rom_ext/doc/si_val.md) * [Signing Guide](../../../../../signing/README.md) * [SiVal Developer Guide](./devguide.md) diff --git a/sw/device/tests/doc/sival/devguide.md b/sw/device/tests/doc/sival/devguide.md index 015320d9cc6ea1..5180817349644d 100644 --- a/sw/device/tests/doc/sival/devguide.md +++ b/sw/device/tests/doc/sival/devguide.md @@ -79,7 +79,7 @@ Silicon bring-up and validation activities have to be planned in a way that minimizes the number of OTP configurations given that there may be limited availability of samples during early post-silicon stages. -`//hw/ip/otp_ctrl/data/earlgrey_skus/sival/BUILD` contains the OTP +`//hw/top_earlgrey/data/otp/sival_skus/BUILD` contains the OTP configurations used in Silicon Validation (SiVal) for the EarlGrey design. ### `:otp_img_test_unlocked0_manuf_empty` @@ -187,7 +187,7 @@ Configuration: ```python { rom = "//sw/device/silicon_creator/rom:mask_rom", - otp = "//hw/ip/otp_ctrl/data/earlgrey_skus/sival:otp_img_prod_manuf_personalized", + otp = "//hw/top_earlgrey/otp/data/sival_skus:otp_img_prod_manuf_personalized", } ``` diff --git a/sw/device/tock/BUILD b/sw/device/tock/BUILD index 4dfe8e6744922c..b184ade70daa39 100644 --- a/sw/device/tock/BUILD +++ b/sw/device/tock/BUILD @@ -45,6 +45,7 @@ filegroup( "//hw/top:ast_rust_regs", "//hw/top:clkmgr_rust_regs", "//hw/top:flash_ctrl_rust_regs", + "//hw/top:otp_ctrl_rust_regs", "//hw/top:pinmux_rust_regs", "//hw/top:pwrmgr_rust_regs", "//hw/top:rstmgr_rust_regs", diff --git a/util/design/BUILD b/util/design/BUILD index f26e0df3c9b159..8c792a55b56f57 100644 --- a/util/design/BUILD +++ b/util/design/BUILD @@ -27,6 +27,7 @@ py_binary( "//util/design/lib:common", "//util/design/lib:otp_mem_map", "//util/design/lib:present", + "//util/topgen", requirement("pyfinite"), ], ) @@ -56,6 +57,16 @@ py_binary( ], ) +py_binary( + name = "gen-otp-mmap", + srcs = ["gen-otp-mmap.py"], + deps = [ + "//util/design/lib:common", + "//util/design/lib:otp_mem_map", + requirement("hjson"), + ], +) + py_binary( name = "gen-otp-rot-auth-json", srcs = ["gen-otp-rot-auth-json.py"], diff --git a/util/design/README.md b/util/design/README.md index 613217136bcb2f..345c8b1f62b34f 100644 --- a/util/design/README.md +++ b/util/design/README.md @@ -1,11 +1,9 @@ # Design-related Tooling Scripts -## Life Cycle and OTP Tools - -### OTP Memory Map Translation Script +## OTP Memory Map Translation Script The `gen-otp-mmap.py` script is used to translate the OTP memory map definition Hjson file into documentation and SV package collateral. -The memory map definition file for top_earlgrey is currently located at `hw/ip/otp_ctrl/data/otp_ctrl_mmap.hjson`. +The memory map definition file for top_earlgrey is currently located at `hw/top_earlgrey/data/otp/otp_ctrl_mmap.hjson`. The script can either be invoked via the makefile ```console @@ -24,7 +22,7 @@ $ ./util/design/gen-otp-mmap.py The seed value used for generating OTP-related random netlist constants can optionally be overridden with the `--seed` switch when calling the script directly. Otherwise that seed value is taken from the Hjson file, or generated on-the-fly if the Hjson file does not contain a seed. -### Life Cycle State Encoding Generator +## Life Cycle State Encoding Generator The `gen-lc-state-enc.py` script is used to generate the redundant life cycle state encoding and print the constants and type definitions into the life cycle state package. The life cycle definition file for top_earlgrey is currently located at `hw/ip/lc_ctrl/data/lc_ctrl_state.hjson`. @@ -46,11 +44,81 @@ $ ./util/design/gen-lc-state-enc.py The seed value used for generating life-cycle-state-related random netlist constants can optionally be overridden with the `--seed` switch when calling the script directly. Otherwise that seed value is taken from the Hjson file, or generated on-the-fly if the Hjson file does not contain a seed. -### OTP Preload Image Generator +## OTP Preload Image Generator The OTP preload image generation tool builds on top of the memory map and life cycle state generation Python classes in order to transform a memory image configuration into a memory hexfile that can be used for OTP preloading in simulation and FPGA emulation runs. The generated hexfile is compatible with the Verilog `$readmemh` command. +OTP image configurations are defined using hjson objects. Currently there are +two ways to build images: + +1. Pre-built OTP image overlays defined in hjson. These is the approach + currently used in most DV test cases. +2. Dynamically built OTP image overlays defined in [Bazel](#bazel). This is the + approach currently used in FPGA and Silicon Validation (SiVal) targets. + +### Bazel + +#### OTP HJSON Map + +OTP image overlays are first defined using the `otp_json` Bazel rule. The +following example shows the definition of a `SECRET2` partition configuration: + +```python +otp_json( + name = "otp_json_secret2_unlocked", + partitions = [ + otp_partition( + name = "SECRET2", + items = { + "RMA_TOKEN": "", + "CREATOR_ROOT_KEY_SHARE0": "", + "CREATOR_ROOT_KEY_SHARE1": "", + }, + lock = False, + ), + ], +) +``` + +See `//rules/otp.bzl` for additional documentation on additional parameters +available in the `otp_json` rule. + +#### OTP Image + +An OTP image is a collection of OTP JSON files used to create an OTP image. +An OTP can have multiple `otp_json` dependencies. Each dependency has the +ability of override the values of the previous dependency, so the order in +which these are listed is important. + +```python +# Represents a device in DEV state with the SECRET0 and SECRET1 partitions in +# locked state. SECRET2 partition is unlocked. +otp_image( + name = "img_dev_individualized", + src = ":otp_json_dev", + overlays = [ + ":otp_json_secret0", + ":otp_json_secret1", + ] + STD_OTP_OVERLAYS_WITHOUT_SECRET_PARTITIONS, +) +``` + +In this example, the `src` attribute points to the baseline OTP JSON +configuration, and the list of overlay dependencies are applied in order +of precedence in the `overlays` attribute. + +The `STD_OTP_OVERLAYS_WITHOUT_SECRET_PARTITIONS` definition imported from +`//rules:otp.bzl` declares a list of `otp_json` targets that are used +as overlays. There are other list of predefined overlays that are used +throughout the code base. + +### FPGA Integration + +See [FPGA bitstreams](../../hw/bitstream/README.md) documentation for more details. + +### DV Flow + The OTP memory image configuration file is basically an Hjson file that lists the names and corresponding values of items defined in the OTP memory map definition. Further, since the life cycle state is stored in OTP, the image generator script also supports generating the correct life cycle state encodings. To that end, the desired life cycle state name can be declared in the image configuration file, and the generator script looks up and assigns the correct netlist constants. @@ -99,14 +167,14 @@ The following snippet shows a memory image configuration that puts the device in } ``` -Common example configuration files that can be used for simulation and emulation are checked in under `hw/ip/otp_ctrl/data`, e.g. `hw/ip/otp_ctrl/data/otp_ctrl_img_dev.hjson` which provisions all buffered partitions and puts the device into the `DEV` life cycle. +Common example configuration files that can be used for simulation and emulation are checked in under `hw/top_earlgrey/data/otp`, e.g. `hw/top_earlgrey/data/otp/otp_ctrl_img_dev.hjson` which provisions all buffered partitions and puts the device into the `DEV` life cycle. Note that the preload image generator script automatically scrambles secret partitions, computes digests of locked partitions using the PRESENT cipher, and computes the OTP ECC bits. The OTP preload image generator expects at least one main image configuration file to be specified with the `--img-cfg` switch, for example: ```console $ cd ${PROJ_ROOT} -$ ./util/design/gen-otp-img.py --img-cfg hw/ip/otp_ctrl/data/otp_ctrl_img_dev.hjson \ +$ ./util/design/gen-otp-img.py --img-cfg hw/top_earlgrey/data/otp/otp_ctrl_img_dev.hjson \ --out otp-img.mem ``` @@ -152,7 +220,7 @@ For example, this can be used to patch in additional software configuration data The generator script call would then look as follows: ```console $ cd ${PROJ_ROOT} -$ ./util/design/gen-otp-img.py --img-cfg hw/ip/otp_ctrl/data/otp_ctrl_img_dev.hjson \ +$ ./util/design/gen-otp-img.py --img-cfg hw/top_earlgrey/data/otp/otp_ctrl_img_dev.hjson \ --add-cfg otp_ctrl_img_sw_cfg.hjson \ --out otp-img.mem ``` diff --git a/util/design/data/BUILD b/util/design/data/BUILD index c8dcacd88fdb58..c8367524bbdc49 100644 --- a/util/design/data/BUILD +++ b/util/design/data/BUILD @@ -2,6 +2,35 @@ # Licensed under the Apache License, Version 2.0, see LICENSE for details. # SPDX-License-Identifier: Apache-2.0 +load("@bazel_skylib//rules:common_settings.bzl", "int_flag", "string_flag") + package(default_visibility = ["//visibility:public"]) exports_files(glob(["**"])) + +# These configurations expose the OTP image generation tool's command line +# arguments to enable dvsim to pass this through Bazel to the underlying OTP +# image generation script. This is required to enable dvsim to invoke OTP image +# generation as part of the Bazel build process, while still enabling the use of +# multiple seeds needed to achieve DV coverage. +int_flag( + name = "img_seed", + build_setting_default = 0, +) + +string_flag( + name = "lc_seed", + # Default must match value in hw/top_earlgrey/data/otp/lc_ctrl.hjson. + build_setting_default = "40182201019264397688411770949626922549663256047001778394918990008320537410392", +) + +string_flag( + name = "otp_seed", + # Default must match value in hw/top_earlgrey/data/otp/otp_ctrl_mmap.hjson. + build_setting_default = "36021179872380457113239299468132194022238108125576166239904535336103582949069", +) + +string_flag( + name = "data_perm", + build_setting_default = "", +) diff --git a/hw/ip/otp_ctrl/data/otp_ctrl_img.c.tpl b/util/design/data/otp_ctrl_img.c.tpl similarity index 98% rename from hw/ip/otp_ctrl/data/otp_ctrl_img.c.tpl rename to util/design/data/otp_ctrl_img.c.tpl index 20def4f02ba833..f3cce72f0b5b45 100644 --- a/hw/ip/otp_ctrl/data/otp_ctrl_img.c.tpl +++ b/util/design/data/otp_ctrl_img.c.tpl @@ -4,7 +4,7 @@ // ${fileheader} // AUTOGENERATED. Do not edit this file by hand. -// See the hw/ip/otp_ctrl/data README for details. +// See the util/design/data hw/ip/otp_ctrl/data README for details. #include diff --git a/util/design/gen-flash-img.py b/util/design/gen-flash-img.py index 45104bcbc7519b..8478f6a02e80ad 100755 --- a/util/design/gen-flash-img.py +++ b/util/design/gen-flash-img.py @@ -21,11 +21,11 @@ import hjson from pyfinite import ffield -from util.design.lib.common import (inverse_permute_bits, - validate_data_perm_option, - vmem_permutation_string) -from util.design.lib.OtpMemMap import OtpMemMap -from util.design.lib.Present import Present +from lib.OtpMemMap import OtpMemMap +from lib.common import (inverse_permute_bits, + validate_data_perm_option, + vmem_permutation_string) +from lib.Present import Present import prince import secded_gen diff --git a/util/design/gen-otp-img.py b/util/design/gen-otp-img.py index 772aab5f21bec1..41dee4ed74359f 100755 --- a/util/design/gen-otp-img.py +++ b/util/design/gen-otp-img.py @@ -17,11 +17,11 @@ from lib.OtpMemImg import OtpMemImg # Get the memory map definition. -MMAP_DEFINITION_FILE = 'hw/ip/otp_ctrl/data/otp_ctrl_mmap.hjson' +MMAP_DEFINITION_FILE = 'hw/top_earlgrey/data/otp/otp_ctrl_mmap.hjson' # Life cycle state and ECC poly definitions. LC_STATE_DEFINITION_FILE = 'hw/ip/lc_ctrl/data/lc_ctrl_state.hjson' # Default image file definition (can be overridden on the command line). -IMAGE_DEFINITION_FILE = 'hw/ip/otp_ctrl/data/otp_ctrl_img_dev.hjson' +IMAGE_DEFINITION_FILE = 'hw/top_earlgrey/data/otp/otp_ctrl_img_dev.hjson' # Default output path (can be overridden on the command line). Note that # "BITWIDTH" will be replaced with the architecture's bitness. MEMORY_MEM_FILE = 'otp-img.BITWIDTH.vmem' diff --git a/util/design/gen-otp-mmap.py b/util/design/gen-otp-mmap.py index a7f4bac4c8c2f4..bd33b33bef1346 100755 --- a/util/design/gen-otp-mmap.py +++ b/util/design/gen-otp-mmap.py @@ -11,7 +11,6 @@ from pathlib import Path from typing import Dict -import hjson from mako import exceptions from mako.template import Template @@ -19,7 +18,7 @@ from lib.OtpMemMap import OtpMemMap # This makes topgen libraries available to template files. -sys.path.append(Path(__file__).parent) +sys.path.append(Path(__file__).parents[1]) TABLE_HEADER_COMMENT = '''