2015-06-23 01:02:30 +08:00
|
|
|
//===- MIParser.cpp - Machine instructions parser implementation ----------===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// 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
|
2015-06-23 01:02:30 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements the parsing of machine instructions.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2019-03-15 06:54:43 +08:00
|
|
|
#include "llvm/CodeGen/MIRParser/MIParser.h"
|
2017-11-08 09:01:31 +08:00
|
|
|
#include "MILexer.h"
|
2017-06-07 06:22:41 +08:00
|
|
|
#include "llvm/ADT/APInt.h"
|
|
|
|
#include "llvm/ADT/APSInt.h"
|
|
|
|
#include "llvm/ADT/ArrayRef.h"
|
|
|
|
#include "llvm/ADT/DenseMap.h"
|
|
|
|
#include "llvm/ADT/None.h"
|
|
|
|
#include "llvm/ADT/Optional.h"
|
|
|
|
#include "llvm/ADT/SmallVector.h"
|
2015-06-23 01:02:30 +08:00
|
|
|
#include "llvm/ADT/StringMap.h"
|
2017-06-07 06:22:41 +08:00
|
|
|
#include "llvm/ADT/StringRef.h"
|
2017-11-08 09:01:31 +08:00
|
|
|
#include "llvm/ADT/StringSwitch.h"
|
2017-06-07 06:22:41 +08:00
|
|
|
#include "llvm/ADT/Twine.h"
|
2018-08-21 04:37:57 +08:00
|
|
|
#include "llvm/Analysis/MemoryLocation.h"
|
2015-08-01 04:49:21 +08:00
|
|
|
#include "llvm/AsmParser/Parser.h"
|
2015-06-27 06:56:48 +08:00
|
|
|
#include "llvm/AsmParser/SlotMapping.h"
|
2019-03-13 04:42:12 +08:00
|
|
|
#include "llvm/CodeGen/GlobalISel/RegisterBank.h"
|
|
|
|
#include "llvm/CodeGen/GlobalISel/RegisterBankInfo.h"
|
2017-05-06 05:09:30 +08:00
|
|
|
#include "llvm/CodeGen/MIRPrinter.h"
|
2015-06-23 01:02:30 +08:00
|
|
|
#include "llvm/CodeGen/MachineBasicBlock.h"
|
2015-07-17 07:37:45 +08:00
|
|
|
#include "llvm/CodeGen/MachineFrameInfo.h"
|
2016-03-08 05:48:43 +08:00
|
|
|
#include "llvm/CodeGen/MachineFunction.h"
|
2015-06-23 01:02:30 +08:00
|
|
|
#include "llvm/CodeGen/MachineInstr.h"
|
2015-07-07 07:07:26 +08:00
|
|
|
#include "llvm/CodeGen/MachineInstrBuilder.h"
|
2015-08-04 07:08:19 +08:00
|
|
|
#include "llvm/CodeGen/MachineMemOperand.h"
|
2017-06-07 06:22:41 +08:00
|
|
|
#include "llvm/CodeGen/MachineOperand.h"
|
2016-03-08 05:48:43 +08:00
|
|
|
#include "llvm/CodeGen/MachineRegisterInfo.h"
|
2017-11-08 09:01:31 +08:00
|
|
|
#include "llvm/CodeGen/TargetInstrInfo.h"
|
2017-11-17 09:07:10 +08:00
|
|
|
#include "llvm/CodeGen/TargetRegisterInfo.h"
|
|
|
|
#include "llvm/CodeGen/TargetSubtargetInfo.h"
|
2017-06-07 06:22:41 +08:00
|
|
|
#include "llvm/IR/BasicBlock.h"
|
2015-07-29 01:28:03 +08:00
|
|
|
#include "llvm/IR/Constants.h"
|
2017-06-07 06:22:41 +08:00
|
|
|
#include "llvm/IR/DataLayout.h"
|
2017-08-24 04:31:27 +08:00
|
|
|
#include "llvm/IR/DebugInfoMetadata.h"
|
2017-06-07 06:22:41 +08:00
|
|
|
#include "llvm/IR/DebugLoc.h"
|
|
|
|
#include "llvm/IR/Function.h"
|
|
|
|
#include "llvm/IR/InstrTypes.h"
|
2016-03-08 05:48:43 +08:00
|
|
|
#include "llvm/IR/Instructions.h"
|
2016-07-30 04:32:59 +08:00
|
|
|
#include "llvm/IR/Intrinsics.h"
|
2017-06-07 06:22:41 +08:00
|
|
|
#include "llvm/IR/Metadata.h"
|
2015-06-27 06:56:48 +08:00
|
|
|
#include "llvm/IR/Module.h"
|
2015-07-28 06:42:41 +08:00
|
|
|
#include "llvm/IR/ModuleSlotTracker.h"
|
2017-06-07 06:22:41 +08:00
|
|
|
#include "llvm/IR/Type.h"
|
|
|
|
#include "llvm/IR/Value.h"
|
2015-07-29 01:28:03 +08:00
|
|
|
#include "llvm/IR/ValueSymbolTable.h"
|
2017-06-07 06:22:41 +08:00
|
|
|
#include "llvm/MC/LaneBitmask.h"
|
[x86/MIR] Implement support for pre- and post-instruction symbols, as
well as MIR parsing support for `MCSymbol` `MachineOperand`s.
The only real way to test pre- and post-instruction symbol support is to
use them in operands, so I ended up implementing that within the patch
as well. I can split out the operand support if folks really want but it
doesn't really seem worth it.
The functional implementation of pre- and post-instruction symbols is
now *completely trivial*. Two tiny bits of code in the (misnamed)
AsmPrinter. It should be completely target independent as well. We emit
these exactly the same way as we emit basic block labels. Most of the
code here is to give full dumping, MIR printing, and MIR parsing support
so that we can write useful tests.
The MIR parsing of MC symbol operands still isn't 100%, as it forces the
symbols to be non-temporary and non-local symbols with names. However,
those names often can encode most (if not all) of the special semantics
desired, and unnamed symbols seem especially annoying to serialize and
de-serialize. While this isn't perfect or full support, it seems plenty
to write tests that exercise usage of these kinds of operands.
The MIR support for pre-and post-instruction symbols was quite
straightforward. I chose to print them out in an as-if-operand syntax
similar to debug locations as this seemed the cleanest way and let me
use nice introducer tokens rather than inventing more magic punctuation
like we use for memoperands.
However, supporting MIR-based parsing of these symbols caused me to
change the design of the symbol support to allow setting arbitrary
symbols. Without this, I don't see any reasonable way to test things
with MIR.
Differential Revision: https://reviews.llvm.org/D50833
llvm-svn: 339962
2018-08-17 07:11:05 +08:00
|
|
|
#include "llvm/MC/MCContext.h"
|
2017-06-07 06:22:41 +08:00
|
|
|
#include "llvm/MC/MCDwarf.h"
|
|
|
|
#include "llvm/MC/MCInstrDesc.h"
|
|
|
|
#include "llvm/MC/MCRegisterInfo.h"
|
|
|
|
#include "llvm/Support/AtomicOrdering.h"
|
|
|
|
#include "llvm/Support/BranchProbability.h"
|
|
|
|
#include "llvm/Support/Casting.h"
|
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
|
|
#include "llvm/Support/LowLevelTypeImpl.h"
|
|
|
|
#include "llvm/Support/MemoryBuffer.h"
|
|
|
|
#include "llvm/Support/SMLoc.h"
|
2015-06-23 01:02:30 +08:00
|
|
|
#include "llvm/Support/SourceMgr.h"
|
2016-03-08 05:48:43 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2016-07-30 04:32:59 +08:00
|
|
|
#include "llvm/Target/TargetIntrinsicInfo.h"
|
2017-06-07 06:22:41 +08:00
|
|
|
#include "llvm/Target/TargetMachine.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#include <cassert>
|
2016-10-13 05:06:45 +08:00
|
|
|
#include <cctype>
|
2017-06-07 06:22:41 +08:00
|
|
|
#include <cstddef>
|
|
|
|
#include <cstdint>
|
|
|
|
#include <limits>
|
|
|
|
#include <string>
|
|
|
|
#include <utility>
|
2015-06-23 01:02:30 +08:00
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
2019-03-13 04:42:12 +08:00
|
|
|
void PerTargetMIParsingState::setTarget(
|
|
|
|
const TargetSubtargetInfo &NewSubtarget) {
|
|
|
|
|
|
|
|
// If the subtarget changed, over conservatively assume everything is invalid.
|
|
|
|
if (&Subtarget == &NewSubtarget)
|
|
|
|
return;
|
|
|
|
|
|
|
|
Names2InstrOpCodes.clear();
|
|
|
|
Names2Regs.clear();
|
|
|
|
Names2RegMasks.clear();
|
|
|
|
Names2SubRegIndices.clear();
|
|
|
|
Names2TargetIndices.clear();
|
|
|
|
Names2DirectTargetFlags.clear();
|
|
|
|
Names2BitmaskTargetFlags.clear();
|
|
|
|
Names2MMOTargetFlags.clear();
|
|
|
|
|
|
|
|
initNames2RegClasses();
|
|
|
|
initNames2RegBanks();
|
|
|
|
}
|
|
|
|
|
|
|
|
void PerTargetMIParsingState::initNames2Regs() {
|
|
|
|
if (!Names2Regs.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
// The '%noreg' register is the register 0.
|
|
|
|
Names2Regs.insert(std::make_pair("noreg", 0));
|
|
|
|
const auto *TRI = Subtarget.getRegisterInfo();
|
|
|
|
assert(TRI && "Expected target register info");
|
|
|
|
|
|
|
|
for (unsigned I = 0, E = TRI->getNumRegs(); I < E; ++I) {
|
|
|
|
bool WasInserted =
|
|
|
|
Names2Regs.insert(std::make_pair(StringRef(TRI->getName(I)).lower(), I))
|
|
|
|
.second;
|
|
|
|
(void)WasInserted;
|
|
|
|
assert(WasInserted && "Expected registers to be unique case-insensitively");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PerTargetMIParsingState::getRegisterByName(StringRef RegName,
|
|
|
|
unsigned &Reg) {
|
|
|
|
initNames2Regs();
|
|
|
|
auto RegInfo = Names2Regs.find(RegName);
|
|
|
|
if (RegInfo == Names2Regs.end())
|
|
|
|
return true;
|
|
|
|
Reg = RegInfo->getValue();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PerTargetMIParsingState::initNames2InstrOpCodes() {
|
|
|
|
if (!Names2InstrOpCodes.empty())
|
|
|
|
return;
|
|
|
|
const auto *TII = Subtarget.getInstrInfo();
|
|
|
|
assert(TII && "Expected target instruction info");
|
|
|
|
for (unsigned I = 0, E = TII->getNumOpcodes(); I < E; ++I)
|
|
|
|
Names2InstrOpCodes.insert(std::make_pair(StringRef(TII->getName(I)), I));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PerTargetMIParsingState::parseInstrName(StringRef InstrName,
|
|
|
|
unsigned &OpCode) {
|
|
|
|
initNames2InstrOpCodes();
|
|
|
|
auto InstrInfo = Names2InstrOpCodes.find(InstrName);
|
|
|
|
if (InstrInfo == Names2InstrOpCodes.end())
|
|
|
|
return true;
|
|
|
|
OpCode = InstrInfo->getValue();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PerTargetMIParsingState::initNames2RegMasks() {
|
|
|
|
if (!Names2RegMasks.empty())
|
|
|
|
return;
|
|
|
|
const auto *TRI = Subtarget.getRegisterInfo();
|
|
|
|
assert(TRI && "Expected target register info");
|
|
|
|
ArrayRef<const uint32_t *> RegMasks = TRI->getRegMasks();
|
|
|
|
ArrayRef<const char *> RegMaskNames = TRI->getRegMaskNames();
|
|
|
|
assert(RegMasks.size() == RegMaskNames.size());
|
|
|
|
for (size_t I = 0, E = RegMasks.size(); I < E; ++I)
|
|
|
|
Names2RegMasks.insert(
|
|
|
|
std::make_pair(StringRef(RegMaskNames[I]).lower(), RegMasks[I]));
|
|
|
|
}
|
|
|
|
|
|
|
|
const uint32_t *PerTargetMIParsingState::getRegMask(StringRef Identifier) {
|
|
|
|
initNames2RegMasks();
|
|
|
|
auto RegMaskInfo = Names2RegMasks.find(Identifier);
|
|
|
|
if (RegMaskInfo == Names2RegMasks.end())
|
|
|
|
return nullptr;
|
|
|
|
return RegMaskInfo->getValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
void PerTargetMIParsingState::initNames2SubRegIndices() {
|
|
|
|
if (!Names2SubRegIndices.empty())
|
|
|
|
return;
|
|
|
|
const TargetRegisterInfo *TRI = Subtarget.getRegisterInfo();
|
|
|
|
for (unsigned I = 1, E = TRI->getNumSubRegIndices(); I < E; ++I)
|
|
|
|
Names2SubRegIndices.insert(
|
|
|
|
std::make_pair(StringRef(TRI->getSubRegIndexName(I)).lower(), I));
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned PerTargetMIParsingState::getSubRegIndex(StringRef Name) {
|
|
|
|
initNames2SubRegIndices();
|
|
|
|
auto SubRegInfo = Names2SubRegIndices.find(Name);
|
|
|
|
if (SubRegInfo == Names2SubRegIndices.end())
|
|
|
|
return 0;
|
|
|
|
return SubRegInfo->getValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
void PerTargetMIParsingState::initNames2TargetIndices() {
|
|
|
|
if (!Names2TargetIndices.empty())
|
|
|
|
return;
|
|
|
|
const auto *TII = Subtarget.getInstrInfo();
|
|
|
|
assert(TII && "Expected target instruction info");
|
|
|
|
auto Indices = TII->getSerializableTargetIndices();
|
|
|
|
for (const auto &I : Indices)
|
|
|
|
Names2TargetIndices.insert(std::make_pair(StringRef(I.second), I.first));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PerTargetMIParsingState::getTargetIndex(StringRef Name, int &Index) {
|
|
|
|
initNames2TargetIndices();
|
|
|
|
auto IndexInfo = Names2TargetIndices.find(Name);
|
|
|
|
if (IndexInfo == Names2TargetIndices.end())
|
|
|
|
return true;
|
|
|
|
Index = IndexInfo->second;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PerTargetMIParsingState::initNames2DirectTargetFlags() {
|
|
|
|
if (!Names2DirectTargetFlags.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
const auto *TII = Subtarget.getInstrInfo();
|
|
|
|
assert(TII && "Expected target instruction info");
|
|
|
|
auto Flags = TII->getSerializableDirectMachineOperandTargetFlags();
|
|
|
|
for (const auto &I : Flags)
|
|
|
|
Names2DirectTargetFlags.insert(
|
|
|
|
std::make_pair(StringRef(I.second), I.first));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PerTargetMIParsingState::getDirectTargetFlag(StringRef Name,
|
|
|
|
unsigned &Flag) {
|
|
|
|
initNames2DirectTargetFlags();
|
|
|
|
auto FlagInfo = Names2DirectTargetFlags.find(Name);
|
|
|
|
if (FlagInfo == Names2DirectTargetFlags.end())
|
|
|
|
return true;
|
|
|
|
Flag = FlagInfo->second;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PerTargetMIParsingState::initNames2BitmaskTargetFlags() {
|
|
|
|
if (!Names2BitmaskTargetFlags.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
const auto *TII = Subtarget.getInstrInfo();
|
|
|
|
assert(TII && "Expected target instruction info");
|
|
|
|
auto Flags = TII->getSerializableBitmaskMachineOperandTargetFlags();
|
|
|
|
for (const auto &I : Flags)
|
|
|
|
Names2BitmaskTargetFlags.insert(
|
|
|
|
std::make_pair(StringRef(I.second), I.first));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PerTargetMIParsingState::getBitmaskTargetFlag(StringRef Name,
|
|
|
|
unsigned &Flag) {
|
|
|
|
initNames2BitmaskTargetFlags();
|
|
|
|
auto FlagInfo = Names2BitmaskTargetFlags.find(Name);
|
|
|
|
if (FlagInfo == Names2BitmaskTargetFlags.end())
|
|
|
|
return true;
|
|
|
|
Flag = FlagInfo->second;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PerTargetMIParsingState::initNames2MMOTargetFlags() {
|
|
|
|
if (!Names2MMOTargetFlags.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
const auto *TII = Subtarget.getInstrInfo();
|
|
|
|
assert(TII && "Expected target instruction info");
|
|
|
|
auto Flags = TII->getSerializableMachineMemOperandTargetFlags();
|
|
|
|
for (const auto &I : Flags)
|
|
|
|
Names2MMOTargetFlags.insert(std::make_pair(StringRef(I.second), I.first));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PerTargetMIParsingState::getMMOTargetFlag(StringRef Name,
|
|
|
|
MachineMemOperand::Flags &Flag) {
|
|
|
|
initNames2MMOTargetFlags();
|
|
|
|
auto FlagInfo = Names2MMOTargetFlags.find(Name);
|
|
|
|
if (FlagInfo == Names2MMOTargetFlags.end())
|
|
|
|
return true;
|
|
|
|
Flag = FlagInfo->second;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PerTargetMIParsingState::initNames2RegClasses() {
|
|
|
|
if (!Names2RegClasses.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
const TargetRegisterInfo *TRI = Subtarget.getRegisterInfo();
|
|
|
|
for (unsigned I = 0, E = TRI->getNumRegClasses(); I < E; ++I) {
|
|
|
|
const auto *RC = TRI->getRegClass(I);
|
|
|
|
Names2RegClasses.insert(
|
|
|
|
std::make_pair(StringRef(TRI->getRegClassName(RC)).lower(), RC));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void PerTargetMIParsingState::initNames2RegBanks() {
|
|
|
|
if (!Names2RegBanks.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
const RegisterBankInfo *RBI = Subtarget.getRegBankInfo();
|
|
|
|
// If the target does not support GlobalISel, we may not have a
|
|
|
|
// register bank info.
|
|
|
|
if (!RBI)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (unsigned I = 0, E = RBI->getNumRegBanks(); I < E; ++I) {
|
|
|
|
const auto &RegBank = RBI->getRegBank(I);
|
|
|
|
Names2RegBanks.insert(
|
|
|
|
std::make_pair(StringRef(RegBank.getName()).lower(), &RegBank));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const TargetRegisterClass *
|
|
|
|
PerTargetMIParsingState::getRegClass(StringRef Name) {
|
|
|
|
auto RegClassInfo = Names2RegClasses.find(Name);
|
|
|
|
if (RegClassInfo == Names2RegClasses.end())
|
|
|
|
return nullptr;
|
|
|
|
return RegClassInfo->getValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
const RegisterBank *PerTargetMIParsingState::getRegBank(StringRef Name) {
|
|
|
|
auto RegBankInfo = Names2RegBanks.find(Name);
|
|
|
|
if (RegBankInfo == Names2RegBanks.end())
|
|
|
|
return nullptr;
|
|
|
|
return RegBankInfo->getValue();
|
|
|
|
}
|
|
|
|
|
2016-07-14 07:27:50 +08:00
|
|
|
PerFunctionMIParsingState::PerFunctionMIParsingState(MachineFunction &MF,
|
2019-03-13 04:42:12 +08:00
|
|
|
SourceMgr &SM, const SlotMapping &IRSlots, PerTargetMIParsingState &T)
|
|
|
|
: MF(MF), SM(&SM), IRSlots(IRSlots), Target(T) {
|
2016-07-14 06:23:23 +08:00
|
|
|
}
|
|
|
|
|
2016-10-11 11:13:01 +08:00
|
|
|
VRegInfo &PerFunctionMIParsingState::getVRegInfo(unsigned Num) {
|
|
|
|
auto I = VRegInfos.insert(std::make_pair(Num, nullptr));
|
|
|
|
if (I.second) {
|
|
|
|
MachineRegisterInfo &MRI = MF.getRegInfo();
|
|
|
|
VRegInfo *Info = new (Allocator) VRegInfo;
|
|
|
|
Info->VReg = MRI.createIncompleteVirtualRegister();
|
|
|
|
I.first->second = Info;
|
|
|
|
}
|
|
|
|
return *I.first->second;
|
|
|
|
}
|
|
|
|
|
2018-03-31 02:15:54 +08:00
|
|
|
VRegInfo &PerFunctionMIParsingState::getVRegInfoNamed(StringRef RegName) {
|
|
|
|
assert(RegName != "" && "Expected named reg.");
|
|
|
|
|
|
|
|
auto I = VRegInfosNamed.insert(std::make_pair(RegName.str(), nullptr));
|
|
|
|
if (I.second) {
|
|
|
|
VRegInfo *Info = new (Allocator) VRegInfo;
|
|
|
|
Info->VReg = MF.getRegInfo().createIncompleteVirtualRegister(RegName);
|
|
|
|
I.first->second = Info;
|
|
|
|
}
|
|
|
|
return *I.first->second;
|
|
|
|
}
|
|
|
|
|
2015-06-23 01:02:30 +08:00
|
|
|
namespace {
|
|
|
|
|
2015-07-07 10:08:46 +08:00
|
|
|
/// A wrapper struct around the 'MachineOperand' struct that includes a source
|
2015-08-20 03:19:16 +08:00
|
|
|
/// range and other attributes.
|
|
|
|
struct ParsedMachineOperand {
|
2015-07-07 10:08:46 +08:00
|
|
|
MachineOperand Operand;
|
|
|
|
StringRef::iterator Begin;
|
|
|
|
StringRef::iterator End;
|
2015-08-20 03:05:34 +08:00
|
|
|
Optional<unsigned> TiedDefIdx;
|
2015-07-07 10:08:46 +08:00
|
|
|
|
2015-08-20 03:19:16 +08:00
|
|
|
ParsedMachineOperand(const MachineOperand &Operand, StringRef::iterator Begin,
|
|
|
|
StringRef::iterator End, Optional<unsigned> &TiedDefIdx)
|
2015-08-20 03:05:34 +08:00
|
|
|
: Operand(Operand), Begin(Begin), End(End), TiedDefIdx(TiedDefIdx) {
|
|
|
|
if (TiedDefIdx)
|
|
|
|
assert(Operand.isReg() && Operand.isUse() &&
|
|
|
|
"Only used register operands can be tied");
|
|
|
|
}
|
2015-07-07 10:08:46 +08:00
|
|
|
};
|
|
|
|
|
2015-06-23 01:02:30 +08:00
|
|
|
class MIParser {
|
|
|
|
MachineFunction &MF;
|
|
|
|
SMDiagnostic &Error;
|
2015-06-23 04:37:46 +08:00
|
|
|
StringRef Source, CurrentSource;
|
|
|
|
MIToken Token;
|
2016-10-11 11:13:01 +08:00
|
|
|
PerFunctionMIParsingState &PFS;
|
2015-07-28 06:42:41 +08:00
|
|
|
/// Maps from slot numbers to function's unnamed basic blocks.
|
|
|
|
DenseMap<unsigned, const BasicBlock *> Slots2BasicBlocks;
|
2015-08-20 07:31:05 +08:00
|
|
|
/// Maps from slot numbers to function's unnamed values.
|
|
|
|
DenseMap<unsigned, const Value *> Slots2Values;
|
2015-06-23 01:02:30 +08:00
|
|
|
|
|
|
|
public:
|
2016-10-11 11:13:01 +08:00
|
|
|
MIParser(PerFunctionMIParsingState &PFS, SMDiagnostic &Error,
|
2016-07-14 07:27:50 +08:00
|
|
|
StringRef Source);
|
2015-06-23 01:02:30 +08:00
|
|
|
|
2016-03-08 08:57:31 +08:00
|
|
|
/// \p SkipChar gives the number of characters to skip before looking
|
|
|
|
/// for the next token.
|
|
|
|
void lex(unsigned SkipChar = 0);
|
2015-06-23 04:37:46 +08:00
|
|
|
|
2015-06-23 01:02:30 +08:00
|
|
|
/// Report an error at the current location with the given message.
|
|
|
|
///
|
|
|
|
/// This function always return true.
|
|
|
|
bool error(const Twine &Msg);
|
|
|
|
|
2015-06-23 04:37:46 +08:00
|
|
|
/// Report an error at the given location with the given message.
|
|
|
|
///
|
|
|
|
/// This function always return true.
|
|
|
|
bool error(StringRef::iterator Loc, const Twine &Msg);
|
|
|
|
|
2015-08-14 07:10:16 +08:00
|
|
|
bool
|
|
|
|
parseBasicBlockDefinitions(DenseMap<unsigned, MachineBasicBlock *> &MBBSlots);
|
|
|
|
bool parseBasicBlocks();
|
2015-07-01 01:47:50 +08:00
|
|
|
bool parse(MachineInstr *&MI);
|
2015-07-28 04:29:27 +08:00
|
|
|
bool parseStandaloneMBB(MachineBasicBlock *&MBB);
|
|
|
|
bool parseStandaloneNamedRegister(unsigned &Reg);
|
2016-10-11 11:13:01 +08:00
|
|
|
bool parseStandaloneVirtualRegister(VRegInfo *&Info);
|
2016-11-15 08:03:14 +08:00
|
|
|
bool parseStandaloneRegister(unsigned &Reg);
|
2015-08-19 06:26:26 +08:00
|
|
|
bool parseStandaloneStackObject(int &FI);
|
2015-08-19 08:13:25 +08:00
|
|
|
bool parseStandaloneMDNode(MDNode *&Node);
|
2015-08-14 07:10:16 +08:00
|
|
|
|
|
|
|
bool
|
|
|
|
parseBasicBlockDefinition(DenseMap<unsigned, MachineBasicBlock *> &MBBSlots);
|
2017-05-06 05:09:30 +08:00
|
|
|
bool parseBasicBlock(MachineBasicBlock &MBB,
|
|
|
|
MachineBasicBlock *&AddFalthroughFrom);
|
2015-08-14 07:10:16 +08:00
|
|
|
bool parseBasicBlockLiveins(MachineBasicBlock &MBB);
|
|
|
|
bool parseBasicBlockSuccessors(MachineBasicBlock &MBB);
|
2015-06-23 01:02:30 +08:00
|
|
|
|
2016-10-11 11:13:01 +08:00
|
|
|
bool parseNamedRegister(unsigned &Reg);
|
|
|
|
bool parseVirtualRegister(VRegInfo *&Info);
|
2018-03-31 02:15:54 +08:00
|
|
|
bool parseNamedVirtualRegister(VRegInfo *&Info);
|
2016-10-11 11:13:01 +08:00
|
|
|
bool parseRegister(unsigned &Reg, VRegInfo *&VRegInfo);
|
2015-07-07 07:07:26 +08:00
|
|
|
bool parseRegisterFlag(unsigned &Flags);
|
2017-01-18 08:59:19 +08:00
|
|
|
bool parseRegisterClassOrBank(VRegInfo &RegInfo);
|
2015-07-14 07:24:34 +08:00
|
|
|
bool parseSubRegisterIndex(unsigned &SubReg);
|
2015-08-20 03:05:34 +08:00
|
|
|
bool parseRegisterTiedDefIndex(unsigned &TiedDefIdx);
|
|
|
|
bool parseRegisterOperand(MachineOperand &Dest,
|
|
|
|
Optional<unsigned> &TiedDefIdx, bool IsDef = false);
|
2015-06-24 07:42:28 +08:00
|
|
|
bool parseImmediateOperand(MachineOperand &Dest);
|
2018-07-17 02:51:40 +08:00
|
|
|
bool parseIRConstant(StringRef::iterator Loc, StringRef StringValue,
|
2015-08-22 05:48:22 +08:00
|
|
|
const Constant *&C);
|
2015-08-06 02:44:00 +08:00
|
|
|
bool parseIRConstant(StringRef::iterator Loc, const Constant *&C);
|
2016-07-29 01:15:12 +08:00
|
|
|
bool parseLowLevelType(StringRef::iterator Loc, LLT &Ty);
|
2015-08-06 02:52:21 +08:00
|
|
|
bool parseTypedImmediateOperand(MachineOperand &Dest);
|
2015-08-01 04:49:21 +08:00
|
|
|
bool parseFPImmediateOperand(MachineOperand &Dest);
|
2015-07-01 02:16:42 +08:00
|
|
|
bool parseMBBReference(MachineBasicBlock *&MBB);
|
2015-06-27 00:46:11 +08:00
|
|
|
bool parseMBBOperand(MachineOperand &Dest);
|
2015-08-19 06:18:52 +08:00
|
|
|
bool parseStackFrameIndex(int &FI);
|
2015-07-17 07:37:45 +08:00
|
|
|
bool parseStackObjectOperand(MachineOperand &Dest);
|
2015-08-13 05:17:02 +08:00
|
|
|
bool parseFixedStackFrameIndex(int &FI);
|
2015-07-17 07:37:45 +08:00
|
|
|
bool parseFixedStackObjectOperand(MachineOperand &Dest);
|
2015-07-29 01:09:52 +08:00
|
|
|
bool parseGlobalValue(GlobalValue *&GV);
|
2015-06-27 06:56:48 +08:00
|
|
|
bool parseGlobalAddressOperand(MachineOperand &Dest);
|
2015-07-21 04:51:18 +08:00
|
|
|
bool parseConstantPoolIndexOperand(MachineOperand &Dest);
|
2016-03-29 02:18:46 +08:00
|
|
|
bool parseSubRegisterIndexOperand(MachineOperand &Dest);
|
2015-07-16 07:38:35 +08:00
|
|
|
bool parseJumpTableIndexOperand(MachineOperand &Dest);
|
2015-07-22 00:59:53 +08:00
|
|
|
bool parseExternalSymbolOperand(MachineOperand &Dest);
|
[x86/MIR] Implement support for pre- and post-instruction symbols, as
well as MIR parsing support for `MCSymbol` `MachineOperand`s.
The only real way to test pre- and post-instruction symbol support is to
use them in operands, so I ended up implementing that within the patch
as well. I can split out the operand support if folks really want but it
doesn't really seem worth it.
The functional implementation of pre- and post-instruction symbols is
now *completely trivial*. Two tiny bits of code in the (misnamed)
AsmPrinter. It should be completely target independent as well. We emit
these exactly the same way as we emit basic block labels. Most of the
code here is to give full dumping, MIR printing, and MIR parsing support
so that we can write useful tests.
The MIR parsing of MC symbol operands still isn't 100%, as it forces the
symbols to be non-temporary and non-local symbols with names. However,
those names often can encode most (if not all) of the special semantics
desired, and unnamed symbols seem especially annoying to serialize and
de-serialize. While this isn't perfect or full support, it seems plenty
to write tests that exercise usage of these kinds of operands.
The MIR support for pre-and post-instruction symbols was quite
straightforward. I chose to print them out in an as-if-operand syntax
similar to debug locations as this seemed the cleanest way and let me
use nice introducer tokens rather than inventing more magic punctuation
like we use for memoperands.
However, supporting MIR-based parsing of these symbols caused me to
change the design of the symbol support to allow setting arbitrary
symbols. Without this, I don't see any reasonable way to test things
with MIR.
Differential Revision: https://reviews.llvm.org/D50833
llvm-svn: 339962
2018-08-17 07:11:05 +08:00
|
|
|
bool parseMCSymbolOperand(MachineOperand &Dest);
|
2015-07-23 05:07:04 +08:00
|
|
|
bool parseMDNode(MDNode *&Node);
|
2018-07-17 02:51:40 +08:00
|
|
|
bool parseDIExpression(MDNode *&Expr);
|
[mir] Serialize DILocation inline when not possible to use a metadata reference
Summary:
Sometimes MIR-level passes create DILocations that were not present in the
LLVM-IR. For example, it may merge two DILocations together to produce a
DILocation that points to line 0.
Previously, the address of these DILocations were printed which prevented the
MIR from being read back into LLVM. With this patch, DILocations will use
metadata references where possible and fall back on serializing them inline like so:
MOV32mr %stack.0.x.addr, 1, _, 0, _, %0, debug-location !DILocation(line: 1, scope: !15)
Reviewers: aprantl, vsk, arphaman
Reviewed By: aprantl
Subscribers: probinson, llvm-commits
Tags: #debug-info
Differential Revision: https://reviews.llvm.org/D55243
llvm-svn: 349035
2018-12-13 22:25:27 +08:00
|
|
|
bool parseDILocation(MDNode *&Expr);
|
2015-07-23 01:58:46 +08:00
|
|
|
bool parseMetadataOperand(MachineOperand &Dest);
|
2015-07-22 06:28:27 +08:00
|
|
|
bool parseCFIOffset(int &Offset);
|
2015-07-24 07:09:07 +08:00
|
|
|
bool parseCFIRegister(unsigned &Reg);
|
2017-12-15 23:17:18 +08:00
|
|
|
bool parseCFIEscapeValues(std::string& Values);
|
2015-07-22 06:28:27 +08:00
|
|
|
bool parseCFIOperand(MachineOperand &Dest);
|
2015-07-29 01:28:03 +08:00
|
|
|
bool parseIRBlock(BasicBlock *&BB, const Function &F);
|
|
|
|
bool parseBlockAddressOperand(MachineOperand &Dest);
|
2016-07-30 04:32:59 +08:00
|
|
|
bool parseIntrinsicOperand(MachineOperand &Dest);
|
2016-08-18 04:25:25 +08:00
|
|
|
bool parsePredicateOperand(MachineOperand &Dest);
|
2015-07-29 07:02:45 +08:00
|
|
|
bool parseTargetIndexOperand(MachineOperand &Dest);
|
2017-03-19 16:14:18 +08:00
|
|
|
bool parseCustomRegisterMaskOperand(MachineOperand &Dest);
|
2015-08-11 07:24:42 +08:00
|
|
|
bool parseLiveoutRegisterMaskOperand(MachineOperand &Dest);
|
2015-08-20 03:05:34 +08:00
|
|
|
bool parseMachineOperand(MachineOperand &Dest,
|
|
|
|
Optional<unsigned> &TiedDefIdx);
|
|
|
|
bool parseMachineOperandAndTargetFlags(MachineOperand &Dest,
|
|
|
|
Optional<unsigned> &TiedDefIdx);
|
2015-08-08 04:21:00 +08:00
|
|
|
bool parseOffset(int64_t &Offset);
|
2015-08-14 04:33:33 +08:00
|
|
|
bool parseAlignment(unsigned &Alignment);
|
2018-01-26 19:47:28 +08:00
|
|
|
bool parseAddrspace(unsigned &Addrspace);
|
2015-08-06 06:26:15 +08:00
|
|
|
bool parseOperandsOffset(MachineOperand &Op);
|
2015-08-20 07:27:07 +08:00
|
|
|
bool parseIRValue(const Value *&V);
|
2016-07-16 02:26:59 +08:00
|
|
|
bool parseMemoryOperandFlag(MachineMemOperand::Flags &Flags);
|
2015-08-13 04:33:26 +08:00
|
|
|
bool parseMemoryPseudoSourceValue(const PseudoSourceValue *&PSV);
|
|
|
|
bool parseMachinePointerInfo(MachinePointerInfo &Dest);
|
2017-07-12 06:23:00 +08:00
|
|
|
bool parseOptionalScope(LLVMContext &Context, SyncScope::ID &SSID);
|
2017-02-14 06:14:08 +08:00
|
|
|
bool parseOptionalAtomicOrdering(AtomicOrdering &Order);
|
2015-08-04 07:08:19 +08:00
|
|
|
bool parseMachineMemoryOperand(MachineMemOperand *&Dest);
|
[x86/MIR] Implement support for pre- and post-instruction symbols, as
well as MIR parsing support for `MCSymbol` `MachineOperand`s.
The only real way to test pre- and post-instruction symbol support is to
use them in operands, so I ended up implementing that within the patch
as well. I can split out the operand support if folks really want but it
doesn't really seem worth it.
The functional implementation of pre- and post-instruction symbols is
now *completely trivial*. Two tiny bits of code in the (misnamed)
AsmPrinter. It should be completely target independent as well. We emit
these exactly the same way as we emit basic block labels. Most of the
code here is to give full dumping, MIR printing, and MIR parsing support
so that we can write useful tests.
The MIR parsing of MC symbol operands still isn't 100%, as it forces the
symbols to be non-temporary and non-local symbols with names. However,
those names often can encode most (if not all) of the special semantics
desired, and unnamed symbols seem especially annoying to serialize and
de-serialize. While this isn't perfect or full support, it seems plenty
to write tests that exercise usage of these kinds of operands.
The MIR support for pre-and post-instruction symbols was quite
straightforward. I chose to print them out in an as-if-operand syntax
similar to debug locations as this seemed the cleanest way and let me
use nice introducer tokens rather than inventing more magic punctuation
like we use for memoperands.
However, supporting MIR-based parsing of these symbols caused me to
change the design of the symbol support to allow setting arbitrary
symbols. Without this, I don't see any reasonable way to test things
with MIR.
Differential Revision: https://reviews.llvm.org/D50833
llvm-svn: 339962
2018-08-17 07:11:05 +08:00
|
|
|
bool parsePreOrPostInstrSymbol(MCSymbol *&Symbol);
|
2015-06-24 00:35:26 +08:00
|
|
|
|
2015-06-23 01:02:30 +08:00
|
|
|
private:
|
2015-06-27 00:46:11 +08:00
|
|
|
/// Convert the integer literal in the current token into an unsigned integer.
|
|
|
|
///
|
|
|
|
/// Return true if an error occurred.
|
|
|
|
bool getUnsigned(unsigned &Result);
|
|
|
|
|
2015-08-04 07:08:19 +08:00
|
|
|
/// Convert the integer literal in the current token into an uint64.
|
|
|
|
///
|
|
|
|
/// Return true if an error occurred.
|
|
|
|
bool getUint64(uint64_t &Result);
|
|
|
|
|
2016-12-16 21:58:01 +08:00
|
|
|
/// Convert the hexadecimal literal in the current token into an unsigned
|
|
|
|
/// APInt with a minimum bitwidth required to represent the value.
|
|
|
|
///
|
|
|
|
/// Return true if the literal does not represent an integer value.
|
|
|
|
bool getHexUint(APInt &Result);
|
|
|
|
|
2015-07-24 07:09:07 +08:00
|
|
|
/// If the current token is of the given kind, consume it and return false.
|
|
|
|
/// Otherwise report an error and return true.
|
|
|
|
bool expectAndConsume(MIToken::TokenKind TokenKind);
|
|
|
|
|
2015-08-14 07:10:16 +08:00
|
|
|
/// If the current token is of the given kind, consume it and return true.
|
|
|
|
/// Otherwise return false.
|
|
|
|
bool consumeIfPresent(MIToken::TokenKind TokenKind);
|
|
|
|
|
2015-07-17 08:24:15 +08:00
|
|
|
bool parseInstruction(unsigned &OpCode, unsigned &Flags);
|
2015-06-24 00:35:26 +08:00
|
|
|
|
2015-08-20 03:05:34 +08:00
|
|
|
bool assignRegisterTies(MachineInstr &MI,
|
2015-08-20 03:19:16 +08:00
|
|
|
ArrayRef<ParsedMachineOperand> Operands);
|
2015-08-20 03:05:34 +08:00
|
|
|
|
2015-08-20 03:19:16 +08:00
|
|
|
bool verifyImplicitOperands(ArrayRef<ParsedMachineOperand> Operands,
|
2015-07-07 10:08:46 +08:00
|
|
|
const MCInstrDesc &MCID);
|
|
|
|
|
2015-07-28 06:42:41 +08:00
|
|
|
const BasicBlock *getIRBlock(unsigned Slot);
|
2015-08-07 07:57:04 +08:00
|
|
|
const BasicBlock *getIRBlock(unsigned Slot, const Function &F);
|
2015-07-29 07:02:45 +08:00
|
|
|
|
2015-08-20 07:31:05 +08:00
|
|
|
const Value *getIRValue(unsigned Slot);
|
|
|
|
|
[x86/MIR] Implement support for pre- and post-instruction symbols, as
well as MIR parsing support for `MCSymbol` `MachineOperand`s.
The only real way to test pre- and post-instruction symbol support is to
use them in operands, so I ended up implementing that within the patch
as well. I can split out the operand support if folks really want but it
doesn't really seem worth it.
The functional implementation of pre- and post-instruction symbols is
now *completely trivial*. Two tiny bits of code in the (misnamed)
AsmPrinter. It should be completely target independent as well. We emit
these exactly the same way as we emit basic block labels. Most of the
code here is to give full dumping, MIR printing, and MIR parsing support
so that we can write useful tests.
The MIR parsing of MC symbol operands still isn't 100%, as it forces the
symbols to be non-temporary and non-local symbols with names. However,
those names often can encode most (if not all) of the special semantics
desired, and unnamed symbols seem especially annoying to serialize and
de-serialize. While this isn't perfect or full support, it seems plenty
to write tests that exercise usage of these kinds of operands.
The MIR support for pre-and post-instruction symbols was quite
straightforward. I chose to print them out in an as-if-operand syntax
similar to debug locations as this seemed the cleanest way and let me
use nice introducer tokens rather than inventing more magic punctuation
like we use for memoperands.
However, supporting MIR-based parsing of these symbols caused me to
change the design of the symbol support to allow setting arbitrary
symbols. Without this, I don't see any reasonable way to test things
with MIR.
Differential Revision: https://reviews.llvm.org/D50833
llvm-svn: 339962
2018-08-17 07:11:05 +08:00
|
|
|
/// Get or create an MCSymbol for a given name.
|
|
|
|
MCSymbol *getOrCreateMCSymbol(StringRef Name);
|
|
|
|
|
2017-07-12 06:23:00 +08:00
|
|
|
/// parseStringConstant
|
|
|
|
/// ::= StringConstant
|
|
|
|
bool parseStringConstant(std::string &Result);
|
2015-06-23 01:02:30 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
2016-10-11 11:13:01 +08:00
|
|
|
MIParser::MIParser(PerFunctionMIParsingState &PFS, SMDiagnostic &Error,
|
2016-07-14 07:27:50 +08:00
|
|
|
StringRef Source)
|
|
|
|
: MF(PFS.MF), Error(Error), Source(Source), CurrentSource(Source), PFS(PFS)
|
|
|
|
{}
|
2015-06-23 04:37:46 +08:00
|
|
|
|
2016-03-08 08:57:31 +08:00
|
|
|
void MIParser::lex(unsigned SkipChar) {
|
2015-06-23 04:37:46 +08:00
|
|
|
CurrentSource = lexMIToken(
|
2016-03-08 08:57:31 +08:00
|
|
|
CurrentSource.data() + SkipChar, Token,
|
2015-06-23 04:37:46 +08:00
|
|
|
[this](StringRef::iterator Loc, const Twine &Msg) { error(Loc, Msg); });
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::error(const Twine &Msg) { return error(Token.location(), Msg); }
|
2015-06-23 01:02:30 +08:00
|
|
|
|
2015-06-23 04:37:46 +08:00
|
|
|
bool MIParser::error(StringRef::iterator Loc, const Twine &Msg) {
|
2016-07-14 07:27:50 +08:00
|
|
|
const SourceMgr &SM = *PFS.SM;
|
2015-06-23 04:37:46 +08:00
|
|
|
assert(Loc >= Source.data() && Loc <= (Source.data() + Source.size()));
|
2015-08-14 07:10:16 +08:00
|
|
|
const MemoryBuffer &Buffer = *SM.getMemoryBuffer(SM.getMainFileID());
|
|
|
|
if (Loc >= Buffer.getBufferStart() && Loc <= Buffer.getBufferEnd()) {
|
|
|
|
// Create an ordinary diagnostic when the source manager's buffer is the
|
|
|
|
// source string.
|
|
|
|
Error = SM.GetMessage(SMLoc::getFromPointer(Loc), SourceMgr::DK_Error, Msg);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// Create a diagnostic for a YAML string literal.
|
|
|
|
Error = SMDiagnostic(SM, SMLoc(), Buffer.getBufferIdentifier(), 1,
|
|
|
|
Loc - Source.data(), SourceMgr::DK_Error, Msg.str(),
|
|
|
|
Source, None, None);
|
2015-06-23 01:02:30 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-07-24 07:09:07 +08:00
|
|
|
static const char *toString(MIToken::TokenKind TokenKind) {
|
|
|
|
switch (TokenKind) {
|
|
|
|
case MIToken::comma:
|
|
|
|
return "','";
|
2015-07-30 02:51:21 +08:00
|
|
|
case MIToken::equal:
|
|
|
|
return "'='";
|
2015-08-14 07:10:16 +08:00
|
|
|
case MIToken::colon:
|
|
|
|
return "':'";
|
2015-07-29 01:28:03 +08:00
|
|
|
case MIToken::lparen:
|
|
|
|
return "'('";
|
|
|
|
case MIToken::rparen:
|
|
|
|
return "')'";
|
2015-07-24 07:09:07 +08:00
|
|
|
default:
|
|
|
|
return "<unknown token>";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::expectAndConsume(MIToken::TokenKind TokenKind) {
|
|
|
|
if (Token.isNot(TokenKind))
|
|
|
|
return error(Twine("expected ") + toString(TokenKind));
|
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-14 07:10:16 +08:00
|
|
|
bool MIParser::consumeIfPresent(MIToken::TokenKind TokenKind) {
|
|
|
|
if (Token.isNot(TokenKind))
|
|
|
|
return false;
|
|
|
|
lex();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::parseBasicBlockDefinition(
|
|
|
|
DenseMap<unsigned, MachineBasicBlock *> &MBBSlots) {
|
|
|
|
assert(Token.is(MIToken::MachineBasicBlockLabel));
|
|
|
|
unsigned ID = 0;
|
|
|
|
if (getUnsigned(ID))
|
|
|
|
return true;
|
|
|
|
auto Loc = Token.location();
|
|
|
|
auto Name = Token.stringValue();
|
|
|
|
lex();
|
|
|
|
bool HasAddressTaken = false;
|
|
|
|
bool IsLandingPad = false;
|
|
|
|
unsigned Alignment = 0;
|
|
|
|
BasicBlock *BB = nullptr;
|
|
|
|
if (consumeIfPresent(MIToken::lparen)) {
|
|
|
|
do {
|
|
|
|
// TODO: Report an error when multiple same attributes are specified.
|
|
|
|
switch (Token.kind()) {
|
|
|
|
case MIToken::kw_address_taken:
|
|
|
|
HasAddressTaken = true;
|
|
|
|
lex();
|
|
|
|
break;
|
|
|
|
case MIToken::kw_landing_pad:
|
|
|
|
IsLandingPad = true;
|
|
|
|
lex();
|
|
|
|
break;
|
|
|
|
case MIToken::kw_align:
|
|
|
|
if (parseAlignment(Alignment))
|
|
|
|
return true;
|
|
|
|
break;
|
|
|
|
case MIToken::IRBlock:
|
|
|
|
// TODO: Report an error when both name and ir block are specified.
|
2017-12-16 06:22:58 +08:00
|
|
|
if (parseIRBlock(BB, MF.getFunction()))
|
2015-08-14 07:10:16 +08:00
|
|
|
return true;
|
|
|
|
lex();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} while (consumeIfPresent(MIToken::comma));
|
|
|
|
if (expectAndConsume(MIToken::rparen))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (expectAndConsume(MIToken::colon))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (!Name.empty()) {
|
|
|
|
BB = dyn_cast_or_null<BasicBlock>(
|
2017-12-16 06:22:58 +08:00
|
|
|
MF.getFunction().getValueSymbolTable()->lookup(Name));
|
2015-08-14 07:10:16 +08:00
|
|
|
if (!BB)
|
|
|
|
return error(Loc, Twine("basic block '") + Name +
|
|
|
|
"' is not defined in the function '" +
|
|
|
|
MF.getName() + "'");
|
|
|
|
}
|
|
|
|
auto *MBB = MF.CreateMachineBasicBlock(BB);
|
|
|
|
MF.insert(MF.end(), MBB);
|
|
|
|
bool WasInserted = MBBSlots.insert(std::make_pair(ID, MBB)).second;
|
|
|
|
if (!WasInserted)
|
|
|
|
return error(Loc, Twine("redefinition of machine basic block with id #") +
|
|
|
|
Twine(ID));
|
|
|
|
if (Alignment)
|
|
|
|
MBB->setAlignment(Alignment);
|
|
|
|
if (HasAddressTaken)
|
|
|
|
MBB->setHasAddressTaken();
|
2015-08-28 07:27:47 +08:00
|
|
|
MBB->setIsEHPad(IsLandingPad);
|
2015-08-14 07:10:16 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::parseBasicBlockDefinitions(
|
|
|
|
DenseMap<unsigned, MachineBasicBlock *> &MBBSlots) {
|
|
|
|
lex();
|
|
|
|
// Skip until the first machine basic block.
|
|
|
|
while (Token.is(MIToken::Newline))
|
|
|
|
lex();
|
|
|
|
if (Token.isErrorOrEOF())
|
|
|
|
return Token.isError();
|
|
|
|
if (Token.isNot(MIToken::MachineBasicBlockLabel))
|
|
|
|
return error("expected a basic block definition before instructions");
|
2015-08-15 02:57:24 +08:00
|
|
|
unsigned BraceDepth = 0;
|
2015-08-14 07:10:16 +08:00
|
|
|
do {
|
|
|
|
if (parseBasicBlockDefinition(MBBSlots))
|
|
|
|
return true;
|
|
|
|
bool IsAfterNewline = false;
|
|
|
|
// Skip until the next machine basic block.
|
|
|
|
while (true) {
|
|
|
|
if ((Token.is(MIToken::MachineBasicBlockLabel) && IsAfterNewline) ||
|
|
|
|
Token.isErrorOrEOF())
|
|
|
|
break;
|
|
|
|
else if (Token.is(MIToken::MachineBasicBlockLabel))
|
|
|
|
return error("basic block definition should be located at the start of "
|
|
|
|
"the line");
|
2015-08-15 02:57:24 +08:00
|
|
|
else if (consumeIfPresent(MIToken::Newline)) {
|
2015-08-14 07:10:16 +08:00
|
|
|
IsAfterNewline = true;
|
2015-08-15 02:57:24 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
IsAfterNewline = false;
|
|
|
|
if (Token.is(MIToken::lbrace))
|
|
|
|
++BraceDepth;
|
|
|
|
if (Token.is(MIToken::rbrace)) {
|
|
|
|
if (!BraceDepth)
|
|
|
|
return error("extraneous closing brace ('}')");
|
|
|
|
--BraceDepth;
|
|
|
|
}
|
2015-08-14 07:10:16 +08:00
|
|
|
lex();
|
|
|
|
}
|
2015-08-15 02:57:24 +08:00
|
|
|
// Verify that we closed all of the '{' at the end of a file or a block.
|
|
|
|
if (!Token.isError() && BraceDepth)
|
|
|
|
return error("expected '}'"); // FIXME: Report a note that shows '{'.
|
2015-08-14 07:10:16 +08:00
|
|
|
} while (!Token.isErrorOrEOF());
|
|
|
|
return Token.isError();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::parseBasicBlockLiveins(MachineBasicBlock &MBB) {
|
|
|
|
assert(Token.is(MIToken::kw_liveins));
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::colon))
|
|
|
|
return true;
|
|
|
|
if (Token.isNewlineOrEOF()) // Allow an empty list of liveins.
|
|
|
|
return false;
|
|
|
|
do {
|
|
|
|
if (Token.isNot(MIToken::NamedRegister))
|
|
|
|
return error("expected a named register");
|
|
|
|
unsigned Reg = 0;
|
2016-10-11 11:13:01 +08:00
|
|
|
if (parseNamedRegister(Reg))
|
2015-08-14 07:10:16 +08:00
|
|
|
return true;
|
|
|
|
lex();
|
2016-12-15 22:36:06 +08:00
|
|
|
LaneBitmask Mask = LaneBitmask::getAll();
|
2016-10-13 05:06:45 +08:00
|
|
|
if (consumeIfPresent(MIToken::colon)) {
|
|
|
|
// Parse lane mask.
|
|
|
|
if (Token.isNot(MIToken::IntegerLiteral) &&
|
|
|
|
Token.isNot(MIToken::HexLiteral))
|
|
|
|
return error("expected a lane mask");
|
2016-12-15 22:36:06 +08:00
|
|
|
static_assert(sizeof(LaneBitmask::Type) == sizeof(unsigned),
|
|
|
|
"Use correct get-function for lane mask");
|
|
|
|
LaneBitmask::Type V;
|
|
|
|
if (getUnsigned(V))
|
2016-10-13 05:06:45 +08:00
|
|
|
return error("invalid lane mask value");
|
2016-12-15 22:36:06 +08:00
|
|
|
Mask = LaneBitmask(V);
|
2016-10-13 05:06:45 +08:00
|
|
|
lex();
|
|
|
|
}
|
|
|
|
MBB.addLiveIn(Reg, Mask);
|
2015-08-14 07:10:16 +08:00
|
|
|
} while (consumeIfPresent(MIToken::comma));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::parseBasicBlockSuccessors(MachineBasicBlock &MBB) {
|
|
|
|
assert(Token.is(MIToken::kw_successors));
|
2015-06-23 04:37:46 +08:00
|
|
|
lex();
|
2015-08-14 07:10:16 +08:00
|
|
|
if (expectAndConsume(MIToken::colon))
|
|
|
|
return true;
|
|
|
|
if (Token.isNewlineOrEOF()) // Allow an empty list of successors.
|
|
|
|
return false;
|
|
|
|
do {
|
|
|
|
if (Token.isNot(MIToken::MachineBasicBlock))
|
|
|
|
return error("expected a machine basic block reference");
|
|
|
|
MachineBasicBlock *SuccMBB = nullptr;
|
|
|
|
if (parseMBBReference(SuccMBB))
|
|
|
|
return true;
|
|
|
|
lex();
|
|
|
|
unsigned Weight = 0;
|
|
|
|
if (consumeIfPresent(MIToken::lparen)) {
|
2016-11-19 03:37:24 +08:00
|
|
|
if (Token.isNot(MIToken::IntegerLiteral) &&
|
|
|
|
Token.isNot(MIToken::HexLiteral))
|
2015-08-14 07:10:16 +08:00
|
|
|
return error("expected an integer literal after '('");
|
|
|
|
if (getUnsigned(Weight))
|
|
|
|
return true;
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::rparen))
|
|
|
|
return true;
|
|
|
|
}
|
2015-12-01 13:29:22 +08:00
|
|
|
MBB.addSuccessor(SuccMBB, BranchProbability::getRaw(Weight));
|
2015-08-14 07:10:16 +08:00
|
|
|
} while (consumeIfPresent(MIToken::comma));
|
2015-12-01 13:29:22 +08:00
|
|
|
MBB.normalizeSuccProbs();
|
2015-08-14 07:10:16 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-05-06 05:09:30 +08:00
|
|
|
bool MIParser::parseBasicBlock(MachineBasicBlock &MBB,
|
|
|
|
MachineBasicBlock *&AddFalthroughFrom) {
|
2015-08-14 07:10:16 +08:00
|
|
|
// Skip the definition.
|
|
|
|
assert(Token.is(MIToken::MachineBasicBlockLabel));
|
|
|
|
lex();
|
|
|
|
if (consumeIfPresent(MIToken::lparen)) {
|
|
|
|
while (Token.isNot(MIToken::rparen) && !Token.isErrorOrEOF())
|
|
|
|
lex();
|
|
|
|
consumeIfPresent(MIToken::rparen);
|
|
|
|
}
|
|
|
|
consumeIfPresent(MIToken::colon);
|
|
|
|
|
|
|
|
// Parse the liveins and successors.
|
|
|
|
// N.B: Multiple lists of successors and liveins are allowed and they're
|
|
|
|
// merged into one.
|
|
|
|
// Example:
|
|
|
|
// liveins: %edi
|
|
|
|
// liveins: %esi
|
|
|
|
//
|
|
|
|
// is equivalent to
|
|
|
|
// liveins: %edi, %esi
|
2017-06-27 18:35:37 +08:00
|
|
|
bool ExplicitSuccessors = false;
|
2015-08-14 07:10:16 +08:00
|
|
|
while (true) {
|
|
|
|
if (Token.is(MIToken::kw_successors)) {
|
|
|
|
if (parseBasicBlockSuccessors(MBB))
|
|
|
|
return true;
|
2017-06-27 18:35:37 +08:00
|
|
|
ExplicitSuccessors = true;
|
2015-08-14 07:10:16 +08:00
|
|
|
} else if (Token.is(MIToken::kw_liveins)) {
|
|
|
|
if (parseBasicBlockLiveins(MBB))
|
|
|
|
return true;
|
|
|
|
} else if (consumeIfPresent(MIToken::Newline)) {
|
|
|
|
continue;
|
|
|
|
} else
|
|
|
|
break;
|
|
|
|
if (!Token.isNewlineOrEOF())
|
|
|
|
return error("expected line break at the end of a list");
|
|
|
|
lex();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse the instructions.
|
2015-08-15 02:57:24 +08:00
|
|
|
bool IsInBundle = false;
|
|
|
|
MachineInstr *PrevMI = nullptr;
|
2017-05-06 05:09:30 +08:00
|
|
|
while (!Token.is(MIToken::MachineBasicBlockLabel) &&
|
|
|
|
!Token.is(MIToken::Eof)) {
|
|
|
|
if (consumeIfPresent(MIToken::Newline))
|
2015-08-14 07:10:16 +08:00
|
|
|
continue;
|
2015-08-15 02:57:24 +08:00
|
|
|
if (consumeIfPresent(MIToken::rbrace)) {
|
|
|
|
// The first parsing pass should verify that all closing '}' have an
|
|
|
|
// opening '{'.
|
|
|
|
assert(IsInBundle);
|
|
|
|
IsInBundle = false;
|
|
|
|
continue;
|
|
|
|
}
|
2015-08-14 07:10:16 +08:00
|
|
|
MachineInstr *MI = nullptr;
|
|
|
|
if (parse(MI))
|
|
|
|
return true;
|
|
|
|
MBB.insert(MBB.end(), MI);
|
2015-08-15 02:57:24 +08:00
|
|
|
if (IsInBundle) {
|
|
|
|
PrevMI->setFlag(MachineInstr::BundledSucc);
|
|
|
|
MI->setFlag(MachineInstr::BundledPred);
|
|
|
|
}
|
|
|
|
PrevMI = MI;
|
|
|
|
if (Token.is(MIToken::lbrace)) {
|
|
|
|
if (IsInBundle)
|
|
|
|
return error("nested instruction bundles are not allowed");
|
|
|
|
lex();
|
|
|
|
// This instruction is the start of the bundle.
|
|
|
|
MI->setFlag(MachineInstr::BundledSucc);
|
|
|
|
IsInBundle = true;
|
|
|
|
if (!Token.is(MIToken::Newline))
|
|
|
|
// The next instruction can be on the same line.
|
|
|
|
continue;
|
|
|
|
}
|
2015-08-14 07:10:16 +08:00
|
|
|
assert(Token.isNewlineOrEOF() && "MI is not fully parsed");
|
|
|
|
lex();
|
|
|
|
}
|
2017-05-06 05:09:30 +08:00
|
|
|
|
|
|
|
// Construct successor list by searching for basic block machine operands.
|
2017-06-27 18:35:37 +08:00
|
|
|
if (!ExplicitSuccessors) {
|
2017-05-06 05:09:30 +08:00
|
|
|
SmallVector<MachineBasicBlock*,4> Successors;
|
|
|
|
bool IsFallthrough;
|
|
|
|
guessSuccessors(MBB, Successors, IsFallthrough);
|
|
|
|
for (MachineBasicBlock *Succ : Successors)
|
|
|
|
MBB.addSuccessor(Succ);
|
|
|
|
|
|
|
|
if (IsFallthrough) {
|
|
|
|
AddFalthroughFrom = &MBB;
|
|
|
|
} else {
|
|
|
|
MBB.normalizeSuccProbs();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-14 07:10:16 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::parseBasicBlocks() {
|
|
|
|
lex();
|
|
|
|
// Skip until the first machine basic block.
|
|
|
|
while (Token.is(MIToken::Newline))
|
|
|
|
lex();
|
|
|
|
if (Token.isErrorOrEOF())
|
|
|
|
return Token.isError();
|
|
|
|
// The first parsing pass should have verified that this token is a MBB label
|
|
|
|
// in the 'parseBasicBlockDefinitions' method.
|
|
|
|
assert(Token.is(MIToken::MachineBasicBlockLabel));
|
2017-05-06 05:09:30 +08:00
|
|
|
MachineBasicBlock *AddFalthroughFrom = nullptr;
|
2015-08-14 07:10:16 +08:00
|
|
|
do {
|
|
|
|
MachineBasicBlock *MBB = nullptr;
|
|
|
|
if (parseMBBReference(MBB))
|
|
|
|
return true;
|
2017-05-06 05:09:30 +08:00
|
|
|
if (AddFalthroughFrom) {
|
|
|
|
if (!AddFalthroughFrom->isSuccessor(MBB))
|
|
|
|
AddFalthroughFrom->addSuccessor(MBB);
|
|
|
|
AddFalthroughFrom->normalizeSuccProbs();
|
|
|
|
AddFalthroughFrom = nullptr;
|
|
|
|
}
|
|
|
|
if (parseBasicBlock(*MBB, AddFalthroughFrom))
|
2015-08-14 07:10:16 +08:00
|
|
|
return true;
|
|
|
|
// The method 'parseBasicBlock' should parse the whole block until the next
|
|
|
|
// block or the end of file.
|
|
|
|
assert(Token.is(MIToken::MachineBasicBlockLabel) || Token.is(MIToken::Eof));
|
|
|
|
} while (Token.isNot(MIToken::Eof));
|
|
|
|
return false;
|
|
|
|
}
|
2015-06-23 04:37:46 +08:00
|
|
|
|
2015-08-14 07:10:16 +08:00
|
|
|
bool MIParser::parse(MachineInstr *&MI) {
|
2015-06-24 00:35:26 +08:00
|
|
|
// Parse any register operands before '='
|
|
|
|
MachineOperand MO = MachineOperand::CreateImm(0);
|
2015-08-20 03:19:16 +08:00
|
|
|
SmallVector<ParsedMachineOperand, 8> Operands;
|
2015-07-30 02:51:21 +08:00
|
|
|
while (Token.isRegister() || Token.isRegisterFlag()) {
|
2015-07-07 10:08:46 +08:00
|
|
|
auto Loc = Token.location();
|
2015-08-20 03:05:34 +08:00
|
|
|
Optional<unsigned> TiedDefIdx;
|
|
|
|
if (parseRegisterOperand(MO, TiedDefIdx, /*IsDef=*/true))
|
2015-07-01 01:47:50 +08:00
|
|
|
return true;
|
2015-08-20 03:05:34 +08:00
|
|
|
Operands.push_back(
|
2015-08-20 03:19:16 +08:00
|
|
|
ParsedMachineOperand(MO, Loc, Token.location(), TiedDefIdx));
|
2015-07-30 02:51:21 +08:00
|
|
|
if (Token.isNot(MIToken::comma))
|
|
|
|
break;
|
2015-06-24 00:35:26 +08:00
|
|
|
lex();
|
|
|
|
}
|
2015-07-30 02:51:21 +08:00
|
|
|
if (!Operands.empty() && expectAndConsume(MIToken::equal))
|
|
|
|
return true;
|
2015-06-24 00:35:26 +08:00
|
|
|
|
2015-07-17 08:24:15 +08:00
|
|
|
unsigned OpCode, Flags = 0;
|
|
|
|
if (Token.isError() || parseInstruction(OpCode, Flags))
|
2015-07-01 01:47:50 +08:00
|
|
|
return true;
|
2015-06-23 01:02:30 +08:00
|
|
|
|
2015-06-24 00:35:26 +08:00
|
|
|
// Parse the remaining machine operands.
|
[x86/MIR] Implement support for pre- and post-instruction symbols, as
well as MIR parsing support for `MCSymbol` `MachineOperand`s.
The only real way to test pre- and post-instruction symbol support is to
use them in operands, so I ended up implementing that within the patch
as well. I can split out the operand support if folks really want but it
doesn't really seem worth it.
The functional implementation of pre- and post-instruction symbols is
now *completely trivial*. Two tiny bits of code in the (misnamed)
AsmPrinter. It should be completely target independent as well. We emit
these exactly the same way as we emit basic block labels. Most of the
code here is to give full dumping, MIR printing, and MIR parsing support
so that we can write useful tests.
The MIR parsing of MC symbol operands still isn't 100%, as it forces the
symbols to be non-temporary and non-local symbols with names. However,
those names often can encode most (if not all) of the special semantics
desired, and unnamed symbols seem especially annoying to serialize and
de-serialize. While this isn't perfect or full support, it seems plenty
to write tests that exercise usage of these kinds of operands.
The MIR support for pre-and post-instruction symbols was quite
straightforward. I chose to print them out in an as-if-operand syntax
similar to debug locations as this seemed the cleanest way and let me
use nice introducer tokens rather than inventing more magic punctuation
like we use for memoperands.
However, supporting MIR-based parsing of these symbols caused me to
change the design of the symbol support to allow setting arbitrary
symbols. Without this, I don't see any reasonable way to test things
with MIR.
Differential Revision: https://reviews.llvm.org/D50833
llvm-svn: 339962
2018-08-17 07:11:05 +08:00
|
|
|
while (!Token.isNewlineOrEOF() && Token.isNot(MIToken::kw_pre_instr_symbol) &&
|
|
|
|
Token.isNot(MIToken::kw_post_instr_symbol) &&
|
|
|
|
Token.isNot(MIToken::kw_debug_location) &&
|
2015-08-15 02:57:24 +08:00
|
|
|
Token.isNot(MIToken::coloncolon) && Token.isNot(MIToken::lbrace)) {
|
2015-07-07 10:08:46 +08:00
|
|
|
auto Loc = Token.location();
|
2015-08-20 03:05:34 +08:00
|
|
|
Optional<unsigned> TiedDefIdx;
|
|
|
|
if (parseMachineOperandAndTargetFlags(MO, TiedDefIdx))
|
2015-07-01 01:47:50 +08:00
|
|
|
return true;
|
2018-10-31 07:28:27 +08:00
|
|
|
if (OpCode == TargetOpcode::DBG_VALUE && MO.isReg())
|
|
|
|
MO.setIsDebug();
|
2015-08-20 03:05:34 +08:00
|
|
|
Operands.push_back(
|
2015-08-20 03:19:16 +08:00
|
|
|
ParsedMachineOperand(MO, Loc, Token.location(), TiedDefIdx));
|
2015-08-15 02:57:24 +08:00
|
|
|
if (Token.isNewlineOrEOF() || Token.is(MIToken::coloncolon) ||
|
|
|
|
Token.is(MIToken::lbrace))
|
2015-06-24 00:35:26 +08:00
|
|
|
break;
|
2015-07-01 01:47:50 +08:00
|
|
|
if (Token.isNot(MIToken::comma))
|
|
|
|
return error("expected ',' before the next machine operand");
|
2015-06-24 00:35:26 +08:00
|
|
|
lex();
|
|
|
|
}
|
|
|
|
|
[x86/MIR] Implement support for pre- and post-instruction symbols, as
well as MIR parsing support for `MCSymbol` `MachineOperand`s.
The only real way to test pre- and post-instruction symbol support is to
use them in operands, so I ended up implementing that within the patch
as well. I can split out the operand support if folks really want but it
doesn't really seem worth it.
The functional implementation of pre- and post-instruction symbols is
now *completely trivial*. Two tiny bits of code in the (misnamed)
AsmPrinter. It should be completely target independent as well. We emit
these exactly the same way as we emit basic block labels. Most of the
code here is to give full dumping, MIR printing, and MIR parsing support
so that we can write useful tests.
The MIR parsing of MC symbol operands still isn't 100%, as it forces the
symbols to be non-temporary and non-local symbols with names. However,
those names often can encode most (if not all) of the special semantics
desired, and unnamed symbols seem especially annoying to serialize and
de-serialize. While this isn't perfect or full support, it seems plenty
to write tests that exercise usage of these kinds of operands.
The MIR support for pre-and post-instruction symbols was quite
straightforward. I chose to print them out in an as-if-operand syntax
similar to debug locations as this seemed the cleanest way and let me
use nice introducer tokens rather than inventing more magic punctuation
like we use for memoperands.
However, supporting MIR-based parsing of these symbols caused me to
change the design of the symbol support to allow setting arbitrary
symbols. Without this, I don't see any reasonable way to test things
with MIR.
Differential Revision: https://reviews.llvm.org/D50833
llvm-svn: 339962
2018-08-17 07:11:05 +08:00
|
|
|
MCSymbol *PreInstrSymbol = nullptr;
|
|
|
|
if (Token.is(MIToken::kw_pre_instr_symbol))
|
|
|
|
if (parsePreOrPostInstrSymbol(PreInstrSymbol))
|
|
|
|
return true;
|
|
|
|
MCSymbol *PostInstrSymbol = nullptr;
|
|
|
|
if (Token.is(MIToken::kw_post_instr_symbol))
|
|
|
|
if (parsePreOrPostInstrSymbol(PostInstrSymbol))
|
|
|
|
return true;
|
|
|
|
|
2015-07-23 05:15:11 +08:00
|
|
|
DebugLoc DebugLocation;
|
|
|
|
if (Token.is(MIToken::kw_debug_location)) {
|
|
|
|
lex();
|
|
|
|
MDNode *Node = nullptr;
|
[mir] Serialize DILocation inline when not possible to use a metadata reference
Summary:
Sometimes MIR-level passes create DILocations that were not present in the
LLVM-IR. For example, it may merge two DILocations together to produce a
DILocation that points to line 0.
Previously, the address of these DILocations were printed which prevented the
MIR from being read back into LLVM. With this patch, DILocations will use
metadata references where possible and fall back on serializing them inline like so:
MOV32mr %stack.0.x.addr, 1, _, 0, _, %0, debug-location !DILocation(line: 1, scope: !15)
Reviewers: aprantl, vsk, arphaman
Reviewed By: aprantl
Subscribers: probinson, llvm-commits
Tags: #debug-info
Differential Revision: https://reviews.llvm.org/D55243
llvm-svn: 349035
2018-12-13 22:25:27 +08:00
|
|
|
if (Token.is(MIToken::exclaim)) {
|
|
|
|
if (parseMDNode(Node))
|
|
|
|
return true;
|
|
|
|
} else if (Token.is(MIToken::md_dilocation)) {
|
|
|
|
if (parseDILocation(Node))
|
|
|
|
return true;
|
|
|
|
} else
|
|
|
|
return error("expected a metadata node after 'debug-location'");
|
2018-10-02 01:50:52 +08:00
|
|
|
if (!isa<DILocation>(Node))
|
|
|
|
return error("referenced metadata is not a DILocation");
|
2015-07-23 05:15:11 +08:00
|
|
|
DebugLocation = DebugLoc(Node);
|
|
|
|
}
|
|
|
|
|
2015-08-04 07:08:19 +08:00
|
|
|
// Parse the machine memory operands.
|
|
|
|
SmallVector<MachineMemOperand *, 2> MemOperands;
|
|
|
|
if (Token.is(MIToken::coloncolon)) {
|
|
|
|
lex();
|
2015-08-14 07:10:16 +08:00
|
|
|
while (!Token.isNewlineOrEOF()) {
|
2015-08-04 07:08:19 +08:00
|
|
|
MachineMemOperand *MemOp = nullptr;
|
|
|
|
if (parseMachineMemoryOperand(MemOp))
|
|
|
|
return true;
|
|
|
|
MemOperands.push_back(MemOp);
|
2015-08-14 07:10:16 +08:00
|
|
|
if (Token.isNewlineOrEOF())
|
2015-08-04 07:08:19 +08:00
|
|
|
break;
|
|
|
|
if (Token.isNot(MIToken::comma))
|
|
|
|
return error("expected ',' before the next machine memory operand");
|
|
|
|
lex();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-23 01:02:30 +08:00
|
|
|
const auto &MCID = MF.getSubtarget().getInstrInfo()->get(OpCode);
|
2015-07-07 10:08:46 +08:00
|
|
|
if (!MCID.isVariadic()) {
|
|
|
|
// FIXME: Move the implicit operand verification to the machine verifier.
|
|
|
|
if (verifyImplicitOperands(Operands, MCID))
|
|
|
|
return true;
|
|
|
|
}
|
2015-06-24 00:35:26 +08:00
|
|
|
|
2015-07-07 07:07:26 +08:00
|
|
|
// TODO: Check for extraneous machine operands.
|
2015-07-23 05:15:11 +08:00
|
|
|
MI = MF.CreateMachineInstr(MCID, DebugLocation, /*NoImplicit=*/true);
|
2015-07-17 08:24:15 +08:00
|
|
|
MI->setFlags(Flags);
|
2015-06-24 00:35:26 +08:00
|
|
|
for (const auto &Operand : Operands)
|
2015-07-07 10:08:46 +08:00
|
|
|
MI->addOperand(MF, Operand.Operand);
|
2015-08-20 03:05:34 +08:00
|
|
|
if (assignRegisterTies(*MI, Operands))
|
|
|
|
return true;
|
[x86/MIR] Implement support for pre- and post-instruction symbols, as
well as MIR parsing support for `MCSymbol` `MachineOperand`s.
The only real way to test pre- and post-instruction symbol support is to
use them in operands, so I ended up implementing that within the patch
as well. I can split out the operand support if folks really want but it
doesn't really seem worth it.
The functional implementation of pre- and post-instruction symbols is
now *completely trivial*. Two tiny bits of code in the (misnamed)
AsmPrinter. It should be completely target independent as well. We emit
these exactly the same way as we emit basic block labels. Most of the
code here is to give full dumping, MIR printing, and MIR parsing support
so that we can write useful tests.
The MIR parsing of MC symbol operands still isn't 100%, as it forces the
symbols to be non-temporary and non-local symbols with names. However,
those names often can encode most (if not all) of the special semantics
desired, and unnamed symbols seem especially annoying to serialize and
de-serialize. While this isn't perfect or full support, it seems plenty
to write tests that exercise usage of these kinds of operands.
The MIR support for pre-and post-instruction symbols was quite
straightforward. I chose to print them out in an as-if-operand syntax
similar to debug locations as this seemed the cleanest way and let me
use nice introducer tokens rather than inventing more magic punctuation
like we use for memoperands.
However, supporting MIR-based parsing of these symbols caused me to
change the design of the symbol support to allow setting arbitrary
symbols. Without this, I don't see any reasonable way to test things
with MIR.
Differential Revision: https://reviews.llvm.org/D50833
llvm-svn: 339962
2018-08-17 07:11:05 +08:00
|
|
|
if (PreInstrSymbol)
|
|
|
|
MI->setPreInstrSymbol(MF, PreInstrSymbol);
|
|
|
|
if (PostInstrSymbol)
|
|
|
|
MI->setPostInstrSymbol(MF, PostInstrSymbol);
|
|
|
|
if (!MemOperands.empty())
|
|
|
|
MI->setMemRefs(MF, MemOperands);
|
2015-07-01 01:47:50 +08:00
|
|
|
return false;
|
2015-06-23 01:02:30 +08:00
|
|
|
}
|
|
|
|
|
2015-07-28 04:29:27 +08:00
|
|
|
bool MIParser::parseStandaloneMBB(MachineBasicBlock *&MBB) {
|
2015-07-01 02:16:42 +08:00
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::MachineBasicBlock))
|
|
|
|
return error("expected a machine basic block reference");
|
|
|
|
if (parseMBBReference(MBB))
|
|
|
|
return true;
|
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::Eof))
|
|
|
|
return error(
|
|
|
|
"expected end of string after the machine basic block reference");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-28 04:29:27 +08:00
|
|
|
bool MIParser::parseStandaloneNamedRegister(unsigned &Reg) {
|
2015-07-15 05:24:41 +08:00
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::NamedRegister))
|
|
|
|
return error("expected a named register");
|
2016-10-11 11:13:01 +08:00
|
|
|
if (parseNamedRegister(Reg))
|
2015-08-19 06:57:36 +08:00
|
|
|
return true;
|
2015-07-15 05:24:41 +08:00
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::Eof))
|
|
|
|
return error("expected end of string after the register reference");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-10-11 11:13:01 +08:00
|
|
|
bool MIParser::parseStandaloneVirtualRegister(VRegInfo *&Info) {
|
2015-07-28 01:42:45 +08:00
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::VirtualRegister))
|
|
|
|
return error("expected a virtual register");
|
2016-10-11 11:13:01 +08:00
|
|
|
if (parseVirtualRegister(Info))
|
2015-08-19 06:57:36 +08:00
|
|
|
return true;
|
2015-07-28 01:42:45 +08:00
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::Eof))
|
|
|
|
return error("expected end of string after the register reference");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-11-15 08:03:14 +08:00
|
|
|
bool MIParser::parseStandaloneRegister(unsigned &Reg) {
|
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::NamedRegister) &&
|
|
|
|
Token.isNot(MIToken::VirtualRegister))
|
|
|
|
return error("expected either a named or virtual register");
|
|
|
|
|
|
|
|
VRegInfo *Info;
|
|
|
|
if (parseRegister(Reg, Info))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::Eof))
|
|
|
|
return error("expected end of string after the register reference");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-19 06:26:26 +08:00
|
|
|
bool MIParser::parseStandaloneStackObject(int &FI) {
|
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::StackObject))
|
|
|
|
return error("expected a stack object");
|
|
|
|
if (parseStackFrameIndex(FI))
|
|
|
|
return true;
|
|
|
|
if (Token.isNot(MIToken::Eof))
|
|
|
|
return error("expected end of string after the stack object reference");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-19 08:13:25 +08:00
|
|
|
bool MIParser::parseStandaloneMDNode(MDNode *&Node) {
|
|
|
|
lex();
|
2017-08-24 04:31:27 +08:00
|
|
|
if (Token.is(MIToken::exclaim)) {
|
|
|
|
if (parseMDNode(Node))
|
|
|
|
return true;
|
|
|
|
} else if (Token.is(MIToken::md_diexpr)) {
|
|
|
|
if (parseDIExpression(Node))
|
|
|
|
return true;
|
[mir] Serialize DILocation inline when not possible to use a metadata reference
Summary:
Sometimes MIR-level passes create DILocations that were not present in the
LLVM-IR. For example, it may merge two DILocations together to produce a
DILocation that points to line 0.
Previously, the address of these DILocations were printed which prevented the
MIR from being read back into LLVM. With this patch, DILocations will use
metadata references where possible and fall back on serializing them inline like so:
MOV32mr %stack.0.x.addr, 1, _, 0, _, %0, debug-location !DILocation(line: 1, scope: !15)
Reviewers: aprantl, vsk, arphaman
Reviewed By: aprantl
Subscribers: probinson, llvm-commits
Tags: #debug-info
Differential Revision: https://reviews.llvm.org/D55243
llvm-svn: 349035
2018-12-13 22:25:27 +08:00
|
|
|
} else if (Token.is(MIToken::md_dilocation)) {
|
|
|
|
if (parseDILocation(Node))
|
|
|
|
return true;
|
2017-08-24 04:31:27 +08:00
|
|
|
} else
|
2015-08-19 08:13:25 +08:00
|
|
|
return error("expected a metadata node");
|
|
|
|
if (Token.isNot(MIToken::Eof))
|
|
|
|
return error("expected end of string after the metadata node");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-07 10:08:46 +08:00
|
|
|
static const char *printImplicitRegisterFlag(const MachineOperand &MO) {
|
|
|
|
assert(MO.isImplicit());
|
|
|
|
return MO.isDef() ? "implicit-def" : "implicit";
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::string getRegisterName(const TargetRegisterInfo *TRI,
|
|
|
|
unsigned Reg) {
|
|
|
|
assert(TargetRegisterInfo::isPhysicalRegister(Reg) && "expected phys reg");
|
|
|
|
return StringRef(TRI->getName(Reg)).lower();
|
|
|
|
}
|
|
|
|
|
2015-09-10 22:04:34 +08:00
|
|
|
/// Return true if the parsed machine operands contain a given machine operand.
|
|
|
|
static bool isImplicitOperandIn(const MachineOperand &ImplicitOperand,
|
|
|
|
ArrayRef<ParsedMachineOperand> Operands) {
|
|
|
|
for (const auto &I : Operands) {
|
|
|
|
if (ImplicitOperand.isIdenticalTo(I.Operand))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-20 03:19:16 +08:00
|
|
|
bool MIParser::verifyImplicitOperands(ArrayRef<ParsedMachineOperand> Operands,
|
|
|
|
const MCInstrDesc &MCID) {
|
2015-07-07 10:08:46 +08:00
|
|
|
if (MCID.isCall())
|
|
|
|
// We can't verify call instructions as they can contain arbitrary implicit
|
|
|
|
// register and register mask operands.
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Gather all the expected implicit operands.
|
|
|
|
SmallVector<MachineOperand, 4> ImplicitOperands;
|
|
|
|
if (MCID.ImplicitDefs)
|
2015-12-05 15:13:35 +08:00
|
|
|
for (const MCPhysReg *ImpDefs = MCID.getImplicitDefs(); *ImpDefs; ++ImpDefs)
|
2015-07-07 10:08:46 +08:00
|
|
|
ImplicitOperands.push_back(
|
|
|
|
MachineOperand::CreateReg(*ImpDefs, true, true));
|
|
|
|
if (MCID.ImplicitUses)
|
2015-12-05 15:13:35 +08:00
|
|
|
for (const MCPhysReg *ImpUses = MCID.getImplicitUses(); *ImpUses; ++ImpUses)
|
2015-07-07 10:08:46 +08:00
|
|
|
ImplicitOperands.push_back(
|
|
|
|
MachineOperand::CreateReg(*ImpUses, false, true));
|
|
|
|
|
|
|
|
const auto *TRI = MF.getSubtarget().getRegisterInfo();
|
|
|
|
assert(TRI && "Expected target register info");
|
2015-09-10 22:04:34 +08:00
|
|
|
for (const auto &I : ImplicitOperands) {
|
|
|
|
if (isImplicitOperandIn(I, Operands))
|
|
|
|
continue;
|
|
|
|
return error(Operands.empty() ? Token.location() : Operands.back().End,
|
2015-07-07 10:08:46 +08:00
|
|
|
Twine("missing implicit register operand '") +
|
2018-06-20 02:39:40 +08:00
|
|
|
printImplicitRegisterFlag(I) + " $" +
|
2015-09-10 22:04:34 +08:00
|
|
|
getRegisterName(TRI, I.getReg()) + "'");
|
2015-07-07 10:08:46 +08:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-17 08:24:15 +08:00
|
|
|
bool MIParser::parseInstruction(unsigned &OpCode, unsigned &Flags) {
|
2018-05-03 08:07:56 +08:00
|
|
|
// Allow frame and fast math flags for OPCODE
|
2018-03-14 03:53:16 +08:00
|
|
|
while (Token.is(MIToken::kw_frame_setup) ||
|
2018-05-03 08:07:56 +08:00
|
|
|
Token.is(MIToken::kw_frame_destroy) ||
|
|
|
|
Token.is(MIToken::kw_nnan) ||
|
|
|
|
Token.is(MIToken::kw_ninf) ||
|
|
|
|
Token.is(MIToken::kw_nsz) ||
|
|
|
|
Token.is(MIToken::kw_arcp) ||
|
|
|
|
Token.is(MIToken::kw_contract) ||
|
|
|
|
Token.is(MIToken::kw_afn) ||
|
2018-09-12 05:35:32 +08:00
|
|
|
Token.is(MIToken::kw_reassoc) ||
|
|
|
|
Token.is(MIToken::kw_nuw) ||
|
|
|
|
Token.is(MIToken::kw_nsw) ||
|
|
|
|
Token.is(MIToken::kw_exact)) {
|
2018-05-03 08:07:56 +08:00
|
|
|
// Mine frame and fast math flags
|
|
|
|
if (Token.is(MIToken::kw_frame_setup))
|
|
|
|
Flags |= MachineInstr::FrameSetup;
|
|
|
|
if (Token.is(MIToken::kw_frame_destroy))
|
|
|
|
Flags |= MachineInstr::FrameDestroy;
|
|
|
|
if (Token.is(MIToken::kw_nnan))
|
|
|
|
Flags |= MachineInstr::FmNoNans;
|
|
|
|
if (Token.is(MIToken::kw_ninf))
|
|
|
|
Flags |= MachineInstr::FmNoInfs;
|
|
|
|
if (Token.is(MIToken::kw_nsz))
|
|
|
|
Flags |= MachineInstr::FmNsz;
|
|
|
|
if (Token.is(MIToken::kw_arcp))
|
|
|
|
Flags |= MachineInstr::FmArcp;
|
|
|
|
if (Token.is(MIToken::kw_contract))
|
|
|
|
Flags |= MachineInstr::FmContract;
|
|
|
|
if (Token.is(MIToken::kw_afn))
|
|
|
|
Flags |= MachineInstr::FmAfn;
|
|
|
|
if (Token.is(MIToken::kw_reassoc))
|
|
|
|
Flags |= MachineInstr::FmReassoc;
|
2018-09-12 05:35:32 +08:00
|
|
|
if (Token.is(MIToken::kw_nuw))
|
|
|
|
Flags |= MachineInstr::NoUWrap;
|
|
|
|
if (Token.is(MIToken::kw_nsw))
|
|
|
|
Flags |= MachineInstr::NoSWrap;
|
|
|
|
if (Token.is(MIToken::kw_exact))
|
|
|
|
Flags |= MachineInstr::IsExact;
|
2018-05-03 08:07:56 +08:00
|
|
|
|
2018-01-09 19:33:22 +08:00
|
|
|
lex();
|
2015-07-17 08:24:15 +08:00
|
|
|
}
|
2015-06-23 04:37:46 +08:00
|
|
|
if (Token.isNot(MIToken::Identifier))
|
|
|
|
return error("expected a machine instruction");
|
|
|
|
StringRef InstrName = Token.stringValue();
|
2019-03-13 04:42:12 +08:00
|
|
|
if (PFS.Target.parseInstrName(InstrName, OpCode))
|
2015-06-23 04:37:46 +08:00
|
|
|
return error(Twine("unknown machine instruction name '") + InstrName + "'");
|
2015-06-24 00:35:26 +08:00
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-10-11 11:13:01 +08:00
|
|
|
bool MIParser::parseNamedRegister(unsigned &Reg) {
|
|
|
|
assert(Token.is(MIToken::NamedRegister) && "Needs NamedRegister token");
|
|
|
|
StringRef Name = Token.stringValue();
|
2019-03-13 04:42:12 +08:00
|
|
|
if (PFS.Target.getRegisterByName(Name, Reg))
|
2016-10-11 11:13:01 +08:00
|
|
|
return error(Twine("unknown register name '") + Name + "'");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-03-31 02:15:54 +08:00
|
|
|
bool MIParser::parseNamedVirtualRegister(VRegInfo *&Info) {
|
|
|
|
assert(Token.is(MIToken::NamedVirtualRegister) && "Expected NamedVReg token");
|
|
|
|
StringRef Name = Token.stringValue();
|
|
|
|
// TODO: Check that the VReg name is not the same as a physical register name.
|
|
|
|
// If it is, then print a warning (when warnings are implemented).
|
|
|
|
Info = &PFS.getVRegInfoNamed(Name);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-10-11 11:13:01 +08:00
|
|
|
bool MIParser::parseVirtualRegister(VRegInfo *&Info) {
|
2018-03-31 02:15:54 +08:00
|
|
|
if (Token.is(MIToken::NamedVirtualRegister))
|
|
|
|
return parseNamedVirtualRegister(Info);
|
2016-10-11 11:13:01 +08:00
|
|
|
assert(Token.is(MIToken::VirtualRegister) && "Needs VirtualRegister token");
|
|
|
|
unsigned ID;
|
|
|
|
if (getUnsigned(ID))
|
|
|
|
return true;
|
|
|
|
Info = &PFS.getVRegInfo(ID);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::parseRegister(unsigned &Reg, VRegInfo *&Info) {
|
2015-06-24 00:35:26 +08:00
|
|
|
switch (Token.kind()) {
|
2015-06-25 01:34:58 +08:00
|
|
|
case MIToken::underscore:
|
|
|
|
Reg = 0;
|
2016-10-11 11:13:01 +08:00
|
|
|
return false;
|
|
|
|
case MIToken::NamedRegister:
|
|
|
|
return parseNamedRegister(Reg);
|
2018-03-31 02:15:54 +08:00
|
|
|
case MIToken::NamedVirtualRegister:
|
2016-10-11 11:13:01 +08:00
|
|
|
case MIToken::VirtualRegister:
|
|
|
|
if (parseVirtualRegister(Info))
|
2015-07-11 06:51:20 +08:00
|
|
|
return true;
|
2016-10-11 11:13:01 +08:00
|
|
|
Reg = Info->VReg;
|
|
|
|
return false;
|
2015-06-24 00:35:26 +08:00
|
|
|
// TODO: Parse other register kinds.
|
|
|
|
default:
|
|
|
|
llvm_unreachable("The current token should be a register");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-18 08:59:19 +08:00
|
|
|
bool MIParser::parseRegisterClassOrBank(VRegInfo &RegInfo) {
|
2017-01-20 08:29:59 +08:00
|
|
|
if (Token.isNot(MIToken::Identifier) && Token.isNot(MIToken::underscore))
|
|
|
|
return error("expected '_', register class, or register bank name");
|
2017-01-18 08:59:19 +08:00
|
|
|
StringRef::iterator Loc = Token.location();
|
|
|
|
StringRef Name = Token.stringValue();
|
|
|
|
|
|
|
|
// Was it a register class?
|
2019-03-13 04:42:12 +08:00
|
|
|
const TargetRegisterClass *RC = PFS.Target.getRegClass(Name);
|
|
|
|
if (RC) {
|
2017-01-18 08:59:19 +08:00
|
|
|
lex();
|
|
|
|
|
|
|
|
switch (RegInfo.Kind) {
|
|
|
|
case VRegInfo::UNKNOWN:
|
|
|
|
case VRegInfo::NORMAL:
|
|
|
|
RegInfo.Kind = VRegInfo::NORMAL;
|
2019-03-13 04:42:12 +08:00
|
|
|
if (RegInfo.Explicit && RegInfo.D.RC != RC) {
|
2017-01-18 08:59:19 +08:00
|
|
|
const TargetRegisterInfo &TRI = *MF.getSubtarget().getRegisterInfo();
|
|
|
|
return error(Loc, Twine("conflicting register classes, previously: ") +
|
|
|
|
Twine(TRI.getRegClassName(RegInfo.D.RC)));
|
|
|
|
}
|
2019-03-13 04:42:12 +08:00
|
|
|
RegInfo.D.RC = RC;
|
2017-01-18 08:59:19 +08:00
|
|
|
RegInfo.Explicit = true;
|
|
|
|
return false;
|
|
|
|
|
|
|
|
case VRegInfo::GENERIC:
|
|
|
|
case VRegInfo::REGBANK:
|
|
|
|
return error(Loc, "register class specification on generic register");
|
|
|
|
}
|
|
|
|
llvm_unreachable("Unexpected register kind");
|
|
|
|
}
|
|
|
|
|
2017-01-20 08:29:59 +08:00
|
|
|
// Should be a register bank or a generic register.
|
|
|
|
const RegisterBank *RegBank = nullptr;
|
|
|
|
if (Name != "_") {
|
2019-03-13 04:42:12 +08:00
|
|
|
RegBank = PFS.Target.getRegBank(Name);
|
|
|
|
if (!RegBank)
|
2017-01-20 08:29:59 +08:00
|
|
|
return error(Loc, "expected '_', register class, or register bank name");
|
|
|
|
}
|
|
|
|
|
2017-01-18 08:59:19 +08:00
|
|
|
lex();
|
|
|
|
|
|
|
|
switch (RegInfo.Kind) {
|
|
|
|
case VRegInfo::UNKNOWN:
|
|
|
|
case VRegInfo::GENERIC:
|
|
|
|
case VRegInfo::REGBANK:
|
2017-01-20 08:29:59 +08:00
|
|
|
RegInfo.Kind = RegBank ? VRegInfo::REGBANK : VRegInfo::GENERIC;
|
|
|
|
if (RegInfo.Explicit && RegInfo.D.RegBank != RegBank)
|
|
|
|
return error(Loc, "conflicting generic register banks");
|
|
|
|
RegInfo.D.RegBank = RegBank;
|
2017-01-18 08:59:19 +08:00
|
|
|
RegInfo.Explicit = true;
|
|
|
|
return false;
|
|
|
|
|
|
|
|
case VRegInfo::NORMAL:
|
2017-01-20 08:29:59 +08:00
|
|
|
return error(Loc, "register bank specification on normal register");
|
2017-01-18 08:59:19 +08:00
|
|
|
}
|
|
|
|
llvm_unreachable("Unexpected register kind");
|
|
|
|
}
|
|
|
|
|
2015-07-07 07:07:26 +08:00
|
|
|
bool MIParser::parseRegisterFlag(unsigned &Flags) {
|
2015-08-06 02:09:03 +08:00
|
|
|
const unsigned OldFlags = Flags;
|
2015-07-07 07:07:26 +08:00
|
|
|
switch (Token.kind()) {
|
|
|
|
case MIToken::kw_implicit:
|
|
|
|
Flags |= RegState::Implicit;
|
|
|
|
break;
|
|
|
|
case MIToken::kw_implicit_define:
|
|
|
|
Flags |= RegState::ImplicitDefine;
|
|
|
|
break;
|
2015-08-20 02:55:47 +08:00
|
|
|
case MIToken::kw_def:
|
|
|
|
Flags |= RegState::Define;
|
|
|
|
break;
|
2015-07-08 04:34:53 +08:00
|
|
|
case MIToken::kw_dead:
|
|
|
|
Flags |= RegState::Dead;
|
|
|
|
break;
|
2015-07-09 05:23:34 +08:00
|
|
|
case MIToken::kw_killed:
|
|
|
|
Flags |= RegState::Kill;
|
|
|
|
break;
|
2015-07-09 07:58:31 +08:00
|
|
|
case MIToken::kw_undef:
|
|
|
|
Flags |= RegState::Undef;
|
|
|
|
break;
|
2015-08-15 03:07:07 +08:00
|
|
|
case MIToken::kw_internal:
|
|
|
|
Flags |= RegState::InternalRead;
|
|
|
|
break;
|
2015-08-06 01:49:03 +08:00
|
|
|
case MIToken::kw_early_clobber:
|
|
|
|
Flags |= RegState::EarlyClobber;
|
|
|
|
break;
|
2015-08-06 01:41:17 +08:00
|
|
|
case MIToken::kw_debug_use:
|
|
|
|
Flags |= RegState::Debug;
|
|
|
|
break;
|
2017-12-13 01:53:59 +08:00
|
|
|
case MIToken::kw_renamable:
|
|
|
|
Flags |= RegState::Renamable;
|
|
|
|
break;
|
2015-07-07 07:07:26 +08:00
|
|
|
default:
|
|
|
|
llvm_unreachable("The current token should be a register flag");
|
|
|
|
}
|
2015-08-06 02:09:03 +08:00
|
|
|
if (OldFlags == Flags)
|
|
|
|
// We know that the same flag is specified more than once when the flags
|
|
|
|
// weren't modified.
|
|
|
|
return error("duplicate '" + Token.stringValue() + "' register flag");
|
2015-07-07 07:07:26 +08:00
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-14 07:24:34 +08:00
|
|
|
bool MIParser::parseSubRegisterIndex(unsigned &SubReg) {
|
2016-07-27 05:49:34 +08:00
|
|
|
assert(Token.is(MIToken::dot));
|
2015-07-14 07:24:34 +08:00
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::Identifier))
|
2016-07-27 05:49:34 +08:00
|
|
|
return error("expected a subregister index after '.'");
|
2015-07-14 07:24:34 +08:00
|
|
|
auto Name = Token.stringValue();
|
2019-03-13 04:42:12 +08:00
|
|
|
SubReg = PFS.Target.getSubRegIndex(Name);
|
2015-07-14 07:24:34 +08:00
|
|
|
if (!SubReg)
|
|
|
|
return error(Twine("use of unknown subregister index '") + Name + "'");
|
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-20 03:05:34 +08:00
|
|
|
bool MIParser::parseRegisterTiedDefIndex(unsigned &TiedDefIdx) {
|
|
|
|
if (!consumeIfPresent(MIToken::kw_tied_def))
|
2016-09-12 19:20:10 +08:00
|
|
|
return true;
|
2015-08-20 03:05:34 +08:00
|
|
|
if (Token.isNot(MIToken::IntegerLiteral))
|
|
|
|
return error("expected an integer literal after 'tied-def'");
|
|
|
|
if (getUnsigned(TiedDefIdx))
|
|
|
|
return true;
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::rparen))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-20 03:19:16 +08:00
|
|
|
bool MIParser::assignRegisterTies(MachineInstr &MI,
|
|
|
|
ArrayRef<ParsedMachineOperand> Operands) {
|
2015-08-20 03:05:34 +08:00
|
|
|
SmallVector<std::pair<unsigned, unsigned>, 4> TiedRegisterPairs;
|
|
|
|
for (unsigned I = 0, E = Operands.size(); I != E; ++I) {
|
|
|
|
if (!Operands[I].TiedDefIdx)
|
|
|
|
continue;
|
|
|
|
// The parser ensures that this operand is a register use, so we just have
|
|
|
|
// to check the tied-def operand.
|
|
|
|
unsigned DefIdx = Operands[I].TiedDefIdx.getValue();
|
|
|
|
if (DefIdx >= E)
|
|
|
|
return error(Operands[I].Begin,
|
|
|
|
Twine("use of invalid tied-def operand index '" +
|
|
|
|
Twine(DefIdx) + "'; instruction has only ") +
|
|
|
|
Twine(E) + " operands");
|
|
|
|
const auto &DefOperand = Operands[DefIdx].Operand;
|
|
|
|
if (!DefOperand.isReg() || !DefOperand.isDef())
|
|
|
|
// FIXME: add note with the def operand.
|
|
|
|
return error(Operands[I].Begin,
|
|
|
|
Twine("use of invalid tied-def operand index '") +
|
|
|
|
Twine(DefIdx) + "'; the operand #" + Twine(DefIdx) +
|
|
|
|
" isn't a defined register");
|
|
|
|
// Check that the tied-def operand wasn't tied elsewhere.
|
|
|
|
for (const auto &TiedPair : TiedRegisterPairs) {
|
|
|
|
if (TiedPair.first == DefIdx)
|
|
|
|
return error(Operands[I].Begin,
|
|
|
|
Twine("the tied-def operand #") + Twine(DefIdx) +
|
|
|
|
" is already tied with another register operand");
|
|
|
|
}
|
|
|
|
TiedRegisterPairs.push_back(std::make_pair(DefIdx, I));
|
|
|
|
}
|
|
|
|
// FIXME: Verify that for non INLINEASM instructions, the def and use tied
|
|
|
|
// indices must be less than tied max.
|
|
|
|
for (const auto &TiedPair : TiedRegisterPairs)
|
|
|
|
MI.tieOperands(TiedPair.first, TiedPair.second);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::parseRegisterOperand(MachineOperand &Dest,
|
|
|
|
Optional<unsigned> &TiedDefIdx,
|
|
|
|
bool IsDef) {
|
2015-07-07 07:07:26 +08:00
|
|
|
unsigned Flags = IsDef ? RegState::Define : 0;
|
|
|
|
while (Token.isRegisterFlag()) {
|
|
|
|
if (parseRegisterFlag(Flags))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (!Token.isRegister())
|
|
|
|
return error("expected a register after register flags");
|
2016-10-11 11:13:01 +08:00
|
|
|
unsigned Reg;
|
|
|
|
VRegInfo *RegInfo;
|
|
|
|
if (parseRegister(Reg, RegInfo))
|
2015-06-24 00:35:26 +08:00
|
|
|
return true;
|
|
|
|
lex();
|
2015-07-14 07:24:34 +08:00
|
|
|
unsigned SubReg = 0;
|
2016-07-27 05:49:34 +08:00
|
|
|
if (Token.is(MIToken::dot)) {
|
2015-07-14 07:24:34 +08:00
|
|
|
if (parseSubRegisterIndex(SubReg))
|
|
|
|
return true;
|
2016-07-16 09:36:18 +08:00
|
|
|
if (!TargetRegisterInfo::isVirtualRegister(Reg))
|
|
|
|
return error("subregister index expects a virtual register");
|
2015-07-14 07:24:34 +08:00
|
|
|
}
|
2017-01-18 08:59:19 +08:00
|
|
|
if (Token.is(MIToken::colon)) {
|
|
|
|
if (!TargetRegisterInfo::isVirtualRegister(Reg))
|
|
|
|
return error("register class specification expects a virtual register");
|
|
|
|
lex();
|
|
|
|
if (parseRegisterClassOrBank(*RegInfo))
|
|
|
|
return true;
|
|
|
|
}
|
2016-09-12 19:20:10 +08:00
|
|
|
MachineRegisterInfo &MRI = MF.getRegInfo();
|
2016-03-08 05:48:43 +08:00
|
|
|
if ((Flags & RegState::Define) == 0) {
|
|
|
|
if (consumeIfPresent(MIToken::lparen)) {
|
|
|
|
unsigned Idx;
|
2016-09-12 19:20:10 +08:00
|
|
|
if (!parseRegisterTiedDefIndex(Idx))
|
|
|
|
TiedDefIdx = Idx;
|
|
|
|
else {
|
|
|
|
// Try a redundant low-level type.
|
|
|
|
LLT Ty;
|
|
|
|
if (parseLowLevelType(Token.location(), Ty))
|
|
|
|
return error("expected tied-def or low-level type after '('");
|
|
|
|
|
|
|
|
if (expectAndConsume(MIToken::rparen))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (MRI.getType(Reg).isValid() && MRI.getType(Reg) != Ty)
|
|
|
|
return error("inconsistent type for generic virtual register");
|
|
|
|
|
|
|
|
MRI.setType(Reg, Ty);
|
|
|
|
}
|
2016-03-08 05:48:43 +08:00
|
|
|
}
|
|
|
|
} else if (consumeIfPresent(MIToken::lparen)) {
|
2016-12-23 05:56:35 +08:00
|
|
|
// Virtual registers may have a tpe with GlobalISel.
|
2016-03-08 05:48:43 +08:00
|
|
|
if (!TargetRegisterInfo::isVirtualRegister(Reg))
|
2016-12-23 05:56:35 +08:00
|
|
|
return error("unexpected type on physical register");
|
2016-07-20 03:48:36 +08:00
|
|
|
|
2016-09-09 19:46:34 +08:00
|
|
|
LLT Ty;
|
|
|
|
if (parseLowLevelType(Token.location(), Ty))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (expectAndConsume(MIToken::rparen))
|
2015-08-20 03:05:34 +08:00
|
|
|
return true;
|
2016-03-08 05:48:43 +08:00
|
|
|
|
2016-09-12 19:20:10 +08:00
|
|
|
if (MRI.getType(Reg).isValid() && MRI.getType(Reg) != Ty)
|
|
|
|
return error("inconsistent type for generic virtual register");
|
|
|
|
|
2016-09-09 19:46:34 +08:00
|
|
|
MRI.setType(Reg, Ty);
|
2016-10-11 11:13:01 +08:00
|
|
|
} else if (TargetRegisterInfo::isVirtualRegister(Reg)) {
|
2016-12-23 06:50:34 +08:00
|
|
|
// Generic virtual registers must have a type.
|
|
|
|
// If we end up here this means the type hasn't been specified and
|
2016-06-09 07:27:46 +08:00
|
|
|
// this is bad!
|
2016-10-11 11:13:01 +08:00
|
|
|
if (RegInfo->Kind == VRegInfo::GENERIC ||
|
|
|
|
RegInfo->Kind == VRegInfo::REGBANK)
|
2016-12-23 06:50:34 +08:00
|
|
|
return error("generic virtual registers must have a type");
|
2015-08-20 03:05:34 +08:00
|
|
|
}
|
2015-07-09 05:23:34 +08:00
|
|
|
Dest = MachineOperand::CreateReg(
|
|
|
|
Reg, Flags & RegState::Define, Flags & RegState::Implicit,
|
2015-07-14 07:24:34 +08:00
|
|
|
Flags & RegState::Kill, Flags & RegState::Dead, Flags & RegState::Undef,
|
2015-08-15 03:07:07 +08:00
|
|
|
Flags & RegState::EarlyClobber, SubReg, Flags & RegState::Debug,
|
2017-12-13 01:53:59 +08:00
|
|
|
Flags & RegState::InternalRead, Flags & RegState::Renamable);
|
|
|
|
|
2015-06-24 00:35:26 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-06-24 07:42:28 +08:00
|
|
|
bool MIParser::parseImmediateOperand(MachineOperand &Dest) {
|
|
|
|
assert(Token.is(MIToken::IntegerLiteral));
|
|
|
|
const APSInt &Int = Token.integerValue();
|
|
|
|
if (Int.getMinSignedBits() > 64)
|
2015-08-06 03:03:42 +08:00
|
|
|
return error("integer literal is too large to be an immediate operand");
|
2015-06-24 07:42:28 +08:00
|
|
|
Dest = MachineOperand::CreateImm(Int.getExtValue());
|
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-22 05:48:22 +08:00
|
|
|
bool MIParser::parseIRConstant(StringRef::iterator Loc, StringRef StringValue,
|
|
|
|
const Constant *&C) {
|
|
|
|
auto Source = StringValue.str(); // The source has to be null terminated.
|
2015-08-01 04:49:21 +08:00
|
|
|
SMDiagnostic Err;
|
2017-12-16 06:22:58 +08:00
|
|
|
C = parseConstantValue(Source, Err, *MF.getFunction().getParent(),
|
2016-07-14 07:27:50 +08:00
|
|
|
&PFS.IRSlots);
|
2015-08-01 04:49:21 +08:00
|
|
|
if (!C)
|
|
|
|
return error(Loc + Err.getColumnNo(), Err.getMessage());
|
2015-08-06 02:44:00 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-22 05:48:22 +08:00
|
|
|
bool MIParser::parseIRConstant(StringRef::iterator Loc, const Constant *&C) {
|
|
|
|
if (parseIRConstant(Loc, StringRef(Loc, Token.range().end() - Loc), C))
|
|
|
|
return true;
|
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-02-05 06:59:56 +08:00
|
|
|
// See LLT implemntation for bit size limits.
|
|
|
|
static bool verifyScalarSize(uint64_t Size) {
|
|
|
|
return Size != 0 && isUInt<16>(Size);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool verifyVectorElementCount(uint64_t NumElts) {
|
|
|
|
return NumElts != 0 && isUInt<16>(NumElts);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool verifyAddrSpace(uint64_t AddrSpace) {
|
|
|
|
return isUInt<24>(AddrSpace);
|
|
|
|
}
|
|
|
|
|
2016-07-29 01:15:12 +08:00
|
|
|
bool MIParser::parseLowLevelType(StringRef::iterator Loc, LLT &Ty) {
|
2018-05-06 04:53:23 +08:00
|
|
|
if (Token.range().front() == 's' || Token.range().front() == 'p') {
|
|
|
|
StringRef SizeStr = Token.range().drop_front();
|
|
|
|
if (SizeStr.size() == 0 || !llvm::all_of(SizeStr, isdigit))
|
2018-05-08 10:02:50 +08:00
|
|
|
return error("expected integers after 's'/'p' type character");
|
2018-05-06 04:53:23 +08:00
|
|
|
}
|
2018-05-05 15:05:51 +08:00
|
|
|
|
|
|
|
if (Token.range().front() == 's') {
|
2019-02-05 06:59:56 +08:00
|
|
|
auto ScalarSize = APSInt(Token.range().drop_front()).getZExtValue();
|
|
|
|
if (!verifyScalarSize(ScalarSize))
|
|
|
|
return error("invalid size for scalar type");
|
|
|
|
|
|
|
|
Ty = LLT::scalar(ScalarSize);
|
2016-07-21 03:09:30 +08:00
|
|
|
lex();
|
|
|
|
return false;
|
2018-05-05 15:05:51 +08:00
|
|
|
} else if (Token.range().front() == 'p') {
|
2017-12-16 06:22:58 +08:00
|
|
|
const DataLayout &DL = MF.getDataLayout();
|
2019-02-05 06:59:56 +08:00
|
|
|
uint64_t AS = APSInt(Token.range().drop_front()).getZExtValue();
|
|
|
|
if (!verifyAddrSpace(AS))
|
|
|
|
return error("invalid address space number");
|
|
|
|
|
2016-09-15 17:20:34 +08:00
|
|
|
Ty = LLT::pointer(AS, DL.getPointerSizeInBits(AS));
|
2016-07-23 00:59:52 +08:00
|
|
|
lex();
|
|
|
|
return false;
|
2016-07-21 03:09:30 +08:00
|
|
|
}
|
2016-03-08 08:20:48 +08:00
|
|
|
|
2016-07-21 03:09:30 +08:00
|
|
|
// Now we're looking for a vector.
|
|
|
|
if (Token.isNot(MIToken::less))
|
2016-07-23 00:59:52 +08:00
|
|
|
return error(Loc,
|
2018-05-08 10:02:50 +08:00
|
|
|
"expected sN, pA, <M x sN>, or <M x pA> for GlobalISel type");
|
2016-07-21 03:09:30 +08:00
|
|
|
lex();
|
|
|
|
|
|
|
|
if (Token.isNot(MIToken::IntegerLiteral))
|
2018-05-08 10:02:50 +08:00
|
|
|
return error(Loc, "expected <M x sN> or <M x pA> for vector type");
|
2016-07-21 03:09:30 +08:00
|
|
|
uint64_t NumElements = Token.integerValue().getZExtValue();
|
2019-02-05 06:59:56 +08:00
|
|
|
if (!verifyVectorElementCount(NumElements))
|
|
|
|
return error("invalid number of vector elements");
|
|
|
|
|
2016-07-21 03:09:30 +08:00
|
|
|
lex();
|
|
|
|
|
|
|
|
if (Token.isNot(MIToken::Identifier) || Token.stringValue() != "x")
|
2018-05-08 10:02:50 +08:00
|
|
|
return error(Loc, "expected <M x sN> or <M x pA> for vector type");
|
2016-07-21 03:09:30 +08:00
|
|
|
lex();
|
|
|
|
|
2018-05-08 10:02:50 +08:00
|
|
|
if (Token.range().front() != 's' && Token.range().front() != 'p')
|
|
|
|
return error(Loc, "expected <M x sN> or <M x pA> for vector type");
|
2018-05-06 04:53:23 +08:00
|
|
|
StringRef SizeStr = Token.range().drop_front();
|
|
|
|
if (SizeStr.size() == 0 || !llvm::all_of(SizeStr, isdigit))
|
2018-05-08 10:02:50 +08:00
|
|
|
return error("expected integers after 's'/'p' type character");
|
|
|
|
|
2019-02-05 06:59:56 +08:00
|
|
|
if (Token.range().front() == 's') {
|
|
|
|
auto ScalarSize = APSInt(Token.range().drop_front()).getZExtValue();
|
|
|
|
if (!verifyScalarSize(ScalarSize))
|
|
|
|
return error("invalid size for scalar type");
|
|
|
|
Ty = LLT::scalar(ScalarSize);
|
|
|
|
} else if (Token.range().front() == 'p') {
|
2018-05-08 10:02:50 +08:00
|
|
|
const DataLayout &DL = MF.getDataLayout();
|
2019-02-05 06:59:56 +08:00
|
|
|
uint64_t AS = APSInt(Token.range().drop_front()).getZExtValue();
|
|
|
|
if (!verifyAddrSpace(AS))
|
|
|
|
return error("invalid address space number");
|
|
|
|
|
2018-05-08 10:02:50 +08:00
|
|
|
Ty = LLT::pointer(AS, DL.getPointerSizeInBits(AS));
|
|
|
|
} else
|
|
|
|
return error(Loc, "expected <M x sN> or <M x pA> for vector type");
|
2016-07-21 03:09:30 +08:00
|
|
|
lex();
|
|
|
|
|
|
|
|
if (Token.isNot(MIToken::greater))
|
2018-05-08 10:02:50 +08:00
|
|
|
return error(Loc, "expected <M x sN> or <M x pA> for vector type");
|
2016-07-21 03:09:30 +08:00
|
|
|
lex();
|
|
|
|
|
2018-05-08 10:02:50 +08:00
|
|
|
Ty = LLT::vector(NumElements, Ty);
|
2016-03-08 08:20:48 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-06 02:52:21 +08:00
|
|
|
bool MIParser::parseTypedImmediateOperand(MachineOperand &Dest) {
|
2018-05-05 15:05:51 +08:00
|
|
|
assert(Token.is(MIToken::Identifier));
|
2018-05-06 04:53:23 +08:00
|
|
|
StringRef TypeStr = Token.range();
|
|
|
|
if (TypeStr.front() != 'i' && TypeStr.front() != 's' &&
|
|
|
|
TypeStr.front() != 'p')
|
|
|
|
return error(
|
2018-05-08 10:02:50 +08:00
|
|
|
"a typed immediate operand should start with one of 'i', 's', or 'p'");
|
2018-05-06 04:53:23 +08:00
|
|
|
StringRef SizeStr = Token.range().drop_front();
|
|
|
|
if (SizeStr.size() == 0 || !llvm::all_of(SizeStr, isdigit))
|
2018-05-08 10:02:50 +08:00
|
|
|
return error("expected integers after 'i'/'s'/'p' type character");
|
2018-05-05 15:05:51 +08:00
|
|
|
|
2015-08-06 02:52:21 +08:00
|
|
|
auto Loc = Token.location();
|
|
|
|
lex();
|
2018-06-05 08:17:13 +08:00
|
|
|
if (Token.isNot(MIToken::IntegerLiteral)) {
|
|
|
|
if (Token.isNot(MIToken::Identifier) ||
|
|
|
|
!(Token.range() == "true" || Token.range() == "false"))
|
|
|
|
return error("expected an integer literal");
|
|
|
|
}
|
2015-08-06 02:52:21 +08:00
|
|
|
const Constant *C = nullptr;
|
|
|
|
if (parseIRConstant(Loc, C))
|
|
|
|
return true;
|
|
|
|
Dest = MachineOperand::CreateCImm(cast<ConstantInt>(C));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-06 02:44:00 +08:00
|
|
|
bool MIParser::parseFPImmediateOperand(MachineOperand &Dest) {
|
|
|
|
auto Loc = Token.location();
|
|
|
|
lex();
|
2016-10-13 05:06:45 +08:00
|
|
|
if (Token.isNot(MIToken::FloatingPointLiteral) &&
|
|
|
|
Token.isNot(MIToken::HexLiteral))
|
2015-08-06 02:44:00 +08:00
|
|
|
return error("expected a floating point literal");
|
|
|
|
const Constant *C = nullptr;
|
|
|
|
if (parseIRConstant(Loc, C))
|
|
|
|
return true;
|
2015-08-01 04:49:21 +08:00
|
|
|
Dest = MachineOperand::CreateFPImm(cast<ConstantFP>(C));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-06-27 00:46:11 +08:00
|
|
|
bool MIParser::getUnsigned(unsigned &Result) {
|
2016-10-13 05:06:45 +08:00
|
|
|
if (Token.hasIntegerValue()) {
|
|
|
|
const uint64_t Limit = uint64_t(std::numeric_limits<unsigned>::max()) + 1;
|
|
|
|
uint64_t Val64 = Token.integerValue().getLimitedValue(Limit);
|
|
|
|
if (Val64 == Limit)
|
|
|
|
return error("expected 32-bit integer (too large)");
|
|
|
|
Result = Val64;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (Token.is(MIToken::HexLiteral)) {
|
2016-12-16 21:58:01 +08:00
|
|
|
APInt A;
|
|
|
|
if (getHexUint(A))
|
2016-10-13 05:06:45 +08:00
|
|
|
return true;
|
2016-12-16 21:58:01 +08:00
|
|
|
if (A.getBitWidth() > 32)
|
2016-10-13 05:06:45 +08:00
|
|
|
return error("expected 32-bit integer (too large)");
|
|
|
|
Result = A.getZExtValue();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
2015-06-27 00:46:11 +08:00
|
|
|
}
|
|
|
|
|
2015-07-01 02:16:42 +08:00
|
|
|
bool MIParser::parseMBBReference(MachineBasicBlock *&MBB) {
|
2015-08-14 07:10:16 +08:00
|
|
|
assert(Token.is(MIToken::MachineBasicBlock) ||
|
|
|
|
Token.is(MIToken::MachineBasicBlockLabel));
|
2015-06-27 00:46:11 +08:00
|
|
|
unsigned Number;
|
|
|
|
if (getUnsigned(Number))
|
|
|
|
return true;
|
2015-07-08 01:46:43 +08:00
|
|
|
auto MBBInfo = PFS.MBBSlots.find(Number);
|
|
|
|
if (MBBInfo == PFS.MBBSlots.end())
|
2015-06-27 00:46:11 +08:00
|
|
|
return error(Twine("use of undefined machine basic block #") +
|
|
|
|
Twine(Number));
|
2015-07-01 02:16:42 +08:00
|
|
|
MBB = MBBInfo->second;
|
2017-12-05 01:18:51 +08:00
|
|
|
// TODO: Only parse the name if it's a MachineBasicBlockLabel. Deprecate once
|
|
|
|
// we drop the <irname> from the bb.<id>.<irname> format.
|
2015-06-27 00:46:11 +08:00
|
|
|
if (!Token.stringValue().empty() && Token.stringValue() != MBB->getName())
|
|
|
|
return error(Twine("the name of machine basic block #") + Twine(Number) +
|
|
|
|
" isn't '" + Token.stringValue() + "'");
|
2015-07-01 02:16:42 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::parseMBBOperand(MachineOperand &Dest) {
|
|
|
|
MachineBasicBlock *MBB;
|
|
|
|
if (parseMBBReference(MBB))
|
|
|
|
return true;
|
2015-06-27 00:46:11 +08:00
|
|
|
Dest = MachineOperand::CreateMBB(MBB);
|
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-19 06:18:52 +08:00
|
|
|
bool MIParser::parseStackFrameIndex(int &FI) {
|
2015-07-17 07:37:45 +08:00
|
|
|
assert(Token.is(MIToken::StackObject));
|
|
|
|
unsigned ID;
|
|
|
|
if (getUnsigned(ID))
|
|
|
|
return true;
|
|
|
|
auto ObjectInfo = PFS.StackObjectSlots.find(ID);
|
|
|
|
if (ObjectInfo == PFS.StackObjectSlots.end())
|
|
|
|
return error(Twine("use of undefined stack object '%stack.") + Twine(ID) +
|
|
|
|
"'");
|
|
|
|
StringRef Name;
|
|
|
|
if (const auto *Alloca =
|
2016-07-29 02:40:00 +08:00
|
|
|
MF.getFrameInfo().getObjectAllocation(ObjectInfo->second))
|
2015-07-17 07:37:45 +08:00
|
|
|
Name = Alloca->getName();
|
|
|
|
if (!Token.stringValue().empty() && Token.stringValue() != Name)
|
|
|
|
return error(Twine("the name of the stack object '%stack.") + Twine(ID) +
|
|
|
|
"' isn't '" + Token.stringValue() + "'");
|
|
|
|
lex();
|
2015-08-19 06:18:52 +08:00
|
|
|
FI = ObjectInfo->second;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::parseStackObjectOperand(MachineOperand &Dest) {
|
|
|
|
int FI;
|
|
|
|
if (parseStackFrameIndex(FI))
|
|
|
|
return true;
|
|
|
|
Dest = MachineOperand::CreateFI(FI);
|
2015-07-17 07:37:45 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-13 05:17:02 +08:00
|
|
|
bool MIParser::parseFixedStackFrameIndex(int &FI) {
|
2015-07-17 07:37:45 +08:00
|
|
|
assert(Token.is(MIToken::FixedStackObject));
|
|
|
|
unsigned ID;
|
|
|
|
if (getUnsigned(ID))
|
|
|
|
return true;
|
|
|
|
auto ObjectInfo = PFS.FixedStackObjectSlots.find(ID);
|
|
|
|
if (ObjectInfo == PFS.FixedStackObjectSlots.end())
|
|
|
|
return error(Twine("use of undefined fixed stack object '%fixed-stack.") +
|
|
|
|
Twine(ID) + "'");
|
|
|
|
lex();
|
2015-08-13 05:17:02 +08:00
|
|
|
FI = ObjectInfo->second;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::parseFixedStackObjectOperand(MachineOperand &Dest) {
|
|
|
|
int FI;
|
|
|
|
if (parseFixedStackFrameIndex(FI))
|
|
|
|
return true;
|
|
|
|
Dest = MachineOperand::CreateFI(FI);
|
2015-07-17 07:37:45 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-29 01:09:52 +08:00
|
|
|
bool MIParser::parseGlobalValue(GlobalValue *&GV) {
|
2015-06-27 06:56:48 +08:00
|
|
|
switch (Token.kind()) {
|
2015-08-06 01:35:55 +08:00
|
|
|
case MIToken::NamedGlobalValue: {
|
2017-12-16 06:22:58 +08:00
|
|
|
const Module *M = MF.getFunction().getParent();
|
2015-08-06 01:35:55 +08:00
|
|
|
GV = M->getNamedValue(Token.stringValue());
|
2015-07-29 01:09:52 +08:00
|
|
|
if (!GV)
|
2015-08-07 07:17:42 +08:00
|
|
|
return error(Twine("use of undefined global value '") + Token.range() +
|
|
|
|
"'");
|
2015-07-29 01:09:52 +08:00
|
|
|
break;
|
2015-06-27 06:56:48 +08:00
|
|
|
}
|
|
|
|
case MIToken::GlobalValue: {
|
|
|
|
unsigned GVIdx;
|
|
|
|
if (getUnsigned(GVIdx))
|
|
|
|
return true;
|
2016-07-14 07:27:50 +08:00
|
|
|
if (GVIdx >= PFS.IRSlots.GlobalValues.size())
|
2015-06-27 06:56:48 +08:00
|
|
|
return error(Twine("use of undefined global value '@") + Twine(GVIdx) +
|
|
|
|
"'");
|
2016-07-14 07:27:50 +08:00
|
|
|
GV = PFS.IRSlots.GlobalValues[GVIdx];
|
2015-06-27 06:56:48 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
llvm_unreachable("The current token should be a global value");
|
|
|
|
}
|
2015-07-29 01:09:52 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::parseGlobalAddressOperand(MachineOperand &Dest) {
|
|
|
|
GlobalValue *GV = nullptr;
|
|
|
|
if (parseGlobalValue(GV))
|
|
|
|
return true;
|
2015-06-27 06:56:48 +08:00
|
|
|
lex();
|
2015-08-06 06:26:15 +08:00
|
|
|
Dest = MachineOperand::CreateGA(GV, /*Offset=*/0);
|
|
|
|
if (parseOperandsOffset(Dest))
|
|
|
|
return true;
|
2015-06-27 06:56:48 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-21 04:51:18 +08:00
|
|
|
bool MIParser::parseConstantPoolIndexOperand(MachineOperand &Dest) {
|
|
|
|
assert(Token.is(MIToken::ConstantPoolItem));
|
|
|
|
unsigned ID;
|
|
|
|
if (getUnsigned(ID))
|
|
|
|
return true;
|
|
|
|
auto ConstantInfo = PFS.ConstantPoolSlots.find(ID);
|
|
|
|
if (ConstantInfo == PFS.ConstantPoolSlots.end())
|
|
|
|
return error("use of undefined constant '%const." + Twine(ID) + "'");
|
|
|
|
lex();
|
|
|
|
Dest = MachineOperand::CreateCPI(ID, /*Offset=*/0);
|
2015-08-06 06:26:15 +08:00
|
|
|
if (parseOperandsOffset(Dest))
|
|
|
|
return true;
|
2015-07-21 04:51:18 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-16 07:38:35 +08:00
|
|
|
bool MIParser::parseJumpTableIndexOperand(MachineOperand &Dest) {
|
|
|
|
assert(Token.is(MIToken::JumpTableIndex));
|
|
|
|
unsigned ID;
|
|
|
|
if (getUnsigned(ID))
|
|
|
|
return true;
|
|
|
|
auto JumpTableEntryInfo = PFS.JumpTableSlots.find(ID);
|
|
|
|
if (JumpTableEntryInfo == PFS.JumpTableSlots.end())
|
|
|
|
return error("use of undefined jump table '%jump-table." + Twine(ID) + "'");
|
|
|
|
lex();
|
|
|
|
Dest = MachineOperand::CreateJTI(JumpTableEntryInfo->second);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-22 00:59:53 +08:00
|
|
|
bool MIParser::parseExternalSymbolOperand(MachineOperand &Dest) {
|
2015-08-06 01:35:55 +08:00
|
|
|
assert(Token.is(MIToken::ExternalSymbol));
|
|
|
|
const char *Symbol = MF.createExternalSymbolName(Token.stringValue());
|
2015-07-22 00:59:53 +08:00
|
|
|
lex();
|
|
|
|
Dest = MachineOperand::CreateES(Symbol);
|
2015-08-06 06:26:15 +08:00
|
|
|
if (parseOperandsOffset(Dest))
|
|
|
|
return true;
|
2015-07-22 00:59:53 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
[x86/MIR] Implement support for pre- and post-instruction symbols, as
well as MIR parsing support for `MCSymbol` `MachineOperand`s.
The only real way to test pre- and post-instruction symbol support is to
use them in operands, so I ended up implementing that within the patch
as well. I can split out the operand support if folks really want but it
doesn't really seem worth it.
The functional implementation of pre- and post-instruction symbols is
now *completely trivial*. Two tiny bits of code in the (misnamed)
AsmPrinter. It should be completely target independent as well. We emit
these exactly the same way as we emit basic block labels. Most of the
code here is to give full dumping, MIR printing, and MIR parsing support
so that we can write useful tests.
The MIR parsing of MC symbol operands still isn't 100%, as it forces the
symbols to be non-temporary and non-local symbols with names. However,
those names often can encode most (if not all) of the special semantics
desired, and unnamed symbols seem especially annoying to serialize and
de-serialize. While this isn't perfect or full support, it seems plenty
to write tests that exercise usage of these kinds of operands.
The MIR support for pre-and post-instruction symbols was quite
straightforward. I chose to print them out in an as-if-operand syntax
similar to debug locations as this seemed the cleanest way and let me
use nice introducer tokens rather than inventing more magic punctuation
like we use for memoperands.
However, supporting MIR-based parsing of these symbols caused me to
change the design of the symbol support to allow setting arbitrary
symbols. Without this, I don't see any reasonable way to test things
with MIR.
Differential Revision: https://reviews.llvm.org/D50833
llvm-svn: 339962
2018-08-17 07:11:05 +08:00
|
|
|
bool MIParser::parseMCSymbolOperand(MachineOperand &Dest) {
|
|
|
|
assert(Token.is(MIToken::MCSymbol));
|
|
|
|
MCSymbol *Symbol = getOrCreateMCSymbol(Token.stringValue());
|
|
|
|
lex();
|
|
|
|
Dest = MachineOperand::CreateMCSymbol(Symbol);
|
|
|
|
if (parseOperandsOffset(Dest))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-03-29 02:18:46 +08:00
|
|
|
bool MIParser::parseSubRegisterIndexOperand(MachineOperand &Dest) {
|
|
|
|
assert(Token.is(MIToken::SubRegisterIndex));
|
|
|
|
StringRef Name = Token.stringValue();
|
2019-03-13 04:42:12 +08:00
|
|
|
unsigned SubRegIndex = PFS.Target.getSubRegIndex(Token.stringValue());
|
2016-03-29 02:18:46 +08:00
|
|
|
if (SubRegIndex == 0)
|
|
|
|
return error(Twine("unknown subregister index '") + Name + "'");
|
|
|
|
lex();
|
|
|
|
Dest = MachineOperand::CreateImm(SubRegIndex);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-23 05:07:04 +08:00
|
|
|
bool MIParser::parseMDNode(MDNode *&Node) {
|
2015-07-23 01:58:46 +08:00
|
|
|
assert(Token.is(MIToken::exclaim));
|
2017-08-24 04:31:27 +08:00
|
|
|
|
2015-07-23 01:58:46 +08:00
|
|
|
auto Loc = Token.location();
|
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::IntegerLiteral) || Token.integerValue().isSigned())
|
|
|
|
return error("expected metadata id after '!'");
|
|
|
|
unsigned ID;
|
|
|
|
if (getUnsigned(ID))
|
|
|
|
return true;
|
2016-07-14 07:27:50 +08:00
|
|
|
auto NodeInfo = PFS.IRSlots.MetadataNodes.find(ID);
|
|
|
|
if (NodeInfo == PFS.IRSlots.MetadataNodes.end())
|
2015-07-23 01:58:46 +08:00
|
|
|
return error(Loc, "use of undefined metadata '!" + Twine(ID) + "'");
|
|
|
|
lex();
|
2015-07-23 05:07:04 +08:00
|
|
|
Node = NodeInfo->second.get();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-08-24 04:31:27 +08:00
|
|
|
bool MIParser::parseDIExpression(MDNode *&Expr) {
|
|
|
|
assert(Token.is(MIToken::md_diexpr));
|
|
|
|
lex();
|
|
|
|
|
|
|
|
// FIXME: Share this parsing with the IL parser.
|
|
|
|
SmallVector<uint64_t, 8> Elements;
|
|
|
|
|
|
|
|
if (expectAndConsume(MIToken::lparen))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (Token.isNot(MIToken::rparen)) {
|
|
|
|
do {
|
|
|
|
if (Token.is(MIToken::Identifier)) {
|
|
|
|
if (unsigned Op = dwarf::getOperationEncoding(Token.stringValue())) {
|
|
|
|
lex();
|
|
|
|
Elements.push_back(Op);
|
|
|
|
continue;
|
|
|
|
}
|
2019-04-30 15:58:57 +08:00
|
|
|
if (unsigned Enc = dwarf::getAttributeEncoding(Token.stringValue())) {
|
|
|
|
lex();
|
|
|
|
Elements.push_back(Enc);
|
|
|
|
continue;
|
|
|
|
}
|
2017-08-24 04:31:27 +08:00
|
|
|
return error(Twine("invalid DWARF op '") + Token.stringValue() + "'");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Token.isNot(MIToken::IntegerLiteral) ||
|
|
|
|
Token.integerValue().isSigned())
|
|
|
|
return error("expected unsigned integer");
|
|
|
|
|
|
|
|
auto &U = Token.integerValue();
|
|
|
|
if (U.ugt(UINT64_MAX))
|
|
|
|
return error("element too large, limit is " + Twine(UINT64_MAX));
|
|
|
|
Elements.push_back(U.getZExtValue());
|
|
|
|
lex();
|
|
|
|
|
|
|
|
} while (consumeIfPresent(MIToken::comma));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (expectAndConsume(MIToken::rparen))
|
|
|
|
return true;
|
|
|
|
|
2017-12-16 06:22:58 +08:00
|
|
|
Expr = DIExpression::get(MF.getFunction().getContext(), Elements);
|
2017-08-24 04:31:27 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
[mir] Serialize DILocation inline when not possible to use a metadata reference
Summary:
Sometimes MIR-level passes create DILocations that were not present in the
LLVM-IR. For example, it may merge two DILocations together to produce a
DILocation that points to line 0.
Previously, the address of these DILocations were printed which prevented the
MIR from being read back into LLVM. With this patch, DILocations will use
metadata references where possible and fall back on serializing them inline like so:
MOV32mr %stack.0.x.addr, 1, _, 0, _, %0, debug-location !DILocation(line: 1, scope: !15)
Reviewers: aprantl, vsk, arphaman
Reviewed By: aprantl
Subscribers: probinson, llvm-commits
Tags: #debug-info
Differential Revision: https://reviews.llvm.org/D55243
llvm-svn: 349035
2018-12-13 22:25:27 +08:00
|
|
|
bool MIParser::parseDILocation(MDNode *&Loc) {
|
|
|
|
assert(Token.is(MIToken::md_dilocation));
|
|
|
|
lex();
|
|
|
|
|
|
|
|
bool HaveLine = false;
|
|
|
|
unsigned Line = 0;
|
|
|
|
unsigned Column = 0;
|
|
|
|
MDNode *Scope = nullptr;
|
|
|
|
MDNode *InlinedAt = nullptr;
|
2018-12-13 23:05:27 +08:00
|
|
|
bool ImplicitCode = false;
|
[mir] Serialize DILocation inline when not possible to use a metadata reference
Summary:
Sometimes MIR-level passes create DILocations that were not present in the
LLVM-IR. For example, it may merge two DILocations together to produce a
DILocation that points to line 0.
Previously, the address of these DILocations were printed which prevented the
MIR from being read back into LLVM. With this patch, DILocations will use
metadata references where possible and fall back on serializing them inline like so:
MOV32mr %stack.0.x.addr, 1, _, 0, _, %0, debug-location !DILocation(line: 1, scope: !15)
Reviewers: aprantl, vsk, arphaman
Reviewed By: aprantl
Subscribers: probinson, llvm-commits
Tags: #debug-info
Differential Revision: https://reviews.llvm.org/D55243
llvm-svn: 349035
2018-12-13 22:25:27 +08:00
|
|
|
|
|
|
|
if (expectAndConsume(MIToken::lparen))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (Token.isNot(MIToken::rparen)) {
|
|
|
|
do {
|
|
|
|
if (Token.is(MIToken::Identifier)) {
|
|
|
|
if (Token.stringValue() == "line") {
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::colon))
|
|
|
|
return true;
|
|
|
|
if (Token.isNot(MIToken::IntegerLiteral) ||
|
|
|
|
Token.integerValue().isSigned())
|
|
|
|
return error("expected unsigned integer");
|
|
|
|
Line = Token.integerValue().getZExtValue();
|
|
|
|
HaveLine = true;
|
|
|
|
lex();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (Token.stringValue() == "column") {
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::colon))
|
|
|
|
return true;
|
|
|
|
if (Token.isNot(MIToken::IntegerLiteral) ||
|
|
|
|
Token.integerValue().isSigned())
|
|
|
|
return error("expected unsigned integer");
|
|
|
|
Column = Token.integerValue().getZExtValue();
|
|
|
|
lex();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (Token.stringValue() == "scope") {
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::colon))
|
|
|
|
return true;
|
|
|
|
if (parseMDNode(Scope))
|
|
|
|
return error("expected metadata node");
|
|
|
|
if (!isa<DIScope>(Scope))
|
|
|
|
return error("expected DIScope node");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (Token.stringValue() == "inlinedAt") {
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::colon))
|
|
|
|
return true;
|
|
|
|
if (Token.is(MIToken::exclaim)) {
|
|
|
|
if (parseMDNode(InlinedAt))
|
|
|
|
return true;
|
|
|
|
} else if (Token.is(MIToken::md_dilocation)) {
|
|
|
|
if (parseDILocation(InlinedAt))
|
|
|
|
return true;
|
|
|
|
} else
|
|
|
|
return error("expected metadata node");
|
|
|
|
if (!isa<DILocation>(InlinedAt))
|
|
|
|
return error("expected DILocation node");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (Token.stringValue() == "isImplicitCode") {
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::colon))
|
|
|
|
return true;
|
|
|
|
if (!Token.is(MIToken::Identifier))
|
|
|
|
return error("expected true/false");
|
|
|
|
// As far as I can see, we don't have any existing need for parsing
|
|
|
|
// true/false in MIR yet. Do it ad-hoc until there's something else
|
|
|
|
// that needs it.
|
|
|
|
if (Token.stringValue() == "true")
|
|
|
|
ImplicitCode = true;
|
|
|
|
else if (Token.stringValue() == "false")
|
|
|
|
ImplicitCode = false;
|
|
|
|
else
|
|
|
|
return error("expected true/false");
|
|
|
|
lex();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return error(Twine("invalid DILocation argument '") +
|
|
|
|
Token.stringValue() + "'");
|
|
|
|
} while (consumeIfPresent(MIToken::comma));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (expectAndConsume(MIToken::rparen))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (!HaveLine)
|
|
|
|
return error("DILocation requires line number");
|
|
|
|
if (!Scope)
|
|
|
|
return error("DILocation requires a scope");
|
|
|
|
|
|
|
|
Loc = DILocation::get(MF.getFunction().getContext(), Line, Column, Scope,
|
|
|
|
InlinedAt, ImplicitCode);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-23 05:07:04 +08:00
|
|
|
bool MIParser::parseMetadataOperand(MachineOperand &Dest) {
|
|
|
|
MDNode *Node = nullptr;
|
2017-08-24 04:31:27 +08:00
|
|
|
if (Token.is(MIToken::exclaim)) {
|
|
|
|
if (parseMDNode(Node))
|
|
|
|
return true;
|
|
|
|
} else if (Token.is(MIToken::md_diexpr)) {
|
|
|
|
if (parseDIExpression(Node))
|
|
|
|
return true;
|
|
|
|
}
|
2015-07-23 05:07:04 +08:00
|
|
|
Dest = MachineOperand::CreateMetadata(Node);
|
2015-07-23 01:58:46 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-22 06:28:27 +08:00
|
|
|
bool MIParser::parseCFIOffset(int &Offset) {
|
|
|
|
if (Token.isNot(MIToken::IntegerLiteral))
|
|
|
|
return error("expected a cfi offset");
|
|
|
|
if (Token.integerValue().getMinSignedBits() > 32)
|
|
|
|
return error("expected a 32 bit integer (the cfi offset is too large)");
|
|
|
|
Offset = (int)Token.integerValue().getExtValue();
|
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-24 07:09:07 +08:00
|
|
|
bool MIParser::parseCFIRegister(unsigned &Reg) {
|
|
|
|
if (Token.isNot(MIToken::NamedRegister))
|
|
|
|
return error("expected a cfi register");
|
|
|
|
unsigned LLVMReg;
|
2016-10-11 11:13:01 +08:00
|
|
|
if (parseNamedRegister(LLVMReg))
|
2015-07-24 07:09:07 +08:00
|
|
|
return true;
|
|
|
|
const auto *TRI = MF.getSubtarget().getRegisterInfo();
|
|
|
|
assert(TRI && "Expected target register info");
|
|
|
|
int DwarfReg = TRI->getDwarfRegNum(LLVMReg, true);
|
|
|
|
if (DwarfReg < 0)
|
|
|
|
return error("invalid DWARF register");
|
|
|
|
Reg = (unsigned)DwarfReg;
|
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-12-15 23:17:18 +08:00
|
|
|
bool MIParser::parseCFIEscapeValues(std::string &Values) {
|
|
|
|
do {
|
|
|
|
if (Token.isNot(MIToken::HexLiteral))
|
|
|
|
return error("expected a hexadecimal literal");
|
|
|
|
unsigned Value;
|
|
|
|
if (getUnsigned(Value))
|
|
|
|
return true;
|
|
|
|
if (Value > UINT8_MAX)
|
|
|
|
return error("expected a 8-bit integer (too large)");
|
|
|
|
Values.push_back(static_cast<uint8_t>(Value));
|
|
|
|
lex();
|
|
|
|
} while (consumeIfPresent(MIToken::comma));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-22 06:28:27 +08:00
|
|
|
bool MIParser::parseCFIOperand(MachineOperand &Dest) {
|
2015-07-24 07:09:07 +08:00
|
|
|
auto Kind = Token.kind();
|
2015-07-22 06:28:27 +08:00
|
|
|
lex();
|
|
|
|
int Offset;
|
2015-07-24 07:09:07 +08:00
|
|
|
unsigned Reg;
|
|
|
|
unsigned CFIIndex;
|
|
|
|
switch (Kind) {
|
2015-08-15 05:55:58 +08:00
|
|
|
case MIToken::kw_cfi_same_value:
|
|
|
|
if (parseCFIRegister(Reg))
|
|
|
|
return true;
|
2016-12-01 07:48:42 +08:00
|
|
|
CFIIndex = MF.addFrameInst(MCCFIInstruction::createSameValue(nullptr, Reg));
|
2015-08-15 05:55:58 +08:00
|
|
|
break;
|
2015-07-24 07:09:07 +08:00
|
|
|
case MIToken::kw_cfi_offset:
|
|
|
|
if (parseCFIRegister(Reg) || expectAndConsume(MIToken::comma) ||
|
|
|
|
parseCFIOffset(Offset))
|
|
|
|
return true;
|
|
|
|
CFIIndex =
|
2016-12-01 07:48:42 +08:00
|
|
|
MF.addFrameInst(MCCFIInstruction::createOffset(nullptr, Reg, Offset));
|
2015-07-24 07:09:07 +08:00
|
|
|
break;
|
2017-12-15 23:17:18 +08:00
|
|
|
case MIToken::kw_cfi_rel_offset:
|
|
|
|
if (parseCFIRegister(Reg) || expectAndConsume(MIToken::comma) ||
|
|
|
|
parseCFIOffset(Offset))
|
|
|
|
return true;
|
|
|
|
CFIIndex = MF.addFrameInst(
|
|
|
|
MCCFIInstruction::createRelOffset(nullptr, Reg, Offset));
|
|
|
|
break;
|
2015-07-28 04:39:03 +08:00
|
|
|
case MIToken::kw_cfi_def_cfa_register:
|
|
|
|
if (parseCFIRegister(Reg))
|
|
|
|
return true;
|
|
|
|
CFIIndex =
|
2016-12-01 07:48:42 +08:00
|
|
|
MF.addFrameInst(MCCFIInstruction::createDefCfaRegister(nullptr, Reg));
|
2015-07-28 04:39:03 +08:00
|
|
|
break;
|
2015-07-24 07:09:07 +08:00
|
|
|
case MIToken::kw_cfi_def_cfa_offset:
|
|
|
|
if (parseCFIOffset(Offset))
|
|
|
|
return true;
|
|
|
|
// NB: MCCFIInstruction::createDefCfaOffset negates the offset.
|
2016-12-01 07:48:42 +08:00
|
|
|
CFIIndex = MF.addFrameInst(
|
2015-07-24 07:09:07 +08:00
|
|
|
MCCFIInstruction::createDefCfaOffset(nullptr, -Offset));
|
|
|
|
break;
|
2017-12-15 23:17:18 +08:00
|
|
|
case MIToken::kw_cfi_adjust_cfa_offset:
|
|
|
|
if (parseCFIOffset(Offset))
|
|
|
|
return true;
|
|
|
|
CFIIndex = MF.addFrameInst(
|
|
|
|
MCCFIInstruction::createAdjustCfaOffset(nullptr, Offset));
|
|
|
|
break;
|
2015-07-30 02:57:23 +08:00
|
|
|
case MIToken::kw_cfi_def_cfa:
|
|
|
|
if (parseCFIRegister(Reg) || expectAndConsume(MIToken::comma) ||
|
|
|
|
parseCFIOffset(Offset))
|
|
|
|
return true;
|
|
|
|
// NB: MCCFIInstruction::createDefCfa negates the offset.
|
|
|
|
CFIIndex =
|
2016-12-01 07:48:42 +08:00
|
|
|
MF.addFrameInst(MCCFIInstruction::createDefCfa(nullptr, Reg, -Offset));
|
2015-07-30 02:57:23 +08:00
|
|
|
break;
|
2017-12-15 23:17:18 +08:00
|
|
|
case MIToken::kw_cfi_remember_state:
|
|
|
|
CFIIndex = MF.addFrameInst(MCCFIInstruction::createRememberState(nullptr));
|
|
|
|
break;
|
2017-11-02 20:00:58 +08:00
|
|
|
case MIToken::kw_cfi_restore:
|
|
|
|
if (parseCFIRegister(Reg))
|
|
|
|
return true;
|
|
|
|
CFIIndex = MF.addFrameInst(MCCFIInstruction::createRestore(nullptr, Reg));
|
|
|
|
break;
|
2017-12-15 23:17:18 +08:00
|
|
|
case MIToken::kw_cfi_restore_state:
|
|
|
|
CFIIndex = MF.addFrameInst(MCCFIInstruction::createRestoreState(nullptr));
|
|
|
|
break;
|
|
|
|
case MIToken::kw_cfi_undefined:
|
|
|
|
if (parseCFIRegister(Reg))
|
|
|
|
return true;
|
|
|
|
CFIIndex = MF.addFrameInst(MCCFIInstruction::createUndefined(nullptr, Reg));
|
|
|
|
break;
|
|
|
|
case MIToken::kw_cfi_register: {
|
|
|
|
unsigned Reg2;
|
|
|
|
if (parseCFIRegister(Reg) || expectAndConsume(MIToken::comma) ||
|
|
|
|
parseCFIRegister(Reg2))
|
|
|
|
return true;
|
2017-11-02 20:00:58 +08:00
|
|
|
|
2017-12-15 23:17:18 +08:00
|
|
|
CFIIndex =
|
|
|
|
MF.addFrameInst(MCCFIInstruction::createRegister(nullptr, Reg, Reg2));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case MIToken::kw_cfi_window_save:
|
|
|
|
CFIIndex = MF.addFrameInst(MCCFIInstruction::createWindowSave(nullptr));
|
|
|
|
break;
|
2018-12-18 18:37:42 +08:00
|
|
|
case MIToken::kw_cfi_aarch64_negate_ra_sign_state:
|
|
|
|
CFIIndex = MF.addFrameInst(MCCFIInstruction::createNegateRAState(nullptr));
|
|
|
|
break;
|
2017-12-15 23:17:18 +08:00
|
|
|
case MIToken::kw_cfi_escape: {
|
|
|
|
std::string Values;
|
|
|
|
if (parseCFIEscapeValues(Values))
|
|
|
|
return true;
|
|
|
|
CFIIndex = MF.addFrameInst(MCCFIInstruction::createEscape(nullptr, Values));
|
|
|
|
break;
|
|
|
|
}
|
2015-07-24 07:09:07 +08:00
|
|
|
default:
|
|
|
|
// TODO: Parse the other CFI operands.
|
|
|
|
llvm_unreachable("The current token should be a cfi operand");
|
|
|
|
}
|
|
|
|
Dest = MachineOperand::CreateCFIIndex(CFIIndex);
|
2015-07-22 06:28:27 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-29 01:28:03 +08:00
|
|
|
bool MIParser::parseIRBlock(BasicBlock *&BB, const Function &F) {
|
|
|
|
switch (Token.kind()) {
|
2015-08-06 01:35:55 +08:00
|
|
|
case MIToken::NamedIRBlock: {
|
|
|
|
BB = dyn_cast_or_null<BasicBlock>(
|
2016-09-17 14:00:02 +08:00
|
|
|
F.getValueSymbolTable()->lookup(Token.stringValue()));
|
2015-07-29 01:28:03 +08:00
|
|
|
if (!BB)
|
2015-08-07 07:17:42 +08:00
|
|
|
return error(Twine("use of undefined IR block '") + Token.range() + "'");
|
2015-07-29 01:28:03 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case MIToken::IRBlock: {
|
|
|
|
unsigned SlotNumber = 0;
|
|
|
|
if (getUnsigned(SlotNumber))
|
|
|
|
return true;
|
2015-08-07 07:57:04 +08:00
|
|
|
BB = const_cast<BasicBlock *>(getIRBlock(SlotNumber, F));
|
2015-07-29 01:28:03 +08:00
|
|
|
if (!BB)
|
|
|
|
return error(Twine("use of undefined IR block '%ir-block.") +
|
|
|
|
Twine(SlotNumber) + "'");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
llvm_unreachable("The current token should be an IR block reference");
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::parseBlockAddressOperand(MachineOperand &Dest) {
|
|
|
|
assert(Token.is(MIToken::kw_blockaddress));
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::lparen))
|
|
|
|
return true;
|
|
|
|
if (Token.isNot(MIToken::GlobalValue) &&
|
2015-08-06 01:35:55 +08:00
|
|
|
Token.isNot(MIToken::NamedGlobalValue))
|
2015-07-29 01:28:03 +08:00
|
|
|
return error("expected a global value");
|
|
|
|
GlobalValue *GV = nullptr;
|
|
|
|
if (parseGlobalValue(GV))
|
|
|
|
return true;
|
|
|
|
auto *F = dyn_cast<Function>(GV);
|
|
|
|
if (!F)
|
|
|
|
return error("expected an IR function reference");
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::comma))
|
|
|
|
return true;
|
|
|
|
BasicBlock *BB = nullptr;
|
2015-08-06 01:35:55 +08:00
|
|
|
if (Token.isNot(MIToken::IRBlock) && Token.isNot(MIToken::NamedIRBlock))
|
2015-07-29 01:28:03 +08:00
|
|
|
return error("expected an IR block reference");
|
|
|
|
if (parseIRBlock(BB, *F))
|
|
|
|
return true;
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::rparen))
|
|
|
|
return true;
|
|
|
|
Dest = MachineOperand::CreateBA(BlockAddress::get(F, BB), /*Offset=*/0);
|
2015-08-06 06:26:15 +08:00
|
|
|
if (parseOperandsOffset(Dest))
|
|
|
|
return true;
|
2015-07-29 01:28:03 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-07-30 04:32:59 +08:00
|
|
|
bool MIParser::parseIntrinsicOperand(MachineOperand &Dest) {
|
|
|
|
assert(Token.is(MIToken::kw_intrinsic));
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::lparen))
|
|
|
|
return error("expected syntax intrinsic(@llvm.whatever)");
|
|
|
|
|
|
|
|
if (Token.isNot(MIToken::NamedGlobalValue))
|
|
|
|
return error("expected syntax intrinsic(@llvm.whatever)");
|
|
|
|
|
|
|
|
std::string Name = Token.stringValue();
|
|
|
|
lex();
|
|
|
|
|
|
|
|
if (expectAndConsume(MIToken::rparen))
|
|
|
|
return error("expected ')' to terminate intrinsic name");
|
|
|
|
|
|
|
|
// Find out what intrinsic we're dealing with, first try the global namespace
|
|
|
|
// and then the target's private intrinsics if that fails.
|
|
|
|
const TargetIntrinsicInfo *TII = MF.getTarget().getIntrinsicInfo();
|
|
|
|
Intrinsic::ID ID = Function::lookupIntrinsicID(Name);
|
|
|
|
if (ID == Intrinsic::not_intrinsic && TII)
|
|
|
|
ID = static_cast<Intrinsic::ID>(TII->lookupName(Name));
|
|
|
|
|
|
|
|
if (ID == Intrinsic::not_intrinsic)
|
|
|
|
return error("unknown intrinsic name");
|
|
|
|
Dest = MachineOperand::CreateIntrinsicID(ID);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-08-18 04:25:25 +08:00
|
|
|
bool MIParser::parsePredicateOperand(MachineOperand &Dest) {
|
|
|
|
assert(Token.is(MIToken::kw_intpred) || Token.is(MIToken::kw_floatpred));
|
|
|
|
bool IsFloat = Token.is(MIToken::kw_floatpred);
|
|
|
|
lex();
|
|
|
|
|
|
|
|
if (expectAndConsume(MIToken::lparen))
|
|
|
|
return error("expected syntax intpred(whatever) or floatpred(whatever");
|
|
|
|
|
|
|
|
if (Token.isNot(MIToken::Identifier))
|
|
|
|
return error("whatever");
|
|
|
|
|
|
|
|
CmpInst::Predicate Pred;
|
|
|
|
if (IsFloat) {
|
|
|
|
Pred = StringSwitch<CmpInst::Predicate>(Token.stringValue())
|
|
|
|
.Case("false", CmpInst::FCMP_FALSE)
|
|
|
|
.Case("oeq", CmpInst::FCMP_OEQ)
|
|
|
|
.Case("ogt", CmpInst::FCMP_OGT)
|
|
|
|
.Case("oge", CmpInst::FCMP_OGE)
|
|
|
|
.Case("olt", CmpInst::FCMP_OLT)
|
|
|
|
.Case("ole", CmpInst::FCMP_OLE)
|
|
|
|
.Case("one", CmpInst::FCMP_ONE)
|
|
|
|
.Case("ord", CmpInst::FCMP_ORD)
|
|
|
|
.Case("uno", CmpInst::FCMP_UNO)
|
|
|
|
.Case("ueq", CmpInst::FCMP_UEQ)
|
|
|
|
.Case("ugt", CmpInst::FCMP_UGT)
|
|
|
|
.Case("uge", CmpInst::FCMP_UGE)
|
|
|
|
.Case("ult", CmpInst::FCMP_ULT)
|
|
|
|
.Case("ule", CmpInst::FCMP_ULE)
|
|
|
|
.Case("une", CmpInst::FCMP_UNE)
|
|
|
|
.Case("true", CmpInst::FCMP_TRUE)
|
|
|
|
.Default(CmpInst::BAD_FCMP_PREDICATE);
|
|
|
|
if (!CmpInst::isFPPredicate(Pred))
|
|
|
|
return error("invalid floating-point predicate");
|
|
|
|
} else {
|
|
|
|
Pred = StringSwitch<CmpInst::Predicate>(Token.stringValue())
|
|
|
|
.Case("eq", CmpInst::ICMP_EQ)
|
|
|
|
.Case("ne", CmpInst::ICMP_NE)
|
|
|
|
.Case("sgt", CmpInst::ICMP_SGT)
|
|
|
|
.Case("sge", CmpInst::ICMP_SGE)
|
|
|
|
.Case("slt", CmpInst::ICMP_SLT)
|
|
|
|
.Case("sle", CmpInst::ICMP_SLE)
|
|
|
|
.Case("ugt", CmpInst::ICMP_UGT)
|
|
|
|
.Case("uge", CmpInst::ICMP_UGE)
|
|
|
|
.Case("ult", CmpInst::ICMP_ULT)
|
|
|
|
.Case("ule", CmpInst::ICMP_ULE)
|
|
|
|
.Default(CmpInst::BAD_ICMP_PREDICATE);
|
|
|
|
if (!CmpInst::isIntPredicate(Pred))
|
|
|
|
return error("invalid integer predicate");
|
|
|
|
}
|
|
|
|
|
|
|
|
lex();
|
|
|
|
Dest = MachineOperand::CreatePredicate(Pred);
|
2016-08-24 05:01:26 +08:00
|
|
|
if (expectAndConsume(MIToken::rparen))
|
2016-08-18 04:25:25 +08:00
|
|
|
return error("predicate should be terminated by ')'.");
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-29 07:02:45 +08:00
|
|
|
bool MIParser::parseTargetIndexOperand(MachineOperand &Dest) {
|
|
|
|
assert(Token.is(MIToken::kw_target_index));
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::lparen))
|
|
|
|
return true;
|
|
|
|
if (Token.isNot(MIToken::Identifier))
|
|
|
|
return error("expected the name of the target index");
|
|
|
|
int Index = 0;
|
2019-03-13 04:42:12 +08:00
|
|
|
if (PFS.Target.getTargetIndex(Token.stringValue(), Index))
|
2015-07-29 07:02:45 +08:00
|
|
|
return error("use of undefined target index '" + Token.stringValue() + "'");
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::rparen))
|
|
|
|
return true;
|
|
|
|
Dest = MachineOperand::CreateTargetIndex(unsigned(Index), /*Offset=*/0);
|
2015-08-06 06:26:15 +08:00
|
|
|
if (parseOperandsOffset(Dest))
|
|
|
|
return true;
|
2015-07-29 07:02:45 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-03-19 16:14:18 +08:00
|
|
|
bool MIParser::parseCustomRegisterMaskOperand(MachineOperand &Dest) {
|
|
|
|
assert(Token.stringValue() == "CustomRegMask" && "Expected a custom RegMask");
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::lparen))
|
|
|
|
return true;
|
|
|
|
|
2018-07-26 08:27:47 +08:00
|
|
|
uint32_t *Mask = MF.allocateRegMask();
|
2017-03-19 16:14:18 +08:00
|
|
|
while (true) {
|
|
|
|
if (Token.isNot(MIToken::NamedRegister))
|
|
|
|
return error("expected a named register");
|
|
|
|
unsigned Reg;
|
|
|
|
if (parseNamedRegister(Reg))
|
|
|
|
return true;
|
|
|
|
lex();
|
|
|
|
Mask[Reg / 32] |= 1U << (Reg % 32);
|
|
|
|
// TODO: Report an error if the same register is used more than once.
|
|
|
|
if (Token.isNot(MIToken::comma))
|
|
|
|
break;
|
|
|
|
lex();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (expectAndConsume(MIToken::rparen))
|
|
|
|
return true;
|
|
|
|
Dest = MachineOperand::CreateRegMask(Mask);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-11 07:24:42 +08:00
|
|
|
bool MIParser::parseLiveoutRegisterMaskOperand(MachineOperand &Dest) {
|
|
|
|
assert(Token.is(MIToken::kw_liveout));
|
2018-07-26 08:27:47 +08:00
|
|
|
uint32_t *Mask = MF.allocateRegMask();
|
2015-08-11 07:24:42 +08:00
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::lparen))
|
|
|
|
return true;
|
|
|
|
while (true) {
|
|
|
|
if (Token.isNot(MIToken::NamedRegister))
|
|
|
|
return error("expected a named register");
|
2016-10-11 11:13:01 +08:00
|
|
|
unsigned Reg;
|
|
|
|
if (parseNamedRegister(Reg))
|
2015-08-11 07:24:42 +08:00
|
|
|
return true;
|
|
|
|
lex();
|
|
|
|
Mask[Reg / 32] |= 1U << (Reg % 32);
|
|
|
|
// TODO: Report an error if the same register is used more than once.
|
|
|
|
if (Token.isNot(MIToken::comma))
|
|
|
|
break;
|
|
|
|
lex();
|
|
|
|
}
|
|
|
|
if (expectAndConsume(MIToken::rparen))
|
|
|
|
return true;
|
|
|
|
Dest = MachineOperand::CreateRegLiveOut(Mask);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-20 03:05:34 +08:00
|
|
|
bool MIParser::parseMachineOperand(MachineOperand &Dest,
|
|
|
|
Optional<unsigned> &TiedDefIdx) {
|
2015-06-24 00:35:26 +08:00
|
|
|
switch (Token.kind()) {
|
2015-07-07 07:07:26 +08:00
|
|
|
case MIToken::kw_implicit:
|
|
|
|
case MIToken::kw_implicit_define:
|
2015-08-20 02:55:47 +08:00
|
|
|
case MIToken::kw_def:
|
2015-07-08 04:34:53 +08:00
|
|
|
case MIToken::kw_dead:
|
2015-07-09 05:23:34 +08:00
|
|
|
case MIToken::kw_killed:
|
2015-07-09 07:58:31 +08:00
|
|
|
case MIToken::kw_undef:
|
2015-08-15 03:07:07 +08:00
|
|
|
case MIToken::kw_internal:
|
2015-08-06 01:49:03 +08:00
|
|
|
case MIToken::kw_early_clobber:
|
2015-08-06 01:41:17 +08:00
|
|
|
case MIToken::kw_debug_use:
|
2017-12-13 01:53:59 +08:00
|
|
|
case MIToken::kw_renamable:
|
2015-06-25 01:34:58 +08:00
|
|
|
case MIToken::underscore:
|
2015-06-24 00:35:26 +08:00
|
|
|
case MIToken::NamedRegister:
|
2015-07-11 06:51:20 +08:00
|
|
|
case MIToken::VirtualRegister:
|
2018-03-31 02:15:54 +08:00
|
|
|
case MIToken::NamedVirtualRegister:
|
2015-08-20 03:05:34 +08:00
|
|
|
return parseRegisterOperand(Dest, TiedDefIdx);
|
2015-06-24 07:42:28 +08:00
|
|
|
case MIToken::IntegerLiteral:
|
|
|
|
return parseImmediateOperand(Dest);
|
2015-08-01 04:49:21 +08:00
|
|
|
case MIToken::kw_half:
|
|
|
|
case MIToken::kw_float:
|
|
|
|
case MIToken::kw_double:
|
|
|
|
case MIToken::kw_x86_fp80:
|
|
|
|
case MIToken::kw_fp128:
|
|
|
|
case MIToken::kw_ppc_fp128:
|
|
|
|
return parseFPImmediateOperand(Dest);
|
2015-06-27 00:46:11 +08:00
|
|
|
case MIToken::MachineBasicBlock:
|
|
|
|
return parseMBBOperand(Dest);
|
2015-07-17 07:37:45 +08:00
|
|
|
case MIToken::StackObject:
|
|
|
|
return parseStackObjectOperand(Dest);
|
|
|
|
case MIToken::FixedStackObject:
|
|
|
|
return parseFixedStackObjectOperand(Dest);
|
2015-06-27 06:56:48 +08:00
|
|
|
case MIToken::GlobalValue:
|
|
|
|
case MIToken::NamedGlobalValue:
|
|
|
|
return parseGlobalAddressOperand(Dest);
|
2015-07-21 04:51:18 +08:00
|
|
|
case MIToken::ConstantPoolItem:
|
|
|
|
return parseConstantPoolIndexOperand(Dest);
|
2015-07-16 07:38:35 +08:00
|
|
|
case MIToken::JumpTableIndex:
|
|
|
|
return parseJumpTableIndexOperand(Dest);
|
2015-07-22 00:59:53 +08:00
|
|
|
case MIToken::ExternalSymbol:
|
|
|
|
return parseExternalSymbolOperand(Dest);
|
[x86/MIR] Implement support for pre- and post-instruction symbols, as
well as MIR parsing support for `MCSymbol` `MachineOperand`s.
The only real way to test pre- and post-instruction symbol support is to
use them in operands, so I ended up implementing that within the patch
as well. I can split out the operand support if folks really want but it
doesn't really seem worth it.
The functional implementation of pre- and post-instruction symbols is
now *completely trivial*. Two tiny bits of code in the (misnamed)
AsmPrinter. It should be completely target independent as well. We emit
these exactly the same way as we emit basic block labels. Most of the
code here is to give full dumping, MIR printing, and MIR parsing support
so that we can write useful tests.
The MIR parsing of MC symbol operands still isn't 100%, as it forces the
symbols to be non-temporary and non-local symbols with names. However,
those names often can encode most (if not all) of the special semantics
desired, and unnamed symbols seem especially annoying to serialize and
de-serialize. While this isn't perfect or full support, it seems plenty
to write tests that exercise usage of these kinds of operands.
The MIR support for pre-and post-instruction symbols was quite
straightforward. I chose to print them out in an as-if-operand syntax
similar to debug locations as this seemed the cleanest way and let me
use nice introducer tokens rather than inventing more magic punctuation
like we use for memoperands.
However, supporting MIR-based parsing of these symbols caused me to
change the design of the symbol support to allow setting arbitrary
symbols. Without this, I don't see any reasonable way to test things
with MIR.
Differential Revision: https://reviews.llvm.org/D50833
llvm-svn: 339962
2018-08-17 07:11:05 +08:00
|
|
|
case MIToken::MCSymbol:
|
|
|
|
return parseMCSymbolOperand(Dest);
|
2016-03-29 02:18:46 +08:00
|
|
|
case MIToken::SubRegisterIndex:
|
|
|
|
return parseSubRegisterIndexOperand(Dest);
|
2017-08-24 04:31:27 +08:00
|
|
|
case MIToken::md_diexpr:
|
2015-07-23 01:58:46 +08:00
|
|
|
case MIToken::exclaim:
|
|
|
|
return parseMetadataOperand(Dest);
|
2015-08-15 05:55:58 +08:00
|
|
|
case MIToken::kw_cfi_same_value:
|
2015-07-24 07:09:07 +08:00
|
|
|
case MIToken::kw_cfi_offset:
|
2017-12-15 23:17:18 +08:00
|
|
|
case MIToken::kw_cfi_rel_offset:
|
2015-07-28 04:39:03 +08:00
|
|
|
case MIToken::kw_cfi_def_cfa_register:
|
2015-07-22 06:28:27 +08:00
|
|
|
case MIToken::kw_cfi_def_cfa_offset:
|
2017-12-15 23:17:18 +08:00
|
|
|
case MIToken::kw_cfi_adjust_cfa_offset:
|
|
|
|
case MIToken::kw_cfi_escape:
|
2015-07-30 02:57:23 +08:00
|
|
|
case MIToken::kw_cfi_def_cfa:
|
2017-12-15 23:17:18 +08:00
|
|
|
case MIToken::kw_cfi_register:
|
|
|
|
case MIToken::kw_cfi_remember_state:
|
2017-11-02 20:00:58 +08:00
|
|
|
case MIToken::kw_cfi_restore:
|
2017-12-15 23:17:18 +08:00
|
|
|
case MIToken::kw_cfi_restore_state:
|
|
|
|
case MIToken::kw_cfi_undefined:
|
|
|
|
case MIToken::kw_cfi_window_save:
|
2018-12-18 18:37:42 +08:00
|
|
|
case MIToken::kw_cfi_aarch64_negate_ra_sign_state:
|
2015-07-22 06:28:27 +08:00
|
|
|
return parseCFIOperand(Dest);
|
2015-07-29 01:28:03 +08:00
|
|
|
case MIToken::kw_blockaddress:
|
|
|
|
return parseBlockAddressOperand(Dest);
|
2016-07-30 04:32:59 +08:00
|
|
|
case MIToken::kw_intrinsic:
|
|
|
|
return parseIntrinsicOperand(Dest);
|
2015-07-29 07:02:45 +08:00
|
|
|
case MIToken::kw_target_index:
|
|
|
|
return parseTargetIndexOperand(Dest);
|
2015-08-11 07:24:42 +08:00
|
|
|
case MIToken::kw_liveout:
|
|
|
|
return parseLiveoutRegisterMaskOperand(Dest);
|
2016-08-18 04:25:25 +08:00
|
|
|
case MIToken::kw_floatpred:
|
|
|
|
case MIToken::kw_intpred:
|
|
|
|
return parsePredicateOperand(Dest);
|
2015-06-24 00:35:26 +08:00
|
|
|
case MIToken::Error:
|
|
|
|
return true;
|
2015-06-30 00:57:06 +08:00
|
|
|
case MIToken::Identifier:
|
2019-03-13 04:42:12 +08:00
|
|
|
if (const auto *RegMask = PFS.Target.getRegMask(Token.stringValue())) {
|
2015-06-30 00:57:06 +08:00
|
|
|
Dest = MachineOperand::CreateRegMask(RegMask);
|
|
|
|
lex();
|
|
|
|
break;
|
2018-05-05 15:05:51 +08:00
|
|
|
} else if (Token.stringValue() == "CustomRegMask") {
|
2017-03-19 16:14:18 +08:00
|
|
|
return parseCustomRegisterMaskOperand(Dest);
|
2018-05-05 15:05:51 +08:00
|
|
|
} else
|
|
|
|
return parseTypedImmediateOperand(Dest);
|
2015-06-24 00:35:26 +08:00
|
|
|
default:
|
2015-08-22 05:12:44 +08:00
|
|
|
// FIXME: Parse the MCSymbol machine operand.
|
2015-06-24 00:35:26 +08:00
|
|
|
return error("expected a machine operand");
|
|
|
|
}
|
2015-06-23 04:37:46 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-20 03:05:34 +08:00
|
|
|
bool MIParser::parseMachineOperandAndTargetFlags(
|
|
|
|
MachineOperand &Dest, Optional<unsigned> &TiedDefIdx) {
|
2015-08-06 08:44:07 +08:00
|
|
|
unsigned TF = 0;
|
|
|
|
bool HasTargetFlags = false;
|
|
|
|
if (Token.is(MIToken::kw_target_flags)) {
|
|
|
|
HasTargetFlags = true;
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::lparen))
|
|
|
|
return true;
|
|
|
|
if (Token.isNot(MIToken::Identifier))
|
|
|
|
return error("expected the name of the target flag");
|
2019-03-13 04:42:12 +08:00
|
|
|
if (PFS.Target.getDirectTargetFlag(Token.stringValue(), TF)) {
|
|
|
|
if (PFS.Target.getBitmaskTargetFlag(Token.stringValue(), TF))
|
2015-08-19 06:52:15 +08:00
|
|
|
return error("use of undefined target flag '" + Token.stringValue() +
|
|
|
|
"'");
|
|
|
|
}
|
2015-08-06 08:44:07 +08:00
|
|
|
lex();
|
2015-08-19 06:52:15 +08:00
|
|
|
while (Token.is(MIToken::comma)) {
|
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::Identifier))
|
|
|
|
return error("expected the name of the target flag");
|
|
|
|
unsigned BitFlag = 0;
|
2019-03-13 04:42:12 +08:00
|
|
|
if (PFS.Target.getBitmaskTargetFlag(Token.stringValue(), BitFlag))
|
2015-08-19 06:52:15 +08:00
|
|
|
return error("use of undefined target flag '" + Token.stringValue() +
|
|
|
|
"'");
|
|
|
|
// TODO: Report an error when using a duplicate bit target flag.
|
|
|
|
TF |= BitFlag;
|
|
|
|
lex();
|
|
|
|
}
|
2015-08-06 08:44:07 +08:00
|
|
|
if (expectAndConsume(MIToken::rparen))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
auto Loc = Token.location();
|
2015-08-20 03:05:34 +08:00
|
|
|
if (parseMachineOperand(Dest, TiedDefIdx))
|
2015-08-06 08:44:07 +08:00
|
|
|
return true;
|
|
|
|
if (!HasTargetFlags)
|
|
|
|
return false;
|
|
|
|
if (Dest.isReg())
|
|
|
|
return error(Loc, "register operands can't have target flags");
|
|
|
|
Dest.setTargetFlags(TF);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-08 04:21:00 +08:00
|
|
|
bool MIParser::parseOffset(int64_t &Offset) {
|
2015-08-06 06:26:15 +08:00
|
|
|
if (Token.isNot(MIToken::plus) && Token.isNot(MIToken::minus))
|
|
|
|
return false;
|
2015-08-07 07:17:42 +08:00
|
|
|
StringRef Sign = Token.range();
|
2015-08-06 06:26:15 +08:00
|
|
|
bool IsNegative = Token.is(MIToken::minus);
|
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::IntegerLiteral))
|
|
|
|
return error("expected an integer literal after '" + Sign + "'");
|
|
|
|
if (Token.integerValue().getMinSignedBits() > 64)
|
|
|
|
return error("expected 64-bit integer (too large)");
|
2015-08-08 04:21:00 +08:00
|
|
|
Offset = Token.integerValue().getExtValue();
|
2015-08-06 06:26:15 +08:00
|
|
|
if (IsNegative)
|
|
|
|
Offset = -Offset;
|
|
|
|
lex();
|
2015-08-08 04:21:00 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-14 04:33:33 +08:00
|
|
|
bool MIParser::parseAlignment(unsigned &Alignment) {
|
|
|
|
assert(Token.is(MIToken::kw_align));
|
|
|
|
lex();
|
2015-08-14 04:55:01 +08:00
|
|
|
if (Token.isNot(MIToken::IntegerLiteral) || Token.integerValue().isSigned())
|
2015-08-14 04:33:33 +08:00
|
|
|
return error("expected an integer literal after 'align'");
|
|
|
|
if (getUnsigned(Alignment))
|
|
|
|
return true;
|
|
|
|
lex();
|
2019-01-31 07:09:28 +08:00
|
|
|
|
|
|
|
if (!isPowerOf2_32(Alignment))
|
|
|
|
return error("expected a power-of-2 literal after 'align'");
|
|
|
|
|
2015-08-14 04:33:33 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-01-26 19:47:28 +08:00
|
|
|
bool MIParser::parseAddrspace(unsigned &Addrspace) {
|
|
|
|
assert(Token.is(MIToken::kw_addrspace));
|
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::IntegerLiteral) || Token.integerValue().isSigned())
|
|
|
|
return error("expected an integer literal after 'addrspace'");
|
|
|
|
if (getUnsigned(Addrspace))
|
|
|
|
return true;
|
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-08 04:21:00 +08:00
|
|
|
bool MIParser::parseOperandsOffset(MachineOperand &Op) {
|
|
|
|
int64_t Offset = 0;
|
|
|
|
if (parseOffset(Offset))
|
|
|
|
return true;
|
2015-08-06 06:26:15 +08:00
|
|
|
Op.setOffset(Offset);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-20 07:27:07 +08:00
|
|
|
bool MIParser::parseIRValue(const Value *&V) {
|
2015-08-04 07:08:19 +08:00
|
|
|
switch (Token.kind()) {
|
2015-08-06 01:35:55 +08:00
|
|
|
case MIToken::NamedIRValue: {
|
2017-12-16 06:22:58 +08:00
|
|
|
V = MF.getFunction().getValueSymbolTable()->lookup(Token.stringValue());
|
2015-08-04 07:08:19 +08:00
|
|
|
break;
|
|
|
|
}
|
2015-08-20 07:31:05 +08:00
|
|
|
case MIToken::IRValue: {
|
|
|
|
unsigned SlotNumber = 0;
|
|
|
|
if (getUnsigned(SlotNumber))
|
|
|
|
return true;
|
|
|
|
V = getIRValue(SlotNumber);
|
|
|
|
break;
|
|
|
|
}
|
2015-08-20 08:20:03 +08:00
|
|
|
case MIToken::NamedGlobalValue:
|
|
|
|
case MIToken::GlobalValue: {
|
|
|
|
GlobalValue *GV = nullptr;
|
|
|
|
if (parseGlobalValue(GV))
|
|
|
|
return true;
|
|
|
|
V = GV;
|
|
|
|
break;
|
|
|
|
}
|
2015-08-22 05:54:12 +08:00
|
|
|
case MIToken::QuotedIRValue: {
|
|
|
|
const Constant *C = nullptr;
|
|
|
|
if (parseIRConstant(Token.location(), Token.stringValue(), C))
|
|
|
|
return true;
|
|
|
|
V = C;
|
|
|
|
break;
|
|
|
|
}
|
2015-08-04 07:08:19 +08:00
|
|
|
default:
|
|
|
|
llvm_unreachable("The current token should be an IR block reference");
|
|
|
|
}
|
2015-08-20 07:31:05 +08:00
|
|
|
if (!V)
|
|
|
|
return error(Twine("use of undefined IR value '") + Token.range() + "'");
|
2015-08-04 07:08:19 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::getUint64(uint64_t &Result) {
|
2016-12-16 21:58:01 +08:00
|
|
|
if (Token.hasIntegerValue()) {
|
|
|
|
if (Token.integerValue().getActiveBits() > 64)
|
|
|
|
return error("expected 64-bit integer (too large)");
|
|
|
|
Result = Token.integerValue().getZExtValue();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (Token.is(MIToken::HexLiteral)) {
|
|
|
|
APInt A;
|
|
|
|
if (getHexUint(A))
|
|
|
|
return true;
|
|
|
|
if (A.getBitWidth() > 64)
|
|
|
|
return error("expected 64-bit integer (too large)");
|
|
|
|
Result = A.getZExtValue();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::getHexUint(APInt &Result) {
|
|
|
|
assert(Token.is(MIToken::HexLiteral));
|
|
|
|
StringRef S = Token.range();
|
|
|
|
assert(S[0] == '0' && tolower(S[1]) == 'x');
|
|
|
|
// This could be a floating point literal with a special prefix.
|
|
|
|
if (!isxdigit(S[2]))
|
|
|
|
return true;
|
|
|
|
StringRef V = S.substr(2);
|
|
|
|
APInt A(V.size()*4, V, 16);
|
2017-09-02 06:17:14 +08:00
|
|
|
|
|
|
|
// If A is 0, then A.getActiveBits() is 0. This isn't a valid bitwidth. Make
|
|
|
|
// sure it isn't the case before constructing result.
|
|
|
|
unsigned NumBits = (A == 0) ? 32 : A.getActiveBits();
|
|
|
|
Result = APInt(NumBits, ArrayRef<uint64_t>(A.getRawData(), A.getNumWords()));
|
2015-08-04 07:08:19 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-07-16 02:26:59 +08:00
|
|
|
bool MIParser::parseMemoryOperandFlag(MachineMemOperand::Flags &Flags) {
|
|
|
|
const auto OldFlags = Flags;
|
2015-08-04 08:24:45 +08:00
|
|
|
switch (Token.kind()) {
|
|
|
|
case MIToken::kw_volatile:
|
|
|
|
Flags |= MachineMemOperand::MOVolatile;
|
|
|
|
break;
|
2015-08-07 00:49:30 +08:00
|
|
|
case MIToken::kw_non_temporal:
|
|
|
|
Flags |= MachineMemOperand::MONonTemporal;
|
|
|
|
break;
|
[CodeGen] Split out the notions of MI invariance and MI dereferenceability.
Summary:
An IR load can be invariant, dereferenceable, neither, or both. But
currently, MI's notion of invariance is IR-invariant &&
IR-dereferenceable.
This patch splits up the notions of invariance and dereferenceability at
the MI level. It's NFC, so adds some probably-unnecessary
"is-dereferenceable" checks, which we can remove later if desired.
Reviewers: chandlerc, tstellarAMD
Subscribers: jholewinski, arsenm, nemanjai, llvm-commits
Differential Revision: https://reviews.llvm.org/D23371
llvm-svn: 281151
2016-09-11 09:38:58 +08:00
|
|
|
case MIToken::kw_dereferenceable:
|
|
|
|
Flags |= MachineMemOperand::MODereferenceable;
|
|
|
|
break;
|
2015-08-07 00:55:53 +08:00
|
|
|
case MIToken::kw_invariant:
|
|
|
|
Flags |= MachineMemOperand::MOInvariant;
|
|
|
|
break;
|
2017-07-13 10:28:54 +08:00
|
|
|
case MIToken::StringConstant: {
|
|
|
|
MachineMemOperand::Flags TF;
|
2019-03-13 04:42:12 +08:00
|
|
|
if (PFS.Target.getMMOTargetFlag(Token.stringValue(), TF))
|
2017-07-13 10:28:54 +08:00
|
|
|
return error("use of undefined target MMO flag '" + Token.stringValue() +
|
|
|
|
"'");
|
|
|
|
Flags |= TF;
|
|
|
|
break;
|
|
|
|
}
|
2015-08-04 08:24:45 +08:00
|
|
|
default:
|
|
|
|
llvm_unreachable("The current token should be a memory operand flag");
|
|
|
|
}
|
2015-08-07 02:26:36 +08:00
|
|
|
if (OldFlags == Flags)
|
|
|
|
// We know that the same flag is specified more than once when the flags
|
|
|
|
// weren't modified.
|
|
|
|
return error("duplicate '" + Token.stringValue() + "' memory operand flag");
|
2015-08-04 08:24:45 +08:00
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-13 04:33:26 +08:00
|
|
|
bool MIParser::parseMemoryPseudoSourceValue(const PseudoSourceValue *&PSV) {
|
|
|
|
switch (Token.kind()) {
|
2015-08-13 04:44:16 +08:00
|
|
|
case MIToken::kw_stack:
|
|
|
|
PSV = MF.getPSVManager().getStack();
|
|
|
|
break;
|
2015-08-13 05:00:22 +08:00
|
|
|
case MIToken::kw_got:
|
|
|
|
PSV = MF.getPSVManager().getGOT();
|
|
|
|
break;
|
2015-08-13 05:11:08 +08:00
|
|
|
case MIToken::kw_jump_table:
|
|
|
|
PSV = MF.getPSVManager().getJumpTable();
|
|
|
|
break;
|
2015-08-13 04:33:26 +08:00
|
|
|
case MIToken::kw_constant_pool:
|
|
|
|
PSV = MF.getPSVManager().getConstantPool();
|
|
|
|
break;
|
2015-08-13 05:23:17 +08:00
|
|
|
case MIToken::FixedStackObject: {
|
|
|
|
int FI;
|
|
|
|
if (parseFixedStackFrameIndex(FI))
|
|
|
|
return true;
|
|
|
|
PSV = MF.getPSVManager().getFixedStack(FI);
|
|
|
|
// The token was already consumed, so use return here instead of break.
|
|
|
|
return false;
|
|
|
|
}
|
2016-06-08 08:47:07 +08:00
|
|
|
case MIToken::StackObject: {
|
|
|
|
int FI;
|
|
|
|
if (parseStackFrameIndex(FI))
|
|
|
|
return true;
|
|
|
|
PSV = MF.getPSVManager().getFixedStack(FI);
|
|
|
|
// The token was already consumed, so use return here instead of break.
|
|
|
|
return false;
|
|
|
|
}
|
2017-06-07 06:22:41 +08:00
|
|
|
case MIToken::kw_call_entry:
|
2015-08-20 08:12:57 +08:00
|
|
|
lex();
|
|
|
|
switch (Token.kind()) {
|
|
|
|
case MIToken::GlobalValue:
|
|
|
|
case MIToken::NamedGlobalValue: {
|
|
|
|
GlobalValue *GV = nullptr;
|
|
|
|
if (parseGlobalValue(GV))
|
|
|
|
return true;
|
|
|
|
PSV = MF.getPSVManager().getGlobalValueCallEntry(GV);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case MIToken::ExternalSymbol:
|
|
|
|
PSV = MF.getPSVManager().getExternalSymbolCallEntry(
|
|
|
|
MF.createExternalSymbolName(Token.stringValue()));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return error(
|
|
|
|
"expected a global value or an external symbol after 'call-entry'");
|
|
|
|
}
|
2015-08-15 05:08:30 +08:00
|
|
|
break;
|
2015-08-13 04:33:26 +08:00
|
|
|
default:
|
|
|
|
llvm_unreachable("The current token should be pseudo source value");
|
|
|
|
}
|
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MIParser::parseMachinePointerInfo(MachinePointerInfo &Dest) {
|
2015-08-13 05:00:22 +08:00
|
|
|
if (Token.is(MIToken::kw_constant_pool) || Token.is(MIToken::kw_stack) ||
|
2015-08-13 05:23:17 +08:00
|
|
|
Token.is(MIToken::kw_got) || Token.is(MIToken::kw_jump_table) ||
|
2016-06-08 08:47:07 +08:00
|
|
|
Token.is(MIToken::FixedStackObject) || Token.is(MIToken::StackObject) ||
|
|
|
|
Token.is(MIToken::kw_call_entry)) {
|
2015-08-13 04:33:26 +08:00
|
|
|
const PseudoSourceValue *PSV = nullptr;
|
|
|
|
if (parseMemoryPseudoSourceValue(PSV))
|
|
|
|
return true;
|
|
|
|
int64_t Offset = 0;
|
|
|
|
if (parseOffset(Offset))
|
|
|
|
return true;
|
|
|
|
Dest = MachinePointerInfo(PSV, Offset);
|
|
|
|
return false;
|
|
|
|
}
|
2015-08-20 08:20:03 +08:00
|
|
|
if (Token.isNot(MIToken::NamedIRValue) && Token.isNot(MIToken::IRValue) &&
|
|
|
|
Token.isNot(MIToken::GlobalValue) &&
|
2015-08-22 05:54:12 +08:00
|
|
|
Token.isNot(MIToken::NamedGlobalValue) &&
|
|
|
|
Token.isNot(MIToken::QuotedIRValue))
|
2015-08-13 04:33:26 +08:00
|
|
|
return error("expected an IR value reference");
|
2015-08-20 07:27:07 +08:00
|
|
|
const Value *V = nullptr;
|
2015-08-13 04:33:26 +08:00
|
|
|
if (parseIRValue(V))
|
|
|
|
return true;
|
|
|
|
if (!V->getType()->isPointerTy())
|
|
|
|
return error("expected a pointer IR value");
|
|
|
|
lex();
|
|
|
|
int64_t Offset = 0;
|
|
|
|
if (parseOffset(Offset))
|
|
|
|
return true;
|
|
|
|
Dest = MachinePointerInfo(V, Offset);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-07-12 06:23:00 +08:00
|
|
|
bool MIParser::parseOptionalScope(LLVMContext &Context,
|
|
|
|
SyncScope::ID &SSID) {
|
|
|
|
SSID = SyncScope::System;
|
|
|
|
if (Token.is(MIToken::Identifier) && Token.stringValue() == "syncscope") {
|
|
|
|
lex();
|
|
|
|
if (expectAndConsume(MIToken::lparen))
|
|
|
|
return error("expected '(' in syncscope");
|
|
|
|
|
|
|
|
std::string SSN;
|
|
|
|
if (parseStringConstant(SSN))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
SSID = Context.getOrInsertSyncScopeID(SSN);
|
|
|
|
if (expectAndConsume(MIToken::rparen))
|
|
|
|
return error("expected ')' in syncscope");
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-02-14 06:14:08 +08:00
|
|
|
bool MIParser::parseOptionalAtomicOrdering(AtomicOrdering &Order) {
|
|
|
|
Order = AtomicOrdering::NotAtomic;
|
|
|
|
if (Token.isNot(MIToken::Identifier))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
Order = StringSwitch<AtomicOrdering>(Token.stringValue())
|
|
|
|
.Case("unordered", AtomicOrdering::Unordered)
|
|
|
|
.Case("monotonic", AtomicOrdering::Monotonic)
|
|
|
|
.Case("acquire", AtomicOrdering::Acquire)
|
|
|
|
.Case("release", AtomicOrdering::Release)
|
|
|
|
.Case("acq_rel", AtomicOrdering::AcquireRelease)
|
|
|
|
.Case("seq_cst", AtomicOrdering::SequentiallyConsistent)
|
|
|
|
.Default(AtomicOrdering::NotAtomic);
|
|
|
|
|
|
|
|
if (Order != AtomicOrdering::NotAtomic) {
|
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-08-21 04:37:57 +08:00
|
|
|
return error("expected an atomic scope, ordering or a size specification");
|
2017-02-14 06:14:08 +08:00
|
|
|
}
|
|
|
|
|
2015-08-04 07:08:19 +08:00
|
|
|
bool MIParser::parseMachineMemoryOperand(MachineMemOperand *&Dest) {
|
|
|
|
if (expectAndConsume(MIToken::lparen))
|
|
|
|
return true;
|
2016-07-16 02:26:59 +08:00
|
|
|
MachineMemOperand::Flags Flags = MachineMemOperand::MONone;
|
2015-08-04 08:24:45 +08:00
|
|
|
while (Token.isMemoryOperandFlag()) {
|
|
|
|
if (parseMemoryOperandFlag(Flags))
|
|
|
|
return true;
|
|
|
|
}
|
2015-08-04 07:08:19 +08:00
|
|
|
if (Token.isNot(MIToken::Identifier) ||
|
|
|
|
(Token.stringValue() != "load" && Token.stringValue() != "store"))
|
|
|
|
return error("expected 'load' or 'store' memory operation");
|
|
|
|
if (Token.stringValue() == "load")
|
|
|
|
Flags |= MachineMemOperand::MOLoad;
|
|
|
|
else
|
|
|
|
Flags |= MachineMemOperand::MOStore;
|
|
|
|
lex();
|
|
|
|
|
2017-11-29 02:57:02 +08:00
|
|
|
// Optional 'store' for operands that both load and store.
|
|
|
|
if (Token.is(MIToken::Identifier) && Token.stringValue() == "store") {
|
|
|
|
Flags |= MachineMemOperand::MOStore;
|
|
|
|
lex();
|
|
|
|
}
|
|
|
|
|
2017-07-12 06:23:00 +08:00
|
|
|
// Optional synchronization scope.
|
|
|
|
SyncScope::ID SSID;
|
2017-12-16 06:22:58 +08:00
|
|
|
if (parseOptionalScope(MF.getFunction().getContext(), SSID))
|
2017-07-12 06:23:00 +08:00
|
|
|
return true;
|
2017-02-14 06:14:08 +08:00
|
|
|
|
|
|
|
// Up to two atomic orderings (cmpxchg provides guarantees on failure).
|
|
|
|
AtomicOrdering Order, FailureOrder;
|
|
|
|
if (parseOptionalAtomicOrdering(Order))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (parseOptionalAtomicOrdering(FailureOrder))
|
|
|
|
return true;
|
|
|
|
|
2018-08-21 04:37:57 +08:00
|
|
|
if (Token.isNot(MIToken::IntegerLiteral) &&
|
|
|
|
Token.isNot(MIToken::kw_unknown_size))
|
|
|
|
return error("expected the size integer literal or 'unknown-size' after "
|
|
|
|
"memory operation");
|
2015-08-04 07:08:19 +08:00
|
|
|
uint64_t Size;
|
2018-08-21 04:37:57 +08:00
|
|
|
if (Token.is(MIToken::IntegerLiteral)) {
|
|
|
|
if (getUint64(Size))
|
|
|
|
return true;
|
|
|
|
} else if (Token.is(MIToken::kw_unknown_size)) {
|
|
|
|
Size = MemoryLocation::UnknownSize;
|
|
|
|
}
|
2015-08-04 07:08:19 +08:00
|
|
|
lex();
|
|
|
|
|
2015-08-13 04:33:26 +08:00
|
|
|
MachinePointerInfo Ptr = MachinePointerInfo();
|
2016-06-04 08:06:31 +08:00
|
|
|
if (Token.is(MIToken::Identifier)) {
|
2017-11-29 02:57:02 +08:00
|
|
|
const char *Word =
|
|
|
|
((Flags & MachineMemOperand::MOLoad) &&
|
|
|
|
(Flags & MachineMemOperand::MOStore))
|
|
|
|
? "on"
|
|
|
|
: Flags & MachineMemOperand::MOLoad ? "from" : "into";
|
2016-06-04 08:06:31 +08:00
|
|
|
if (Token.stringValue() != Word)
|
|
|
|
return error(Twine("expected '") + Word + "'");
|
|
|
|
lex();
|
|
|
|
|
|
|
|
if (parseMachinePointerInfo(Ptr))
|
|
|
|
return true;
|
|
|
|
}
|
2018-08-21 04:37:57 +08:00
|
|
|
unsigned BaseAlignment = (Size != MemoryLocation::UnknownSize ? Size : 1);
|
2015-08-18 06:05:15 +08:00
|
|
|
AAMDNodes AAInfo;
|
2015-08-18 06:09:52 +08:00
|
|
|
MDNode *Range = nullptr;
|
2015-08-18 06:05:15 +08:00
|
|
|
while (consumeIfPresent(MIToken::comma)) {
|
|
|
|
switch (Token.kind()) {
|
|
|
|
case MIToken::kw_align:
|
|
|
|
if (parseAlignment(BaseAlignment))
|
|
|
|
return true;
|
|
|
|
break;
|
2018-01-26 19:47:28 +08:00
|
|
|
case MIToken::kw_addrspace:
|
|
|
|
if (parseAddrspace(Ptr.AddrSpace))
|
|
|
|
return true;
|
|
|
|
break;
|
2015-08-18 06:05:15 +08:00
|
|
|
case MIToken::md_tbaa:
|
|
|
|
lex();
|
|
|
|
if (parseMDNode(AAInfo.TBAA))
|
|
|
|
return true;
|
|
|
|
break;
|
2015-08-18 06:06:40 +08:00
|
|
|
case MIToken::md_alias_scope:
|
|
|
|
lex();
|
|
|
|
if (parseMDNode(AAInfo.Scope))
|
|
|
|
return true;
|
|
|
|
break;
|
2015-08-18 06:08:02 +08:00
|
|
|
case MIToken::md_noalias:
|
|
|
|
lex();
|
|
|
|
if (parseMDNode(AAInfo.NoAlias))
|
|
|
|
return true;
|
|
|
|
break;
|
2015-08-18 06:09:52 +08:00
|
|
|
case MIToken::md_range:
|
|
|
|
lex();
|
|
|
|
if (parseMDNode(Range))
|
|
|
|
return true;
|
|
|
|
break;
|
2015-08-18 06:05:15 +08:00
|
|
|
// TODO: Report an error on duplicate metadata nodes.
|
|
|
|
default:
|
2015-08-18 06:09:52 +08:00
|
|
|
return error("expected 'align' or '!tbaa' or '!alias.scope' or "
|
|
|
|
"'!noalias' or '!range'");
|
2015-08-18 06:05:15 +08:00
|
|
|
}
|
2015-08-08 04:48:30 +08:00
|
|
|
}
|
2015-08-04 07:08:19 +08:00
|
|
|
if (expectAndConsume(MIToken::rparen))
|
|
|
|
return true;
|
2017-02-14 06:14:08 +08:00
|
|
|
Dest = MF.getMachineMemOperand(Ptr, Flags, Size, BaseAlignment, AAInfo, Range,
|
2017-07-12 06:23:00 +08:00
|
|
|
SSID, Order, FailureOrder);
|
2015-08-04 07:08:19 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
[x86/MIR] Implement support for pre- and post-instruction symbols, as
well as MIR parsing support for `MCSymbol` `MachineOperand`s.
The only real way to test pre- and post-instruction symbol support is to
use them in operands, so I ended up implementing that within the patch
as well. I can split out the operand support if folks really want but it
doesn't really seem worth it.
The functional implementation of pre- and post-instruction symbols is
now *completely trivial*. Two tiny bits of code in the (misnamed)
AsmPrinter. It should be completely target independent as well. We emit
these exactly the same way as we emit basic block labels. Most of the
code here is to give full dumping, MIR printing, and MIR parsing support
so that we can write useful tests.
The MIR parsing of MC symbol operands still isn't 100%, as it forces the
symbols to be non-temporary and non-local symbols with names. However,
those names often can encode most (if not all) of the special semantics
desired, and unnamed symbols seem especially annoying to serialize and
de-serialize. While this isn't perfect or full support, it seems plenty
to write tests that exercise usage of these kinds of operands.
The MIR support for pre-and post-instruction symbols was quite
straightforward. I chose to print them out in an as-if-operand syntax
similar to debug locations as this seemed the cleanest way and let me
use nice introducer tokens rather than inventing more magic punctuation
like we use for memoperands.
However, supporting MIR-based parsing of these symbols caused me to
change the design of the symbol support to allow setting arbitrary
symbols. Without this, I don't see any reasonable way to test things
with MIR.
Differential Revision: https://reviews.llvm.org/D50833
llvm-svn: 339962
2018-08-17 07:11:05 +08:00
|
|
|
bool MIParser::parsePreOrPostInstrSymbol(MCSymbol *&Symbol) {
|
|
|
|
assert((Token.is(MIToken::kw_pre_instr_symbol) ||
|
|
|
|
Token.is(MIToken::kw_post_instr_symbol)) &&
|
|
|
|
"Invalid token for a pre- post-instruction symbol!");
|
|
|
|
lex();
|
|
|
|
if (Token.isNot(MIToken::MCSymbol))
|
|
|
|
return error("expected a symbol after 'pre-instr-symbol'");
|
|
|
|
Symbol = getOrCreateMCSymbol(Token.stringValue());
|
|
|
|
lex();
|
|
|
|
if (Token.isNewlineOrEOF() || Token.is(MIToken::coloncolon) ||
|
|
|
|
Token.is(MIToken::lbrace))
|
|
|
|
return false;
|
|
|
|
if (Token.isNot(MIToken::comma))
|
|
|
|
return error("expected ',' before the next machine operand");
|
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-08-07 07:57:04 +08:00
|
|
|
static void initSlots2BasicBlocks(
|
|
|
|
const Function &F,
|
|
|
|
DenseMap<unsigned, const BasicBlock *> &Slots2BasicBlocks) {
|
|
|
|
ModuleSlotTracker MST(F.getParent(), /*ShouldInitializeAllMetadata=*/false);
|
2015-07-28 06:42:41 +08:00
|
|
|
MST.incorporateFunction(F);
|
|
|
|
for (auto &BB : F) {
|
|
|
|
if (BB.hasName())
|
|
|
|
continue;
|
|
|
|
int Slot = MST.getLocalSlot(&BB);
|
|
|
|
if (Slot == -1)
|
|
|
|
continue;
|
|
|
|
Slots2BasicBlocks.insert(std::make_pair(unsigned(Slot), &BB));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-07 07:57:04 +08:00
|
|
|
static const BasicBlock *getIRBlockFromSlot(
|
|
|
|
unsigned Slot,
|
|
|
|
const DenseMap<unsigned, const BasicBlock *> &Slots2BasicBlocks) {
|
2015-07-28 06:42:41 +08:00
|
|
|
auto BlockInfo = Slots2BasicBlocks.find(Slot);
|
|
|
|
if (BlockInfo == Slots2BasicBlocks.end())
|
|
|
|
return nullptr;
|
|
|
|
return BlockInfo->second;
|
|
|
|
}
|
|
|
|
|
2015-08-07 07:57:04 +08:00
|
|
|
const BasicBlock *MIParser::getIRBlock(unsigned Slot) {
|
|
|
|
if (Slots2BasicBlocks.empty())
|
2017-12-16 06:22:58 +08:00
|
|
|
initSlots2BasicBlocks(MF.getFunction(), Slots2BasicBlocks);
|
2015-08-07 07:57:04 +08:00
|
|
|
return getIRBlockFromSlot(Slot, Slots2BasicBlocks);
|
|
|
|
}
|
|
|
|
|
|
|
|
const BasicBlock *MIParser::getIRBlock(unsigned Slot, const Function &F) {
|
2017-12-16 06:22:58 +08:00
|
|
|
if (&F == &MF.getFunction())
|
2015-08-07 07:57:04 +08:00
|
|
|
return getIRBlock(Slot);
|
|
|
|
DenseMap<unsigned, const BasicBlock *> CustomSlots2BasicBlocks;
|
|
|
|
initSlots2BasicBlocks(F, CustomSlots2BasicBlocks);
|
|
|
|
return getIRBlockFromSlot(Slot, CustomSlots2BasicBlocks);
|
|
|
|
}
|
|
|
|
|
2015-08-20 07:31:05 +08:00
|
|
|
static void mapValueToSlot(const Value *V, ModuleSlotTracker &MST,
|
|
|
|
DenseMap<unsigned, const Value *> &Slots2Values) {
|
|
|
|
int Slot = MST.getLocalSlot(V);
|
|
|
|
if (Slot == -1)
|
|
|
|
return;
|
|
|
|
Slots2Values.insert(std::make_pair(unsigned(Slot), V));
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates the mapping from slot numbers to function's unnamed IR values.
|
|
|
|
static void initSlots2Values(const Function &F,
|
|
|
|
DenseMap<unsigned, const Value *> &Slots2Values) {
|
|
|
|
ModuleSlotTracker MST(F.getParent(), /*ShouldInitializeAllMetadata=*/false);
|
|
|
|
MST.incorporateFunction(F);
|
|
|
|
for (const auto &Arg : F.args())
|
|
|
|
mapValueToSlot(&Arg, MST, Slots2Values);
|
|
|
|
for (const auto &BB : F) {
|
|
|
|
mapValueToSlot(&BB, MST, Slots2Values);
|
|
|
|
for (const auto &I : BB)
|
|
|
|
mapValueToSlot(&I, MST, Slots2Values);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const Value *MIParser::getIRValue(unsigned Slot) {
|
|
|
|
if (Slots2Values.empty())
|
2017-12-16 06:22:58 +08:00
|
|
|
initSlots2Values(MF.getFunction(), Slots2Values);
|
2015-08-20 07:31:05 +08:00
|
|
|
auto ValueInfo = Slots2Values.find(Slot);
|
|
|
|
if (ValueInfo == Slots2Values.end())
|
|
|
|
return nullptr;
|
|
|
|
return ValueInfo->second;
|
|
|
|
}
|
|
|
|
|
[x86/MIR] Implement support for pre- and post-instruction symbols, as
well as MIR parsing support for `MCSymbol` `MachineOperand`s.
The only real way to test pre- and post-instruction symbol support is to
use them in operands, so I ended up implementing that within the patch
as well. I can split out the operand support if folks really want but it
doesn't really seem worth it.
The functional implementation of pre- and post-instruction symbols is
now *completely trivial*. Two tiny bits of code in the (misnamed)
AsmPrinter. It should be completely target independent as well. We emit
these exactly the same way as we emit basic block labels. Most of the
code here is to give full dumping, MIR printing, and MIR parsing support
so that we can write useful tests.
The MIR parsing of MC symbol operands still isn't 100%, as it forces the
symbols to be non-temporary and non-local symbols with names. However,
those names often can encode most (if not all) of the special semantics
desired, and unnamed symbols seem especially annoying to serialize and
de-serialize. While this isn't perfect or full support, it seems plenty
to write tests that exercise usage of these kinds of operands.
The MIR support for pre-and post-instruction symbols was quite
straightforward. I chose to print them out in an as-if-operand syntax
similar to debug locations as this seemed the cleanest way and let me
use nice introducer tokens rather than inventing more magic punctuation
like we use for memoperands.
However, supporting MIR-based parsing of these symbols caused me to
change the design of the symbol support to allow setting arbitrary
symbols. Without this, I don't see any reasonable way to test things
with MIR.
Differential Revision: https://reviews.llvm.org/D50833
llvm-svn: 339962
2018-08-17 07:11:05 +08:00
|
|
|
MCSymbol *MIParser::getOrCreateMCSymbol(StringRef Name) {
|
|
|
|
// FIXME: Currently we can't recognize temporary or local symbols and call all
|
|
|
|
// of the appropriate forms to create them. However, this handles basic cases
|
|
|
|
// well as most of the special aspects are recognized by a prefix on their
|
|
|
|
// name, and the input names should already be unique. For test cases, keeping
|
|
|
|
// the symbol name out of the symbol table isn't terribly important.
|
|
|
|
return MF.getContext().getOrCreateSymbol(Name);
|
|
|
|
}
|
|
|
|
|
2017-07-12 06:23:00 +08:00
|
|
|
bool MIParser::parseStringConstant(std::string &Result) {
|
|
|
|
if (Token.isNot(MIToken::StringConstant))
|
|
|
|
return error("expected string constant");
|
|
|
|
Result = Token.stringValue();
|
|
|
|
lex();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-07-14 06:23:23 +08:00
|
|
|
bool llvm::parseMachineBasicBlockDefinitions(PerFunctionMIParsingState &PFS,
|
|
|
|
StringRef Src,
|
2015-08-14 07:10:16 +08:00
|
|
|
SMDiagnostic &Error) {
|
2016-07-14 07:27:50 +08:00
|
|
|
return MIParser(PFS, Error, Src).parseBasicBlockDefinitions(PFS.MBBSlots);
|
2015-08-14 07:10:16 +08:00
|
|
|
}
|
|
|
|
|
2016-10-11 11:13:01 +08:00
|
|
|
bool llvm::parseMachineInstructions(PerFunctionMIParsingState &PFS,
|
2016-07-14 07:27:50 +08:00
|
|
|
StringRef Src, SMDiagnostic &Error) {
|
|
|
|
return MIParser(PFS, Error, Src).parseBasicBlocks();
|
2015-06-23 01:02:30 +08:00
|
|
|
}
|
2015-07-01 02:16:42 +08:00
|
|
|
|
2016-10-11 11:13:01 +08:00
|
|
|
bool llvm::parseMBBReference(PerFunctionMIParsingState &PFS,
|
2016-07-14 07:27:50 +08:00
|
|
|
MachineBasicBlock *&MBB, StringRef Src,
|
2016-07-14 06:23:23 +08:00
|
|
|
SMDiagnostic &Error) {
|
2016-07-14 07:27:50 +08:00
|
|
|
return MIParser(PFS, Error, Src).parseStandaloneMBB(MBB);
|
2016-11-15 08:03:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool llvm::parseRegisterReference(PerFunctionMIParsingState &PFS,
|
|
|
|
unsigned &Reg, StringRef Src,
|
|
|
|
SMDiagnostic &Error) {
|
|
|
|
return MIParser(PFS, Error, Src).parseStandaloneRegister(Reg);
|
2015-07-01 02:16:42 +08:00
|
|
|
}
|
2015-07-15 05:24:41 +08:00
|
|
|
|
2016-10-11 11:13:01 +08:00
|
|
|
bool llvm::parseNamedRegisterReference(PerFunctionMIParsingState &PFS,
|
2016-07-14 07:27:50 +08:00
|
|
|
unsigned &Reg, StringRef Src,
|
2015-07-15 05:24:41 +08:00
|
|
|
SMDiagnostic &Error) {
|
2016-07-14 07:27:50 +08:00
|
|
|
return MIParser(PFS, Error, Src).parseStandaloneNamedRegister(Reg);
|
2015-07-15 05:24:41 +08:00
|
|
|
}
|
2015-07-28 01:42:45 +08:00
|
|
|
|
2016-10-11 11:13:01 +08:00
|
|
|
bool llvm::parseVirtualRegisterReference(PerFunctionMIParsingState &PFS,
|
|
|
|
VRegInfo *&Info, StringRef Src,
|
2015-07-28 01:42:45 +08:00
|
|
|
SMDiagnostic &Error) {
|
2016-10-11 11:13:01 +08:00
|
|
|
return MIParser(PFS, Error, Src).parseStandaloneVirtualRegister(Info);
|
2015-07-28 01:42:45 +08:00
|
|
|
}
|
2015-08-19 06:26:26 +08:00
|
|
|
|
2016-10-11 11:13:01 +08:00
|
|
|
bool llvm::parseStackObjectReference(PerFunctionMIParsingState &PFS,
|
2016-07-14 07:27:50 +08:00
|
|
|
int &FI, StringRef Src,
|
2015-08-19 06:26:26 +08:00
|
|
|
SMDiagnostic &Error) {
|
2016-07-14 07:27:50 +08:00
|
|
|
return MIParser(PFS, Error, Src).parseStandaloneStackObject(FI);
|
2015-08-19 06:26:26 +08:00
|
|
|
}
|
2015-08-19 08:13:25 +08:00
|
|
|
|
2016-10-11 11:13:01 +08:00
|
|
|
bool llvm::parseMDNode(PerFunctionMIParsingState &PFS,
|
2016-07-14 07:27:50 +08:00
|
|
|
MDNode *&Node, StringRef Src, SMDiagnostic &Error) {
|
|
|
|
return MIParser(PFS, Error, Src).parseStandaloneMDNode(Node);
|
2015-08-19 08:13:25 +08:00
|
|
|
}
|