gpu: nvgpu: unit: Add unit testing FW

Full documentation for this is in the unit testing
confluence page.

JIRA NVGPU-525
Bug 2261555

Change-Id: I463e6267eb0eb12b7313f8b275266e8faabe5ccf
Signed-off-by: Alex Waterman <alexw@nvidia.com>
Reviewed-on: https://git-master.nvidia.com/r/1683915
GVS: Gerrit_Virtual_Submit
Reviewed-by: Konsta Holtta <kholtta@nvidia.com>
Reviewed-by: Terje Bergstrom <tbergstrom@nvidia.com>
Reviewed-by: mobile promotions <svcmobile_promotions@nvidia.com>
Tested-by: mobile promotions <svcmobile_promotions@nvidia.com>
This commit is contained in:
Alex Waterman
2018-06-27 14:45:07 -07:00
committed by mobile promotions
parent 6e746a97cc
commit 691bf90445
17 changed files with 1343 additions and 17 deletions

134
userspace/src/args.c Normal file
View File

@@ -0,0 +1,134 @@
/*
* Copyright (c) 2018, 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 <stdio.h>
#include <getopt.h>
#include <stdlib.h>
#include <string.h>
#include <unit/core.h>
#include <unit/args.h>
#include <unit/io.h>
static struct option core_opts[] = {
{ "help", 0, NULL, 'h' },
{ "verbose", 0, NULL, 'v' },
{ "quiet", 0, NULL, 'q' },
{ "no-color", 0, NULL, 'C' },
{ "unit-load-path", 1, NULL, 'L' },
{ NULL, 0, NULL, 0 }
};
static const char *core_opts_str = "hvqCL:";
void core_print_help(struct unit_fw *fw)
{
const char **line, *help_msg[] = {
"NvGpu Unit Testing FW. Basic usage\n",
"\n",
" $ nvgpu_unit [options] <unit>\n",
"\n",
"Basic usage consists of one or more options and a particular unit test to\n",
"execute.\n",
"\n",
"Available options are as follows:\n",
"\n",
" -h, --help Print this help message and exit.\n",
" -v, --verbose Increment the verbosity level. Can be specified\n",
" multiple times.\n",
" -q, --quiet Set the verbose level back to 0.\n",
" -C, --no-color Disable color printing; for example, if writing\n",
" output to a file the color escape sequences will\n",
" corrupt that file.\n",
" -L, --unit-load-path <PATH>\n",
" Path to where the unit test libraries reside.\n",
"\n",
"Note: mandatory arguments to long arguments are mandatory for short\n",
"arguments as well.\n",
NULL
};
line = help_msg;
while (*line != NULL) {
core_msg(fw, "%s", *line);
line++;
}
}
static void set_arg_defaults(struct unit_fw_args *args)
{
args->unit_load_path = DEFAULT_ARG_UNIT_LOAD_PATH;
}
/*
* Parse command line arguments.
*/
int core_parse_args(struct unit_fw *fw, int argc, char **argv)
{
int c, opt_index;
struct unit_fw_args *args;
args = malloc(sizeof(*args));
if (!args)
return -1;
memset(args, 0, sizeof(*args));
set_arg_defaults(args);
fw->args = args;
while (1) {
c = getopt_long(argc, argv,
core_opts_str, core_opts, &opt_index);
if (c == -1)
break;
switch (c) {
case 'h':
args->help = true;
break;
case 'v':
args->verbose_lvl += 1;
break;
case 'q':
args->verbose_lvl = 0;
break;
case 'C':
args->no_color = true;
break;
case 'L':
args->unit_load_path = optarg;
break;
case '?':
args->help = true;
return -1;
default:
core_err(fw, "bug?!\n");
return -1;
}
}
return 0;
}

90
userspace/src/exec.c Normal file
View File

