From b5d7c135a57cf112d9141be557aa0027b6fca3a6 Mon Sep 17 00:00:00 2001 From: Harry Callahan Date: Fri, 10 Mar 2023 12:18:19 +0000 Subject: [PATCH] Run 'make -k -C hw top' to re-generate alert_handler/rv_plic docs Signed-off-by: Harry Callahan --- SUMMARY.md | 4 + .../ip_autogen/alert_handler/README.md | 911 +----------------- .../alert_handler/doc/programmers_guide.md | 122 +++ .../alert_handler/doc/theory_of_operation.md | 785 +++++++++++++++ hw/top_earlgrey/ip_autogen/rv_plic/README.md | 223 +---- .../rv_plic/doc/programmers_guide.md | 100 ++ .../rv_plic/doc/theory_of_operation.md | 119 +++ 7 files changed, 1132 insertions(+), 1132 deletions(-) create mode 100644 hw/top_earlgrey/ip_autogen/alert_handler/doc/programmers_guide.md create mode 100644 hw/top_earlgrey/ip_autogen/alert_handler/doc/theory_of_operation.md create mode 100644 hw/top_earlgrey/ip_autogen/rv_plic/doc/programmers_guide.md create mode 100644 hw/top_earlgrey/ip_autogen/rv_plic/doc/theory_of_operation.md diff --git a/SUMMARY.md b/SUMMARY.md index 8735b7ff23eec..aa30609b6281c 100644 --- a/SUMMARY.md +++ b/SUMMARY.md @@ -12,14 +12,18 @@ - [Design Verification](./hw/top_earlgrey/dv/README.md) - [Analog Sensor Top](./hw/top_earlgrey/ip/ast/README.md) - [Alert Handler](./hw/top_earlgrey/ip_autogen/alert_handler/README.md) + - [Theory of Operation](./hw/top_earlgrey/ip_autogen/alert_handler/doc/theory_of_operation.md) - [Design Verification](./hw/top_earlgrey/ip_autogen/alert_handler/dv/README.md) - [Testplan](./hw/top_earlgrey/ip_autogen/alert_handler/data/alert_handler_testplan.hjson) + - [Programmer's Guide](./hw/top_earlgrey/ip_autogen/alert_handler/doc/programmers_guide.md) - [Interface and Registers](./hw/top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson) - [Device Interface Functions](./sw/device/lib/dif/dif_alert_handler.h) - [Checklist](./hw/top_earlgrey/ip_autogen/alert_handler/doc/checklist.md) - [Interrupt Controller](./hw/top_earlgrey/ip_autogen/rv_plic/README.md) + - [Theory of Operation](./hw/top_earlgrey/ip_autogen/rv_plic/doc/theory_of_operation.md) - [Design Verification](./hw/top_earlgrey/ip_autogen/rv_plic/doc/dv/README.md) - [Testplan](./hw/top_earlgrey/ip_autogen/rv_plic/data/rv_plic_fpv_testplan.hjson) + - [Programmer's Guide](./hw/top_earlgrey/ip_autogen/rv_plic/doc/programmers_guide.md) - [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) diff --git a/hw/top_earlgrey/ip_autogen/alert_handler/README.md b/hw/top_earlgrey/ip_autogen/alert_handler/README.md index 7f45d196129cf..69e388bdda458 100644 --- a/hw/top_earlgrey/ip_autogen/alert_handler/README.md +++ b/hw/top_earlgrey/ip_autogen/alert_handler/README.md @@ -5,7 +5,7 @@ # Overview This document specifies the functionality of the alert handler mechanism. -The alert handler 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 alert handler is a module that is a peripheral on the chip interconnect bus, and thus follows the [Comportability Specification](../../../../doc/contributing/hw/comportability/README.md). It gathers alerts - defined as interrupt-type signals from other peripherals that are designated as potential security threats - throughout the design, and converts them to interrupts that the processor can handle. If the processor does not handle them, the alert handler mechanism provides hardware responses to handle the threat. @@ -68,912 +68,3 @@ If that counter exceeds a programmable maximum value, then the escalation protoc The details for alert signaling, classification, and escalation are all given in the Theory of Operations section. - -# Theory of Operations - -## Block Diagram - -The figure below shows a block diagram of the alert handler module, as well as a few examples of alert senders in other peripheral modules. -In this diagram, there are seven sources of alerts: three sources from external modules (two from `periph0` and one from `periph1`), and four local sources (`alert_ping_fail`, `alert_sig_int`, `esc_ping_fail`, `esc_sig_int`). -The local sources represent alerts that are created by this module itself. See the later section on special local alerts. - -![Alert Handler Block Diagram](./doc/alert_handler_block_diagram.svg) - -Also shown are internal modules for classification, interrupt generation, accumulation, escalation, ping generation and alert-channel low-power control. -These are described later in the document. -Note that the differential alert sender and receiver blocks used for alert signaling support both _asynchronous_ and _synchronous_ clocking schemes, and hence peripherals able to raise alerts may be placed in clock domains different from that of the alert handler (Jittered clock domains are also supported in the asynchronous clocking scheme). -Proper care must however be taken when formulating the timing constraints for the diff pairs, and when determining clock-dependent parameters (such as the ping timeout) of the design. -On the escalation sender / receiver side, the differential signaling blocks employ a fully synchronous clocking scheme throughout. - -## Hardware Interfaces - -### Parameters - -The following table lists the main parameters used throughout the alert handler design. -Note that the alert handler is generated based on the system configuration, and hence these parameters are placed into a package as "localparams". -The parameterization rules are explained in more detail in the architectural description. - -Localparam | Default (Max) | Top Earlgrey | Description ----------------|-----------------------|--------------|--------------- -`NAlerts` | 8 (248) | see RTL | Number of alert instances. Maximum number bounded by LFSR implementation that generates ping timing. -`NLpg` | 1 | see RTL | Number of unique low-power groups as determined by topgen. -`LpgMap` | {0} | see RTL | Array mapping each alert to a unique low-power group as determined by topgen. -`EscCntWidth` | 32 (32) | 32 | Width of the escalation counters in bit. -`AccuCntWidth` | 16 (32) | 16 | Width of the alert accumulation counters in bit. -`AsyncOn` | '0 (2^`NAlerts`-1) | see RTL | This is a bit array specifying whether a certain alert sender / receiver pair goes across an asynchronous boundary or not. - -The next table lists free parameters in the `prim_alert_sender` and -`prim_alert receiver` submodules. - -Parameter | Default (Max) | Description ----------------|------------------|--------------- -`AsyncOn` | `1'b0` (`1'b1`) | 0: Synchronous, 1: Asynchronous, determines whether additional synchronizer flops and logic need to be instantiated. - - -### Signals - -* [Interface Tables](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#interfaces) - -The table below lists other alert handler module signals. -The number of alert instances is parametric and hence alert and ping diff pairs are grouped together in packed arrays. -The diff pair signals are indexed with the corresponding alert instance ``. - -Signal | Direction | Type | Description --------------------------|------------------|---------------- |--------------- -`edn_o` | `output` | `otp_edn_req_t` | Entropy request to the entropy distribution network for LFSR reseeding and ephemeral key derivation. -`edn_i` | `input` | `otp_edn_rsp_t` | Entropy acknowledgment to the entropy distribution network for LFSR reseeding and ephemeral key derivation. -`alert_tx_i[]` | `input` | packed `alert_tx_t` array | Incoming alert or ping response(s), differentially encoded. Index range: `[NAlerts-1:0]` -`alert_rx_o[]` | `output` | packed `alert_rx_t` array | Outgoing alert acknowledgment and ping requests, differentially encoded. Index range: `[NAlerts-1:0]` -`esc_tx_o[]` | `output` | packed `esc_tx_t` array | Escalation or ping request, differentially encoded. Index corresponds to severity level, and ranges from 0 to 3. -`esc_rx_i[]` | `input` | packed `esc_rx_t` array | Escalation ping response, differentially encoded. Index corresponds to severity level, and ranges from 0 to 3. -`lpg_cg_en_i[]` | `input` | packed `mubi4_t` array | Incoming clock gated indication from clock manager. Index range: `[NLpg-1:0]` -`lpg_rst_en_i[]` | `input` | packed `mubi4_t` array | Incoming reset asserted indication from reset manager. Index range: `[NLpg-1:0]` -`crashdump_o` | `output` | packed `struct` | This is a collection of alert handler state registers that can be latched by hardware debugging circuitry, if needed. - -#### Entropy Network Connections - -The LFSR ping timer needs to be periodically reseeded. -Therefore, the alert handler is connected to the entropy distribution network via the `edn_i/o` signals. - -#### Alert Channels - -For each alert, there is a pair of input and two pairs of output signals. -These signals are connected to a differential sender module within the source, and a differential receiver module within the alert handler. -Both of these modules are described in more detail in the following section. -These signal pairs carry differentially encoded messages that enable two types of signaling: a native alert and a ping/response test of the alert mechanism. -The latter is to ensure that all alert senders are always active and have not been the target of an attack. - -#### Escalation Channels - -For each escalation action in the system, there is a pair of input and a pair of output signals, encapsulated in the `esc_rx_t` and `esc_tx_t` types. -These signals are connected to a differential sender module within the alert handler, and a differential receiver module within the module that performs a particular escalation action (for example the reset manager or life cycle controllers). -The signal pairs carry differentially encoded messages that enable two types of signaling: a native escalation and a ping/response test of the escalation mechanism. -The latter is to ensure that all escalation receivers are always active and have not been the target of an attack. - -#### Low-power Indication Signals - -The `lpg_cg_en_i` and `lpg_rst_en_i` are two arrays with multibit indication signals from the [clock](../../ip/clkmgr/README.md) and [reset managers](../../ip/rstmgr/README.md). -These indication signals convey whether a specific group of alert senders are either clock gated or in reset. -As explained in [more detail below](#low-power-management-of-alert-channels), this information is used to temporarily halt the ping timer mechanism on channels that are in a low-power state in order to prevent false positives. - -#### Crashdump Output - -The `crashdump_o` struct outputs a snapshot of CSRs and alert handler state bits that can be read by hardware debugging circuitry: - -```systemverilog - typedef struct packed { - // alerts - logic [NAlerts-1:0] alert_cause; // alert cause bits - logic [6:0] loc_alert_cause; // local alert cause bits - // class state - logic [3:0][15:0] class_accum_cnt; // current accumulator value - logic [3:0][31:0] class_esc_cnt; // current escalation counter value - cstate_e [3:0] class_esc_state; // current escalation protocol state - } alert_crashdump_t; -``` - -This can be useful for extracting more information about possible failures or bugs without having to use the tile-link bus interface (which may become unresponsive under certain circumstances). -It is recommended for the top level to store this information in an always-on location. - -Note that the crashdump state is continuously output via `crashdump_o` until the latching trigger condition is true for the first time (see [`CLASSA_CRASHDUMP_TRIGGER_SHADOWED`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_crashdump_trigger_shadowed)). -After that, the `crashdump_o` is held constant until all classes that have escalated are cleared. -This is done so that it is possible to capture the true alert cause before spurious alert events start to pop up due to escalation countermeasures with excessive side effects (like life cycle scrapping for example). -If classes that have escalated are not configured as clearable, then it is not possible to re-arm the crashdump latching mechanism at runtime and the alert handler has to be reset. - -## Design Details - -This section gives the full design details of the alert handler module and its submodules. - - -### Alert Definition - -Alerts are defined as events that have security implications, and should be handled by the main processor, or escalated to other hardware modules to take action. -Each peripheral has the option to define one or more alert signals. -Those peripherals should instantiate one module (`prim_alert_sender`) to convert the event associated with that alert into a signal to the alert handler module. -The alert handler instantiates one receiver module (`prim_alert_receiver`) per alert, then handles the classification, accumulation, and escalation of the received signal. -The differential signaling submodules may either use a synchronous or asynchronous clocking scheme, since the message type to be transferred is a single discrete event. - - -### Differential Alert Signaling - -Each alert sender is connected to the corresponding alert receiver via the 3 differential pairs `alert_tx_i/o.alert_p/n`, `alert_rx_i/o.ack_p/n` and `alert_rx_i/o.ping_p/n`, as illustrated below: - -![Alert Handler Alert RXTX](./doc/alert_handler_alert_rxtx.svg) - -Alerts are encoded differentially and signaled using a full handshake on the `alert_tx_i/o.alert_p/n` and `alert_rx_i/o.ack_p/n` wires. -The use of a full handshake protocol allows this mechanism to be used with an asynchronous clocking strategy, where peripherals may reside in a different clock domain than the alert handler. -The full handshake guarantees that alert messages are correctly back-pressured and no alert is "lost" at the asynchronous boundary due to (possibly variable) clock ratios greater or less than 1.0. -The "native alert message" will be repeated on the output wires as long as the alert event is still true within the peripheral. - -The wave pattern below illustrates differential full handshake mechanism. - -```wavejson -{ - signal: [ - { name: 'clk_i', wave: 'p...............' }, - { name: 'alert_req_i', wave: '01.|..|..|...|..' }, - { name: 'alert_ack_o', wave: '0..|..|..|10.|..' }, - { name: 'alert_tx_o/i.alert_p', wave: '01.|..|0.|..1|..' , node: '.a.....c....e'}, - { name: 'alert_tx_o/i.alert_n', wave: '10.|..|1.|..0|..' }, - { name: 'alert_rx_i/o.ack_p', wave: '0..|1.|..|0..|1.' , node: '....b.....d..'}, - { name: 'alert_rx_i/o.ack_n', wave: '1..|0.|..|1..|0.' }, - { name: 'alert_o', wave: '0..|10|..|...|10' }, - ], - edge: [ - 'a~>b Phase 0/1', - 'b~>c Phase 1/2', - 'c~>d Phase 2/3', - 'd~>e 2 Pause Cycles', - ], - head: { - text: 'Alert signaling and repeat pattern', - }, - foot: { - text: 'Native alert at time 1 with 4-phase handshake; repeated alert at time 12;', - tick: 0, - } -} -``` - -The handshake pattern is repeated as long as the alert is true. -The sender will wait for 2 cycles between handshakes. - -Note that the alert is immediately propagated to `alert_o` once the initial level change on `alert_tx_i.alert_p/n` has been received and synchronized to the local clock on the receiver side. -This ensures that the first occurrence of an alert is always propagated - even if the handshake lines have been manipulated to emulate backpressure. -(In such a scenario, all subsequent alerts would be back-pressured and eventually the ping testing mechanism described in the next subsection would detect that the wires have been tampered with.) - -The alert sender and receiver modules can either be used synchronously or asynchronously. -The signaling protocol remains the same in both cases, but the additional synchronizer flops at the diff pair inputs may be omitted, which results in lower signaling latency. - -### Ping Testing - -In order to ensure that the event sending modules have not been compromised, the alert receiver module `prim_alert_receiver` will "ping" or line-test the senders periodically every few microseconds. -Pings timing is randomized so their appearance can not be predicted. - - -The ping timing is generated by a central LFSR-based timer within the alert handler that randomly asserts the `ping_req_i` signal of a particular `prim_alert_receiver` module. -Once `ping_req_i` is asserted, the receiver module encodes the ping message as a level change on the differential `alert_rx_o.ping_p/n` output, and waits until the sender responds with a full handshake on the `alert_tx_i.alert_p/n` and `alert_rx_o.ack_p/n` lines. -Once that handshake is complete, the `ping_ok_o` signal is asserted. -The LFSR timer has a programmable ping timeout, after which it will automatically assert a "pingfail" alert. -That timeout is a function of the clock ratios present in the system, and has to be programmed accordingly at system startup (as explained later in the LFSR timer subsection). - -The following wave diagram illustrates a correct ping sequence, viewed from the receiver side: - -```wavejson -{ - signal: [ - { name: 'clk_i', wave: 'p..............' }, - { name: 'ping_req_i', wave: '01.|..|..|..|.0' }, - { name: 'ping_ok_o', wave: '0..|..|..|..|10' , node: '.............e'}, - { name: 'alert_rx_o.ping_p', wave: '01.|..|..|..|..' , node: '.a'}, - { name: 'alert_rx_o.ping_n', wave: '10.|..|..|..|..' , node: '.b'}, - { name: 'alert_tx_i.alert_p', wave: '0..|1.|..|0.|..' , node: '....c'}, - { name: 'alert_tx_i.alert_n', wave: '1..|0.|..|1.|..' }, - { name: 'alert_rx_o.ack_p', wave: '0..|..|1.|..|0.' , node: '.............d'}, - { name: 'alert_rx_o.ack_n', wave: '1..|..|0.|..|1.' }, - ], - edge: [ - 'a-b', - 'b~>c ping response', - 'd->e response complete', - ], - head: { - text: 'Ping testing', - }, - foot: { - text: 'Level change at time 1 triggers a full handshake (ping response) at time 4', - tick: 0, - } -} -``` - -In the unlikely case that a ping request collides with a native alert at the sender side, the native alert is held back until the ping handshake has been completed. -This slightly delays the transmission of a native alert, but the alert will eventually be signaled. -Further, if an alert is sent out right before a ping requests comes in at the sender side, the receiver will treat the alert as a ping response. -However, the "true" ping response will be returned right after the alert handshake completed, and thus the alert will eventually be signaled with a slight delay. - -Note that in both collision cases mentioned, the delay will be in the order of the handshake length, plus the constant amount of pause cycles between handshakes (2 sender cycles). - - -### Monitoring of Signal Integrity Issues - -All differential pairs are monitored for signal integrity issues, and if an encoding failure is detected, the receiver module asserts a signal integrity alert via `integ_fail_o`. In particular, this covers the following failure cases: - -1. The `alert_tx_i.alert_p/n` pair is not correctly encoded on the receiver side. -This can be directly flagged as an integrity failure on the receiver side. - -2. The `alert_rx_i.ping_p/n` or the `alert_rx_i.ack_p/n` pairs are not correctly encoded on the sender side. -This is signaled to the receiver by setting the `alert_tx_o.alert_p/n` wires to the same value, and that value will be continuously toggled. -This implicitly triggers a signal integrity alert on the receiver side. - -Some of these failure patterns are illustrated in the wave diagram below: - -```wavejson -{ - signal: [ - { name: 'clk_i', wave: 'p..............' }, - { name: 'alert_tx_o.alert_p', wave: '0.1...|0..10101' , node: '..a.......d'}, - { name: 'alert_tx_o.alert_n', wave: '1.....|....0101' }, - { name: 'alert_rx_i.ack_p', wave: '0.....|.1......' , node: '........c'}, - { name: 'alert_rx_i.ack_n', wave: '1.....|........' }, - { name: 'integ_fail_o', wave: '0...1.|0....1..' , node: '....b.......e'}, - ], - edge: [ - 'a~>b sigint issue detected', - 'c~>d', - 'd~>e indirect sigint issue detected', - ], - head: { - text: 'Detection of Signal Integrity Issues', - }, - foot: { - text: 'signal integrity issues occur at times 2 and 8; synchronizer latency is 2 cycles.', - tick: 0, - } -} -``` - -Note that if signal integrity failures occur during ping or alert handshaking, it is possible that the protocol state-machines lock up and the alert sender and receiver modules become unresponsive. However, the above mechanisms ensure that this will always trigger either a signal integrity alert or eventually a "pingfail" alert. - -### Skew on Asynchronous Differential Pairs - -Note that there is likely a (small) skew present within each differential pair of the signaling mechanism above. Since these pairs cross clock domain boundaries, it may thus happen that a level change appears in staggered manner after resynchronization, as illustrated below: - -```wavejson -{ - signal: [ - { name: 'clk_i', wave: 'p...........' }, - { name: 'diff_p', wave: '0.1.|.0.|..1' , node: '......a....d' }, - { name: 'diff_n', wave: '1.0.|..1|.0.' , node: '.......b..c.' }, - ], - edge: [ - 'a-~>b skew', - 'c-~>d skew' - ], - head: { - text: 'Skewed diff pair', - }, - foot: { - text: 'Correctly sampled diff pair at time 2; staggered samples at time 6-7 and 10-11', - tick: 0, - } -} -``` - -This behavior is permissible, but needs to be accounted for in the protocol logic. -Further, the skew within the differential pair should be constrained to be smaller than the shortest clock period in the system. -This ensures that the staggered level changes appear at most 1 cycle apart from each other. - - -### LFSR Timer - -The `ping_req_i` inputs of all signaling modules (`prim_alert_receiver`, `prim_esc_sender`) instantiated within the alert handler are connected to a central ping timer that alternatingly pings either an alert line or an escalation line after waiting for a pseudo-random amount of clock cycles. -Further, this ping timer also randomly selects a particular alert line to be pinged (escalation senders are always pinged in-order due to the [ping monitoring mechanism](#monitoring-of-pings-at-the-escalation-receiver-side) on the escalation side). -That should make it more difficult to predict the next ping occurrence based on past observations. - -The ping timer is implemented using an [LFSR-based PRNG of Galois type](../../ip/prim/doc/prim_lfsr.md). -This ping timer is reseeded with fresh entropy from EDN roughly every 500k cycles which corresponds to around 16 ping operations on average. -The LFSR is 32bits wide, but only 24bits of its state are actually being used to generate the random timer count and select the alert line to be pinged. -I.e., the 32bits first go through a fixed permutation function, and then bits `[23:16]` are used to determine which alert line to ping. -The random cycle count is created by OR'ing bits `[15:0]` with the constant `3'b100` as follows: - -``` -cycle_cnt = permuted[15:0] | 3'b100; -``` - -This constant DC offset introduces a minimum ping spacing of 4 cycles (1 cycle + margin) to ensure that the handshake protocols of the sender/receiver pairs work. - -After selecting one of the peripherals to ping, the LFSR timer waits until either the corresponding `*_ping_ok[]` signal is asserted, or until the programmable ping timeout value is reached. -In both cases, the LFSR timer proceeds with the next ping, but in the second case it will additionally raise a "pingfail" alert. -The ping enable signal remains asserted during the time where the LFSR counter waits. - -The timeout value is a function of the ratios between the alert handler clock and peripheral clocks present in the system, and can be programmed at startup time via the register [`PING_TIMEOUT_CYC_SHADOWED`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#ping_timeout_cyc_shadowed). - -Note that the ping timer directly flags a "pingfail" alert if a spurious "ping ok" message comes in that has not been requested. - - -As described in the programmers guide below, the ping timer has to be enabled explicitly. -Only alerts that have been *enabled and locked* will be pinged in order to avoid spurious alerts. -Escalation channels are always enabled, and hence will always be pinged once this mechanism has been turned on. - -In addition to the ping timer mechanism described above, the escalation receivers contain monitoring counters that monitor the liveness of the alert handler (described in more detail in [this section](#monitoring-of-pings-at-the-escalation-receiver-side). -This mechanism requires that the maximum wait time between escalation receiver pings is bounded. -To that end, escalation senders are pinged in-order every second ping operation (i.e., the wait time is randomized, but the selection of the escalation line is not). - -### Alert Receiving - -The alert handler module contains one alert receiver module (`prim_alert_receiver`) per sending module. -This receiver module has three outputs based upon the signaling of the input alert. -Primary is the signal of a received native alert, shown in the top-level diagram as `alert_triggered[]`. -Also generated are two other outputs, one that signals a differential encoding error (`alert_integ_fail[]`), and one that signals the receipt of a ping response (`alert_ping_ok[]`). -Each "triggered" alert received is sent into the classification block for individual configuration. -All of the `integ_fail` signals are OR'ed together to create one alert for classification. -The ping responses are fed to the LFSR timer, which determines whether a ping has correctly completed within the timeout window or not. - - -### Alert Classification and Interrupts - -Each of the incoming and local alert signals can be classified generically to one of four classes, or disabled for no classification at all. -These are the classes A, B, C, and D. -There is no pre-determined definition of a class, that is left to software. -But for guidance, software can consider that some alert types are similar to others; some alert types are more "noisy" than others (i.e. when triggered they stay on for long periods of time); some are more critical than others, etc. - -For each alert class (A-D), an interrupt is generally sent. -Like all other peripheral interrupts, there is a triad of registers: enable, status, test. -Thus like all other interrupts, software should handle the source of the interrupt (in this case, the original alert), then clear the state. -Since the interrupt class is disassociated with the original alert (due to the classification process), software can access cause registers to determine which alerts have fired since the last clearing. -Since alerts are expected to be rare (if ever) events, the complexity of dealing with multiple interrupts per class firing during the same time period should not be of concern. See the programming section on interrupt clearing. - -Each of the four interrupts can optionally trigger a timeout counter that triggers escalation if the interrupt is not handled and cleared within a certain time frame. -This feature is explained in more detail in the next subsection about escalation mechanisms. - -Note that an interrupt always fires once an alert has been registered in the corresponding class. -Interrupts are not dependent on escalation mechanisms like alert accumulation or timeout as described in the next subsection. - - -### Escalation Mechanisms - -There are two mechanisms per class that can trigger the corresponding escalation -protocol: - -1. The first consists of an accumulation counter that counts the amount of alert occurrences within a particular class. - An alert classified to class A indicates that on every received alert trigger, the accumulation counter for class A is incremented. - Note: since alerts are expected to be rare or never occur, the module does not attempt to count every alert per cycle, but rather all triggers per class are ORd before sending to the accumulation counter as an increment signal. - Once the threshold has been reached, the next occurrence triggers the escalation escalation protocol for this particular class. - The counter is a saturation counter, meaning that it will not wrap around once it hits the maximum representable count. - This mechanism has two associated CSRs: - - - Accumulation max value. - This is the total number (sum of all alerts classified in this group) of alerts required to enter escalation phase (see below). - Example register is [`CLASSA_ACCUM_THRESH_SHADOWED`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_accum_thresh_shadowed). - - Current accumulation register. - This clearable register indicates how many alerts have been accumulated to date. - Software should clear before it reaches the accumulation setting to avoid escalation. - Example register is [`CLASSA_ACCUM_CNT`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_accum_cnt). - -2. The second way is an interrupt timeout counter which triggers escalation if an alert interrupt is not handled within the programmable timeout window. - Once the counter hits the timeout threshold, the escalation protocol is triggered. - The corresponding CSRs are: - - - Interrupt timeout value in cycles [`CLASSA_TIMEOUT_CYC_SHADOWED`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_timeout_cyc_shadowed). - The interrupt timeout is disabled if this is set to 0 (default). - - The current interrupt timeout value can be read via [`CLASSA_ESC_CNT`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_esc_cnt) if [`CLASSA_STATE`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_state) is in the `Timeout` state. - Software should clear the corresponding interrupt state bit [`INTR_STATE.CLASSA`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#intr_state) before the timeout expires to avoid escalation. - -Technically, the interrupt timeout feature (2. above) is implemented using the same counter used to time the escalation phases. -This is possible since escalation phases or interrupt timeout periods are non-overlapping (escalation always takes precedence should it be triggered). - - -### Programmable Escalation Protocol - -There are four output escalation signals, 0, 1, 2, and 3. -There is no predetermined definition of an escalation signal, that is left to the top-level integration. -Examples could be processor Non Maskable Interrupt (NMI), privilege lowering, secret wiping, chip reset, etc. -Typically the assumption is that escalation level 0 is the first to trigger, followed by 1, 2, and then 3, emulating a "fuse" that is lit that can't be stopped once the first triggers (this is however not a requirement). -See register section for discussion of counter clearing and register locking to determine the finality of accumulation -triggers. - -Each class can be programmed with its own escalation protocol. -If one of the two mechanisms described above fires, a timer for that particular class is started. -The timer can be programmed with up to 4 delays (e.g., [`CLASSA_PHASE0_CYC`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_phase0_cyc)), each representing a distinct escalation phase (0 - 3). -Each of the four escalation severity outputs (0 - 3) are by default configured to be asserted during the corresponding phase, e.g., severity 0 in phase 0, severity 1 in phase 1, etc. -However, this mapping can be freely reassigned by modifying the corresponding enable/phase mappings (e.g., [`CLASSA_CTRL_SHADOWED.E0_MAP`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_ctrl_shadowed) for enable bit 0 of class A). -This mapping will be locked in together with the alert enable configuration after initial configuration. - -SW can stop a triggered escalation protocol by clearing the corresponding escalation counter (e.g., [`CLASSA_ESC_CNT`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_esc_cnt)). -Protection of this clearing is up to software, see the register control section that follows for [`CLASSA_CTRL_SHADOWED.LOCK`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_ctrl_shadowed). - -It should be noted that each of the escalation phases have a duration of at least 1 clock cycle, even if the cycle count of a particular phase has been -set to 0. - -The next waveform shows the gathering of alerts of one class until eventually the escalation protocol is engaged. -In this diagram, two different alerts are shown for class A, and the gathering and escalation configuration values are shown. - -```wavejson -{ - signal: [ - { name: 'clk_i', wave: 'p...................' }, - { name: 'CLASSA_ACCUM_THRESH_SHADOWED', wave: '2...................', data: ['15'] }, - { name: 'CLASSA_PHASE0_CYC_SHADOWED', wave: '2...................', data: ['1e3 cycles'] }, - { name: 'CLASSA_PHASE1_CYC_SHADOWED', wave: '2...................', data: ['1e4 cycles'] }, - { name: 'CLASSA_PHASE2_CYC_SHADOWED', wave: '2...................', data: ['1e5 cycles'] }, - { name: 'CLASSA_PHASE3_CYC_SHADOWED', wave: '2...................', data: ['1e6 cycles'] }, - { name: 'alert_triggered[0]', wave: '010|.10.............' }, - { name: 'alert_triggered[1]', wave: '0..|10..............' }, - { name: 'CLASSA_ACCUM_CNT', wave: '33.|33..............', data: ['0', '1','15','16'] }, - { name: 'irq_o[0]', wave: '01.|................' }, - { name: 'CLASSA_STATE', wave: '3..|.3|3.|3..|3..|3.', data: ['Idle', ' Phase0','Phase1','Phase2','Phase3','Terminal'] }, - { name: 'CLASSA_ESC_CNT', wave: '3..|.3|33|333|333|3.', data: ['0','1','1','2','1','2','3','1','2','3','0'] }, - { name: 'esc_tx_o.esc_p[0]', wave: '0..|.1|.0...........', node: '.....a..b' }, - { name: 'esc_tx_o.esc_n[0]', wave: '1..|.0|.1...........' }, - { name: 'esc_tx_o.esc_p[1]', wave: '0..|..|1.|.0........', node: '.......c...d' }, - { name: 'esc_tx_o.esc_n[1]', wave: '1..|..|0.|.1........' }, - { name: 'esc_tx_o.esc_p[2]', wave: '0..|.....|1..|.0....', node: '..........e....f' }, - { name: 'esc_tx_o.esc_n[2]', wave: '1..|.....|0..|.1....' }, - { name: 'esc_tx_o.esc_p[3]', wave: '0..|.........|1..|.0', node: '..............g....h' }, - { name: 'esc_tx_o.esc_n[3]', wave: '1..|.........|0..|.1' }, - ], - edge: [ - 'a->b 1e3 + 1 cycles', - 'c->d 1e4 + 1 cycles', - 'e->f 1e5 + 1 cycles', - 'g->h 1e6 + 1 cycles', - ], - head: { - text: 'Alert class gathering and escalation triggers (fully synchronous case)', - }, - foot: { - text: 'alert class A gathers 16 alerts, triggers first escalation, followed by three more', - tick: 0, - } -} -``` - -In this diagram, the first alert triggers an interrupt to class A. -The assumption is that the processor is wedged or taken over, in which case it does not handle the interrupt. -Once enough interrupts gather (16 in this case), the first escalation phase is entered, followed by three more (each phase has its own programmable length). -Note that the accumulator threshold is set to 15 in order to trigger on the 16th occurrence. -If escalation shall be triggered on the first occurrence within an alert class, the accumulation threshold shall be set to 0. -Also note that it takes one cycle to activate escalation and enter phase 0. - -The next wave shows a case where an interrupt remains unhandled and hence the interrupt timeout counter triggers escalation. - -```wavejson -{ - signal: [ - { name: 'clk_i', wave: 'p.....................' }, - { name: 'CLASSA_TIMEOUT_CYC_SHADOWED', wave: '2.....................', data: ['1e4 cycles'] }, - { name: 'alert_triggered[0]', wave: '010.|.................' }, - { name: 'irq_o[0]', wave: '01..|.................', node: '.a..|.b' }, - { name: 'CLASSA_ESC_STATE', wave: '33..|.3|3.|3..|3...|3.', data: ['Idle', 'Timeout',' Phase0','Phase1','Phase2','Phase3','Terminal'] }, - { name: 'CLASSA_ESC_CNT', wave: '3333|33|33|333|3333|3.', data: ['0', '1','2','3','1e4','1','1','2','1','2','3','1','2','3','4','0'] }, - { name: 'esc_tx_o.esc_p[0]', wave: '0...|.1|.0............' }, - { name: 'esc_tx_o.esc_n[0]', wave: '1...|.0|.1............' }, - { name: 'esc_tx_o.esc_p[1]', wave: '0...|..|1.|.0.........' }, - { name: 'esc_tx_o.esc_n[1]', wave: '1...|..|0.|.1.........' }, - { name: 'esc_tx_o.esc_p[2]', wave: '0...|.....|1..|.0.....' }, - { name: 'esc_tx_o.esc_n[2]', wave: '1...|.....|0..|.1.....' }, - { name: 'esc_tx_o.esc_p[3]', wave: '0...|.........|1...|.0' }, - { name: 'esc_tx_o.esc_n[3]', wave: '1...|.........|0...|.1' }, - ], - edge: [ - 'a->b 1e4 cycles', - ], - head: { - text: 'Escalation due to an interrupt timeout (fully synchronous case)', - }, - foot: { - text: 'alert class A triggers an interrupt and the timeout counter, which eventually triggers escalation after 1e4 cycles.', - tick: 0, - } -} -``` - -It should be noted here that the differential escalation signaling protocol distinguishes 'true' escalation conditions from mere pings by encoding them as pulses that are N + 1 cycles long. -This is reflected in the two wave diagrams above. -Refer to the subsequent section on escalation signaling for more details. - -### Escalation Signaling - -For each of the four escalation severities, the alert handler instantiates a `prim_esc_sender` module and each of the four escalation countermeasures instantiates an `prim_esc_receiver` module. -The signaling mechanism has similarities with the alert signaling mechanism - but it is a fully synchronous protocol. -Hence, it must be ensured at the top-level that all escalation sender and receiver modules are using the same clock and reset -signals. - -As illustrated in the following block diagram, a sender-receiver pair is connected with two differential lines, one going from sender to receiver and the other going from receiver to sender. - -![Alert Handler Escalation RXTX](./doc/alert_handler_escalation_rxtx.svg) - -Upon receiving an escalation enable pulse of width N > 0 at the `esc_req_i` input, the escalation sender encodes that signal as a differential pulse of width N+1 on `esc_tx.esc_p/n`. -The receiver decodes that message and asserts the `esc_req_o` output after one cycle of delay. -Further, it acknowledges the receipt of that message by continuously toggling the `esc_rx.resp_p/n` signals as long as the escalation signal is asserted. -Any failure to respond correctly will trigger a `integ_fail_o` alert, as illustrated below: - -```wavejson -{ - signal: [ - { name: 'clk_i', wave: 'p..................' }, - { name: 'ping_req_i', wave: '0........|.........' }, - { name: 'ping_ok_o', wave: '0........|.........' }, - { name: 'integ_fail_o', wave: '0........|..1010...' , node: '............b.d' }, - { name: 'ping_fail_o', wave: '0........|.........' }, - { name: 'esc_req_i', wave: '01....0..|.1....0..' }, - { name: 'esc_rx_i/o.resp_p', wave: '0.101010.|.........', node: '............a.c' }, - { name: 'esc_rx_i/o.resp_n', wave: '1.010101.|.........' }, - { name: 'esc_tx_o/i.esc_p', wave: '01.....0.|.1.....0.' }, - { name: 'esc_tx_o/i.esc_n', wave: '10.....1.|.0.....1.' }, - { name: 'esc_req_o', wave: '0.1....0.|..?....0.'}, - ], - edge: [ - 'a~>b missing response', - 'c~>d', - ], - head: { - text: 'Escalation signaling and response', - }, - foot: { - text: 'escalation enable pulse shown at input sender at time 1 and 11; missing response and repeated integfail at time 12 and 14', - tick: 0, - } -} -``` - -Further, any differential signal mismatch on both the `esc_tx_i.esc_p/n` and `esc_rx_i.resp_p/n` lines will trigger an `integ_fail_o` alert. -Mismatches on `esc_rx_i.resp_p/n` can be directly detected at the sender. -Mismatches on the `esc_tx_i.esc_p/n` line will be signaled back to the sender by setting both the positive and negative response wires to the same value - and that value is being toggled each cycle. -This implicitly triggers a signal integrity alert on the sender side. -In addition to that, a signal integrity error on the `esc_tx_i.esc_p/n` lines will lead to assertion of the `esc_req_o` output, since it cannot be guaranteed that the back signalling mechanism always works when the sender / receiver pair is being tampered with. - -This back-signaling mechanism can be leveraged to fast-track escalation and use another countermeasure in case it is detected that a particular escalation signaling path has been tampered with. - -Some signal integrity failure cases are illustrated in the wave diagram below: - -```wavejson -{ - signal: [ - { name: 'clk_i', wave: 'p...........' }, - { name: 'ping_req_i', wave: '0....|......' }, - { name: 'ping_ok_o', wave: '0....|......' }, - { name: 'integ_fail_o', wave: '0.1.0|.1....' , node: '..b....e' }, - { name: 'esc_req_i', wave: '0....|......' }, - { name: 'esc_rx_i.resp_p', wave: '0.1.0|..1010', node: '..a..' }, - { name: 'esc_rx_i.resp_n', wave: '1....|.01010', node: '.......d' }, - { name: 'esc_tx_i.esc_p', wave: '0....|1.....', node: '......c..' }, - { name: 'esc_tx_i.esc_n', wave: '1....|......' }, - { name: 'esc_req_o', wave: '0....|1.....'}, - ], - edge: [ - 'a~>b', - 'c->d', - 'd->e', - ], - head: { - text: 'possible signal integrity failure cases', - }, - foot: { - text: 'direct signal integrity failure at time 2; indirect failure at time 6', - tick: 0, - } -} -``` - - -### Ping Testing of the Escalation Signals - - -Similarly to the alert signaling scheme, the escalation signaling lines can be pinged / line tested in order to test whether the escalation receiver has been tampered with. -This is achieved by asserting `ping_req_i` at the escalation sender module. -A ping request is encoded as a single cycle pulse on the `esc_tx_o.esc_p/n` outputs. -Hence, the receiver module will not decode this single cycle pulse as an escalation enable message, but it will respond to it with a "1010" pattern on the `esc_rx_i.resp_p/n` lines. -The escalation sender module will assert `ping_ok_o` if that pattern is received correctly after one cycle of latency. -Otherwise, the escalation sender will first assert `integ_fail_o` later, after the programmable ping timeout is reached, the LFSR timer will raise a "pingfail" alert. -The `integ_fail_o` triggers in this case since "no ping response" and "wrong ping response" are ambiguous in this setting, and it has been decided to not suppress integrity failures when expecting a ping response. - -This mechanism is illustrated below from the viewpoint of the sender module. - -```wavejson -{ - signal: [ - { name: 'clk_i', wave: 'p..............' }, - { name: 'ping_req_i', wave: '01....0|.1.....' , node: '.a'}, - { name: 'ping_ok_o', wave: '0....10|.......' , node: '.....e....g'}, - { name: 'integ_fail_o', wave: '0......|..10101' }, - { name: 'esc_req_i', wave: '0......|.......' }, - { name: 'esc_rx_i.resp_p', wave: '0.1010.|.......' , node: '..c..d....f'}, - { name: 'esc_rx_i.resp_n', wave: '1.0101.|.......' }, - { name: 'esc_tx_o.esc_p', wave: '010....|.10....' , node: '.b'}, - { name: 'esc_tx_o.esc_n', wave: '101....|.01....' }, - ], - edge: [ - 'a->b', - 'b->c', - 'd->e correct response', - 'f->g missing response', - ], - head: { - text: 'ping testing of escalation lines', - }, - foot: { - text: 'ping trig at sender input at time 1 and 9; correct response at time 5; missing response at time 10', - tick: 0, - } -} -``` - -Note that the escalation signal always takes precedence, and the `ping_req_i` will just be acknowledged with `ping_ok_o` in case `esc_req_i` is already asserted. -An ongoing ping sequence will be aborted immediately. - -Another thing to note is that the ping and escalation response sequences have to start _exactly_ one cycle after either a ping or escalation event has been signaled. -Otherwise the escalation sender will assert `integ_fail_o` immediately. - -### Monitoring of Pings at the Escalation Receiver Side - -Escalation receivers contain a mechanism to monitor the liveness of the alert handler itself. -In particular, the receivers passively monitor the ping requests sent out by the alert handler using a timeout counter. -If ping requests are absent for too long, the corresponding escalation action will be automatically asserted until reset. - -The monitoring mechanism builds on top of the following properties of the alert handler system: -1. the ping mechanism can only be enabled, but not disabled. -This allows us to start the timeout counter once the first ping request arrives at a particular escalation receiver. - -2. the escalation receivers are in the same clock/reset domain as the alert handler. -This ensures that we are seeing the same clock frequency, and the mechanism is properly reset together with the alert handler logic. - -3. the maximum cycle count between subsequent pings on the same escalation line is bounded, even though the wait counts are randomized. -This allows us to compute a safe and fixed timeout threshold based on design constants. - - -### Low-power Management of Alert Channels - -Due to the various clock and reset domains in the OpenTitan system, the alert handler ping mechanism needs to have additional logic to deal with alert senders that are either held in reset, or that are clock gated. -This is needed to ensure that no false alarms are produced by the ping mechanism when an alert channel (sender / receiver pair) does not respond due to the sender being either in reset or clock gated. - -Since the FSMs associated with an alert channel may end up in an inconsistent state when the sender is reset or gated while an asynchronous event handshake is in progress, this logic also needs to be able to re-initialize affected alert channels whenever the channels comes back from reset / clock gated state. - -#### Assumptions - -The following diagram shows a typical arrangement of alert sender (TX) and receiver (RX) pairs. - -![Alert Handler Low-Power Overview](./doc/alert_handler_lp_overview.svg) - -It is assumed that: - -1. The alert handler clock domain cannot be gated by SW. - This means that this clock domain is only ever disabled as part of the power-down sequence of the corresponding power domain. -2. The alert senders are in general located within different clock and reset domains than the alert receivers within the alert handler, and thus use the asynchronous event handshake mode. -3. Some alert senders may be located in an always-on (AON) power domain, within different clock and reset groups than the alert handler. -4. The alert handler may be located in an non-AON power domain, and may thus undergo a reset cycle where it cannot be guaranteed that all alert senders are reset as well (i.e., some alert senders may retain their state). - -Further, we assume that we can get the following side-band information from the clock and reset managers in the system: - -- All relevant reset signals pertaining to alert sender domains -- All relevant clock enable signals pertaining to alert sender domains - -#### Scenarios - -With the assumptions above, the following two problematic scenarios can occur. - -##### Alert Handler in Reset - -It may happen that the alert handler is reset while some alert senders (e.g. those located in the AON domain) are not. -In general, if the associated alert channels are idle during an alert handler reset cycle, no problems arise. - -However, if an alert channel is reset while it is handling a ping request or an alert event, the sender / receiver FSMs may end up in an inconsistent state upon deassertion of the alert handler reset. -This can either lead to spurious alert or ping events, or a completely locked up alert channel which will be flagged eventually by the ping mechanism. - -##### Alert Sender in Reset or Clock-gated - -If any of the alert senders is either put into reset or its clock is disabled while the alert handler is operational, the ping mechanism inside the alert handler will eventually report a ping failure because of missing ping responses from the affected alert channel(s). - -Further, if the alert sender is reset while the corresponding alert channel is handling a ping request or an alert event, the sender / receiver FSMs may end up in an inconsistent state after reset deassertion. - -#### Employed Solution - -As elaborated before, the side effects of resetting / clock gating either the alert handler or any of the alert senders are inconsistent FSM states, leading to locked up alert channels, or spurious alert or ping events. -To address these issues, we have to: - -1. make sure spurious events (alert and ping_ok outputs of the alert receivers) are suppressed if an alert channel is clock gated or in reset, -2. provide a mechanism for resetting an alert channel to an operational state once the associated clock is re-enabled, or the associated reset is released, -3. trigger this reset mechanism on all alert channels whenever the alert handler itself has been reset. - -To attain this, the idea is to make use of side-band information available from the clock and reset managers to detect whether an alert channel (or a group of alert channels with the same clock and reset on the sender side) has to be put into a low-power state. -In the following we will refer to such a clock / reset domain grouping as a low-power group (LPG). - -The mechanism is illustrated below for a single LPG (in practice, this logic is replicated for each LPG that is identified in the system): - -![Alert Handler LPG Ctrl](./doc/alert_handler_lpg_ctrl.svg) - -The clock gating enable (`lpg_cg_en`) and reset enable (`lpg_rst_en`) indications are routed as multibit signals to the alert handler, where they are synchronized to the alert handler clock and logically combined using an OR function to form a combined low-power indication signal that is multibit encoded. - -This multibit indication signal is then routed to all alert receivers, where it is used to trigger re-initialization of each alert channel, and bypass the ping requests from the ping mechanism. - -To that end, two extra *init states* are added to the alert receiver FSMs to perform this in-band reset, as indicated in the state diagram below: - -![Alert Handler Receiver FSM](./doc/alert_handler_receiver_fsm.svg) - -Whenever the `init_trig` multibit signal of an LPG is asserted, all corresponding sender FSMs are moved into the `InitReq` state. -In that state, the alert receivers immediately acknowledge ping requests from the ping mechanism, and ignore alert events from the sender side. -In addition to that, the receivers intentionally place a signal integrity error onto the `ping_p` / `ping_n`, `ack_p` / `ack_n` lines going from receivers to the senders. -This causes the senders to 1) move into the signal integrity error state, and 2) respond by placing a signal integrity error onto the `alert_p` / `alert_n` lines, which serves as an initialization "acknowledge" signal in this case. -Since the sender FSMs fall back into the `Idle` state once the signal integrity error disappears, this procedure essentially implements an in-band reset mechanism with an acknowledgement handshake that can be used to determine whether the reset has been successful. - -#### Implementation Aspects - -##### Ping Mechanism Bypass - -Note that the ping bypass mechanism is to be implemented in a way that pings are only ack'ed immediately if 1) the FSM is in the `InitReq` state, and 2) the `init_trig` signal is still asserted. - -This allows to subject the initialization process of each alert channel to the ping mechanism for channels that are recovering from a reset or clock gated cycle on the sender side. -I.e., alert channels that get stuck during the initialization process can be detected by the ping mechanism since ping requests are not immediately ack'ed anymore once `init_trig` is deasserted. - -##### FSM Encoding - -Since there are many alert channels in the design, the receiver and sender FSMs themselves are not sparsely encoded. -Instead, we rely on the ping mechanism to detect alert channels that are in a bad state. -The specific implementation of the ping bypass mentioned in the previous subsection ensures that the ping mechanism can also be used to monitor the initialization sequence of alert channels. - -##### Latency / Skew Considerations - -Due to asynchronous transitions and different path latencies in the system, a change in reset or clock gating state will experience a different latency through the alert channels than through the indication signals (`rst_n` and `clk_en`) that are connected to the low-power control logic. - -It is consequently possible for a group of alert senders to already be in reset or clock gated state, while the corresponding LPG logic does not yet know about this state change - and vice versa. - -In practice, this means that ping requests may be pending for several cycles until the LPG logic detects a reset or clock-gated condition and disables the corresponding alert channel(s). -Fortunately, such delay can be tolerated by setting the ping timeout to a sufficiently large value (see [`CLASSA_TIMEOUT_CYC_SHADOWED`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_timeout_cyc_shadowed)). - -As for alert events, this latency difference should not pose a problem. -Alert events may get stuck in the alert sender due to a reset or clock-gated condition - but this is to be expected. - -##### Integration Considerations - -Note that due to the aforementioned latency tolerance built into the ping timer, it is permissible to connect **any** reset or clock enable indication signal from the relevant clock group to the LPG logic. -I.e., the only requirement is that the indication signals are logically related to the resets and clocks routed to the alert senders, and that the skew between reset / clock state changes and the indication signals is bounded. - -The topgen script is extended so that it can identify all LPGs and the associated alert channels. -This information is then used to parameterize the alert handler design, and make the necessary top-level connections from the reset and clock management controllers to the alert handler. - -### Hardening Against Glitch Attacks - -In addition to the differential alert and escalation signalling scheme, the internal state machines and counters are hardened against glitch attacks as described bellow: - -1. Ping Timer: - - The FSM is sparsely encoded. - - The LFSR and the counter are duplicated. - - If the FSM or counter are glitched into an invalid state, all internal ping fail alerts will be permanently asserted. - -2. Escalation Timers: - - The escalation timer FSMs are sparsely encoded. - - The escalation timer counters are duplicated. - - The escalation accumulators are duplicated. - - If one of these FSMs, counters or accumulators are glitched into an invalid state, all escalation actions will be triggered and the affected FSM goes into a terminal `FsmError` state. - -3. CSRs: - - Critical configuration CSRs are shadowed. - - The shadow CSRs can trigger additional internal alerts for CSR storage and update failures. - These internal alerts are fed back into the alert classifier in the same manner as the ping and integrity failure alerts. - -4. LPGs: - - Clock-gated and reset-asserted indication signals that are routed from clock and reset managers to the alert handler are encoded with multibit signals. - -# Programmers Guide - - -## Power-up and Reset Considerations - -False alerts during power-up and reset are not possible since the alerts are disabled by default, and need to be configured and locked in by the firmware. -The ping timer won't start until initial configuration is over and the registers are locked in. - -The low-power state management of alert channels is handled entirely by hardware and hence this is transparent to software. -Note however that the LPGs inherit the security properties of the associated clock groups and resets. -This means that the low-power state of certain alerts can be controlled by SW by means of clock gating or block reset. -For example, certain crypto blocks are located in a transactional clock group which can be clock gated by SW - and this also affects the associated alerts of these crypto blocks. -See [clock](../../ip/clkmgr/README.md) and [reset managers](../../ip/rstmgr/README.md) for more detail. - - -## Initialization - -To initialize the block, software running at a high privilege levels (early in the security settings process) should do the following: - -1. For each alert and each local alert: - - - Determine if alert is enabled (should only be false if alert is known to be faulty). - Set [`ALERT_EN_SHADOWED_0.EN_A_0`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#alert_en_shadowed_0) and [`LOC_ALERT_EN_SHADOWED_0.EN_LA_0`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#loc_alert_en_shadowed_0) accordingly. - - - Determine which class (A..D) the alert is associated with. - Set [`ALERT_CLASS_SHADOWED_0.CLASS_A_0`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#alert_class_shadowed_0) and [`LOC_ALERT_CLASS_SHADOWED_0.CLASS_LA_0`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#loc_alert_class_shadowed_0) accordingly. - - - Optionally lock each alert configuration by writing 0 to [`ALERT_REGWEN_0.EN_0`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#alert_regwen_0) or [`LOC_ALERT_REGWEN_0.EN_0`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#loc_alert_regwen_0). - Note however that only **locked and enabled** alerts are going to be pinged using the ping mechanism. - This ensures that spurious ping failures cannot occur when previously enabled alerts are being disabled again (before locking). - - -2. Set the ping timeout value [`PING_TIMEOUT_CYC_SHADOWED`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#ping_timeout_cyc_shadowed). - This value is dependent on the clock ratios present in the system. - -3. For each class (A..D): - - - Determine whether to enable escalation mechanisms (accumulation / interrupt timeout) for this particular class. Set [`CLASSA_CTRL_SHADOWED.EN`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_ctrl_shadowed) accordingly. - - - Determine if this class of alerts allows clearing of escalation once it has begun. - Set [`CLASSA_CTRL_SHADOWED.LOCK`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_ctrl_shadowed) to true if clearing should be disabled. - If true, once escalation protocol begins, it can not be stopped, the assumption being that it ends in a chip reset else it will be rendered useless thenceforth. - - - Determine the number of alerts required to be accumulated before escalation protocol kicks in. Set [`CLASSA_ACCUM_THRESH`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_accum_thresh) accordingly. - - - Determine whether the interrupt associated with that class needs a timeout. - If yes, set [`CLASSA_TIMEOUT_CYC_SHADOWED`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_timeout_cyc_shadowed) to an appropriate value greater than zero (zero corresponds to an infinite timeout and disables the mechanism). - - - For each escalation phase (0..3): - - Determine length of each escalation phase by setting [`CLASSA_PHASE0_CYC`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_phase0_cyc) accordingly - - - For each escalation signal (0..3): - - Determine whether to enable the escalation signal, and set the [`CLASSA_CTRL_SHADOWED.E0_EN`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_ctrl_shadowed) bit accordingly (default is enabled). - Note that setting all of the `E*_EN` bits to 0 within a class has the same effect of disabling the entire class by setting [`CLASSA_CTRL_SHADOWED.EN`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_ctrl_shadowed) to zero. - - Determine the phase -> escalation mapping of this class and program it via the [`CLASSA_CTRL_SHADOWED.E0_MAP`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_ctrl_shadowed) values if it needs to be changed from the default mapping (0->0, 1->1, 2->2, 3->3). - - - Optionally lock the class configuration by writing 0 to [`CLASSA_CTRL_SHADOWED.REGWEN`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_ctrl_shadowed). - -4. After initial configuration at startup, enable the ping timer mechanism by writing 1 to [`PING_TIMER_EN`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#ping_timer_en). -It is also recommended to lock the ping timer configuration by clearing [`PING_TIMER_REGWEN`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#ping_timer_regwen). -Note that only **locked and enabled** alerts are going to be pinged using the ping mechanism. -This ensures that spurious ping failures cannot occur when previously enabled alerts are being disabled again (before locking). - -## Interrupt Handling - -For every alert that is enabled, an interrupt will be triggered on class A, B, C, or D. -To handle an interrupt of a particular class, software should execute the following steps: - -1. If needed, check the escalation state of this class by reading [`CLASSA_STATE`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_state). - This reveals whether escalation protocol has been triggered and in which escalation phase the class is. - In case interrupt timeouts are enabled the class will be in timeout state unless escalation has already been triggered. - The current interrupt or escalation cycle counter can be read via [`CLASSA_ESC_CNT`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_esc_cnt). - -2. Since the interrupt does not indicate which alert triggered, SW must read the cause registers [`LOC_ALERT_CAUSE`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#loc_alert_cause) and [`ALERT_CAUSE`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#alert_cause) etc. - The cause bits of all alerts are concatenated and chunked into 32bit words. - Hence the register file contains as many cause words as needed to cover all alerts present in the system. - Each cause register contains a sticky bit that is set by the incoming alert, and is clearable with a write by software. - This should only be cleared after software has cleared the event trigger, if applicable. - It is possible that the event requires no clearing (e.g. a parity error), or can't be cleared (a breach in the metal mesh protecting the chip). - - Note that in the rare case when multiple events are triggered at or about the same time, all events should be cleared before proceeding. - -3. After the event is cleared (if needed or possible), software should handle the interrupt as follows: - - - Resetting the accumulation register for the class by writing [`CLASSA_CLR`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_clr). - This also aborts the escalation protocol if it has been triggered. - If for some reason it is desired to never allow the accumulator or escalation to be cleared, software can initialize the [`CLASSA_CLR_REGWEN`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_clr_regwen) register to zero. - If [`CLASSA_CLR_REGWEN`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_clr_regwen) is already false when an alert interrupt is detected (either due to software control or hardware trigger via [`CLASSA_CTRL_SHADOWED.LOCK`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#classa_ctrl_shadowed)), then the accumulation counter can not be cleared and this step has no effect. - - - After the accumulation counter is reset (if applicable), software should clear the class A interrupt state bit [`INTR_STATE.CLASSA`](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#intr_state). - Clearing the class A interrupt state bit also clears and stops the interrupt timeout counter (if enabled). - -Note that testing interrupts by writing to the interrupt test registers does also trigger the internal interrupt timeout (if enabled), since the interrupt state is used as enable signal for the timer. -However, alert accumulation will not be triggered by this testing mechanism. - -## Device Interface Functions (DIFs) - -- [Device Interface Functions](../../../sw/device/lib/dif/dif_alert_handler.h) - -## Register Table - -* [Register Table](../../top_earlgrey/ip_autogen/alert_handler/data/alert_handler.hjson#registers) - - -# Additional Notes - -## Timing Constraints - -The skew within all differential signal pairs must be constrained to be smaller than the period of the fastest clock operating the alert handler receivers. -The maximum propagation delay of differential pair should also be constrained (although it may be longer than the clock periods involved). - - -## Fast-track Alerts - -Note that it is possible to program a certain class to provide a fast-track response for critical alerts by setting its accumulation trigger value to 1, and configuring the escalation protocol such that the appropriate escalation measure is triggered within escalation phase 0. -This results in a minimal escalation latency of 4 clock cycles from alert sender input to escalation receiver output in the case where all involved signaling modules are completely synchronous with the alert handler. -In case the alert sender is asynchronous w.r.t. to the alert handler, the actual latency depends on the clock periods involved. -Assuming both clocks have the same frequency alert propagation takes at least 6-8 clock alert handler clock cycles. - -For alerts that mandate an asynchronous response (i.e. without requiring a clock to be active), it is highly recommended to build a separate network at the top-level. -That network should OR' the critical sources together and route the asynchronous alert signal directly to the highest severity countermeasure device. -Examples for alert conditions of this sort would be attacks on the secure clock. diff --git a/hw/top_earlgrey/ip_autogen/alert_handler/doc/programmers_guide.md b/hw/top_earlgrey/ip_autogen/alert_handler/doc/programmers_guide.md new file mode 100644 index 0000000000000..60f9fc61bdfd7 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/alert_handler/doc/programmers_guide.md @@ -0,0 +1,122 @@ +# Programmer's Guide + + +## Power-up and Reset Considerations + +False alerts during power-up and reset are not possible since the alerts are disabled by default, and need to be configured and locked in by the firmware. +The ping timer won't start until initial configuration is over and the registers are locked in. + +The low-power state management of alert channels is handled entirely by hardware and hence this is transparent to software. +Note however that the LPGs inherit the security properties of the associated clock groups and resets. +This means that the low-power state of certain alerts can be controlled by SW by means of clock gating or block reset. +For example, certain crypto blocks are located in a transactional clock group which can be clock gated by SW - and this also affects the associated alerts of these crypto blocks. +See [clock](../../../../ip/clkmgr/README.md) and [reset managers](../../../../ip/rstmgr/README.md) for more detail. + + +## Initialization + +To initialize the block, software running at a high privilege levels (early in the security settings process) should do the following: + +1. For each alert and each local alert: + + - Determine if alert is enabled (should only be false if alert is known to be faulty). + Set [`ALERT_EN_SHADOWED_0.EN_A_0`](../data/alert_handler.hjson#alert_en_shadowed_0) and [`LOC_ALERT_EN_SHADOWED_0.EN_LA_0`](../data/alert_handler.hjson#loc_alert_en_shadowed_0) accordingly. + + - Determine which class (A..D) the alert is associated with. + Set [`ALERT_CLASS_SHADOWED_0.CLASS_A_0`](../data/alert_handler.hjson#alert_class_shadowed_0) and [`LOC_ALERT_CLASS_SHADOWED_0.CLASS_LA_0`](../data/alert_handler.hjson#loc_alert_class_shadowed_0) accordingly. + + - Optionally lock each alert configuration by writing 0 to [`ALERT_REGWEN_0.EN_0`](../data/alert_handler.hjson#alert_regwen_0) or [`LOC_ALERT_REGWEN_0.EN_0`](../data/alert_handler.hjson#loc_alert_regwen_0). + Note however that only **locked and enabled** alerts are going to be pinged using the ping mechanism. + This ensures that spurious ping failures cannot occur when previously enabled alerts are being disabled again (before locking). + + +2. Set the ping timeout value [`PING_TIMEOUT_CYC_SHADOWED`](../data/alert_handler.hjson#ping_timeout_cyc_shadowed). + This value is dependent on the clock ratios present in the system. + +3. For each class (A..D): + + - Determine whether to enable escalation mechanisms (accumulation / interrupt timeout) for this particular class. Set [`CLASSA_CTRL_SHADOWED.EN`](../data/alert_handler.hjson#classa_ctrl_shadowed) accordingly. + + - Determine if this class of alerts allows clearing of escalation once it has begun. + Set [`CLASSA_CTRL_SHADOWED.LOCK`](../data/alert_handler.hjson#classa_ctrl_shadowed) to true if clearing should be disabled. + If true, once escalation protocol begins, it can not be stopped, the assumption being that it ends in a chip reset else it will be rendered useless thenceforth. + + - Determine the number of alerts required to be accumulated before escalation protocol kicks in. Set [`CLASSA_ACCUM_THRESH`](../data/alert_handler.hjson#classa_accum_thresh) accordingly. + + - Determine whether the interrupt associated with that class needs a timeout. + If yes, set [`CLASSA_TIMEOUT_CYC_SHADOWED`](../data/alert_handler.hjson#classa_timeout_cyc_shadowed) to an appropriate value greater than zero (zero corresponds to an infinite timeout and disables the mechanism). + + - For each escalation phase (0..3): + - Determine length of each escalation phase by setting [`CLASSA_PHASE0_CYC`](../data/alert_handler.hjson#classa_phase0_cyc) accordingly + + - For each escalation signal (0..3): + - Determine whether to enable the escalation signal, and set the [`CLASSA_CTRL_SHADOWED.E0_EN`](../data/alert_handler.hjson#classa_ctrl_shadowed) bit accordingly (default is enabled). + Note that setting all of the `E*_EN` bits to 0 within a class has the same effect of disabling the entire class by setting [`CLASSA_CTRL_SHADOWED.EN`](../data/alert_handler.hjson#classa_ctrl_shadowed) to zero. + - Determine the phase -> escalation mapping of this class and program it via the [`CLASSA_CTRL_SHADOWED.E0_MAP`](../data/alert_handler.hjson#classa_ctrl_shadowed) values if it needs to be changed from the default mapping (0->0, 1->1, 2->2, 3->3). + + - Optionally lock the class configuration by writing 0 to [`CLASSA_CTRL_SHADOWED.REGWEN`](../data/alert_handler.hjson#classa_ctrl_shadowed). + +4. After initial configuration at startup, enable the ping timer mechanism by writing 1 to [`PING_TIMER_EN`](../data/alert_handler.hjson#ping_timer_en). +It is also recommended to lock the ping timer configuration by clearing [`PING_TIMER_REGWEN`](../data/alert_handler.hjson#ping_timer_regwen). +Note that only **locked and enabled** alerts are going to be pinged using the ping mechanism. +This ensures that spurious ping failures cannot occur when previously enabled alerts are being disabled again (before locking). + +## Interrupt Handling + +For every alert that is enabled, an interrupt will be triggered on class A, B, C, or D. +To handle an interrupt of a particular class, software should execute the following steps: + +1. If needed, check the escalation state of this class by reading [`CLASSA_STATE`](../data/alert_handler.hjson#classa_state). + This reveals whether escalation protocol has been triggered and in which escalation phase the class is. + In case interrupt timeouts are enabled the class will be in timeout state unless escalation has already been triggered. + The current interrupt or escalation cycle counter can be read via [`CLASSA_ESC_CNT`](../data/alert_handler.hjson#classa_esc_cnt). + +2. Since the interrupt does not indicate which alert triggered, SW must read the cause registers [`LOC_ALERT_CAUSE`](../data/alert_handler.hjson#loc_alert_cause) and [`ALERT_CAUSE`](../data/alert_handler.hjson#alert_cause) etc. + The cause bits of all alerts are concatenated and chunked into 32bit words. + Hence the register file contains as many cause words as needed to cover all alerts present in the system. + Each cause register contains a sticky bit that is set by the incoming alert, and is clearable with a write by software. + This should only be cleared after software has cleared the event trigger, if applicable. + It is possible that the event requires no clearing (e.g. a parity error), or can't be cleared (a breach in the metal mesh protecting the chip). + + Note that in the rare case when multiple events are triggered at or about the same time, all events should be cleared before proceeding. + +3. After the event is cleared (if needed or possible), software should handle the interrupt as follows: + + - Resetting the accumulation register for the class by writing [`CLASSA_CLR`](../data/alert_handler.hjson#classa_clr). + This also aborts the escalation protocol if it has been triggered. + If for some reason it is desired to never allow the accumulator or escalation to be cleared, software can initialize the [`CLASSA_CLR_REGWEN`](../data/alert_handler.hjson#classa_clr_regwen) register to zero. + If [`CLASSA_CLR_REGWEN`](../data/alert_handler.hjson#classa_clr_regwen) is already false when an alert interrupt is detected (either due to software control or hardware trigger via [`CLASSA_CTRL_SHADOWED.LOCK`](../data/alert_handler.hjson#classa_ctrl_shadowed)), then the accumulation counter can not be cleared and this step has no effect. + + - After the accumulation counter is reset (if applicable), software should clear the class A interrupt state bit [`INTR_STATE.CLASSA`](../data/alert_handler.hjson#intr_state). + Clearing the class A interrupt state bit also clears and stops the interrupt timeout counter (if enabled). + +Note that testing interrupts by writing to the interrupt test registers does also trigger the internal interrupt timeout (if enabled), since the interrupt state is used as enable signal for the timer. +However, alert accumulation will not be triggered by this testing mechanism. + +## Device Interface Functions (DIFs) + +- [Device Interface Functions](../../../../../sw/device/lib/dif/dif_alert_handler.h) + +## Register Table + +* [Register Table](../data/alert_handler.hjson#registers) + + +# Additional Notes + +## Timing Constraints + +The skew within all differential signal pairs must be constrained to be smaller than the period of the fastest clock operating the alert handler receivers. +The maximum propagation delay of differential pair should also be constrained (although it may be longer than the clock periods involved). + + +## Fast-track Alerts + +Note that it is possible to program a certain class to provide a fast-track response for critical alerts by setting its accumulation trigger value to 1, and configuring the escalation protocol such that the appropriate escalation measure is triggered within escalation phase 0. +This results in a minimal escalation latency of 4 clock cycles from alert sender input to escalation receiver output in the case where all involved signaling modules are completely synchronous with the alert handler. +In case the alert sender is asynchronous w.r.t. to the alert handler, the actual latency depends on the clock periods involved. +Assuming both clocks have the same frequency alert propagation takes at least 6-8 clock alert handler clock cycles. + +For alerts that mandate an asynchronous response (i.e. without requiring a clock to be active), it is highly recommended to build a separate network at the top-level. +That network should OR' the critical sources together and route the asynchronous alert signal directly to the highest severity countermeasure device. +Examples for alert conditions of this sort would be attacks on the secure clock. diff --git a/hw/top_earlgrey/ip_autogen/alert_handler/doc/theory_of_operation.md b/hw/top_earlgrey/ip_autogen/alert_handler/doc/theory_of_operation.md new file mode 100644 index 0000000000000..bd8d3f3dbbe12 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/alert_handler/doc/theory_of_operation.md @@ -0,0 +1,785 @@ +# Theory of Operation + +## Block Diagram + +The figure below shows a block diagram of the alert handler module, as well as a few examples of alert senders in other peripheral modules. +In this diagram, there are seven sources of alerts: three sources from external modules (two from `periph0` and one from `periph1`), and four local sources (`alert_ping_fail`, `alert_sig_int`, `esc_ping_fail`, `esc_sig_int`). +The local sources represent alerts that are created by this module itself. See the later section on special local alerts. + +![Alert Handler Block Diagram](alert_handler_block_diagram.svg) + +Also shown are internal modules for classification, interrupt generation, accumulation, escalation, ping generation and alert-channel low-power control. +These are described later in the document. +Note that the differential alert sender and receiver blocks used for alert signaling support both _asynchronous_ and _synchronous_ clocking schemes, and hence peripherals able to raise alerts may be placed in clock domains different from that of the alert handler (Jittered clock domains are also supported in the asynchronous clocking scheme). +Proper care must however be taken when formulating the timing constraints for the diff pairs, and when determining clock-dependent parameters (such as the ping timeout) of the design. +On the escalation sender / receiver side, the differential signaling blocks employ a fully synchronous clocking scheme throughout. + +## Hardware Interfaces + +### Parameters + +The following table lists the main parameters used throughout the alert handler design. +Note that the alert handler is generated based on the system configuration, and hence these parameters are placed into a package as "localparams". +The parameterization rules are explained in more detail in the architectural description. + +Localparam | Default (Max) | Top Earlgrey | Description +---------------|-----------------------|--------------|--------------- +`NAlerts` | 8 (248) | see RTL | Number of alert instances. Maximum number bounded by LFSR implementation that generates ping timing. +`NLpg` | 1 | see RTL | Number of unique low-power groups as determined by topgen. +`LpgMap` | {0} | see RTL | Array mapping each alert to a unique low-power group as determined by topgen. +`EscCntWidth` | 32 (32) | 32 | Width of the escalation counters in bit. +`AccuCntWidth` | 16 (32) | 16 | Width of the alert accumulation counters in bit. +`AsyncOn` | '0 (2^`NAlerts`-1) | see RTL | This is a bit array specifying whether a certain alert sender / receiver pair goes across an asynchronous boundary or not. + +The next table lists free parameters in the `prim_alert_sender` and +`prim_alert receiver` submodules. + +Parameter | Default (Max) | Description +---------------|------------------|--------------- +`AsyncOn` | `1'b0` (`1'b1`) | 0: Synchronous, 1: Asynchronous, determines whether additional synchronizer flops and logic need to be instantiated. + + +### Signals + +* [Interface Tables](../data/alert_handler.hjson#interfaces) + +The table below lists other alert handler module signals. +The number of alert instances is parametric and hence alert and ping diff pairs are grouped together in packed arrays. +The diff pair signals are indexed with the corresponding alert instance ``. + +Signal | Direction | Type | Description +-------------------------|------------------|---------------- |--------------- +`edn_o` | `output` | `otp_edn_req_t` | Entropy request to the entropy distribution network for LFSR reseeding and ephemeral key derivation. +`edn_i` | `input` | `otp_edn_rsp_t` | Entropy acknowledgment to the entropy distribution network for LFSR reseeding and ephemeral key derivation. +`alert_tx_i[]` | `input` | packed `alert_tx_t` array | Incoming alert or ping response(s), differentially encoded. Index range: `[NAlerts-1:0]` +`alert_rx_o[]` | `output` | packed `alert_rx_t` array | Outgoing alert acknowledgment and ping requests, differentially encoded. Index range: `[NAlerts-1:0]` +`esc_tx_o[]` | `output` | packed `esc_tx_t` array | Escalation or ping request, differentially encoded. Index corresponds to severity level, and ranges from 0 to 3. +`esc_rx_i[]` | `input` | packed `esc_rx_t` array | Escalation ping response, differentially encoded. Index corresponds to severity level, and ranges from 0 to 3. +`lpg_cg_en_i[]` | `input` | packed `mubi4_t` array | Incoming clock gated indication from clock manager. Index range: `[NLpg-1:0]` +`lpg_rst_en_i[]` | `input` | packed `mubi4_t` array | Incoming reset asserted indication from reset manager. Index range: `[NLpg-1:0]` +`crashdump_o` | `output` | packed `struct` | This is a collection of alert handler state registers that can be latched by hardware debugging circuitry, if needed. + +#### Entropy Network Connections + +The LFSR ping timer needs to be periodically reseeded. +Therefore, the alert handler is connected to the entropy distribution network via the `edn_i/o` signals. + +#### Alert Channels + +For each alert, there is a pair of input and two pairs of output signals. +These signals are connected to a differential sender module within the source, and a differential receiver module within the alert handler. +Both of these modules are described in more detail in the following section. +These signal pairs carry differentially encoded messages that enable two types of signaling: a native alert and a ping/response test of the alert mechanism. +The latter is to ensure that all alert senders are always active and have not been the target of an attack. + +#### Escalation Channels + +For each escalation action in the system, there is a pair of input and a pair of output signals, encapsulated in the `esc_rx_t` and `esc_tx_t` types. +These signals are connected to a differential sender module within the alert handler, and a differential receiver module within the module that performs a particular escalation action (for example the reset manager or life cycle controllers). +The signal pairs carry differentially encoded messages that enable two types of signaling: a native escalation and a ping/response test of the escalation mechanism. +The latter is to ensure that all escalation receivers are always active and have not been the target of an attack. + +#### Low-power Indication Signals + +The `lpg_cg_en_i` and `lpg_rst_en_i` are two arrays with multibit indication signals from the [clock](../../../../ip/clkmgr/README.md) and [reset managers](../../../../ip/rstmgr/README.md). +These indication signals convey whether a specific group of alert senders are either clock gated or in reset. +As explained in [more detail below](#low-power-management-of-alert-channels), this information is used to temporarily halt the ping timer mechanism on channels that are in a low-power state in order to prevent false positives. + +#### Crashdump Output + +The `crashdump_o` struct outputs a snapshot of CSRs and alert handler state bits that can be read by hardware debugging circuitry: + +```systemverilog + typedef struct packed { + // alerts + logic [NAlerts-1:0] alert_cause; // alert cause bits + logic [6:0] loc_alert_cause; // local alert cause bits + // class state + logic [3:0][15:0] class_accum_cnt; // current accumulator value + logic [3:0][31:0] class_esc_cnt; // current escalation counter value + cstate_e [3:0] class_esc_state; // current escalation protocol state + } alert_crashdump_t; +``` + +This can be useful for extracting more information about possible failures or bugs without having to use the tile-link bus interface (which may become unresponsive under certain circumstances). +It is recommended for the top level to store this information in an always-on location. + +Note that the crashdump state is continuously output via `crashdump_o` until the latching trigger condition is true for the first time (see [`CLASSA_CRASHDUMP_TRIGGER_SHADOWED`](../data/alert_handler.hjson#classa_crashdump_trigger_shadowed)). +After that, the `crashdump_o` is held constant until all classes that have escalated are cleared. +This is done so that it is possible to capture the true alert cause before spurious alert events start to pop up due to escalation countermeasures with excessive side effects (like life cycle scrapping for example). +If classes that have escalated are not configured as clearable, then it is not possible to re-arm the crashdump latching mechanism at runtime and the alert handler has to be reset. + +## Design Details + +This section gives the full design details of the alert handler module and its submodules. + + +### Alert Definition + +Alerts are defined as events that have security implications, and should be handled by the main processor, or escalated to other hardware modules to take action. +Each peripheral has the option to define one or more alert signals. +Those peripherals should instantiate one module (`prim_alert_sender`) to convert the event associated with that alert into a signal to the alert handler module. +The alert handler instantiates one receiver module (`prim_alert_receiver`) per alert, then handles the classification, accumulation, and escalation of the received signal. +The differential signaling submodules may either use a synchronous or asynchronous clocking scheme, since the message type to be transferred is a single discrete event. + + +### Differential Alert Signaling + +Each alert sender is connected to the corresponding alert receiver via the 3 differential pairs `alert_tx_i/o.alert_p/n`, `alert_rx_i/o.ack_p/n` and `alert_rx_i/o.ping_p/n`, as illustrated below: + +![Alert Handler Alert RXTX](alert_handler_alert_rxtx.svg) + +Alerts are encoded differentially and signaled using a full handshake on the `alert_tx_i/o.alert_p/n` and `alert_rx_i/o.ack_p/n` wires. +The use of a full handshake protocol allows this mechanism to be used with an asynchronous clocking strategy, where peripherals may reside in a different clock domain than the alert handler. +The full handshake guarantees that alert messages are correctly back-pressured and no alert is "lost" at the asynchronous boundary due to (possibly variable) clock ratios greater or less than 1.0. +The "native alert message" will be repeated on the output wires as long as the alert event is still true within the peripheral. + +The wave pattern below illustrates differential full handshake mechanism. + +```wavejson +{ + signal: [ + { name: 'clk_i', wave: 'p...............' }, + { name: 'alert_req_i', wave: '01.|..|..|...|..' }, + { name: 'alert_ack_o', wave: '0..|..|..|10.|..' }, + { name: 'alert_tx_o/i.alert_p', wave: '01.|..|0.|..1|..' , node: '.a.....c....e'}, + { name: 'alert_tx_o/i.alert_n', wave: '10.|..|1.|..0|..' }, + { name: 'alert_rx_i/o.ack_p', wave: '0..|1.|..|0..|1.' , node: '....b.....d..'}, + { name: 'alert_rx_i/o.ack_n', wave: '1..|0.|..|1..|0.' }, + { name: 'alert_o', wave: '0..|10|..|...|10' }, + ], + edge: [ + 'a~>b Phase 0/1', + 'b~>c Phase 1/2', + 'c~>d Phase 2/3', + 'd~>e 2 Pause Cycles', + ], + head: { + text: 'Alert signaling and repeat pattern', + }, + foot: { + text: 'Native alert at time 1 with 4-phase handshake; repeated alert at time 12;', + tick: 0, + } +} +``` + +The handshake pattern is repeated as long as the alert is true. +The sender will wait for 2 cycles between handshakes. + +Note that the alert is immediately propagated to `alert_o` once the initial level change on `alert_tx_i.alert_p/n` has been received and synchronized to the local clock on the receiver side. +This ensures that the first occurrence of an alert is always propagated - even if the handshake lines have been manipulated to emulate backpressure. +(In such a scenario, all subsequent alerts would be back-pressured and eventually the ping testing mechanism described in the next subsection would detect that the wires have been tampered with.) + +The alert sender and receiver modules can either be used synchronously or asynchronously. +The signaling protocol remains the same in both cases, but the additional synchronizer flops at the diff pair inputs may be omitted, which results in lower signaling latency. + +### Ping Testing + +In order to ensure that the event sending modules have not been compromised, the alert receiver module `prim_alert_receiver` will "ping" or line-test the senders periodically every few microseconds. +Pings timing is randomized so their appearance can not be predicted. + + +The ping timing is generated by a central LFSR-based timer within the alert handler that randomly asserts the `ping_req_i` signal of a particular `prim_alert_receiver` module. +Once `ping_req_i` is asserted, the receiver module encodes the ping message as a level change on the differential `alert_rx_o.ping_p/n` output, and waits until the sender responds with a full handshake on the `alert_tx_i.alert_p/n` and `alert_rx_o.ack_p/n` lines. +Once that handshake is complete, the `ping_ok_o` signal is asserted. +The LFSR timer has a programmable ping timeout, after which it will automatically assert a "pingfail" alert. +That timeout is a function of the clock ratios present in the system, and has to be programmed accordingly at system startup (as explained later in the LFSR timer subsection). + +The following wave diagram illustrates a correct ping sequence, viewed from the receiver side: + +```wavejson +{ + signal: [ + { name: 'clk_i', wave: 'p..............' }, + { name: 'ping_req_i', wave: '01.|..|..|..|.0' }, + { name: 'ping_ok_o', wave: '0..|..|..|..|10' , node: '.............e'}, + { name: 'alert_rx_o.ping_p', wave: '01.|..|..|..|..' , node: '.a'}, + { name: 'alert_rx_o.ping_n', wave: '10.|..|..|..|..' , node: '.b'}, + { name: 'alert_tx_i.alert_p', wave: '0..|1.|..|0.|..' , node: '....c'}, + { name: 'alert_tx_i.alert_n', wave: '1..|0.|..|1.|..' }, + { name: 'alert_rx_o.ack_p', wave: '0..|..|1.|..|0.' , node: '.............d'}, + { name: 'alert_rx_o.ack_n', wave: '1..|..|0.|..|1.' }, + ], + edge: [ + 'a-b', + 'b~>c ping response', + 'd->e response complete', + ], + head: { + text: 'Ping testing', + }, + foot: { + text: 'Level change at time 1 triggers a full handshake (ping response) at time 4', + tick: 0, + } +} +``` + +In the unlikely case that a ping request collides with a native alert at the sender side, the native alert is held back until the ping handshake has been completed. +This slightly delays the transmission of a native alert, but the alert will eventually be signaled. +Further, if an alert is sent out right before a ping requests comes in at the sender side, the receiver will treat the alert as a ping response. +However, the "true" ping response will be returned right after the alert handshake completed, and thus the alert will eventually be signaled with a slight delay. + +Note that in both collision cases mentioned, the delay will be in the order of the handshake length, plus the constant amount of pause cycles between handshakes (2 sender cycles). + + +### Monitoring of Signal Integrity Issues + +All differential pairs are monitored for signal integrity issues, and if an encoding failure is detected, the receiver module asserts a signal integrity alert via `integ_fail_o`. In particular, this covers the following failure cases: + +1. The `alert_tx_i.alert_p/n` pair is not correctly encoded on the receiver side. +This can be directly flagged as an integrity failure on the receiver side. + +2. The `alert_rx_i.ping_p/n` or the `alert_rx_i.ack_p/n` pairs are not correctly encoded on the sender side. +This is signaled to the receiver by setting the `alert_tx_o.alert_p/n` wires to the same value, and that value will be continuously toggled. +This implicitly triggers a signal integrity alert on the receiver side. + +Some of these failure patterns are illustrated in the wave diagram below: + +```wavejson +{ + signal: [ + { name: 'clk_i', wave: 'p..............' }, + { name: 'alert_tx_o.alert_p', wave: '0.1...|0..10101' , node: '..a.......d'}, + { name: 'alert_tx_o.alert_n', wave: '1.....|....0101' }, + { name: 'alert_rx_i.ack_p', wave: '0.....|.1......' , node: '........c'}, + { name: 'alert_rx_i.ack_n', wave: '1.....|........' }, + { name: 'integ_fail_o', wave: '0...1.|0....1..' , node: '....b.......e'}, + ], + edge: [ + 'a~>b sigint issue detected', + 'c~>d', + 'd~>e indirect sigint issue detected', + ], + head: { + text: 'Detection of Signal Integrity Issues', + }, + foot: { + text: 'signal integrity issues occur at times 2 and 8; synchronizer latency is 2 cycles.', + tick: 0, + } +} +``` + +Note that if signal integrity failures occur during ping or alert handshaking, it is possible that the protocol state-machines lock up and the alert sender and receiver modules become unresponsive. However, the above mechanisms ensure that this will always trigger either a signal integrity alert or eventually a "pingfail" alert. + +### Skew on Asynchronous Differential Pairs + +Note that there is likely a (small) skew present within each differential pair of the signaling mechanism above. Since these pairs cross clock domain boundaries, it may thus happen that a level change appears in staggered manner after resynchronization, as illustrated below: + +```wavejson +{ + signal: [ + { name: 'clk_i', wave: 'p...........' }, + { name: 'diff_p', wave: '0.1.|.0.|..1' , node: '......a....d' }, + { name: 'diff_n', wave: '1.0.|..1|.0.' , node: '.......b..c.' }, + ], + edge: [ + 'a-~>b skew', + 'c-~>d skew' + ], + head: { + text: 'Skewed diff pair', + }, + foot: { + text: 'Correctly sampled diff pair at time 2; staggered samples at time 6-7 and 10-11', + tick: 0, + } +} +``` + +This behavior is permissible, but needs to be accounted for in the protocol logic. +Further, the skew within the differential pair should be constrained to be smaller than the shortest clock period in the system. +This ensures that the staggered level changes appear at most 1 cycle apart from each other. + + +### LFSR Timer + +The `ping_req_i` inputs of all signaling modules (`prim_alert_receiver`, `prim_esc_sender`) instantiated within the alert handler are connected to a central ping timer that alternatingly pings either an alert line or an escalation line after waiting for a pseudo-random amount of clock cycles. +Further, this ping timer also randomly selects a particular alert line to be pinged (escalation senders are always pinged in-order due to the [ping monitoring mechanism](#monitoring-of-pings-at-the-escalation-receiver-side) on the escalation side). +That should make it more difficult to predict the next ping occurrence based on past observations. + +The ping timer is implemented using an [LFSR-based PRNG of Galois type](../../../../ip/prim/doc/prim_lfsr.md). +This ping timer is reseeded with fresh entropy from EDN roughly every 500k cycles which corresponds to around 16 ping operations on average. +The LFSR is 32bits wide, but only 24bits of its state are actually being used to generate the random timer count and select the alert line to be pinged. +I.e., the 32bits first go through a fixed permutation function, and then bits `[23:16]` are used to determine which alert line to ping. +The random cycle count is created by OR'ing bits `[15:0]` with the constant `3'b100` as follows: + +``` +cycle_cnt = permuted[15:0] | 3'b100; +``` + +This constant DC offset introduces a minimum ping spacing of 4 cycles (1 cycle + margin) to ensure that the handshake protocols of the sender/receiver pairs work. + +After selecting one of the peripherals to ping, the LFSR timer waits until either the corresponding `*_ping_ok[]` signal is asserted, or until the programmable ping timeout value is reached. +In both cases, the LFSR timer proceeds with the next ping, but in the second case it will additionally raise a "pingfail" alert. +The ping enable signal remains asserted during the time where the LFSR counter waits. + +The timeout value is a function of the ratios between the alert handler clock and peripheral clocks present in the system, and can be programmed at startup time via the register [`PING_TIMEOUT_CYC_SHADOWED`](../data/alert_handler.hjson#ping_timeout_cyc_shadowed). + +Note that the ping timer directly flags a "pingfail" alert if a spurious "ping ok" message comes in that has not been requested. + + +As described in the programmers guide below, the ping timer has to be enabled explicitly. +Only alerts that have been *enabled and locked* will be pinged in order to avoid spurious alerts. +Escalation channels are always enabled, and hence will always be pinged once this mechanism has been turned on. + +In addition to the ping timer mechanism described above, the escalation receivers contain monitoring counters that monitor the liveness of the alert handler (described in more detail in [this section](#monitoring-of-pings-at-the-escalation-receiver-side). +This mechanism requires that the maximum wait time between escalation receiver pings is bounded. +To that end, escalation senders are pinged in-order every second ping operation (i.e., the wait time is randomized, but the selection of the escalation line is not). + +### Alert Receiving + +The alert handler module contains one alert receiver module (`prim_alert_receiver`) per sending module. +This receiver module has three outputs based upon the signaling of the input alert. +Primary is the signal of a received native alert, shown in the top-level diagram as `alert_triggered[]`. +Also generated are two other outputs, one that signals a differential encoding error (`alert_integ_fail[]`), and one that signals the receipt of a ping response (`alert_ping_ok[]`). +Each "triggered" alert received is sent into the classification block for individual configuration. +All of the `integ_fail` signals are OR'ed together to create one alert for classification. +The ping responses are fed to the LFSR timer, which determines whether a ping has correctly completed within the timeout window or not. + + +### Alert Classification and Interrupts + +Each of the incoming and local alert signals can be classified generically to one of four classes, or disabled for no classification at all. +These are the classes A, B, C, and D. +There is no pre-determined definition of a class, that is left to software. +But for guidance, software can consider that some alert types are similar to others; some alert types are more "noisy" than others (i.e. when triggered they stay on for long periods of time); some are more critical than others, etc. + +For each alert class (A-D), an interrupt is generally sent. +Like all other peripheral interrupts, there is a triad of registers: enable, status, test. +Thus like all other interrupts, software should handle the source of the interrupt (in this case, the original alert), then clear the state. +Since the interrupt class is disassociated with the original alert (due to the classification process), software can access cause registers to determine which alerts have fired since the last clearing. +Since alerts are expected to be rare (if ever) events, the complexity of dealing with multiple interrupts per class firing during the same time period should not be of concern. See the programming section on interrupt clearing. + +Each of the four interrupts can optionally trigger a timeout counter that triggers escalation if the interrupt is not handled and cleared within a certain time frame. +This feature is explained in more detail in the next subsection about escalation mechanisms. + +Note that an interrupt always fires once an alert has been registered in the corresponding class. +Interrupts are not dependent on escalation mechanisms like alert accumulation or timeout as described in the next subsection. + + +### Escalation Mechanisms + +There are two mechanisms per class that can trigger the corresponding escalation +protocol: + +1. The first consists of an accumulation counter that counts the amount of alert occurrences within a particular class. + An alert classified to class A indicates that on every received alert trigger, the accumulation counter for class A is incremented. + Note: since alerts are expected to be rare or never occur, the module does not attempt to count every alert per cycle, but rather all triggers per class are ORd before sending to the accumulation counter as an increment signal. + Once the threshold has been reached, the next occurrence triggers the escalation escalation protocol for this particular class. + The counter is a saturation counter, meaning that it will not wrap around once it hits the maximum representable count. + This mechanism has two associated CSRs: + + - Accumulation max value. + This is the total number (sum of all alerts classified in this group) of alerts required to enter escalation phase (see below). + Example register is [`CLASSA_ACCUM_THRESH_SHADOWED`](../data/alert_handler.hjson#classa_accum_thresh_shadowed). + - Current accumulation register. + This clearable register indicates how many alerts have been accumulated to date. + Software should clear before it reaches the accumulation setting to avoid escalation. + Example register is [`CLASSA_ACCUM_CNT`](../data/alert_handler.hjson#classa_accum_cnt). + +2. The second way is an interrupt timeout counter which triggers escalation if an alert interrupt is not handled within the programmable timeout window. + Once the counter hits the timeout threshold, the escalation protocol is triggered. + The corresponding CSRs are: + + - Interrupt timeout value in cycles [`CLASSA_TIMEOUT_CYC_SHADOWED`](../data/alert_handler.hjson#classa_timeout_cyc_shadowed). + The interrupt timeout is disabled if this is set to 0 (default). + - The current interrupt timeout value can be read via [`CLASSA_ESC_CNT`](../data/alert_handler.hjson#classa_esc_cnt) if [`CLASSA_STATE`](../data/alert_handler.hjson#classa_state) is in the `Timeout` state. + Software should clear the corresponding interrupt state bit [`INTR_STATE.CLASSA`](../data/alert_handler.hjson#intr_state) before the timeout expires to avoid escalation. + +Technically, the interrupt timeout feature (2. above) is implemented using the same counter used to time the escalation phases. +This is possible since escalation phases or interrupt timeout periods are non-overlapping (escalation always takes precedence should it be triggered). + + +### Programmable Escalation Protocol + +There are four output escalation signals, 0, 1, 2, and 3. +There is no predetermined definition of an escalation signal, that is left to the top-level integration. +Examples could be processor Non Maskable Interrupt (NMI), privilege lowering, secret wiping, chip reset, etc. +Typically the assumption is that escalation level 0 is the first to trigger, followed by 1, 2, and then 3, emulating a "fuse" that is lit that can't be stopped once the first triggers (this is however not a requirement). +See register section for discussion of counter clearing and register locking to determine the finality of accumulation +triggers. + +Each class can be programmed with its own escalation protocol. +If one of the two mechanisms described above fires, a timer for that particular class is started. +The timer can be programmed with up to 4 delays (e.g., [`CLASSA_PHASE0_CYC`](../data/alert_handler.hjson#classa_phase0_cyc)), each representing a distinct escalation phase (0 - 3). +Each of the four escalation severity outputs (0 - 3) are by default configured to be asserted during the corresponding phase, e.g., severity 0 in phase 0, severity 1 in phase 1, etc. +However, this mapping can be freely reassigned by modifying the corresponding enable/phase mappings (e.g., [`CLASSA_CTRL_SHADOWED.E0_MAP`](../data/alert_handler.hjson#classa_ctrl_shadowed) for enable bit 0 of class A). +This mapping will be locked in together with the alert enable configuration after initial configuration. + +SW can stop a triggered escalation protocol by clearing the corresponding escalation counter (e.g., [`CLASSA_ESC_CNT`](../data/alert_handler.hjson#classa_esc_cnt)). +Protection of this clearing is up to software, see the register control section that follows for [`CLASSA_CTRL_SHADOWED.LOCK`](../data/alert_handler.hjson#classa_ctrl_shadowed). + +It should be noted that each of the escalation phases have a duration of at least 1 clock cycle, even if the cycle count of a particular phase has been +set to 0. + +The next waveform shows the gathering of alerts of one class until eventually the escalation protocol is engaged. +In this diagram, two different alerts are shown for class A, and the gathering and escalation configuration values are shown. + +```wavejson +{ + signal: [ + { name: 'clk_i', wave: 'p...................' }, + { name: 'CLASSA_ACCUM_THRESH_SHADOWED', wave: '2...................', data: ['15'] }, + { name: 'CLASSA_PHASE0_CYC_SHADOWED', wave: '2...................', data: ['1e3 cycles'] }, + { name: 'CLASSA_PHASE1_CYC_SHADOWED', wave: '2...................', data: ['1e4 cycles'] }, + { name: 'CLASSA_PHASE2_CYC_SHADOWED', wave: '2...................', data: ['1e5 cycles'] }, + { name: 'CLASSA_PHASE3_CYC_SHADOWED', wave: '2...................', data: ['1e6 cycles'] }, + { name: 'alert_triggered[0]', wave: '010|.10.............' }, + { name: 'alert_triggered[1]', wave: '0..|10..............' }, + { name: 'CLASSA_ACCUM_CNT', wave: '33.|33..............', data: ['0', '1','15','16'] }, + { name: 'irq_o[0]', wave: '01.|................' }, + { name: 'CLASSA_STATE', wave: '3..|.3|3.|3..|3..|3.', data: ['Idle', ' Phase0','Phase1','Phase2','Phase3','Terminal'] }, + { name: 'CLASSA_ESC_CNT', wave: '3..|.3|33|333|333|3.', data: ['0','1','1','2','1','2','3','1','2','3','0'] }, + { name: 'esc_tx_o.esc_p[0]', wave: '0..|.1|.0...........', node: '.....a..b' }, + { name: 'esc_tx_o.esc_n[0]', wave: '1..|.0|.1...........' }, + { name: 'esc_tx_o.esc_p[1]', wave: '0..|..|1.|.0........', node: '.......c...d' }, + { name: 'esc_tx_o.esc_n[1]', wave: '1..|..|0.|.1........' }, + { name: 'esc_tx_o.esc_p[2]', wave: '0..|.....|1..|.0....', node: '..........e....f' }, + { name: 'esc_tx_o.esc_n[2]', wave: '1..|.....|0..|.1....' }, + { name: 'esc_tx_o.esc_p[3]', wave: '0..|.........|1..|.0', node: '..............g....h' }, + { name: 'esc_tx_o.esc_n[3]', wave: '1..|.........|0..|.1' }, + ], + edge: [ + 'a->b 1e3 + 1 cycles', + 'c->d 1e4 + 1 cycles', + 'e->f 1e5 + 1 cycles', + 'g->h 1e6 + 1 cycles', + ], + head: { + text: 'Alert class gathering and escalation triggers (fully synchronous case)', + }, + foot: { + text: 'alert class A gathers 16 alerts, triggers first escalation, followed by three more', + tick: 0, + } +} +``` + +In this diagram, the first alert triggers an interrupt to class A. +The assumption is that the processor is wedged or taken over, in which case it does not handle the interrupt. +Once enough interrupts gather (16 in this case), the first escalation phase is entered, followed by three more (each phase has its own programmable length). +Note that the accumulator threshold is set to 15 in order to trigger on the 16th occurrence. +If escalation shall be triggered on the first occurrence within an alert class, the accumulation threshold shall be set to 0. +Also note that it takes one cycle to activate escalation and enter phase 0. + +The next wave shows a case where an interrupt remains unhandled and hence the interrupt timeout counter triggers escalation. + +```wavejson +{ + signal: [ + { name: 'clk_i', wave: 'p.....................' }, + { name: 'CLASSA_TIMEOUT_CYC_SHADOWED', wave: '2.....................', data: ['1e4 cycles'] }, + { name: 'alert_triggered[0]', wave: '010.|.................' }, + { name: 'irq_o[0]', wave: '01..|.................', node: '.a..|.b' }, + { name: 'CLASSA_ESC_STATE', wave: '33..|.3|3.|3..|3...|3.', data: ['Idle', 'Timeout',' Phase0','Phase1','Phase2','Phase3','Terminal'] }, + { name: 'CLASSA_ESC_CNT', wave: '3333|33|33|333|3333|3.', data: ['0', '1','2','3','1e4','1','1','2','1','2','3','1','2','3','4','0'] }, + { name: 'esc_tx_o.esc_p[0]', wave: '0...|.1|.0............' }, + { name: 'esc_tx_o.esc_n[0]', wave: '1...|.0|.1............' }, + { name: 'esc_tx_o.esc_p[1]', wave: '0...|..|1.|.0.........' }, + { name: 'esc_tx_o.esc_n[1]', wave: '1...|..|0.|.1.........' }, + { name: 'esc_tx_o.esc_p[2]', wave: '0...|.....|1..|.0.....' }, + { name: 'esc_tx_o.esc_n[2]', wave: '1...|.....|0..|.1.....' }, + { name: 'esc_tx_o.esc_p[3]', wave: '0...|.........|1...|.0' }, + { name: 'esc_tx_o.esc_n[3]', wave: '1...|.........|0...|.1' }, + ], + edge: [ + 'a->b 1e4 cycles', + ], + head: { + text: 'Escalation due to an interrupt timeout (fully synchronous case)', + }, + foot: { + text: 'alert class A triggers an interrupt and the timeout counter, which eventually triggers escalation after 1e4 cycles.', + tick: 0, + } +} +``` + +It should be noted here that the differential escalation signaling protocol distinguishes 'true' escalation conditions from mere pings by encoding them as pulses that are N + 1 cycles long. +This is reflected in the two wave diagrams above. +Refer to the subsequent section on escalation signaling for more details. + +### Escalation Signaling + +For each of the four escalation severities, the alert handler instantiates a `prim_esc_sender` module and each of the four escalation countermeasures instantiates an `prim_esc_receiver` module. +The signaling mechanism has similarities with the alert signaling mechanism - but it is a fully synchronous protocol. +Hence, it must be ensured at the top-level that all escalation sender and receiver modules are using the same clock and reset +signals. + +As illustrated in the following block diagram, a sender-receiver pair is connected with two differential lines, one going from sender to receiver and the other going from receiver to sender. + +![Alert Handler Escalation RXTX](alert_handler_escalation_rxtx.svg) + +Upon receiving an escalation enable pulse of width N > 0 at the `esc_req_i` input, the escalation sender encodes that signal as a differential pulse of width N+1 on `esc_tx.esc_p/n`. +The receiver decodes that message and asserts the `esc_req_o` output after one cycle of delay. +Further, it acknowledges the receipt of that message by continuously toggling the `esc_rx.resp_p/n` signals as long as the escalation signal is asserted. +Any failure to respond correctly will trigger a `integ_fail_o` alert, as illustrated below: + +```wavejson +{ + signal: [ + { name: 'clk_i', wave: 'p..................' }, + { name: 'ping_req_i', wave: '0........|.........' }, + { name: 'ping_ok_o', wave: '0........|.........' }, + { name: 'integ_fail_o', wave: '0........|..1010...' , node: '............b.d' }, + { name: 'ping_fail_o', wave: '0........|.........' }, + { name: 'esc_req_i', wave: '01....0..|.1....0..' }, + { name: 'esc_rx_i/o.resp_p', wave: '0.101010.|.........', node: '............a.c' }, + { name: 'esc_rx_i/o.resp_n', wave: '1.010101.|.........' }, + { name: 'esc_tx_o/i.esc_p', wave: '01.....0.|.1.....0.' }, + { name: 'esc_tx_o/i.esc_n', wave: '10.....1.|.0.....1.' }, + { name: 'esc_req_o', wave: '0.1....0.|..?....0.'}, + ], + edge: [ + 'a~>b missing response', + 'c~>d', + ], + head: { + text: 'Escalation signaling and response', + }, + foot: { + text: 'escalation enable pulse shown at input sender at time 1 and 11; missing response and repeated integfail at time 12 and 14', + tick: 0, + } +} +``` + +Further, any differential signal mismatch on both the `esc_tx_i.esc_p/n` and `esc_rx_i.resp_p/n` lines will trigger an `integ_fail_o` alert. +Mismatches on `esc_rx_i.resp_p/n` can be directly detected at the sender. +Mismatches on the `esc_tx_i.esc_p/n` line will be signaled back to the sender by setting both the positive and negative response wires to the same value - and that value is being toggled each cycle. +This implicitly triggers a signal integrity alert on the sender side. +In addition to that, a signal integrity error on the `esc_tx_i.esc_p/n` lines will lead to assertion of the `esc_req_o` output, since it cannot be guaranteed that the back signalling mechanism always works when the sender / receiver pair is being tampered with. + +This back-signaling mechanism can be leveraged to fast-track escalation and use another countermeasure in case it is detected that a particular escalation signaling path has been tampered with. + +Some signal integrity failure cases are illustrated in the wave diagram below: + +```wavejson +{ + signal: [ + { name: 'clk_i', wave: 'p...........' }, + { name: 'ping_req_i', wave: '0....|......' }, + { name: 'ping_ok_o', wave: '0....|......' }, + { name: 'integ_fail_o', wave: '0.1.0|.1....' , node: '..b....e' }, + { name: 'esc_req_i', wave: '0....|......' }, + { name: 'esc_rx_i.resp_p', wave: '0.1.0|..1010', node: '..a..' }, + { name: 'esc_rx_i.resp_n', wave: '1....|.01010', node: '.......d' }, + { name: 'esc_tx_i.esc_p', wave: '0....|1.....', node: '......c..' }, + { name: 'esc_tx_i.esc_n', wave: '1....|......' }, + { name: 'esc_req_o', wave: '0....|1.....'}, + ], + edge: [ + 'a~>b', + 'c->d', + 'd->e', + ], + head: { + text: 'possible signal integrity failure cases', + }, + foot: { + text: 'direct signal integrity failure at time 2; indirect failure at time 6', + tick: 0, + } +} +``` + + +### Ping Testing of the Escalation Signals + + +Similarly to the alert signaling scheme, the escalation signaling lines can be pinged / line tested in order to test whether the escalation receiver has been tampered with. +This is achieved by asserting `ping_req_i` at the escalation sender module. +A ping request is encoded as a single cycle pulse on the `esc_tx_o.esc_p/n` outputs. +Hence, the receiver module will not decode this single cycle pulse as an escalation enable message, but it will respond to it with a "1010" pattern on the `esc_rx_i.resp_p/n` lines. +The escalation sender module will assert `ping_ok_o` if that pattern is received correctly after one cycle of latency. +Otherwise, the escalation sender will first assert `integ_fail_o` later, after the programmable ping timeout is reached, the LFSR timer will raise a "pingfail" alert. +The `integ_fail_o` triggers in this case since "no ping response" and "wrong ping response" are ambiguous in this setting, and it has been decided to not suppress integrity failures when expecting a ping response. + +This mechanism is illustrated below from the viewpoint of the sender module. + +```wavejson +{ + signal: [ + { name: 'clk_i', wave: 'p..............' }, + { name: 'ping_req_i', wave: '01....0|.1.....' , node: '.a'}, + { name: 'ping_ok_o', wave: '0....10|.......' , node: '.....e....g'}, + { name: 'integ_fail_o', wave: '0......|..10101' }, + { name: 'esc_req_i', wave: '0......|.......' }, + { name: 'esc_rx_i.resp_p', wave: '0.1010.|.......' , node: '..c..d....f'}, + { name: 'esc_rx_i.resp_n', wave: '1.0101.|.......' }, + { name: 'esc_tx_o.esc_p', wave: '010....|.10....' , node: '.b'}, + { name: 'esc_tx_o.esc_n', wave: '101....|.01....' }, + ], + edge: [ + 'a->b', + 'b->c', + 'd->e correct response', + 'f->g missing response', + ], + head: { + text: 'ping testing of escalation lines', + }, + foot: { + text: 'ping trig at sender input at time 1 and 9; correct response at time 5; missing response at time 10', + tick: 0, + } +} +``` + +Note that the escalation signal always takes precedence, and the `ping_req_i` will just be acknowledged with `ping_ok_o` in case `esc_req_i` is already asserted. +An ongoing ping sequence will be aborted immediately. + +Another thing to note is that the ping and escalation response sequences have to start _exactly_ one cycle after either a ping or escalation event has been signaled. +Otherwise the escalation sender will assert `integ_fail_o` immediately. + +### Monitoring of Pings at the Escalation Receiver Side + +Escalation receivers contain a mechanism to monitor the liveness of the alert handler itself. +In particular, the receivers passively monitor the ping requests sent out by the alert handler using a timeout counter. +If ping requests are absent for too long, the corresponding escalation action will be automatically asserted until reset. + +The monitoring mechanism builds on top of the following properties of the alert handler system: +1. the ping mechanism can only be enabled, but not disabled. +This allows us to start the timeout counter once the first ping request arrives at a particular escalation receiver. + +2. the escalation receivers are in the same clock/reset domain as the alert handler. +This ensures that we are seeing the same clock frequency, and the mechanism is properly reset together with the alert handler logic. + +3. the maximum cycle count between subsequent pings on the same escalation line is bounded, even though the wait counts are randomized. +This allows us to compute a safe and fixed timeout threshold based on design constants. + + +### Low-power Management of Alert Channels + +Due to the various clock and reset domains in the OpenTitan system, the alert handler ping mechanism needs to have additional logic to deal with alert senders that are either held in reset, or that are clock gated. +This is needed to ensure that no false alarms are produced by the ping mechanism when an alert channel (sender / receiver pair) does not respond due to the sender being either in reset or clock gated. + +Since the FSMs associated with an alert channel may end up in an inconsistent state when the sender is reset or gated while an asynchronous event handshake is in progress, this logic also needs to be able to re-initialize affected alert channels whenever the channels comes back from reset / clock gated state. + +#### Assumptions + +The following diagram shows a typical arrangement of alert sender (TX) and receiver (RX) pairs. + +![Alert Handler Low-Power Overview](alert_handler_lp_overview.svg) + +It is assumed that: + +1. The alert handler clock domain cannot be gated by SW. + This means that this clock domain is only ever disabled as part of the power-down sequence of the corresponding power domain. +2. The alert senders are in general located within different clock and reset domains than the alert receivers within the alert handler, and thus use the asynchronous event handshake mode. +3. Some alert senders may be located in an always-on (AON) power domain, within different clock and reset groups than the alert handler. +4. The alert handler may be located in an non-AON power domain, and may thus undergo a reset cycle where it cannot be guaranteed that all alert senders are reset as well (i.e., some alert senders may retain their state). + +Further, we assume that we can get the following side-band information from the clock and reset managers in the system: + +- All relevant reset signals pertaining to alert sender domains +- All relevant clock enable signals pertaining to alert sender domains + +#### Scenarios + +With the assumptions above, the following two problematic scenarios can occur. + +##### Alert Handler in Reset + +It may happen that the alert handler is reset while some alert senders (e.g. those located in the AON domain) are not. +In general, if the associated alert channels are idle during an alert handler reset cycle, no problems arise. + +However, if an alert channel is reset while it is handling a ping request or an alert event, the sender / receiver FSMs may end up in an inconsistent state upon deassertion of the alert handler reset. +This can either lead to spurious alert or ping events, or a completely locked up alert channel which will be flagged eventually by the ping mechanism. + +##### Alert Sender in Reset or Clock-gated + +If any of the alert senders is either put into reset or its clock is disabled while the alert handler is operational, the ping mechanism inside the alert handler will eventually report a ping failure because of missing ping responses from the affected alert channel(s). + +Further, if the alert sender is reset while the corresponding alert channel is handling a ping request or an alert event, the sender / receiver FSMs may end up in an inconsistent state after reset deassertion. + +#### Employed Solution + +As elaborated before, the side effects of resetting / clock gating either the alert handler or any of the alert senders are inconsistent FSM states, leading to locked up alert channels, or spurious alert or ping events. +To address these issues, we have to: + +1. make sure spurious events (alert and ping_ok outputs of the alert receivers) are suppressed if an alert channel is clock gated or in reset, +2. provide a mechanism for resetting an alert channel to an operational state once the associated clock is re-enabled, or the associated reset is released, +3. trigger this reset mechanism on all alert channels whenever the alert handler itself has been reset. + +To attain this, the idea is to make use of side-band information available from the clock and reset managers to detect whether an alert channel (or a group of alert channels with the same clock and reset on the sender side) has to be put into a low-power state. +In the following we will refer to such a clock / reset domain grouping as a low-power group (LPG). + +The mechanism is illustrated below for a single LPG (in practice, this logic is replicated for each LPG that is identified in the system): + +![Alert Handler LPG Ctrl](alert_handler_lpg_ctrl.svg) + +The clock gating enable (`lpg_cg_en`) and reset enable (`lpg_rst_en`) indications are routed as multibit signals to the alert handler, where they are synchronized to the alert handler clock and logically combined using an OR function to form a combined low-power indication signal that is multibit encoded. + +This multibit indication signal is then routed to all alert receivers, where it is used to trigger re-initialization of each alert channel, and bypass the ping requests from the ping mechanism. + +To that end, two extra *init states* are added to the alert receiver FSMs to perform this in-band reset, as indicated in the state diagram below: + +![Alert Handler Receiver FSM](alert_handler_receiver_fsm.svg) + +Whenever the `init_trig` multibit signal of an LPG is asserted, all corresponding sender FSMs are moved into the `InitReq` state. +In that state, the alert receivers immediately acknowledge ping requests from the ping mechanism, and ignore alert events from the sender side. +In addition to that, the receivers intentionally place a signal integrity error onto the `ping_p` / `ping_n`, `ack_p` / `ack_n` lines going from receivers to the senders. +This causes the senders to 1) move into the signal integrity error state, and 2) respond by placing a signal integrity error onto the `alert_p` / `alert_n` lines, which serves as an initialization "acknowledge" signal in this case. +Since the sender FSMs fall back into the `Idle` state once the signal integrity error disappears, this procedure essentially implements an in-band reset mechanism with an acknowledgement handshake that can be used to determine whether the reset has been successful. + +#### Implementation Aspects + +##### Ping Mechanism Bypass + +Note that the ping bypass mechanism is to be implemented in a way that pings are only ack'ed immediately if 1) the FSM is in the `InitReq` state, and 2) the `init_trig` signal is still asserted. + +This allows to subject the initialization process of each alert channel to the ping mechanism for channels that are recovering from a reset or clock gated cycle on the sender side. +I.e., alert channels that get stuck during the initialization process can be detected by the ping mechanism since ping requests are not immediately ack'ed anymore once `init_trig` is deasserted. + +##### FSM Encoding + +Since there are many alert channels in the design, the receiver and sender FSMs themselves are not sparsely encoded. +Instead, we rely on the ping mechanism to detect alert channels that are in a bad state. +The specific implementation of the ping bypass mentioned in the previous subsection ensures that the ping mechanism can also be used to monitor the initialization sequence of alert channels. + +##### Latency / Skew Considerations + +Due to asynchronous transitions and different path latencies in the system, a change in reset or clock gating state will experience a different latency through the alert channels than through the indication signals (`rst_n` and `clk_en`) that are connected to the low-power control logic. + +It is consequently possible for a group of alert senders to already be in reset or clock gated state, while the corresponding LPG logic does not yet know about this state change - and vice versa. + +In practice, this means that ping requests may be pending for several cycles until the LPG logic detects a reset or clock-gated condition and disables the corresponding alert channel(s). +Fortunately, such delay can be tolerated by setting the ping timeout to a sufficiently large value (see [`CLASSA_TIMEOUT_CYC_SHADOWED`](../data/alert_handler.hjson#classa_timeout_cyc_shadowed)). + +As for alert events, this latency difference should not pose a problem. +Alert events may get stuck in the alert sender due to a reset or clock-gated condition - but this is to be expected. + +##### Integration Considerations + +Note that due to the aforementioned latency tolerance built into the ping timer, it is permissible to connect **any** reset or clock enable indication signal from the relevant clock group to the LPG logic. +I.e., the only requirement is that the indication signals are logically related to the resets and clocks routed to the alert senders, and that the skew between reset / clock state changes and the indication signals is bounded. + +The topgen script is extended so that it can identify all LPGs and the associated alert channels. +This information is then used to parameterize the alert handler design, and make the necessary top-level connections from the reset and clock management controllers to the alert handler. + +### Hardening Against Glitch Attacks + +In addition to the differential alert and escalation signalling scheme, the internal state machines and counters are hardened against glitch attacks as described bellow: + +1. Ping Timer: + - The FSM is sparsely encoded. + - The LFSR and the counter are duplicated. + - If the FSM or counter are glitched into an invalid state, all internal ping fail alerts will be permanently asserted. + +2. Escalation Timers: + - The escalation timer FSMs are sparsely encoded. + - The escalation timer counters are duplicated. + - The escalation accumulators are duplicated. + - If one of these FSMs, counters or accumulators are glitched into an invalid state, all escalation actions will be triggered and the affected FSM goes into a terminal `FsmError` state. + +3. CSRs: + - Critical configuration CSRs are shadowed. + - The shadow CSRs can trigger additional internal alerts for CSR storage and update failures. + These internal alerts are fed back into the alert classifier in the same manner as the ping and integrity failure alerts. + +4. LPGs: + - Clock-gated and reset-asserted indication signals that are routed from clock and reset managers to the alert handler are encoded with multibit signals. diff --git a/hw/top_earlgrey/ip_autogen/rv_plic/README.md b/hw/top_earlgrey/ip_autogen/rv_plic/README.md index d642009cb0434..1436784f7a380 100644 --- a/hw/top_earlgrey/ip_autogen/rv_plic/README.md +++ b/hw/top_earlgrey/ip_autogen/rv_plic/README.md @@ -4,7 +4,7 @@ This document specifies the Interrupt Controller (RV_PLIC) functionality. This module conforms to the -[Comportable guideline for peripheral functionality](../../../doc/contributing/hw/comportability/README.md). +[Comportable guideline for peripheral functionality](../../../../doc/contributing/hw/comportability/README.md). See that document for integration overview within the broader top level system. @@ -24,224 +24,3 @@ incoming interrupt signals (``intr_src_i``) and can notify multiple targets. ## Compatibility The RV_PLIC is compatible with any RISC-V core implementing the RISC-V privilege specification. - -# Theory of Operations - -## Block Diagram - -![RV_PLIC Block Diagram](./doc/block_diagram.svg) - -## Hardware Interfaces - -* [Interface Tables](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#interfaces) - -## Design Details - -### Identifier - -Each interrupt source has a unique ID assigned based upon its bit position -within the input `intr_src_i`. ID ranges from 0 to N, the number of interrupt -sources. ID 0 is reserved and represents no interrupt. The bit 0 of -`intr_src_i` shall be tied to 0 from the outside of RV_PLIC. The -`intr_src_i[i]` bit has an ID of `i`. This ID is used when targets "claim" the -interrupt and to "complete" the interrupt event. - -### Priority and Threshold - -Interrupt sources have configurable priority values. The maximum value of the -priority is configurable through the localparam `MAX_PRIO` in the rv_plic -top-level module. For each target there is a threshold value ([`THRESHOLD0`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#threshold0) for -target 0). RV_PLIC notifies a target of an interrupt only if it's priority is -strictly greater than the target's threshold. Note this means an interrupt with -a priority is 0 is effectively prevented from causing an interrupt at any target -and a target can suppress all interrupts by setting it's threshold to the max -priority value. - -`MAX_PRIO` parameter is most area contributing option in RV_PLIC. If `MAX_PRIO` -is big, then finding the highest priority in Process module may consume a lot of -logic gates. - -### Interrupt Gateways - -The Gateway observes incoming interrupt sources and converts them to a common -interrupt format used internally by RV_PLIC. It can be parameterized to detect -interrupts events on an edge (when the signal changes from **0** to **1**) or -level basis (where the signal remains at **1**). -The choice is a system-integration decision and can be configured via the design parameter `LevelEdgeTrig` for each interrupt request. - -When the gateway detects an interrupt event it raises the interrupt pending bit -([`IP`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#ip)) for that interrupt source. When an interrupt is claimed by a target the -relevant bit of [`IP`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#ip) is cleared. A bit in [`IP`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#ip) will not be reasserted until the -target signals completion of the interrupt. Any new interrupt event between a -bit in [`IP`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#ip) asserting and completing that interrupt is ignored. In particular -this means that for edge triggered interrupts if a new edge is seen after the -source's [`IP`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#ip) bit is asserted but before completion, that edge will be ignored -(counting missed edges as discussed in the RISC-V PLIC specification is not -supported). - -Note that there is no ability for a level triggered interrupt to be cancelled. -If the interrupt drops after the gateway has set a bit in [`IP`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#ip), the bit will -remain set until the interrupt is completed. The SW handler should be conscious -of this and check the interrupt still requires handling in the handler if this -behaviour is possible. - -### Interrupt Enables - -Each target has a set of Interrupt Enable ([`IE0`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#ie0) for target 0) registers. Each -bit in the [`IE0`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#ie0) registers controls the corresponding interrupt source. If an -interrupt source is disabled for a target, then interrupt events from that -source won't trigger an interrupt at the target. RV_PLIC doesn't have a global -interrupt disable feature. - -### Interrupt Claims - -"Claiming" an interrupt is done by a target reading the associated -Claim/Completion register for the target ([`CC0`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#cc0) for target 0). The return value -of the [`CC0`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#cc0) read represents the ID of the pending interrupt that has the -highest priority. If two or more pending interrupts have the same priority, -RV_PLIC chooses the one with lowest ID. Only interrupts that that are enabled -for the target can be claimed. The target priority threshold doesn't matter -(this only factors into whether an interrupt is signalled to the target) so -lower priority interrupt IDs can be returned on a read from [`CC0`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#cc0). If no -interrupt is pending (or all pending interrupts are disabled for the target) a -read of [`CC0`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#cc0) returns an ID of 0. - -### Interrupt Completion - -After an interrupt is claimed, the relevant bit of interrupt pending ([`IP`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#ip)) is -cleared, regardless of the status of the `intr_src_i` input value. Until a -target "completes" the interrupt, it won't be re-asserted if a new event for the -interrupt occurs. A target completes the interrupt by writing the ID of the -interrupt to the Claim/Complete register ([`CC0`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#cc0) for target 0). The write event -is forwarded to the Gateway logic, which resets the interrupt status to accept a -new interrupt event. The assumption is that the processor has cleaned up the -originating interrupt event during the time between claim and complete such that -`intr_src_i[ID]` will have de-asserted (unless a new interrupt has occurred). - -```wavejson -{ signal: [ - { name: 'clk', wave: 'p...........' }, - { name: 'intr_src_i[i]', wave: '01....0.1...', node:'.a....e.f...'}, - { name: 'irq_o', wave: '0.1.0......1', node:'..b.d......h'}, - { name: 'irq_id_o', wave: '=.=.=......=', - data: ["0","i","0","i"] }, - { name: 'claim', wave: '0..10.......', node:'...c........'}, - { name: 'complete', wave: '0.........10', node:'..........g.'}, - ], - head:{ - text: 'Interrupt Flow', - tick: 0, - }, -} -``` - -In the example above an interrupt for source ID `i` is configured as a level -interrupt and is raised at a, this results in the target being notified of the -interrupt at b. The target claims the interrupt at c (reading `i` from it's -Claim/Complete register) so `irq_o` deasserts though `intr_src_i[i]` remains -raised. The SW handles the interrupt and it drops at e. However a new interrupt -quickly occurs at f. As complete hasn't been signaled yet `irq_o` isn't -asserted. At g the interrupt is completed (by writing `i` to it's -Claim/Complete register) so at h `irq_o` is asserted due to the new interrupt. - - -# Programmers Guide - -## Initialization - -After reset, RV_PLIC doesn't generate any interrupts to any targets even if -interrupt sources are set, as all priorities and thresholds are 0 by default and -all ``IE`` values are 0. Software should configure the above three registers. - -[`PRIO0`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#prio0) .. [`PRIO31`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#prio1) registers are unique. So, only one of the targets -shall configure them. - -```c -// Pseudo-code below -void plic_init() { - // Configure priority - // Note that PRIO0 register doesn't affect as intr_src_i[0] is tied to 0. - for (int i = 0; i < N_SOURCE; ++i) { - *(PRIO + i) = value(i); - } -} - -void plic_threshold(tid, threshold) { - *(THRESHOLD + tid) = threshold; -} - -void plic_enable(tid, iid) { - // iid: 0-based ID - int offset = ceil(N_SOURCE / 32) * tid + (iid >> 5); - - *(IE + offset) = *(IE + offset) | (1 << (iid % 32)); -} -``` - -## Handling Interrupt Request Events - -If software receives an interrupt request, it is recommended to follow the steps -shown below (assuming target 0 which uses [`CC0`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#cc0) for claim/complete). - -1. Claim the interrupts right after entering to the interrupt service routine - by reading the [`CC0`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#cc0) register. -2. Determine which interrupt should be serviced based on the values read from - the [`CC0`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#cc0) register. -3. Execute ISR, clearing the originating peripheral interrupt. -4. Write Interrupt ID to [`CC0`](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#cc0) -5. Repeat as necessary for other pending interrupts. - -It is possible to have multiple interrupt events claimed. If software claims one -interrupt request, then the process module advertises any pending interrupts -with lower priority unless new higher priority interrupt events occur. If a -higher interrupt event occurs after previous interrupt is claimed, the RV_PLIC -IP advertises the higher priority interrupt. Software may utilize an event -manager inside a loop so that interrupt claiming and completion can be -separated. - -~~~~c -void interrupt_service() { - uint32_t tid = /* ... */; - uint32_t iid = *(CC + tid); - if (iid == 0) { - // Interrupt is claimed by one of other targets. - return; - } - - do { - // Process interrupts... - // ... - - // Finish. - *(CC + tid) = iid; - iid = *(CC + tid); - } while (iid != 0); -} -~~~~ - -As a reference, default interrupt service routines are auto-generated for each -IP, and are documented [here](/sw/apis/isr__testutils_8h.html). - -## Device Interface Functions (DIFs) - -- [Device Interface Functions](../../../sw/device/lib/dif/dif_rv_plic.h) - -## Registers - -The RV_PLIC in the top level is generated by topgen tool so that the number of -interrupt sources may be different. - -- IE: CEILING(N_SOURCE / DW) X N_TARGET - Each bit enables corresponding interrupt source. Each target has IE set. -- PRIO: N_SOURCE - Universal set across all targets. Lower n bits are valid. n is determined by - MAX_PRIO parameter -- THRESHOLD: N_TARGET - Priority threshold per target. Only priority of the interrupt greater than - threshold can raise interrupt notification to the target. -- IP: CEILING(N_SOURCE / DW) - Pending bits right after the gateways. Read-only -- CC: N_TARGET - Claim by read, complete by write - -* [Register Table](../../top_earlgrey/ip_autogen/rv_plic/data/rv_plic.hjson#interfaces) diff --git a/hw/top_earlgrey/ip_autogen/rv_plic/doc/programmers_guide.md b/hw/top_earlgrey/ip_autogen/rv_plic/doc/programmers_guide.md new file mode 100644 index 0000000000000..cdbd8f2c62a3a --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/rv_plic/doc/programmers_guide.md @@ -0,0 +1,100 @@ +# Programmer's Guide + +## Initialization + +After reset, RV_PLIC doesn't generate any interrupts to any targets even if +interrupt sources are set, as all priorities and thresholds are 0 by default and +all ``IE`` values are 0. Software should configure the above three registers. + +[`PRIO0`](../data/rv_plic.hjson#prio0) .. [`PRIO31`](../data/rv_plic.hjson#prio1) registers are unique. So, only one of the targets +shall configure them. + +```c +// Pseudo-code below +void plic_init() { + // Configure priority + // Note that PRIO0 register doesn't affect as intr_src_i[0] is tied to 0. + for (int i = 0; i < N_SOURCE; ++i) { + *(PRIO + i) = value(i); + } +} + +void plic_threshold(tid, threshold) { + *(THRESHOLD + tid) = threshold; +} + +void plic_enable(tid, iid) { + // iid: 0-based ID + int offset = ceil(N_SOURCE / 32) * tid + (iid >> 5); + + *(IE + offset) = *(IE + offset) | (1 << (iid % 32)); +} +``` + +## Handling Interrupt Request Events + +If software receives an interrupt request, it is recommended to follow the steps +shown below (assuming target 0 which uses [`CC0`](../data/rv_plic.hjson#cc0) for claim/complete). + +1. Claim the interrupts right after entering to the interrupt service routine + by reading the [`CC0`](../data/rv_plic.hjson#cc0) register. +2. Determine which interrupt should be serviced based on the values read from + the [`CC0`](../data/rv_plic.hjson#cc0) register. +3. Execute ISR, clearing the originating peripheral interrupt. +4. Write Interrupt ID to [`CC0`](../data/rv_plic.hjson#cc0) +5. Repeat as necessary for other pending interrupts. + +It is possible to have multiple interrupt events claimed. If software claims one +interrupt request, then the process module advertises any pending interrupts +with lower priority unless new higher priority interrupt events occur. If a +higher interrupt event occurs after previous interrupt is claimed, the RV_PLIC +IP advertises the higher priority interrupt. Software may utilize an event +manager inside a loop so that interrupt claiming and completion can be +separated. + +~~~~c +void interrupt_service() { + uint32_t tid = /* ... */; + uint32_t iid = *(CC + tid); + if (iid == 0) { + // Interrupt is claimed by one of other targets. + return; + } + + do { + // Process interrupts... + // ... + + // Finish. + *(CC + tid) = iid; + iid = *(CC + tid); + } while (iid != 0); +} +~~~~ + +As a reference, default interrupt service routines are auto-generated for each +IP, and are documented [here](/sw/apis/isr__testutils_8h.html). + +## Device Interface Functions (DIFs) + +- [Device Interface Functions](../../../../../sw/device/lib/dif/dif_rv_plic.h) + +## Registers + +The RV_PLIC in the top level is generated by topgen tool so that the number of +interrupt sources may be different. + +- IE: CEILING(N_SOURCE / DW) X N_TARGET + Each bit enables corresponding interrupt source. Each target has IE set. +- PRIO: N_SOURCE + Universal set across all targets. Lower n bits are valid. n is determined by + MAX_PRIO parameter +- THRESHOLD: N_TARGET + Priority threshold per target. Only priority of the interrupt greater than + threshold can raise interrupt notification to the target. +- IP: CEILING(N_SOURCE / DW) + Pending bits right after the gateways. Read-only +- CC: N_TARGET + Claim by read, complete by write + +* [Register Table](../data/rv_plic.hjson#interfaces) diff --git a/hw/top_earlgrey/ip_autogen/rv_plic/doc/theory_of_operation.md b/hw/top_earlgrey/ip_autogen/rv_plic/doc/theory_of_operation.md new file mode 100644 index 0000000000000..aebdf43bff497 --- /dev/null +++ b/hw/top_earlgrey/ip_autogen/rv_plic/doc/theory_of_operation.md @@ -0,0 +1,119 @@ +# Theory of Operation + +## Block Diagram + +![RV_PLIC Block Diagram](block_diagram.svg) + +## Hardware Interfaces + +* [Interface Tables](../data/rv_plic.hjson#interfaces) + +## Design Details + +### Identifier + +Each interrupt source has a unique ID assigned based upon its bit position +within the input `intr_src_i`. ID ranges from 0 to N, the number of interrupt +sources. ID 0 is reserved and represents no interrupt. The bit 0 of +`intr_src_i` shall be tied to 0 from the outside of RV_PLIC. The +`intr_src_i[i]` bit has an ID of `i`. This ID is used when targets "claim" the +interrupt and to "complete" the interrupt event. + +### Priority and Threshold + +Interrupt sources have configurable priority values. The maximum value of the +priority is configurable through the localparam `MAX_PRIO` in the rv_plic +top-level module. For each target there is a threshold value ([`THRESHOLD0`](../data/rv_plic.hjson#threshold0) for +target 0). RV_PLIC notifies a target of an interrupt only if it's priority is +strictly greater than the target's threshold. Note this means an interrupt with +a priority is 0 is effectively prevented from causing an interrupt at any target +and a target can suppress all interrupts by setting it's threshold to the max +priority value. + +`MAX_PRIO` parameter is most area contributing option in RV_PLIC. If `MAX_PRIO` +is big, then finding the highest priority in Process module may consume a lot of +logic gates. + +### Interrupt Gateways + +The Gateway observes incoming interrupt sources and converts them to a common +interrupt format used internally by RV_PLIC. It can be parameterized to detect +interrupts events on an edge (when the signal changes from **0** to **1**) or +level basis (where the signal remains at **1**). +The choice is a system-integration decision and can be configured via the design parameter `LevelEdgeTrig` for each interrupt request. + +When the gateway detects an interrupt event it raises the interrupt pending bit +([`IP`](../data/rv_plic.hjson#ip)) for that interrupt source. When an interrupt is claimed by a target the +relevant bit of [`IP`](../data/rv_plic.hjson#ip) is cleared. A bit in [`IP`](../data/rv_plic.hjson#ip) will not be reasserted until the +target signals completion of the interrupt. Any new interrupt event between a +bit in [`IP`](../data/rv_plic.hjson#ip) asserting and completing that interrupt is ignored. In particular +this means that for edge triggered interrupts if a new edge is seen after the +source's [`IP`](../data/rv_plic.hjson#ip) bit is asserted but before completion, that edge will be ignored +(counting missed edges as discussed in the RISC-V PLIC specification is not +supported). + +Note that there is no ability for a level triggered interrupt to be cancelled. +If the interrupt drops after the gateway has set a bit in [`IP`](../data/rv_plic.hjson#ip), the bit will +remain set until the interrupt is completed. The SW handler should be conscious +of this and check the interrupt still requires handling in the handler if this +behaviour is possible. + +### Interrupt Enables + +Each target has a set of Interrupt Enable ([`IE0`](../data/rv_plic.hjson#ie0) for target 0) registers. Each +bit in the [`IE0`](../data/rv_plic.hjson#ie0) registers controls the corresponding interrupt source. If an +interrupt source is disabled for a target, then interrupt events from that +source won't trigger an interrupt at the target. RV_PLIC doesn't have a global +interrupt disable feature. + +### Interrupt Claims + +"Claiming" an interrupt is done by a target reading the associated +Claim/Completion register for the target ([`CC0`](../data/rv_plic.hjson#cc0) for target 0). The return value +of the [`CC0`](../data/rv_plic.hjson#cc0) read represents the ID of the pending interrupt that has the +highest priority. If two or more pending interrupts have the same priority, +RV_PLIC chooses the one with lowest ID. Only interrupts that that are enabled +for the target can be claimed. The target priority threshold doesn't matter +(this only factors into whether an interrupt is signalled to the target) so +lower priority interrupt IDs can be returned on a read from [`CC0`](../data/rv_plic.hjson#cc0). If no +interrupt is pending (or all pending interrupts are disabled for the target) a +read of [`CC0`](../data/rv_plic.hjson#cc0) returns an ID of 0. + +### Interrupt Completion + +After an interrupt is claimed, the relevant bit of interrupt pending ([`IP`](../data/rv_plic.hjson#ip)) is +cleared, regardless of the status of the `intr_src_i` input value. Until a +target "completes" the interrupt, it won't be re-asserted if a new event for the +interrupt occurs. A target completes the interrupt by writing the ID of the +interrupt to the Claim/Complete register ([`CC0`](../data/rv_plic.hjson#cc0) for target 0). The write event +is forwarded to the Gateway logic, which resets the interrupt status to accept a +new interrupt event. The assumption is that the processor has cleaned up the +originating interrupt event during the time between claim and complete such that +`intr_src_i[ID]` will have de-asserted (unless a new interrupt has occurred). + +```wavejson +{ signal: [ + { name: 'clk', wave: 'p...........' }, + { name: 'intr_src_i[i]', wave: '01....0.1...', node:'.a....e.f...'}, + { name: 'irq_o', wave: '0.1.0......1', node:'..b.d......h'}, + { name: 'irq_id_o', wave: '=.=.=......=', + data: ["0","i","0","i"] }, + { name: 'claim', wave: '0..10.......', node:'...c........'}, + { name: 'complete', wave: '0.........10', node:'..........g.'}, + ], + head:{ + text: 'Interrupt Flow', + tick: 0, + }, +} +``` + +In the example above an interrupt for source ID `i` is configured as a level +interrupt and is raised at a, this results in the target being notified of the +interrupt at b. The target claims the interrupt at c (reading `i` from it's +Claim/Complete register) so `irq_o` deasserts though `intr_src_i[i]` remains +raised. The SW handles the interrupt and it drops at e. However a new interrupt +quickly occurs at f. As complete hasn't been signaled yet `irq_o` isn't +asserted. At g the interrupt is completed (by writing `i` to it's +Claim/Complete register) so at h `irq_o` is asserted due to the new interrupt. +