2002-12-02 01:23:04 +00:00
|
|
|
//===- Record.cpp - Record implementation ---------------------------------===//
|
2005-04-22 00:00:37 +00:00
|
|
|
//
|
2003-10-20 20:20:30 +00:00
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
2007-12-29 20:37:13 +00:00
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
2005-04-22 00:00:37 +00:00
|
|
|
//
|
2003-10-20 20:20:30 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
2002-12-02 01:23:04 +00:00
|
|
|
//
|
2006-03-31 21:53:49 +00:00
|
|
|
// Implement the tablegen record classes.
|
2002-12-02 01:23:04 +00:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2011-10-01 16:41:13 +00:00
|
|
|
#include "llvm/TableGen/Record.h"
|
2011-07-29 19:07:11 +00:00
|
|
|
#include "llvm/ADT/DenseMap.h"
|
|
|
|
#include "llvm/ADT/FoldingSet.h"
|
2012-03-05 10:36:16 +00:00
|
|
|
#include "llvm/ADT/Hashing.h"
|
2016-12-04 05:48:06 +00:00
|
|
|
#include "llvm/ADT/SmallString.h"
|
2012-12-03 16:50:05 +00:00
|
|
|
#include "llvm/ADT/SmallVector.h"
|
2007-11-20 22:25:16 +00:00
|
|
|
#include "llvm/ADT/StringExtras.h"
|
2011-07-29 19:07:14 +00:00
|
|
|
#include "llvm/ADT/StringMap.h"
|
2016-08-23 17:14:32 +00:00
|
|
|
#include "llvm/Support/Compiler.h"
|
2012-12-03 16:50:05 +00:00
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
|
|
#include "llvm/TableGen/Error.h"
|
2016-08-23 17:14:32 +00:00
|
|
|
#include <cassert>
|
|
|
|
#include <cstdint>
|
|
|
|
#include <new>
|
2005-12-26 05:08:55 +00:00
|
|
|
|
2004-08-01 03:55:39 +00:00
|
|
|
using namespace llvm;
|
2002-12-02 01:23:04 +00:00
|
|
|
|
2016-12-04 05:48:20 +00:00
|
|
|
static BumpPtrAllocator Allocator;
|
|
|
|
|
2011-07-29 19:06:58 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// std::string wrapper for DenseMap purposes
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2012-03-05 10:36:16 +00:00
|
|
|
namespace llvm {
|
|
|
|
|
2016-07-14 14:53:14 +00:00
|
|
|
/// This is a wrapper for std::string suitable for using as a key to a DenseMap.
|
|
|
|
/// Because there isn't a particularly
|
2011-07-29 19:06:58 +00:00
|
|
|
/// good way to indicate tombstone or empty keys for strings, we want
|
|
|
|
/// to wrap std::string to indicate that this is a "special" string
|
|
|
|
/// not expected to take on certain values (those of the tombstone and
|
|
|
|
/// empty keys). This makes things a little safer as it clarifies
|
|
|
|
/// that DenseMap is really not appropriate for general strings.
|
|
|
|
|
|
|
|
class TableGenStringKey {
|
|
|
|
public:
|
2016-12-04 05:48:03 +00:00
|
|
|
TableGenStringKey(StringRef str) : data(str) {}
|
2011-07-29 19:06:58 +00:00
|
|
|
TableGenStringKey(const char *str) : data(str) {}
|
|
|
|
|
2016-12-04 05:48:16 +00:00
|
|
|
StringRef str() const { return data; }
|
2012-03-05 10:36:16 +00:00
|
|
|
|
|
|
|
friend hash_code hash_value(const TableGenStringKey &Value) {
|
|
|
|
using llvm::hash_value;
|
|
|
|
return hash_value(Value.str());
|
|
|
|
}
|
2016-08-23 17:14:32 +00:00
|
|
|
|
2011-07-29 19:06:58 +00:00
|
|
|
private:
|
|
|
|
std::string data;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Specialize DenseMapInfo for TableGenStringKey.
|
|
|
|
template<> struct DenseMapInfo<TableGenStringKey> {
|
|
|
|
static inline TableGenStringKey getEmptyKey() {
|
|
|
|
TableGenStringKey Empty("<<<EMPTY KEY>>>");
|
|
|
|
return Empty;
|
|
|
|
}
|
2016-08-23 17:14:32 +00:00
|
|
|
|
2011-07-29 19:06:58 +00:00
|
|
|
static inline TableGenStringKey getTombstoneKey() {
|
|
|
|
TableGenStringKey Tombstone("<<<TOMBSTONE KEY>>>");
|
|
|
|
return Tombstone;
|
|
|
|
}
|
2016-08-23 17:14:32 +00:00
|
|
|
|
2011-07-29 19:06:58 +00:00
|
|
|
static unsigned getHashValue(const TableGenStringKey& Val) {
|
2012-03-05 10:36:16 +00:00
|
|
|
using llvm::hash_value;
|
|
|
|
return hash_value(Val);
|
2011-07-29 19:06:58 +00:00
|
|
|
}
|
2016-08-23 17:14:32 +00:00
|
|
|
|
2011-07-29 19:06:58 +00:00
|
|
|
static bool isEqual(const TableGenStringKey& LHS,
|
|
|
|
const TableGenStringKey& RHS) {
|
|
|
|
return LHS.str() == RHS.str();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-08-23 17:14:32 +00:00
|
|
|
} // end namespace llvm
|
2011-07-29 19:06:58 +00:00
|
|
|
|
2002-12-02 01:23:04 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Type implementations
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2011-07-18 17:02:57 +00:00
|
|
|
BitRecTy BitRecTy::Shared;
|
2016-07-05 21:22:55 +00:00
|
|
|
CodeRecTy CodeRecTy::Shared;
|
2011-07-18 17:02:57 +00:00
|
|
|
IntRecTy IntRecTy::Shared;
|
|
|
|
StringRecTy StringRecTy::Shared;
|
|
|
|
DagRecTy DagRecTy::Shared;
|
|
|
|
|
2016-01-29 20:50:44 +00:00
|
|
|
LLVM_DUMP_METHOD void RecTy::dump() const { print(errs()); }
|
2002-12-02 01:23:04 +00:00
|
|
|
|
2011-07-18 17:02:57 +00:00
|
|
|
ListRecTy *RecTy::getListTy() {
|
|
|
|
if (!ListTy)
|
2016-12-04 05:48:20 +00:00
|
|
|
ListTy = new(Allocator) ListRecTy(this);
|
|
|
|
return ListTy;
|
2011-07-18 17:02:57 +00:00
|
|
|
}
|
|
|
|
|
2015-05-30 07:36:01 +00:00
|
|
|
bool RecTy::typeIsConvertibleTo(const RecTy *RHS) const {
|
|
|
|
assert(RHS && "NULL pointer");
|
2013-01-07 02:30:19 +00:00
|
|
|
return Kind == RHS->getRecTyKind();
|
|
|
|
}
|
|
|
|
|
2015-05-30 07:36:01 +00:00
|
|
|
bool BitRecTy::typeIsConvertibleTo(const RecTy *RHS) const{
|
2015-06-06 01:34:01 +00:00
|
|
|
if (RecTy::typeIsConvertibleTo(RHS) || RHS->getRecTyKind() == IntRecTyKind)
|
2013-01-07 02:30:19 +00:00
|
|
|
return true;
|
2015-06-06 01:34:01 +00:00
|
|
|
if (const BitsRecTy *BitsTy = dyn_cast<BitsRecTy>(RHS))
|
2013-01-07 02:30:19 +00:00
|
|
|
return BitsTy->getNumBits() == 1;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-07-18 17:02:57 +00:00
|
|
|
BitsRecTy *BitsRecTy::get(unsigned Sz) {
|
2016-12-04 05:48:20 +00:00
|
|
|
static std::vector<BitsRecTy*> Shared;
|
2011-07-18 17:02:57 +00:00
|
|
|
if (Sz >= Shared.size())
|
|
|
|
Shared.resize(Sz + 1);
|
2016-12-04 05:48:20 +00:00
|
|
|
BitsRecTy *&Ty = Shared[Sz];
|
2011-07-18 17:02:57 +00:00
|
|
|
if (!Ty)
|
2016-12-04 05:48:20 +00:00
|
|
|
Ty = new(Allocator) BitsRecTy(Sz);
|
|
|
|
return Ty;
|
2011-07-18 17:02:57 +00:00
|
|
|
}
|
|
|
|
|
2007-11-20 22:25:16 +00:00
|
|
|
std::string BitsRecTy::getAsString() const {
|
|
|
|
return "bits<" + utostr(Size) + ">";
|
|
|
|
}
|
|
|
|
|
2015-05-30 07:36:01 +00:00
|
|
|
bool BitsRecTy::typeIsConvertibleTo(const RecTy *RHS) const {
|
|
|
|
if (RecTy::typeIsConvertibleTo(RHS)) //argument and the sender are same type
|
2013-01-07 02:30:19 +00:00
|
|
|
return cast<BitsRecTy>(RHS)->Size == Size;
|
|
|
|
RecTyKind kind = RHS->getRecTyKind();
|
|
|
|
return (kind == BitRecTyKind && Size == 1) || (kind == IntRecTyKind);
|
|
|
|
}
|
|
|
|
|
2015-05-30 07:36:01 +00:00
|
|
|
bool IntRecTy::typeIsConvertibleTo(const RecTy *RHS) const {
|
2013-01-07 02:30:19 +00:00
|
|
|
RecTyKind kind = RHS->getRecTyKind();
|
|
|
|
return kind==BitRecTyKind || kind==BitsRecTyKind || kind==IntRecTyKind;
|
|
|
|
}
|
|
|
|
|
2015-06-01 06:44:18 +00:00
|
|
|
std::string StringRecTy::getAsString() const {
|
|
|
|
return "string";
|
|
|
|
}
|
2006-03-31 21:53:49 +00:00
|
|
|
|
2007-11-20 22:25:16 +00:00
|
|
|
std::string ListRecTy::getAsString() const {
|
|
|
|
return "list<" + Ty->getAsString() + ">";
|
|
|
|
}
|
|
|
|
|
2015-05-30 07:36:01 +00:00
|
|
|
bool ListRecTy::typeIsConvertibleTo(const RecTy *RHS) const {
|
|
|
|
if (const auto *ListTy = dyn_cast<ListRecTy>(RHS))
|
|
|
|
return Ty->typeIsConvertibleTo(ListTy->getElementType());
|
2013-01-07 02:30:19 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-06-01 06:44:18 +00:00
|
|
|
std::string DagRecTy::getAsString() const {
|
|
|
|
return "dag";
|
|
|
|
}
|
|
|
|
|
2011-07-18 17:02:57 +00:00
|
|
|
RecordRecTy *RecordRecTy::get(Record *R) {
|
2015-04-22 02:59:06 +00:00
|
|
|
return dyn_cast<RecordRecTy>(R->getDefInit()->getType());
|
2011-07-18 17:02:57 +00:00
|
|
|
}
|
|
|
|
|
2007-11-20 22:25:16 +00:00
|
|
|
std::string RecordRecTy::getAsString() const {
|
|
|
|
return Rec->getName();
|
|
|
|
}
|
2003-08-04 04:50:57 +00:00
|
|
|
|
2015-05-30 07:36:01 +00:00
|
|
|
bool RecordRecTy::typeIsConvertibleTo(const RecTy *RHS) const {
|
2013-01-07 02:30:19 +00:00
|
|
|
const RecordRecTy *RTy = dyn_cast<RecordRecTy>(RHS);
|
|
|
|
if (!RTy)
|
|
|
|
return false;
|
|
|
|
|
2015-05-30 07:36:01 +00:00
|
|
|
if (RTy->getRecord() == Rec || Rec->isSubClassOf(RTy->getRecord()))
|
2010-06-17 23:00:16 +00:00
|
|
|
return true;
|
|
|
|
|
2016-01-18 19:52:37 +00:00
|
|
|
for (const auto &SCPair : RTy->getRecord()->getSuperClasses())
|
|
|
|
if (Rec->isSubClassOf(SCPair.first))
|
2010-06-17 23:00:16 +00:00
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
2003-08-03 18:17:22 +00:00
|
|
|
}
|
|
|
|
|
2009-06-08 20:23:18 +00:00
|
|
|
RecTy *llvm::resolveTypes(RecTy *T1, RecTy *T2) {
|
2012-09-19 02:14:59 +00:00
|
|
|
if (T1->typeIsConvertibleTo(T2))
|
|
|
|
return T2;
|
|
|
|
if (T2->typeIsConvertibleTo(T1))
|
|
|
|
return T1;
|
|
|
|
|
|
|
|
// If one is a Record type, check superclasses
|
2012-10-05 03:31:58 +00:00
|
|
|
if (RecordRecTy *RecTy1 = dyn_cast<RecordRecTy>(T1)) {
|
2012-09-19 02:14:59 +00:00
|
|
|
// See if T2 inherits from a type T1 also inherits from
|
2016-01-18 19:52:37 +00:00
|
|
|
for (const auto &SuperPair1 : RecTy1->getRecord()->getSuperClasses()) {
|
|
|
|
RecordRecTy *SuperRecTy1 = RecordRecTy::get(SuperPair1.first);
|
2012-09-19 02:14:59 +00:00
|
|
|
RecTy *NewType1 = resolveTypes(SuperRecTy1, T2);
|
2015-04-22 05:27:11 +00:00
|
|
|
if (NewType1)
|
2012-09-19 02:14:59 +00:00
|
|
|
return NewType1;
|
|
|
|
}
|
|
|
|
}
|
2012-10-05 03:31:58 +00:00
|
|
|
if (RecordRecTy *RecTy2 = dyn_cast<RecordRecTy>(T2)) {
|
2012-09-19 02:14:59 +00:00
|
|
|
// See if T1 inherits from a type T2 also inherits from
|
2016-01-18 19:52:37 +00:00
|
|
|
for (const auto &SuperPair2 : RecTy2->getRecord()->getSuperClasses()) {
|
|
|
|
RecordRecTy *SuperRecTy2 = RecordRecTy::get(SuperPair2.first);
|
2012-09-19 02:14:59 +00:00
|
|
|
RecTy *NewType2 = resolveTypes(T1, SuperRecTy2);
|
2015-04-22 05:27:11 +00:00
|
|
|
if (NewType2)
|
2012-09-19 02:14:59 +00:00
|
|
|
return NewType2;
|
2009-06-08 20:23:18 +00:00
|
|
|
}
|
|
|
|
}
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2009-06-08 20:23:18 +00:00
|
|
|
}
|
|
|
|
|
2002-12-02 01:23:04 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Initializer implementations
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2011-12-20 02:50:00 +00:00
|
|
|
void Init::anchor() { }
|
2016-01-29 20:50:44 +00:00
|
|
|
LLVM_DUMP_METHOD void Init::dump() const { return print(errs()); }
|
2002-12-02 01:23:04 +00:00
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
UnsetInit *UnsetInit::get() {
|
|
|
|
static UnsetInit TheInit;
|
2011-07-29 19:07:09 +00:00
|
|
|
return &TheInit;
|
2011-07-29 19:07:07 +00:00
|
|
|
}
|
|
|
|
|
2015-05-30 07:34:51 +00:00
|
|
|
Init *UnsetInit::convertInitializerTo(RecTy *Ty) const {
|
|
|
|
if (auto *BRT = dyn_cast<BitsRecTy>(Ty)) {
|
|
|
|
SmallVector<Init *, 16> NewBits(BRT->getNumBits());
|
|
|
|
|
|
|
|
for (unsigned i = 0; i != BRT->getNumBits(); ++i)
|
|
|
|
NewBits[i] = UnsetInit::get();
|
|
|
|
|
|
|
|
return BitsInit::get(NewBits);
|
|
|
|
}
|
|
|
|
|
|
|
|
// All other types can just be returned.
|
|
|
|
return const_cast<UnsetInit *>(this);
|
|
|
|
}
|
2011-12-20 02:50:00 +00:00
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
BitInit *BitInit::get(bool V) {
|
|
|
|
static BitInit True(true);
|
|
|
|
static BitInit False(false);
|
2011-07-29 19:07:10 +00:00
|
|
|
|
|
|
|
return V ? &True : &False;
|
2011-07-29 19:07:07 +00:00
|
|
|
}
|
|
|
|
|
2015-05-30 07:34:51 +00:00
|
|
|
Init *BitInit::convertInitializerTo(RecTy *Ty) const {
|
|
|
|
if (isa<BitRecTy>(Ty))
|
|
|
|
return const_cast<BitInit *>(this);
|
|
|
|
|
|
|
|
if (isa<IntRecTy>(Ty))
|
|
|
|
return IntInit::get(getValue());
|
|
|
|
|
|
|
|
if (auto *BRT = dyn_cast<BitsRecTy>(Ty)) {
|
|
|
|
// Can only convert single bit.
|
|
|
|
if (BRT->getNumBits() == 1)
|
|
|
|
return BitsInit::get(const_cast<BitInit *>(this));
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2011-07-29 19:07:11 +00:00
|
|
|
static void
|
2011-07-29 22:43:06 +00:00
|
|
|
ProfileBitsInit(FoldingSetNodeID &ID, ArrayRef<Init *> Range) {
|
2011-07-29 19:07:11 +00:00
|
|
|
ID.AddInteger(Range.size());
|
|
|
|
|
2015-04-29 07:13:14 +00:00
|
|
|
for (Init *I : Range)
|
|
|
|
ID.AddPointer(I);
|
2011-07-29 19:07:11 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
BitsInit *BitsInit::get(ArrayRef<Init *> Range) {
|
2015-04-24 05:38:48 +00:00
|
|
|
static FoldingSet<BitsInit> ThePool;
|
2016-12-04 05:48:20 +00:00
|
|
|
static std::vector<BitsInit*> TheActualPool;
|
2011-07-29 19:07:11 +00:00
|
|
|
|
|
|
|
FoldingSetNodeID ID;
|
|
|
|
ProfileBitsInit(ID, Range);
|
|
|
|
|
2014-04-09 04:50:04 +00:00
|
|
|
void *IP = nullptr;
|
2011-07-29 22:43:06 +00:00
|
|
|
if (BitsInit *I = ThePool.FindNodeOrInsertPos(ID, IP))
|
2011-07-29 19:07:11 +00:00
|
|
|
return I;
|
|
|
|
|
2016-12-04 05:48:20 +00:00
|
|
|
void *Mem = Allocator.Allocate(totalSizeToAlloc<Init *>(Range.size()),
|
|
|
|
alignof(BitsInit));
|
|
|
|
BitsInit *I = new(Mem) BitsInit(Range.size());
|
2016-01-18 19:52:24 +00:00
|
|
|
std::uninitialized_copy(Range.begin(), Range.end(),
|
|
|
|
I->getTrailingObjects<Init *>());
|
2011-07-29 19:07:11 +00:00
|
|
|
ThePool.InsertNode(I, IP);
|
2016-12-04 05:48:20 +00:00
|
|
|
TheActualPool.push_back(I);
|
2011-07-29 19:07:11 +00:00
|
|
|
return I;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BitsInit::Profile(FoldingSetNodeID &ID) const {
|
2016-01-18 19:52:24 +00:00
|
|
|
ProfileBitsInit(ID, makeArrayRef(getTrailingObjects<Init *>(), NumBits));
|
2011-07-29 19:07:07 +00:00
|
|
|
}
|
|
|
|
|
2015-05-30 07:34:51 +00:00
|
|
|
Init *BitsInit::convertInitializerTo(RecTy *Ty) const {
|
|
|
|
if (isa<BitRecTy>(Ty)) {
|
|
|
|
if (getNumBits() != 1) return nullptr; // Only accept if just one bit!
|
|
|
|
return getBit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (auto *BRT = dyn_cast<BitsRecTy>(Ty)) {
|
|
|
|
// If the number of bits is right, return it. Otherwise we need to expand
|
|
|
|
// or truncate.
|
|
|
|
if (getNumBits() != BRT->getNumBits()) return nullptr;
|
|
|
|
return const_cast<BitsInit *>(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isa<IntRecTy>(Ty)) {
|
|
|
|
int64_t Result = 0;
|
|
|
|
for (unsigned i = 0, e = getNumBits(); i != e; ++i)
|
|
|
|
if (auto *Bit = dyn_cast<BitInit>(getBit(i)))
|
|
|
|
Result |= static_cast<int64_t>(Bit->getValue()) << i;
|
|
|
|
else
|
|
|
|
return nullptr;
|
|
|
|
return IntInit::get(Result);
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *
|
2011-07-29 19:07:05 +00:00
|
|
|
BitsInit::convertInitializerBitRange(const std::vector<unsigned> &Bits) const {
|
2011-07-29 22:43:06 +00:00
|
|
|
SmallVector<Init *, 16> NewBits(Bits.size());
|
2011-07-29 19:07:00 +00:00
|
|
|
|
2002-12-02 01:23:04 +00:00
|
|
|
for (unsigned i = 0, e = Bits.size(); i != e; ++i) {
|
2011-07-29 19:07:00 +00:00
|
|
|
if (Bits[i] >= getNumBits())
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2011-07-29 19:07:00 +00:00
|
|
|
NewBits[i] = getBit(Bits[i]);
|
2002-12-02 01:23:04 +00:00
|
|
|
}
|
2011-07-29 19:07:07 +00:00
|
|
|
return BitsInit::get(NewBits);
|
2002-12-02 01:23:04 +00:00
|
|
|
}
|
|
|
|
|
2007-11-22 21:05:25 +00:00
|
|
|
std::string BitsInit::getAsString() const {
|
|
|
|
std::string Result = "{ ";
|
2002-12-02 01:23:04 +00:00
|
|
|
for (unsigned i = 0, e = getNumBits(); i != e; ++i) {
|
2007-11-22 21:05:25 +00:00
|
|
|
if (i) Result += ", ";
|
2011-07-29 22:43:06 +00:00
|
|
|
if (Init *Bit = getBit(e-i-1))
|
2007-11-22 21:05:25 +00:00
|
|
|
Result += Bit->getAsString();
|
2002-12-06 04:42:10 +00:00
|
|
|
else
|
2007-11-22 21:05:25 +00:00
|
|
|
Result += "*";
|
2002-12-02 01:23:04 +00:00
|
|
|
}
|
2007-11-22 21:05:25 +00:00
|
|
|
return Result + " }";
|
2002-12-02 01:23:04 +00:00
|
|
|
}
|
|
|
|
|
Re-work bit/bits value resolving in tblgen
- This patch is inspired by the failure of the following code snippet
which is used to convert enumerable values into encoding bits to
improve the readability of td files.
class S<int s> {
bits<2> V = !if(!eq(s, 8), {0, 0},
!if(!eq(s, 16), {0, 1},
!if(!eq(s, 32), {1, 0},
!if(!eq(s, 64), {1, 1}, {?, ?}))));
}
Later, PR8330 is found to report not exactly the same bug relevant
issue to bit/bits values.
- Instead of resolving bit/bits values separately through
resolveBitReference(), this patch adds getBit() for all Inits and
resolves bit value by resolving plus getting the specified bit. This
unifies the resolving of bit with other values and removes redundant
logic for resolving bit only. In addition,
BitsInit::resolveReferences() is optimized to take advantage of this
origanization by resolving VarBitInit's variable reference first and
then getting bits from it.
- The type interference in '!if' operator is revised to support possible
combinations of int and bits/bit in MHS and RHS.
- As there may be illegal assignments from integer value to bit, says
assign 2 to a bit, but we only check this during instantiation in some
cases, e.g.
bit V = !if(!eq(x, 17), 0, 2);
Verbose diagnostic message is generated when invalid value is
resolveed to help locating the error.
- PR8330 is fixed as well.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163360 91177308-0d34-0410-b5e6-96231b3b80d8
2012-09-06 23:32:48 +00:00
|
|
|
// Fix bit initializer to preserve the behavior that bit reference from a unset
|
|
|
|
// bits initializer will resolve into VarBitInit to keep the field name and bit
|
|
|
|
// number used in targets with fixed insn length.
|
|
|
|
static Init *fixBitInit(const RecordVal *RV, Init *Before, Init *After) {
|
2015-04-22 02:09:45 +00:00
|
|
|
if (RV || !isa<UnsetInit>(After))
|
Re-work bit/bits value resolving in tblgen
- This patch is inspired by the failure of the following code snippet
which is used to convert enumerable values into encoding bits to
improve the readability of td files.
class S<int s> {
bits<2> V = !if(!eq(s, 8), {0, 0},
!if(!eq(s, 16), {0, 1},
!if(!eq(s, 32), {1, 0},
!if(!eq(s, 64), {1, 1}, {?, ?}))));
}
Later, PR8330 is found to report not exactly the same bug relevant
issue to bit/bits values.
- Instead of resolving bit/bits values separately through
resolveBitReference(), this patch adds getBit() for all Inits and
resolves bit value by resolving plus getting the specified bit. This
unifies the resolving of bit with other values and removes redundant
logic for resolving bit only. In addition,
BitsInit::resolveReferences() is optimized to take advantage of this
origanization by resolving VarBitInit's variable reference first and
then getting bits from it.
- The type interference in '!if' operator is revised to support possible
combinations of int and bits/bit in MHS and RHS.
- As there may be illegal assignments from integer value to bit, says
assign 2 to a bit, but we only check this during instantiation in some
cases, e.g.
bit V = !if(!eq(x, 17), 0, 2);
Verbose diagnostic message is generated when invalid value is
resolveed to help locating the error.
- PR8330 is fixed as well.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163360 91177308-0d34-0410-b5e6-96231b3b80d8
2012-09-06 23:32:48 +00:00
|
|
|
return After;
|
|
|
|
return Before;
|
|
|
|
}
|
|
|
|
|
2003-07-30 04:05:07 +00:00
|
|
|
// resolveReferences - If there are any field references that refer to fields
|
|
|
|
// that have been filled in, we can propagate the values now.
|
|
|
|
//
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *BitsInit::resolveReferences(Record &R, const RecordVal *RV) const {
|
2002-12-02 16:43:30 +00:00
|
|
|
bool Changed = false;
|
2011-07-29 22:43:06 +00:00
|
|
|
SmallVector<Init *, 16> NewBits(getNumBits());
|
2002-12-02 16:43:30 +00:00
|
|
|
|
2014-04-09 04:50:04 +00:00
|
|
|
Init *CachedInit = nullptr;
|
|
|
|
Init *CachedBitVar = nullptr;
|
Re-work bit/bits value resolving in tblgen
- This patch is inspired by the failure of the following code snippet
which is used to convert enumerable values into encoding bits to
improve the readability of td files.
class S<int s> {
bits<2> V = !if(!eq(s, 8), {0, 0},
!if(!eq(s, 16), {0, 1},
!if(!eq(s, 32), {1, 0},
!if(!eq(s, 64), {1, 1}, {?, ?}))));
}
Later, PR8330 is found to report not exactly the same bug relevant
issue to bit/bits values.
- Instead of resolving bit/bits values separately through
resolveBitReference(), this patch adds getBit() for all Inits and
resolves bit value by resolving plus getting the specified bit. This
unifies the resolving of bit with other values and removes redundant
logic for resolving bit only. In addition,
BitsInit::resolveReferences() is optimized to take advantage of this
origanization by resolving VarBitInit's variable reference first and
then getting bits from it.
- The type interference in '!if' operator is revised to support possible
combinations of int and bits/bit in MHS and RHS.
- As there may be illegal assignments from integer value to bit, says
assign 2 to a bit, but we only check this during instantiation in some
cases, e.g.
bit V = !if(!eq(x, 17), 0, 2);
Verbose diagnostic message is generated when invalid value is
resolveed to help locating the error.
- PR8330 is fixed as well.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163360 91177308-0d34-0410-b5e6-96231b3b80d8
2012-09-06 23:32:48 +00:00
|
|
|
bool CachedBitVarChanged = false;
|
|
|
|
|
|
|
|
for (unsigned i = 0, e = getNumBits(); i != e; ++i) {
|
2016-01-18 19:52:24 +00:00
|
|
|
Init *CurBit = getBit(i);
|
Re-work bit/bits value resolving in tblgen
- This patch is inspired by the failure of the following code snippet
which is used to convert enumerable values into encoding bits to
improve the readability of td files.
class S<int s> {
bits<2> V = !if(!eq(s, 8), {0, 0},
!if(!eq(s, 16), {0, 1},
!if(!eq(s, 32), {1, 0},
!if(!eq(s, 64), {1, 1}, {?, ?}))));
}
Later, PR8330 is found to report not exactly the same bug relevant
issue to bit/bits values.
- Instead of resolving bit/bits values separately through
resolveBitReference(), this patch adds getBit() for all Inits and
resolves bit value by resolving plus getting the specified bit. This
unifies the resolving of bit with other values and removes redundant
logic for resolving bit only. In addition,
BitsInit::resolveReferences() is optimized to take advantage of this
origanization by resolving VarBitInit's variable reference first and
then getting bits from it.
- The type interference in '!if' operator is revised to support possible
combinations of int and bits/bit in MHS and RHS.
- As there may be illegal assignments from integer value to bit, says
assign 2 to a bit, but we only check this during instantiation in some
cases, e.g.
bit V = !if(!eq(x, 17), 0, 2);
Verbose diagnostic message is generated when invalid value is
resolveed to help locating the error.
- PR8330 is fixed as well.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163360 91177308-0d34-0410-b5e6-96231b3b80d8
2012-09-06 23:32:48 +00:00
|
|
|
Init *CurBitVar = CurBit->getBitVar();
|
2002-12-06 03:55:39 +00:00
|
|
|
|
2011-07-29 19:07:00 +00:00
|
|
|
NewBits[i] = CurBit;
|
Re-work bit/bits value resolving in tblgen
- This patch is inspired by the failure of the following code snippet
which is used to convert enumerable values into encoding bits to
improve the readability of td files.
class S<int s> {
bits<2> V = !if(!eq(s, 8), {0, 0},
!if(!eq(s, 16), {0, 1},
!if(!eq(s, 32), {1, 0},
!if(!eq(s, 64), {1, 1}, {?, ?}))));
}
Later, PR8330 is found to report not exactly the same bug relevant
issue to bit/bits values.
- Instead of resolving bit/bits values separately through
resolveBitReference(), this patch adds getBit() for all Inits and
resolves bit value by resolving plus getting the specified bit. This
unifies the resolving of bit with other values and removes redundant
logic for resolving bit only. In addition,
BitsInit::resolveReferences() is optimized to take advantage of this
origanization by resolving VarBitInit's variable reference first and
then getting bits from it.
- The type interference in '!if' operator is revised to support possible
combinations of int and bits/bit in MHS and RHS.
- As there may be illegal assignments from integer value to bit, says
assign 2 to a bit, but we only check this during instantiation in some
cases, e.g.
bit V = !if(!eq(x, 17), 0, 2);
Verbose diagnostic message is generated when invalid value is
resolveed to help locating the error.
- PR8330 is fixed as well.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163360 91177308-0d34-0410-b5e6-96231b3b80d8
2012-09-06 23:32:48 +00:00
|
|
|
|
|
|
|
if (CurBitVar == CachedBitVar) {
|
|
|
|
if (CachedBitVarChanged) {
|
|
|
|
Init *Bit = CachedInit->getBit(CurBit->getBitNum());
|
|
|
|
NewBits[i] = fixBitInit(RV, CurBit, Bit);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
CachedBitVar = CurBitVar;
|
|
|
|
CachedBitVarChanged = false;
|
|
|
|
|
|
|
|
Init *B;
|
|
|
|
do {
|
|
|
|
B = CurBitVar;
|
|
|
|
CurBitVar = CurBitVar->resolveReferences(R, RV);
|
|
|
|
CachedBitVarChanged |= B != CurBitVar;
|
|
|
|
Changed |= B != CurBitVar;
|
|
|
|
} while (B != CurBitVar);
|
|
|
|
CachedInit = CurBitVar;
|
|
|
|
|
|
|
|
if (CachedBitVarChanged) {
|
|
|
|
Init *Bit = CurBitVar->getBit(CurBit->getBitNum());
|
|
|
|
NewBits[i] = fixBitInit(RV, CurBit, Bit);
|
|
|
|
}
|
2002-12-02 16:43:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (Changed)
|
2011-07-29 19:07:07 +00:00
|
|
|
return BitsInit::get(NewBits);
|
2011-07-29 19:07:00 +00:00
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
return const_cast<BitsInit *>(this);
|
2002-12-02 16:43:30 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
IntInit *IntInit::get(int64_t V) {
|
2016-12-04 05:48:20 +00:00
|
|
|
static DenseMap<int64_t, IntInit*> ThePool;
|
2011-07-29 19:07:12 +00:00
|
|
|
|
2016-12-04 05:48:20 +00:00
|
|
|
IntInit *&I = ThePool[V];
|
|
|
|
if (!I) I = new(Allocator) IntInit(V);
|
|
|
|
return I;
|
2011-07-29 19:07:07 +00:00
|
|
|
}
|
|
|
|
|
2007-11-22 21:05:25 +00:00
|
|
|
std::string IntInit::getAsString() const {
|
|
|
|
return itostr(Value);
|
|
|
|
}
|
|
|
|
|
2015-05-30 07:34:51 +00:00
|
|
|
static bool canFitInBitfield(int64_t Value, unsigned NumBits) {
|
|
|
|
// For example, with NumBits == 4, we permit Values from [-7 .. 15].
|
|
|
|
return (NumBits >= sizeof(Value) * 8) ||
|
|
|
|
(Value >> NumBits == 0) || (Value >> (NumBits-1) == -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
Init *IntInit::convertInitializerTo(RecTy *Ty) const {
|
|
|
|
if (isa<IntRecTy>(Ty))
|
|
|
|
return const_cast<IntInit *>(this);
|
|
|
|
|
|
|
|
if (isa<BitRecTy>(Ty)) {
|
|
|
|
int64_t Val = getValue();
|
|
|
|
if (Val != 0 && Val != 1) return nullptr; // Only accept 0 or 1 for a bit!
|
|
|
|
return BitInit::get(Val != 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (auto *BRT = dyn_cast<BitsRecTy>(Ty)) {
|
|
|
|
int64_t Value = getValue();
|
|
|
|
// Make sure this bitfield is large enough to hold the integer value.
|
|
|
|
if (!canFitInBitfield(Value, BRT->getNumBits()))
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
SmallVector<Init *, 16> NewBits(BRT->getNumBits());
|
|
|
|
for (unsigned i = 0; i != BRT->getNumBits(); ++i)
|
|
|
|
NewBits[i] = BitInit::get(Value & (1LL << i));
|
|
|
|
|
|
|
|
return BitsInit::get(NewBits);
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *
|
2011-07-29 19:07:05 +00:00
|
|
|
IntInit::convertInitializerBitRange(const std::vector<unsigned> &Bits) const {
|
2011-07-29 22:43:06 +00:00
|
|
|
SmallVector<Init *, 16> NewBits(Bits.size());
|
2002-12-02 01:23:04 +00:00
|
|
|
|
|
|
|
for (unsigned i = 0, e = Bits.size(); i != e; ++i) {
|
2011-07-29 19:07:00 +00:00
|
|
|
if (Bits[i] >= 64)
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2011-07-29 19:07:00 +00:00
|
|
|
|
2011-07-29 19:07:07 +00:00
|
|
|
NewBits[i] = BitInit::get(Value & (INT64_C(1) << Bits[i]));
|
2011-07-11 18:25:51 +00:00
|
|
|
}
|
2011-07-29 19:07:07 +00:00
|
|
|
return BitsInit::get(NewBits);
|
|
|
|
}
|
|
|
|
|
2016-07-05 21:22:55 +00:00
|
|
|
CodeInit *CodeInit::get(StringRef V) {
|
2016-12-04 05:48:20 +00:00
|
|
|
static DenseMap<StringRef, CodeInit*> ThePool;
|
2016-07-05 21:22:55 +00:00
|
|
|
|
2016-12-04 05:48:20 +00:00
|
|
|
auto I = ThePool.insert(std::make_pair(V, nullptr));
|
|
|
|
if (I.second) {
|
|
|
|
StringRef VCopy = V.copy(Allocator);
|
|
|
|
I.first->first = VCopy;
|
|
|
|
I.first->second = new(Allocator) CodeInit(VCopy);
|
|
|
|
}
|
|
|
|
return I.first->second;
|
2016-07-05 21:22:55 +00:00
|
|
|
}
|
|
|
|
|
2012-01-13 03:16:35 +00:00
|
|
|
StringInit *StringInit::get(StringRef V) {
|
2016-12-04 05:48:20 +00:00
|
|
|
static DenseMap<StringRef, StringInit*> ThePool;
|
2011-07-29 19:07:14 +00:00
|
|
|
|
2016-12-04 05:48:20 +00:00
|
|
|
auto I = ThePool.insert(std::make_pair(V, nullptr));
|
|
|
|
if (I.second) {
|
|
|
|
StringRef VCopy = V.copy(Allocator);
|
|
|
|
I.first->first = VCopy;
|
|
|
|
I.first->second = new(Allocator) StringInit(VCopy);
|
|
|
|
}
|
|
|
|
return I.first->second;
|
2011-07-29 19:07:07 +00:00
|
|
|
}
|
|
|
|
|
2015-05-30 07:34:51 +00:00
|
|
|
Init *StringInit::convertInitializerTo(RecTy *Ty) const {
|
|
|
|
if (isa<StringRecTy>(Ty))
|
|
|
|
return const_cast<StringInit *>(this);
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2016-07-05 21:22:55 +00:00
|
|
|
Init *CodeInit::convertInitializerTo(RecTy *Ty) const {
|
|
|
|
if (isa<CodeRecTy>(Ty))
|
|
|
|
return const_cast<CodeInit *>(this);
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2011-07-29 19:07:16 +00:00
|
|
|
static void ProfileListInit(FoldingSetNodeID &ID,
|
2011-07-29 22:43:06 +00:00
|
|
|
ArrayRef<Init *> Range,
|
2011-07-29 19:07:16 +00:00
|
|
|
RecTy *EltTy) {
|
|
|
|
ID.AddInteger(Range.size());
|
|
|
|
ID.AddPointer(EltTy);
|
|
|
|
|
2015-04-29 07:13:14 +00:00
|
|
|
for (Init *I : Range)
|
|
|
|
ID.AddPointer(I);
|
2011-07-29 19:07:16 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
ListInit *ListInit::get(ArrayRef<Init *> Range, RecTy *EltTy) {
|
2015-04-24 05:38:48 +00:00
|
|
|
static FoldingSet<ListInit> ThePool;
|
2016-12-04 05:48:20 +00:00
|
|
|
static std::vector<ListInit*> TheActualPool;
|
2011-07-29 19:07:16 +00:00
|
|
|
|
|
|
|
FoldingSetNodeID ID;
|
|
|
|
ProfileListInit(ID, Range, EltTy);
|
|
|
|
|
2014-04-09 04:50:04 +00:00
|
|
|
void *IP = nullptr;
|
2011-07-29 22:43:06 +00:00
|
|
|
if (ListInit *I = ThePool.FindNodeOrInsertPos(ID, IP))
|
2011-07-29 19:07:16 +00:00
|
|
|
return I;
|
|
|
|
|
2016-12-04 05:48:20 +00:00
|
|
|
void *Mem = Allocator.Allocate(totalSizeToAlloc<Init *>(Range.size()),
|
|
|
|
alignof(ListInit));
|
|
|
|
ListInit *I = new(Mem) ListInit(Range.size(), EltTy);
|
2016-01-18 19:52:24 +00:00
|
|
|
std::uninitialized_copy(Range.begin(), Range.end(),
|
|
|
|
I->getTrailingObjects<Init *>());
|
2011-07-29 19:07:16 +00:00
|
|
|
ThePool.InsertNode(I, IP);
|
2016-12-04 05:48:20 +00:00
|
|
|
TheActualPool.push_back(I);
|
2011-07-29 19:07:16 +00:00
|
|
|
return I;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ListInit::Profile(FoldingSetNodeID &ID) const {
|
2015-04-29 07:13:05 +00:00
|
|
|
RecTy *EltTy = cast<ListRecTy>(getType())->getElementType();
|
2011-07-29 19:07:16 +00:00
|
|
|
|
2016-01-18 19:52:24 +00:00
|
|
|
ProfileListInit(ID, getValues(), EltTy);
|
2011-07-11 18:25:51 +00:00
|
|
|
}
|
|
|
|
|
2015-05-30 07:34:51 +00:00
|
|
|
Init *ListInit::convertInitializerTo(RecTy *Ty) const {
|
2016-12-05 06:41:47 +00:00
|
|
|
if (getType() == Ty)
|
|
|
|
return const_cast<ListInit*>(this);
|
|
|
|
|
2015-05-30 07:34:51 +00:00
|
|
|
if (auto *LRT = dyn_cast<ListRecTy>(Ty)) {
|
|
|
|
std::vector<Init*> Elements;
|
2016-12-05 06:41:47 +00:00
|
|
|
Elements.reserve(getValues().size());
|
2015-05-30 07:34:51 +00:00
|
|
|
|
|
|
|
// Verify that all of the elements of the list are subclasses of the
|
|
|
|
// appropriate class!
|
2016-12-05 06:41:47 +00:00
|
|
|
bool Changed = false;
|
|
|
|
RecTy *ElementType = LRT->getElementType();
|
2015-06-02 04:15:51 +00:00
|
|
|
for (Init *I : getValues())
|
2016-12-05 06:41:47 +00:00
|
|
|
if (Init *CI = I->convertInitializerTo(ElementType)) {
|
2015-05-30 07:34:51 +00:00
|
|
|
Elements.push_back(CI);
|
2016-12-05 06:41:47 +00:00
|
|
|
if (CI != I)
|
|
|
|
Changed = true;
|
|
|
|
} else
|
2015-05-30 07:34:51 +00:00
|
|
|
return nullptr;
|
|
|
|
|
2016-12-05 06:41:47 +00:00
|
|
|
if (!Changed)
|
|
|
|
return const_cast<ListInit*>(this);
|
|
|
|
return ListInit::get(Elements, Ty);
|
2015-05-30 07:34:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *
|
2011-07-29 19:07:05 +00:00
|
|
|
ListInit::convertInitListSlice(const std::vector<unsigned> &Elements) const {
|
2011-07-29 22:43:06 +00:00
|
|
|
std::vector<Init*> Vals;
|
Add initial support for list slices. This currently allows you to do stuff
like this:
def B {
list<int> X = [10, 20, 30, 4, 1, 1231, 20] [2-4,2,2,0-6];
}
... which isn't particularly useful, but more is to come.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@15247 91177308-0d34-0410-b5e6-96231b3b80d8
2004-07-26 23:21:34 +00:00
|
|
|
for (unsigned i = 0, e = Elements.size(); i != e; ++i) {
|
2015-06-02 04:15:57 +00:00
|
|
|
if (Elements[i] >= size())
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
Add initial support for list slices. This currently allows you to do stuff
like this:
def B {
list<int> X = [10, 20, 30, 4, 1, 1231, 20] [2-4,2,2,0-6];
}
... which isn't particularly useful, but more is to come.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@15247 91177308-0d34-0410-b5e6-96231b3b80d8
2004-07-26 23:21:34 +00:00
|
|
|
Vals.push_back(getElement(Elements[i]));
|
|
|
|
}
|
2011-07-29 19:07:07 +00:00
|
|
|
return ListInit::get(Vals, getType());
|
Add initial support for list slices. This currently allows you to do stuff
like this:
def B {
list<int> X = [10, 20, 30, 4, 1, 1231, 20] [2-4,2,2,0-6];
}
... which isn't particularly useful, but more is to come.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@15247 91177308-0d34-0410-b5e6-96231b3b80d8
2004-07-26 23:21:34 +00:00
|
|
|
}
|
|
|
|
|
2007-02-27 22:08:27 +00:00
|
|
|
Record *ListInit::getElementAsRecord(unsigned i) const {
|
2016-01-18 19:59:05 +00:00
|
|
|
assert(i < NumValues && "List element index out of range!");
|
2016-01-18 19:52:24 +00:00
|
|
|
DefInit *DI = dyn_cast<DefInit>(getElement(i));
|
2014-04-09 04:50:04 +00:00
|
|
|
if (!DI)
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError("Expected record in list!");
|
2007-02-27 22:08:27 +00:00
|
|
|
return DI->getDef();
|
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *ListInit::resolveReferences(Record &R, const RecordVal *RV) const {
|
|
|
|
std::vector<Init*> Resolved;
|
2015-06-02 04:15:57 +00:00
|
|
|
Resolved.reserve(size());
|
2004-07-27 01:01:21 +00:00
|
|
|
bool Changed = false;
|
|
|
|
|
2015-06-02 04:15:51 +00:00
|
|
|
for (Init *CurElt : getValues()) {
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *E;
|
2004-07-27 01:01:21 +00:00
|
|
|
|
|
|
|
do {
|
|
|
|
E = CurElt;
|
2005-04-19 03:36:21 +00:00
|
|
|
CurElt = CurElt->resolveReferences(R, RV);
|
2004-07-27 01:01:21 +00:00
|
|
|
Changed |= E != CurElt;
|
|
|
|
} while (E != CurElt);
|
|
|
|
Resolved.push_back(E);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Changed)
|
2011-07-29 19:07:07 +00:00
|
|
|
return ListInit::get(Resolved, getType());
|
2011-07-29 22:43:06 +00:00
|
|
|
return const_cast<ListInit *>(this);
|
2004-07-27 01:01:21 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *ListInit::resolveListElementReference(Record &R, const RecordVal *IRV,
|
|
|
|
unsigned Elt) const {
|
2015-06-02 04:15:57 +00:00
|
|
|
if (Elt >= size())
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr; // Out of range reference.
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *E = getElement(Elt);
|
2009-11-22 03:58:57 +00:00
|
|
|
// If the element is set to some value, or if we are resolving a reference
|
|
|
|
// to a specific variable and that variable is explicitly unset, then
|
|
|
|
// replace the VarListElementInit with it.
|
2012-10-10 20:24:47 +00:00
|
|
|
if (IRV || !isa<UnsetInit>(E))
|
2009-11-22 03:58:57 +00:00
|
|
|
return E;
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2009-06-08 20:23:18 +00:00
|
|
|
}
|
|
|
|
|
2007-11-22 21:05:25 +00:00
|
|
|
std::string ListInit::getAsString() const {
|
|
|
|
std::string Result = "[";
|
2016-01-18 19:52:24 +00:00
|
|
|
for (unsigned i = 0, e = NumValues; i != e; ++i) {
|
2007-11-22 21:05:25 +00:00
|
|
|
if (i) Result += ", ";
|
2016-01-18 19:52:24 +00:00
|
|
|
Result += getElement(i)->getAsString();
|
2002-12-02 01:23:04 +00:00
|
|
|
}
|
2007-11-22 21:05:25 +00:00
|
|
|
return Result + "]";
|
2002-12-02 01:23:04 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *OpInit::resolveListElementReference(Record &R, const RecordVal *IRV,
|
|
|
|
unsigned Elt) const {
|
2011-10-04 18:55:36 +00:00
|
|
|
Init *Resolved = resolveReferences(R, IRV);
|
2012-10-10 20:24:43 +00:00
|
|
|
OpInit *OResolved = dyn_cast<OpInit>(Resolved);
|
2011-10-04 18:55:36 +00:00
|
|
|
if (OResolved) {
|
2014-04-09 04:50:04 +00:00
|
|
|
Resolved = OResolved->Fold(&R, nullptr);
|
2011-10-04 18:55:36 +00:00
|
|
|
}
|
2009-05-14 20:54:48 +00:00
|
|
|
|
2011-10-04 18:55:36 +00:00
|
|
|
if (Resolved != this) {
|
2015-04-29 07:13:05 +00:00
|
|
|
TypedInit *Typed = cast<TypedInit>(Resolved);
|
|
|
|
if (Init *New = Typed->resolveListElementReference(R, IRV, Elt))
|
|
|
|
return New;
|
|
|
|
return VarListElementInit::get(Typed, Elt);
|
2009-05-14 20:54:48 +00:00
|
|
|
}
|
2009-11-22 04:24:42 +00:00
|
|
|
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2009-05-14 20:54:48 +00:00
|
|
|
}
|
|
|
|
|
Re-work bit/bits value resolving in tblgen
- This patch is inspired by the failure of the following code snippet
which is used to convert enumerable values into encoding bits to
improve the readability of td files.
class S<int s> {
bits<2> V = !if(!eq(s, 8), {0, 0},
!if(!eq(s, 16), {0, 1},
!if(!eq(s, 32), {1, 0},
!if(!eq(s, 64), {1, 1}, {?, ?}))));
}
Later, PR8330 is found to report not exactly the same bug relevant
issue to bit/bits values.
- Instead of resolving bit/bits values separately through
resolveBitReference(), this patch adds getBit() for all Inits and
resolves bit value by resolving plus getting the specified bit. This
unifies the resolving of bit with other values and removes redundant
logic for resolving bit only. In addition,
BitsInit::resolveReferences() is optimized to take advantage of this
origanization by resolving VarBitInit's variable reference first and
then getting bits from it.
- The type interference in '!if' operator is revised to support possible
combinations of int and bits/bit in MHS and RHS.
- As there may be illegal assignments from integer value to bit, says
assign 2 to a bit, but we only check this during instantiation in some
cases, e.g.
bit V = !if(!eq(x, 17), 0, 2);
Verbose diagnostic message is generated when invalid value is
resolveed to help locating the error.
- PR8330 is fixed as well.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163360 91177308-0d34-0410-b5e6-96231b3b80d8
2012-09-06 23:32:48 +00:00
|
|
|
Init *OpInit::getBit(unsigned Bit) const {
|
|
|
|
if (getType() == BitRecTy::get())
|
|
|
|
return const_cast<OpInit*>(this);
|
|
|
|
return VarBitInit::get(const_cast<OpInit*>(this), Bit);
|
|
|
|
}
|
|
|
|
|
2016-01-18 20:36:06 +00:00
|
|
|
static void
|
|
|
|
ProfileUnOpInit(FoldingSetNodeID &ID, unsigned Opcode, Init *Op, RecTy *Type) {
|
|
|
|
ID.AddInteger(Opcode);
|
|
|
|
ID.AddPointer(Op);
|
|
|
|
ID.AddPointer(Type);
|
|
|
|
}
|
2011-07-29 19:07:18 +00:00
|
|
|
|
2016-01-18 20:36:06 +00:00
|
|
|
UnOpInit *UnOpInit::get(UnaryOp Opc, Init *LHS, RecTy *Type) {
|
|
|
|
static FoldingSet<UnOpInit> ThePool;
|
2016-12-04 05:48:20 +00:00
|
|
|
static std::vector<UnOpInit*> TheActualPool;
|
2011-07-29 19:07:18 +00:00
|
|
|
|
2016-01-18 20:36:06 +00:00
|
|
|
FoldingSetNodeID ID;
|
|
|
|
ProfileUnOpInit(ID, Opc, LHS, Type);
|
|
|
|
|
|
|
|
void *IP = nullptr;
|
|
|
|
if (UnOpInit *I = ThePool.FindNodeOrInsertPos(ID, IP))
|
|
|
|
return I;
|
|
|
|
|
2016-12-04 05:48:20 +00:00
|
|
|
UnOpInit *I = new(Allocator) UnOpInit(Opc, LHS, Type);
|
2016-01-18 20:36:06 +00:00
|
|
|
ThePool.InsertNode(I, IP);
|
2016-12-04 05:48:20 +00:00
|
|
|
TheActualPool.push_back(I);
|
2016-01-18 20:36:06 +00:00
|
|
|
return I;
|
|
|
|
}
|
|
|
|
|
|
|
|
void UnOpInit::Profile(FoldingSetNodeID &ID) const {
|
|
|
|
ProfileUnOpInit(ID, getOpcode(), getOperand(), getType());
|
2011-07-29 19:07:07 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *UnOpInit::Fold(Record *CurRec, MultiClass *CurMultiClass) const {
|
2009-05-14 21:22:49 +00:00
|
|
|
switch (getOpcode()) {
|
|
|
|
case CAST: {
|
2015-05-28 06:38:28 +00:00
|
|
|
if (isa<StringRecTy>(getType())) {
|
2012-10-10 20:24:47 +00:00
|
|
|
if (StringInit *LHSs = dyn_cast<StringInit>(LHS))
|
2009-06-29 20:05:29 +00:00
|
|
|
return LHSs;
|
2009-05-14 21:22:49 +00:00
|
|
|
|
2012-10-10 20:24:47 +00:00
|
|
|
if (DefInit *LHSd = dyn_cast<DefInit>(LHS))
|
2015-05-28 06:38:32 +00:00
|
|
|
return StringInit::get(LHSd->getAsString());
|
2012-01-30 20:47:04 +00:00
|
|
|
|
2012-10-10 20:24:47 +00:00
|
|
|
if (IntInit *LHSi = dyn_cast<IntInit>(LHS))
|
2012-01-30 20:47:04 +00:00
|
|
|
return StringInit::get(LHSi->getAsString());
|
2009-11-22 04:24:42 +00:00
|
|
|
} else {
|
2016-12-05 06:00:36 +00:00
|
|
|
if (StringInit *Name = dyn_cast<StringInit>(LHS)) {
|
2009-06-29 20:05:29 +00:00
|
|
|
// From TGParser::ParseIDValue
|
|
|
|
if (CurRec) {
|
|
|
|
if (const RecordVal *RV = CurRec->getValue(Name)) {
|
2010-10-06 00:19:21 +00:00
|
|
|
if (RV->getType() != getType())
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError("type mismatch in cast");
|
2011-07-29 19:07:07 +00:00
|
|
|
return VarInit::get(Name, RV->getType());
|
2009-05-14 21:22:49 +00:00
|
|
|
}
|
|
|
|
|
2011-10-19 13:02:42 +00:00
|
|
|
Init *TemplateArgName = QualifyName(*CurRec, CurMultiClass, Name,
|
|
|
|
":");
|
2015-05-04 01:35:39 +00:00
|
|
|
|
2009-06-29 20:05:29 +00:00
|
|
|
if (CurRec->isTemplateArg(TemplateArgName)) {
|
|
|
|
const RecordVal *RV = CurRec->getValue(TemplateArgName);
|
|
|
|
assert(RV && "Template arg doesn't exist??");
|
2009-05-14 21:22:49 +00:00
|
|
|
|
2010-10-06 00:19:21 +00:00
|
|
|
if (RV->getType() != getType())
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError("type mismatch in cast");
|
2009-05-14 21:22:49 +00:00
|
|
|
|
2011-07-29 19:07:07 +00:00
|
|
|
return VarInit::get(TemplateArgName, RV->getType());
|
2009-05-14 21:22:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-06-29 20:05:29 +00:00
|
|
|
if (CurMultiClass) {
|
2015-05-04 01:35:39 +00:00
|
|
|
Init *MCName = QualifyName(CurMultiClass->Rec, CurMultiClass, Name,
|
|
|
|
"::");
|
2011-10-19 13:02:42 +00:00
|
|
|
|
2009-06-29 20:05:29 +00:00
|
|
|
if (CurMultiClass->Rec.isTemplateArg(MCName)) {
|
|
|
|
const RecordVal *RV = CurMultiClass->Rec.getValue(MCName);
|
|
|
|
assert(RV && "Template arg doesn't exist??");
|
2009-11-22 04:24:42 +00:00
|
|
|
|
2010-10-06 00:19:21 +00:00
|
|
|
if (RV->getType() != getType())
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError("type mismatch in cast");
|
2009-11-22 04:24:42 +00:00
|
|
|
|
2011-07-29 19:07:07 +00:00
|
|
|
return VarInit::get(MCName, RV->getType());
|
2009-06-29 20:05:29 +00:00
|
|
|
}
|
|
|
|
}
|
2014-12-12 21:48:03 +00:00
|
|
|
assert(CurRec && "NULL pointer");
|
2016-12-05 06:00:36 +00:00
|
|
|
if (Record *D = (CurRec->getRecords()).getDef(Name->getValue()))
|
2011-07-18 17:02:57 +00:00
|
|
|
return DefInit::get(D);
|
2009-05-14 21:22:49 +00:00
|
|
|
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(CurRec->getLoc(),
|
2016-12-05 06:00:36 +00:00
|
|
|
"Undefined reference:'" + Name->getValue() + "'\n");
|
2009-06-29 20:05:29 +00:00
|
|
|
}
|
2015-07-31 01:12:06 +00:00
|
|
|
|
|
|
|
if (isa<IntRecTy>(getType())) {
|
|
|
|
if (BitsInit *BI = dyn_cast<BitsInit>(LHS)) {
|
|
|
|
if (Init *NewInit = BI->convertInitializerTo(IntRecTy::get()))
|
|
|
|
return NewInit;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2009-05-14 21:22:49 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2011-01-07 17:05:37 +00:00
|
|
|
case HEAD: {
|
2012-10-10 20:24:47 +00:00
|
|
|
if (ListInit *LHSl = dyn_cast<ListInit>(LHS)) {
|
2015-05-16 05:42:13 +00:00
|
|
|
assert(!LHSl->empty() && "Empty list in head");
|
2009-05-14 22:38:31 +00:00
|
|
|
return LHSl->getElement(0);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2011-01-07 17:05:37 +00:00
|
|
|
case TAIL: {
|
2012-10-10 20:24:47 +00:00
|
|
|
if (ListInit *LHSl = dyn_cast<ListInit>(LHS)) {
|
2015-05-16 05:42:13 +00:00
|
|
|
assert(!LHSl->empty() && "Empty list in tail");
|
2011-07-29 19:07:07 +00:00
|
|
|
// Note the +1. We can't just pass the result of getValues()
|
|
|
|
// directly.
|
2015-05-13 06:57:51 +00:00
|
|
|
return ListInit::get(LHSl->getValues().slice(1), LHSl->getType());
|
2009-05-14 22:38:31 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2011-01-07 17:05:37 +00:00
|
|
|
case EMPTY: {
|
2015-05-16 05:42:11 +00:00
|
|
|
if (ListInit *LHSl = dyn_cast<ListInit>(LHS))
|
|
|
|
return IntInit::get(LHSl->empty());
|
|
|
|
if (StringInit *LHSs = dyn_cast<StringInit>(LHS))
|
|
|
|
return IntInit::get(LHSs->getValue().empty());
|
2009-11-22 04:24:42 +00:00
|
|
|
|
2009-05-14 22:38:31 +00:00
|
|
|
break;
|
|
|
|
}
|
2009-05-14 21:22:49 +00:00
|
|
|
}
|
2011-07-29 22:43:06 +00:00
|
|
|
return const_cast<UnOpInit *>(this);
|
2009-05-14 21:22:49 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *UnOpInit::resolveReferences(Record &R, const RecordVal *RV) const {
|
|
|
|
Init *lhs = LHS->resolveReferences(R, RV);
|
2009-11-22 04:24:42 +00:00
|
|
|
|
2009-05-14 21:22:49 +00:00
|
|
|
if (LHS != lhs)
|
2014-04-09 04:50:04 +00:00
|
|
|
return (UnOpInit::get(getOpcode(), lhs, getType()))->Fold(&R, nullptr);
|
|
|
|
return Fold(&R, nullptr);
|
2009-05-14 21:22:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string UnOpInit::getAsString() const {
|
|
|
|
std::string Result;
|
2016-01-04 06:28:49 +00:00
|
|
|
switch (getOpcode()) {
|
2009-05-14 21:22:49 +00:00
|
|
|
case CAST: Result = "!cast<" + getType()->getAsString() + ">"; break;
|
2011-01-07 17:05:37 +00:00
|
|
|
case HEAD: Result = "!head"; break;
|
|
|
|
case TAIL: Result = "!tail"; break;
|
|
|
|
case EMPTY: Result = "!empty"; break;
|
2009-05-14 21:22:49 +00:00
|
|
|
}
|
|
|
|
return Result + "(" + LHS->getAsString() + ")";
|
|
|
|
}
|
2009-05-14 20:54:48 +00:00
|
|
|
|
2016-01-18 20:36:06 +00:00
|
|
|
static void
|
|
|
|
ProfileBinOpInit(FoldingSetNodeID &ID, unsigned Opcode, Init *LHS, Init *RHS,
|
|
|
|
RecTy *Type) {
|
|
|
|
ID.AddInteger(Opcode);
|
|
|
|
ID.AddPointer(LHS);
|
|
|
|
ID.AddPointer(RHS);
|
|
|
|
ID.AddPointer(Type);
|
|
|
|
}
|
2011-07-29 19:07:19 +00:00
|
|
|
|
2016-01-18 20:36:06 +00:00
|
|
|
BinOpInit *BinOpInit::get(BinaryOp Opc, Init *LHS,
|
|
|
|
Init *RHS, RecTy *Type) {
|
|
|
|
static FoldingSet<BinOpInit> ThePool;
|
2016-12-04 05:48:20 +00:00
|
|
|
static std::vector<BinOpInit*> TheActualPool;
|
2011-07-29 19:07:19 +00:00
|
|
|
|
2016-01-18 20:36:06 +00:00
|
|
|
FoldingSetNodeID ID;
|
|
|
|
ProfileBinOpInit(ID, Opc, LHS, RHS, Type);
|
2011-07-29 19:07:19 +00:00
|
|
|
|
2016-01-18 20:36:06 +00:00
|
|
|
void *IP = nullptr;
|
|
|
|
if (BinOpInit *I = ThePool.FindNodeOrInsertPos(ID, IP))
|
|
|
|
return I;
|
|
|
|
|
2016-12-04 05:48:20 +00:00
|
|
|
BinOpInit *I = new(Allocator) BinOpInit(Opc, LHS, RHS, Type);
|
2016-01-18 20:36:06 +00:00
|
|
|
ThePool.InsertNode(I, IP);
|
2016-12-04 05:48:20 +00:00
|
|
|
TheActualPool.push_back(I);
|
2016-01-18 20:36:06 +00:00
|
|
|
return I;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BinOpInit::Profile(FoldingSetNodeID &ID) const {
|
|
|
|
ProfileBinOpInit(ID, getOpcode(), getLHS(), getRHS(), getType());
|
2011-07-29 19:07:07 +00:00
|
|
|
}
|
|
|
|
|
2016-12-05 05:21:18 +00:00
|
|
|
static StringInit *ConcatStringInits(const StringInit *I0,
|
|
|
|
const StringInit *I1) {
|
|
|
|
SmallString<80> Concat(I0->getValue());
|
|
|
|
Concat.append(I1->getValue());
|
|
|
|
return StringInit::get(Concat);
|
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *BinOpInit::Fold(Record *CurRec, MultiClass *CurMultiClass) const {
|
2006-03-31 21:53:49 +00:00
|
|
|
switch (getOpcode()) {
|
2007-05-15 01:23:24 +00:00
|
|
|
case CONCAT: {
|
2012-10-10 20:24:43 +00:00
|
|
|
DagInit *LHSs = dyn_cast<DagInit>(LHS);
|
|
|
|
DagInit *RHSs = dyn_cast<DagInit>(RHS);
|
2007-05-15 01:23:24 +00:00
|
|
|
if (LHSs && RHSs) {
|
2012-10-10 20:24:43 +00:00
|
|
|
DefInit *LOp = dyn_cast<DefInit>(LHSs->getOperator());
|
|
|
|
DefInit *ROp = dyn_cast<DefInit>(RHSs->getOperator());
|
2014-04-09 04:50:04 +00:00
|
|
|
if (!LOp || !ROp || LOp->getDef() != ROp->getDef())
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError("Concated Dag operators do not match!");
|
2011-07-29 22:43:06 +00:00
|
|
|
std::vector<Init*> Args;
|
2016-12-05 06:00:46 +00:00
|
|
|
std::vector<StringInit*> ArgNames;
|
2007-05-15 01:23:24 +00:00
|
|
|
for (unsigned i = 0, e = LHSs->getNumArgs(); i != e; ++i) {
|
|
|
|
Args.push_back(LHSs->getArg(i));
|
|
|
|
ArgNames.push_back(LHSs->getArgName(i));
|
|
|
|
}
|
|
|
|
for (unsigned i = 0, e = RHSs->getNumArgs(); i != e; ++i) {
|
|
|
|
Args.push_back(RHSs->getArg(i));
|
|
|
|
ArgNames.push_back(RHSs->getArgName(i));
|
|
|
|
}
|
2016-12-05 06:00:41 +00:00
|
|
|
return DagInit::get(LHSs->getOperator(), nullptr, Args, ArgNames);
|
2007-05-15 01:23:24 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2014-05-07 10:13:19 +00:00
|
|
|
case LISTCONCAT: {
|
|
|
|
ListInit *LHSs = dyn_cast<ListInit>(LHS);
|
|
|
|
ListInit *RHSs = dyn_cast<ListInit>(RHS);
|
|
|
|
if (LHSs && RHSs) {
|
|
|
|
std::vector<Init *> Args;
|
|
|
|
Args.insert(Args.end(), LHSs->begin(), LHSs->end());
|
|
|
|
Args.insert(Args.end(), RHSs->begin(), RHSs->end());
|
|
|
|
return ListInit::get(
|
2015-05-04 01:35:42 +00:00
|
|
|
Args, cast<ListRecTy>(LHSs->getType())->getElementType());
|
2014-05-07 10:13:19 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2006-03-31 21:53:49 +00:00
|
|
|
case STRCONCAT: {
|
2012-10-10 20:24:43 +00:00
|
|
|
StringInit *LHSs = dyn_cast<StringInit>(LHS);
|
|
|
|
StringInit *RHSs = dyn_cast<StringInit>(RHS);
|
2016-12-05 05:21:18 +00:00
|
|
|
if (LHSs && RHSs)
|
|
|
|
return ConcatStringInits(LHSs, RHSs);
|
2006-03-31 21:53:49 +00:00
|
|
|
break;
|
|
|
|
}
|
2010-01-05 19:11:42 +00:00
|
|
|
case EQ: {
|
2010-06-16 23:24:12 +00:00
|
|
|
// try to fold eq comparison for 'bit' and 'int', otherwise fallback
|
|
|
|
// to string objects.
|
Re-work bit/bits value resolving in tblgen
- This patch is inspired by the failure of the following code snippet
which is used to convert enumerable values into encoding bits to
improve the readability of td files.
class S<int s> {
bits<2> V = !if(!eq(s, 8), {0, 0},
!if(!eq(s, 16), {0, 1},
!if(!eq(s, 32), {1, 0},
!if(!eq(s, 64), {1, 1}, {?, ?}))));
}
Later, PR8330 is found to report not exactly the same bug relevant
issue to bit/bits values.
- Instead of resolving bit/bits values separately through
resolveBitReference(), this patch adds getBit() for all Inits and
resolves bit value by resolving plus getting the specified bit. This
unifies the resolving of bit with other values and removes redundant
logic for resolving bit only. In addition,
BitsInit::resolveReferences() is optimized to take advantage of this
origanization by resolving VarBitInit's variable reference first and
then getting bits from it.
- The type interference in '!if' operator is revised to support possible
combinations of int and bits/bit in MHS and RHS.
- As there may be illegal assignments from integer value to bit, says
assign 2 to a bit, but we only check this during instantiation in some
cases, e.g.
bit V = !if(!eq(x, 17), 0, 2);
Verbose diagnostic message is generated when invalid value is
resolveed to help locating the error.
- PR8330 is fixed as well.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163360 91177308-0d34-0410-b5e6-96231b3b80d8
2012-09-06 23:32:48 +00:00
|
|
|
IntInit *L =
|
2012-10-10 20:24:43 +00:00
|
|
|
dyn_cast_or_null<IntInit>(LHS->convertInitializerTo(IntRecTy::get()));
|
Re-work bit/bits value resolving in tblgen
- This patch is inspired by the failure of the following code snippet
which is used to convert enumerable values into encoding bits to
improve the readability of td files.
class S<int s> {
bits<2> V = !if(!eq(s, 8), {0, 0},
!if(!eq(s, 16), {0, 1},
!if(!eq(s, 32), {1, 0},
!if(!eq(s, 64), {1, 1}, {?, ?}))));
}
Later, PR8330 is found to report not exactly the same bug relevant
issue to bit/bits values.
- Instead of resolving bit/bits values separately through
resolveBitReference(), this patch adds getBit() for all Inits and
resolves bit value by resolving plus getting the specified bit. This
unifies the resolving of bit with other values and removes redundant
logic for resolving bit only. In addition,
BitsInit::resolveReferences() is optimized to take advantage of this
origanization by resolving VarBitInit's variable reference first and
then getting bits from it.
- The type interference in '!if' operator is revised to support possible
combinations of int and bits/bit in MHS and RHS.
- As there may be illegal assignments from integer value to bit, says
assign 2 to a bit, but we only check this during instantiation in some
cases, e.g.
bit V = !if(!eq(x, 17), 0, 2);
Verbose diagnostic message is generated when invalid value is
resolveed to help locating the error.
- PR8330 is fixed as well.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163360 91177308-0d34-0410-b5e6-96231b3b80d8
2012-09-06 23:32:48 +00:00
|
|
|
IntInit *R =
|
2012-10-10 20:24:43 +00:00
|
|
|
dyn_cast_or_null<IntInit>(RHS->convertInitializerTo(IntRecTy::get()));
|
2010-06-16 23:24:12 +00:00
|
|
|
|
|
|
|
if (L && R)
|
2011-07-29 19:07:07 +00:00
|
|
|
return IntInit::get(L->getValue() == R->getValue());
|
2010-06-16 23:24:12 +00:00
|
|
|
|
2012-10-10 20:24:43 +00:00
|
|
|
StringInit *LHSs = dyn_cast<StringInit>(LHS);
|
|
|
|
StringInit *RHSs = dyn_cast<StringInit>(RHS);
|
2010-06-16 23:24:12 +00:00
|
|
|
|
|
|
|
// Make sure we've resolved
|
2010-01-05 19:11:42 +00:00
|
|
|
if (LHSs && RHSs)
|
2011-07-29 19:07:07 +00:00
|
|
|
return IntInit::get(LHSs->getValue() == RHSs->getValue());
|
2010-01-05 19:11:42 +00:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2013-01-25 14:49:08 +00:00
|
|
|
case ADD:
|
2014-08-05 09:43:25 +00:00
|
|
|
case AND:
|
2016-11-15 06:49:28 +00:00
|
|
|
case OR:
|
2006-03-31 21:53:49 +00:00
|
|
|
case SHL:
|
|
|
|
case SRA:
|
|
|
|
case SRL: {
|
2014-07-17 17:04:27 +00:00
|
|
|
IntInit *LHSi =
|
|
|
|
dyn_cast_or_null<IntInit>(LHS->convertInitializerTo(IntRecTy::get()));
|
|
|
|
IntInit *RHSi =
|
|
|
|
dyn_cast_or_null<IntInit>(RHS->convertInitializerTo(IntRecTy::get()));
|
2006-03-31 21:53:49 +00:00
|
|
|
if (LHSi && RHSi) {
|
2008-10-17 01:33:43 +00:00
|
|
|
int64_t LHSv = LHSi->getValue(), RHSv = RHSi->getValue();
|
|
|
|
int64_t Result;
|
2006-03-31 21:53:49 +00:00
|
|
|
switch (getOpcode()) {
|
2012-02-07 05:05:23 +00:00
|
|
|
default: llvm_unreachable("Bad opcode!");
|
2013-01-25 14:49:08 +00:00
|
|
|
case ADD: Result = LHSv + RHSv; break;
|
2014-08-05 09:43:25 +00:00
|
|
|
case AND: Result = LHSv & RHSv; break;
|
2016-11-15 06:49:28 +00:00
|
|
|
case OR: Result = LHSv | RHSv; break;
|
2006-03-31 21:53:49 +00:00
|
|
|
case SHL: Result = LHSv << RHSv; break;
|
|
|
|
case SRA: Result = LHSv >> RHSv; break;
|
2008-10-17 01:33:43 +00:00
|
|
|
case SRL: Result = (uint64_t)LHSv >> (uint64_t)RHSv; break;
|
2006-03-31 21:53:49 +00:00
|
|
|
}
|
2011-07-29 19:07:07 +00:00
|
|
|
return IntInit::get(Result);
|
2006-03-31 21:53:49 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2011-07-29 22:43:06 +00:00
|
|
|
return const_cast<BinOpInit *>(this);
|
2006-03-31 21:53:49 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *BinOpInit::resolveReferences(Record &R, const RecordVal *RV) const {
|
|
|
|
Init *lhs = LHS->resolveReferences(R, RV);
|
|
|
|
Init *rhs = RHS->resolveReferences(R, RV);
|
2009-11-22 04:24:42 +00:00
|
|
|
|
2006-03-31 21:53:49 +00:00
|
|
|
if (LHS != lhs || RHS != rhs)
|
2014-04-09 04:50:04 +00:00
|
|
|
return (BinOpInit::get(getOpcode(), lhs, rhs, getType()))->Fold(&R,nullptr);
|
|
|
|
return Fold(&R, nullptr);
|
2006-03-31 21:53:49 +00:00
|
|
|
}
|
|
|
|
|
2007-11-22 21:05:25 +00:00
|
|
|
std::string BinOpInit::getAsString() const {
|
|
|
|
std::string Result;
|
2016-01-04 06:28:49 +00:00
|
|
|
switch (getOpcode()) {
|
2007-11-22 21:05:25 +00:00
|
|
|
case CONCAT: Result = "!con"; break;
|
2013-01-25 14:49:08 +00:00
|
|
|
case ADD: Result = "!add"; break;
|
2014-08-05 09:43:25 +00:00
|
|
|
case AND: Result = "!and"; break;
|
2016-11-15 06:49:28 +00:00
|
|
|
case OR: Result = "!or"; break;
|
2007-11-22 21:05:25 +00:00
|
|
|
case SHL: Result = "!shl"; break;
|
|
|
|
case SRA: Result = "!sra"; break;
|
|
|
|
case SRL: Result = "!srl"; break;
|
2010-01-05 19:11:42 +00:00
|
|
|
case EQ: Result = "!eq"; break;
|
2014-05-07 10:13:19 +00:00
|
|
|
case LISTCONCAT: Result = "!listconcat"; break;
|
2007-11-22 21:05:25 +00:00
|
|
|
case STRCONCAT: Result = "!strconcat"; break;
|
2006-03-31 21:53:49 +00:00
|
|
|
}
|
2007-11-22 21:05:25 +00:00
|
|
|
return Result + "(" + LHS->getAsString() + ", " + RHS->getAsString() + ")";
|
2006-03-31 21:53:49 +00:00
|
|
|
}
|
|
|
|
|
2016-01-18 20:36:06 +00:00
|
|
|
static void
|
|
|
|
ProfileTernOpInit(FoldingSetNodeID &ID, unsigned Opcode, Init *LHS, Init *MHS,
|
|
|
|
Init *RHS, RecTy *Type) {
|
|
|
|
ID.AddInteger(Opcode);
|
|
|
|
ID.AddPointer(LHS);
|
|
|
|
ID.AddPointer(MHS);
|
|
|
|
ID.AddPointer(RHS);
|
|
|
|
ID.AddPointer(Type);
|
|
|
|
}
|
|
|
|
|
|
|
|
TernOpInit *TernOpInit::get(TernaryOp Opc, Init *LHS, Init *MHS, Init *RHS,
|
2015-05-04 01:35:39 +00:00
|
|
|
RecTy *Type) {
|
2016-01-18 20:36:06 +00:00
|
|
|
static FoldingSet<TernOpInit> ThePool;
|
2016-12-04 05:48:20 +00:00
|
|
|
static std::vector<TernOpInit*> TheActualPool;
|
2016-01-18 20:36:06 +00:00
|
|
|
|
|
|
|
FoldingSetNodeID ID;
|
|
|
|
ProfileTernOpInit(ID, Opc, LHS, MHS, RHS, Type);
|
|
|
|
|
|
|
|
void *IP = nullptr;
|
|
|
|
if (TernOpInit *I = ThePool.FindNodeOrInsertPos(ID, IP))
|
|
|
|
return I;
|
|
|
|
|
2016-12-04 05:48:20 +00:00
|
|
|
TernOpInit *I = new(Allocator) TernOpInit(Opc, LHS, MHS, RHS, Type);
|
2016-01-18 20:36:06 +00:00
|
|
|
ThePool.InsertNode(I, IP);
|
2016-12-04 05:48:20 +00:00
|
|
|
TheActualPool.push_back(I);
|
2016-01-18 20:36:06 +00:00
|
|
|
return I;
|
|
|
|
}
|
|
|
|
|
|
|
|
void TernOpInit::Profile(FoldingSetNodeID &ID) const {
|
|
|
|
ProfileTernOpInit(ID, getOpcode(), getLHS(), getMHS(), getRHS(), getType());
|
2011-07-29 19:07:07 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
static Init *ForeachHelper(Init *LHS, Init *MHS, Init *RHS, RecTy *Type,
|
|
|
|
Record *CurRec, MultiClass *CurMultiClass);
|
2009-05-14 22:23:47 +00:00
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
static Init *EvaluateOperation(OpInit *RHSo, Init *LHS, Init *Arg,
|
|
|
|
RecTy *Type, Record *CurRec,
|
|
|
|
MultiClass *CurMultiClass) {
|
2009-05-14 22:23:47 +00:00
|
|
|
// If this is a dag, recurse
|
2015-04-30 05:12:52 +00:00
|
|
|
if (auto *TArg = dyn_cast<TypedInit>(Arg))
|
2015-05-28 06:38:28 +00:00
|
|
|
if (isa<DagRecTy>(TArg->getType()))
|
2015-04-30 05:12:52 +00:00
|
|
|
return ForeachHelper(LHS, Arg, RHSo, Type, CurRec, CurMultiClass);
|
2009-05-14 22:23:47 +00:00
|
|
|
|
2015-04-29 07:13:12 +00:00
|
|
|
std::vector<Init *> NewOperands;
|
2015-06-06 01:34:04 +00:00
|
|
|
for (unsigned i = 0; i < RHSo->getNumOperands(); ++i) {
|
2015-04-30 05:12:52 +00:00
|
|
|
if (auto *RHSoo = dyn_cast<OpInit>(RHSo->getOperand(i))) {
|
|
|
|
if (Init *Result = EvaluateOperation(RHSoo, LHS, Arg,
|
|
|
|
Type, CurRec, CurMultiClass))
|
2009-05-14 22:23:47 +00:00
|
|
|
NewOperands.push_back(Result);
|
2015-04-30 05:12:52 +00:00
|
|
|
else
|
2009-05-14 22:23:47 +00:00
|
|
|
NewOperands.push_back(Arg);
|
2009-11-22 04:24:42 +00:00
|
|
|
} else if (LHS->getAsString() == RHSo->getOperand(i)->getAsString()) {
|
2009-05-14 22:23:47 +00:00
|
|
|
NewOperands.push_back(Arg);
|
2009-11-22 04:24:42 +00:00
|
|
|
} else {
|
2009-05-14 22:23:47 +00:00
|
|
|
NewOperands.push_back(RHSo->getOperand(i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now run the operator and use its result as the new leaf
|
2011-07-29 19:07:05 +00:00
|
|
|
const OpInit *NewOp = RHSo->clone(NewOperands);
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *NewVal = NewOp->Fold(CurRec, CurMultiClass);
|
2014-04-09 04:50:04 +00:00
|
|
|
return (NewVal != NewOp) ? NewVal : nullptr;
|
2009-05-14 22:23:47 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
static Init *ForeachHelper(Init *LHS, Init *MHS, Init *RHS, RecTy *Type,
|
|
|
|
Record *CurRec, MultiClass *CurMultiClass) {
|
2009-05-14 22:23:47 +00:00
|
|
|
|
2012-10-10 20:24:43 +00:00
|
|
|
OpInit *RHSo = dyn_cast<OpInit>(RHS);
|
2009-05-14 22:23:47 +00:00
|
|
|
|
2015-05-04 01:35:39 +00:00
|
|
|
if (!RHSo)
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(CurRec->getLoc(), "!foreach requires an operator\n");
|
2009-05-14 22:23:47 +00:00
|
|
|
|
2012-10-10 20:24:43 +00:00
|
|
|
TypedInit *LHSt = dyn_cast<TypedInit>(LHS);
|
2009-05-14 22:23:47 +00:00
|
|
|
|
2012-10-25 20:33:17 +00:00
|
|
|
if (!LHSt)
|
|
|
|
PrintFatalError(CurRec->getLoc(), "!foreach requires typed variable\n");
|
2009-05-14 22:23:47 +00:00
|
|
|
|
2015-06-06 01:33:55 +00:00
|
|
|
DagInit *MHSd = dyn_cast<DagInit>(MHS);
|
|
|
|
if (MHSd && isa<DagRecTy>(Type)) {
|
|
|
|
Init *Val = MHSd->getOperator();
|
2015-06-06 01:33:58 +00:00
|
|
|
if (Init *Result = EvaluateOperation(RHSo, LHS, Val,
|
|
|
|
Type, CurRec, CurMultiClass))
|
2015-06-06 01:33:55 +00:00
|
|
|
Val = Result;
|
|
|
|
|
2016-12-05 06:00:46 +00:00
|
|
|
std::vector<std::pair<Init *, StringInit*> > args;
|
2015-06-06 01:33:55 +00:00
|
|
|
for (unsigned int i = 0; i < MHSd->getNumArgs(); ++i) {
|
|
|
|
Init *Arg = MHSd->getArg(i);
|
2016-12-05 06:00:46 +00:00
|
|
|
StringInit *ArgName = MHSd->getArgName(i);
|
2015-06-06 01:33:55 +00:00
|
|
|
|
|
|
|
// Process args
|
2015-06-06 01:33:58 +00:00
|
|
|
if (Init *Result = EvaluateOperation(RHSo, LHS, Arg, Type,
|
|
|
|
CurRec, CurMultiClass))
|
2015-06-06 01:33:55 +00:00
|
|
|
Arg = Result;
|
2009-05-14 22:23:47 +00:00
|
|
|
|
2015-06-06 01:33:55 +00:00
|
|
|
// TODO: Process arg names
|
|
|
|
args.push_back(std::make_pair(Arg, ArgName));
|
|
|
|
}
|
2009-05-14 22:23:47 +00:00
|
|
|
|
2016-12-05 06:00:41 +00:00
|
|
|
return DagInit::get(Val, nullptr, args);
|
2015-06-06 01:33:55 +00:00
|
|
|
}
|
2009-05-14 22:23:47 +00:00
|
|
|
|
2015-06-06 01:33:55 +00:00
|
|
|
ListInit *MHSl = dyn_cast<ListInit>(MHS);
|
|
|
|
if (MHSl && isa<ListRecTy>(Type)) {
|
|
|
|
std::vector<Init *> NewOperands;
|
|
|
|
std::vector<Init *> NewList(MHSl->begin(), MHSl->end());
|
|
|
|
|
|
|
|
for (Init *&Item : NewList) {
|
|
|
|
NewOperands.clear();
|
2015-06-06 01:34:04 +00:00
|
|
|
for(unsigned i = 0; i < RHSo->getNumOperands(); ++i) {
|
2015-06-06 01:33:55 +00:00
|
|
|
// First, replace the foreach variable with the list item
|
|
|
|
if (LHS->getAsString() == RHSo->getOperand(i)->getAsString())
|
|
|
|
NewOperands.push_back(Item);
|
|
|
|
else
|
|
|
|
NewOperands.push_back(RHSo->getOperand(i));
|
2009-05-14 22:23:47 +00:00
|
|
|
}
|
|
|
|
|
2015-06-06 01:33:55 +00:00
|
|
|
// Now run the operator and use its result as the new list item
|
|
|
|
const OpInit *NewOp = RHSo->clone(NewOperands);
|
|
|
|
Init *NewItem = NewOp->Fold(CurRec, CurMultiClass);
|
|
|
|
if (NewItem != NewOp)
|
|
|
|
Item = NewItem;
|
2009-05-14 22:23:47 +00:00
|
|
|
}
|
2015-06-06 01:33:55 +00:00
|
|
|
return ListInit::get(NewList, MHSl->getType());
|
2009-05-14 22:23:47 +00:00
|
|
|
}
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2009-05-14 22:23:47 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *TernOpInit::Fold(Record *CurRec, MultiClass *CurMultiClass) const {
|
2009-05-14 21:54:42 +00:00
|
|
|
switch (getOpcode()) {
|
|
|
|
case SUBST: {
|
2012-10-10 20:24:43 +00:00
|
|
|
DefInit *LHSd = dyn_cast<DefInit>(LHS);
|
|
|
|
VarInit *LHSv = dyn_cast<VarInit>(LHS);
|
|
|
|
StringInit *LHSs = dyn_cast<StringInit>(LHS);
|
2009-05-14 21:54:42 +00:00
|
|
|
|
2012-10-10 20:24:43 +00:00
|
|
|
DefInit *MHSd = dyn_cast<DefInit>(MHS);
|
|
|
|
VarInit *MHSv = dyn_cast<VarInit>(MHS);
|
|
|
|
StringInit *MHSs = dyn_cast<StringInit>(MHS);
|
2009-05-14 21:54:42 +00:00
|
|
|
|
2012-10-10 20:24:43 +00:00
|
|
|
DefInit *RHSd = dyn_cast<DefInit>(RHS);
|
|
|
|
VarInit *RHSv = dyn_cast<VarInit>(RHS);
|
|
|
|
StringInit *RHSs = dyn_cast<StringInit>(RHS);
|
2009-05-14 21:54:42 +00:00
|
|
|
|
2015-05-14 05:54:02 +00:00
|
|
|
if (LHSd && MHSd && RHSd) {
|
|
|
|
Record *Val = RHSd->getDef();
|
|
|
|
if (LHSd->getAsString() == RHSd->getAsString())
|
|
|
|
Val = MHSd->getDef();
|
|
|
|
return DefInit::get(Val);
|
|
|
|
}
|
|
|
|
if (LHSv && MHSv && RHSv) {
|
|
|
|
std::string Val = RHSv->getName();
|
|
|
|
if (LHSv->getAsString() == RHSv->getAsString())
|
|
|
|
Val = MHSv->getName();
|
|
|
|
return VarInit::get(Val, getType());
|
|
|
|
}
|
|
|
|
if (LHSs && MHSs && RHSs) {
|
|
|
|
std::string Val = RHSs->getValue();
|
|
|
|
|
|
|
|
std::string::size_type found;
|
|
|
|
std::string::size_type idx = 0;
|
2015-05-16 05:42:03 +00:00
|
|
|
while (true) {
|
2015-05-14 05:54:02 +00:00
|
|
|
found = Val.find(LHSs->getValue(), idx);
|
2015-05-16 05:42:03 +00:00
|
|
|
if (found == std::string::npos)
|
|
|
|
break;
|
|
|
|
Val.replace(found, LHSs->getValue().size(), MHSs->getValue());
|
|
|
|
idx = found + MHSs->getValue().size();
|
|
|
|
}
|
2015-05-14 05:54:02 +00:00
|
|
|
|
|
|
|
return StringInit::get(Val);
|
2009-05-14 21:54:42 +00:00
|
|
|
}
|
|
|
|
break;
|
2009-11-22 04:24:42 +00:00
|
|
|
}
|
2009-05-14 21:54:42 +00:00
|
|
|
|
|
|
|
case FOREACH: {
|
2015-05-14 05:53:59 +00:00
|
|
|
if (Init *Result = ForeachHelper(LHS, MHS, RHS, getType(),
|
|
|
|
CurRec, CurMultiClass))
|
2009-05-14 22:23:47 +00:00
|
|
|
return Result;
|
2009-05-14 21:54:42 +00:00
|
|
|
break;
|
|
|
|
}
|
2009-05-14 23:26:46 +00:00
|
|
|
|
|
|
|
case IF: {
|
2012-10-10 20:24:43 +00:00
|
|
|
IntInit *LHSi = dyn_cast<IntInit>(LHS);
|
2011-07-29 22:43:06 +00:00
|
|
|
if (Init *I = LHS->convertInitializerTo(IntRecTy::get()))
|
2012-10-10 20:24:43 +00:00
|
|
|
LHSi = dyn_cast<IntInit>(I);
|
2009-05-14 23:26:46 +00:00
|
|
|
if (LHSi) {
|
2015-05-04 01:35:39 +00:00
|
|
|
if (LHSi->getValue())
|
2009-05-14 23:26:46 +00:00
|
|
|
return MHS;
|
2015-05-04 01:35:39 +00:00
|
|
|
return RHS;
|
2009-05-14 23:26:46 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2009-05-14 21:54:42 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
return const_cast<TernOpInit *>(this);
|
2009-05-14 21:54:42 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *TernOpInit::resolveReferences(Record &R,
|
|
|
|
const RecordVal *RV) const {
|
|
|
|
Init *lhs = LHS->resolveReferences(R, RV);
|
2009-06-08 19:16:56 +00:00
|
|
|
|
2016-01-04 06:28:49 +00:00
|
|
|
if (getOpcode() == IF && lhs != LHS) {
|
2012-10-10 20:24:43 +00:00
|
|
|
IntInit *Value = dyn_cast<IntInit>(lhs);
|
2011-07-29 22:43:06 +00:00
|
|
|
if (Init *I = lhs->convertInitializerTo(IntRecTy::get()))
|
2012-10-10 20:24:43 +00:00
|
|
|
Value = dyn_cast<IntInit>(I);
|
2014-04-09 04:50:04 +00:00
|
|
|
if (Value) {
|
2009-06-08 19:16:56 +00:00
|
|
|
// Short-circuit
|
|
|
|
if (Value->getValue()) {
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *mhs = MHS->resolveReferences(R, RV);
|
2011-07-29 19:07:07 +00:00
|
|
|
return (TernOpInit::get(getOpcode(), lhs, mhs,
|
2014-04-09 04:50:04 +00:00
|
|
|
RHS, getType()))->Fold(&R, nullptr);
|
2009-06-08 19:16:56 +00:00
|
|
|
}
|
2015-05-04 01:35:39 +00:00
|
|
|
Init *rhs = RHS->resolveReferences(R, RV);
|
|
|
|
return (TernOpInit::get(getOpcode(), lhs, MHS,
|
|
|
|
rhs, getType()))->Fold(&R, nullptr);
|
2009-06-08 19:16:56 +00:00
|
|
|
}
|
|
|
|
}
|
2009-11-22 04:24:42 +00:00
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *mhs = MHS->resolveReferences(R, RV);
|
|
|
|
Init *rhs = RHS->resolveReferences(R, RV);
|
2009-06-08 19:16:56 +00:00
|
|
|
|
2009-05-14 21:54:42 +00:00
|
|
|
if (LHS != lhs || MHS != mhs || RHS != rhs)
|
2011-07-29 19:07:07 +00:00
|
|
|
return (TernOpInit::get(getOpcode(), lhs, mhs, rhs,
|
2014-04-09 04:50:04 +00:00
|
|
|
getType()))->Fold(&R, nullptr);
|
|
|
|
return Fold(&R, nullptr);
|
2009-05-14 21:54:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string TernOpInit::getAsString() const {
|
|
|
|
std::string Result;
|
2016-01-04 06:28:49 +00:00
|
|
|
switch (getOpcode()) {
|
2009-05-14 21:54:42 +00:00
|
|
|
case SUBST: Result = "!subst"; break;
|
2009-11-22 04:24:42 +00:00
|
|
|
case FOREACH: Result = "!foreach"; break;
|
|
|
|
case IF: Result = "!if"; break;
|
2015-04-30 05:54:22 +00:00
|
|
|
}
|
|
|
|
return Result + "(" + LHS->getAsString() + ", " + MHS->getAsString() + ", " +
|
|
|
|
RHS->getAsString() + ")";
|
2009-05-14 21:54:42 +00:00
|
|
|
}
|
2009-04-23 21:25:15 +00:00
|
|
|
|
2016-12-05 06:00:36 +00:00
|
|
|
RecTy *TypedInit::getFieldType(StringInit *FieldName) const {
|
2012-10-05 03:32:00 +00:00
|
|
|
if (RecordRecTy *RecordType = dyn_cast<RecordRecTy>(getType()))
|
|
|
|
if (RecordVal *Field = RecordType->getRecord()->getValue(FieldName))
|
2010-09-03 21:00:49 +00:00
|
|
|
return Field->getType();
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2010-09-03 21:00:49 +00:00
|
|
|
}
|
|
|
|
|
2015-05-30 07:34:51 +00:00
|
|
|
Init *
|
|
|
|
TypedInit::convertInitializerTo(RecTy *Ty) const {
|
|
|
|
if (isa<IntRecTy>(Ty)) {
|
|
|
|
if (getType()->typeIsConvertibleTo(Ty))
|
|
|
|
return const_cast<TypedInit *>(this);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isa<StringRecTy>(Ty)) {
|
|
|
|
if (isa<StringRecTy>(getType()))
|
|
|
|
return const_cast<TypedInit *>(this);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2016-07-05 21:22:55 +00:00
|
|
|
if (isa<CodeRecTy>(Ty)) {
|
|
|
|
if (isa<CodeRecTy>(getType()))
|
|
|
|
return const_cast<TypedInit *>(this);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2015-05-30 07:34:51 +00:00
|
|
|
if (isa<BitRecTy>(Ty)) {
|
|
|
|
// Accept variable if it is already of bit type!
|
|
|
|
if (isa<BitRecTy>(getType()))
|
|
|
|
return const_cast<TypedInit *>(this);
|
|
|
|
if (auto *BitsTy = dyn_cast<BitsRecTy>(getType())) {
|
|
|
|
// Accept only bits<1> expression.
|
|
|
|
if (BitsTy->getNumBits() == 1)
|
|
|
|
return const_cast<TypedInit *>(this);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
// Ternary !if can be converted to bit, but only if both sides are
|
|
|
|
// convertible to a bit.
|
|
|
|
if (const auto *TOI = dyn_cast<TernOpInit>(this)) {
|
|
|
|
if (TOI->getOpcode() == TernOpInit::TernaryOp::IF &&
|
|
|
|
TOI->getMHS()->convertInitializerTo(BitRecTy::get()) &&
|
|
|
|
TOI->getRHS()->convertInitializerTo(BitRecTy::get()))
|
|
|
|
return const_cast<TypedInit *>(this);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (auto *BRT = dyn_cast<BitsRecTy>(Ty)) {
|
|
|
|
if (BRT->getNumBits() == 1 && isa<BitRecTy>(getType()))
|
|
|
|
return BitsInit::get(const_cast<TypedInit *>(this));
|
|
|
|
|
|
|
|
if (getType()->typeIsConvertibleTo(BRT)) {
|
|
|
|
SmallVector<Init *, 16> NewBits(BRT->getNumBits());
|
|
|
|
|
|
|
|
for (unsigned i = 0; i != BRT->getNumBits(); ++i)
|
|
|
|
NewBits[i] = VarBitInit::get(const_cast<TypedInit *>(this), i);
|
|
|
|
return BitsInit::get(NewBits);
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (auto *DLRT = dyn_cast<ListRecTy>(Ty)) {
|
|
|
|
if (auto *SLRT = dyn_cast<ListRecTy>(getType()))
|
|
|
|
if (SLRT->getElementType()->typeIsConvertibleTo(DLRT->getElementType()))
|
|
|
|
return const_cast<TypedInit *>(this);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (auto *DRT = dyn_cast<DagRecTy>(Ty)) {
|
|
|
|
if (getType()->typeIsConvertibleTo(DRT))
|
|
|
|
return const_cast<TypedInit *>(this);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (auto *SRRT = dyn_cast<RecordRecTy>(Ty)) {
|
|
|
|
// Ensure that this is compatible with Rec.
|
|
|
|
if (RecordRecTy *DRRT = dyn_cast<RecordRecTy>(getType()))
|
|
|
|
if (DRRT->getRecord()->isSubClassOf(SRRT->getRecord()) ||
|
|
|
|
DRRT->getRecord() == SRRT->getRecord())
|
|
|
|
return const_cast<TypedInit *>(this);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *
|
2011-07-29 19:07:05 +00:00
|
|
|
TypedInit::convertInitializerBitRange(const std::vector<unsigned> &Bits) const {
|
2012-10-05 03:31:58 +00:00
|
|
|
BitsRecTy *T = dyn_cast<BitsRecTy>(getType());
|
2014-04-09 04:50:04 +00:00
|
|
|
if (!T) return nullptr; // Cannot subscript a non-bits variable.
|
2002-12-02 01:23:04 +00:00
|
|
|
unsigned NumBits = T->getNumBits();
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
SmallVector<Init *, 16> NewBits(Bits.size());
|
2002-12-02 01:23:04 +00:00
|
|
|
for (unsigned i = 0, e = Bits.size(); i != e; ++i) {
|
2011-07-29 19:07:00 +00:00
|
|
|
if (Bits[i] >= NumBits)
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2011-07-29 19:07:00 +00:00
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
NewBits[i] = VarBitInit::get(const_cast<TypedInit *>(this), Bits[i]);
|
2002-12-02 01:23:04 +00:00
|
|
|
}
|
2011-07-29 19:07:07 +00:00
|
|
|
return BitsInit::get(NewBits);
|
2002-12-02 01:23:04 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *
|
2011-07-29 19:07:05 +00:00
|
|
|
TypedInit::convertInitListSlice(const std::vector<unsigned> &Elements) const {
|
2012-10-05 03:31:58 +00:00
|
|
|
ListRecTy *T = dyn_cast<ListRecTy>(getType());
|
2014-04-09 04:50:04 +00:00
|
|
|
if (!T) return nullptr; // Cannot subscript a non-list variable.
|
2004-07-27 01:01:21 +00:00
|
|
|
|
|
|
|
if (Elements.size() == 1)
|
2011-07-29 22:43:06 +00:00
|
|
|
return VarListElementInit::get(const_cast<TypedInit *>(this), Elements[0]);
|
2004-07-27 01:01:21 +00:00
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
std::vector<Init*> ListInits;
|
2004-07-27 01:01:21 +00:00
|
|
|
ListInits.reserve(Elements.size());
|
|
|
|
for (unsigned i = 0, e = Elements.size(); i != e; ++i)
|
2011-07-29 22:43:06 +00:00
|
|
|
ListInits.push_back(VarListElementInit::get(const_cast<TypedInit *>(this),
|
|
|
|
Elements[i]));
|
2011-07-29 19:07:07 +00:00
|
|
|
return ListInit::get(ListInits, T);
|
2004-07-27 01:01:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-12-04 05:48:03 +00:00
|
|
|
VarInit *VarInit::get(StringRef VN, RecTy *T) {
|
2011-10-19 13:02:33 +00:00
|
|
|
Init *Value = StringInit::get(VN);
|
|
|
|
return VarInit::get(Value, T);
|
|
|
|
}
|
|
|
|
|
|
|
|
VarInit *VarInit::get(Init *VN, RecTy *T) {
|
|
|
|
typedef std::pair<RecTy *, Init *> Key;
|
2016-12-04 05:48:20 +00:00
|
|
|
static DenseMap<Key, VarInit*> ThePool;
|
2011-07-29 19:07:21 +00:00
|
|
|
|
|
|
|
Key TheKey(std::make_pair(T, VN));
|
|
|
|
|
2016-12-04 05:48:20 +00:00
|
|
|
VarInit *&I = ThePool[TheKey];
|
|
|
|
if (!I)
|
|
|
|
I = new(Allocator) VarInit(VN, T);
|
|
|
|
return I;
|
2011-07-29 19:07:07 +00:00
|
|
|
}
|
|
|
|
|
2016-12-04 05:48:16 +00:00
|
|
|
StringRef VarInit::getName() const {
|
2015-04-29 07:13:05 +00:00
|
|
|
StringInit *NameString = cast<StringInit>(getNameInit());
|
2011-10-19 13:02:33 +00:00
|
|
|
return NameString->getValue();
|
|
|
|
}
|
|
|
|
|
Re-work bit/bits value resolving in tblgen
- This patch is inspired by the failure of the following code snippet
which is used to convert enumerable values into encoding bits to
improve the readability of td files.
class S<int s> {
bits<2> V = !if(!eq(s, 8), {0, 0},
!if(!eq(s, 16), {0, 1},
!if(!eq(s, 32), {1, 0},
!if(!eq(s, 64), {1, 1}, {?, ?}))));
}
Later, PR8330 is found to report not exactly the same bug relevant
issue to bit/bits values.
- Instead of resolving bit/bits values separately through
resolveBitReference(), this patch adds getBit() for all Inits and
resolves bit value by resolving plus getting the specified bit. This
unifies the resolving of bit with other values and removes redundant
logic for resolving bit only. In addition,
BitsInit::resolveReferences() is optimized to take advantage of this
origanization by resolving VarBitInit's variable reference first and
then getting bits from it.
- The type interference in '!if' operator is revised to support possible
combinations of int and bits/bit in MHS and RHS.
- As there may be illegal assignments from integer value to bit, says
assign 2 to a bit, but we only check this during instantiation in some
cases, e.g.
bit V = !if(!eq(x, 17), 0, 2);
Verbose diagnostic message is generated when invalid value is
resolveed to help locating the error.
- PR8330 is fixed as well.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163360 91177308-0d34-0410-b5e6-96231b3b80d8
2012-09-06 23:32:48 +00:00
|
|
|
Init *VarInit::getBit(unsigned Bit) const {
|
|
|
|
if (getType() == BitRecTy::get())
|
|
|
|
return const_cast<VarInit*>(this);
|
|
|
|
return VarBitInit::get(const_cast<VarInit*>(this), Bit);
|
2002-12-02 01:23:04 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *VarInit::resolveListElementReference(Record &R,
|
|
|
|
const RecordVal *IRV,
|
|
|
|
unsigned Elt) const {
|
2014-04-09 04:50:04 +00:00
|
|
|
if (R.isTemplateArg(getNameInit())) return nullptr;
|
|
|
|
if (IRV && IRV->getNameInit() != getNameInit()) return nullptr;
|
2004-07-27 01:01:21 +00:00
|
|
|
|
2012-01-13 03:16:35 +00:00
|
|
|
RecordVal *RV = R.getValue(getNameInit());
|
2009-11-21 22:44:20 +00:00
|
|
|
assert(RV && "Reference to a non-existent variable?");
|
2012-10-10 20:24:43 +00:00
|
|
|
ListInit *LI = dyn_cast<ListInit>(RV->getValue());
|
2015-05-04 01:35:39 +00:00
|
|
|
if (!LI)
|
|
|
|
return VarListElementInit::get(cast<TypedInit>(RV->getValue()), Elt);
|
2009-11-22 04:24:42 +00:00
|
|
|
|
2015-06-02 04:15:57 +00:00
|
|
|
if (Elt >= LI->size())
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr; // Out of range reference.
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *E = LI->getElement(Elt);
|
2009-11-22 03:58:57 +00:00
|
|
|
// If the element is set to some value, or if we are resolving a reference
|
|
|
|
// to a specific variable and that variable is explicitly unset, then
|
|
|
|
// replace the VarListElementInit with it.
|
2012-10-10 20:24:47 +00:00
|
|
|
if (IRV || !isa<UnsetInit>(E))
|
2009-11-22 03:58:57 +00:00
|
|
|
return E;
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2004-07-27 01:01:21 +00:00
|
|
|
}
|
|
|
|
|
2016-12-05 06:00:36 +00:00
|
|
|
RecTy *VarInit::getFieldType(StringInit *FieldName) const {
|
2012-10-05 03:31:58 +00:00
|
|
|
if (RecordRecTy *RTy = dyn_cast<RecordRecTy>(getType()))
|
2002-12-03 06:00:33 +00:00
|
|
|
if (const RecordVal *RV = RTy->getRecord()->getValue(FieldName))
|
|
|
|
return RV->getType();
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2002-12-03 06:00:33 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *VarInit::getFieldInit(Record &R, const RecordVal *RV,
|
2016-12-05 06:00:36 +00:00
|
|
|
StringInit *FieldName) const {
|
2012-10-05 03:32:00 +00:00
|
|
|
if (isa<RecordRecTy>(getType()))
|
2010-03-25 06:23:34 +00:00
|
|
|
if (const RecordVal *Val = R.getValue(VarName)) {
|
2012-10-10 20:24:47 +00:00
|
|
|
if (RV != Val && (RV || isa<UnsetInit>(Val->getValue())))
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *TheInit = Val->getValue();
|
2004-02-28 16:31:53 +00:00
|
|
|
assert(TheInit != this && "Infinite loop detected!");
|
2011-07-29 22:43:06 +00:00
|
|
|
if (Init *I = TheInit->getFieldInit(R, RV, FieldName))
|
2002-12-03 06:00:33 +00:00
|
|
|
return I;
|
2015-05-04 01:35:39 +00:00
|
|
|
return nullptr;
|
2004-02-28 16:31:53 +00:00
|
|
|
}
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2002-12-03 06:00:33 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *VarInit::resolveReferences(Record &R, const RecordVal *RV) const {
|
2003-07-30 04:05:07 +00:00
|
|
|
if (RecordVal *Val = R.getValue(VarName))
|
2014-04-09 04:50:04 +00:00
|
|
|
if (RV == Val || (!RV && !isa<UnsetInit>(Val->getValue())))
|
2003-07-30 04:05:07 +00:00
|
|
|
return Val->getValue();
|
2011-07-29 22:43:06 +00:00
|
|
|
return const_cast<VarInit *>(this);
|
2003-07-30 04:05:07 +00:00
|
|
|
}
|
2005-04-22 00:00:37 +00:00
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
VarBitInit *VarBitInit::get(TypedInit *T, unsigned B) {
|
|
|
|
typedef std::pair<TypedInit *, unsigned> Key;
|
2016-12-04 05:48:20 +00:00
|
|
|
static DenseMap<Key, VarBitInit*> ThePool;
|
2011-07-29 19:07:22 +00:00
|
|
|
|
|
|
|
Key TheKey(std::make_pair(T, B));
|
|
|
|
|
2016-12-04 05:48:20 +00:00
|
|
|
VarBitInit *&I = ThePool[TheKey];
|
|
|
|
if (!I)
|
|
|
|
I = new(Allocator) VarBitInit(T, B);
|
|
|
|
return I;
|
2011-07-29 19:07:07 +00:00
|
|
|
}
|
|
|
|
|
2015-05-30 07:34:51 +00:00
|
|
|
Init *VarBitInit::convertInitializerTo(RecTy *Ty) const {
|
|
|
|
if (isa<BitRecTy>(Ty))
|
|
|
|
return const_cast<VarBitInit *>(this);
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2007-11-22 21:05:25 +00:00
|
|
|
std::string VarBitInit::getAsString() const {
|
2015-05-04 01:35:39 +00:00
|
|
|
return TI->getAsString() + "{" + utostr(Bit) + "}";
|
2007-11-22 21:05:25 +00:00
|
|
|
}
|
2002-12-02 17:44:35 +00:00
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *VarBitInit::resolveReferences(Record &R, const RecordVal *RV) const {
|
Re-work bit/bits value resolving in tblgen
- This patch is inspired by the failure of the following code snippet
which is used to convert enumerable values into encoding bits to
improve the readability of td files.
class S<int s> {
bits<2> V = !if(!eq(s, 8), {0, 0},
!if(!eq(s, 16), {0, 1},
!if(!eq(s, 32), {1, 0},
!if(!eq(s, 64), {1, 1}, {?, ?}))));
}
Later, PR8330 is found to report not exactly the same bug relevant
issue to bit/bits values.
- Instead of resolving bit/bits values separately through
resolveBitReference(), this patch adds getBit() for all Inits and
resolves bit value by resolving plus getting the specified bit. This
unifies the resolving of bit with other values and removes redundant
logic for resolving bit only. In addition,
BitsInit::resolveReferences() is optimized to take advantage of this
origanization by resolving VarBitInit's variable reference first and
then getting bits from it.
- The type interference in '!if' operator is revised to support possible
combinations of int and bits/bit in MHS and RHS.
- As there may be illegal assignments from integer value to bit, says
assign 2 to a bit, but we only check this during instantiation in some
cases, e.g.
bit V = !if(!eq(x, 17), 0, 2);
Verbose diagnostic message is generated when invalid value is
resolveed to help locating the error.
- PR8330 is fixed as well.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163360 91177308-0d34-0410-b5e6-96231b3b80d8
2012-09-06 23:32:48 +00:00
|
|
|
Init *I = TI->resolveReferences(R, RV);
|
|
|
|
if (TI != I)
|
|
|
|
return I->getBit(getBitNum());
|
|
|
|
|
|
|
|
return const_cast<VarBitInit*>(this);
|
2002-12-02 17:44:35 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
VarListElementInit *VarListElementInit::get(TypedInit *T,
|
|
|
|
unsigned E) {
|
|
|
|
typedef std::pair<TypedInit *, unsigned> Key;
|
2016-12-04 05:48:20 +00:00
|
|
|
static DenseMap<Key, VarListElementInit*> ThePool;
|
2011-07-29 19:07:23 +00:00
|
|
|
|
|
|
|
Key TheKey(std::make_pair(T, E));
|
|
|
|
|
2016-12-04 05:48:20 +00:00
|
|
|
VarListElementInit *&I = ThePool[TheKey];
|
|
|
|
if (!I) I = new(Allocator) VarListElementInit(T, E);
|
|
|
|
return I;
|
2011-07-29 19:07:07 +00:00
|
|
|
}
|
|
|
|
|
2007-11-22 21:05:25 +00:00
|
|
|
std::string VarListElementInit::getAsString() const {
|
|
|
|
return TI->getAsString() + "[" + utostr(Element) + "]";
|
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *
|
|
|
|
VarListElementInit::resolveReferences(Record &R, const RecordVal *RV) const {
|
|
|
|
if (Init *I = getVariable()->resolveListElementReference(R, RV,
|
2011-07-11 23:06:52 +00:00
|
|
|
getElementNum()))
|
2004-07-27 01:01:21 +00:00
|
|
|
return I;
|
2011-07-29 22:43:06 +00:00
|
|
|
return const_cast<VarListElementInit *>(this);
|
2004-07-27 01:01:21 +00:00
|
|
|
}
|
|
|
|
|
Re-work bit/bits value resolving in tblgen
- This patch is inspired by the failure of the following code snippet
which is used to convert enumerable values into encoding bits to
improve the readability of td files.
class S<int s> {
bits<2> V = !if(!eq(s, 8), {0, 0},
!if(!eq(s, 16), {0, 1},
!if(!eq(s, 32), {1, 0},
!if(!eq(s, 64), {1, 1}, {?, ?}))));
}
Later, PR8330 is found to report not exactly the same bug relevant
issue to bit/bits values.
- Instead of resolving bit/bits values separately through
resolveBitReference(), this patch adds getBit() for all Inits and
resolves bit value by resolving plus getting the specified bit. This
unifies the resolving of bit with other values and removes redundant
logic for resolving bit only. In addition,
BitsInit::resolveReferences() is optimized to take advantage of this
origanization by resolving VarBitInit's variable reference first and
then getting bits from it.
- The type interference in '!if' operator is revised to support possible
combinations of int and bits/bit in MHS and RHS.
- As there may be illegal assignments from integer value to bit, says
assign 2 to a bit, but we only check this during instantiation in some
cases, e.g.
bit V = !if(!eq(x, 17), 0, 2);
Verbose diagnostic message is generated when invalid value is
resolveed to help locating the error.
- PR8330 is fixed as well.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163360 91177308-0d34-0410-b5e6-96231b3b80d8
2012-09-06 23:32:48 +00:00
|
|
|
Init *VarListElementInit::getBit(unsigned Bit) const {
|
|
|
|
if (getType() == BitRecTy::get())
|
|
|
|
return const_cast<VarListElementInit*>(this);
|
|
|
|
return VarBitInit::get(const_cast<VarListElementInit*>(this), Bit);
|
2004-07-27 01:01:21 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *VarListElementInit:: resolveListElementReference(Record &R,
|
|
|
|
const RecordVal *RV,
|
|
|
|
unsigned Elt) const {
|
2015-05-04 01:35:39 +00:00
|
|
|
if (Init *Result = TI->resolveListElementReference(R, RV, Element)) {
|
2012-10-10 20:24:47 +00:00
|
|
|
if (TypedInit *TInit = dyn_cast<TypedInit>(Result)) {
|
2015-06-06 01:33:58 +00:00
|
|
|
if (Init *Result2 = TInit->resolveListElementReference(R, RV, Elt))
|
|
|
|
return Result2;
|
2015-04-24 05:38:48 +00:00
|
|
|
return VarListElementInit::get(TInit, Elt);
|
2011-09-30 20:59:49 +00:00
|
|
|
}
|
|
|
|
return Result;
|
|
|
|
}
|
2015-05-04 01:35:39 +00:00
|
|
|
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2004-07-27 01:01:21 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
DefInit *DefInit::get(Record *R) {
|
2011-07-18 17:02:57 +00:00
|
|
|
return R->getDefInit();
|
|
|
|
}
|
|
|
|
|
2015-05-30 07:34:51 +00:00
|
|
|
Init *DefInit::convertInitializerTo(RecTy *Ty) const {
|
|
|
|
if (auto *RRT = dyn_cast<RecordRecTy>(Ty))
|
|
|
|
if (getDef()->isSubClassOf(RRT->getRecord()))
|
|
|
|
return const_cast<DefInit *>(this);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2016-12-05 06:00:36 +00:00
|
|
|
RecTy *DefInit::getFieldType(StringInit *FieldName) const {
|
2002-12-03 06:00:33 +00:00
|
|
|
if (const RecordVal *RV = Def->getValue(FieldName))
|
|
|
|
return RV->getType();
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2002-12-03 06:00:33 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *DefInit::getFieldInit(Record &R, const RecordVal *RV,
|
2016-12-05 06:00:36 +00:00
|
|
|
StringInit *FieldName) const {
|
2002-12-03 06:00:33 +00:00
|
|
|
return Def->getValue(FieldName)->getValue();
|
|
|
|
}
|
|
|
|
|
2007-11-22 21:05:25 +00:00
|
|
|
std::string DefInit::getAsString() const {
|
|
|
|
return Def->getName();
|
2002-12-02 01:23:04 +00:00
|
|
|
}
|
|
|
|
|
2016-12-05 06:00:36 +00:00
|
|
|
FieldInit *FieldInit::get(Init *R, StringInit *FN) {
|
|
|
|
typedef std::pair<Init *, StringInit *> Key;
|
2016-12-04 05:48:20 +00:00
|
|
|
static DenseMap<Key, FieldInit*> ThePool;
|
2011-07-29 19:07:24 +00:00
|
|
|
|
|
|
|
Key TheKey(std::make_pair(R, FN));
|
|
|
|
|
2016-12-04 05:48:20 +00:00
|
|
|
FieldInit *&I = ThePool[TheKey];
|
|
|
|
if (!I) I = new(Allocator) FieldInit(R, FN);
|
|
|
|
return I;
|
2011-07-29 19:07:07 +00:00
|
|
|
}
|
|
|
|
|
Re-work bit/bits value resolving in tblgen
- This patch is inspired by the failure of the following code snippet
which is used to convert enumerable values into encoding bits to
improve the readability of td files.
class S<int s> {
bits<2> V = !if(!eq(s, 8), {0, 0},
!if(!eq(s, 16), {0, 1},
!if(!eq(s, 32), {1, 0},
!if(!eq(s, 64), {1, 1}, {?, ?}))));
}
Later, PR8330 is found to report not exactly the same bug relevant
issue to bit/bits values.
- Instead of resolving bit/bits values separately through
resolveBitReference(), this patch adds getBit() for all Inits and
resolves bit value by resolving plus getting the specified bit. This
unifies the resolving of bit with other values and removes redundant
logic for resolving bit only. In addition,
BitsInit::resolveReferences() is optimized to take advantage of this
origanization by resolving VarBitInit's variable reference first and
then getting bits from it.
- The type interference in '!if' operator is revised to support possible
combinations of int and bits/bit in MHS and RHS.
- As there may be illegal assignments from integer value to bit, says
assign 2 to a bit, but we only check this during instantiation in some
cases, e.g.
bit V = !if(!eq(x, 17), 0, 2);
Verbose diagnostic message is generated when invalid value is
resolveed to help locating the error.
- PR8330 is fixed as well.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163360 91177308-0d34-0410-b5e6-96231b3b80d8
2012-09-06 23:32:48 +00:00
|
|
|
Init *FieldInit::getBit(unsigned Bit) const {
|
|
|
|
if (getType() == BitRecTy::get())
|
|
|
|
return const_cast<FieldInit*>(this);
|
|
|
|
return VarBitInit::get(const_cast<FieldInit*>(this), Bit);
|
2004-07-27 01:01:21 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *FieldInit::resolveListElementReference(Record &R, const RecordVal *RV,
|
|
|
|
unsigned Elt) const {
|
|
|
|
if (Init *ListVal = Rec->getFieldInit(R, RV, FieldName))
|
2012-10-10 20:24:43 +00:00
|
|
|
if (ListInit *LI = dyn_cast<ListInit>(ListVal)) {
|
2015-06-02 04:15:57 +00:00
|
|
|
if (Elt >= LI->size()) return nullptr;
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *E = LI->getElement(Elt);
|
2004-07-27 01:01:21 +00:00
|
|
|
|
2009-11-22 03:58:57 +00:00
|
|
|
// If the element is set to some value, or if we are resolving a
|
|
|
|
// reference to a specific variable and that variable is explicitly
|
|
|
|
// unset, then replace the VarListElementInit with it.
|
2012-10-10 20:24:47 +00:00
|
|
|
if (RV || !isa<UnsetInit>(E))
|
2009-11-22 03:58:57 +00:00
|
|
|
return E;
|
2004-07-27 01:01:21 +00:00
|
|
|
}
|
2014-04-09 04:50:04 +00:00
|
|
|
return nullptr;
|
2003-08-01 05:58:58 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *FieldInit::resolveReferences(Record &R, const RecordVal *RV) const {
|
|
|
|
Init *NewRec = RV ? Rec->resolveReferences(R, RV) : Rec;
|
2005-04-19 03:36:21 +00:00
|
|
|
|
2015-05-04 01:35:39 +00:00
|
|
|
if (Init *BitsVal = NewRec->getFieldInit(R, RV, FieldName)) {
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *BVR = BitsVal->resolveReferences(R, RV);
|
|
|
|
return BVR->isComplete() ? BVR : const_cast<FieldInit *>(this);
|
2002-12-03 06:00:33 +00:00
|
|
|
}
|
2005-04-19 03:36:21 +00:00
|
|
|
|
2015-05-04 01:35:39 +00:00
|
|
|
if (NewRec != Rec)
|
2011-07-29 19:07:07 +00:00
|
|
|
return FieldInit::get(NewRec, FieldName);
|
2011-07-29 22:43:06 +00:00
|
|
|
return const_cast<FieldInit *>(this);
|
2002-12-02 17:44:35 +00:00
|
|
|
}
|
|
|
|
|
2016-12-05 06:00:41 +00:00
|
|
|
static void ProfileDagInit(FoldingSetNodeID &ID, Init *V, StringInit *VN,
|
2013-02-15 12:30:38 +00:00
|
|
|
ArrayRef<Init *> ArgRange,
|
2016-12-05 06:00:46 +00:00
|
|
|
ArrayRef<StringInit *> NameRange) {
|
2011-07-29 19:07:26 +00:00
|
|
|
ID.AddPointer(V);
|
2016-12-05 06:00:41 +00:00
|
|
|
ID.AddPointer(VN);
|
2011-07-29 19:07:26 +00:00
|
|
|
|
2016-12-05 06:00:46 +00:00
|
|
|
ArrayRef<Init *>::iterator Arg = ArgRange.begin();
|
|
|
|
ArrayRef<StringInit *>::iterator Name = NameRange.begin();
|
2011-07-29 19:07:26 +00:00
|
|
|
while (Arg != ArgRange.end()) {
|
|
|
|
assert(Name != NameRange.end() && "Arg name underflow!");
|
|
|
|
ID.AddPointer(*Arg++);
|
2016-12-05 06:00:46 +00:00
|
|
|
ID.AddPointer(*Name++);
|
2011-07-29 19:07:26 +00:00
|
|
|
}
|
|
|
|
assert(Name == NameRange.end() && "Arg name overflow!");
|
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
DagInit *
|
2016-12-05 06:00:41 +00:00
|
|
|
DagInit::get(Init *V, StringInit *VN, ArrayRef<Init *> ArgRange,
|
2016-12-05 06:00:46 +00:00
|
|
|
ArrayRef<StringInit *> NameRange) {
|
2015-04-24 05:38:48 +00:00
|
|
|
static FoldingSet<DagInit> ThePool;
|
2016-12-04 05:48:20 +00:00
|
|
|
static std::vector<DagInit*> TheActualPool;
|
2011-07-29 19:07:26 +00:00
|
|
|
|
|
|
|
FoldingSetNodeID ID;
|
|
|
|
ProfileDagInit(ID, V, VN, ArgRange, NameRange);
|
|
|
|
|
2014-04-09 04:50:04 +00:00
|
|
|
void *IP = nullptr;
|
2011-07-29 22:43:06 +00:00
|
|
|
if (DagInit *I = ThePool.FindNodeOrInsertPos(ID, IP))
|
2011-07-29 19:07:26 +00:00
|
|
|
return I;
|
|
|
|
|
2016-12-04 05:48:20 +00:00
|
|
|
DagInit *I = new(Allocator) DagInit(V, VN, ArgRange, NameRange);
|
2011-07-29 19:07:26 +00:00
|
|
|
ThePool.InsertNode(I, IP);
|
2016-12-04 05:48:20 +00:00
|
|
|
TheActualPool.push_back(I);
|
2011-07-29 19:07:26 +00:00
|
|
|
return I;
|
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
DagInit *
|
2016-12-05 06:00:41 +00:00
|
|
|
DagInit::get(Init *V, StringInit *VN,
|
2016-12-05 06:00:46 +00:00
|
|
|
const std::vector<std::pair<Init*, StringInit*> > &args) {
|
2011-07-29 22:43:06 +00:00
|
|
|
std::vector<Init *> Args;
|
2016-12-05 06:00:46 +00:00
|
|
|
std::vector<StringInit *> Names;
|
2011-07-29 19:07:07 +00:00
|
|
|
|
2015-04-29 07:13:14 +00:00
|
|
|
for (const auto &Arg : args) {
|
|
|
|
Args.push_back(Arg.first);
|
|
|
|
Names.push_back(Arg.second);
|
2011-07-29 19:07:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return DagInit::get(V, VN, Args, Names);
|
|
|
|
}
|
|
|
|
|
2011-07-29 19:07:26 +00:00
|
|
|
void DagInit::Profile(FoldingSetNodeID &ID) const {
|
|
|
|
ProfileDagInit(ID, Val, ValName, Args, ArgNames);
|
2011-07-29 19:07:07 +00:00
|
|
|
}
|
|
|
|
|
2015-05-30 07:34:51 +00:00
|
|
|
Init *DagInit::convertInitializerTo(RecTy *Ty) const {
|
|
|
|
if (isa<DagRecTy>(Ty))
|
|
|
|
return const_cast<DagInit *>(this);
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *DagInit::resolveReferences(Record &R, const RecordVal *RV) const {
|
|
|
|
std::vector<Init*> NewArgs;
|
2006-01-31 06:02:35 +00:00
|
|
|
for (unsigned i = 0, e = Args.size(); i != e; ++i)
|
|
|
|
NewArgs.push_back(Args[i]->resolveReferences(R, RV));
|
2009-11-22 04:24:42 +00:00
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *Op = Val->resolveReferences(R, RV);
|
2009-11-22 04:24:42 +00:00
|
|
|
|
2006-03-30 22:50:40 +00:00
|
|
|
if (Args != NewArgs || Op != Val)
|
2011-07-29 19:07:07 +00:00
|
|
|
return DagInit::get(Op, ValName, NewArgs, ArgNames);
|
2009-11-22 04:24:42 +00:00
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
return const_cast<DagInit *>(this);
|
2006-01-31 06:02:35 +00:00
|
|
|
}
|
|
|
|
|
2007-11-22 21:05:25 +00:00
|
|
|
std::string DagInit::getAsString() const {
|
|
|
|
std::string Result = "(" + Val->getAsString();
|
2016-12-05 06:00:41 +00:00
|
|
|
if (ValName)
|
|
|
|
Result += ":" + ValName->getAsUnquotedString();
|
2015-01-15 11:41:30 +00:00
|
|
|
if (!Args.empty()) {
|
2007-11-22 21:05:25 +00:00
|
|
|
Result += " " + Args[0]->getAsString();
|
2016-12-05 06:00:46 +00:00
|
|
|
if (ArgNames[0]) Result += ":$" + ArgNames[0]->getAsUnquotedString();
|
2003-08-10 22:14:13 +00:00
|
|
|
for (unsigned i = 1, e = Args.size(); i != e; ++i) {
|
2007-11-22 21:05:25 +00:00
|
|
|
Result += ", " + Args[i]->getAsString();
|
2016-12-05 06:00:46 +00:00
|
|
|
if (ArgNames[i]) Result += ":$" + ArgNames[i]->getAsUnquotedString();
|
2003-08-10 22:14:13 +00:00
|
|
|
}
|
2003-08-04 20:44:17 +00:00
|
|
|
}
|
2007-11-22 21:05:25 +00:00
|
|
|
return Result + ")";
|
2003-08-04 20:44:17 +00:00
|
|
|
}
|
|
|
|
|
2002-12-02 01:23:04 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Other implementations
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2015-06-04 07:40:12 +00:00
|
|
|
RecordVal::RecordVal(Init *N, RecTy *T, bool P)
|
2016-12-05 05:21:13 +00:00
|
|
|
: Name(N), TyAndPrefix(T, P) {
|
|
|
|
Value = UnsetInit::get()->convertInitializerTo(T);
|
2002-12-02 01:23:04 +00:00
|
|
|
assert(Value && "Cannot create unset value for current type!");
|
|
|
|
}
|
|
|
|
|
2016-12-04 05:48:03 +00:00
|
|
|
RecordVal::RecordVal(StringRef N, RecTy *T, bool P)
|
2016-12-05 05:21:13 +00:00
|
|
|
: Name(StringInit::get(N)), TyAndPrefix(T, P) {
|
|
|
|
Value = UnsetInit::get()->convertInitializerTo(T);
|
2011-09-02 20:12:07 +00:00
|
|
|
assert(Value && "Cannot create unset value for current type!");
|
|
|
|
}
|
|
|
|
|
2016-12-04 05:48:16 +00:00
|
|
|
StringRef RecordVal::getName() const {
|
2015-06-04 07:40:12 +00:00
|
|
|
return cast<StringInit>(getNameInit())->getValue();
|
2011-09-02 20:12:07 +00:00
|
|
|
}
|
|
|
|
|
2016-01-29 20:50:44 +00:00
|
|
|
LLVM_DUMP_METHOD void RecordVal::dump() const { errs() << *this; }
|
2002-12-02 01:23:04 +00:00
|
|
|
|
2009-07-03 00:10:29 +00:00
|
|
|
void RecordVal::print(raw_ostream &OS, bool PrintSem) const {
|
2002-12-02 01:23:04 +00:00
|
|
|
if (getPrefix()) OS << "field ";
|
2011-10-19 13:02:52 +00:00
|
|
|
OS << *getType() << " " << getNameInitAsString();
|
2005-04-19 03:36:21 +00:00
|
|
|
|
|
|
|
if (getValue())
|
2002-12-02 01:23:04 +00:00
|
|
|
OS << " = " << *getValue();
|
2005-04-19 03:36:21 +00:00
|
|
|
|
2002-12-02 01:23:04 +00:00
|
|
|
if (PrintSem) OS << ";\n";
|
|
|
|
}
|
|
|
|
|
2009-08-23 09:47:37 +00:00
|
|
|
unsigned Record::LastID = 0;
|
|
|
|
|
2011-10-19 13:02:45 +00:00
|
|
|
void Record::init() {
|
|
|
|
checkName();
|
2011-10-19 13:04:13 +00:00
|
|
|
|
|
|
|
// Every record potentially has a def at the top. This value is
|
|
|
|
// replaced with the top-level def name at instantiation time.
|
2016-08-23 17:14:32 +00:00
|
|
|
RecordVal DN("NAME", StringRecTy::get(), false);
|
2011-10-19 13:04:13 +00:00
|
|
|
addValue(DN);
|
2011-10-19 13:02:45 +00:00
|
|
|
}
|
|
|
|
|
2011-08-10 18:27:46 +00:00
|
|
|
void Record::checkName() {
|
|
|
|
// Ensure the record name has string type.
|
2015-04-29 07:13:05 +00:00
|
|
|
const TypedInit *TypedName = cast<const TypedInit>(Name);
|
2015-06-06 01:34:00 +00:00
|
|
|
if (!isa<StringRecTy>(TypedName->getType()))
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record name is not a string!");
|
2011-08-10 18:27:46 +00:00
|
|
|
}
|
|
|
|
|
2011-07-18 17:02:57 +00:00
|
|
|
DefInit *Record::getDefInit() {
|
2015-06-24 06:19:19 +00:00
|
|
|
if (!TheInit)
|
2016-12-04 05:48:20 +00:00
|
|
|
TheInit = new(Allocator) DefInit(this, new(Allocator) RecordRecTy(this));
|
|
|
|
return TheInit;
|
2011-07-18 17:02:57 +00:00
|
|
|
}
|
|
|
|
|
2016-12-04 05:48:16 +00:00
|
|
|
StringRef Record::getName() const {
|
2015-04-29 07:13:05 +00:00
|
|
|
return cast<StringInit>(Name)->getValue();
|
2011-08-10 18:27:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Record::setName(Init *NewName) {
|
2011-10-19 13:03:25 +00:00
|
|
|
Name = NewName;
|
2011-08-10 18:27:46 +00:00
|
|
|
checkName();
|
|
|
|
// DO NOT resolve record values to the name at this point because
|
|
|
|
// there might be default values for arguments of this def. Those
|
|
|
|
// arguments might not have been resolved yet so we don't want to
|
|
|
|
// prematurely assume values for those arguments were not passed to
|
|
|
|
// this def.
|
|
|
|
//
|
|
|
|
// Nonetheless, it may be that some of this Record's values
|
|
|
|
// reference the record name. Indeed, the reason for having the
|
|
|
|
// record name be an Init is to provide this flexibility. The extra
|
|
|
|
// resolve steps after completely instantiating defs takes care of
|
|
|
|
// this. See TGParser::ParseDef and TGParser::ParseDefm.
|
|
|
|
}
|
|
|
|
|
2016-12-04 05:48:03 +00:00
|
|
|
void Record::setName(StringRef Name) {
|
2011-08-10 18:27:46 +00:00
|
|
|
setName(StringInit::get(Name));
|
2011-10-19 13:02:29 +00:00
|
|
|
}
|
|
|
|
|
2005-04-19 03:36:21 +00:00
|
|
|
void Record::resolveReferencesTo(const RecordVal *RV) {
|
|
|
|
for (unsigned i = 0, e = Values.size(); i != e; ++i) {
|
2012-03-07 16:39:35 +00:00
|
|
|
if (RV == &Values[i]) // Skip resolve the same field as the given one
|
|
|
|
continue;
|
2011-07-29 22:43:06 +00:00
|
|
|
if (Init *V = Values[i].getValue())
|
Re-work bit/bits value resolving in tblgen
- This patch is inspired by the failure of the following code snippet
which is used to convert enumerable values into encoding bits to
improve the readability of td files.
class S<int s> {
bits<2> V = !if(!eq(s, 8), {0, 0},
!if(!eq(s, 16), {0, 1},
!if(!eq(s, 32), {1, 0},
!if(!eq(s, 64), {1, 1}, {?, ?}))));
}
Later, PR8330 is found to report not exactly the same bug relevant
issue to bit/bits values.
- Instead of resolving bit/bits values separately through
resolveBitReference(), this patch adds getBit() for all Inits and
resolves bit value by resolving plus getting the specified bit. This
unifies the resolving of bit with other values and removes redundant
logic for resolving bit only. In addition,
BitsInit::resolveReferences() is optimized to take advantage of this
origanization by resolving VarBitInit's variable reference first and
then getting bits from it.
- The type interference in '!if' operator is revised to support possible
combinations of int and bits/bit in MHS and RHS.
- As there may be illegal assignments from integer value to bit, says
assign 2 to a bit, but we only check this during instantiation in some
cases, e.g.
bit V = !if(!eq(x, 17), 0, 2);
Verbose diagnostic message is generated when invalid value is
resolveed to help locating the error.
- PR8330 is fixed as well.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@163360 91177308-0d34-0410-b5e6-96231b3b80d8
2012-09-06 23:32:48 +00:00
|
|
|
if (Values[i].setValue(V->resolveReferences(*this, RV)))
|
2015-04-30 05:54:22 +00:00
|
|
|
PrintFatalError(getLoc(), "Invalid value is found when setting '" +
|
|
|
|
Values[i].getNameInitAsString() +
|
|
|
|
"' after resolving references" +
|
|
|
|
(RV ? " against '" + RV->getNameInitAsString() +
|
|
|
|
"' of (" + RV->getValue()->getAsUnquotedString() +
|
|
|
|
")"
|
|
|
|
: "") + "\n");
|
2005-04-19 03:36:21 +00:00
|
|
|
}
|
2011-10-19 13:03:30 +00:00
|
|
|
Init *OldName = getNameInit();
|
|
|
|
Init *NewName = Name->resolveReferences(*this, RV);
|
|
|
|
if (NewName != OldName) {
|
|
|
|
// Re-register with RecordKeeper.
|
|
|
|
setName(NewName);
|
|
|
|
}
|
2002-12-02 01:23:04 +00:00
|
|
|
}
|
|
|
|
|
2016-01-29 20:50:44 +00:00
|
|
|
LLVM_DUMP_METHOD void Record::dump() const { errs() << *this; }
|
2002-12-02 01:23:04 +00:00
|
|
|
|
2009-07-03 00:10:29 +00:00
|
|
|
raw_ostream &llvm::operator<<(raw_ostream &OS, const Record &R) {
|
2011-10-19 13:02:57 +00:00
|
|
|
OS << R.getNameInitAsString();
|
2002-12-02 01:23:04 +00:00
|
|
|
|
2015-10-24 12:46:45 +00:00
|
|
|
ArrayRef<Init *> TArgs = R.getTemplateArgs();
|
2002-12-02 01:23:04 +00:00
|
|
|
if (!TArgs.empty()) {
|
|
|
|
OS << "<";
|
2015-06-04 07:40:14 +00:00
|
|
|
bool NeedComma = false;
|
|
|
|
for (const Init *TA : TArgs) {
|
|
|
|
if (NeedComma) OS << ", ";
|
|
|
|
NeedComma = true;
|
|
|
|
const RecordVal *RV = R.getValue(TA);
|
2002-12-02 01:23:04 +00:00
|
|
|
assert(RV && "Template argument record not found??");
|
|
|
|
RV->print(OS, false);
|
|
|
|
}
|
|
|
|
OS << ">";
|
|
|
|
}
|
|
|
|
|
|
|
|
OS << " {";
|
2016-01-18 19:52:37 +00:00
|
|
|
ArrayRef<std::pair<Record *, SMRange>> SC = R.getSuperClasses();
|
2002-12-02 01:23:04 +00:00
|
|
|
if (!SC.empty()) {
|
|
|
|
OS << "\t//";
|
2016-01-18 19:52:37 +00:00
|
|
|
for (const auto &SuperPair : SC)
|
|
|
|
OS << " " << SuperPair.first->getNameInitAsString();
|
2002-12-02 01:23:04 +00:00
|
|
|
}
|
|
|
|
OS << "\n";
|
|
|
|
|
2015-06-04 07:40:14 +00:00
|
|
|
for (const RecordVal &Val : R.getValues())
|
|
|
|
if (Val.getPrefix() && !R.isTemplateArg(Val.getName()))
|
|
|
|
OS << Val;
|
|
|
|
for (const RecordVal &Val : R.getValues())
|
|
|
|
if (!Val.getPrefix() && !R.isTemplateArg(Val.getName()))
|
|
|
|
OS << Val;
|
2002-12-02 01:23:04 +00:00
|
|
|
|
|
|
|
return OS << "}\n";
|
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
Init *Record::getValueInit(StringRef FieldName) const {
|
2003-08-01 06:27:59 +00:00
|
|
|
const RecordVal *R = getValue(FieldName);
|
2014-04-09 04:50:04 +00:00
|
|
|
if (!R || !R->getValue())
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() +
|
2014-03-29 17:17:15 +00:00
|
|
|
"' does not have a field named `" + FieldName + "'!\n");
|
2003-08-01 06:27:59 +00:00
|
|
|
return R->getValue();
|
|
|
|
}
|
|
|
|
|
2009-09-18 18:31:37 +00:00
|
|
|
std::string Record::getValueAsString(StringRef FieldName) const {
|
2003-08-01 04:37:57 +00:00
|
|
|
const RecordVal *R = getValue(FieldName);
|
2014-04-09 04:50:04 +00:00
|
|
|
if (!R || !R->getValue())
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() +
|
2014-03-29 17:17:15 +00:00
|
|
|
"' does not have a field named `" + FieldName + "'!\n");
|
2003-08-01 04:37:57 +00:00
|
|
|
|
2012-10-10 20:24:43 +00:00
|
|
|
if (StringInit *SI = dyn_cast<StringInit>(R->getValue()))
|
2003-08-01 04:37:57 +00:00
|
|
|
return SI->getValue();
|
2016-07-05 21:22:55 +00:00
|
|
|
if (CodeInit *CI = dyn_cast<CodeInit>(R->getValue()))
|
|
|
|
return CI->getValue();
|
|
|
|
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() + "', field `" +
|
2014-03-29 17:17:15 +00:00
|
|
|
FieldName + "' does not have a string initializer!");
|
2003-08-01 04:37:57 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
BitsInit *Record::getValueAsBitsInit(StringRef FieldName) const {
|
2003-08-01 04:46:24 +00:00
|
|
|
const RecordVal *R = getValue(FieldName);
|
2014-04-09 04:50:04 +00:00
|
|
|
if (!R || !R->getValue())
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() +
|
2014-03-29 17:17:15 +00:00
|
|
|
"' does not have a field named `" + FieldName + "'!\n");
|
2003-08-01 04:46:24 +00:00
|
|
|
|
2012-10-10 20:24:43 +00:00
|
|
|
if (BitsInit *BI = dyn_cast<BitsInit>(R->getValue()))
|
2003-08-01 04:46:24 +00:00
|
|
|
return BI;
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() + "', field `" +
|
2014-03-29 17:17:15 +00:00
|
|
|
FieldName + "' does not have a BitsInit initializer!");
|
2003-08-01 04:46:24 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
ListInit *Record::getValueAsListInit(StringRef FieldName) const {
|
2003-08-01 06:15:10 +00:00
|
|
|
const RecordVal *R = getValue(FieldName);
|
2014-04-09 04:50:04 +00:00
|
|
|
if (!R || !R->getValue())
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() +
|
2014-03-29 17:17:15 +00:00
|
|
|
"' does not have a field named `" + FieldName + "'!\n");
|
2003-08-01 06:15:10 +00:00
|
|
|
|
2012-10-10 20:24:43 +00:00
|
|
|
if (ListInit *LI = dyn_cast<ListInit>(R->getValue()))
|
2003-08-01 06:15:10 +00:00
|
|
|
return LI;
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() + "', field `" +
|
2014-03-29 17:17:15 +00:00
|
|
|
FieldName + "' does not have a list initializer!");
|
2003-08-01 06:15:10 +00:00
|
|
|
}
|
2003-08-01 04:46:24 +00:00
|
|
|
|
2009-11-22 04:24:42 +00:00
|
|
|
std::vector<Record*>
|
2009-09-18 18:31:37 +00:00
|
|
|
Record::getValueAsListOfDefs(StringRef FieldName) const {
|
2011-07-29 22:43:06 +00:00
|
|
|
ListInit *List = getValueAsListInit(FieldName);
|
2005-10-28 21:46:31 +00:00
|
|
|
std::vector<Record*> Defs;
|
2015-06-02 04:15:51 +00:00
|
|
|
for (Init *I : List->getValues()) {
|
|
|
|
if (DefInit *DI = dyn_cast<DefInit>(I))
|
2005-10-28 21:46:31 +00:00
|
|
|
Defs.push_back(DI->getDef());
|
2015-05-04 01:35:39 +00:00
|
|
|
else
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() + "', field `" +
|
2014-03-29 17:17:15 +00:00
|
|
|
FieldName + "' list is not entirely DefInit!");
|
2005-10-28 21:46:31 +00:00
|
|
|
}
|
|
|
|
return Defs;
|
|
|
|
}
|
|
|
|
|
2009-09-18 18:31:37 +00:00
|
|
|
int64_t Record::getValueAsInt(StringRef FieldName) const {
|
2003-08-01 06:15:10 +00:00
|
|
|
const RecordVal *R = getValue(FieldName);
|
2014-04-09 04:50:04 +00:00
|
|
|
if (!R || !R->getValue())
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() +
|
2014-03-29 17:17:15 +00:00
|
|
|
"' does not have a field named `" + FieldName + "'!\n");
|
2003-08-01 06:15:10 +00:00
|
|
|
|
2012-10-10 20:24:43 +00:00
|
|
|
if (IntInit *II = dyn_cast<IntInit>(R->getValue()))
|
2003-08-01 06:15:10 +00:00
|
|
|
return II->getValue();
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() + "', field `" +
|
2014-03-29 17:17:15 +00:00
|
|
|
FieldName + "' does not have an int initializer!");
|
2003-08-01 06:15:10 +00:00
|
|
|
}
|
2003-08-01 04:37:57 +00:00
|
|
|
|
2009-11-22 04:24:42 +00:00
|
|
|
std::vector<int64_t>
|
2009-09-18 18:31:37 +00:00
|
|
|
Record::getValueAsListOfInts(StringRef FieldName) const {
|
2011-07-29 22:43:06 +00:00
|
|
|
ListInit *List = getValueAsListInit(FieldName);
|
2008-10-17 01:33:43 +00:00
|
|
|
std::vector<int64_t> Ints;
|
2015-06-02 04:15:51 +00:00
|
|
|
for (Init *I : List->getValues()) {
|
|
|
|
if (IntInit *II = dyn_cast<IntInit>(I))
|
2007-11-11 11:19:37 +00:00
|
|
|
Ints.push_back(II->getValue());
|
2015-05-04 01:35:39 +00:00
|
|
|
else
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() + "', field `" +
|
2014-03-29 17:17:15 +00:00
|
|
|
FieldName + "' does not have a list of ints initializer!");
|
2007-11-11 11:19:37 +00:00
|
|
|
}
|
|
|
|
return Ints;
|
|
|
|
}
|
|
|
|
|
2011-06-27 21:06:21 +00:00
|
|
|
std::vector<std::string>
|
|
|
|
Record::getValueAsListOfStrings(StringRef FieldName) const {
|
2011-07-29 22:43:06 +00:00
|
|
|
ListInit *List = getValueAsListInit(FieldName);
|
2011-06-27 21:06:21 +00:00
|
|
|
std::vector<std::string> Strings;
|
2015-06-02 04:15:51 +00:00
|
|
|
for (Init *I : List->getValues()) {
|
|
|
|
if (StringInit *SI = dyn_cast<StringInit>(I))
|
|
|
|
Strings.push_back(SI->getValue());
|
2015-05-04 01:35:39 +00:00
|
|
|
else
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() + "', field `" +
|
2014-03-29 17:17:15 +00:00
|
|
|
FieldName + "' does not have a list of strings initializer!");
|
2011-06-27 21:06:21 +00:00
|
|
|
}
|
|
|
|
return Strings;
|
|
|
|
}
|
|
|
|
|
2009-09-18 18:31:37 +00:00
|
|
|
Record *Record::getValueAsDef(StringRef FieldName) const {
|
2003-08-02 01:27:37 +00:00
|
|
|
const RecordVal *R = getValue(FieldName);
|
2014-04-09 04:50:04 +00:00
|
|
|
if (!R || !R->getValue())
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() +
|
2014-03-29 17:17:15 +00:00
|
|
|
"' does not have a field named `" + FieldName + "'!\n");
|
2003-08-02 01:27:37 +00:00
|
|
|
|
2012-10-10 20:24:43 +00:00
|
|
|
if (DefInit *DI = dyn_cast<DefInit>(R->getValue()))
|
2003-08-02 01:27:37 +00:00
|
|
|
return DI->getDef();
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() + "', field `" +
|
2014-03-29 17:17:15 +00:00
|
|
|
FieldName + "' does not have a def initializer!");
|
2003-08-02 01:27:37 +00:00
|
|
|
}
|
|
|
|
|
2009-09-18 18:31:37 +00:00
|
|
|
bool Record::getValueAsBit(StringRef FieldName) const {
|
2003-08-03 21:58:13 +00:00
|
|
|
const RecordVal *R = getValue(FieldName);
|
2014-04-09 04:50:04 +00:00
|
|
|
if (!R || !R->getValue())
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() +
|
2014-03-29 17:17:15 +00:00
|
|
|
"' does not have a field named `" + FieldName + "'!\n");
|
2003-08-03 21:58:13 +00:00
|
|
|
|
2012-10-10 20:24:43 +00:00
|
|
|
if (BitInit *BI = dyn_cast<BitInit>(R->getValue()))
|
2003-08-07 06:00:43 +00:00
|
|
|
return BI->getValue();
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() + "', field `" +
|
2014-03-29 17:17:15 +00:00
|
|
|
FieldName + "' does not have a bit initializer!");
|
2003-08-07 06:00:43 +00:00
|
|
|
}
|
|
|
|
|
2012-08-23 19:34:46 +00:00
|
|
|
bool Record::getValueAsBitOrUnset(StringRef FieldName, bool &Unset) const {
|
|
|
|
const RecordVal *R = getValue(FieldName);
|
2014-04-09 04:50:04 +00:00
|
|
|
if (!R || !R->getValue())
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() +
|
|
|
|
"' does not have a field named `" + FieldName.str() + "'!\n");
|
2012-08-23 19:34:46 +00:00
|
|
|
|
2015-04-22 02:09:45 +00:00
|
|
|
if (isa<UnsetInit>(R->getValue())) {
|
2012-08-23 19:34:46 +00:00
|
|
|
Unset = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Unset = false;
|
2012-10-10 20:24:43 +00:00
|
|
|
if (BitInit *BI = dyn_cast<BitInit>(R->getValue()))
|
2012-08-23 19:34:46 +00:00
|
|
|
return BI->getValue();
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() + "', field `" +
|
2014-03-29 17:17:15 +00:00
|
|
|
FieldName + "' does not have a bit initializer!");
|
2012-08-23 19:34:46 +00:00
|
|
|
}
|
|
|
|
|
2011-07-29 22:43:06 +00:00
|
|
|
DagInit *Record::getValueAsDag(StringRef FieldName) const {
|
2003-08-07 06:00:43 +00:00
|
|
|
const RecordVal *R = getValue(FieldName);
|
2014-04-09 04:50:04 +00:00
|
|
|
if (!R || !R->getValue())
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() +
|
2014-03-29 17:17:15 +00:00
|
|
|
"' does not have a field named `" + FieldName + "'!\n");
|
2003-08-07 06:00:43 +00:00
|
|
|
|
2012-10-10 20:24:43 +00:00
|
|
|
if (DagInit *DI = dyn_cast<DagInit>(R->getValue()))
|
2003-08-07 06:00:43 +00:00
|
|
|
return DI;
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError(getLoc(), "Record `" + getName() + "', field `" +
|
2014-03-29 17:17:15 +00:00
|
|
|
FieldName + "' does not have a dag initializer!");
|
2003-08-03 21:58:13 +00:00
|
|
|
}
|
|
|
|
|
2016-01-29 20:50:44 +00:00
|
|
|
LLVM_DUMP_METHOD void MultiClass::dump() const {
|
2009-07-03 00:10:29 +00:00
|
|
|
errs() << "Record:\n";
|
2009-04-24 16:55:41 +00:00
|
|
|
Rec.dump();
|
2009-11-22 04:24:42 +00:00
|
|
|
|
2009-07-03 00:10:29 +00:00
|
|
|
errs() << "Defs:\n";
|
2015-05-04 01:35:39 +00:00
|
|
|
for (const auto &Proto : DefPrototypes)
|
2015-04-29 07:13:14 +00:00
|
|
|
Proto->dump();
|
2009-04-24 16:55:41 +00:00
|
|
|
}
|
|
|
|
|
2016-01-29 20:50:44 +00:00
|
|
|
LLVM_DUMP_METHOD void RecordKeeper::dump() const { errs() << *this; }
|
2002-12-02 01:23:04 +00:00
|
|
|
|
2009-07-03 00:10:29 +00:00
|
|
|
raw_ostream &llvm::operator<<(raw_ostream &OS, const RecordKeeper &RK) {
|
2002-12-02 01:23:04 +00:00
|
|
|
OS << "------------- Classes -----------------\n";
|
2015-05-12 05:25:10 +00:00
|
|
|
for (const auto &C : RK.getClasses())
|
2014-08-24 19:10:49 +00:00
|
|
|
OS << "class " << *C.second;
|
2005-04-22 00:00:37 +00:00
|
|
|
|
2002-12-02 01:23:04 +00:00
|
|
|
OS << "------------- Defs -----------------\n";
|
2015-05-12 05:25:10 +00:00
|
|
|
for (const auto &D : RK.getDefs())
|
2014-08-24 19:10:49 +00:00
|
|
|
OS << "def " << *D.second;
|
2002-12-02 01:23:04 +00:00
|
|
|
return OS;
|
|
|
|
}
|
2003-08-01 04:09:58 +00:00
|
|
|
|
2016-07-14 14:53:11 +00:00
|
|
|
std::vector<Record *>
|
2016-12-04 05:48:03 +00:00
|
|
|
RecordKeeper::getAllDerivedDefinitions(StringRef ClassName) const {
|
2010-12-13 00:20:52 +00:00
|
|
|
Record *Class = getClass(ClassName);
|
2003-08-01 04:37:57 +00:00
|
|
|
if (!Class)
|
2012-10-25 20:33:17 +00:00
|
|
|
PrintFatalError("ERROR: Couldn't find the `" + ClassName + "' class!\n");
|
2003-08-01 04:09:58 +00:00
|
|
|
|
2003-08-01 04:37:57 +00:00
|
|
|
std::vector<Record*> Defs;
|
2014-08-24 19:10:49 +00:00
|
|
|
for (const auto &D : getDefs())
|
|
|
|
if (D.second->isSubClassOf(Class))
|
2014-08-24 19:10:57 +00:00
|
|
|
Defs.push_back(D.second.get());
|
2003-08-01 04:09:58 +00:00
|
|
|
|
2003-08-01 04:37:57 +00:00
|
|
|
return Defs;
|
2003-08-01 04:09:58 +00:00
|
|
|
}
|
2003-11-11 22:41:34 +00:00
|
|
|
|
2016-12-05 05:21:18 +00:00
|
|
|
static Init *GetStrConcat(Init *I0, Init *I1) {
|
|
|
|
// Shortcut for the common case of concatenating two strings.
|
|
|
|
if (const StringInit *I0s = dyn_cast<StringInit>(I0))
|
|
|
|
if (const StringInit *I1s = dyn_cast<StringInit>(I1))
|
|
|
|
return ConcatStringInits(I0s, I1s);
|
|
|
|
return BinOpInit::get(BinOpInit::STRCONCAT, I0, I1, StringRecTy::get());
|
|
|
|
}
|
|
|
|
|
2011-10-19 13:02:36 +00:00
|
|
|
Init *llvm::QualifyName(Record &CurRec, MultiClass *CurMultiClass,
|
2016-12-04 05:48:03 +00:00
|
|
|
Init *Name, StringRef Scoper) {
|
2016-12-05 05:21:18 +00:00
|
|
|
Init *NewName = GetStrConcat(CurRec.getNameInit(), StringInit::get(Scoper));
|
|
|
|
NewName = GetStrConcat(NewName, Name);
|
2011-10-19 13:02:36 +00:00
|
|
|
if (CurMultiClass && Scoper != "::") {
|
2016-12-05 05:21:18 +00:00
|
|
|
Init *Prefix = GetStrConcat(CurMultiClass->Rec.getNameInit(),
|
|
|
|
StringInit::get("::"));
|
|
|
|
NewName = GetStrConcat(Prefix, NewName);
|
2011-10-19 13:02:36 +00:00
|
|
|
}
|
|
|
|
|
2016-12-05 05:21:18 +00:00
|
|
|
if (BinOpInit *BinOp = dyn_cast<BinOpInit>(NewName))
|
|
|
|
NewName = BinOp->Fold(&CurRec, CurMultiClass);
|
|
|
|
return NewName;
|
2011-10-19 13:02:36 +00:00
|
|
|
}
|