blob: 2686b9aed5cd55ef876b277f4dc8424bb6ea0c1c [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
Simon Glassafb5a8a2023-08-21 21:16:49 -06006#include <efi.h>
7#include <initcall.h>
8#include <log.h>
Simon Glass6b42d382023-08-21 21:16:54 -06009#include <relocate.h>
Simon Glassafb5a8a2023-08-21 21:16:49 -060010#include <asm/global_data.h>
11
12DECLARE_GLOBAL_DATA_PTR;
13
Simon Glass0e33bbf2023-08-21 21:16:50 -060014static ulong calc_reloc_ofs(void)
15{
16#ifdef CONFIG_EFI_APP
17 return (ulong)image_base;
18#endif
19 /*
20 * Sandbox is relocated by the OS, so symbols always appear at
21 * the relocated address.
22 */
23 if (IS_ENABLED(CONFIG_SANDBOX) || (gd->flags & GD_FLG_RELOC))
24 return gd->reloc_off;
25
26 return 0;
27}
Simon Glass6b42d382023-08-21 21:16:54 -060028
29/**
30 * initcall_is_event() - Get the event number for an initcall
31 *
32 * func: Function pointer to check
33 * Return: Event number, if this is an event, else 0
34 */
35static int initcall_is_event(init_fnc_t func)
36{
37 ulong val = (ulong)func;
38
39 if ((val & INITCALL_IS_EVENT) == INITCALL_IS_EVENT)
40 return val & INITCALL_EVENT_TYPE;
41
42 return 0;
43}
44
Simon Glassafb5a8a2023-08-21 21:16:49 -060045/*
46 * To enable debugging. add #define DEBUG at the top of the including file.
47 *
48 * To find a symbol, use grep on u-boot.map
49 */
50int initcall_run_list(const init_fnc_t init_sequence[])
51{
Simon Glass4a45d5d2024-06-23 14:30:22 -060052 ulong reloc_ofs;
Simon Glass45a95bf2023-08-21 21:16:51 -060053 const init_fnc_t *ptr;
Simon Glass6b42d382023-08-21 21:16:54 -060054 enum event_t type;
Simon Glass45a95bf2023-08-21 21:16:51 -060055 init_fnc_t func;
56 int ret = 0;
Simon Glassafb5a8a2023-08-21 21:16:49 -060057
Caleb Connolly3c160cd2024-03-05 14:55:13 +000058 for (ptr = init_sequence; func = *ptr, func; ptr++) {
Simon Glass4a45d5d2024-06-23 14:30:22 -060059 reloc_ofs = calc_reloc_ofs();
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();
Caleb Connolly3c160cd2024-03-05 14:55:13 +000075 if (ret)
76 break;
Simon Glass4e47b6c2023-08-21 21:16:52 -060077 }
78
79 if (ret) {
Simon Glass6b42d382023-08-21 21:16:54 -060080 if (CONFIG_IS_ENABLED(EVENT)) {
81 char buf[60];
82
83 /* don't worry about buf size as we are dying here */
84 if (type) {
85 sprintf(buf, "event %d/%s", type,
86 event_type_name(type));
87 } else {
Simon Glass4a45d5d2024-06-23 14:30:22 -060088 sprintf(buf, "call %p",
89 (char *)func - reloc_ofs);
Simon Glass6b42d382023-08-21 21:16:54 -060090 }
91
92 printf("initcall failed at %s (err=%dE)\n", buf, ret);
93 } else {
94 printf("initcall failed at call %p (err=%d)\n",
95 (char *)func - reloc_ofs, ret);
96 }
Simon Glass4e47b6c2023-08-21 21:16:52 -060097
98 return ret;
Simon Glassafb5a8a2023-08-21 21:16:49 -060099 }
Simon Glass45a95bf2023-08-21 21:16:51 -0600100
Simon Glassafb5a8a2023-08-21 21:16:49 -0600101 return 0;
102}