mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-11-03 21:48:15 +00:00 
			
		
		
		
	The relocation offset can change in some initcall sequences. Handle this and make sure it is used for all debugging statements in init_run_list() Update the trace test to match. Signed-off-by: Simon Glass <sjg@chromium.org> Reviewed-by: Caleb Connolly <caleb.connolly@linaro.org>
		
			
				
	
	
		
			103 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			103 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// SPDX-License-Identifier: GPL-2.0+
 | 
						|
/*
 | 
						|
 * Copyright (c) 2013 The Chromium OS Authors.
 | 
						|
 */
 | 
						|
 | 
						|
#include <efi.h>
 | 
						|
#include <initcall.h>
 | 
						|
#include <log.h>
 | 
						|
#include <relocate.h>
 | 
						|
#include <asm/global_data.h>
 | 
						|
 | 
						|
DECLARE_GLOBAL_DATA_PTR;
 | 
						|
 | 
						|
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;
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * 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;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * 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[])
 | 
						|
{
 | 
						|
	ulong reloc_ofs;
 | 
						|
	const init_fnc_t *ptr;
 | 
						|
	enum event_t type;
 | 
						|
	init_fnc_t func;
 | 
						|
	int ret = 0;
 | 
						|
 | 
						|
	for (ptr = init_sequence; func = *ptr, func; ptr++) {
 | 
						|
		reloc_ofs = calc_reloc_ofs();
 | 
						|
		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) {
 | 
						|
			debug("initcall: %p (relocated to %p)\n",
 | 
						|
			      (char *)func - reloc_ofs, (char *)func);
 | 
						|
		} else {
 | 
						|
			debug("initcall: %p\n", (char *)func - reloc_ofs);
 | 
						|
		}
 | 
						|
 | 
						|
		ret = type ? event_notify_null(type) : func();
 | 
						|
		if (ret)
 | 
						|
			break;
 | 
						|
	}
 | 
						|
 | 
						|
	if (ret) {
 | 
						|
		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",
 | 
						|
					(char *)func - reloc_ofs);
 | 
						|
			}
 | 
						|
 | 
						|
			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);
 | 
						|
		}
 | 
						|
 | 
						|
		return ret;
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 |