mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-24 03:33:20 +01:00
76132aaa0b
In the assembler, we should emit build attributes based on the target selected with command-line options. This matches the GNU assembler's behaviour. We only do this for build attributes which describe the hardware that is expected to be available, not the ones that describe ABI compatibility. This is done by moving some of the attribute emission code to ARMTargetStreamer, so that it can be shared between the assembly and code-generation code paths. Since the assembler only creates a MCSubtargetInfo, not an ARMSubtarget, the code had to be changed to check raw features, and not use the convenience functions in ARMSubtarget. If different attributes are later specified using the .eabi_attribute directive, then they will take precedence, as happens when the same .eabi_attribute is specified twice. This must be enabled by an option, because we don't want to do this when parsing inline assembly. The attributes would match the ones emitted at the start of the file, so wouldn't actually change the emitted object file, but the extra directives would be added to every inline assembly block when emitting assembly, which we'd like to avoid. The majority of the changes in the build-attributes.ll test are just re-ordering the directives, because the hardware attributes are now emitted before the ABI ones. However, I did fix one bug which I spotted: Tag_CPU_arch_profile was not being emitted for v6M. Differential revision: https://reviews.llvm.org/D31812 llvm-svn: 300547
192 lines
6.7 KiB
C++
192 lines
6.7 KiB
C++
//===- llvm/MC/MCSubtargetInfo.h - Subtarget Information --------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file describes the subtarget options of a Target machine.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLVM_MC_MCSUBTARGETINFO_H
|
|
#define LLVM_MC_MCSUBTARGETINFO_H
|
|
|
|
#include "llvm/ADT/ArrayRef.h"
|
|
#include "llvm/ADT/StringRef.h"
|
|
#include "llvm/ADT/Triple.h"
|
|
#include "llvm/MC/MCInstrItineraries.h"
|
|
#include "llvm/MC/MCSchedule.h"
|
|
#include "llvm/MC/SubtargetFeature.h"
|
|
#include <algorithm>
|
|
#include <cassert>
|
|
#include <cstdint>
|
|
#include <string>
|
|
|
|
namespace llvm {
|
|
class MachineInstr;
|
|
class MCInst;
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
///
|
|
/// MCSubtargetInfo - Generic base class for all target subtargets.
|
|
///
|
|
class MCSubtargetInfo {
|
|
Triple TargetTriple; // Target triple
|
|
std::string CPU; // CPU being targeted.
|
|
ArrayRef<SubtargetFeatureKV> ProcFeatures; // Processor feature list
|
|
ArrayRef<SubtargetFeatureKV> ProcDesc; // Processor descriptions
|
|
|
|
// Scheduler machine model
|
|
const SubtargetInfoKV *ProcSchedModels;
|
|
const MCWriteProcResEntry *WriteProcResTable;
|
|
const MCWriteLatencyEntry *WriteLatencyTable;
|
|
const MCReadAdvanceEntry *ReadAdvanceTable;
|
|
const MCSchedModel *CPUSchedModel;
|
|
|
|
const InstrStage *Stages; // Instruction itinerary stages
|
|
const unsigned *OperandCycles; // Itinerary operand cycles
|
|
const unsigned *ForwardingPaths; // Forwarding paths
|
|
FeatureBitset FeatureBits; // Feature bits for current CPU + FS
|
|
|
|
public:
|
|
MCSubtargetInfo(const MCSubtargetInfo &) = default;
|
|
MCSubtargetInfo(const Triple &TT, StringRef CPU, StringRef FS,
|
|
ArrayRef<SubtargetFeatureKV> PF,
|
|
ArrayRef<SubtargetFeatureKV> PD,
|
|
const SubtargetInfoKV *ProcSched,
|
|
const MCWriteProcResEntry *WPR, const MCWriteLatencyEntry *WL,
|
|
const MCReadAdvanceEntry *RA, const InstrStage *IS,
|
|
const unsigned *OC, const unsigned *FP);
|
|
MCSubtargetInfo() = delete;
|
|
MCSubtargetInfo &operator=(const MCSubtargetInfo &) = delete;
|
|
MCSubtargetInfo &operator=(MCSubtargetInfo &&) = delete;
|
|
|
|
virtual ~MCSubtargetInfo() {}
|
|
|
|
/// getTargetTriple - Return the target triple string.
|
|
const Triple &getTargetTriple() const { return TargetTriple; }
|
|
|
|
/// getCPU - Return the CPU string.
|
|
StringRef getCPU() const {
|
|
return CPU;
|
|
}
|
|
|
|
/// getFeatureBits - Return the feature bits.
|
|
///
|
|
const FeatureBitset& getFeatureBits() const {
|
|
return FeatureBits;
|
|
}
|
|
|
|
/// setFeatureBits - Set the feature bits.
|
|
///
|
|
void setFeatureBits(const FeatureBitset &FeatureBits_) {
|
|
FeatureBits = FeatureBits_;
|
|
}
|
|
|
|
bool hasFeature(unsigned Feature) const {
|
|
return FeatureBits[Feature];
|
|
}
|
|
|
|
protected:
|
|
/// Initialize the scheduling model and feature bits.
|
|
///
|
|
/// FIXME: Find a way to stick this in the constructor, since it should only
|
|
/// be called during initialization.
|
|
void InitMCProcessorInfo(StringRef CPU, StringRef FS);
|
|
|
|
public:
|
|
/// Set the features to the default for the given CPU with an appended feature
|
|
/// string.
|
|
void setDefaultFeatures(StringRef CPU, StringRef FS);
|
|
|
|
/// ToggleFeature - Toggle a feature and returns the re-computed feature
|
|
/// bits. This version does not change the implied bits.
|
|
FeatureBitset ToggleFeature(uint64_t FB);
|
|
|
|
/// ToggleFeature - Toggle a feature and returns the re-computed feature
|
|
/// bits. This version does not change the implied bits.
|
|
FeatureBitset ToggleFeature(const FeatureBitset& FB);
|
|
|
|
/// ToggleFeature - Toggle a set of features and returns the re-computed
|
|
/// feature bits. This version will also change all implied bits.
|
|
FeatureBitset ToggleFeature(StringRef FS);
|
|
|
|
/// Apply a feature flag and return the re-computed feature bits, including
|
|
/// all feature bits implied by the flag.
|
|
FeatureBitset ApplyFeatureFlag(StringRef FS);
|
|
|
|
/// getSchedModelForCPU - Get the machine model of a CPU.
|
|
///
|
|
const MCSchedModel &getSchedModelForCPU(StringRef CPU) const;
|
|
|
|
/// Get the machine model for this subtarget's CPU.
|
|
const MCSchedModel &getSchedModel() const { return *CPUSchedModel; }
|
|
|
|
/// Return an iterator at the first process resource consumed by the given
|
|
/// scheduling class.
|
|
const MCWriteProcResEntry *getWriteProcResBegin(
|
|
const MCSchedClassDesc *SC) const {
|
|
return &WriteProcResTable[SC->WriteProcResIdx];
|
|
}
|
|
const MCWriteProcResEntry *getWriteProcResEnd(
|
|
const MCSchedClassDesc *SC) const {
|
|
return getWriteProcResBegin(SC) + SC->NumWriteProcResEntries;
|
|
}
|
|
|
|
const MCWriteLatencyEntry *getWriteLatencyEntry(const MCSchedClassDesc *SC,
|
|
unsigned DefIdx) const {
|
|
assert(DefIdx < SC->NumWriteLatencyEntries &&
|
|
"MachineModel does not specify a WriteResource for DefIdx");
|
|
|
|
return &WriteLatencyTable[SC->WriteLatencyIdx + DefIdx];
|
|
}
|
|
|
|
int getReadAdvanceCycles(const MCSchedClassDesc *SC, unsigned UseIdx,
|
|
unsigned WriteResID) const {
|
|
// TODO: The number of read advance entries in a class can be significant
|
|
// (~50). Consider compressing the WriteID into a dense ID of those that are
|
|
// used by ReadAdvance and representing them as a bitset.
|
|
for (const MCReadAdvanceEntry *I = &ReadAdvanceTable[SC->ReadAdvanceIdx],
|
|
*E = I + SC->NumReadAdvanceEntries; I != E; ++I) {
|
|
if (I->UseIdx < UseIdx)
|
|
continue;
|
|
if (I->UseIdx > UseIdx)
|
|
break;
|
|
// Find the first WriteResIdx match, which has the highest cycle count.
|
|
if (!I->WriteResourceID || I->WriteResourceID == WriteResID) {
|
|
return I->Cycles;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/// getInstrItineraryForCPU - Get scheduling itinerary of a CPU.
|
|
///
|
|
InstrItineraryData getInstrItineraryForCPU(StringRef CPU) const;
|
|
|
|
/// Initialize an InstrItineraryData instance.
|
|
void initInstrItins(InstrItineraryData &InstrItins) const;
|
|
|
|
/// Check whether the CPU string is valid.
|
|
bool isCPUStringValid(StringRef CPU) const {
|
|
auto Found = std::lower_bound(ProcDesc.begin(), ProcDesc.end(), CPU);
|
|
return Found != ProcDesc.end() && StringRef(Found->Key) == CPU;
|
|
}
|
|
|
|
/// Returns string representation of scheduler comment
|
|
virtual std::string getSchedInfoStr(const MachineInstr &MI) const {
|
|
return std::string();
|
|
}
|
|
|
|
virtual std::string getSchedInfoStr(MCInst const &MCI) const {
|
|
return std::string();
|
|
}
|
|
};
|
|
|
|
} // end namespace llvm
|
|
|
|
#endif // LLVM_MC_MCSUBTARGETINFO_H
|