1 //===-- HexagonMCTargetDesc.cpp - Hexagon Target Descriptions -------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 // This file provides Hexagon specific target descriptions.
11 //===----------------------------------------------------------------------===//
13 #include "MCTargetDesc/HexagonMCTargetDesc.h"
14 #include "HexagonDepArch.h"
15 #include "HexagonTargetStreamer.h"
16 #include "MCTargetDesc/HexagonInstPrinter.h"
17 #include "MCTargetDesc/HexagonMCAsmInfo.h"
18 #include "MCTargetDesc/HexagonMCELFStreamer.h"
19 #include "MCTargetDesc/HexagonMCInstrInfo.h"
20 #include "TargetInfo/HexagonTargetInfo.h"
21 #include "llvm/ADT/StringExtras.h"
22 #include "llvm/ADT/StringRef.h"
23 #include "llvm/BinaryFormat/ELF.h"
24 #include "llvm/MC/MCAsmBackend.h"
25 #include "llvm/MC/MCCodeEmitter.h"
26 #include "llvm/MC/MCContext.h"
27 #include "llvm/MC/MCDwarf.h"
28 #include "llvm/MC/MCELFStreamer.h"
29 #include "llvm/MC/MCInstrAnalysis.h"
30 #include "llvm/MC/MCInstrInfo.h"
31 #include "llvm/MC/MCObjectWriter.h"
32 #include "llvm/MC/MCRegisterInfo.h"
33 #include "llvm/MC/MCStreamer.h"
34 #include "llvm/MC/MCSubtargetInfo.h"
35 #include "llvm/Support/ErrorHandling.h"
36 #include "llvm/Support/TargetRegistry.h"
37 #include "llvm/Support/raw_ostream.h"
45 #define GET_INSTRINFO_MC_DESC
46 #include "HexagonGenInstrInfo.inc"
48 #define GET_SUBTARGETINFO_MC_DESC
49 #include "HexagonGenSubtargetInfo.inc"
51 #define GET_REGINFO_MC_DESC
52 #include "HexagonGenRegisterInfo.inc"
54 cl::opt
<bool> llvm::HexagonDisableCompound
56 cl::desc("Disable looking for compound instructions for Hexagon"));
58 cl::opt
<bool> llvm::HexagonDisableDuplex
60 cl::desc("Disable looking for duplex instructions for Hexagon"));
62 namespace { // These flags are to be deprecated
63 cl::opt
<bool> MV5("mv5", cl::Hidden
, cl::desc("Build for Hexagon V5"),
65 cl::opt
<bool> MV55("mv55", cl::Hidden
, cl::desc("Build for Hexagon V55"),
67 cl::opt
<bool> MV60("mv60", cl::Hidden
, cl::desc("Build for Hexagon V60"),
69 cl::opt
<bool> MV62("mv62", cl::Hidden
, cl::desc("Build for Hexagon V62"),
71 cl::opt
<bool> MV65("mv65", cl::Hidden
, cl::desc("Build for Hexagon V65"),
73 cl::opt
<bool> MV66("mv66", cl::Hidden
, cl::desc("Build for Hexagon V66"),
76 cl::opt
<Hexagon::ArchEnum
>
78 cl::desc("Enable Hexagon Vector eXtensions"),
80 clEnumValN(Hexagon::ArchEnum::V60
, "v60", "Build for HVX v60"),
81 clEnumValN(Hexagon::ArchEnum::V62
, "v62", "Build for HVX v62"),
82 clEnumValN(Hexagon::ArchEnum::V65
, "v65", "Build for HVX v65"),
83 clEnumValN(Hexagon::ArchEnum::V66
, "v66", "Build for HVX v66"),
84 // Sentinel for no value specified.
85 clEnumValN(Hexagon::ArchEnum::Generic
, "", "")),
86 // Sentinel for flag not present.
87 cl::init(Hexagon::ArchEnum::NoArch
), cl::ValueOptional
);
91 DisableHVX("mno-hvx", cl::Hidden
,
92 cl::desc("Disable Hexagon Vector eXtensions"));
95 static StringRef DefaultArch
= "hexagonv60";
97 static StringRef
HexagonGetArchVariant() {
113 StringRef
Hexagon_MC::selectHexagonCPU(StringRef CPU
) {
114 StringRef ArchV
= HexagonGetArchVariant();
115 if (!ArchV
.empty() && !CPU
.empty()) {
117 report_fatal_error("conflicting architectures specified.");
128 unsigned llvm::HexagonGetLastSlot() { return HexagonItinerariesV5FU::SLOT3
; }
132 class HexagonTargetAsmStreamer
: public HexagonTargetStreamer
{
134 HexagonTargetAsmStreamer(MCStreamer
&S
,
135 formatted_raw_ostream
&OS
,
138 : HexagonTargetStreamer(S
) {}
140 void prettyPrintAsm(MCInstPrinter
&InstPrinter
, raw_ostream
&OS
,
141 const MCInst
&Inst
, const MCSubtargetInfo
&STI
) override
{
142 assert(HexagonMCInstrInfo::isBundle(Inst
));
143 assert(HexagonMCInstrInfo::bundleSize(Inst
) <= HEXAGON_PACKET_SIZE
);
146 raw_string_ostream
TempStream(Buffer
);
147 InstPrinter
.printInst(&Inst
, TempStream
, "", STI
);
149 StringRef
Contents(Buffer
);
150 auto PacketBundle
= Contents
.rsplit('\n');
151 auto HeadTail
= PacketBundle
.first
.split('\n');
152 StringRef Separator
= "\n";
153 StringRef Indent
= "\t";
155 while (!HeadTail
.first
.empty()) {
157 auto Duplex
= HeadTail
.first
.split('\v');
158 if (!Duplex
.second
.empty()) {
159 OS
<< Indent
<< Duplex
.first
<< Separator
;
160 InstTxt
= Duplex
.second
;
161 } else if (!HeadTail
.first
.trim().startswith("immext")) {
162 InstTxt
= Duplex
.first
;
164 if (!InstTxt
.empty())
165 OS
<< Indent
<< InstTxt
<< Separator
;
166 HeadTail
= HeadTail
.second
.split('\n');
169 if (HexagonMCInstrInfo::isMemReorderDisabled(Inst
))
170 OS
<< "\n\t} :mem_noshuf" << PacketBundle
.second
;
172 OS
<< "\t}" << PacketBundle
.second
;
176 class HexagonTargetELFStreamer
: public HexagonTargetStreamer
{
178 MCELFStreamer
&getStreamer() {
179 return static_cast<MCELFStreamer
&>(Streamer
);
181 HexagonTargetELFStreamer(MCStreamer
&S
, MCSubtargetInfo
const &STI
)
182 : HexagonTargetStreamer(S
) {
183 MCAssembler
&MCA
= getStreamer().getAssembler();
184 MCA
.setELFHeaderEFlags(Hexagon_MC::GetELFFlags(STI
));
188 void EmitCommonSymbolSorted(MCSymbol
*Symbol
, uint64_t Size
,
189 unsigned ByteAlignment
,
190 unsigned AccessSize
) override
{
191 HexagonMCELFStreamer
&HexagonELFStreamer
=
192 static_cast<HexagonMCELFStreamer
&>(getStreamer());
193 HexagonELFStreamer
.HexagonMCEmitCommonSymbol(Symbol
, Size
, ByteAlignment
,
197 void EmitLocalCommonSymbolSorted(MCSymbol
*Symbol
, uint64_t Size
,
198 unsigned ByteAlignment
,
199 unsigned AccessSize
) override
{
200 HexagonMCELFStreamer
&HexagonELFStreamer
=
201 static_cast<HexagonMCELFStreamer
&>(getStreamer());
202 HexagonELFStreamer
.HexagonMCEmitLocalCommonSymbol(
203 Symbol
, Size
, ByteAlignment
, AccessSize
);
207 } // end anonymous namespace
209 llvm::MCInstrInfo
*llvm::createHexagonMCInstrInfo() {
210 MCInstrInfo
*X
= new MCInstrInfo();
211 InitHexagonMCInstrInfo(X
);
215 static MCRegisterInfo
*createHexagonMCRegisterInfo(const Triple
&TT
) {
216 MCRegisterInfo
*X
= new MCRegisterInfo();
217 InitHexagonMCRegisterInfo(X
, Hexagon::R31
);
221 static MCAsmInfo
*createHexagonMCAsmInfo(const MCRegisterInfo
&MRI
,
223 MCAsmInfo
*MAI
= new HexagonMCAsmInfo(TT
);
225 // VirtualFP = (R30 + #0).
226 MCCFIInstruction Inst
=
227 MCCFIInstruction::createDefCfa(nullptr,
228 MRI
.getDwarfRegNum(Hexagon::R30
, true), 0);
229 MAI
->addInitialFrameState(Inst
);
234 static MCInstPrinter
*createHexagonMCInstPrinter(const Triple
&T
,
235 unsigned SyntaxVariant
,
236 const MCAsmInfo
&MAI
,
237 const MCInstrInfo
&MII
,
238 const MCRegisterInfo
&MRI
)
240 if (SyntaxVariant
== 0)
241 return new HexagonInstPrinter(MAI
, MII
, MRI
);
246 static MCTargetStreamer
*
247 createMCAsmTargetStreamer(MCStreamer
&S
, formatted_raw_ostream
&OS
,
248 MCInstPrinter
*IP
, bool IsVerboseAsm
) {
249 return new HexagonTargetAsmStreamer(S
, OS
, IsVerboseAsm
, *IP
);
252 static MCStreamer
*createMCStreamer(Triple
const &T
, MCContext
&Context
,
253 std::unique_ptr
<MCAsmBackend
> &&MAB
,
254 std::unique_ptr
<MCObjectWriter
> &&OW
,
255 std::unique_ptr
<MCCodeEmitter
> &&Emitter
,
257 return createHexagonELFStreamer(T
, Context
, std::move(MAB
), std::move(OW
),
261 static MCTargetStreamer
*
262 createHexagonObjectTargetStreamer(MCStreamer
&S
, const MCSubtargetInfo
&STI
) {
263 return new HexagonTargetELFStreamer(S
, STI
);
266 static void LLVM_ATTRIBUTE_UNUSED
clearFeature(MCSubtargetInfo
* STI
, uint64_t F
) {
267 uint64_t FB
= STI
->getFeatureBits().to_ullong();
268 if (FB
& (1ULL << F
))
269 STI
->ToggleFeature(F
);
272 static bool LLVM_ATTRIBUTE_UNUSED
checkFeature(MCSubtargetInfo
* STI
, uint64_t F
) {
273 uint64_t FB
= STI
->getFeatureBits().to_ullong();
274 return (FB
& (1ULL << F
)) != 0;
278 std::string
selectHexagonFS(StringRef CPU
, StringRef FS
) {
279 SmallVector
<StringRef
, 3> Result
;
281 Result
.push_back(FS
);
284 case Hexagon::ArchEnum::V5
:
285 case Hexagon::ArchEnum::V55
:
287 case Hexagon::ArchEnum::V60
:
288 Result
.push_back("+hvxv60");
290 case Hexagon::ArchEnum::V62
:
291 Result
.push_back("+hvxv62");
293 case Hexagon::ArchEnum::V65
:
294 Result
.push_back("+hvxv65");
296 case Hexagon::ArchEnum::V66
:
297 Result
.push_back("+hvxv66");
299 case Hexagon::ArchEnum::Generic
:{
300 Result
.push_back(StringSwitch
<StringRef
>(CPU
)
301 .Case("hexagonv60", "+hvxv60")
302 .Case("hexagonv62", "+hvxv62")
303 .Case("hexagonv65", "+hvxv65")
304 .Case("hexagonv66", "+hvxv66"));
307 case Hexagon::ArchEnum::NoArch
:
308 // Sentinal if -mhvx isn't specified
311 return join(Result
.begin(), Result
.end(), ",");
315 static bool isCPUValid(std::string CPU
)
317 std::vector
<std::string
> table
{
318 "generic", "hexagonv5", "hexagonv55", "hexagonv60",
319 "hexagonv62", "hexagonv65", "hexagonv66",
322 return std::find(table
.begin(), table
.end(), CPU
) != table
.end();
326 std::pair
<std::string
, std::string
> selectCPUAndFS(StringRef CPU
,
328 std::pair
<std::string
, std::string
> Result
;
329 Result
.first
= Hexagon_MC::selectHexagonCPU(CPU
);
330 Result
.second
= selectHexagonFS(Result
.first
, FS
);
335 FeatureBitset
Hexagon_MC::completeHVXFeatures(const FeatureBitset
&S
) {
336 using namespace Hexagon
;
337 // Make sure that +hvx-length turns hvx on, and that "hvx" alone
338 // turns on hvxvNN, corresponding to the existing ArchVNN.
339 FeatureBitset FB
= S
;
340 unsigned CpuArch
= ArchV5
;
341 for (unsigned F
: {ArchV66
, ArchV65
, ArchV62
, ArchV60
, ArchV55
, ArchV5
}) {
348 for (unsigned F
: {ExtensionHVX
, ExtensionHVX64B
, ExtensionHVX128B
}) {
354 bool HasHvxVer
= false;
355 for (unsigned F
: {ExtensionHVXV60
, ExtensionHVXV62
, ExtensionHVXV65
,
364 if (!UseHvx
|| HasHvxVer
)
367 // HasHvxVer is false, and UseHvx is true.
370 FB
.set(ExtensionHVXV66
);
373 FB
.set(ExtensionHVXV65
);
376 FB
.set(ExtensionHVXV62
);
379 FB
.set(ExtensionHVXV60
);
385 MCSubtargetInfo
*Hexagon_MC::createHexagonMCSubtargetInfo(const Triple
&TT
,
388 std::pair
<std::string
, std::string
> Features
= selectCPUAndFS(CPU
, FS
);
389 StringRef CPUName
= Features
.first
;
390 StringRef ArchFS
= Features
.second
;
392 if (!isCPUValid(CPUName
.str())) {
393 errs() << "error: invalid CPU \"" << CPUName
.str().c_str()
398 MCSubtargetInfo
*X
= createHexagonMCSubtargetInfoImpl(TT
, CPUName
, ArchFS
);
399 if (HexagonDisableDuplex
) {
400 llvm::FeatureBitset Features
= X
->getFeatureBits();
401 X
->setFeatureBits(Features
.set(Hexagon::FeatureDuplex
, false));
404 X
->setFeatureBits(completeHVXFeatures(X
->getFeatureBits()));
408 unsigned Hexagon_MC::GetELFFlags(const MCSubtargetInfo
&STI
) {
409 static std::map
<StringRef
,unsigned> ElfFlags
= {
410 {"hexagonv5", ELF::EF_HEXAGON_MACH_V5
},
411 {"hexagonv55", ELF::EF_HEXAGON_MACH_V55
},
412 {"hexagonv60", ELF::EF_HEXAGON_MACH_V60
},
413 {"hexagonv62", ELF::EF_HEXAGON_MACH_V62
},
414 {"hexagonv65", ELF::EF_HEXAGON_MACH_V65
},
415 {"hexagonv66", ELF::EF_HEXAGON_MACH_V66
},
418 auto F
= ElfFlags
.find(STI
.getCPU());
419 assert(F
!= ElfFlags
.end() && "Unrecognized Architecture");
424 class HexagonMCInstrAnalysis
: public MCInstrAnalysis
{
426 HexagonMCInstrAnalysis(MCInstrInfo
const *Info
) : MCInstrAnalysis(Info
) {}
428 bool isUnconditionalBranch(MCInst
const &Inst
) const override
{
429 //assert(!HexagonMCInstrInfo::isBundle(Inst));
430 return MCInstrAnalysis::isUnconditionalBranch(Inst
);
433 bool isConditionalBranch(MCInst
const &Inst
) const override
{
434 //assert(!HexagonMCInstrInfo::isBundle(Inst));
435 return MCInstrAnalysis::isConditionalBranch(Inst
);
438 bool evaluateBranch(MCInst
const &Inst
, uint64_t Addr
,
439 uint64_t Size
, uint64_t &Target
) const override
{
440 //assert(!HexagonMCInstrInfo::isBundle(Inst));
441 if(!HexagonMCInstrInfo::isExtendable(*Info
, Inst
))
443 auto const &Extended(HexagonMCInstrInfo::getExtendableOperand(*Info
, Inst
));
444 assert(Extended
.isExpr());
446 if(!Extended
.getExpr()->evaluateAsAbsolute(Value
))
454 static MCInstrAnalysis
*createHexagonMCInstrAnalysis(const MCInstrInfo
*Info
) {
455 return new HexagonMCInstrAnalysis(Info
);
458 // Force static initialization.
459 extern "C" void LLVMInitializeHexagonTargetMC() {
460 // Register the MC asm info.
461 RegisterMCAsmInfoFn
X(getTheHexagonTarget(), createHexagonMCAsmInfo
);
463 // Register the MC instruction info.
464 TargetRegistry::RegisterMCInstrInfo(getTheHexagonTarget(),
465 createHexagonMCInstrInfo
);
467 // Register the MC register info.
468 TargetRegistry::RegisterMCRegInfo(getTheHexagonTarget(),
469 createHexagonMCRegisterInfo
);
471 // Register the MC subtarget info.
472 TargetRegistry::RegisterMCSubtargetInfo(getTheHexagonTarget(),
473 Hexagon_MC::createHexagonMCSubtargetInfo
);
475 // Register the MC Code Emitter
476 TargetRegistry::RegisterMCCodeEmitter(getTheHexagonTarget(),
477 createHexagonMCCodeEmitter
);
479 // Register the asm backend
480 TargetRegistry::RegisterMCAsmBackend(getTheHexagonTarget(),
481 createHexagonAsmBackend
);
484 // Register the MC instruction analyzer.
485 TargetRegistry::RegisterMCInstrAnalysis(getTheHexagonTarget(),
486 createHexagonMCInstrAnalysis
);
488 // Register the obj streamer
489 TargetRegistry::RegisterELFStreamer(getTheHexagonTarget(),
492 // Register the obj target streamer
493 TargetRegistry::RegisterObjectTargetStreamer(getTheHexagonTarget(),
494 createHexagonObjectTargetStreamer
);
496 // Register the asm streamer
497 TargetRegistry::RegisterAsmTargetStreamer(getTheHexagonTarget(),
498 createMCAsmTargetStreamer
);
500 // Register the MC Inst Printer
501 TargetRegistry::RegisterMCInstPrinter(getTheHexagonTarget(),
502 createHexagonMCInstPrinter
);