Files
linux-nvgpu/userspace/units/gr/intr/nvgpu-gr-intr.c
Shashank Singh 9512b9f1de gpu: nvgpu: remove user managed addr space capability flag
Remove NVGPU_GPU_IOCTL_ALLOC_AS_FLAGS_USERSPACE_MANAGED and
NVGPU_AS_ALLOC_USERSPACE_MANAGED flags which are used for supporting
userspace managed address-space. This functionality is not implemented
fully in kernel neither going to be implemented in near future.

Jira NVGPU-9832
Bug 4034184

Change-Id: I3787d92c44682b02d440e52c7a0c8c0553742dcc
Signed-off-by: Shashank Singh <shashsingh@nvidia.com>
Reviewed-on: https://git-master.nvidia.com/r/c/linux-nvgpu/+/2882168
Tested-by: mobile promotions <svcmobile_promotions@nvidia.com>
Reviewed-by: mobile promotions <svcmobile_promotions@nvidia.com>
2023-05-04 11:39:30 -07:00

1279 lines
36 KiB
C

/*
* Copyright (c) 2019-2023, NVIDIA CORPORATION. All rights reserved.
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, including without limitation
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
* and/or sell copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE.
*/
#include <stdlib.h>
#include <unistd.h>
#include <unit/unit.h>
#include <unit/io.h>
#include <nvgpu/posix/io.h>
#include <nvgpu/posix/kmem.h>
#include <nvgpu/posix/posix-fault-injection.h>
#include <nvgpu/types.h>
#include <nvgpu/gk20a.h>
#include <nvgpu/channel.h>
#include <nvgpu/runlist.h>
#include <nvgpu/tsg.h>
#include <nvgpu/class.h>
#include <nvgpu/gr/ctx.h>
#include <nvgpu/gr/gr_intr.h>
#include <nvgpu/tsg_subctx.h>
#include <nvgpu/hw/gv11b/hw_gr_gv11b.h>
#include "hal/gr/intr/gr_intr_gv11b.h"
#include "hal/gr/intr/gr_intr_gp10b.h"
#include "common/gr/gr_intr_priv.h"
#include "common/gr/gr_priv.h"
#include "../nvgpu-gr.h"
#include "nvgpu-gr-intr.h"
#define TPC_EXCEPTION_TEX (0x1U << 0U)
#define TPC_EXCEPTION_SM (0x1U << 1U)
#define TPC_SM0_ESR_SEL (0x1U << 0U)
#define TPC_SM1_ESR_SEL (0x1U << 1U)
#define GR_TEST_TRAPPED_ADDR_DATAHIGH 0x0F000000
#define GR_TEST_CHANNEL_MAP_TLB_SIZE 0x2
struct test_gr_intr_sw_mthd_exceptions {
int trapped_addr;
int data[2];
};
struct gr_gops_intr_orgs {
void (*handle_tpc_sm_ecc_exception)(struct gk20a *g,
u32 gpc, u32 tpc);
void (*handle_tpc_mpc_exception)(struct gk20a *g,
u32 gpc, u32 tpc);
void (*handle_tpc_pe_exception)(struct gk20a *g,
u32 gpc, u32 tpc);
u32 (*get_ctxsw_checksum_mismatch_mailbox_val)(void);
void (*handle_ssync_hww)(struct gk20a *g,
u32 *ssync_esr);
void (*handle_gcc_exception)(struct gk20a *g, u32 gpc,
u32 gpc_exception, u32 *corrected_err, u32 *uncorrected_err);
void (*handle_gpc_gpcmmu_exception)(struct gk20a *g, u32 gpc,
u32 gpc_exception, u32 *corrected_err, u32 *uncorrected_err);
void (*handle_gpc_prop_exception)(struct gk20a *g, u32 gpc,
u32 gpc_exception);
void (*handle_gpc_zcull_exception)(struct gk20a *g, u32 gpc,
u32 gpc_exception);
void (*handle_gpc_setup_exception)(struct gk20a *g, u32 gpc,
u32 gpc_exception);
void (*handle_gpc_pes_exception)(struct gk20a *g, u32 gpc,
u32 gpc_exception);
void (*handle_gpc_gpccs_exception)(struct gk20a *g, u32 gpc,
u32 gpc_exception, u32 *corrected_err, u32 *uncorrected_err);
u64 (*get_sm_hww_warp_esr_pc)(struct gk20a *g, u32 offset);
bool (*handle_exceptions)(struct gk20a *g,
bool *is_gpc_exception);
#ifdef CONFIG_NVGPU_RECOVERY
void (*recover)(struct gk20a *g, u32 bitmask, u32 id,
unsigned int id_type, unsigned int rc_type,
struct mmu_fault_info *mmufault);
#endif
};
struct gr_gops_intr_orgs gr_test_intr_gops;
static void gr_test_save_intr_gops(struct gk20a *g)
{
gr_test_intr_gops.handle_tpc_sm_ecc_exception =
g->ops.gr.intr.handle_tpc_sm_ecc_exception;
gr_test_intr_gops.handle_tpc_mpc_exception =
g->ops.gr.intr.handle_tpc_mpc_exception;
gr_test_intr_gops.handle_tpc_pe_exception =
g->ops.gr.intr.handle_tpc_pe_exception;
gr_test_intr_gops.get_ctxsw_checksum_mismatch_mailbox_val =
g->ops.gr.intr.get_ctxsw_checksum_mismatch_mailbox_val;
gr_test_intr_gops.handle_ssync_hww =
g->ops.gr.intr.handle_ssync_hww;
gr_test_intr_gops.handle_gcc_exception =
g->ops.gr.intr.handle_gcc_exception;
gr_test_intr_gops.handle_gpc_gpcmmu_exception =
g->ops.gr.intr.handle_gpc_gpcmmu_exception;
gr_test_intr_gops.handle_gpc_prop_exception =
g->ops.gr.intr.handle_gpc_prop_exception;
gr_test_intr_gops.handle_gpc_zcull_exception =
g->ops.gr.intr.handle_gpc_zcull_exception;
gr_test_intr_gops.handle_gpc_setup_exception =
g->ops.gr.intr.handle_gpc_setup_exception;
gr_test_intr_gops.handle_gpc_pes_exception =
g->ops.gr.intr.handle_gpc_pes_exception;
gr_test_intr_gops.handle_gpc_gpccs_exception =
g->ops.gr.intr.handle_gpc_gpccs_exception;
gr_test_intr_gops.get_sm_hww_warp_esr_pc =
g->ops.gr.intr.get_sm_hww_warp_esr_pc;
gr_test_intr_gops.handle_exceptions =
g->ops.gr.intr.handle_exceptions;
#ifdef CONFIG_NVGPU_RECOVERY
gr_test_intr_gops.recover = g->ops.fifo.recover;
#endif
}
#ifdef CONFIG_NVGPU_RECOVERY
static void gr_test_intr_fifo_recover(struct gk20a *g, u32 bitmask, u32 id,
unsigned int id_type, unsigned int rc_type,
struct mmu_fault_info *mmufault)
{
/* Remove once recovery support get disable for safety */
}
#endif
static u32 stub_channel_count(struct gk20a *g)
{
return 4;
}
static int stub_runlist_update(struct gk20a *g, struct nvgpu_runlist *rl,
struct nvgpu_channel *ch, bool add, bool wait_for_finish)
{
return 0;
}
static bool gr_test_intr_handle_exceptions(struct gk20a *g,
bool *is_gpc_exception)
{
*is_gpc_exception = true;
return true;
}
static int gr_test_intr_allocate_ch(struct unit_module *m,
struct gk20a *g)
{
int err;
struct nvgpu_channel *ch = NULL;
u32 tsgid = getpid();
err = nvgpu_channel_setup_sw(g);
if (err != 0) {
unit_return_fail(m, "failed channel setup\n");
}
ch = nvgpu_channel_open_new(g, NVGPU_INVALID_RUNLIST_ID,
false, tsgid, tsgid);
if (ch == NULL) {
unit_return_fail(m, "failed channel open\n");
}
/* Set pending interrupt for notify and semaphore */
nvgpu_posix_io_writel_reg_space(g, gr_intr_r(),
gr_intr_notify_pending_f() |
gr_intr_semaphore_pending_f()|
gr_intr_illegal_notify_pending_f()|
gr_intr_fecs_error_pending_f() |
gr_intr_class_error_pending_f() |
(0x1 << 31));
err = g->ops.gr.intr.stall_isr(g);
if (err != 0) {
unit_err(m, "failed stall isr\n");
}
nvgpu_channel_close(ch);
return (err == 0) ? UNIT_SUCCESS: UNIT_FAIL;
}
static int gr_test_intr_block_ptr_as_current_ctx(struct unit_module *m,
struct gk20a *g, struct nvgpu_channel *ch, struct nvgpu_tsg *tsg,
u32 pid)
{
int err, i;
struct nvgpu_gr_intr *intr = g->gr->intr;
u32 tsgid = nvgpu_inst_block_ptr(g, &ch->inst_block);
struct nvgpu_tsg_sm_error_state *sm_error_states = NULL;
err = EXPECT_BUG(g->ops.gr.intr.stall_isr(g));
if (err != 0) {
unit_return_fail(m, "failed stall isr\n");
}
nvgpu_posix_io_writel_reg_space(g, gr_fecs_current_ctx_r(),
tsgid);
err = g->ops.gr.intr.stall_isr(g);
if (err != 0) {
unit_return_fail(m, "failed stall isr\n");
}
/* Cover the case where gv11b_gr_intr_read_sm_error_state fails */
sm_error_states = tsg->sm_error_states;
tsg->sm_error_states = NULL;
err = g->ops.gr.intr.stall_isr(g);
if (err != 0) {
unit_return_fail(m, "failed stall isr\n");
}
tsg->sm_error_states = sm_error_states;
/* Make all entry valid so code with flush one */
for (i = 0; i < GR_TEST_CHANNEL_MAP_TLB_SIZE; i++) {
intr->chid_tlb[i].curr_ctx = pid;
}
err = g->ops.gr.intr.stall_isr(g);
if (err != 0) {
unit_return_fail(m, "failed stall isr\n");
}
return UNIT_SUCCESS;
}
static int gr_test_intr_cache_current_ctx(struct gk20a *g,
struct nvgpu_channel *ch, u32 pid)
{
int i;
struct nvgpu_gr_intr *intr = g->gr->intr;
nvgpu_gr_intr_flush_channel_tlb(g);
nvgpu_posix_io_writel_reg_space(g, gr_fecs_current_ctx_r(),
pid);
/* setting the cache */
for (i = 0; i < GR_TEST_CHANNEL_MAP_TLB_SIZE; i++) {
intr->chid_tlb[i].chid = ch->chid;
intr->chid_tlb[i].tsgid = ch->tsgid;
intr->chid_tlb[i].curr_ctx = pid;
}
return g->ops.gr.intr.stall_isr(g);
}
static u64 nvgpu_gmmu_map_locked_stub(struct vm_gk20a *vm,
u64 vaddr,
struct nvgpu_sgt *sgt,
u64 buffer_offset,
u64 size,
u32 pgsz_idx,
u8 kind_v,
u32 ctag_offset,
u32 flags,
enum gk20a_mem_rw_flag rw_flag,
bool clear_ctags,
bool sparse,
bool priv,
struct vm_gk20a_mapping_batch *batch,
enum nvgpu_aperture aperture)
{
return 1;
}
static void nvgpu_gmmu_unmap_locked_stub(struct vm_gk20a *vm,
u64 vaddr,
u64 size,
u32 pgsz_idx,
bool va_allocated,
enum gk20a_mem_rw_flag rw_flag,
bool sparse,
struct vm_gk20a_mapping_batch *batch)
{
return;
}
static int gr_test_intr_allocate_ch_tsg(struct unit_module *m,
struct gk20a *g)
{
u32 tsgid = getpid();
struct nvgpu_gr_ctx_mappings *mappings = NULL;
struct nvgpu_channel *ch = NULL;
struct nvgpu_tsg *tsg = NULL;
struct vm_gk20a *vm = NULL;
bool sema_init, notify_init;
int err;
err = nvgpu_channel_setup_sw(g);
if (err != 0) {
unit_return_fail(m, "failed channel setup\n");
}
err = nvgpu_tsg_setup_sw(g);
if (err != 0) {
unit_return_fail(m, "failed tsg setup\n");
}
tsg = nvgpu_tsg_open(g, tsgid);
if (tsg == NULL) {
unit_return_fail(m, "failed tsg open\n");
}
ch = nvgpu_channel_open_new(g, NVGPU_INVALID_RUNLIST_ID,
false, tsgid, tsgid);
if (ch == NULL) {
unit_err(m, "failed channel open\n");
goto ch_cleanup;
}
/* Setup VM */
vm = nvgpu_vm_init(g, SZ_4K, SZ_4K << 10,
nvgpu_safe_sub_u64(1ULL << 37, SZ_4K << 10),
(1ULL << 32), 0ULL,
false, false, "dummy");
if (!vm) {
unit_err(m, "failed to allocate VM");
goto ch_cleanup;
}
ch->g = g;
ch->vm = vm;
err = nvgpu_tsg_bind_channel(tsg, ch);
if (err != 0) {
unit_err(m, "failed tsg channel bind\n");
goto ch_cleanup;
}
g->ops.mm.gmmu.map = nvgpu_gmmu_map_locked_stub;
g->ops.mm.gmmu.unmap = nvgpu_gmmu_unmap_locked_stub;
err = nvgpu_tsg_subctx_alloc_gr_subctx(g, ch);
if (err != 0) {
unit_err(m, "failed to alloc gr subctx");
goto ch_cleanup;
}
err = nvgpu_tsg_subctx_setup_subctx_header(g, ch);
if (err != 0) {
unit_err(m, "failed to setup subctx header");
goto ch_cleanup;
}
mappings = nvgpu_gr_ctx_alloc_or_get_mappings(g, tsg, ch);
if (mappings == NULL) {
unit_err(m, "failed to allocate gr_ctx mappings");
goto ch_cleanup;
}
err = gr_test_intr_block_ptr_as_current_ctx(m, g, ch, tsg, tsgid);
if (err != 0) {
unit_err(m, "isr failed with block_ptr as current_ctx\n");
goto tsg_unbind;
}
err = gr_test_intr_cache_current_ctx(g, ch, tsgid);
if (err != 0) {
unit_err(m, "isr failed with cache current_ctx\n");
goto tsg_unbind;
}
/* Set pending interrupt for notify and semaphore */
nvgpu_posix_io_writel_reg_space(g, gr_intr_r(),
gr_intr_notify_pending_f() |
gr_intr_semaphore_pending_f()|
gr_intr_firmware_method_pending_f());
err = g->ops.gr.intr.stall_isr(g);
if (err != 0) {
unit_err(m, "failed stall isr\n");
}
/* Set the semaphore and notify wait queue uninitialized*/
sema_init = ch->semaphore_wq.initialized;
notify_init = ch->notifier_wq.initialized;
ch->semaphore_wq.initialized = false;
ch->notifier_wq.initialized = false;
err = g->ops.gr.intr.stall_isr(g);
if (err != 0) {
unit_err(m, "failed stall isr for wait_queue\n");
}
ch->semaphore_wq.initialized = sema_init;
ch->notifier_wq.initialized = notify_init;
tsg_unbind:
err = nvgpu_tsg_unbind_channel(tsg, ch, true);
if (err != 0) {
unit_err(m, "failed tsg channel unbind\n");
}
ch_cleanup:
nvgpu_channel_close(ch);
return (err == 0) ? UNIT_SUCCESS: UNIT_FAIL;
}
int test_gr_intr_setup_channel(struct unit_module *m,
struct gk20a *g, void *args)
{
u32 tsgid = getpid();
int err;
struct nvgpu_fifo *f = &g->fifo;
nvgpu_posix_io_writel_reg_space(g, gr_fecs_current_ctx_r(),
tsgid);
g->ops.channel.count = stub_channel_count;
g->ops.runlist.update = stub_runlist_update;
if (f != NULL) {
f->g = g;
}
/* Test with channel and tsg */
err = gr_test_intr_allocate_ch_tsg(m, g);
if (err != 0) {
unit_return_fail(m, "isr test with channel and tsg failed\n");
}
/* Test with channel and without tsg */
err = gr_test_intr_allocate_ch(m, g);
if (err != 0) {
unit_return_fail(m, "isr test with channel and tsg failed\n");
}
return UNIT_SUCCESS;
}
static int gr_test_nonstall_isr(struct unit_module *m,
struct gk20a *g)
{
int err;
/* Call without setting any non stall interrupt */
err = g->ops.gr.intr.nonstall_isr(g);
if (err != 0) {
return err;
}
/* Call with setting any non stall interrupt */
nvgpu_posix_io_writel_reg_space(g, gr_intr_nonstall_r(),
gr_intr_nonstall_trap_pending_f());
err = g->ops.gr.intr.nonstall_isr(g);
if (err == 0) {
err = UNIT_FAIL;
unit_return_fail(m, "nonstall_isr failed\n");
}
return UNIT_SUCCESS;
}
static int test_gr_intr_error_injections(struct unit_module *m,
struct gk20a *g, void *args)
{
int err = 0;
struct nvgpu_posix_fault_inj *kmem_fi =
nvgpu_kmem_get_fault_injection();
struct nvgpu_gr_intr *test_gr_intr;
struct nvgpu_gr_isr_data isr_data;
u32 gr_intr;
/* Fail gr_intr struct allocation */
nvgpu_posix_enable_fault_injection(kmem_fi, true, 0);
test_gr_intr = nvgpu_gr_intr_init_support(g);
if (test_gr_intr != NULL) {
unit_return_fail(m, "nvgpu_gr_intr_init_support failed\n");
}
nvgpu_posix_enable_fault_injection(kmem_fi, false, 0);
/* Free NULL gr_intr struct */
nvgpu_gr_intr_remove_support(g, NULL);
/* call isr without any interrupt */
gr_intr = nvgpu_posix_io_readl_reg_space(g, gr_intr_r());
nvgpu_posix_io_writel_reg_space(g, gr_intr_r(), 0);
err = g->ops.gr.intr.stall_isr(g);
if (err != 0) {
unit_return_fail(m, "isr failed without interrupts\n");
}
nvgpu_posix_io_writel_reg_space(g, gr_intr_r(), gr_intr);
/* Call fecs_interrupt handler with fecs error set */
isr_data.ch = NULL;
nvgpu_posix_io_writel_reg_space(g, gr_fecs_host_int_status_r(), 0);
isr_data.fecs_intr = g->ops.gr.falcon.fecs_host_intr_status(g,
&(isr_data.fecs_host_intr_status));
err = g->ops.gr.intr.handle_fecs_error(g, NULL, &isr_data);
if (err != 0) {
unit_return_fail(m,
"gr.intr.handle_fecs_error failed\n");
}
/* Fault injection - gpc exception with reset */
g->ops.gr.intr.handle_exceptions =
gr_test_intr_handle_exceptions;
nvgpu_posix_io_writel_reg_space(g, gr_intr_r(),
gr_intr_exception_pending_f());
err = g->ops.gr.intr.stall_isr(g);
if (err != 0) {
unit_return_fail(m, "sw_method failed for invalid data\n");
}
g->ops.gr.intr.handle_exceptions =
gr_test_intr_gops.handle_exceptions;
nvgpu_posix_io_writel_reg_space(g, gr_intr_r(), gr_intr);
return UNIT_SUCCESS;
}
int test_gr_intr_without_channel(struct unit_module *m,
struct gk20a *g, void *args)
{
int err;
gr_test_save_intr_gops(g);
#ifdef CONFIG_NVGPU_RECOVERY
g->ops.fifo.recover = gr_test_intr_fifo_recover;
#endif
err = test_gr_intr_error_injections(m, g, args);
if (err != 0) {
unit_return_fail(m, "gr_test_intr_error_injections failed\n");
}
/* Set trapped address datahigh bit */
nvgpu_posix_io_writel_reg_space(g, gr_trapped_addr_r(),
GR_TEST_TRAPPED_ADDR_DATAHIGH);
/* Set exception for FE, MEMFMT, PD, SCC, DS, SSYNC, MME, SKED */
nvgpu_posix_io_writel_reg_space(g, gr_exception_r(),
gr_exception_fe_m() | gr_exception_memfmt_m() |
gr_exception_pd_m() | gr_exception_scc_m() |
gr_exception_ds_m() | gr_exception_ssync_m() |
gr_exception_mme_m() | gr_exception_sked_m());
err = g->ops.gr.intr.stall_isr(g);
if (err != 0) {
unit_return_fail(m, "stall_isr failed\n");
}
err = gr_test_nonstall_isr(m, g);
if (err != 0) {
unit_return_fail(m, "nonstall_isr failed\n");
}
/* Set handle ssync_hww to NULL */
g->ops.gr.intr.handle_ssync_hww = NULL;
nvgpu_posix_io_writel_reg_space(g, gr_exception_r(),
gr_exception_ssync_m());
err = g->ops.gr.intr.stall_isr(g);
if (err != 0) {
unit_return_fail(m, "stall_isr handle_ssync_hww failed\n");
}
g->ops.gr.intr.handle_ssync_hww =
gr_test_intr_gops.handle_ssync_hww;
return UNIT_SUCCESS;
}
struct test_gr_intr_sw_mthd_exceptions sw_excep[] = {
[0] = {
.trapped_addr = NVC3C0_SET_SKEDCHECK,
.data[0] = NVC397_SET_SKEDCHECK_18_ENABLE,
.data[1] = NVC397_SET_SKEDCHECK_18_DISABLE,
},
[1] = {
.trapped_addr = NVC3C0_SET_SHADER_CUT_COLLECTOR,
.data[0] = NVC397_SET_SHADER_CUT_COLLECTOR_STATE_ENABLE,
.data[1] = NVC397_SET_SHADER_CUT_COLLECTOR_STATE_DISABLE,
},
[2] = {
.trapped_addr = 0,
.data[0] = 0,
.data[1] = 0,
}
};
int test_gr_intr_sw_exceptions(struct unit_module *m,
struct gk20a *g, void *args)
{
int err;
int i, j, data_cnt;
int arry_cnt = sizeof(sw_excep)/
sizeof(struct test_gr_intr_sw_mthd_exceptions);
/* Set illegal method and class error pending */
nvgpu_posix_io_writel_reg_space(g, gr_intr_r(),
gr_intr_illegal_method_pending_f() |
gr_intr_class_error_pending_f());
/* valid class num */
nvgpu_posix_io_writel_reg_space(g,
gr_fe_object_table_r(0), VOLTA_COMPUTE_A);
for (i = 0; i < arry_cnt; i++) {
/* method & sub channel */
nvgpu_posix_io_writel_reg_space(g, gr_trapped_addr_r(),
sw_excep[i].trapped_addr);
data_cnt = (i < (arry_cnt - 1)) ? 2 : 1;
for (j = 0; j < data_cnt; j++) {
/* data */
nvgpu_posix_io_writel_reg_space(g,
gr_trapped_data_lo_r(), sw_excep[i].data[j]);
err = g->ops.gr.intr.stall_isr(g);
if (err != 0) {
unit_return_fail(m, "stall isr failed\n");
}
}
}
/* Fault injection - sw_method with invalid class */
err = g->ops.gr.intr.handle_sw_method(g, 0, 0, 0, 0);
if (err == 0) {
unit_return_fail(m, "sw_method passed for invalid class\n");
}
return UNIT_SUCCESS;
}
static void gr_intr_gpc_gpcmmu_esr_regs(struct gk20a *g)
{
u32 esr_reg = gr_gpc0_mmu_gpcmmu_global_esr_ecc_corrected_m() |
gr_gpc0_mmu_gpcmmu_global_esr_ecc_uncorrected_m();
nvgpu_posix_io_writel_reg_space(g,
gr_gpc0_mmu_gpcmmu_global_esr_r(), esr_reg);
}
static void gr_intr_gpc_gpccs_esr_regs(struct gk20a *g)
{
u32 esr_reg = gr_gpc0_gpccs_hww_esr_ecc_corrected_m() |
gr_gpc0_gpccs_hww_esr_ecc_uncorrected_m();
nvgpu_posix_io_writel_reg_space(g,
gr_gpc0_gpccs_hww_esr_r(), esr_reg);
}
struct test_gr_intr_gpc_ecc_status {
u32 status_val;
u32 status_reg;
u32 corr_reg;
u32 uncorr_reg;
};
struct test_gr_intr_gpc_ecc_status gpc_ecc_reg[] = {
[0] = { /* L1 tag ecc regs */
.status_val = 0x5FF, //0xFFF,
.status_reg = gr_pri_gpc0_tpc0_sm_l1_tag_ecc_status_r(),
.corr_reg = gr_pri_gpc0_tpc0_sm_l1_tag_ecc_corrected_err_count_r(),
.uncorr_reg = gr_pri_gpc0_tpc0_sm_l1_tag_ecc_uncorrected_err_count_r(),
},
[1] = { /* LRF ecc regs */
.status_val = 0xFFFFFFF,
.status_reg = gr_pri_gpc0_tpc0_sm_lrf_ecc_status_r(),
.corr_reg = gr_pri_gpc0_tpc0_sm_lrf_ecc_corrected_err_count_r(),
.uncorr_reg = gr_pri_gpc0_tpc0_sm_lrf_ecc_uncorrected_err_count_r(),
},
[2] = { /* CBU ecc regs */
.status_val = 0xF00FF,
.status_reg = gr_pri_gpc0_tpc0_sm_cbu_ecc_status_r(),
.corr_reg = gr_pri_gpc0_tpc0_sm_cbu_ecc_corrected_err_count_r(),
.uncorr_reg = gr_pri_gpc0_tpc0_sm_cbu_ecc_uncorrected_err_count_r(),
},
[3] = { /* L1 data regs */
.status_val = 0xF0F,
.status_reg = gr_pri_gpc0_tpc0_sm_l1_data_ecc_status_r(),
.corr_reg = gr_pri_gpc0_tpc0_sm_l1_data_ecc_corrected_err_count_r(),
.uncorr_reg = gr_pri_gpc0_tpc0_sm_l1_data_ecc_uncorrected_err_count_r(),
},
[4] = { /* ICACHE regs */
.status_val = 0xF00FF,
.status_reg = gr_pri_gpc0_tpc0_sm_icache_ecc_status_r(),
.corr_reg = gr_pri_gpc0_tpc0_sm_icache_ecc_corrected_err_count_r(),
.uncorr_reg = gr_pri_gpc0_tpc0_sm_icache_ecc_uncorrected_err_count_r(),
},
[5] = { /* MMU_L1TLB regs */
.status_val = 0xF000F,
.status_reg = gr_gpc0_mmu_l1tlb_ecc_status_r(),
.corr_reg = gr_gpc0_mmu_l1tlb_ecc_corrected_err_count_r(),
.uncorr_reg = gr_gpc0_mmu_l1tlb_ecc_uncorrected_err_count_r(),
},
[6] = { /* GPCCS_FALCON regs */
.status_val = 0xF33,
.status_reg = gr_gpc0_gpccs_falcon_ecc_status_r(),
.corr_reg = gr_gpc0_gpccs_falcon_ecc_corrected_err_count_r(),
.uncorr_reg = gr_gpc0_gpccs_falcon_ecc_uncorrected_err_count_r(),
},
[7] = { /* GCC_L15 regs */
.status_val = 0xF33,
.status_reg = gr_pri_gpc0_gcc_l15_ecc_status_r(),
.corr_reg = gr_pri_gpc0_gcc_l15_ecc_corrected_err_count_r(),
.uncorr_reg = gr_pri_gpc0_gcc_l15_ecc_uncorrected_err_count_r(),
},
};
static void gr_intr_gpc_ecc_err_injections(struct gk20a *g)
{
u32 corr_cnt = 20U, uncorr_cnt = 20U;
int i, j;
u32 status_val, ecc_status;
u32 corr_overflow, uncorr_overflow;
u32 corr_err, uncorr_err;
u32 status_reg, corr_reg, uncorr_reg;
u32 gpc_exception;
int arry_cnt = sizeof(gpc_ecc_reg)/
sizeof(struct test_gr_intr_gpc_ecc_status);
for (i = 0; i < arry_cnt; i++) {
status_val = gpc_ecc_reg[i].status_val;
status_reg = gpc_ecc_reg[i].status_reg;
corr_reg = gpc_ecc_reg[i].corr_reg;
uncorr_reg = gpc_ecc_reg[i].uncorr_reg;
corr_cnt = 20U;
uncorr_cnt = 20U;
switch (i) {
case 0: /* L1 tag ecc regs */
corr_overflow = 0x100;
uncorr_overflow = 0x400;
corr_err = 0x33;
uncorr_err = 0xCC;
break;
case 1: /* LRF ecc regs */
corr_overflow = 0x1 << 24;
uncorr_overflow = 0x1 << 26;
corr_err = 0xFF;
uncorr_err = 0xFF00;
break;
case 2: /* CBU ecc regs */
corr_overflow = 0x1 << 16;
uncorr_overflow = 0x1 << 18;
corr_err = 0xF;
uncorr_err = 0xF0;
break;
case 3: /* L1 data ecc regs */
corr_overflow = 0x1 << 8;
uncorr_overflow = 0x1 << 10;
corr_err = 0x3;
uncorr_err = 0xC;
break;
case 4: /* Icache ecc regs */
corr_overflow = 0x1 << 16;
uncorr_overflow = 0x1 << 18;
corr_err = 0xF;
uncorr_err = 0xF0;
break;
case 5: /* MMU_L1TLB ecc regs */
corr_overflow = 0x1 << 16;
uncorr_overflow = 0x1 << 18;
corr_err = 0x5;
uncorr_err = 0xA;
break;
case 6: /* GPCCS_FALCON ecc regs */
corr_overflow = 0x1 << 8;
uncorr_overflow = 0x1 << 11;
corr_err = 0x3;
uncorr_err = 0x30;
break;
case 7: /* GCC_L15 ecc regs */
corr_overflow = 0x1 << 8;
uncorr_overflow = 0x1 << 10;
corr_err = 0x3;
uncorr_err = 0x30;
break;
}
for (j = 0; j < 5; j++) {
if (j == 0) {
ecc_status = status_val &
~(corr_overflow | uncorr_overflow);
corr_cnt = 20U;
uncorr_cnt = 20U;
} else if (j == 1) {
corr_cnt = 0U;
uncorr_cnt = 20U;
ecc_status = status_val & (corr_overflow | uncorr_err);
} else if (j == 2) {
corr_cnt = 20U;
uncorr_cnt = 0U;
ecc_status = status_val & (uncorr_overflow | corr_err);
} else if (j == 3){
ecc_status = (corr_overflow | uncorr_overflow);
corr_cnt = 0U;
uncorr_cnt = 0U;
} else {
ecc_status = status_val &
~(corr_overflow | uncorr_overflow);
corr_cnt = 0U;
uncorr_cnt = 0U;
}
nvgpu_posix_io_writel_reg_space(g, corr_reg, corr_cnt);
nvgpu_posix_io_writel_reg_space(g, uncorr_reg, uncorr_cnt);
nvgpu_posix_io_writel_reg_space(g, status_reg, ecc_status);
if (i <= 4) {
EXPECT_BUG(g->ops.gr.intr.handle_tpc_sm_ecc_exception(g, 0 , 0));
} else if (i == 5) {
gpc_exception =
gr_gpc0_gpccs_gpc_exception_gpcmmu_m();
EXPECT_BUG(g->ops.gr.intr.handle_gpc_gpcmmu_exception(g,
0, gpc_exception, &corr_cnt, &uncorr_cnt));
} else if (i == 6) {
gpc_exception =
gr_gpc0_gpccs_gpc_exception_gpccs_m();
EXPECT_BUG(g->ops.gr.intr.handle_gpc_gpccs_exception(g,
0, gpc_exception, &corr_cnt, &uncorr_cnt));
} else if (i == 7) {
gpc_exception = 0x1 << 2;
EXPECT_BUG(g->ops.gr.intr.handle_gcc_exception(g,
0, gpc_exception, &corr_cnt, &uncorr_cnt));
}
}
}
}
static void gr_intr_gpc_ecc_err_regs(struct gk20a *g)
{
u32 cnt = 20U;
int i;
int arry_cnt = sizeof(gpc_ecc_reg)/
sizeof(struct test_gr_intr_gpc_ecc_status);
for (i = 0; i < arry_cnt; i++) {
nvgpu_posix_io_writel_reg_space(g,
gpc_ecc_reg[i].corr_reg, cnt);
nvgpu_posix_io_writel_reg_space(g,
gpc_ecc_reg[i].uncorr_reg, cnt);
nvgpu_posix_io_writel_reg_space(g,
gpc_ecc_reg[i].status_reg,
gpc_ecc_reg[i].status_val);
}
}
static void gr_test_enable_gpc_exception_intr(struct gk20a *g)
{
/* Set exception pending */
nvgpu_posix_io_writel_reg_space(g, gr_intr_r(),
gr_intr_exception_pending_f());
/* Set gpc exception */
nvgpu_posix_io_writel_reg_space(g, gr_exception_r(),
gr_exception_gpc_m());
/* Set gpc exception1 */
nvgpu_posix_io_writel_reg_space(g, gr_exception1_r(),
gr_exception1_gpc_0_pending_f());
}
static int gr_test_gpc_exception_intr(struct gk20a *g)
{
int err = UNIT_SUCCESS;
/* enable gpc exception interrupt bit */
gr_test_enable_gpc_exception_intr(g);
/* Call interrupt routine */
err = g->ops.gr.intr.stall_isr(g);
return err;
}
static void gr_test_set_gpc_exceptions(struct gk20a *g, bool full)
{
u32 gpc_exception = 0;
/* Set exceptions for gpcmmu/gcc/tpc */
gpc_exception = gr_gpc0_gpccs_gpc_exception_gpcmmu_m() |
gr_gpc0_gpccs_gpc_exception_gpccs_m() |
gr_gpcs_gpccs_gpc_exception_en_gcc_f(1U) |
gr_gpcs_gpccs_gpc_exception_en_tpc_f(1U);
if (full) {
/* Set exceptions for prop/zcull/setup/pes/gpccs */
gpc_exception |= gr_gpc0_gpccs_gpc_exception_prop_m() |
gr_gpc0_gpccs_gpc_exception_zcull_m() |
gr_gpc0_gpccs_gpc_exception_setup_m() |
gr_gpc0_gpccs_gpc_exception_pes0_m() |
gr_gpc0_gpccs_gpc_exception_pes1_m();
}
nvgpu_posix_io_writel_reg_space(g, gr_gpc0_gpccs_gpc_exception_r(),
gpc_exception);
}
static int gr_test_set_gpc_tpc_exceptions(struct gk20a *g)
{
int err;
u32 gpc_exception =
gr_gpcs_gpccs_gpc_exception_en_tpc_f(1U);
/* Set trapped addr with sub_chan > 4 */
nvgpu_posix_io_writel_reg_space(g, gr_trapped_addr_r(), (0x5 << 16U));
nvgpu_posix_io_writel_reg_space(g, gr_gpc0_gpccs_gpc_exception_r(),
gpc_exception);
nvgpu_posix_io_writel_reg_space(g,
gr_gpc0_tpc0_tpccs_tpc_exception_r(),
0);
err = gr_test_gpc_exception_intr(g);
nvgpu_posix_io_writel_reg_space(g, gr_trapped_addr_r(), 0);
return err;
}
static void gr_test_set_tpc_exceptions(struct gk20a *g)
{
u32 tpc_exception = 0;
/* Tpc exceptions for mpc/pe */
tpc_exception = gr_gpc0_tpc0_tpccs_tpc_exception_mpc_m() |
gr_gpc0_tpc0_tpccs_tpc_exception_pe_m();
/* Tpc exceptions for tex/sm */
tpc_exception |= TPC_EXCEPTION_TEX | TPC_EXCEPTION_SM;
nvgpu_posix_io_writel_reg_space(g, gr_gpc0_tpc0_tpccs_tpc_exception_r(),
tpc_exception);
}
static void gr_test_set_gpc_pes_exception(struct gk20a *g)
{
/* Handle either pes0 or pes1 exception */
u32 gpc_exception = gr_gpc0_gpccs_gpc_exception_pes0_m();
g->ops.gr.intr.handle_gpc_pes_exception(g, 0, gpc_exception);
gpc_exception = gr_gpc0_gpccs_gpc_exception_pes1_m();
g->ops.gr.intr.handle_gpc_pes_exception(g, 0, gpc_exception);
}
static void gr_test_set_tpc_esr_sm(struct gk20a *g)
{
u32 global_esr_mask = 0U;
nvgpu_posix_io_writel_reg_space(g,
gr_gpc0_tpc0_sm_tpc_esr_sm_sel_r(),
TPC_SM0_ESR_SEL | TPC_SM1_ESR_SEL);
/* set global esr for sm */
global_esr_mask = nvgpu_posix_io_readl_reg_space(g,
gr_gpc0_tpc0_sm0_hww_global_esr_r());
global_esr_mask |=
gr_gpc0_tpc0_sm0_hww_global_esr_multiple_warp_errors_pending_f();
nvgpu_posix_io_writel_reg_space(g,
gr_gpc0_tpc0_sm0_hww_global_esr_r(), global_esr_mask);
}
static int gr_test_set_gpc_exceptions_without_handle(struct gk20a *g)
{
int err;
g->ops.gr.intr.handle_gcc_exception = NULL;
g->ops.gr.intr.handle_gpc_gpcmmu_exception = NULL;
g->ops.gr.intr.handle_gpc_prop_exception = NULL;
g->ops.gr.intr.handle_gpc_zcull_exception = NULL;
g->ops.gr.intr.handle_gpc_setup_exception = NULL;
g->ops.gr.intr.handle_gpc_pes_exception = NULL;
g->ops.gr.intr.handle_gpc_gpccs_exception = NULL;
err = gr_test_gpc_exception_intr(g);
if (err)
return err;
g->ops.gr.intr.handle_gcc_exception =
gr_test_intr_gops.handle_gcc_exception;
g->ops.gr.intr.handle_gpc_gpcmmu_exception =
gr_test_intr_gops.handle_gpc_gpcmmu_exception;
g->ops.gr.intr.handle_gpc_prop_exception =
gr_test_intr_gops.handle_gpc_prop_exception;
g->ops.gr.intr.handle_gpc_zcull_exception =
gr_test_intr_gops.handle_gpc_zcull_exception;
g->ops.gr.intr.handle_gpc_setup_exception =
gr_test_intr_gops.handle_gpc_setup_exception;
g->ops.gr.intr.handle_gpc_pes_exception =
gr_test_intr_gops.handle_gpc_pes_exception;
g->ops.gr.intr.handle_gpc_gpccs_exception =
gr_test_intr_gops.handle_gpc_gpccs_exception;
/* Set exception pending */
nvgpu_posix_io_writel_reg_space(g, gr_intr_r(),
gr_intr_exception_pending_f());
/* Set gpc exception */
nvgpu_posix_io_writel_reg_space(g, gr_exception_r(),
gr_exception_gpc_m());
/* Set gpc exception1 */
nvgpu_posix_io_writel_reg_space(g, gr_exception1_r(), 0);
err = g->ops.gr.intr.stall_isr(g);
return err;
}
static int gr_test_set_tpc_exceptions_without_handle(struct gk20a *g)
{
u32 tpc_exception = 0;
int err;
u32 gpc_exception =
gr_gpcs_gpccs_gpc_exception_en_tpc_f(1U);
g->ops.gr.intr.handle_tpc_sm_ecc_exception = NULL;
g->ops.gr.intr.handle_tpc_mpc_exception = NULL;
g->ops.gr.intr.handle_tpc_pe_exception = NULL;
g->ops.gr.intr.get_sm_hww_warp_esr_pc = NULL;
nvgpu_posix_io_writel_reg_space(g, gr_gpc0_gpccs_gpc_exception_r(),
gpc_exception);
/* Tpc exceptions for mpc/pe */
tpc_exception = gr_gpc0_tpc0_tpccs_tpc_exception_mpc_m() |
gr_gpc0_tpc0_tpccs_tpc_exception_pe_m();
/* Tpc exceptions for tex/sm */
tpc_exception |= TPC_EXCEPTION_SM;
nvgpu_posix_io_writel_reg_space(g, gr_gpc0_tpc0_tpccs_tpc_exception_r(),
tpc_exception);
err = gr_test_gpc_exception_intr(g);
if (err != 0) {
return err;
}
g->ops.gr.intr.handle_tpc_sm_ecc_exception =
gr_test_intr_gops.handle_tpc_sm_ecc_exception;
g->ops.gr.intr.handle_tpc_mpc_exception =
gr_test_intr_gops.handle_tpc_mpc_exception;
g->ops.gr.intr.handle_tpc_pe_exception =
gr_test_intr_gops.handle_tpc_pe_exception;
gpc_exception = gr_gpcs_gpccs_gpc_exception_en_tpc_f(3U);
nvgpu_posix_io_writel_reg_space(g, gr_gpc0_gpccs_gpc_exception_r(),
gpc_exception);
/* Tpc exceptions for sm and multiple tpc*/
tpc_exception = TPC_EXCEPTION_SM;
nvgpu_posix_io_writel_reg_space(g, gr_gpc0_tpc0_tpccs_tpc_exception_r(),
tpc_exception);
gr_test_set_tpc_esr_sm(g);
err = gr_test_gpc_exception_intr(g);
g->ops.gr.intr.get_sm_hww_warp_esr_pc =
gr_test_intr_gops.get_sm_hww_warp_esr_pc;
return err;
}
int test_gr_intr_gpc_exceptions(struct unit_module *m,
struct gk20a *g, void *args)
{
int err;
/**
* Negative test to verify gpc_exception interrupt without
* enabling any gpc_exception.
*/
err = gr_test_gpc_exception_intr(g);
if (err != 0) {
unit_return_fail(m, "isr failed without gpc exceptions\n");
}
/**
* Negative test to verify gpc_tpc_exception interrupt with
* enabling gpc_tpc_exception and with enabling any tpc_exception.
*/
err = gr_test_set_gpc_tpc_exceptions(g);
if (err != 0) {
unit_return_fail(m, "gr_test_set_gpc_tpc_exceptions failed\n");
}
/**
* Negative test to verify gpc_exception interrupt with
* enabling all gpc_exceptions, but without setting the ecc status
* registers.
*/
gr_test_set_gpc_exceptions(g, true);
gr_test_set_tpc_exceptions(g);
err = gr_test_gpc_exception_intr(g);
if (err != 0) {
unit_return_fail(m, "gpc exceptions without ecc status failed\n");
}
/**
* Negative test to verify tpc_exception interrupt with NULL handle
* and enabling gpc_tpc_exceptions.
*/
err = gr_test_set_tpc_exceptions_without_handle(g);
if (err != 0) {
unit_return_fail(m,
"gr_test_set_tpc_exceptions_without_handle failed\n");
}
err = gr_test_set_gpc_exceptions_without_handle(g);
if (err != 0) {
unit_return_fail(m,
"gr_test_set_gpc_exceptions_without_handle failed\n");
}
/**
* Negative test to verify gpc_exception interrupt with
* enabling all gpc_exceptions and by setting the ecc status
* registers.
*/
gr_test_set_gpc_exceptions(g, false);
gr_test_set_tpc_exceptions(g);
gr_test_set_tpc_esr_sm(g);
gr_intr_gpc_gpcmmu_esr_regs(g);
gr_intr_gpc_gpccs_esr_regs(g);
gr_intr_gpc_ecc_err_regs(g);
/* enable gpc exception interrupt bit */
gr_test_enable_gpc_exception_intr(g);
/* Call interrupt routine */
err = EXPECT_BUG(g->ops.gr.intr.stall_isr(g));
if (err == 0) {
unit_return_fail(m, "stall isr failed\n");
}
/**
* Negative tests for gpc_exceptions ecc registers values
* for overflow and corrected and uncorrected errors.
*/
gr_intr_gpc_ecc_err_injections(g);
gr_test_set_gpc_pes_exception(g);
return UNIT_SUCCESS;
}
static void gr_intr_fecs_ecc_err_regs(struct gk20a *g, int index)
{
u32 corr_cnt = 20U, uncorr_cnt = 20U;
u32 ecc_status =
gr_fecs_falcon_ecc_status_corrected_err_imem_m() |
gr_fecs_falcon_ecc_status_corrected_err_dmem_m() |
gr_fecs_falcon_ecc_status_uncorrected_err_imem_m() |
gr_fecs_falcon_ecc_status_uncorrected_err_dmem_m() |
gr_fecs_falcon_ecc_status_corrected_err_total_counter_overflow_m() |
gr_fecs_falcon_ecc_status_uncorrected_err_total_counter_overflow_m();
if (index == 0) {
ecc_status = 0U;
corr_cnt = 0U;
uncorr_cnt = 0U;
} else if (index == 2) {
corr_cnt = 0U;
ecc_status =
gr_fecs_falcon_ecc_status_corrected_err_total_counter_overflow_m();
} else if (index == 3) {
uncorr_cnt = 0U;
ecc_status =
gr_fecs_falcon_ecc_status_uncorrected_err_total_counter_overflow_m();
}
nvgpu_posix_io_writel_reg_space(g,
gr_fecs_falcon_ecc_corrected_err_count_r(), corr_cnt);
nvgpu_posix_io_writel_reg_space(g,
gr_fecs_falcon_ecc_uncorrected_err_count_r(), uncorr_cnt);
nvgpu_posix_io_writel_reg_space(g,
gr_fecs_falcon_ecc_status_r(), ecc_status);
}
int test_gr_intr_fecs_exceptions(struct unit_module *m,
struct gk20a *g, void *args)
{
int err, i, j = 0;
int arry_cnt = 13;
u32 mismatch_value =
gr_fecs_ctxsw_mailbox_value_ctxsw_checksum_mismatch_v();
u32 fecs_status[13] = {
0,
gr_fecs_host_int_enable_ctxsw_intr0_enable_f(),
gr_fecs_host_int_enable_ctxsw_intr1_enable_f(),
gr_fecs_host_int_enable_fault_during_ctxsw_enable_f(),
gr_fecs_host_int_enable_umimp_firmware_method_enable_f(),
gr_fecs_host_int_enable_umimp_illegal_method_enable_f(),
gr_fecs_host_int_enable_watchdog_enable_f(),
gr_fecs_host_int_enable_ctxsw_intr0_enable_f(),
gr_fecs_host_int_enable_ctxsw_intr0_enable_f(),
gr_fecs_host_int_enable_ecc_corrected_enable_f() |
gr_fecs_host_int_enable_ecc_uncorrected_enable_f(),
gr_fecs_host_int_enable_ecc_corrected_enable_f(),
gr_fecs_host_int_enable_ecc_corrected_enable_f(),
gr_fecs_host_int_enable_ecc_uncorrected_enable_f(),
};
for (i = 0; i < arry_cnt; i++) {
/* Set fecs error pending */
nvgpu_posix_io_writel_reg_space(g, gr_intr_r(),
gr_intr_fecs_error_pending_f());
/* Set fecs host register status */
nvgpu_posix_io_writel_reg_space(g, gr_fecs_host_int_status_r(),
fecs_status[i]);
if ((i > 6) && (fecs_status[i] ==
gr_fecs_host_int_enable_ctxsw_intr0_enable_f())) {
/* Set valid mailbox values */
nvgpu_posix_io_writel_reg_space(g,
gr_fecs_ctxsw_mailbox_r(6U),
mismatch_value);
g->ops.gr.intr.get_ctxsw_checksum_mismatch_mailbox_val =
gr_test_intr_gops.get_ctxsw_checksum_mismatch_mailbox_val;
}
if (i == 7) {
g->ops.gr.intr.get_ctxsw_checksum_mismatch_mailbox_val = NULL;
}
/* Set fecs ecc registers */
if (i >= 9) {
gr_intr_fecs_ecc_err_regs(g, j);
j += 1;
}
if (i == 10) {
/* Injection of ECC corrected error will trigger BUG(). */
err = EXPECT_BUG(g->ops.gr.intr.stall_isr(g));
if (err == 0) {
unit_return_fail(m, "failed in fecs error interrupts\n");
}
} else {
err = g->ops.gr.intr.stall_isr(g);
if (err != 0) {
unit_return_fail(m, "failed in fecs error interrupts\n");
}
}
}
return UNIT_SUCCESS;
}
struct unit_module_test nvgpu_gr_intr_tests[] = {
UNIT_TEST(gr_intr_setup, test_gr_init_setup_ready, NULL, 2),
UNIT_TEST(gr_intr_channel_free, test_gr_intr_without_channel, NULL, 2),
UNIT_TEST(gr_intr_sw_method, test_gr_intr_sw_exceptions, NULL, 2),
UNIT_TEST(gr_intr_fecs_exceptions, test_gr_intr_fecs_exceptions, NULL, 2),
UNIT_TEST(gr_intr_gpc_exceptions, test_gr_intr_gpc_exceptions, NULL, 2),
UNIT_TEST(gr_intr_with_channel, test_gr_intr_setup_channel, NULL, 2),
UNIT_TEST(gr_intr_cleanup, test_gr_init_setup_cleanup, NULL, 0),
};
UNIT_MODULE(nvgpu_gr_intr, nvgpu_gr_intr_tests, UNIT_PRIO_NVGPU_TEST);