2009-12-06 02:57:05 +01:00
|
|
|
/** @file
|
|
|
|
Add custom commands for BeagleBoard development.
|
|
|
|
|
2010-04-29 14:24:22 +02:00
|
|
|
Copyright (c) 2008 - 2010, Apple Inc. All rights reserved.<BR>
|
2014-08-19 15:29:52 +02:00
|
|
|
|
2010-04-29 14:24:22 +02:00
|
|
|
This program and the accompanying materials
|
2009-12-06 02:57:05 +01:00
|
|
|
are licensed and made available under the terms and conditions of the BSD License
|
|
|
|
which accompanies this distribution. The full text of the license may be found at
|
|
|
|
http://opensource.org/licenses/bsd-license.php
|
|
|
|
|
|
|
|
THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
|
|
|
|
|
|
|
|
**/
|
|
|
|
|
|
|
|
#include <PiDxe.h>
|
|
|
|
#include <Library/ArmLib.h>
|
|
|
|
#include <Library/CacheMaintenanceLib.h>
|
|
|
|
#include <Library/EblCmdLib.h>
|
|
|
|
#include <Library/BaseLib.h>
|
|
|
|
#include <Library/DebugLib.h>
|
|
|
|
#include <Library/UefiBootServicesTableLib.h>
|
|
|
|
#include <Library/UefiRuntimeServicesTableLib.h>
|
|
|
|
#include <Library/MemoryAllocationLib.h>
|
|
|
|
#include <Library/UefiLib.h>
|
|
|
|
#include <Library/PcdLib.h>
|
|
|
|
#include <Library/EfiFileLib.h>
|
2010-02-01 19:59:27 +01:00
|
|
|
#include <Library/ArmDisassemblerLib.h>
|
2010-02-20 00:49:51 +01:00
|
|
|
#include <Library/PeCoffGetEntryPointLib.h>
|
2010-04-21 19:48:09 +02:00
|
|
|
#include <Library/PerformanceLib.h>
|
|
|
|
#include <Library/TimerLib.h>
|
2009-12-06 02:57:05 +01:00
|
|
|
|
2010-02-20 00:49:51 +01:00
|
|
|
#include <Guid/DebugImageInfoTable.h>
|
2010-04-21 19:48:09 +02:00
|
|
|
|
2010-02-20 00:49:51 +01:00
|
|
|
#include <Protocol/DebugSupport.h>
|
|
|
|
#include <Protocol/LoadedImage.h>
|
2010-01-27 03:47:47 +01:00
|
|
|
|
|
|
|
/**
|
2010-02-20 00:49:51 +01:00
|
|
|
Simple arm disassembler via a library
|
2010-01-27 03:47:47 +01:00
|
|
|
|
2010-02-20 00:49:51 +01:00
|
|
|
Argv[0] - symboltable
|
2014-08-19 15:29:52 +02:00
|
|
|
Argv[1] - Optional quoted format string
|
2010-02-20 00:49:51 +01:00
|
|
|
Argv[2] - Optional flag
|
|
|
|
|
|
|
|
@param Argc Number of command arguments in Argv
|
2014-08-19 15:29:52 +02:00
|
|
|
@param Argv Array of strings that represent the parsed command line.
|
2011-09-23 00:53:54 +02:00
|
|
|
Argv[0] is the command name
|
2010-02-20 00:49:51 +01:00
|
|
|
|
|
|
|
@return EFI_SUCCESS
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EblSymbolTable (
|
|
|
|
IN UINTN Argc,
|
|
|
|
IN CHAR8 **Argv
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_DEBUG_IMAGE_INFO_TABLE_HEADER *DebugImageTableHeader = NULL;
|
|
|
|
EFI_DEBUG_IMAGE_INFO *DebugTable;
|
|
|
|
UINTN Entry;
|
|
|
|
CHAR8 *Format;
|
|
|
|
CHAR8 *Pdb;
|
|
|
|
UINT32 PeCoffSizeOfHeaders;
|
|
|
|
UINT32 ImageBase;
|
|
|
|
BOOLEAN Elf;
|
2014-08-19 15:29:52 +02:00
|
|
|
|
2010-02-20 00:49:51 +01:00
|
|
|
// Need to add lots of error checking on the passed in string
|
2010-09-08 19:58:18 +02:00
|
|
|
// Default string is for RealView debugger or gdb depending on toolchain used.
|
|
|
|
if (Argc > 1) {
|
|
|
|
Format = Argv[1];
|
|
|
|
} else {
|
|
|
|
#if __GNUC__
|
|
|
|
// Assume gdb
|
|
|
|
Format = "add-symbol-file %a 0x%x";
|
|
|
|
#else
|
2014-08-19 15:29:52 +02:00
|
|
|
// Default to RVCT
|
2010-09-08 19:58:18 +02:00
|
|
|
Format = "load /a /ni /np %a &0x%x";
|
|
|
|
#endif
|
|
|
|
}
|
2010-02-20 00:49:51 +01:00
|
|
|
Elf = (Argc > 2) ? FALSE : TRUE;
|
2014-08-19 15:29:52 +02:00
|
|
|
|
2010-02-20 00:49:51 +01:00
|
|
|
Status = EfiGetSystemConfigurationTable (&gEfiDebugImageInfoTableGuid, (VOID **)&DebugImageTableHeader);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
2014-08-19 15:29:52 +02:00
|
|
|
|
2010-02-20 00:49:51 +01:00
|
|
|
DebugTable = DebugImageTableHeader->EfiDebugImageInfoTable;
|
|
|
|
if (DebugTable == NULL) {
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (Entry = 0; Entry < DebugImageTableHeader->TableSize; Entry++, DebugTable++) {
|
|
|
|
if (DebugTable->NormalImage != NULL) {
|
|
|
|
if ((DebugTable->NormalImage->ImageInfoType == EFI_DEBUG_IMAGE_INFO_TYPE_NORMAL) && (DebugTable->NormalImage->LoadedImageProtocolInstance != NULL)) {
|
|
|
|
ImageBase = (UINT32)DebugTable->NormalImage->LoadedImageProtocolInstance->ImageBase;
|
|
|
|
PeCoffSizeOfHeaders = PeCoffGetSizeOfHeaders ((VOID *)(UINTN)ImageBase);
|
|
|
|
Pdb = PeCoffLoaderGetPdbPointer (DebugTable->NormalImage->LoadedImageProtocolInstance->ImageBase);
|
2010-03-02 01:06:19 +01:00
|
|
|
if (Pdb != NULL) {
|
|
|
|
if (Elf) {
|
|
|
|
// ELF and Mach-O images don't include the header so the linked address does not include header
|
|
|
|
ImageBase += PeCoffSizeOfHeaders;
|
2014-08-19 15:29:52 +02:00
|
|
|
}
|
2010-03-02 01:06:19 +01:00
|
|
|
AsciiPrint (Format, Pdb, ImageBase);
|
|
|
|
AsciiPrint ("\n");
|
|
|
|
} else {
|
|
|
|
}
|
2010-02-20 00:49:51 +01:00
|
|
|
}
|
2014-08-19 15:29:52 +02:00
|
|
|
}
|
2010-02-20 00:49:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
Simple arm disassembler via a library
|
|
|
|
|
|
|
|
Argv[0] - disasm
|
|
|
|
Argv[1] - Address to start disassembling from
|
|
|
|
ARgv[2] - Number of instructions to disassembly (optional)
|
2010-01-27 03:47:47 +01:00
|
|
|
|
|
|
|
@param Argc Number of command arguments in Argv
|
2014-08-19 15:29:52 +02:00
|
|
|
@param Argv Array of strings that represent the parsed command line.
|
2011-09-23 00:53:54 +02:00
|
|
|
Argv[0] is the command name
|
2010-01-27 03:47:47 +01:00
|
|
|
|
|
|
|
@return EFI_SUCCESS
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
2010-02-01 19:59:27 +01:00
|
|
|
EblDisassembler (
|
2010-01-27 03:47:47 +01:00
|
|
|
IN UINTN Argc,
|
|
|
|
IN CHAR8 **Argv
|
|
|
|
)
|
|
|
|
{
|
2010-02-02 05:34:19 +01:00
|
|
|
UINT8 *Ptr, *CurrentAddress;
|
2010-02-01 19:59:27 +01:00
|
|
|
UINT32 Address;
|
|
|
|
UINT32 Count;
|
|
|
|
CHAR8 Buffer[80];
|
2010-02-05 07:50:09 +01:00
|
|
|
UINT32 ItBlock;
|
2014-08-19 15:29:52 +02:00
|
|
|
|
2010-02-01 19:59:27 +01:00
|
|
|
if (Argc < 2) {
|
|
|
|
return EFI_INVALID_PARAMETER;
|
|
|
|
}
|
2014-08-19 15:29:52 +02:00
|
|
|
|
2010-02-01 19:59:27 +01:00
|
|
|
Address = AsciiStrHexToUintn (Argv[1]);
|
2010-02-02 05:34:19 +01:00
|
|
|
Count = (Argc > 2) ? (UINT32)AsciiStrHexToUintn (Argv[2]) : 20;
|
2010-02-01 19:59:27 +01:00
|
|
|
|
2014-08-19 15:29:52 +02:00
|
|
|
Ptr = (UINT8 *)(UINTN)Address;
|
2010-02-05 07:50:09 +01:00
|
|
|
ItBlock = 0;
|
2010-02-02 05:34:19 +01:00
|
|
|
do {
|
|
|
|
CurrentAddress = Ptr;
|
2010-02-05 07:50:09 +01:00
|
|
|
DisassembleInstruction (&Ptr, TRUE, TRUE, &ItBlock, Buffer, sizeof (Buffer));
|
2010-02-02 05:34:19 +01:00
|
|
|
AsciiPrint ("0x%08x: %a\n", CurrentAddress, Buffer);
|
|
|
|
} while (Count-- > 0);
|
2014-08-19 15:29:52 +02:00
|
|
|
|
2010-02-01 19:59:27 +01:00
|
|
|
|
2010-01-27 03:47:47 +01:00
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-04-21 19:48:09 +02:00
|
|
|
CHAR8 *
|
|
|
|
ImageHandleToPdbFileName (
|
|
|
|
IN EFI_HANDLE Handle
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
EFI_LOADED_IMAGE_PROTOCOL *LoadedImage;
|
2010-04-22 00:04:35 +02:00
|
|
|
CHAR8 *Pdb;
|
|
|
|
CHAR8 *StripLeading;
|
2010-04-21 19:48:09 +02:00
|
|
|
|
|
|
|
Status = gBS->HandleProtocol (Handle, &gEfiLoadedImageProtocolGuid, (VOID **)&LoadedImage);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2010-04-22 00:04:35 +02:00
|
|
|
Pdb = PeCoffLoaderGetPdbPointer (LoadedImage->ImageBase);
|
|
|
|
StripLeading = AsciiStrStr (Pdb, "\\ARM\\");
|
|
|
|
if (StripLeading == NULL) {
|
|
|
|
StripLeading = AsciiStrStr (Pdb, "/ARM/");
|
|
|
|
if (StripLeading == NULL) {
|
|
|
|
return Pdb;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Hopefully we hacked off the unneeded part
|
|
|
|
return (StripLeading + 5);
|
2010-04-21 19:48:09 +02:00
|
|
|
}
|
|
|
|
|
2010-04-22 00:04:35 +02:00
|
|
|
|
2010-04-21 19:48:09 +02:00
|
|
|
CHAR8 *mTokenList[] = {
|
|
|
|
"SEC",
|
|
|
|
"PEI",
|
|
|
|
"DXE",
|
|
|
|
"BDS",
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
Simple arm disassembler via a library
|
|
|
|
|
|
|
|
Argv[0] - disasm
|
|
|
|
Argv[1] - Address to start disassembling from
|
|
|
|
ARgv[2] - Number of instructions to disassembly (optional)
|
|
|
|
|
|
|
|
@param Argc Number of command arguments in Argv
|
2014-08-19 15:29:52 +02:00
|
|
|
@param Argv Array of strings that represent the parsed command line.
|
2011-09-23 00:53:54 +02:00
|
|
|
Argv[0] is the command name
|
2010-04-21 19:48:09 +02:00
|
|
|
|
|
|
|
@return EFI_SUCCESS
|
|
|
|
|
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
EblPerformance (
|
|
|
|
IN UINTN Argc,
|
|
|
|
IN CHAR8 **Argv
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UINTN Key;
|
|
|
|
CONST VOID *Handle;
|
|
|
|
CONST CHAR8 *Token, *Module;
|
|
|
|
UINT64 Start, Stop, TimeStamp;
|
|
|
|
UINT64 Delta, TicksPerSecond, Milliseconds, Microseconds;
|
|
|
|
UINTN Index;
|
|
|
|
|
|
|
|
TicksPerSecond = GetPerformanceCounterProperties (NULL, NULL);
|
|
|
|
|
|
|
|
Key = 0;
|
|
|
|
do {
|
|
|
|
Key = GetPerformanceMeasurement (Key, (CONST VOID **)&Handle, &Token, &Module, &Start, &Stop);
|
|
|
|
if (Key != 0) {
|
|
|
|
if (AsciiStriCmp ("StartImage:", Token) == 0) {
|
|
|
|
if (Stop == 0) {
|
|
|
|
// The entry for EBL is still running so the stop time will be zero. Skip it
|
|
|
|
AsciiPrint (" running %a\n", ImageHandleToPdbFileName ((EFI_HANDLE)Handle));
|
|
|
|
} else {
|
|
|
|
Delta = Stop - Start;
|
|
|
|
Microseconds = DivU64x64Remainder (MultU64x32 (Delta, 1000000), TicksPerSecond, NULL);
|
|
|
|
AsciiPrint ("%10ld us %a\n", Microseconds, ImageHandleToPdbFileName ((EFI_HANDLE)Handle));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} while (Key != 0);
|
|
|
|
|
|
|
|
AsciiPrint ("\n");
|
|
|
|
|
|
|
|
TimeStamp = 0;
|
|
|
|
Key = 0;
|
|
|
|
do {
|
|
|
|
Key = GetPerformanceMeasurement (Key, (CONST VOID **)&Handle, &Token, &Module, &Start, &Stop);
|
|
|
|
if (Key != 0) {
|
|
|
|
for (Index = 0; mTokenList[Index] != NULL; Index++) {
|
|
|
|
if (AsciiStriCmp (mTokenList[Index], Token) == 0) {
|
|
|
|
Delta = Stop - Start;
|
|
|
|
TimeStamp += Delta;
|
|
|
|
Milliseconds = DivU64x64Remainder (MultU64x32 (Delta, 1000), TicksPerSecond, NULL);
|
|
|
|
AsciiPrint ("%6a %6ld ms\n", Token, Milliseconds);
|
|
|
|
break;
|
|
|
|
}
|
2014-08-19 15:29:52 +02:00
|
|
|
}
|
2010-04-21 19:48:09 +02:00
|
|
|
}
|
|
|
|
} while (Key != 0);
|
|
|
|
|
|
|
|
AsciiPrint ("Total Time = %ld ms\n\n", DivU64x64Remainder (MultU64x32 (TimeStamp, 1000), TicksPerSecond, NULL));
|
|
|
|
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-12-06 02:57:05 +01:00
|
|
|
GLOBAL_REMOVE_IF_UNREFERENCED const EBL_COMMAND_TABLE mLibCmdTemplate[] =
|
|
|
|
{
|
2010-01-27 03:47:47 +01:00
|
|
|
{
|
2010-02-01 19:59:27 +01:00
|
|
|
"disasm address [count]",
|
|
|
|
" disassemble count instructions",
|
2010-01-27 03:47:47 +01:00
|
|
|
NULL,
|
2010-02-01 19:59:27 +01:00
|
|
|
EblDisassembler
|
2010-02-20 00:49:51 +01:00
|
|
|
},
|
2010-04-21 19:48:09 +02:00
|
|
|
{
|
|
|
|
"performance",
|
|
|
|
" Display boot performance info",
|
|
|
|
NULL,
|
|
|
|
EblPerformance
|
|
|
|
},
|
2010-02-20 00:49:51 +01:00
|
|
|
{
|
2010-03-02 01:06:19 +01:00
|
|
|
"symboltable [\"format string\"] [PECOFF]",
|
2010-02-20 00:49:51 +01:00
|
|
|
" show symbol table commands for debugger",
|
|
|
|
NULL,
|
|
|
|
EblSymbolTable
|
2010-01-27 03:47:47 +01:00
|
|
|
}
|
2009-12-06 02:57:05 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
VOID
|
|
|
|
EblInitializeExternalCmd (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
EblAddCommands (mLibCmdTemplate, sizeof (mLibCmdTemplate)/sizeof (EBL_COMMAND_TABLE));
|
|
|
|
return;
|
|
|
|
}
|