Skip to content

Commit

Permalink
[LoongArch64] add crossgen2 for LoongArch64. (#72017)
Browse files Browse the repository at this point in the history
* [LoongArch64] add crossgen2 for LoongArch64.

* add `aot/ILCompiler.Compiler/Compiler/DependencyAnalysis/Target_LoongArch64`

* amend the code format.

* delete unused comments.
  • Loading branch information
shushanhf authored Jul 15, 2022
1 parent 6677466 commit 0937f26
Show file tree
Hide file tree
Showing 36 changed files with 1,580 additions and 26 deletions.
2 changes: 1 addition & 1 deletion src/coreclr/jit/emitloongarch64.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3213,7 +3213,7 @@ size_t emitter::emitOutputInstr(insGroup* ig, instrDesc* id, BYTE** dp)

dstRW += 4;

emitRecordRelocation(dstRW2, id->idAddr()->iiaAddr, IMAGE_REL_LOONGARCH64_PC);
emitRecordRelocation(dstRW2 - writeableOffset, id->idAddr()->iiaAddr, IMAGE_REL_LOONGARCH64_PC);

dstRW2 += 4;

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,13 @@ public override ObjectData GetData(NodeFactory factory, bool relocsOnly)
arm64Emitter.Builder.AddSymbol(this);
return arm64Emitter.Builder.ToObjectData();

case TargetArchitecture.LoongArch64:
LoongArch64.LoongArch64Emitter loongarch64Emitter = new LoongArch64.LoongArch64Emitter(factory, relocsOnly);
EmitCode(factory, ref loongarch64Emitter, relocsOnly);
loongarch64Emitter.Builder.RequireInitialAlignment(alignment);
loongarch64Emitter.Builder.AddSymbol(this);
return loongarch64Emitter.Builder.ToObjectData();

default:
throw new NotImplementedException();
}
Expand All @@ -77,5 +84,6 @@ public override ObjectData GetData(NodeFactory factory, bool relocsOnly)
protected abstract void EmitCode(NodeFactory factory, ref X86.X86Emitter instructionEncoder, bool relocsOnly);
protected abstract void EmitCode(NodeFactory factory, ref ARM.ARMEmitter instructionEncoder, bool relocsOnly);
protected abstract void EmitCode(NodeFactory factory, ref ARM64.ARM64Emitter instructionEncoder, bool relocsOnly);
protected abstract void EmitCode(NodeFactory factory, ref LoongArch64.LoongArch64Emitter instructionEncoder, bool relocsOnly);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -303,6 +303,8 @@ public void EmitReloc(ISymbolNode symbol, RelocType relocType, int delta = 0)
case RelocType.IMAGE_REL_BASED_ARM64_PAGEBASE_REL21:
case RelocType.IMAGE_REL_BASED_ARM64_PAGEOFFSET_12L:
case RelocType.IMAGE_REL_BASED_ARM64_PAGEOFFSET_12A:
case RelocType.IMAGE_REL_BASED_LOONGARCH64_PC:
case RelocType.IMAGE_REL_BASED_LOONGARCH64_JIR:
Debug.Assert(delta == 0);
// Do not vacate space for this kind of relocation, because
// the space is embedded in the instruction.
Expand Down
102 changes: 102 additions & 0 deletions src/coreclr/tools/Common/Compiler/DependencyAnalysis/Relocation.cs
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,8 @@ public enum RelocType
IMAGE_REL_BASED_THUMB_BRANCH24 = 0x13, // Thumb2: based B, BL
IMAGE_REL_BASED_THUMB_MOV32_PCREL = 0x14, // Thumb2: based MOVW/MOVT
IMAGE_REL_BASED_ARM64_BRANCH26 = 0x15, // Arm64: B, BL
IMAGE_REL_BASED_LOONGARCH64_PC = 0x16, // LoongArch64: pcaddu12i+imm12
IMAGE_REL_BASED_LOONGARCH64_JIR = 0x17, // LoongArch64: pcaddu18i+jirl
IMAGE_REL_BASED_RELPTR32 = 0x7C, // 32-bit relative address from byte starting reloc
// This is a special NGEN-specific relocation type
// for relative pointer (used to make NGen relocation
Expand Down Expand Up @@ -294,7 +296,97 @@ private static unsafe void PutArm64Rel28(uint* pCode, long imm28)
Debug.Assert(GetArm64Rel28(pCode) == imm28);
}

private static unsafe int GetLoongArch64PC12(uint* pCode)
{
uint pcInstr = *pCode;

// first get the hight 20 bits,
int imm = (int)(((pcInstr >> 5) & 0xFFFFF) << 12);

// then get the low 12 bits,
pcInstr = *(pCode + 1);
imm += ((int)(((pcInstr >> 10) & 0xFFF) << 20)) >> 20;

return imm;
}

