blob: 480490ea239d050ea66d479184cc19aed3de5e04 [file] [log] [blame]
Simon Glassafb5a8a2023-08-21 21:16:49 -06001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * Copyright (c) 2013 The Chromium OS Authors.
4 */
5
6#include <common.h>
7#include <efi.h>
8#include <initcall.h>
9#include <log.h>
Simon Glass6b42d382023-08-21 21:16:54 -060010#include <relocate.h>
Simon Glassafb5a8a2023-08-21 21:16:49 -060011#include <asm/global_data.h>
12
13DECLARE_GLOBAL_DATA_PTR;
14
Simon Glass0e33bbf2023-08-21 21:16:50 -060015static ulong calc_reloc_ofs(void)
16{
17#ifdef CONFIG_EFI_APP
18 return (ulong)image_base;
19#endif
20 /*
21 * Sandbox is relocated by the OS, so symbols always appear at
22 * the relocated address.
23 */
24 if (IS_ENABLED(CONFIG_SANDBOX) || (gd->flags & GD_FLG_RELOC))
25 return gd->reloc_off;
26
27 return 0;
28}
Simon Glass6b42d382023-08-21 21:16:54 -060029
30/**
31 * initcall_is_event() - Get the event number for an initcall
32 *
33 * func: Function pointer to check
34 * Return: Event number, if this is an event, else 0
35 */
36static int initcall_is_event(init_fnc_t func)
37{
38 ulong val = (ulong)func;
39
40 if ((val & INITCALL_IS_EVENT) == INITCALL_IS_EVENT)
41 return val & INITCALL_EVENT_TYPE;
42
43 return 0;
44}
45
Simon Glassafb5a8a2023-08-21 21:16:49 -060046/*
47 * To enable debugging. add #define DEBUG at the top of the including file.
48 *
49 * To find a symbol, use grep on u-boot.map
50 */
51int initcall_run_list(const init_fnc_t init_sequence[])
52{
Simon Glass0e33bbf2023-08-21 21:16:50 -060053 ulong reloc_ofs = calc_reloc_ofs();
Simon Glass45a95bf2023-08-21 21:16:51 -060054 const init_fnc_t *ptr;
Simon Glass6b42d382023-08-21 21:16:54 -060055 enum event_t type;
Simon Glass45a95bf2023-08-21 21:16:51 -060056 init_fnc_t func;
57 int ret = 0;
Simon Glassafb5a8a2023-08-21 21:16:49 -060058
Simon Glass45a95bf2023-08-21 21:16:51 -060059 for (ptr = init_sequence; func = *ptr, !ret && func; ptr++) {
Simon Glass6b42d382023-08-21 21:16:54 -060060 type = initcall_is_event(func);
61
62 if (type) {
63 if (!CONFIG_IS_ENABLED(EVENT))
64 continue;
65 debug("initcall: event %d/%s\n", type,
66 event_type_name(type));
67 } else if (reloc_ofs) {
Simon Glassafb5a8a2023-08-21 21:16:49 -060068 debug("initcall: %p (relocated to %p)\n",
Simon Glass6b42d382023-08-21 21:16:54 -060069 (char *)func - reloc_ofs, (char *)func);
Simon Glass45a95bf2023-08-21 21:16:51 -060070 } else {
71 debug("initcall: %p\n", (char *)func - reloc_ofs);
72 }
Simon Glassafb5a8a2023-08-21 21:16:49 -060073
Simon Glass6b42d382023-08-21 21:16:54 -060074 ret = type ? event_notify_null(type) : func();
Simon Glass4e47b6c2023-08-21 21:16:52 -060075 }
76
77 if (ret) {
Simon Glass6b42d382023-08-21 21:16:54 -060078 if (CONFIG_IS_ENABLED(EVENT)) {
79 char buf[60];
80
81 /* don't worry about buf size as we are dying here */
82 if (type) {
83 sprintf(buf, "event %d/%s", type,
84 event_type_name(type));
85 } else {
86 sprintf(buf, "call %p", func);
87 }
88
89 printf("initcall failed at %s (err=%dE)\n", buf, ret);
90 } else {
91 printf("initcall failed at call %p (err=%d)\n",
92 (char *)func - reloc_ofs, ret);
93 }
Simon Glass4e47b6c2023-08-21 21:16:52 -060094
95 return ret;
Simon Glassafb5a8a2023-08-21 21:16:49 -060096 }
Simon Glass45a95bf2023-08-21 21:16:51 -060097
Simon Glassafb5a8a2023-08-21 21:16:49 -060098 return 0;
99}
Simon Glasscad6b822023-08-21 21:16:55 -0600100
101void initcall_manual_reloc(init_fnc_t init_sequence[])
102{
103 init_fnc_t *ptr;
104
105 for (ptr = init_sequence; *ptr; ptr++) {
106 if (!initcall_is_event(*ptr))
107 MANUAL_RELOC(*ptr);
108 }
109}