mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-04-06 05:58:06 +02:00
Remote RISC-V 32 bit support.
This commit is contained in:
parent
e7f96059b2
commit
5c1ef5b508
@ -11,7 +11,7 @@ type
|
||||
// Target information, could be different from host debugger
|
||||
TMachineType = (mtNone, mtSPARC, mt386, mt68K, mtPPC, mtPPC64, mtARM, mtARM64,
|
||||
mtOLD_ALPHA, mtIA_64, mtX86_64, mtAVR8, mtALPHA,
|
||||
mtMIPS, mtMIPSEL,mtLA64, mtXTENSA);
|
||||
mtMIPS, mtMIPSEL,mtLA64, mtXTENSA, mtRISCV);
|
||||
TBitness = (bNone, b32, b64);
|
||||
TByteOrder = (boNone, boLSB, boMSB);
|
||||
TOperatingSystem = (osNone, osBSD, osDarwin, osEmbedded, osLinux, osUnix, osMac, osWindows);
|
||||
|
2867
components/fpdebug/fpdbgdisasriscv.pas
Normal file
2867
components/fpdebug/fpdbgdisasriscv.pas
Normal file
File diff suppressed because it is too large
Load Diff
297
components/fpdebug/fpdbgriscvclasses.pas
Normal file
297
components/fpdebug/fpdbgriscvclasses.pas
Normal file
@ -0,0 +1,297 @@
|
||||
unit FpDbgRiscvClasses;
|
||||
|
||||
{$mode objfpc}{$H+}
|
||||
{$packrecords c}
|
||||
{$modeswitch advancedrecords}
|
||||
|
||||
interface
|
||||
|
||||
uses
|
||||
Classes,
|
||||
SysUtils,
|
||||
FpDbgClasses,
|
||||
DbgIntfBaseTypes, DbgIntfDebuggerBase,
|
||||
{$ifdef FORCE_LAZLOGGER_DUMMY} LazLoggerDummy {$else} LazLoggerBase {$endif},
|
||||
FpDbgRsp, FpDbgRspClasses, FpDbgCommon, FpdMemoryTools,
|
||||
FpErrorMessages;
|
||||
|
||||
const
|
||||
SPindex = 2;
|
||||
SPindexDwarf = 2; // Also known as a1, Index refers to active window
|
||||
ReturnAddressIndex = 1;
|
||||
ReturnAddressIndexDwarf = 1;
|
||||
PCIndexDwarf = 32; // Dwarf index
|
||||
nPC = 'pc';
|
||||
nReturnPC = 'ra';
|
||||
nSP = 'sp';
|
||||
|
||||
type
|
||||
{ TDbgRiscvThread }
|
||||
|
||||
TDbgRiscvThread = class(TDbgRspThread)
|
||||
private
|
||||
const
|
||||
lastCPURegIndex = 32; // 32 registers + PC
|
||||
// Offsets to load specific registers from register data
|
||||
// These are byte offsets, to be used when reading from the raw byte register data
|
||||
RegArrayByteLength = 33*4; // Depends on qemu options, but this seems to be the smallest size to handle. Only show basic registers, so rest can be ignored for now.
|
||||
PCindex = 32;
|
||||
protected
|
||||
procedure LoadRegisterCache; override;
|
||||
procedure SaveRegisterCache; override;
|
||||
function GetReturnPC: TDbgPtr;
|
||||
function GetStackUnwinder: TDbgStackUnwinder; override;
|
||||
public
|
||||
destructor Destroy; override;
|
||||
procedure LoadRegisterValues; override;
|
||||
procedure SetRegisterValue(AName: string; AValue: QWord); override;
|
||||
function GetInstructionPointerRegisterValue: TDbgPtr; override;
|
||||
function GetStackBasePointerRegisterValue: TDbgPtr; override;
|
||||
// procedure SetInstructionPointerRegisterValue(AValue: TDbgPtr); override;
|
||||
function GetStackPointerRegisterValue: TDbgPtr; override;
|
||||
// procedure SetStackPointerRegisterValue(AValue: TDbgPtr); override;
|
||||
end;
|
||||
|
||||
{ TDbgRiscvProcess }
|
||||
|
||||
TDbgRiscvProcess = class(TDbgRspProcess)
|
||||
private const
|
||||
FNumRegisters = 33; // x0..x31,pc
|
||||
protected
|
||||
function CreateThread(AthreadIdentifier: THandle; out IsMainThread: boolean): TDbgThread; override;
|
||||
function CreateBreakPointTargetHandler: TFpBreakPointTargetHandler; override;
|
||||
public
|
||||
class function isSupported(target: TTargetDescriptor): boolean; override;
|
||||
constructor Create(const AFileName: string; AnOsClasses: TOSDbgClasses;
|
||||
AMemManager: TFpDbgMemManager; AMemModel: TFpDbgMemModel;
|
||||
AProcessConfig: TDbgProcessConfig = nil); override;
|
||||
destructor Destroy; override;
|
||||
end;
|
||||
|
||||
TRiscvBreakInfo = object
|
||||
const
|
||||
_CODE: Word = $9002; // c.ebreak -> this is a 16 bit instruction
|
||||
end;
|
||||
|
||||
TRiscvBreakPointTargetHandler = specialize TRspBreakPointTargetHandler<Word, TRiscvBreakInfo>;
|
||||
|
||||
implementation
|
||||
|
||||
uses
|
||||
FpDbgDisasRiscv, FpDbgDwarfDataClasses;
|
||||
|
||||
var
|
||||
DBG_VERBOSE, DBG_WARNINGS: PLazLoggerLogGroup;
|
||||
|
||||
{ TDbgRiscvThread }
|
||||
|
||||
procedure TDbgRiscvThread.LoadRegisterCache;
|
||||
var
|
||||
regs: TBytes;
|
||||
i: integer;
|
||||
begin
|
||||
if not FRegs.Initialized then
|
||||
begin
|
||||
SetLength(regs, RegArrayByteLength);
|
||||
FRegs.Initialized := TDbgRiscvProcess(Process).RspConnection.ReadRegisters(regs[0], length(regs));
|
||||
// 32 bit LE registers
|
||||
for i := 0 to lastCPURegIndex do // x0..x31, PC
|
||||
FRegs.regs[i] := regs[4*i] + (regs[4*i + 1] shl 8) + (regs[4*i + 2] shl 16) + (regs[4*i + 3] shl 24);
|
||||
end;
|
||||
end;
|
||||
|
||||
procedure TDbgRiscvThread.SaveRegisterCache;
|
||||
procedure CopyDWordToByteArray( const val: DWord; regs: PByte);
|
||||
begin
|
||||
regs[0] := val;
|
||||
regs[1] := val shr 8;
|
||||
regs[2] := val shr 16;
|
||||
regs[3] := val shr 24;
|
||||
end;
|
||||
|
||||
var
|
||||
regs: TBytes;
|
||||
i: Integer;
|
||||
begin
|
||||
exit; // TODO: Need to determine which other registers will also change in case of a subroutine call on the debugger side.
|
||||
|
||||
if FRegsChanged then
|
||||
begin
|
||||
SetLength(regs, RegArrayByteLength);
|
||||
for i := 0 to lastCPURegIndex do
|
||||
CopyDWordToByteArray(FRegs.regs[i], @regs[4*i]);
|
||||
end;
|
||||
end;
|
||||
|
||||
function TDbgRiscvThread.GetReturnPC: TDbgPtr;
|
||||
begin
|
||||
Result := 0;
|
||||
if TDbgRiscvProcess(Process).FIsTerminating then
|
||||
begin
|
||||
DebugLn(DBG_WARNINGS, 'TDbgRiscvProcess.GetStackPointerRegisterValue called while FIsTerminating is set.');
|
||||
exit;
|
||||
end;
|
||||
|
||||
if not ReadThreadState then
|
||||
exit;
|
||||
|
||||
DebugLn(DBG_VERBOSE, 'TDbgRiscvProcess.GetStackPointerRegisterValue requesting stack registers.');
|
||||
ReadDebugReg(PCindex, result);
|
||||
end;
|
||||
|
||||
function TDbgRiscvThread.GetStackUnwinder: TDbgStackUnwinder;
|
||||
begin
|
||||
if FUnwinder = nil then
|
||||
FUnwinder := TDbgStackUnwinderRiscv.Create(Process);
|
||||
Result := FUnwinder;
|
||||
end;
|
||||
|
||||
destructor TDbgRiscvThread.Destroy;
|
||||
begin
|
||||
if Assigned(FUnwinder) then
|
||||
FreeAndNil(FUnwinder);
|
||||
inherited Destroy;
|
||||
end;
|
||||
|
||||
procedure TDbgRiscvThread.LoadRegisterValues;
|
||||
var
|
||||
i: integer;
|
||||
begin
|
||||
if TDbgRiscvProcess(Process).FIsTerminating or (TDbgRiscvProcess(Process).FStatus = SIGHUP) then
|
||||
begin
|
||||
DebugLn(DBG_WARNINGS, 'TDbgRiscvProcess.LoadRegisterValues called while FIsTerminating is set.');
|
||||
exit;
|
||||
end;
|
||||
|
||||
if not ReadThreadState then
|
||||
exit;
|
||||
|
||||
LoadRegisterCache;
|
||||
|
||||
if FRegs.Initialized then
|
||||
begin
|
||||
{ FRegs.regs starts with 32 core registers, then PC
|
||||
Todo: Where does floating point and vector registers fit in? }
|
||||
|
||||
for i := 0 to 32 do
|
||||
FRegisterValueList.DbgRegisterAutoCreate[RiscvABIRegisterNames[i]].SetValue(FRegs.regs[i], IntToStr(FRegs.regs[i]), 4, i);
|
||||
|
||||
FRegisterValueListValid := true;
|
||||
end
|
||||
else
|
||||
DebugLn(DBG_WARNINGS, 'Warning: Could not update registers');
|
||||
end;
|
||||
|
||||
procedure TDbgRiscvThread.SetRegisterValue(AName: string; AValue: QWord);
|
||||
var
|
||||
i: integer;
|
||||
res: boolean;
|
||||
begin
|
||||
res := false;
|
||||
for i := low(RiscvABIRegisterNames) to high(RiscvABIRegisterNames) do
|
||||
if AName = RiscvABIRegisterNames[i] then
|
||||
begin
|
||||
res := TDbgRspProcess(Process).RspConnection.WriteDebugReg(i, byte(AValue));
|
||||
break;
|
||||
end;
|
||||
|
||||
if not res then
|
||||
DebugLn(DBG_WARNINGS, 'Error setting register %s to %u', [AName, AValue]);
|
||||
end;
|
||||
|
||||
function TDbgRiscvThread.GetInstructionPointerRegisterValue: TDbgPtr;
|
||||
begin
|
||||
Result := 0;
|
||||
if TDbgRiscvProcess(Process).FIsTerminating then
|
||||
begin
|
||||
DebugLn(DBG_WARNINGS, 'TDbgRiscvProcess.GetInstructionPointerRegisterValue called while FIsTerminating is set.');
|
||||
exit;
|
||||
end;
|
||||
|
||||
if not ReadThreadState then
|
||||
exit;
|
||||
|
||||
DebugLn(DBG_VERBOSE, 'TDbgRiscvProcess.GetInstructionPointerRegisterValue requesting PC.');
|
||||
ReadDebugReg(PCindex, Result);
|
||||
end;
|
||||
|
||||
function TDbgRiscvThread.GetStackBasePointerRegisterValue: TDbgPtr;
|
||||
begin
|
||||
Result := 0;
|
||||
if TDbgRiscvProcess(Process).FIsTerminating then
|
||||
begin
|
||||
DebugLn(DBG_WARNINGS, 'TDbgRiscvProcess.GetStackBasePointerRegisterValue called while FIsTerminating is set.');
|
||||
exit;
|
||||
end;
|
||||
|
||||
if not ReadThreadState then
|
||||
exit;
|
||||
|
||||
DebugLn(DBG_VERBOSE, 'TDbgRiscvProcess.GetStackBasePointerRegisterValue requesting base registers.');
|
||||
// Todo: check FPC implementation of stack frame for riscv
|
||||
Result := GetStackPointerRegisterValue;
|
||||
end;
|
||||
|
||||
function TDbgRiscvThread.GetStackPointerRegisterValue: TDbgPtr;
|
||||
begin
|
||||
Result := 0;
|
||||
if TDbgRiscvProcess(Process).FIsTerminating then
|
||||
begin
|
||||
DebugLn(DBG_WARNINGS, 'TDbgRiscvProcess.GetStackPointerRegisterValue called while FIsTerminating is set.');
|
||||
exit;
|
||||
end;
|
||||
|
||||
if not ReadThreadState then
|
||||
exit;
|
||||
|
||||
DebugLn(DBG_VERBOSE, 'TDbgRiscvProcess.GetStackPointerRegisterValue requesting stack registers.');
|
||||
if not ReadDebugReg(SPindex, result) then
|
||||
Result := 0;
|
||||
end;
|
||||
|
||||
{ TDbgRiscvProcess }
|
||||
|
||||
function TDbgRiscvProcess.CreateThread(AthreadIdentifier: THandle; out IsMainThread: boolean): TDbgThread;
|
||||
begin
|
||||
IsMainThread:=False;
|
||||
if AthreadIdentifier<>feInvalidHandle then
|
||||
begin
|
||||
IsMainThread := AthreadIdentifier=ProcessID;
|
||||
result := TDbgRiscvThread.Create(Self, AthreadIdentifier, AthreadIdentifier);
|
||||
end
|
||||
else
|
||||
result := nil;
|
||||
end;
|
||||
|
||||
function
|
||||
TDbgRiscvProcess.CreateBreakPointTargetHandler: TFpBreakPointTargetHandler;
|
||||
begin
|
||||
Result := TRiscvBreakPointTargetHandler.Create(Self);
|
||||
end;
|
||||
|
||||
constructor TDbgRiscvProcess.Create(const AFileName: string;
|
||||
AnOsClasses: TOSDbgClasses; AMemManager: TFpDbgMemManager;
|
||||
AMemModel: TFpDbgMemModel; AProcessConfig: TDbgProcessConfig);
|
||||
begin
|
||||
FRegArrayLength := FNumRegisters;
|
||||
inherited Create(AFileName, AnOsClasses, AMemManager, AMemModel, AProcessConfig);
|
||||
end;
|
||||
|
||||
destructor TDbgRiscvProcess.Destroy;
|
||||
begin
|
||||
inherited Destroy;
|
||||
end;
|
||||
|
||||
class function TDbgRiscvProcess.isSupported(target: TTargetDescriptor): boolean;
|
||||
begin
|
||||
result := (target.OS = osEmbedded) and (target.machineType = mtRISCV);
|
||||
end;
|
||||
|
||||
initialization
|
||||
DBG_VERBOSE := DebugLogger.FindOrRegisterLogGroup('DBG_VERBOSE' {$IFDEF DBG_VERBOSE} , True {$ENDIF} );
|
||||
DBG_WARNINGS := DebugLogger.FindOrRegisterLogGroup('DBG_WARNINGS' {$IFDEF DBG_WARNINGS} , True {$ENDIF} );
|
||||
RegisterDbgOsClasses(TOSDbgClasses.Create(
|
||||
TDbgRiscvProcess,
|
||||
TDbgRiscvThread,
|
||||
TRiscvAsmDecoder));
|
||||
end.
|
@ -237,6 +237,14 @@ File(s) with other licenses (see also header in file(s):
|
||||
<Filename Value="fpdbgrspclasses.pas"/>
|
||||
<UnitName Value="FpDbgRspClasses"/>
|
||||
</Item>
|
||||
<Item>
|
||||
<Filename Value="fpdbgdisasriscv.pas"/>
|
||||
<UnitName Value="fpdbgdisasriscv"/>
|
||||
</Item>
|
||||
<Item>
|
||||
<Filename Value="fpdbgriscvclasses.pas"/>
|
||||
<UnitName Value="fpdbgriscvclasses"/>
|
||||
</Item>
|
||||
<Item>
|
||||
<Filename Value="fpdbgdisasxtensa.pas"/>
|
||||
<UnitName Value="FpDbgDisasXtensa"/>
|
||||
|
@ -16,8 +16,8 @@ uses
|
||||
FpDbgDwarfDataClasses, FpDbgDwarfFreePascal, fpDbgSymTableContext,
|
||||
fpDbgSymTable, FpDbgAvrClasses, FpDbgDisasAvr, FpDbgRsp, FpDbgCommon,
|
||||
FpImgReaderWinPETypes, FpDbgHardcodedFreepascalInfo, FpDbgCallContextInfo,
|
||||
FpWatchResultData, FpDbgDwarfCFI, FpDbgCpuX86, FpDbgRspClasses,
|
||||
FpDbgDisasXtensa, FpDbgXtensaClasses;
|
||||
FpWatchResultData, FpDbgDwarfCFI, FpDbgCpuX86, FpDbgRspClasses,
|
||||
FpDbgDisasRiscv, FpDbgRiscvClasses, FpDbgDisasXtensa, FpDbgXtensaClasses;
|
||||
|
||||
implementation
|
||||
|
||||
|
@ -142,6 +142,7 @@ begin
|
||||
EM_X86_64: result := mtX86_64;
|
||||
EM_AVR: result := mtAVR8;
|
||||
EM_XTENSA: result := mtXTENSA;
|
||||
EM_RISCV: result := mtRISCV;
|
||||
EM_ALPHA: result := mtALPHA;
|
||||
else
|
||||
result := mtNone;
|
||||
@ -150,7 +151,7 @@ begin
|
||||
// If OS is not encoded in header, take some guess based on machine type
|
||||
if FTargetInfo.OS = osNone then
|
||||
begin
|
||||
if result in [mtAVR8, mtXTENSA] then
|
||||
if result in [mtAVR8, mtXTENSA, mtRISCV] then
|
||||
FTargetInfo.OS := osEmbedded
|
||||
else
|
||||
// Default to the same as host...
|
||||
|
@ -155,6 +155,7 @@ const
|
||||
EM_X86_64 = 62;
|
||||
EM_AVR = 83;
|
||||
EM_XTENSA = 94;
|
||||
EM_RISCV = 243;
|
||||
EM_ALPHA = $9026; //unofficial, but used by gnu toolchain
|
||||
|
||||
//elf version {Elf32_Hdr.e_version}
|
||||
|
@ -16,7 +16,7 @@ uses
|
||||
FpPascalParser, FPDbgController, FpDbgDwarfDataClasses, FpDbgDwarfFreePascal,
|
||||
FpDbgDwarf, FpDbgUtil,
|
||||
DebuggerPropertiesBase,
|
||||
FpDbgRsp, FpDbgAvrClasses, FpDbgXtensaClasses;
|
||||
FpDbgRsp, FpDbgRspClasses, FpDbgAvrClasses, FpDbgXtensaClasses, FpDbgRiscvClasses;
|
||||
|
||||
type
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user