@@ -0,0 +1,90 @@
/*
* Copyright (c) 2018, 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 <unit/io.h>
#include <unit/core.h>
#include <unit/unit.h>
#include <unit/module.h>
#include <unit/results.h>
#include <nvgpu/posix/probe.h>
/*
* Execute a module and all its subtests. This function builds a gk20a for the
* test to use by executing nvgpu_posix_probe() and nvgpu_posix_cleanup();
*/
static int core_exec_module(struct unit_fw *fw,
struct unit_module *module)
{
unsigned int i;
struct gk20a *g = fw->nvgpu.nvgpu_posix_probe();
if (!g)
return -1;
core_vbs(fw, 1, "Execing module: %s\n", module->name);
/*
* Execute each test within the module. No reinit is done between tests.
* Thats up to the module itself to handle. Any setup/teardown between
* unit tests must be handled within the module.
*/
for (i = 0; i < module->nr_tests; i++) {
struct unit_module_test *t = module->tests + i;
int test_status;
core_msg(fw, "Running %s.%s\n", module->name, t->name);
test_status = t->fn(module, g, t->args);
if (test_status != UNIT_SUCCESS)
core_msg_color(fw, C_RED,
" Unit error! Test %s.%s FAILED!\n",
module->name, t->name);
core_add_test_record(fw, module, t,
test_status == UNIT_SUCCESS);
}
fw->nvgpu.nvgpu_posix_cleanup(g);
return 0;
}
/*
* Execute all modules loaded by the unit test framework.
*/
int core_exec(struct unit_fw *fw)
{
int ret;
struct unit_module **modules;
for (modules = fw->modules; *modules != NULL; modules++) {
ret = core_exec_module(fw, *modules);
if (ret != 0)
return ret;
}
return 0;
}

105
userspace/src/io.c Normal file
View File

@@ -0,0 +1,105 @@
/*
* Copyright (c) 2018, 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 <stdio.h>
#include <stdarg.h>
#include <string.h>
#include <unit/io.h>
#include <unit/args.h>
#include <unit/core.h>
#include <unit/unit.h>
#define MAX_LOG_LINE_LENGTH 4096
static void __core_print_file(struct unit_fw *fw, FILE *filp,
const char *prefix, const char *msg,
const char *color)
{
if (color == NULL || args(fw)->no_color)
color = "";
fprintf(filp, "[%s%s%s] %s%s%s",
color, prefix, C_RESET,
color, msg, C_RESET);
}
__attribute__((format (printf, 3, 4)))
void __core_print_stdout(struct unit_fw *fw, const char *color,
const char *fmt, ...)
{
va_list args;
char buf[MAX_LOG_LINE_LENGTH];
va_start(args, fmt);
vsnprintf(buf, MAX_LOG_LINE_LENGTH, fmt, args);
va_end(args);
buf[MAX_LOG_LINE_LENGTH - 1] = 0;
__core_print_file(fw, stdout, "C", buf, color);
}
__attribute__((format (printf, 2, 3)))
void __core_print_stderr(struct unit_fw *fw, const char *fmt, ...)
{
va_list args;
char buf[MAX_LOG_LINE_LENGTH];
va_start(args, fmt);
vsnprintf(buf, MAX_LOG_LINE_LENGTH, fmt, args);
va_end(args);
buf[MAX_LOG_LINE_LENGTH - 1] = 0;
__core_print_file(fw, stdout, "E", buf, C_RED);
}
__attribute__((format (printf, 3, 4)))
void __unit_info_color(struct unit_module *unit, const char *color,
const char *fmt, ...)
{
va_list args;
char buf[MAX_LOG_LINE_LENGTH];
char *msg_start;
int written;
/*
* Default color for module prints is blue. Users can still turn this
* off with '-C'.
*/
if (color == NULL)
color = C_BLUE;
/*
* First prepend the unit name to the print.
*/
written = snprintf(buf, MAX_LOG_LINE_LENGTH, " [%s] ", unit->name);
msg_start = buf + written;
va_start(args, fmt);
vsnprintf(msg_start, MAX_LOG_LINE_LENGTH - written, fmt, args);
va_end(args);
__core_print_file(unit->fw, stdout, "T", buf, color);
}

167
userspace/src/module.c Normal file
View File

