Files
linux-nvgpu/userspace/units/falcon/falcon_tests/falcon.c
srajum b2345cd01a gpu: nvgpu: fixing unit tests for ga10b
- Add support for unit tests to run on orin platform.

JIRA NVGPU-9909 

Change-Id: If4ca69b77d0d8483c0e9f6a6a5a64c3c3e050d65
Signed-off-by: srajum <srajum@nvidia.com>
Reviewed-on: https://git-master.nvidia.com/r/c/linux-nvgpu/+/2737876
Reviewed-by: Dinesh T <dt@nvidia.com>
Reviewed-by: svc-mobile-coverity <svc-mobile-coverity@nvidia.com>
Reviewed-by: svc-mobile-misra <svc-mobile-misra@nvidia.com>
Reviewed-by: Ankur Kishore <ankkishore@nvidia.com>
GVS: Gerrit_Virtual_Submit <buildbot_gerritrpt@nvidia.com>
2023-04-28 02:08:09 -07:00

1381 lines
38 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 <pthread.h>
#include <unit/unit.h>
#include <unit/io.h>
#include <nvgpu/posix/posix-fault-injection.h>
#include <nvgpu/firmware.h>
#include <nvgpu/gk20a.h>
#include <nvgpu/hal_init.h>
#include <nvgpu/posix/io.h>
#include <nvgpu/hw/gp10b/hw_fuse_gp10b.h>
#include <nvgpu/hw/gv11b/hw_falcon_gv11b.h>
#include "../falcon_utf.h"
#include "nvgpu-falcon.h"
#include "common/acr/acr_priv.h"
struct utf_falcon *utf_falcons[FALCON_ID_END];
static struct nvgpu_falcon *pmu_flcn;
static struct nvgpu_falcon *gpccs_flcn;
static struct nvgpu_falcon *uninit_flcn;
static u8 *rand_test_data_unaligned;
static u32 *rand_test_data;
#define NV_PMC_BOOT_0_ARCHITECTURE_GV110 (0x00000015 << \
NVGPU_GPU_ARCHITECTURE_SHIFT)
#define NV_PMC_BOOT_0_ARCHITECTURE_GA100 (0x00000017 << \
NVGPU_GPU_ARCHITECTURE_SHIFT)
#define NV_PMC_BOOT_0_IMPLEMENTATION_B 0xB
#define MAX_MEM_TYPE (MEM_IMEM + 1)
#define RAND_DATA_SIZE (SZ_4K)
static struct utf_falcon *get_utf_falcon_from_addr(struct gk20a *g, u32 addr)
{
struct utf_falcon *flcn = NULL;
u32 flcn_base;
u32 i;
for (i = 0; i < FALCON_ID_END; i++) {
if (utf_falcons[i] == NULL || utf_falcons[i]->flcn == NULL) {
continue;
}
flcn_base = utf_falcons[i]->flcn->flcn_base;
if ((addr >= flcn_base) &&
(addr < (flcn_base + UTF_FALCON_MAX_REG_OFFSET))) {
flcn = utf_falcons[i];
break;
}
}
return flcn;
}
static void writel_access_reg_fn(struct gk20a *g,
struct nvgpu_reg_access *access)
{
struct utf_falcon *flcn = NULL;
flcn = get_utf_falcon_from_addr(g, access->addr);
if (flcn != NULL) {
nvgpu_utf_falcon_writel_access_reg_fn(g, flcn, access);
} else {
nvgpu_posix_io_writel_reg_space(g, access->addr, access->value);
}
nvgpu_posix_io_record_access(g, access);
}
static void readl_access_reg_fn(struct gk20a *g,
struct nvgpu_reg_access *access)
{
struct utf_falcon *flcn = NULL;
flcn = get_utf_falcon_from_addr(g, access->addr);
if (flcn != NULL) {
nvgpu_utf_falcon_readl_access_reg_fn(g, flcn, access);
} else {
access->value = nvgpu_posix_io_readl_reg_space(g, access->addr);
}
}
static struct nvgpu_posix_io_callbacks utf_falcon_reg_callbacks = {
.writel = writel_access_reg_fn,
.writel_check = writel_access_reg_fn,
.bar1_writel = writel_access_reg_fn,
.usermode_writel = writel_access_reg_fn,
.__readl = readl_access_reg_fn,
.readl = readl_access_reg_fn,
.bar1_readl = readl_access_reg_fn,
};
static void utf_falcon_register_io(struct gk20a *g)
{
nvgpu_posix_register_io(g, &utf_falcon_reg_callbacks);
}
static void init_rand_buffer(void)
{
u32 i;
/*
* Fill the test buffer with random data. Always use the same seed to
* make the test deterministic.
*/
srand(0);
for (i = 0; i < RAND_DATA_SIZE/sizeof(u32); i++) {
rand_test_data[i] = (u32) rand();
}
}
static int init_falcon_test_env(struct unit_module *m, struct gk20a *g)
{
int err = 0;
utf_falcon_register_io(g);
/*
* Fuse register fuse_opt_priv_sec_en_r() is read during init_hal hence
* add it to reg space
*/
if (nvgpu_posix_io_add_reg_space(g,
fuse_opt_priv_sec_en_r(), 0x4) != 0) {
unit_err(m, "Add reg space failed!\n");
return -ENOMEM;
}
/* HAL init parameters for gv11b and ga10b */
if (strcmp(g->name, "ga10b") == 0) {
g->params.gpu_arch = NV_PMC_BOOT_0_ARCHITECTURE_GA100;
} else {
g->params.gpu_arch = NV_PMC_BOOT_0_ARCHITECTURE_GV110;
}
g->params.gpu_impl = NV_PMC_BOOT_0_IMPLEMENTATION_B;
/* HAL init required for getting the falcon ops initialized. */
err = nvgpu_init_hal(g);
if (err != 0) {
return -ENODEV;
}
/* Initialize utf & nvgpu falcon for test usage */
utf_falcons[FALCON_ID_PMU] = nvgpu_utf_falcon_init(m, g, FALCON_ID_PMU);
if (utf_falcons[FALCON_ID_PMU] == NULL) {
return -ENODEV;
}
utf_falcons[FALCON_ID_GPCCS] =
nvgpu_utf_falcon_init(m, g, FALCON_ID_GPCCS);
if (utf_falcons[FALCON_ID_GPCCS] == NULL) {
return -ENODEV;
}
/* Set falcons for test usage */
pmu_flcn = &g->pmu_flcn;
gpccs_flcn = &g->gpccs_flcn;
uninit_flcn = &g->fecs_flcn;
/* Create a test buffer to be filled with random data */
rand_test_data = (u32 *) nvgpu_kzalloc(g, RAND_DATA_SIZE);
if (rand_test_data == NULL) {
return -ENOMEM;
}
init_rand_buffer();
return 0;
}
static int free_falcon_test_env(struct unit_module *m, struct gk20a *g,
void *__args)
{
if (pmu_flcn == NULL || !pmu_flcn->is_falcon_supported) {
unit_return_fail(m, "test environment not initialized.");
}
nvgpu_kfree(g, rand_test_data);
nvgpu_utf_falcon_free(g, utf_falcons[FALCON_ID_GPCCS]);
nvgpu_utf_falcon_free(g, utf_falcons[FALCON_ID_PMU]);
return UNIT_SUCCESS;
}
#ifdef CONFIG_NVGPU_FALCON_NON_FUSA
/*
* This function will compare rand_test_data with falcon flcn's imem/dmem
* based on type from offset src of size. It returns 0 on match else
* non-zero value.
*/
static int falcon_read_compare(struct unit_module *m, struct gk20a *g,
enum falcon_mem_type type,
u32 src, u32 size, bool aligned_rand_data)
{
u8 *dest = NULL, *destp, *cmp_test_data = NULL;
u32 total_block_read = 0;
u32 byte_read_count = 0;
u32 byte_cnt = size;
int err;
dest = (u8 *) nvgpu_kzalloc(g, byte_cnt);
if (dest == NULL) {
unit_err(m, "Memory allocation failed\n");
return -ENOMEM;
}
destp = dest;
total_block_read = byte_cnt >> 8;
do {
byte_read_count =
total_block_read ? FALCON_BLOCK_SIZE : byte_cnt;
if (!byte_read_count) {
break;
}
if (type == MEM_IMEM) {
err = nvgpu_falcon_copy_from_imem(pmu_flcn, src,
destp, byte_read_count, 0);
} else if (type == MEM_DMEM) {
err = nvgpu_falcon_copy_from_dmem(pmu_flcn, src,
destp, byte_read_count, 0);
} else {
unit_err(m, "Invalid read type\n");
err = -EINVAL;
goto free_dest;
}
if (err) {
unit_err(m, "Failed to copy from falcon memory\n");
goto free_dest;
}
destp += byte_read_count;
src += byte_read_count;
byte_cnt -= byte_read_count;
} while (total_block_read--);
if (aligned_rand_data) {
cmp_test_data = (u8 *) rand_test_data;
} else {
cmp_test_data = rand_test_data_unaligned;
}
if (memcmp((void *) dest, (void *) cmp_test_data, size) != 0) {
unit_err(m, "Mismatch comparing copied data\n");
err = -EINVAL;
}
free_dest:
nvgpu_kfree(g, dest);
return err;
}
#endif
/*
* This function will check that falcon memory read/write functions with
* specified parameters complete with return value exp_err.
*/
static int falcon_check_read_write(struct gk20a *g,
struct unit_module *m,
struct nvgpu_falcon *flcn,
enum falcon_mem_type type,
u32 dst, u32 byte_cnt, int exp_err)
{
u8 *dest = NULL;
int ret = -1;
int err;
dest = (u8 *) nvgpu_kzalloc(g, byte_cnt);
if (dest == NULL) {
unit_err(m, "Memory allocation failed\n");
goto out;
}
if (type == MEM_IMEM) {
err = nvgpu_falcon_copy_to_imem(flcn, dst,
(u8 *) rand_test_data,
byte_cnt, 0, false, 0);
if (err != exp_err) {
unit_err(m, "Copy to IMEM should %s\n",
exp_err ? "fail" : "pass");
goto free_dest;
}
#ifdef CONFIG_NVGPU_FALCON_NON_FUSA
err = nvgpu_falcon_copy_from_imem(flcn, dst,
dest, byte_cnt, 0);
if (err != exp_err) {
unit_err(m, "Copy from IMEM should %s\n",
exp_err ? "fail" : "pass");
goto free_dest;
}
#endif
} else if (type == MEM_DMEM) {
err = nvgpu_falcon_copy_to_dmem(flcn, dst,
(u8 *) rand_test_data,
byte_cnt, 0);
if (err != exp_err) {
unit_err(m, "Copy to DMEM should %s\n",
exp_err ? "fail" : "pass");
goto free_dest;
}
#ifdef CONFIG_NVGPU_FALCON_NON_FUSA
err = nvgpu_falcon_copy_from_dmem(flcn, dst,
dest, byte_cnt, 0);
if (err != exp_err) {
unit_err(m, "Copy from DMEM should %s\n",
exp_err ? "fail" : "pass");
goto free_dest;
}
#endif
}
ret = 0;
free_dest:
nvgpu_kfree(g, dest);
out:
return ret;
}
static int verify_valid_falcon_sw_init(struct unit_module *m, struct gk20a *g,
u32 flcn_id)
{
int err;
err = nvgpu_falcon_sw_init(g, flcn_id);
if (err != 0) {
unit_err(m, "falcon init with valid ID %d failed\n", flcn_id);
return err;
}
nvgpu_falcon_sw_free(g, flcn_id);
return 0;
}
/*
* Valid/Invalid: Passing valid ID should succeed the call to function
* nvgpu_falcon_sw_init|free. Otherwise it should fail with error.
*/
int test_falcon_sw_init_free(struct unit_module *m, struct gk20a *g,
void *__args)
{
int err;
/* initialize test setup */
if (init_falcon_test_env(m, g) != 0) {
unit_return_fail(m, "Module init failed\n");
}
err = nvgpu_falcon_sw_init(g, FALCON_ID_INVALID);
if (err != -ENODEV) {
unit_return_fail(m, "falcon with invalid id initialized\n");
}
nvgpu_falcon_sw_free(g, FALCON_ID_INVALID);
err = verify_valid_falcon_sw_init(m, g, FALCON_ID_FECS);
if (err != 0) {
unit_return_fail(m, "FECS falcon sw not initialized\n");
}
#ifdef CONFIG_NVGPU_DGPU
err = verify_valid_falcon_sw_init(m, g, FALCON_ID_GSPLITE);
if (err != 0) {
unit_return_fail(m, "GSPLITE falcon sw not initialized\n");
}
err = verify_valid_falcon_sw_init(m, g, FALCON_ID_NVDEC);
if (err != 0) {
unit_return_fail(m, "NVDEC falcon sw not initialized\n");
}
err = verify_valid_falcon_sw_init(m, g, FALCON_ID_SEC2);
if (err != 0) {
unit_return_fail(m, "SEC2 falcon sw not initialized\n");
}
err = verify_valid_falcon_sw_init(m, g, FALCON_ID_MINION);
if (err != 0) {
unit_return_fail(m, "MINION falcon sw not initialized\n");
}
#endif
return UNIT_SUCCESS;
}
int test_falcon_get_id(struct unit_module *m, struct gk20a *g,
void *__args)
{
if (nvgpu_falcon_get_id(gpccs_flcn) == FALCON_ID_GPCCS) {
return UNIT_SUCCESS;
} else {
return UNIT_FAIL;
}
}
static void flcn_mem_scrub_pass(void *data)
{
struct nvgpu_falcon *flcn = (struct nvgpu_falcon *) data;
u32 dmactl_addr = flcn->flcn_base + falcon_falcon_dmactl_r();
struct gk20a *g = flcn->g;
u32 unit_status;
unit_status = nvgpu_posix_io_readl_reg_space(g, dmactl_addr);
unit_status &= ~(falcon_falcon_dmactl_dmem_scrubbing_m() |
falcon_falcon_dmactl_imem_scrubbing_m());
nvgpu_posix_io_writel_reg_space(g, dmactl_addr, unit_status);
}
static int flcn_reset_state_check(void *data)
{
struct nvgpu_falcon *flcn = (struct nvgpu_falcon *) data;
struct gk20a *g = flcn->g;
u32 unit_status;
unit_status = nvgpu_posix_io_readl_reg_space(g, flcn->flcn_base +
falcon_falcon_cpuctl_r());
if (unit_status | falcon_falcon_cpuctl_hreset_f(1))
return 0;
else
return -1;
}
static void flcn_mem_scrub_fail(void *data)
{
struct nvgpu_falcon *flcn = (struct nvgpu_falcon *) data;
u32 dmactl_addr = flcn->flcn_base + falcon_falcon_dmactl_r();
struct gk20a *g = flcn->g;
u32 unit_status;
unit_status = nvgpu_posix_io_readl_reg_space(g, dmactl_addr);
unit_status |= (falcon_falcon_dmactl_dmem_scrubbing_m() |
falcon_falcon_dmactl_imem_scrubbing_m());
nvgpu_posix_io_writel_reg_space(g, dmactl_addr, unit_status);
}
/*
* Valid: Reset of initialized Falcon succeeds and if not backed by engine
* dependent reset then check CPU control register for bit
* falcon_falcon_cpuctl_hreset_f(1).
* Invalid: Reset of uninitialized and null falcon fails with error -EINVAL.
*/
int test_falcon_reset(struct unit_module *m, struct gk20a *g, void *__args)
{
struct {
struct nvgpu_falcon *flcn;
void (*pre_reset)(void *);
int exp_err;
int (*reset_state_check)(void *);
} test_data[] = {{NULL, NULL, -EINVAL, NULL},
{uninit_flcn, NULL, -EINVAL, NULL},
{gpccs_flcn, flcn_mem_scrub_pass, 0,
flcn_reset_state_check} };
int size = ARRAY_SIZE(test_data);
void *data;
int err, i;
for (i = 0; i < size; i++) {
if (test_data[i].pre_reset) {
test_data[i].pre_reset((void *)test_data[i].flcn);
}
err = nvgpu_falcon_reset(test_data[i].flcn);
if (err != test_data[i].exp_err) {
unit_return_fail(m, "falcon reset err: %d "
"expected err: %d\n",
err, test_data[i].exp_err);
}
if (test_data[i].reset_state_check) {
data = (void *)test_data[i].flcn;
err = test_data[i].reset_state_check(data);
if (err) {
unit_return_fail(m, "falcon reset state "
"mismatch\n");
}
}
}
return UNIT_SUCCESS;
}
/*
* Invalid: Calling this interface on uninitialized falcon should
* return -EINVAL.
* Valid: Set the mem scrubbing status as done and call should return 0.
* Set the mem scrubbing status as pending and call should return
* -ETIMEDOUT.
*/
int test_falcon_mem_scrub(struct unit_module *m, struct gk20a *g, void *__args)
{
struct {
struct nvgpu_falcon *flcn;
void (*pre_scrub)(void *);
int exp_err;
} test_data[] = {{uninit_flcn, NULL, -EINVAL},
{gpccs_flcn, flcn_mem_scrub_pass, 0},
{gpccs_flcn, flcn_mem_scrub_fail, -ETIMEDOUT} };
int size = ARRAY_SIZE(test_data);
int err, i;
for (i = 0; i < size; i++) {
if (test_data[i].pre_scrub) {
test_data[i].pre_scrub(test_data[i].flcn);
}
err = nvgpu_falcon_mem_scrub_wait(test_data[i].flcn);
if (err != test_data[i].exp_err) {
unit_return_fail(m, "falcon mem scrub err: %d "
"expected err: %d\n",
err, test_data[i].exp_err);
}
}
return UNIT_SUCCESS;
}
/*
* FIXME: Following masks are not yet available in the hw headers.
*/
#define falcon_falcon_idlestate_falcon_busy_m() (U32(0x1U) << 0U)
#define falcon_falcon_idlestate_ext_busy_m() (U32(0x7fffU) << 1U)
static void flcn_idle_pass(void *data)
{
struct nvgpu_falcon *flcn = (struct nvgpu_falcon *) data;
u32 idlestate_addr = flcn->flcn_base + falcon_falcon_idlestate_r();
struct gk20a *g = flcn->g;
u32 unit_status;
unit_status = nvgpu_posix_io_readl_reg_space(g, idlestate_addr);
unit_status &= ~(falcon_falcon_idlestate_falcon_busy_m() |
falcon_falcon_idlestate_ext_busy_m());
nvgpu_posix_io_writel_reg_space(g, idlestate_addr, unit_status);
}
/*
* This is to cover the falcon CPU idle & ext units busy branch in if condition
* in gk20a_is_falcon_idle.
*/
static void flcn_idle_fail_ext_busy(void *data)
{
struct nvgpu_falcon *flcn = (struct nvgpu_falcon *) data;
u32 idlestate_addr = flcn->flcn_base + falcon_falcon_idlestate_r();
struct gk20a *g = flcn->g;
u32 unit_status;
unit_status = nvgpu_posix_io_readl_reg_space(g, idlestate_addr);
unit_status |= falcon_falcon_idlestate_ext_busy_m();
nvgpu_posix_io_writel_reg_space(g, idlestate_addr, unit_status);
}
static void flcn_idle_fail(void *data)
{
struct nvgpu_falcon *flcn = (struct nvgpu_falcon *) data;
u32 idlestate_addr = flcn->flcn_base + falcon_falcon_idlestate_r();
struct gk20a *g = flcn->g;
u32 unit_status;
unit_status = nvgpu_posix_io_readl_reg_space(g, idlestate_addr);
unit_status |= (falcon_falcon_idlestate_falcon_busy_m() |
falcon_falcon_idlestate_ext_busy_m());
nvgpu_posix_io_writel_reg_space(g, idlestate_addr, unit_status);
}
/*
* Invalid: Calling this interface on uninitialized falcon should
* return -EINVAL.
* Valid: Set the Falcon idle state as idle in falcon_falcon_idlestate_r and
* call should return 0. Set it to non-idle and call should return
* -ETIMEDOUT.
*/
int test_falcon_idle(struct unit_module *m, struct gk20a *g, void *__args)
{
struct {
struct nvgpu_falcon *flcn;
void (*pre_idle)(void *);
int exp_err;
} test_data[] = {{uninit_flcn, NULL, -EINVAL},
{gpccs_flcn, flcn_idle_pass, 0},
{gpccs_flcn, flcn_idle_fail_ext_busy, -ETIMEDOUT},
{gpccs_flcn, flcn_idle_fail, -ETIMEDOUT} };
int size = ARRAY_SIZE(test_data);
int err, i;
for (i = 0; i < size; i++) {
if (test_data[i].pre_idle) {
test_data[i].pre_idle(test_data[i].flcn);
}
err = nvgpu_falcon_wait_idle(test_data[i].flcn);
if (err != test_data[i].exp_err) {
unit_return_fail(m, "falcon wait for idle err: %d "
"expected err: %d\n",
err, test_data[i].exp_err);
}
}
return UNIT_SUCCESS;
}
static void flcn_halt_pass(void *data)
{
struct nvgpu_falcon *flcn = (struct nvgpu_falcon *) data;
u32 cpuctl_addr = flcn->flcn_base + falcon_falcon_cpuctl_r();
struct gk20a *g = flcn->g;
u32 unit_status;
unit_status = nvgpu_posix_io_readl_reg_space(g, cpuctl_addr);
unit_status |= falcon_falcon_cpuctl_halt_intr_m();
nvgpu_posix_io_writel_reg_space(g, cpuctl_addr, unit_status);
}
static void flcn_halt_fail(void *data)
{
struct nvgpu_falcon *flcn = (struct nvgpu_falcon *) data;
u32 cpuctl_addr = flcn->flcn_base + falcon_falcon_cpuctl_r();
struct gk20a *g = flcn->g;
u32 unit_status;
unit_status = nvgpu_posix_io_readl_reg_space(g, cpuctl_addr);
unit_status &= ~falcon_falcon_cpuctl_halt_intr_m();
nvgpu_posix_io_writel_reg_space(g, cpuctl_addr, unit_status);
}
/*
* Invalid: Calling this interface on uninitialized falcon should return
* -EINVAL.
*
* Valid: Set the Falcon halt state as halted in falcon_falcon_cpuctl_r and
* call should return 0. Set it to non-halted and call should return
* -ETIMEDOUT.
*/
int test_falcon_halt(struct unit_module *m, struct gk20a *g, void *__args)
{
#define FALCON_WAIT_HALT 200
struct {
struct nvgpu_falcon *flcn;
void (*pre_halt)(void *);
int exp_err;
} test_data[] = {{uninit_flcn, NULL, -EINVAL},
{gpccs_flcn, flcn_halt_pass, 0},
{gpccs_flcn, flcn_halt_fail, -ETIMEDOUT} };
int size = ARRAY_SIZE(test_data);
int err, i;
for (i = 0; i < size; i++) {
if (test_data[i].pre_halt) {
test_data[i].pre_halt(test_data[i].flcn);
}
err = nvgpu_falcon_wait_for_halt(test_data[i].flcn,
FALCON_WAIT_HALT);
if (err != test_data[i].exp_err) {
unit_return_fail(m, "falcon wait for halt err: %d "
"expected err: %d\n",
err, test_data[i].exp_err);
}
}
return UNIT_SUCCESS;
}
/*
* Valid/Invalid: Status of read and write from Falcon
* Valid: Read and write of word-multiple and non-word-multiple data from
* initialized Falcon succeeds.
* Invalid: Read and write for uninitialized Falcon fails
* with error -EINVAL.
*/
int test_falcon_mem_rw_init(struct unit_module *m, struct gk20a *g,
void *__args)
{
u32 dst = 0;
int err = 0, i;
/* write/read to/from uninitialized falcon */
for (i = 0; i < MAX_MEM_TYPE; i++) {
err = falcon_check_read_write(g, m, uninit_flcn, i, dst,
RAND_DATA_SIZE, -EINVAL);
if (err) {
return UNIT_FAIL;
}
}
/* write/read to/from initialized falcon */
for (i = 0; i < MAX_MEM_TYPE; i++) {
err = falcon_check_read_write(g, m, pmu_flcn, i, dst,
RAND_DATA_SIZE, 0);
if (err) {
return UNIT_FAIL;
}
}
/* write/read to/from initialized falcon with non-word-multiple data */
for (i = 0; i < MAX_MEM_TYPE; i++) {
err = falcon_check_read_write(g, m, pmu_flcn, i, dst,
RAND_DATA_SIZE - 1, 0);
if (err) {
return UNIT_FAIL;
}
}
return UNIT_SUCCESS;
}
/*
* Invalid: Read and write for invalid Falcon port should fail
* with error -EINVAL.
*/
int test_falcon_mem_rw_inval_port(struct unit_module *m, struct gk20a *g,
void *__args)
{
int err = 0, size = RAND_DATA_SIZE, port = 2;
if (pmu_flcn == NULL || !pmu_flcn->is_falcon_supported) {
unit_return_fail(m, "test environment not initialized.");
}
/* write to invalid port */
unit_info(m, "Writing %d bytes to imem port %d\n", size, port);
err = nvgpu_falcon_copy_to_imem(pmu_flcn, 0, (u8 *) rand_test_data,
size, port, false, 0);
if (err != -EINVAL) {
unit_return_fail(m, "Copy to IMEM invalid port should fail\n");
}
#ifdef CONFIG_NVGPU_FALCON_NON_FUSA
err = nvgpu_falcon_copy_from_imem(pmu_flcn, 0,
NULL, size, port);
if (err != -EINVAL) {
unit_err(m, "Copy from IMEM invalid port should fail\n");
}
#endif
return UNIT_SUCCESS;
}
/*
* Reading and writing data from/to unaligned data should succeed.
*/
int test_falcon_mem_rw_unaligned_cpu_buffer(struct unit_module *m,
struct gk20a *g,
void *__args)
{
rand_test_data_unaligned = (u8 *)&rand_test_data[0] + 1;
u32 byte_cnt = RAND_DATA_SIZE - 8;
u32 dst = 0;
int err = UNIT_FAIL;
if (pmu_flcn == NULL || !pmu_flcn->is_falcon_supported) {
unit_return_fail(m, "test environment not initialized.");
}
/* write data to valid range in imem from unaligned data */
unit_info(m, "Writing %d bytes to imem\n", byte_cnt);
err = nvgpu_falcon_copy_to_imem(pmu_flcn, dst,
(u8 *) rand_test_data_unaligned,
byte_cnt, 0, false, 0);
if (err) {
unit_return_fail(m, "Failed to copy to IMEM\n");
}
#ifdef CONFIG_NVGPU_FALCON_NON_FUSA
/* verify data written to imem matches */
unit_info(m, "Reading %d bytes from imem\n", byte_cnt);
err = falcon_read_compare(m, g, MEM_IMEM, dst, byte_cnt, false);
if (err) {
unit_err(m, "IMEM read data does not match %d\n", err);
return UNIT_FAIL;
}
#endif
/* write data to valid range in dmem from unaligned data */
unit_info(m, "Writing %d bytes to dmem\n", byte_cnt);
err = nvgpu_falcon_copy_to_dmem(pmu_flcn, dst,
(u8 *) rand_test_data_unaligned,
byte_cnt, 0);
if (err) {
unit_return_fail(m, "Failed to copy to DMEM\n");
}
#ifdef CONFIG_NVGPU_FALCON_NON_FUSA
/* verify data written to dmem matches */
unit_info(m, "Reading %d bytes from dmem\n", byte_cnt);
err = falcon_read_compare(m, g, MEM_DMEM, dst, byte_cnt, false);
if (err) {
unit_err(m, "DMEM read data does not match %d\n", err);
return UNIT_FAIL;
}
#endif
/*
* write data of size 1K to valid range in imem from unaligned data
* to verify the buffering logic in falcon_copy_to_dmem_unaligned_src.
*/
unit_info(m, "Writing %d bytes to imem\n", (u32) SZ_1K);
err = nvgpu_falcon_copy_to_imem(pmu_flcn, dst,
(u8 *) rand_test_data_unaligned,
SZ_1K, 0, false, 0);
if (err) {
unit_return_fail(m, "Failed to copy to IMEM\n");
}
/*
* write data of size 1K to valid range in dmem from unaligned data
* to verify the buffering logic in falcon_copy_to_imem_unaligned_src.
*/
unit_info(m, "Writing %d bytes to dmem\n", (u32) SZ_1K);
err = nvgpu_falcon_copy_to_dmem(pmu_flcn, dst,
(u8 *) rand_test_data_unaligned,
SZ_1K, 0);
if (err) {
unit_return_fail(m, "Failed to copy to DMEM\n");
}
return UNIT_SUCCESS;
}
/*
* Valid/Invalid: Reading and writing data in accessible range should work
* and fail otherwise.
* Valid: Data read from or written to Falcon memory in bounds is valid
* operation and should return success.
* Invalid: Reading and writing data out of Falcon memory bounds should
* return error -EINVAL.
*/
int test_falcon_mem_rw_range(struct unit_module *m, struct gk20a *g,
void *__args)
{
u32 byte_cnt = RAND_DATA_SIZE;
u32 dst = 0;
int err = UNIT_FAIL;
if (pmu_flcn == NULL || !pmu_flcn->is_falcon_supported) {
unit_return_fail(m, "test environment not initialized.");
}
/* write data to valid range in imem */
unit_info(m, "Writing %d bytes to imem\n", byte_cnt);
err = nvgpu_falcon_copy_to_imem(pmu_flcn, dst, (u8 *) rand_test_data,
byte_cnt, 0, false, 0);
if (err) {
unit_return_fail(m, "Failed to copy to IMEM\n");
}
#ifdef CONFIG_NVGPU_FALCON_NON_FUSA
/* verify data written to imem matches */
unit_info(m, "Reading %d bytes from imem\n", byte_cnt);
err = falcon_read_compare(m, g, MEM_IMEM, dst, byte_cnt, true);
if (err) {
unit_err(m, "IMEM read data does not match %d\n", err);
return UNIT_FAIL;
}
#endif
/* write data to valid range in dmem */
unit_info(m, "Writing %d bytes to dmem\n", byte_cnt);
err = nvgpu_falcon_copy_to_dmem(pmu_flcn, dst, (u8 *) rand_test_data,
byte_cnt, 0);
if (err) {
unit_return_fail(m, "Failed to copy to DMEM\n");
}
#ifdef CONFIG_NVGPU_FALCON_NON_FUSA
/* verify data written to dmem matches */
unit_info(m, "Reading %d bytes from dmem\n", byte_cnt);
err = falcon_read_compare(m, g, MEM_DMEM, dst, byte_cnt, true);
if (err) {
unit_err(m, "DMEM read data does not match %d\n", err);
return UNIT_FAIL;
}
#endif
dst = UTF_FALCON_IMEM_DMEM_SIZE - RAND_DATA_SIZE;
byte_cnt *= 2;
/* write/read data to/from invalid range in imem */
err = falcon_check_read_write(g, m, pmu_flcn, MEM_IMEM, dst,
byte_cnt, -EINVAL);
if (err) {
return UNIT_FAIL;
}
/* write/read data to/from invalid range in dmem */
err = falcon_check_read_write(g, m, pmu_flcn, MEM_DMEM, dst,
byte_cnt, -EINVAL);
if (err) {
return UNIT_FAIL;
}
dst = UTF_FALCON_IMEM_DMEM_SIZE;
/* write/read data to/from invalid offset in imem */
err = falcon_check_read_write(g, m, pmu_flcn, MEM_IMEM, dst,
byte_cnt, -EINVAL);
if (err) {
return UNIT_FAIL;
}
/* write/read data to/from invalid offset in dmem */
err = falcon_check_read_write(g, m, pmu_flcn, MEM_DMEM, dst,
byte_cnt, -EINVAL);
if (err) {
return UNIT_FAIL;
}
return UNIT_SUCCESS;
}
/*
* Writing data to falcon's DMEM should not succeed when DMEMC
* read returns invalid value due to HW fault.
*/
int test_falcon_mem_rw_fault(struct unit_module *m, struct gk20a *g,
void *__args)
{
struct nvgpu_posix_fault_inj *falcon_memcpy_fi =
nvgpu_utf_falcon_memcpy_get_fault_injection();
u32 byte_cnt = RAND_DATA_SIZE;
u32 dst = 0;
int err = UNIT_FAIL;
if (pmu_flcn == NULL || !pmu_flcn->is_falcon_supported) {
unit_return_fail(m, "test environment not initialized.");
}
/* cause write failure */
nvgpu_posix_enable_fault_injection(falcon_memcpy_fi, true, 0);
unit_info(m, "Writing %d bytes to dmem with hw fault injected.\n",
byte_cnt);
err = nvgpu_falcon_copy_to_dmem(pmu_flcn, dst, (u8 *) rand_test_data,
byte_cnt, 0);
nvgpu_posix_enable_fault_injection(falcon_memcpy_fi, false, 0);
if (err == 0) {
unit_return_fail(m, "Copy to DMEM succeeded with faulty hw.\n");
}
return UNIT_SUCCESS;
}
/*
* Valid/Invalid: Reading and writing data at offset that is word (4-byte)
* aligned data should work and fail otherwise.
* Valid: Data read/written from/to Falcon memory from word (4-byte) aligned
* offset is valid operation and should return success.
* Invalid: Reading and writing data out of non-word-aligned offset in Falcon
* memory should return error -EINVAL.
*/
int test_falcon_mem_rw_aligned(struct unit_module *m, struct gk20a *g,
void *__args)
{
u32 byte_cnt = RAND_DATA_SIZE;
u32 dst = 0, i;
int err = 0;
if (pmu_flcn == NULL || !pmu_flcn->is_falcon_supported) {
unit_return_fail(m, "test environment not initialized.");
}
for (i = 0; i < MAX_MEM_TYPE; i++) {
/*
* Copy to/from offset dst = 3 that is not word aligned should
* fail.
*/
dst = 0x3;
err = falcon_check_read_write(g, m, pmu_flcn, i, dst,
byte_cnt, -EINVAL);
if (err) {
return UNIT_FAIL;
}
/*
* Copy to/from offset dst = 4 that is word aligned should
* succeed.
*/
dst = 0x4;
err = falcon_check_read_write(g, m, pmu_flcn, i, dst,
byte_cnt, 0);
if (err) {
return UNIT_FAIL;
}
}
return UNIT_SUCCESS;
}
/*
* Reading/writing zero bytes should return error -EINVAL.
*/
int test_falcon_mem_rw_zero(struct unit_module *m, struct gk20a *g,
void *__args)
{
u32 byte_cnt = 0;
u32 dst = 0, i;
int err = 0;
if (pmu_flcn == NULL || !pmu_flcn->is_falcon_supported) {
unit_return_fail(m, "test environment not initialized.");
}
for (i = 0; i < MAX_MEM_TYPE; i++) {
/* write/read zero bytes should fail*/
err = falcon_check_read_write(g, m, pmu_flcn, i, dst,
byte_cnt, -EINVAL);
if (err) {
return UNIT_FAIL;
}
}
return UNIT_SUCCESS;
}
/*
* Invalid: Calling read interface on uninitialized falcon should return value 0
* and do nothing with write interface.
* Invalid: Pass invalid mailbox number and verify that read returns zero and
* write does not fail.
*
* Valid: Write the value of a mailbox register through this interface and
* verify the expected value in register falcon_falcon_mailbox0_r.
* Read the value through this interface and verify that it matches
* the register value.
*/
int test_falcon_mailbox(struct unit_module *m, struct gk20a *g, void *__args)
{
#define SAMPLE_MAILBOX_DATA 0xDEADBEED
u32 val, reg_data, mailbox_addr, i;
nvgpu_falcon_mailbox_write(uninit_flcn, FALCON_MAILBOX_0,
SAMPLE_MAILBOX_DATA);
val = nvgpu_falcon_mailbox_read(uninit_flcn, FALCON_MAILBOX_0);
if (val != 0) {
unit_return_fail(m, "Invalid falcon's mailbox read"
"should return zero\n");
}
for (i = FALCON_MAILBOX_0; i <= FALCON_MAILBOX_COUNT; i++) {
nvgpu_falcon_mailbox_write(gpccs_flcn, i, SAMPLE_MAILBOX_DATA);
val = nvgpu_falcon_mailbox_read(gpccs_flcn, i);
if (i == FALCON_MAILBOX_COUNT) {
if (val != 0) {
unit_return_fail(m, "Invalid mailbox read "
"should return zero\n");
} else {
continue;
}
}
mailbox_addr = i != 0U ? falcon_falcon_mailbox1_r() :
falcon_falcon_mailbox0_r();
mailbox_addr = gpccs_flcn->flcn_base + mailbox_addr;
reg_data = nvgpu_posix_io_readl_reg_space(g, mailbox_addr);
if (val != SAMPLE_MAILBOX_DATA || val != reg_data) {
unit_return_fail(m, "Failed reading/writing mailbox\n");
}
}
return UNIT_SUCCESS;
}
static bool falcon_check_reg_group(struct gk20a *g,
struct nvgpu_reg_access *sequence,
u32 size)
{
u32 i;
for (i = 0; i < size; i++) {
if (nvgpu_posix_io_readl_reg_space(g, sequence[i].addr) !=
sequence[i].value) {
break;
}
}
return i == size;
}
/*
* Invalid: Calling bootstrap interfaces on uninitialized falcon should return
* -EINVAL.
* Invalid: Invoke nvgpu_falcon_hs_ucode_load_bootstrap with invalid ucode data
* and verify that call fails.
*
* Valid: Invoke nvgpu_falcon_hs_ucode_load_bootstrap with initialized
* falcon with ACR firmware, verify the expected state of falcon
* registers - falcon_falcon_dmactl_r, falcon_falcon_bootvec_r,
* falcon_falcon_cpuctl_r.
*/
int test_falcon_bootstrap(struct unit_module *m, struct gk20a *g, void *__args)
{
/* Define a group of expected register writes */
struct nvgpu_reg_access bootstrap_group[] = {
{ .addr = 0x0041a10c,
.value = falcon_falcon_dmactl_require_ctx_f(0) },
{ .addr = 0x0041a104,
.value = falcon_falcon_bootvec_vec_f(0) },
{ .addr = 0x0041a100,
.value = falcon_falcon_cpuctl_startcpu_f(1) |
falcon_falcon_cpuctl_hreset_f(1) },
};
struct acr_fw_header *fw_hdr = NULL;
struct bin_hdr *hs_bin_hdr = NULL;
struct nvgpu_firmware *acr_fw;
u32 *ucode_header = NULL;
#ifdef CONFIG_NVGPU_FALCON_NON_FUSA
u32 boot_vector = 0xF000;
#endif
u32 *ucode = NULL;
u32 valid_size;
int err;
#ifdef CONFIG_NVGPU_FALCON_NON_FUSA
/** Invalid falcon bootstrap. */
err = nvgpu_falcon_bootstrap(uninit_flcn, boot_vector);
if (err != -EINVAL) {
unit_return_fail(m, "Invalid falcon bootstrap should "
"fail\n");
}
/** Valid falcon bootstrap. */
err = nvgpu_falcon_bootstrap(gpccs_flcn, boot_vector);
if (err) {
unit_return_fail(m, "GPCCS falcon bootstrap failed\n");
}
#endif
if (!g->ops.pmu.is_debug_mode_enabled(g)) {
acr_fw = nvgpu_request_firmware(g, HSBIN_ACR_PROD_UCODE, 0);
if (acr_fw == NULL) {
unit_return_fail(m, "%s ucode get fail for %s",
HSBIN_ACR_PROD_UCODE, g->name);
}
} else {
acr_fw = nvgpu_request_firmware(g, HSBIN_ACR_DBG_UCODE, 0);
if (acr_fw == NULL) {
unit_return_fail(m, "%s ucode get fail for %s",
HSBIN_ACR_DBG_UCODE, g->name);
}
}
hs_bin_hdr = (struct bin_hdr *)(void *)acr_fw->data;
fw_hdr = (struct acr_fw_header *)(void *)(acr_fw->data +
hs_bin_hdr->header_offset);
ucode_header = (u32 *)(void *)(acr_fw->data + fw_hdr->hdr_offset);
ucode = (u32 *)(void *)(acr_fw->data + hs_bin_hdr->data_offset);
/** Invalid falcon hs_ucode_load_bootstrap. */
err = nvgpu_falcon_hs_ucode_load_bootstrap(uninit_flcn,
ucode, ucode_header);
if (err != -EINVAL) {
unit_return_fail(m, "Invalid falcon bootstrap should "
"fail\n");
}
/** Valid falcon hs_ucode_load_bootstrap with falcon reset failure. */
flcn_mem_scrub_fail(gpccs_flcn);
err = nvgpu_falcon_hs_ucode_load_bootstrap(gpccs_flcn,
ucode, ucode_header);
if (err == 0) {
unit_return_fail(m, "ACR bootstrap should have failed "
"as falcon reset is failed.\n");
}
flcn_mem_scrub_pass(gpccs_flcn);
/**
* Valid falcon hs_ucode_load_bootstrap with invalid non-secure
* code size.
*/
valid_size = ucode_header[OS_CODE_SIZE];
ucode_header[OS_CODE_SIZE] = g->ops.falcon.get_mem_size(gpccs_flcn,
MEM_IMEM);
ucode_header[OS_CODE_SIZE] += 4;
err = nvgpu_falcon_hs_ucode_load_bootstrap(gpccs_flcn,
ucode, ucode_header);
if (err == 0) {
unit_return_fail(m, "ACR bootstrap should have failed "
"as non-secure code size > IMEM size.\n");
}
ucode_header[OS_CODE_SIZE] = valid_size;
/**
* Valid falcon hs_ucode_load_bootstrap with invalid secure
* code size.
*/
valid_size = ucode_header[APP_0_CODE_SIZE];
ucode_header[APP_0_CODE_SIZE] = g->ops.falcon.get_mem_size(gpccs_flcn,
MEM_IMEM);
ucode_header[APP_0_CODE_SIZE] += 4;
err = nvgpu_falcon_hs_ucode_load_bootstrap(gpccs_flcn,
ucode, ucode_header);
if (err == 0) {
unit_return_fail(m, "ACR bootstrap should have failed "
"as secure code size > IMEM size.\n");
}
ucode_header[APP_0_CODE_SIZE] = valid_size;
/**
* Valid falcon hs_ucode_load_bootstrap with invalid dmem data
* size.
*/
valid_size = ucode_header[OS_DATA_SIZE];
ucode_header[OS_DATA_SIZE] = g->ops.falcon.get_mem_size(gpccs_flcn,
MEM_DMEM);
ucode_header[OS_DATA_SIZE] += 4;
err = nvgpu_falcon_hs_ucode_load_bootstrap(gpccs_flcn,
ucode, ucode_header);
if (err == 0) {
unit_return_fail(m, "ACR bootstrap should have failed "
"as dmem data size > DMEM size.\n");
}
ucode_header[OS_DATA_SIZE] = valid_size;
/** Valid falcon hs_ucode_load_bootstrap. */
err = nvgpu_falcon_hs_ucode_load_bootstrap(gpccs_flcn,
ucode, ucode_header);
if (err) {
unit_return_fail(m, "GPCCS falcon bootstrap failed\n");
}
if (falcon_check_reg_group(g, bootstrap_group,
ARRAY_SIZE(bootstrap_group)) == false) {
unit_return_fail(m, "Failed checking bootstrap sequence\n");
}
return UNIT_SUCCESS;
}
static void flcn_irq_not_supported(struct nvgpu_falcon *flcn)
{
flcn->is_interrupt_enabled = false;
}
static void flcn_irq_supported(struct nvgpu_falcon *flcn)
{
flcn->is_interrupt_enabled = true;
}
static bool check_flcn_irq_status(struct nvgpu_falcon *flcn, bool enable,
u32 irq_mask, u32 irq_dest)
{
u32 tmp_mask, tmp_dest;
if (enable) {
tmp_mask = nvgpu_posix_io_readl_reg_space(flcn->g,
flcn->flcn_base + falcon_falcon_irqmset_r());
tmp_dest = nvgpu_posix_io_readl_reg_space(flcn->g,
flcn->flcn_base + falcon_falcon_irqdest_r());
if (tmp_mask != irq_mask || tmp_dest != irq_dest) {
return false;
} else {
return true;
}
} else {
tmp_mask = nvgpu_posix_io_readl_reg_space(flcn->g,
flcn->flcn_base + falcon_falcon_irqmclr_r());
if (tmp_mask != 0xffffffff) {
return false;
} else {
return true;
}
}
}
int test_falcon_irq(struct unit_module *m, struct gk20a *g, void *__args)
{
struct {
struct nvgpu_falcon *flcn;
bool enable;
u32 intr_mask;
u32 intr_dest;
void (*pre_irq)(struct nvgpu_falcon *);
bool (*post_irq)(struct nvgpu_falcon *, bool, u32, u32);
} test_data[] = {{uninit_flcn, true, 0, 0, NULL, NULL},
{gpccs_flcn, true, 0, 0, flcn_irq_not_supported, NULL},
{gpccs_flcn, true, 0xdeadbeee, 0xbeeedead,
flcn_irq_supported, check_flcn_irq_status},
{gpccs_flcn, false, 0xdeadbeee, 0xbeeedead,
flcn_irq_supported, check_flcn_irq_status} };
int size = ARRAY_SIZE(test_data);
bool intr_enabled;
bool err;
int i;
intr_enabled = gpccs_flcn->is_interrupt_enabled;
for (i = 0; i < size; i++) {
if (test_data[i].pre_irq) {
test_data[i].pre_irq(test_data[i].flcn);
}
nvgpu_falcon_set_irq(test_data[i].flcn, test_data[i].enable,
test_data[i].intr_mask,
test_data[i].intr_dest);
if (test_data[i].post_irq) {
err = test_data[i].post_irq(test_data[i].flcn,
test_data[i].enable,
test_data[i].intr_mask,
test_data[i].intr_dest);
if (!err) {
unit_return_fail(m, "falcon set_irq err");
}
}
}
gpccs_flcn->is_interrupt_enabled = intr_enabled;
return UNIT_SUCCESS;
}
struct unit_module_test falcon_tests[] = {
UNIT_TEST(falcon_sw_init_free, test_falcon_sw_init_free, NULL, 0),
UNIT_TEST(falcon_get_id, test_falcon_get_id, NULL, 0),
UNIT_TEST(falcon_reset, test_falcon_reset, NULL, 0),
UNIT_TEST(falcon_mem_scrub, test_falcon_mem_scrub, NULL, 0),
UNIT_TEST(falcon_idle, test_falcon_idle, NULL, 0),
UNIT_TEST(falcon_halt, test_falcon_halt, NULL, 0),
UNIT_TEST(falcon_mem_rw_init, test_falcon_mem_rw_init, NULL, 0),
UNIT_TEST(falcon_mem_rw_inval_port,
test_falcon_mem_rw_inval_port, NULL, 0),
UNIT_TEST(falcon_mem_rw_unaligned_cpu_buffer,
test_falcon_mem_rw_unaligned_cpu_buffer, NULL, 0),
UNIT_TEST(falcon_mem_rw_range, test_falcon_mem_rw_range, NULL, 0),
UNIT_TEST(falcon_mem_rw_fault, test_falcon_mem_rw_fault, NULL, 0),
UNIT_TEST(falcon_mem_rw_aligned, test_falcon_mem_rw_aligned, NULL, 0),
UNIT_TEST(falcon_mem_rw_zero, test_falcon_mem_rw_zero, NULL, 0),
UNIT_TEST(falcon_mailbox, test_falcon_mailbox, NULL, 0),
UNIT_TEST(falcon_bootstrap, test_falcon_bootstrap, NULL, 0),
UNIT_TEST(falcon_irq, test_falcon_irq, NULL, 0),
/* Cleanup */
UNIT_TEST(falcon_free_test_env, free_falcon_test_env, NULL, 0),
};
UNIT_MODULE(falcon, falcon_tests, UNIT_PRIO_NVGPU_TEST);