Spdk/lib/init/subsystem.c
paul luse a6dbe3721e update Intel copyright notices
per Intel policy to include file commit date using git cmd
below.  The policy does not apply to non-Intel (C) notices.

git log --follow -C90% --format=%ad --date default <file> | tail -1

and then pull just the 4 digit year from the result.

Intel copyrights were not added to files where Intel either had
no contribution ot the contribution lacked substance (ie license
header updates, formatting changes, etc).  Contribution date used
"--follow -C95%" to get the most accurate date.

Note that several files in this patch didn't end the license/(c)
block with a blank comment line so these were added as the vast
majority of files do have this last blank line.  Simply there for
consistency.

Signed-off-by: paul luse <paul.e.luse@intel.com>
Change-Id: Id5b7ce4f658fe87132f14139ead58d6e285c04d4
Reviewed-on: https://review.spdk.io/gerrit/c/spdk/spdk/+/15192
Tested-by: SPDK CI Jenkins <sys_sgci@intel.com>
Reviewed-by: Jim Harris <james.r.harris@intel.com>
Reviewed-by: Ben Walker <benjamin.walker@intel.com>
Community-CI: Mellanox Build Bot
2022-11-10 08:28:53 +00:00

255 lines
6.0 KiB
C

/* SPDX-License-Identifier: BSD-3-Clause
* Copyright (C) 2016 Intel Corporation.
* All rights reserved.
*/
#include "spdk/stdinc.h"
#include "spdk/init.h"
#include "spdk/log.h"
#include "spdk/queue.h"
#include "spdk/thread.h"
#include "spdk_internal/init.h"
#include "spdk/env.h"
#include "spdk/json.h"
#include "subsystem.h"
TAILQ_HEAD(spdk_subsystem_list, spdk_subsystem);
struct spdk_subsystem_list g_subsystems = TAILQ_HEAD_INITIALIZER(g_subsystems);
TAILQ_HEAD(spdk_subsystem_depend_list, spdk_subsystem_depend);
struct spdk_subsystem_depend_list g_subsystems_deps = TAILQ_HEAD_INITIALIZER(g_subsystems_deps);
static struct spdk_subsystem *g_next_subsystem;
static bool g_subsystems_initialized = false;
static bool g_subsystems_init_interrupted = false;
static spdk_subsystem_init_fn g_subsystem_start_fn = NULL;
static void *g_subsystem_start_arg = NULL;
static spdk_msg_fn g_subsystem_stop_fn = NULL;
static void *g_subsystem_stop_arg = NULL;
static struct spdk_thread *g_fini_thread = NULL;
void
spdk_add_subsystem(struct spdk_subsystem *subsystem)
{
TAILQ_INSERT_TAIL(&g_subsystems, subsystem, tailq);
}
void
spdk_add_subsystem_depend(struct spdk_subsystem_depend *depend)
{
TAILQ_INSERT_TAIL(&g_subsystems_deps, depend, tailq);
}
static struct spdk_subsystem *
_subsystem_find(struct spdk_subsystem_list *list, const char *name)
{
struct spdk_subsystem *iter;
TAILQ_FOREACH(iter, list, tailq) {
if (strcmp(name, iter->name) == 0) {
return iter;
}
}
return NULL;
}
struct spdk_subsystem *
subsystem_find(const char *name)
{
return _subsystem_find(&g_subsystems, name);
}
struct spdk_subsystem *
subsystem_get_first(void)
{
return TAILQ_FIRST(&g_subsystems);
}
struct spdk_subsystem *
subsystem_get_next(struct spdk_subsystem *cur_subsystem)
{
return TAILQ_NEXT(cur_subsystem, tailq);
}
struct spdk_subsystem_depend *
subsystem_get_first_depend(void)
{
return TAILQ_FIRST(&g_subsystems_deps);
}
struct spdk_subsystem_depend *
subsystem_get_next_depend(struct spdk_subsystem_depend *cur_depend)
{
return TAILQ_NEXT(cur_depend, tailq);
}
static void
subsystem_sort(void)
{
bool depends_on, depends_on_sorted;
struct spdk_subsystem *subsystem, *subsystem_tmp;
struct spdk_subsystem_depend *subsystem_dep;
struct spdk_subsystem_list subsystems_list = TAILQ_HEAD_INITIALIZER(subsystems_list);
while (!TAILQ_EMPTY(&g_subsystems)) {
TAILQ_FOREACH_SAFE(subsystem, &g_subsystems, tailq, subsystem_tmp) {
depends_on = false;
TAILQ_FOREACH(subsystem_dep, &g_subsystems_deps, tailq) {
if (strcmp(subsystem->name, subsystem_dep->name) == 0) {
depends_on = true;
depends_on_sorted = !!_subsystem_find(&subsystems_list, subsystem_dep->depends_on);
if (depends_on_sorted) {
continue;
}
break;
}
}
if (depends_on == false) {
TAILQ_REMOVE(&g_subsystems, subsystem, tailq);
TAILQ_INSERT_TAIL(&subsystems_list, subsystem, tailq);
} else {
if (depends_on_sorted == true) {
TAILQ_REMOVE(&g_subsystems, subsystem, tailq);
TAILQ_INSERT_TAIL(&subsystems_list, subsystem, tailq);
}
}
}
}
TAILQ_FOREACH_SAFE(subsystem, &subsystems_list, tailq, subsystem_tmp) {
TAILQ_REMOVE(&subsystems_list, subsystem, tailq);
TAILQ_INSERT_TAIL(&g_subsystems, subsystem, tailq);
}
}
void
spdk_subsystem_init_next(int rc)
{
/* The initialization is interrupted by the spdk_subsystem_fini, so just return */
if (g_subsystems_init_interrupted) {
return;
}
if (rc) {
SPDK_ERRLOG("Init subsystem %s failed\n", g_next_subsystem->name);
g_subsystem_start_fn(rc, g_subsystem_start_arg);
return;
}
if (!g_next_subsystem) {
g_next_subsystem = TAILQ_FIRST(&g_subsystems);
} else {
g_next_subsystem = TAILQ_NEXT(g_next_subsystem, tailq);
}
if (!g_next_subsystem) {
g_subsystems_initialized = true;
g_subsystem_start_fn(0, g_subsystem_start_arg);
return;
}
if (g_next_subsystem->init) {
g_next_subsystem->init();
} else {
spdk_subsystem_init_next(0);
}
}
void
spdk_subsystem_init(spdk_subsystem_init_fn cb_fn, void *cb_arg)
{
struct spdk_subsystem_depend *dep;
g_subsystem_start_fn = cb_fn;
g_subsystem_start_arg = cb_arg;
/* Verify that all dependency name and depends_on subsystems are registered */
TAILQ_FOREACH(dep, &g_subsystems_deps, tailq) {
if (!subsystem_find(dep->name)) {
SPDK_ERRLOG("subsystem %s is missing\n", dep->name);
g_subsystem_start_fn(-1, g_subsystem_start_arg);
return;
}
if (!subsystem_find(dep->depends_on)) {
SPDK_ERRLOG("subsystem %s dependency %s is missing\n",
dep->name, dep->depends_on);
g_subsystem_start_fn(-1, g_subsystem_start_arg);
return;
}
}
subsystem_sort();
spdk_subsystem_init_next(0);
}
static void
subsystem_fini_next(void *arg1)
{
assert(g_fini_thread == spdk_get_thread());
if (!g_next_subsystem) {
/* If the initialized flag is false, then we've failed to initialize
* the very first subsystem and no de-init is needed
*/
if (g_subsystems_initialized) {
g_next_subsystem = TAILQ_LAST(&g_subsystems, spdk_subsystem_list);
}
} else {
if (g_subsystems_initialized || g_subsystems_init_interrupted) {
g_next_subsystem = TAILQ_PREV(g_next_subsystem, spdk_subsystem_list, tailq);
} else {
g_subsystems_init_interrupted = true;
}
}
while (g_next_subsystem) {
if (g_next_subsystem->fini) {
g_next_subsystem->fini();
return;
}
g_next_subsystem = TAILQ_PREV(g_next_subsystem, spdk_subsystem_list, tailq);
}
g_subsystem_stop_fn(g_subsystem_stop_arg);
return;
}
void
spdk_subsystem_fini_next(void)
{
if (g_fini_thread != spdk_get_thread()) {
spdk_thread_send_msg(g_fini_thread, subsystem_fini_next, NULL);
} else {
subsystem_fini_next(NULL);
}
}
void
spdk_subsystem_fini(spdk_msg_fn cb_fn, void *cb_arg)
{
g_subsystem_stop_fn = cb_fn;
g_subsystem_stop_arg = cb_arg;
g_fini_thread = spdk_get_thread();
spdk_subsystem_fini_next();
}
void
subsystem_config_json(struct spdk_json_write_ctx *w, struct spdk_subsystem *subsystem)
{
if (subsystem && subsystem->write_config_json) {
subsystem->write_config_json(w);
} else {
spdk_json_write_null(w);
}
}