2023-08-22 03:16:49 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0+
|
|
|
|
/*
|
|
|
|
* Copyright (c) 2013 The Chromium OS Authors.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <common.h>
|
|
|
|
#include <efi.h>
|
|
|
|
#include <initcall.h>
|
|
|
|
#include <log.h>
|
2023-08-22 03:16:54 +00:00
|
|
|
#include <relocate.h>
|
2023-08-22 03:16:49 +00:00
|
|
|
#include <asm/global_data.h>
|
|
|
|
|
|
|
|
DECLARE_GLOBAL_DATA_PTR;
|
|
|
|
|
2023-08-22 03:16:50 +00:00
|
|
|
static ulong calc_reloc_ofs(void)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_EFI_APP
|
|
|
|
return (ulong)image_base;
|
|
|
|
#endif
|
|
|
|
/*
|
|
|
|
* Sandbox is relocated by the OS, so symbols always appear at
|
|
|
|
* the relocated address.
|
|
|
|
*/
|
|
|
|
if (IS_ENABLED(CONFIG_SANDBOX) || (gd->flags & GD_FLG_RELOC))
|
|
|
|
return gd->reloc_off;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2023-08-22 03:16:54 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* initcall_is_event() - Get the event number for an initcall
|
|
|
|
*
|
|
|
|
* func: Function pointer to check
|
|
|
|
* Return: Event number, if this is an event, else 0
|
|
|
|
*/
|
|
|
|
static int initcall_is_event(init_fnc_t func)
|
|
|
|
{
|
|
|
|
ulong val = (ulong)func;
|
|
|
|
|
|
|
|
if ((val & INITCALL_IS_EVENT) == INITCALL_IS_EVENT)
|
|
|
|
return val & INITCALL_EVENT_TYPE;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2023-08-22 03:16:49 +00:00
|
|
|
/*
|
|
|
|
* To enable debugging. add #define DEBUG at the top of the including file.
|
|
|
|
*
|
|
|
|
* To find a symbol, use grep on u-boot.map
|
|
|
|
*/
|
|
|
|
int initcall_run_list(const init_fnc_t init_sequence[])
|
|
|
|
{
|
2023-08-22 03:16:50 +00:00
|
|
|
ulong reloc_ofs = calc_reloc_ofs();
|
2023-08-22 03:16:51 +00:00
|
|
|
const init_fnc_t *ptr;
|
2023-08-22 03:16:54 +00:00
|
|
|
enum event_t type;
|
2023-08-22 03:16:51 +00:00
|
|
|
init_fnc_t func;
|
|
|
|
int ret = 0;
|
2023-08-22 03:16:49 +00:00
|
|
|
|
2023-08-22 03:16:51 +00:00
|
|
|
for (ptr = init_sequence; func = *ptr, !ret && func; ptr++) {
|
2023-08-22 03:16:54 +00:00
|
|
|
type = initcall_is_event(func);
|
|
|
|
|
|
|
|
if (type) {
|
|
|
|
if (!CONFIG_IS_ENABLED(EVENT))
|
|
|
|
continue;
|
|
|
|
debug("initcall: event %d/%s\n", type,
|
|
|
|
event_type_name(type));
|
|
|
|
} else if (reloc_ofs) {
|
2023-08-22 03:16:49 +00:00
|
|
|
debug("initcall: %p (relocated to %p)\n",
|
2023-08-22 03:16:54 +00:00
|
|
|
(char *)func - reloc_ofs, (char *)func);
|
2023-08-22 03:16:51 +00:00
|
|
|
} else {
|
|
|
|
debug("initcall: %p\n", (char *)func - reloc_ofs);
|
|
|
|
}
|
2023-08-22 03:16:49 +00:00
|
|
|
|
2023-08-22 03:16:54 +00:00
|
|
|
ret = type ? event_notify_null(type) : func();
|
2023-08-22 03:16:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (ret) {
|
2023-08-22 03:16:54 +00:00
|
|
|
if (CONFIG_IS_ENABLED(EVENT)) {
|
|
|
|
char buf[60];
|
|
|
|
|
|
|
|
/* don't worry about buf size as we are dying here */
|
|
|
|
if (type) {
|
|
|
|
sprintf(buf, "event %d/%s", type,
|
|
|
|
event_type_name(type));
|
|
|
|
} else {
|
|
|
|
sprintf(buf, "call %p", func);
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("initcall failed at %s (err=%dE)\n", buf, ret);
|
|
|
|
} else {
|
|
|
|
printf("initcall failed at call %p (err=%d)\n",
|
|
|
|
(char *)func - reloc_ofs, ret);
|
|
|
|
}
|
2023-08-22 03:16:52 +00:00
|
|
|
|
|
|
|
return ret;
|
2023-08-22 03:16:49 +00:00
|
|
|
}
|
2023-08-22 03:16:51 +00:00
|
|
|
|
2023-08-22 03:16:49 +00:00
|
|
|
return 0;
|
|
|
|
}
|