@@ -0,0 +1,167 @@
/*
* Copyright (c) 2018, 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 <errno.h>
#include <dlfcn.h>
#include <dirent.h>
#include <string.h>
#include <stdlib.h>
#include <sys/types.h>
#include <unit/core.h>
#include <unit/io.h>
#include <unit/args.h>
#include <unit/unit.h>
static int check_module(struct unit_fw *fw, struct unit_module *mod)
{
unsigned int i;
/*
* Make sure this module has reasonable data.
*/
if (mod->name == NULL) {
core_err(fw, "Unnamed module!");
return -1;
}
if (mod->tests == NULL || mod->nr_tests == 0) {
core_err(fw, "%s: Empty module!\n", mod->name);
return -1;
}
for (i = 0; i < mod->nr_tests; i++) {
struct unit_module_test *test = &mod->tests[i];
if (test->name == NULL) {
core_err(fw, "%s: Unnamed test\n", mod->name);
return -1;
}
if (test->fn == NULL) {
core_err(fw, "%s: Test %s missing function \n",
mod->name, test->name);
return -1;
}
}
return 0;
}
static struct unit_module *load_one_module(struct unit_fw *fw,
struct dirent *dent)
{
void *lib_handle;
struct unit_module *mod;
core_vbs(fw, 1, "Loading: %s\n", dent->d_name);
lib_handle = dlopen(dent->d_name, RTLD_NOW);
if (lib_handle == NULL) {
core_err(fw, "Failed to load %s: %s\n",
dent->d_name, dlerror());
return NULL;
}
mod = dlsym(lib_handle, "__unit_module__");
if (mod == NULL) {
core_vbs(fw, 1,
"Failed to resolve __unit_module__ in %s: %s\n",
dent->d_name, dlerror());
return NULL;
}
mod->lib_handle = lib_handle;
mod->fw = fw;
core_vbs(fw, 1, " '%s' contains %lu tests\n", mod->name, mod->nr_tests);
return mod;
}
/*
* Load all the modules we can from the module load path. Return the list of
* loaded module as an array of pointers to modules. The returned list of
* modules is NULL terminated.
*/
struct unit_module **core_load_modules(struct unit_fw *fw)
{
int nr_modules = 0, i;
DIR *load_dir;
struct dirent *ent;
const char *load_path = args(fw)->unit_load_path;
struct unit_module **modules, *mod;
core_vbs(fw, 1, "Loading modules from %s\n", load_path);
/*
* Open and count the number of files in the dir.
*/
load_dir = opendir(load_path);
if (!load_dir) {
core_err(fw, "%s: Unable to open dir (%s)\n",
load_path, strerror(errno));
return NULL;
}
while (readdir(load_dir) != NULL)
nr_modules += 1;
/* '.' and '..' should be skipped. */
nr_modules -= 2;
/*
* Now allocate necessary space for storing pointers to the modules and
* load the modules. +1 for the last NULL entry.
*/
modules = malloc(sizeof(*modules) * (nr_modules + 1));
if (!modules) {
core_err(fw, "Out of mem! (huh?)\n");
goto err;
}
rewinddir(load_dir);
i = 0;
while ((ent = readdir(load_dir)) != NULL) {
if (strcmp(".", ent->d_name) == 0 ||
strcmp("..", ent->d_name) == 0)
continue;
mod = load_one_module(fw, ent);
if (mod == NULL)
continue;
if (check_module(fw, mod) != 0)
continue;
modules[i] = mod;
i++;
}
modules[i] = NULL;
return modules;
err:
closedir(load_dir);
return NULL;
}

78
userspace/src/nvgpu.c Normal file
View File

@@ -0,0 +1,78 @@
/*
* Copyright (c) 2018, 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 <dlfcn.h>
#include <stdlib.h>
#include <unit/io.h>
#include <unit/core.h>
/*
* Load libnvgpu-drv.so. This is done with dlopen() since this will make
* resolving addresses into symbols easier in the future.
*
* Also, this makes people think carefully about what functions to call in
* nvgpu-drv from the unit test FW. The interaction should really be limited
* and doing explicit name lookups is a good way to prevent too much coupling.
*/
int core_load_nvgpu(struct unit_fw *fw)
{
const char *msg;
/*
* Specify a GLOBAL binding so that subsequently loaded unit tests see
* the nvgpu-drv library. They will of course need it (and will access
* it directly). I.e they will link against nvgpu-drv and this should
* satisfy that linkage.
*/
fw->nvgpu_so = dlopen("libnvgpu-drv.so", RTLD_NOW | RTLD_GLOBAL);
if (fw->nvgpu_so == NULL) {
msg = dlerror();
core_err(fw, "Failed to load nvgpu-drv: %s\n", msg);
return -1;
}
/*
* We directly check the value of the returned symbol for these
* functions against NULL because if it is NULL then something is
* terribly wrong.
*/
fw->nvgpu.nvgpu_posix_probe = dlsym(fw->nvgpu_so,
"nvgpu_posix_probe");
if (fw->nvgpu.nvgpu_posix_probe == NULL) {
msg = dlerror();
core_err(fw, "Failed to resolve nvgpu_posix_probe: %s\n", msg);
return -1;
}
fw->nvgpu.nvgpu_posix_cleanup = dlsym(fw->nvgpu_so,
"nvgpu_posix_cleanup");
if (fw->nvgpu.nvgpu_posix_cleanup == NULL) {
msg = dlerror();
core_err(fw, "Failed to resolve nvgpu_posix_cleanup: %s\n", msg);
return -1;
}
return 0;
}

