| //===- subzero/src/IceGlobalInits.cpp - Global declarations ---------------===// |
| // |
| // The Subzero Code Generator |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| /// |
| /// \file |
| /// \brief Implements the notion of function declarations, global variable |
| /// declarations, and the corresponding variable initializers in Subzero. |
| /// |
| //===----------------------------------------------------------------------===// |
| |
| #include "IceGlobalInits.h" |
| |
| #include "IceDefs.h" |
| #include "IceGlobalContext.h" |
| #include "IceTypes.h" |
| #include "llvm/ADT/STLExtras.h" |
| #include "llvm/IR/Function.h" |
| #include "llvm/IR/Value.h" |
| |
| namespace { |
| char hexdigit(unsigned X) { return X < 10 ? '0' + X : 'A' + X - 10; } |
| |
| void dumpLinkage(Ice::Ostream &Stream, |
| llvm::GlobalValue::LinkageTypes Linkage) { |
| if (!Ice::BuildDefs::dump()) |
| return; |
| switch (Linkage) { |
| case llvm::GlobalValue::ExternalLinkage: |
| Stream << "external"; |
| return; |
| case llvm::GlobalValue::InternalLinkage: |
| Stream << "internal"; |
| return; |
| default: |
| break; |
| } |
| std::string Buffer; |
| llvm::raw_string_ostream StrBuf(Buffer); |
| StrBuf << "Unknown linkage value: " << Linkage; |
| llvm::report_fatal_error(StrBuf.str()); |
| } |
| |
| void dumpCallingConv(Ice::Ostream &, llvm::CallingConv::ID CallingConv) { |
| if (!Ice::BuildDefs::dump()) |
| return; |
| if (CallingConv == llvm::CallingConv::C) |
| return; |
| std::string Buffer; |
| llvm::raw_string_ostream StrBuf(Buffer); |
| StrBuf << "Unknown calling convention: " << CallingConv; |
| llvm::report_fatal_error(StrBuf.str()); |
| } |
| |
| } // end of anonymous namespace |
| |
| namespace Ice { |
| |
| const Intrinsics::FullIntrinsicInfo * |
| FunctionDeclaration::getIntrinsicInfo(const GlobalContext *Ctx, |
| bool *IsIntrinsic) const { |
| *IsIntrinsic = false; |
| if (!hasName()) |
| return nullptr; |
| bool BadIntrinsic; |
| const Intrinsics::FullIntrinsicInfo *Info = |
| Ctx->getIntrinsicsInfo().find(getName(), BadIntrinsic); |
| *IsIntrinsic = Info || BadIntrinsic; |
| return Info; |
| } |
| |
| bool FunctionDeclaration::validateRegularTypeSignature() const { |
| for (SizeT i = 0; i < Signature.getNumArgs(); ++i) { |
| if (!isCallParameterType(Signature.getArgType(i))) |
| return false; |
| } |
| return isCallReturnType(Signature.getReturnType()); |
| } |
| |
| bool FunctionDeclaration::validateIntrinsicTypeSignature( |
| const Intrinsics::FullIntrinsicInfo *Info) const { |
| if (Signature.getNumArgs() != Info->getNumArgs()) |
| return false; |
| for (SizeT i = 0; i < Signature.getNumArgs(); ++i) { |
| if (Signature.getArgType(i) != Info->getArgType(i)) |
| return false; |
| } |
| return Signature.getReturnType() == Info->getReturnType(); |
| } |
| |
| std::string |
| FunctionDeclaration::getTypeSignatureError(const GlobalContext *Ctx) { |
| std::string Buffer; |
| llvm::raw_string_ostream StrBuf(Buffer); |
| StrBuf << "Invalid"; |
| bool IsIntrinsic; |
| const Intrinsics::FullIntrinsicInfo *Info = |
| getIntrinsicInfo(Ctx, &IsIntrinsic); |
| if (IsIntrinsic && Info == nullptr) { |
| StrBuf << " intrinsic name: " << getName(); |
| return StrBuf.str(); |
| } |
| StrBuf << " type signature for"; |
| if (IsIntrinsic) |
| StrBuf << " intrinsic"; |
| StrBuf << " " << getName() << ": " << getSignature(); |
| return StrBuf.str(); |
| } |
| |
| void FunctionDeclaration::dumpType(Ostream &Stream) const { |
| if (!Ice::BuildDefs::dump()) |
| return; |
| Stream << Signature; |
| } |
| |
| void FunctionDeclaration::dump(Ostream &Stream) const { |
| if (!Ice::BuildDefs::dump()) |
| return; |
| if (IsProto) |
| Stream << "declare "; |
| ::dumpLinkage(Stream, Linkage); |
| ::dumpCallingConv(Stream, CallingConv); |
| Stream << Signature.getReturnType() << " @" << Name << "("; |
| bool IsFirst = true; |
| for (Type ArgTy : Signature.getArgList()) { |
| if (IsFirst) |
| IsFirst = false; |
| else |
| Stream << ", "; |
| Stream << ArgTy; |
| } |
| Stream << ")"; |
| } |
| |
| void VariableDeclaration::dumpType(Ostream &Stream) const { |
| if (!Ice::BuildDefs::dump()) |
| return; |
| if (Initializers.size() == 1) { |
| Initializers.front()->dumpType(Stream); |
| } else { |
| Stream << "<{ "; |
| bool IsFirst = true; |
| for (const auto *Init : Initializers) { |
| if (IsFirst) { |
| IsFirst = false; |
| } else { |
| Stream << ", "; |
| } |
| Init->dumpType(Stream); |
| } |
| Stream << " }>"; |
| } |
| } |
| |
| void VariableDeclaration::dump(Ostream &Stream) const { |
| if (!Ice::BuildDefs::dump()) |
| return; |
| Stream << "@" << Name << " = "; |
| ::dumpLinkage(Stream, Linkage); |
| Stream << " " << (IsConstant ? "constant" : "global") << " "; |
| |
| // Add initializer. |
| if (Initializers.size() == 1) { |
| Initializers.front()->dump(Stream); |
| } else { |
| dumpType(Stream); |
| Stream << " <{ "; |
| bool IsFirst = true; |
| for (const auto *Init : Initializers) { |
| if (IsFirst) { |
| IsFirst = false; |
| } else { |
| Stream << ", "; |
| } |
| Init->dump(Stream); |
| } |
| Stream << " }>"; |
| } |
| |
| // Add alignment. |
| if (Alignment > 0) |
| Stream << ", align " << Alignment; |
| Stream << "\n"; |
| } |
| |
| void VariableDeclaration::Initializer::dumpType(Ostream &Stream) const { |
| if (!Ice::BuildDefs::dump()) |
| return; |
| Stream << "[" << getNumBytes() << " x " << Ice::IceType_i8 << "]"; |
| } |
| |
| void VariableDeclaration::DataInitializer::dump(Ostream &Stream) const { |
| if (!Ice::BuildDefs::dump()) |
| return; |
| dumpType(Stream); |
| Stream << " c\""; |
| // Code taken from PrintEscapedString() in AsmWriter.cpp. Keep the strings in |
| // the same format as the .ll file for practical diffing. |
| for (SizeT i = 0; i < ContentsSize; ++i) { |
| uint8_t C = Contents[i]; |
| if (isprint(C) && C != '\\' && C != '"') |
| Stream << C; |
| else |
| Stream << '\\' << hexdigit(C >> 4) << hexdigit(C & 0x0F); |
| } |
| Stream << "\""; |
| } |
| |
| void VariableDeclaration::ZeroInitializer::dump(Ostream &Stream) const { |
| if (!Ice::BuildDefs::dump()) |
| return; |
| dumpType(Stream); |
| Stream << " zeroinitializer"; |
| } |
| |
| void VariableDeclaration::RelocInitializer::dumpType(Ostream &Stream) const { |
| if (!Ice::BuildDefs::dump()) |
| return; |
| Stream << Ice::IceType_i32; |
| } |
| |
| void VariableDeclaration::RelocInitializer::dump(Ostream &Stream) const { |
| if (!Ice::BuildDefs::dump()) |
| return; |
| const RelocOffsetT Offset = getOffset(); |
| if (Offset != 0) { |
| dumpType(Stream); |
| Stream << " add ("; |
| } |
| dumpType(Stream); |
| Stream << " ptrtoint ("; |
| Declaration->dumpType(Stream); |
| Stream << "* @" << Declaration->getName() << " to "; |
| dumpType(Stream); |
| Stream << ")"; |
| if (Offset != 0) { |
| Stream << ", "; |
| dumpType(Stream); |
| Stream << " " << Offset << ")"; |
| } |
| } |
| |
| } // end of namespace Ice |