// case:EA_HANDLE_CNS_RELOC
// pcaddu12i reg, off-hi-20bits
// addi_d reg, reg, off-lo-12bits
// case:EA_PTR_DSP_RELOC
// pcaddu12i reg, off-hi-20bits
// ld_d reg, reg, off-lo-12bits
private static unsafe void PutLoongArch64PC12(uint* pCode, long imm32)
{
// Verify that we got a valid offset
Debug.Assert((int)imm32 == imm32);

uint pcInstr = *pCode;

Debug.Assert((pcInstr & 0xFE000000) == 0x1c000000); // Must be pcaddu12i

int relOff = (int)imm32 & 0x800;
int imm = (int)imm32 + relOff;
relOff = ((imm & 0x7ff) - relOff) & 0xfff;

// Assemble the pc-relative hight20bits of 'imm32' into the pcaddu12i instruction
pcInstr |= (uint)(((imm >> 12) & 0xFFFFF) << 5);

*pCode = pcInstr; // write the assembled instruction

pcInstr = *(pCode + 1);

// Assemble the pc-relative low12bits of 'imm32' into the addid or ld instruction
pcInstr |= (uint)(relOff << 10);

*(pCode + 1) = pcInstr; // write the assembled instruction

Debug.Assert(GetLoongArch64PC12(pCode) == imm32);
}

private static unsafe long GetLoongArch64JIR(uint* pCode)
{
uint pcInstr = *pCode;

// first get the high 20 bits,
long imm = ((long)((pcInstr >> 5) & 0xFFFFF) << 18);

// then get the low 18 bits
pcInstr = *(pCode + 1);
imm += ((long)((short)((pcInstr >> 10) & 0xFFFF))) << 2;

return imm;
}

private static unsafe void PutLoongArch64JIR(uint* pCode, long imm38)
{
// Verify that we got a valid offset
Debug.Assert((imm38 >= -0x2000000000L) && (imm38 < 0x2000000000L));

Debug.Assert((imm38 & 0x3) == 0); // the low two bits must be zero

uint pcInstr = *pCode;

Debug.Assert(pcInstr == 0x1e00000e); // Must be pcaddu18i R14, 0

long relOff = imm38 & 0x20000;
long imm = imm38 + relOff;
relOff = (((imm & 0x1ffff) - relOff) >> 2) & 0xffff;

// Assemble the pc-relative hight20bits of 'imm38' into the pcaddu12i instruction
pcInstr |= (uint)(((imm >> 18) & 0xFFFFF) << 5);

*pCode = pcInstr; // write the assembled instruction

pcInstr = *(pCode + 1);

// Assemble the pc-relative low18bits of 'imm38' into the addid or ld instruction
pcInstr |= (uint)(relOff << 10);

*(pCode + 1) = pcInstr; // write the assembled instruction

Debug.Assert(GetLoongArch64JIR(pCode) == imm38);
}