128
userspace/src/results.c Normal file
View File

@@ -0,0 +1,128 @@
/*
* Copyright (c) 2018, 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 <string.h>
#include <unit/io.h>
#include <unit/core.h>
#include <unit/unit.h>
#include <unit/results.h>
static int __init_results(struct unit_fw *fw)
{
struct unit_results *results;
if (fw->results != NULL)
return 0;
results = malloc(sizeof(*results));
if (results == NULL)
return -1;
memset(results, 0, sizeof(*results));
fw->results = results;
return 0;
}
static void add_record(struct unit_test_list *list,
struct unit_test_record *tr)
{
/*
* First entry.
*/
if (list->head == NULL) {
list->head = tr;
list->last = tr;
return;
}
/*
* Add to the end of the list and update the pointer to the last entry
* in the list. This gives us O(1) add time.
*/
list->last->next = tr;
list->last = tr;
}
int core_add_test_record(struct unit_fw *fw,
struct unit_module *mod,
struct unit_module_test *test,
bool success)
{
struct unit_test_record *tr;
/*
* Dones nothing if results are already inited.
*/
if (__init_results(fw) != 0)
return -1;
tr = malloc(sizeof(*tr));
if (tr == NULL)
return -1;
tr->mod = mod;
tr->test = test;
tr->status = success;
tr->next = NULL;
if (success)
add_record(&fw->results->passing, tr);
else
add_record(&fw->results->failing, tr);
fw->results->nr_tests += 1;
if (success)
fw->results->nr_passing += 1;
return 0;
}
void core_print_test_status(struct unit_fw *fw)
{
struct unit_test_list *failing_tests = &fw->results->failing;
struct unit_test_record *rec;
/*
* Print stats for the tests.
*/
core_msg(fw, "\n");
core_msg(fw, "Test results:\n");
core_msg(fw, "-------------\n");
core_msg(fw, "\n");
core_msg(fw, " Passing: %d\n", fw->results->nr_passing);
core_msg(fw, " Failing: %d\n",
fw->results->nr_tests - fw->results->nr_passing);
core_msg(fw, " Total: %d\n", fw->results->nr_tests);
core_msg(fw, "\n");
core_msg(fw, "Failing tests:\n");
core_msg(fw, "\n");
for_record_in_test_list(failing_tests, rec) {
core_msg(fw, " %s.%s\n",
rec->mod->name,
rec->test->name);
}
}

76
userspace/src/unit_main.c Normal file
View File

@@ -0,0 +1,76 @@
/*
* Copyright (c) 2018, 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.
*/
/**
* NvGpu unit testing framework!
*/
#include <stdlib.h>
#include <string.h>
#include <unit/io.h>
#include <unit/core.h>
#include <unit/args.h>
#include <unit/module.h>
#include <unit/results.h>
int main(int argc, char **argv)
{
struct unit_fw *fw;
int ret;
fw = malloc(sizeof(*fw));
if (!fw)
return 1;
memset(fw, 0, sizeof(*fw));
ret = core_parse_args(fw, argc, argv);
if (ret) {
core_err(fw, "Enable to parse args.\n");
core_err(fw, "Exiting!\n");
return 1;
}
core_vbs(fw, 1, "Welcome to the nvgpu unit testing framework!\n");
if (args(fw)->help) {
core_print_help(fw);
return 1;
}
ret = core_load_nvgpu(fw);
if (ret != 0)
return ret;
fw->modules = core_load_modules(fw);
if (fw->modules == NULL)
return -1;
ret = core_exec(fw);
if (ret != 0)
return ret;
core_print_test_status(fw);
return 0;
}