This document describes the functionality of the reset controller and its interaction with the rest of the OpenTitan system.
The OpenTitan reset topology and reset controller block diagram are shown in the diagram below. The reset controller is closely related to the power controller, please refer to that spec for details on how reset controller inputs are controlled.
The topology can be summarized as follows:
TRSTn
rv_dm
module)The test domain does not have sub reset trees. TRSTn
is used directly by all components in the domain.
The Core domain consists of all remaining logic and contains 4 sub reset trees, see table below.
The reset trees are cascaded upon one another in this order:
rst_por_n
-> rst_lc_n
-> rst_module_n
rst_por_n
-> rst_sys_n
-> rst_module_n
This means when a particular reset asserts, all downstream resets also assert.The primary difference between rst_lc_n
and rst_sys_n
is that the former controls the reset state of most logic in the system, while the latter controls the reset state only of the debug domain. This separation is required because the debug domain may request the system to reset while retaining debug info and control. This is particularly useful if one wanted to debug something early during the boot flow, and thus needed to set a break point after requesting a debug reset.
The reset topology also contains additional properties:
hartreset
in dmcontrol
, are not implemented, as it causes a security policy inconsistency with the remaining system.TRSTn
) should be explicitly asserted through system integration.TRSTn
only needs to be released for RMA transitions and nothing else. .The reset manager handles the reset of the core domain, and also holds relevant reset information in CSR registers, such as:
Additionally, the reset manager, along with the power manager, accepts requests from the system and asserts resets for the appropriate clock trees. These requests primarily come from the following sources:
rv_dm
.alert_handler
or pwrmgr
itself.OpenTitan supports the shadow configuration registers. These are registers stored in two constantly checking copies to ensure the values are not maliciously or accidentally disturbed. For these components, the reset manager outputs a shadow reset dedicated to resetting only the shadow storage. This reset separation ensures that a targetted attack on the reset line cannot easily defeat shadow registers.
The reset manager implements reset consistency checks to ensure that triggered resets are supposed to happen and not due to some fault in the system. Every leaf reset in the system has an associated consistency checker.
The consistency check ensures that when a leaf reset asserts, either its parent reset must have asserted, or the software request, if available, has asserted. While this sounds simple in principle, the check itself crosses up to 3 clock domains and must be carefully managed.
First, the parent and leaf resets are used to asynchronously assert a flag indication. This flag indication is then synchronized into the reset manager's local clock domain.
The reset manager then checks as follows:
If a leaf reset has asserted, check to see either its parent or software request (synchronous to the local domain) has asserted.
If the condition is not true, it is possible the parent reset indication is still being synchronized, thus we wait for the parent indication.
It is also possible the parent indication was seen first, but the leaf condition was not, in this case, we wait for the leaf indication.
A timeout period corresponding to the maximum synchronization delay is used to cover both waits.
If all reset conditions are satisfied, wait for the reset release to gracefully complete the cycle.
The following table lists the instantiation parameters of rstmgr
.
Parameter | Default | Description |
---|---|---|
SecCheck | 1 | Enables reset consistency checks on the leaf reset. Each check contains a small FSM. |
SecMaxSyncDelay | 2 | The default synchronization delay assumptions used in reset consistency checks. If a design uses a sync cell with more stages of delay, that value should be supplied. |
The reset manager generates the resets required by the system by synchronizing reset tree components to appropriate output clocks. As a result, a particular reset tree (for example rst_lc_n
) may have multiple outputs depending on the clock domains of its consumers.
Each reset tree is discussed in detail below.
The POR reset tree, rst_por_n
, is the root reset of the entire device. If this reset ever asserts, everything in the design is reset.
The ast
input aon_pok
is used as the root reset indication. It is filtered and stretched to cover any slow voltage ramp scenarios. The stretch parameters are design time configurations.
Life cycle reset, rst_lc_n
asserts under the following conditions:
rst_por_n
asserts.The rst_lc_n
tree contains both always-on and non-always-on versions. How many non-always-on versions is dependent on how many power domains are supported by the system.
System reset, rst_sys_n
, assertion depends on life cycle state.
When in PROD and PROD_END states, rst_sys_n
is identical to rst_lc_n
.
When in TEST, RMA and DEV states, rst_sys_n
is identical to rst_lc_n
unless the reset request is ndmreset_req
. ndmreset_req
is issued by the debug module of the system, it requests for all logic, except those needed to maintain debug state to reset.
Since ndmreset_req
is valid only during TEST, RMA and DEV states, it is the only place where the reset is differentiated. During these states, when ndmreset_req
is issued, all logic except the debug module and associated glue logic are reset.
The rst_sys_n
tree contains both always-on and non-always-on versions. How many non-always-on versions is dependent on how many power domains are supported by the system.
The reset trees discussed above are not directly output to the system for consumption. Instead, the output leaf resets are synchronized versions of the various root resets. How many leaf resets there are and to which clock is decided by the system and templated through the reset manager module.
Assuming a leaf output has N power domains and M clock domains, it potentially means one reset tree may output NxM outputs to satisfy all the reset scenario combinations.
It is alluded above that reset trees may contain both always-on and non-always-on versions. This distinction is required to support power manager's various low power states. When a power domain goes offline, all of its components must reset, regardless of the reset tree to which it belongs.
For example, assume a system with two power domains - Domain A
is always-on, and Domain B
is non-always-on. When Domain B
is powered off, all of Domain B
's resets, from rst_lc_n
, rst_sys_n
to rst_module_n
are asserted. However, the corresponding resets for Domain A
are left untouched because it has not been powered off.
Certain leaf resets can be directly controlled by software. Due to security considerations, most leaf resets cannot be controlled, only a few blocks are given exceptions. The only blocks currently allowed to software reset are spi_device
, usbdev
, spi_host
and i2c
.
The criteria for selecting which block is software reset controllable is meant to be overly restrictive. Unless there is a clear need, the default option is to not provide reset control.
In general, the following rules apply:
The following table summarizes the different reset requests and which part of each reset tree, along with what power domain is affected.
Reset Request Type | Example | POR Reset Tree | LC Reset Tree | SYS Reset Tree | Module Specific Reset |
---|---|---|---|---|---|
POR | VCC toggle, POR_N pad toggle | all domains | all domains | all domains | all domains |
HW reset Request | aon_timer reset request, alert_handler escalation request | all domains | all domains | all domains | |
Directed SW system reset request | rstmgr SW_RESET | all domains | all domains | all domains | |
Ndm reset request (PROD/PROD_END) | rv_dm non-debug-module reset request in PROD | all domains | all domains | all domains | |
Ndm reset request (Other states) | rv_dm non-debug-module reset request in DEV | all domains | all domains | ||
SW low power entry | wait-for-interrupt deep sleep entry | non-aon domains | non-aon domains | non-aon domains | |
SW controlled reset request | rstmgr SW_RST_CTRL_N | all domains |
The reset information register is a reflection of the reset state from the perspective of the system. In OpenTitan, since there is only 1 host, it is thus from the perspective of the processor. This also suggests that if the design had multiple processors, there would need to be multiple such registers.
If a reset does not cause the processor to reset, there is no reason for the reset information to change (this is also why there is a strong security link between the reset of the processor and the rest of the system). The following are the currently defined reset reasons and their meaning:
Reset Cause | Description |
---|---|
POR | Cold boot, the system was reset through POR circuitry. |
LOW_POWER_EXIT | Warm boot, the system was reset through low power exit. |
NDM RESET | Warm boot, the system was reset through rv_dm non-debug-module request. |
SW_REQ | Warm boot, the system was reset through {{#regref rstmgr.RESET_REQ }}. |
HW_REQ | Warm boot, the system was reset through peripheral requests. There may be multiple such requests. |
The reset info register is write 1 clear. It is software responsibility to clear old reset reasons; the reset manager simply records based on the rules below.
Excluding power on reset, which is always recorded when the device POR circuitry is triggered, the other resets are recorded when authorized by the reset manager. Reset manager authorization is based on reset categories as indicated by the power manager. The power manager has three reset categories that are mutually exclusive:
The reset categories are sent to the reset manager so that it can decide which reason to record when the processor reset is observed. Non-debug-module resets are allowed only when no resets have been triggered by pwrmgr.
Since a reset could be motivated by multiple reasons (a security escalation during low power transition for example), the reset information registers constantly record all reset causes in which it is allowed. The only case where this is not done is POR
, where active recording is silenced until the first processor reset release.
Even though four reset causes are labeled as warm boot, their effects on the system are not identical.
LOW_POWER_EXIT
, it means only the non-always-on domains have been reset.NDM_RESET
, it means only the rst_sys_n
tree has asserted for all power domains.HW_REQ
or SW_REQ
, it means everything other than power / clock / reset managers have reset.This behavioral difference may be important to software, as it implies the configuration of the system may need to be different.
The reset manager manages crash dump information for software debugging across unexpected resets and watchdogs. When enabled, the latest alert information and latest CPU information are captured in always-on registers.
When the software resumes after the reset, it is then able to examine the last CPU state or the last set of alert information to understand why the system has reset.
The enable for such debug capture can be locked such that it never captures.
The alert information register contains the value of the alert crash dump prior to a triggered reset. Since this information differs in length between system implementation, the alert information register only displays 32-bits at a time. The {{#regref rstmgr.ALERT_INFO_ATTR }} register indicates how many 32-bit data segments must be read.
To enable alert crash dump capture, set {{#regref rstmgr.ALERT_INFO_CTRL.EN }} to 1. Once the system has reset, check {{#regref rstmgr.ALERT_INFO_ATTR.CNT_AVAIL }} for how many reads need to be done. Set {{#regref rstmgr.ALERT_INFO_CTRL.INDEX }} to the desired segment, and then read the output from {{#regref rstmgr.ALERT_INFO }}.
The CPU information register contains the value of the CPU state prior to a triggered reset. Since this information differs in length between system implementation, the information register only displays 32-bits at a time.
For more details on the CPU dump details, please see crash dump.
The {{#regref rstmgr.CPU_INFO_ATTR }} register indicates how many 32-bit data segments must be read. Software then simply needs to write in {{#regref rstmgr.CPU_INFO_CTRL.INDEX }} which segment it wishes and then read out the {{#regref rstmgr.CPU_INFO }} register.
{{< dif_listing “sw/device/lib/dif/dif_rstmgr.h” >}}