public Relocation(RelocType relocType, int offset, ISymbolNode target)
{
Expand Down Expand Up @@ -334,6 +426,12 @@ public static unsafe void WriteValue(RelocType relocType, void* location, long v
case RelocType.IMAGE_REL_BASED_ARM64_PAGEOFFSET_12A:
PutArm64Rel12((uint*)location, (int)value);
break;
case RelocType.IMAGE_REL_BASED_LOONGARCH64_PC:
PutLoongArch64PC12((uint*)location, value);
break;
case RelocType.IMAGE_REL_BASED_LOONGARCH64_JIR:
PutLoongArch64JIR((uint*)location, value);
break;
default:
Debug.Fail("Invalid RelocType: " + relocType);
break;
Expand Down Expand Up @@ -366,6 +464,10 @@ public static unsafe long ReadValue(RelocType relocType, void* location)
return GetArm64Rel21((uint*)location);
case RelocType.IMAGE_REL_BASED_ARM64_PAGEOFFSET_12A:
return GetArm64Rel12((uint*)location);
case RelocType.IMAGE_REL_BASED_LOONGARCH64_PC:
return (long)GetLoongArch64PC12((uint*)location);
case RelocType.IMAGE_REL_BASED_LOONGARCH64_JIR:
return (long)GetLoongArch64JIR((uint*)location);
default:
Debug.Fail("Invalid RelocType: " + relocType);
return 0;
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.

using System;

namespace ILCompiler.DependencyAnalysis.LoongArch64
{
public enum AddrModeSize
{
Int8 = 1,
Int16 = 2,
Int32 = 4,
Int64 = 8,
Int128 = 16
}

public struct AddrMode
{
public readonly Register BaseReg;
public readonly Register? IndexReg;
public readonly int Offset;
public readonly byte Scale;
public readonly AddrModeSize Size;

public AddrMode(Register baseRegister, Register? indexRegister, int offset, byte scale, AddrModeSize size)
{
BaseReg = baseRegister;
IndexReg = indexRegister;
Offset = offset;
Scale = scale;
Size = size;
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,134 @@
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.

using System;
using System.Diagnostics;

namespace ILCompiler.DependencyAnalysis.LoongArch64
{
public struct LoongArch64Emitter
{
public LoongArch64Emitter(NodeFactory factory, bool relocsOnly)
{
Builder = new ObjectDataBuilder(factory, relocsOnly);
TargetRegister = new TargetRegisterMap(factory.Target.OperatingSystem);
}

public ObjectDataBuilder Builder;
public TargetRegisterMap TargetRegister;

// Assembly stub creation api. TBD, actually make this general purpose

public void EmitBreak()
{
Builder.EmitUInt(0x002a0005);
}

public void EmitMOV(Register regDst, ushort imm16)
{
Debug.Assert((uint)regDst <= 0x1f);
Debug.Assert(imm16 <= 0xfff);
uint instruction = 0x03800000u | (uint)((imm16 & 0xfff) << 10) | (uint)regDst;
Builder.EmitUInt(instruction);
}

public void EmitMOV(Register regDst, Register regSrc)
{
Builder.EmitUInt((uint)(0x03800000 | ((uint)regSrc << 5) | (uint)regDst));
}

public void EmitMOV(Register regDst, ISymbolNode symbol)
{
Builder.EmitReloc(symbol, RelocType.IMAGE_REL_BASED_LOONGARCH64_PC);
// pcaddu12i reg, off-hi-20bits
Builder.EmitUInt(0x1c000000u | (uint)regDst);

// addi_d reg, reg, off-lo-12bits
Builder.EmitUInt(0x02c00000u | (uint)(((uint)regDst << 5) | (uint)regDst));
}

// pcaddi regDst, 0
public void EmitPC(Register regDst)
{
Debug.Assert((uint)regDst > 0 && (uint)regDst < 32);
Builder.EmitUInt(0x18000000 | (uint)regDst);
}

// addi.d regDst, regSrc, imm12
public void EmitADD(Register regDst, Register regSrc, int imm)
{
Debug.Assert((imm >= -2048) && (imm <= 2047));

Builder.EmitUInt((uint)(0x02c00000 | (uint)((imm & 0xfff) << 10) | ((uint)regSrc << 5) | (uint)regDst));
}

// xori regDst, regSrc, imm12
public void EmitXOR(Register regDst, Register regSrc, int imm)
{
Debug.Assert((imm >= 0) && (imm <= 0xfff));

Builder.EmitUInt((uint)(0x03c00000 | (uint)((imm & 0xfff) << 10) | ((uint)regSrc << 5) | (uint)regDst));
}

// ld_d regDst, regAddr, offset
public void EmitLD(Register regDst, Register regSrc, int offset)
{
Debug.Assert((offset >= -2048) && (offset <= 2047));

Builder.EmitUInt((uint)(0x28c00000 | (uint)((offset & 0xfff) << 10) | ((uint)regSrc << 5) | (uint)regDst));
}

public void EmitRET()
{
// jirl R0,R1,0
Builder.EmitUInt(0x4c000020);
}

public void EmitJMP(Register reg)
{
Builder.EmitUInt(0x4c000000u | ((uint)reg << 5));
}

public void EmitJMP(ISymbolNode symbol)
{
if (symbol.RepresentsIndirectionCell)
{
// pcaddi R21, 0
EmitPC(Register.R21);

EmitLD(Register.R21, Register.R21, 0x10);

// ld_d R21, R21, 0
EmitLD(Register.R21, Register.R21, 0);

// jirl R0,R21,0
Builder.EmitUInt(0x4c0002a0);

Builder.EmitReloc(symbol, RelocType.IMAGE_REL_BASED_DIR64);
}
else
{
//Builder.EmitReloc(symbol, RelocType.IMAGE_REL_BASED_LOONGARCH64_PC);
Builder.EmitUInt(0xffffffff); // bad code.
throw new NotImplementedException();
}
}

public void EmitRETIfEqual(Register regSrc)
{
// BNEZ regSrc, 8
Builder.EmitUInt((uint)(0x44000000 | (2 << 10) | ((uint)regSrc << 5)));
EmitRET();
}

public void EmitJE(Register regSrc, ISymbolNode symbol)
{
uint offset = symbol.RepresentsIndirectionCell ? 7u : 2u;

// BNEZ regSrc, offset
Builder.EmitUInt((uint)(0x44000000 | (offset << 10) | ((uint)regSrc << 5)));

EmitJMP(symbol);
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace ILCompiler.DependencyAnalysis.LoongArch64
{
public enum Register
{
R0 = 0,
R1 = 1,
R2 = 2,
R3 = 3,
R4 = 4,
R5 = 5,
R6 = 6,
R7 = 7,
R8 = 8,
R9 = 9,
R10 = 10,
R11 = 11,
R12 = 12,
R13 = 13,
R14 = 14,
R15 = 15,
R16 = 16,
R17 = 17,
R18 = 18,
R19 = 19,
R20 = 20,
R21 = 21,
R22 = 22,
R23 = 23,
R24 = 24,
R25 = 25,
R26 = 26,
R27 = 27,
R28 = 28,
R29 = 29,
R30 = 30,
R31 = 31,

None = 32,
NoIndex = 128,
}
}
Loading

0 comments on commit 0937f26

Please sign in to comment.