diff --git a/llvm/lib/Target/PowerPC/AsmParser/PPCAsmParser.cpp b/llvm/lib/Target/PowerPC/AsmParser/PPCAsmParser.cpp --- a/llvm/lib/Target/PowerPC/AsmParser/PPCAsmParser.cpp +++ b/llvm/lib/Target/PowerPC/AsmParser/PPCAsmParser.cpp @@ -282,6 +282,11 @@ return (unsigned) Imm.Val; } + unsigned getFpReg() const { + assert(isEvenRegNumber() && "Invalid access!"); + return (unsigned)(Imm.Val >> 1); + } + unsigned getVSReg() const { assert(isVSRegNumber() && "Invalid access!"); return (unsigned) Imm.Val; @@ -502,6 +507,11 @@ Inst.addOperand(MCOperand::createReg(FRegs[getReg()])); } + void addRegFpRCOperands(MCInst &Inst, unsigned N) const { + assert(N == 1 && "Invalid number of operands!"); + Inst.addOperand(MCOperand::createReg(FpRegs[getFpReg()])); + } + void addRegVFRCOperands(MCInst &Inst, unsigned N) const { assert(N == 1 && "Invalid number of operands!"); Inst.addOperand(MCOperand::createReg(VFRegs[getReg()])); diff --git a/llvm/lib/Target/PowerPC/Disassembler/PPCDisassembler.cpp b/llvm/lib/Target/PowerPC/Disassembler/PPCDisassembler.cpp --- a/llvm/lib/Target/PowerPC/Disassembler/PPCDisassembler.cpp +++ b/llvm/lib/Target/PowerPC/Disassembler/PPCDisassembler.cpp @@ -112,6 +112,14 @@ return decodeRegisterClass(Inst, RegNo, FRegs); } +static DecodeStatus DecodeFpRCRegisterClass(MCInst &Inst, uint64_t RegNo, + uint64_t Address, + const MCDisassembler *Decoder) { + assert(RegNo <= 30 && "Expecting a register number no more than 30."); + assert((RegNo & 1) == 0 && "Expecting an even register number."); + return decodeRegisterClass(Inst, RegNo >> 1, FpRegs); +} + static DecodeStatus DecodeVFRCRegisterClass(MCInst &Inst, uint64_t RegNo, uint64_t Address, const MCDisassembler *Decoder) { diff --git a/llvm/lib/Target/PowerPC/MCTargetDesc/PPCMCTargetDesc.h b/llvm/lib/Target/PowerPC/MCTargetDesc/PPCMCTargetDesc.h --- a/llvm/lib/Target/PowerPC/MCTargetDesc/PPCMCTargetDesc.h +++ b/llvm/lib/Target/PowerPC/MCTargetDesc/PPCMCTargetDesc.h @@ -136,6 +136,12 @@ X##22, X##23, X##24, X##25, X##26, X##27, X##28, X##29, X##30, X##31 \ } +#define PPC_REGS_EVEN0_30(X) \ + { \ + X##0, X##2, X##4, X##6, X##8, X##10, X##12, X##14, X##16, X##18, X##20, \ + X##22, X##24, X##26, X##28, X##30 \ + } + #define PPC_REGS0_63(X) \ { \ X##0, X##1, X##2, X##3, X##4, X##5, X##6, X##7, X##8, X##9, X##10, X##11, \ @@ -182,6 +188,7 @@ static const MCPhysReg RRegs[32] = PPC_REGS0_31(PPC::R); \ static const MCPhysReg XRegs[32] = PPC_REGS0_31(PPC::X); \ static const MCPhysReg FRegs[32] = PPC_REGS0_31(PPC::F); \ + static const MCPhysReg FpRegs[16] = PPC_REGS_EVEN0_30(PPC::Fp); \ static const MCPhysReg VSRpRegs[32] = PPC_REGS0_31(PPC::VSRp); \ static const MCPhysReg SPERegs[32] = PPC_REGS0_31(PPC::S); \ static const MCPhysReg VFRegs[32] = PPC_REGS0_31(PPC::VF); \ diff --git a/llvm/lib/Target/PowerPC/PPCInstrDFP.td b/llvm/lib/Target/PowerPC/PPCInstrDFP.td new file mode 100644 --- /dev/null +++ b/llvm/lib/Target/PowerPC/PPCInstrDFP.td @@ -0,0 +1,28 @@ +//===-- PPCInstrDFP.td - PowerPC Decimal Floating Point ----*- tablegen -*-===// +// +// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. +// See https://llvm.org/LICENSE.txt for license information. +// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception +// +//===----------------------------------------------------------------------===// +// +// This file describes the PowerPC Decimal Floating Point (DFP) instructions. +// +//===----------------------------------------------------------------------===// + +// We provide no sceduling info for the DFP instructions. +// While they are not pseudo instructions we don't intend on scheduling them. +let hasNoSchedulingInfo = 1 in { +defm DADD : XForm_28r<59, 2, (outs f8rc:$RST), (ins f8rc:$RA, f8rc:$RB), + "dadd", "$RST, $RA, $RB", IIC_FPGeneral, []>; + +defm DADDQ : XForm_28r<63, 2, (outs fpairrc:$RST), (ins fpairrc:$RA, fpairrc:$RB), + "daddq", "$RST, $RA, $RB", IIC_FPGeneral, []>; + +defm DSUB : XForm_28r<59, 514, (outs f8rc:$RST), (ins f8rc:$RA, f8rc:$RB), + "dsub", "$RST, $RA, $RB", IIC_FPGeneral, []>; + +defm DSUBQ : XForm_28r<63, 514, (outs fpairrc:$RST), (ins fpairrc:$RA, fpairrc:$RB), + "dsubq", "$RST, $RA, $RB", IIC_FPGeneral, []>; +} + diff --git a/llvm/lib/Target/PowerPC/PPCInstrInfo.td b/llvm/lib/Target/PowerPC/PPCInstrInfo.td --- a/llvm/lib/Target/PowerPC/PPCInstrInfo.td +++ b/llvm/lib/Target/PowerPC/PPCInstrInfo.td @@ -3321,6 +3321,7 @@ include "PPCInstrFutureMMA.td" include "PPCInstrFuture.td" include "PPCInstrMMA.td" +include "PPCInstrDFP.td" // Patterns for arithmetic i1 operations. def : Pat<(add i1:$a, i1:$b), diff --git a/llvm/lib/Target/PowerPC/PPCRegisterInfo.h b/llvm/lib/Target/PowerPC/PPCRegisterInfo.h --- a/llvm/lib/Target/PowerPC/PPCRegisterInfo.h +++ b/llvm/lib/Target/PowerPC/PPCRegisterInfo.h @@ -179,9 +179,12 @@ case 'a': if (RegName[1] == 'c' && RegName[2] == 'c') return RegName + 3; - break; - case 'r': + break; case 'f': + if (RegName[1] == 'p') + return RegName + 2; + LLVM_FALLTHROUGH; + case 'r': case 'v': if (RegName[1] == 's') { if (RegName[2] == 'p') diff --git a/llvm/lib/Target/PowerPC/PPCRegisterInfo.td b/llvm/lib/Target/PowerPC/PPCRegisterInfo.td --- a/llvm/lib/Target/PowerPC/PPCRegisterInfo.td +++ b/llvm/lib/Target/PowerPC/PPCRegisterInfo.td @@ -20,6 +20,8 @@ def sub_vsx1 : SubRegIndex<128, 128>; def sub_gp8_x0 : SubRegIndex<64>; def sub_gp8_x1 : SubRegIndex<64, 64>; +def sub_fp0 : SubRegIndex<64>; +def sub_fp1 : SubRegIndex<64, 64>; } @@ -58,6 +60,15 @@ let HWEncoding{4-0} = num; } +// FPPair - A pair of 64-bit floating-point registers. +class FPPair EvenIndex> : PPCReg { + assert !eq(EvenIndex{0}, 0), "Index should be even."; + let HWEncoding{4-0} = EvenIndex; + let SubRegs = [!cast("F"#EvenIndex), !cast("F"#!add(EvenIndex, 1))]; + let DwarfNumbers = [-1, -1]; + let SubRegIndices = [sub_fp0, sub_fp1]; +} + // VF - One of the 32 64-bit floating-point subregisters of the vector // registers (used by VSX). class VF num, string n> : PPCReg { @@ -136,6 +147,11 @@ DwarfRegNum<[!add(Index, 32), !add(Index, 32)]>; } +// Floating-point pair registers +foreach Index = { 0, 2, 4, 6, 8, 10, 12, 14, 16, 18, 20, 22, 24, 26, 28, 30 } in { + def Fp#Index : FPPair<"fp"#Index, Index>; +} + // 64-bit Floating-point subregisters of Altivec registers // Note: the register names are v0-v31 or vs32-vs63 depending on the use. // Custom C++ code is used to produce the correct name and encoding. @@ -349,6 +365,20 @@ (sequence "F%u", 31, 14))>; def F4RC : RegisterClass<"PPC", [f32], 32, (add F8RC)>; +// Floating point pair registers. +// Note that the type used for this register class is ppcf128. This is not +// completely correct. However, since we are not pattern matching any +// instrctions for these registers and we are not register allocating or +// scheduling any of these instructions it should be safe to do this. +// The reason we didn't use the correct type (Decimal Floating Point) is that +// at the time of this implementation the correct type is not available. +def FpRC : + RegisterClass<"PPC", [ppcf128], 128, + (add Fp0, Fp2, Fp4, Fp6, Fp8, Fp10, Fp12, Fp14, Fp16, Fp18, + Fp20, Fp22, Fp24, Fp26, Fp28, Fp30)> { + let Size = 128; +} + def VRRC : RegisterClass<"PPC", [v16i8,v8i16,v4i32,v2i64,v1i128,v4f32,v2f64, f128], 128, @@ -504,6 +534,12 @@ def f4rc : RegisterOperand { let ParserMatchClass = PPCRegF4RCAsmOperand; } +def PPCRegFpRCAsmOperand : AsmOperandClass { + let Name = "RegFpRC"; let PredicateMethod = "isEvenRegNumber"; +} +def fpairrc : RegisterOperand { + let ParserMatchClass = PPCRegFpRCAsmOperand; +} def PPCRegVRRCAsmOperand : AsmOperandClass { let Name = "RegVRRC"; let PredicateMethod = "isRegNumber"; } diff --git a/llvm/test/MC/Disassembler/PowerPC/ppc64-encoding-dfp.txt b/llvm/test/MC/Disassembler/PowerPC/ppc64-encoding-dfp.txt new file mode 100644 --- /dev/null +++ b/llvm/test/MC/Disassembler/PowerPC/ppc64-encoding-dfp.txt @@ -0,0 +1,25 @@ +# RUN: llvm-mc --disassemble %s -triple powerpc64-unknown-unknown -mcpu=pwr7 | FileCheck %s + +# CHECK: dadd 2, 3, 4 +0xec 0x43 0x20 0x04 + +# CHECK: dadd. 2, 3, 4 +0xec 0x43 0x20 0x05 + +# CHECK: daddq 2, 6, 4 +0xfc 0x46 0x20 0x04 + +# CHECK: daddq. 2, 6, 4 +0xfc 0x46 0x20 0x05 + +# CHECK: dsub 2, 3, 4 +0xec 0x43 0x24 0x04 + +# CHECK: dsub. 2, 3, 4 +0xec 0x43 0x24 0x05 + +# CHECK: dsubq 2, 6, 4 +0xfc 0x46 0x24 0x04 + +# CHECK: dsubq. 2, 6, 4 +0xfc 0x46 0x24 0x05 diff --git a/llvm/test/MC/PowerPC/ppc64-encoding-dfp.s b/llvm/test/MC/PowerPC/ppc64-encoding-dfp.s new file mode 100644 --- /dev/null +++ b/llvm/test/MC/PowerPC/ppc64-encoding-dfp.s @@ -0,0 +1,29 @@ + +# RUN: llvm-mc -triple powerpc64-unknown-unknown --show-encoding %s | FileCheck -check-prefix=CHECK-BE %s +# RUN: llvm-mc -triple powerpc64le-unknown-unknown --show-encoding %s | FileCheck -check-prefix=CHECK-LE %s + + +# CHECK-BE: dadd 2, 3, 4 # encoding: [0xec,0x43,0x20,0x04] +# CHECK-LE: dadd 2, 3, 4 # encoding: [0x04,0x20,0x43,0xec] + dadd 2, 3, 4 +# CHECK-BE: dadd. 2, 3, 4 # encoding: [0xec,0x43,0x20,0x05] +# CHECK-LE: dadd. 2, 3, 4 # encoding: [0x05,0x20,0x43,0xec] + dadd. 2, 3, 4 +# CHECK-BE: daddq 2, 6, 4 # encoding: [0xfc,0x46,0x20,0x04] +# CHECK-LE: daddq 2, 6, 4 # encoding: [0x04,0x20,0x46,0xfc] + daddq 2, 6, 4 +# CHECK-BE: daddq. 2, 6, 4 # encoding: [0xfc,0x46,0x20,0x05] +# CHECK-LE: daddq. 2, 6, 4 # encoding: [0x05,0x20,0x46,0xfc] + daddq. 2, 6, 4 +# CHECK-BE: dsub 2, 3, 4 # encoding: [0xec,0x43,0x24,0x04] +# CHECK-LE: dsub 2, 3, 4 # encoding: [0x04,0x24,0x43,0xec] + dsub 2, 3, 4 +# CHECK-BE: dsub. 2, 3, 4 # encoding: [0xec,0x43,0x24,0x05] +# CHECK-LE: dsub. 2, 3, 4 # encoding: [0x05,0x24,0x43,0xec] + dsub. 2, 3, 4 +# CHECK-BE: dsubq 2, 6, 4 # encoding: [0xfc,0x46,0x24,0x04] +# CHECK-LE: dsubq 2, 6, 4 # encoding: [0x04,0x24,0x46,0xfc] + dsubq 2, 6, 4 +# CHECK-BE: dsubq. 2, 6, 4 # encoding: [0xfc,0x46,0x24,0x05] +# CHECK-LE: dsubq. 2, 6, 4 # encoding: [0x05,0x24,0x46,0xfc] + dsubq. 2, 6, 4