blob: 6fb5bf9ef179901879f648ac635d05e58e11a52a [file] [log] [blame]
// Copyright lowRISC contributors.
// Licensed under the Apache License, Version 2.0, see LICENSE for details.
// SPDX-License-Identifier: Apache-2.0
// clang-format off
${gencmd}
<%
irq_peripheral_names = sorted({p.name for p in helper.irq_peripherals})
# For some rv_timer DIFs, tha hart argument follows the instance handle.
def args(p):
extra_arg = ", kHart" if p.name == "rv_timer" else ""
return f"&{p.inst_name}{extra_arg}"
%>\
#include "sw/device/lib/base/freestanding/limits.h"
#include "sw/device/lib/base/mmio.h"
% for n in sorted(irq_peripheral_names + ["rv_plic"]):
#include "sw/device/lib/dif/dif_${n}.h"
% endfor
#include "sw/device/lib/handler.h"
#include "sw/device/lib/irq.h"
#include "sw/device/lib/runtime/log.h"
#include "sw/device/lib/testing/check.h"
#include "sw/device/lib/testing/rv_plic_testutils.h"
#include "sw/device/lib/testing/test_framework/test_main.h"
#include "sw/device/lib/testing/test_framework/test_status.h"
#include "hw/top_earlgrey/sw/autogen/top_earlgrey.h"
% for p in helper.irq_peripherals:
static dif_${p.name}_t ${p.inst_name};
% endfor
static dif_rv_plic_t plic;
static const top_earlgrey_plic_target_t kHart = kTopEarlgreyPlicTargetIbex0;
/**
* Flag indicating which peripheral is under test.
*
* Declared volatile because it is referenced in the main program flow as well
* as the ISR.
*/
static volatile top_earlgrey_plic_peripheral_t peripheral_expected;
/**
* Flags indicating the IRQ expected to have triggered and serviced within the
* peripheral.
*
* Declared volatile because it is referenced in the main program flow as well
* as the ISR.
*/
% for n in irq_peripheral_names:
static volatile dif_${n}_irq_t ${n}_irq_expected;
static volatile dif_${n}_irq_t ${n}_irq_serviced;
% endfor
/**
* Provides external IRQ handling for this test.
*
* This function overrides the default external IRQ handler in
* `sw/device/lib/handler.h`.
*
* For each IRQ, it performs the following:
* 1. Claims the IRQ fired (finds PLIC IRQ index).
* 2. Checks that the index belongs to the expected peripheral.
* 3. Checks that the correct and the only IRQ from the expected peripheral
* triggered.
* 4. Clears the IRQ at the peripheral.
* 5. Completes the IRQ service at PLIC.
*/
void handler_irq_external(void) {
dif_rv_plic_irq_id_t plic_irq_id;
CHECK_DIF_OK(dif_rv_plic_irq_claim(&plic, kHart, &plic_irq_id));
top_earlgrey_plic_peripheral_t peripheral = (top_earlgrey_plic_peripheral_t)
top_earlgrey_plic_interrupt_for_peripheral[plic_irq_id];
CHECK(peripheral == peripheral_expected,
"Interrupt from incorrect peripheral: exp = %d, obs = %d",
peripheral_expected, peripheral);
switch (peripheral) {
% for p in helper.irq_peripherals:
case ${p.plic_name}: {
dif_${p.name}_irq_t irq = (dif_${p.name}_irq_t)(
plic_irq_id -
(dif_rv_plic_irq_id_t)${p.plic_start_irq});
CHECK(irq == ${p.name}_irq_expected,
"Incorrect ${p.inst_name} IRQ triggered: exp = %d, obs = %d",
${p.name}_irq_expected, irq);
${p.name}_irq_serviced = irq;
dif_${p.name}_irq_state_snapshot_t snapshot;
CHECK_DIF_OK(dif_${p.name}_irq_get_state(${args(p)}, &snapshot));
CHECK(snapshot == (dif_${p.name}_irq_state_snapshot_t)(1 << irq),
"Only ${p.inst_name} IRQ %d expected to fire. Actual interrupt "
"status = %x",
irq, snapshot);
CHECK_DIF_OK(dif_${p.name}_irq_acknowledge(&${p.inst_name}, irq));
break;
}
% endfor
default:
LOG_FATAL("ISR is not implemented!");
test_status_set(kTestStatusFailed);
}
// Complete the IRQ at PLIC.
CHECK_DIF_OK(dif_rv_plic_irq_complete(&plic, kHart, plic_irq_id));
}
/**
* Initializes the handles to all peripherals.
*/
static void peripherals_init(void) {
mmio_region_t base_addr;
% for p in helper.irq_peripherals:
base_addr = mmio_region_from_addr(${p.base_addr_name});
CHECK_DIF_OK(dif_${p.name}_init(base_addr, &${p.inst_name}));
% endfor
base_addr = mmio_region_from_addr(TOP_EARLGREY_RV_PLIC_BASE_ADDR);
CHECK_DIF_OK(dif_rv_plic_init(base_addr, &plic));
}
/**
* Clears pending IRQs in all peripherals.
*/
static void peripheral_irqs_clear(void) {
% for p in helper.irq_peripherals:
CHECK_DIF_OK(dif_${p.name}_irq_acknowledge_all(${args(p)}));
% endfor
}
/**
* Enables all IRQs in all peripherals.
*/
static void peripheral_irqs_enable(void) {
% for n in irq_peripheral_names:
<%
if n == "aon_timer": continue
%>\
dif_${n}_irq_state_snapshot_t ${n}_irqs =
(dif_${n}_irq_state_snapshot_t)UINT_MAX;
% endfor
% for p in helper.irq_peripherals:
<%
if p.name == "aon_timer": continue
%>\
CHECK_DIF_OK(
dif_${p.name}_irq_restore_all(${args(p)}, &${p.name}_irqs));
% endfor
}
/**
* Triggers all IRQs in all peripherals one by one.
*
* Walks through all instances of all peripherals and triggers an interrupt one
* by one, by forcing with the `intr_test` CSR. On trigger, the CPU instantly
* jumps into the ISR. The main flow of execution thus proceeds to check that
* the correct IRQ was serviced immediately. The ISR, in turn checks if the
* expected IRQ from the expected peripheral triggered.
*/
static void peripheral_irqs_trigger(void) {
% for p in helper.irq_peripherals:
peripheral_expected = ${p.plic_name};
for (dif_${p.name}_irq_t irq = ${p.start_irq};
irq <= ${p.end_irq}; ++irq) {
${p.name}_irq_expected = irq;
LOG_INFO("Triggering ${p.inst_name} IRQ %d.", irq);
CHECK_DIF_OK(dif_${p.name}_irq_force(&${p.inst_name}, irq));
CHECK(${p.name}_irq_serviced == irq,
"Incorrect ${p.inst_name} IRQ serviced: exp = %d, obs = %d", irq,
${p.name}_irq_serviced);
LOG_INFO("IRQ %d from ${p.inst_name} is serviced.", irq);
}
${"" if loop.last else "\n"}\
% endfor
}
const test_config_t kTestConfig;
bool test_main(void) {
irq_global_ctrl(true);
irq_external_ctrl(true);
peripherals_init();
rv_plic_testutils_irq_range_enable(
&plic, kHart, kTopEarlgreyPlicIrqIdNone + 1, kTopEarlgreyPlicIrqIdLast);
peripheral_irqs_clear();
peripheral_irqs_enable();
peripheral_irqs_trigger();
return true;
}
// clang-format on