diff --git a/llvm/include/llvm/AsmParser/LLParser.h b/llvm/include/llvm/AsmParser/LLParser.h --- a/llvm/include/llvm/AsmParser/LLParser.h +++ b/llvm/include/llvm/AsmParser/LLParser.h @@ -613,7 +613,8 @@ unsigned Opc, bool IsFP); bool parseLogical(Instruction *&Inst, PerFunctionState &PFS, unsigned Opc); bool parseCompare(Instruction *&Inst, PerFunctionState &PFS, unsigned Opc); - bool parseCast(Instruction *&Inst, PerFunctionState &PFS, unsigned Opc); + bool parseCast(Instruction *&Inst, PerFunctionState &PFS, unsigned Opc, + bool NonNeg = false); bool parseSelect(Instruction *&Inst, PerFunctionState &PFS); bool parseVAArg(Instruction *&Inst, PerFunctionState &PFS); bool parseExtractElement(Instruction *&Inst, PerFunctionState &PFS); diff --git a/llvm/include/llvm/AsmParser/LLToken.h b/llvm/include/llvm/AsmParser/LLToken.h --- a/llvm/include/llvm/AsmParser/LLToken.h +++ b/llvm/include/llvm/AsmParser/LLToken.h @@ -110,6 +110,7 @@ kw_nsw, kw_exact, kw_inbounds, + kw_nneg, kw_inrange, kw_addrspace, kw_section, diff --git a/llvm/include/llvm/Bitcode/LLVMBitCodes.h b/llvm/include/llvm/Bitcode/LLVMBitCodes.h --- a/llvm/include/llvm/Bitcode/LLVMBitCodes.h +++ b/llvm/include/llvm/Bitcode/LLVMBitCodes.h @@ -505,6 +505,10 @@ AllowReassoc = (1 << 7) }; +/// NonNegOperatorOptionalFlags - Flags for serializing +/// NonNegOperator's SubclassOptionalData contents. +enum NonNegOperatorOptionalFlags { NNO_NON_NEG = 0 }; + /// PossiblyExactOperatorOptionalFlags - Flags for serializing /// PossiblyExactOperator's SubclassOptionalData contents. enum PossiblyExactOperatorOptionalFlags { PEO_EXACT = 0 }; diff --git a/llvm/include/llvm/IR/Constants.h b/llvm/include/llvm/IR/Constants.h --- a/llvm/include/llvm/IR/Constants.h +++ b/llvm/include/llvm/IR/Constants.h @@ -1056,7 +1056,8 @@ static Constant *getAShr(Constant *C1, Constant *C2, bool isExact = false); static Constant *getTrunc(Constant *C, Type *Ty, bool OnlyIfReduced = false); static Constant *getSExt(Constant *C, Type *Ty, bool OnlyIfReduced = false); - static Constant *getZExt(Constant *C, Type *Ty, bool OnlyIfReduced = false); + static Constant *getZExt(Constant *C, Type *Ty, bool OnlyIfReduced = false, + bool NonNeg = false); static Constant *getFPTrunc(Constant *C, Type *Ty, bool OnlyIfReduced = false); static Constant *getFPExtend(Constant *C, Type *Ty, diff --git a/llvm/include/llvm/IR/Instruction.h b/llvm/include/llvm/IR/Instruction.h --- a/llvm/include/llvm/IR/Instruction.h +++ b/llvm/include/llvm/IR/Instruction.h @@ -369,12 +369,19 @@ /// which supports this flag. See LangRef.html for the meaning of this flag. void setIsExact(bool b = true); + /// Set or clear the sext flag on this instruction, which must be an operator + /// which supports this flag. In our case NonNegOperator defined in Operator.h. + void setNonNeg(bool b = true); + /// Determine whether the no unsigned wrap flag is set. bool hasNoUnsignedWrap() const LLVM_READONLY; /// Determine whether the no signed wrap flag is set. bool hasNoSignedWrap() const LLVM_READONLY; + /// Determine whether the the value of this Instr is NonNeg. + bool hasNonNeg() const LLVM_READONLY; + /// Return true if this operator has flags which may cause this instruction /// to evaluate to poison despite having non-poison inputs. bool hasPoisonGeneratingFlags() const LLVM_READONLY; diff --git a/llvm/include/llvm/IR/Operator.h b/llvm/include/llvm/IR/Operator.h --- a/llvm/include/llvm/IR/Operator.h +++ b/llvm/include/llvm/IR/Operator.h @@ -331,6 +331,36 @@ } }; +/// An utility class for the zext instructions that were +/// created by a conversion from sext instructions. +class NonNegOperator : public Operator { +public: + enum { NonNeg = (1 << 0) }; + +private: + friend class Instruction; + friend class ConstantExpr; + + void setNonNeg(bool B) { + SubclassOptionalData = (SubclassOptionalData & ~NonNeg) | (B * NonNeg); + } + +public: + /// + bool hasNonNeg() const { return (SubclassOptionalData & NonNeg) != 0; } + + static bool classof(const Instruction *I) { + return I->getOpcode() == Instruction::ZExt; + } + static bool classof(const ConstantExpr *CE) { + return CE->getOpcode() == Instruction::ZExt; + } + static bool classof(const Value *V) { + return (isa(V) && classof(cast(V))) || + (isa(V) && classof(cast(V))); + } +}; + /// A helper template for defining operators for individual opcodes. template class ConcreteOperator : public SuperClass { @@ -373,7 +403,8 @@ : public ConcreteOperator { }; -class ZExtOperator : public ConcreteOperator {}; +class ZExtOperator : public ConcreteOperator { +}; class GEPOperator : public ConcreteOperator { diff --git a/llvm/lib/AsmParser/LLLexer.cpp b/llvm/lib/AsmParser/LLLexer.cpp --- a/llvm/lib/AsmParser/LLLexer.cpp +++ b/llvm/lib/AsmParser/LLLexer.cpp @@ -564,6 +564,7 @@ KEYWORD(fast); KEYWORD(nuw); KEYWORD(nsw); + KEYWORD(nneg); KEYWORD(exact); KEYWORD(inbounds); KEYWORD(inrange); diff --git a/llvm/lib/AsmParser/LLParser.cpp b/llvm/lib/AsmParser/LLParser.cpp --- a/llvm/lib/AsmParser/LLParser.cpp +++ b/llvm/lib/AsmParser/LLParser.cpp @@ -3740,9 +3740,13 @@ case lltok::kw_inttoptr: case lltok::kw_ptrtoint: { unsigned Opc = Lex.getUIntVal(); + bool NonNeg = false; Type *DestTy = nullptr; Constant *SrcVal; Lex.Lex(); + if (Opc == Instruction::ZExt && EatIfPresent(lltok::kw_nneg)) + NonNeg = true; + if (parseToken(lltok::lparen, "expected '(' after constantexpr cast") || parseGlobalTypeAndValue(SrcVal) || parseToken(lltok::kw_to, "expected 'to' in constantexpr cast") || @@ -3753,9 +3757,17 @@ return error(ID.Loc, "invalid cast opcode for cast from '" + getTypeString(SrcVal->getType()) + "' to '" + getTypeString(DestTy) + "'"); - ID.ConstantVal = ConstantExpr::getCast((Instruction::CastOps)Opc, - SrcVal, DestTy); + ID.Kind = ValID::t_Constant; + if (NonNeg) { + ID.ConstantVal = + ConstantExpr::getZExt(SrcVal, DestTy, + /* OnlyIfReduced */ false, /* NonNeg */ true); + + } else { + ID.ConstantVal = + ConstantExpr::getCast((Instruction::CastOps)Opc, SrcVal, DestTy); + } return false; } case lltok::kw_extractvalue: @@ -6364,8 +6376,15 @@ } // Casts. + case lltok::kw_zext: { + bool NonNeg = EatIfPresent(lltok::kw_nneg); + bool Res = parseCast(Inst, PFS, KeywordVal, NonNeg); + if (Res != 0) { + return Res; + } + return 0; + } case lltok::kw_trunc: - case lltok::kw_zext: case lltok::kw_sext: case lltok::kw_fptrunc: case lltok::kw_fpext: @@ -7143,7 +7162,7 @@ /// parseCast /// ::= CastOpc TypeAndValue 'to' Type bool LLParser::parseCast(Instruction *&Inst, PerFunctionState &PFS, - unsigned Opc) { + unsigned Opc, bool NonNeg/*=false*/) { LocTy Loc; Value *Op; Type *DestTy = nullptr; @@ -7159,6 +7178,7 @@ getTypeString(DestTy) + "'"); } Inst = CastInst::Create((Instruction::CastOps)Opc, Op, DestTy); + if(NonNeg) Inst->setNonNeg(true); return false; } diff --git a/llvm/lib/Bitcode/Reader/BitcodeReader.cpp b/llvm/lib/Bitcode/Reader/BitcodeReader.cpp --- a/llvm/lib/Bitcode/Reader/BitcodeReader.cpp +++ b/llvm/lib/Bitcode/Reader/BitcodeReader.cpp @@ -1582,6 +1582,9 @@ if (Instruction::isCast(BC->Opcode)) { I = CastInst::Create((Instruction::CastOps)BC->Opcode, Ops[0], BC->getType(), "constexpr", InsertBB); + if (isa(I) && (BC->Flags & NonNegOperator::NonNeg)) + I->setNonNeg(true); + } else if (Instruction::isUnaryOp(BC->Opcode)) { I = UnaryOperator::Create((Instruction::UnaryOps)BC->Opcode, Ops[0], "constexpr", InsertBB); @@ -3210,18 +3213,22 @@ } break; } - case bitc::CST_CODE_CE_CAST: { // CE_CAST: [opcode, opty, opval] + case bitc::CST_CODE_CE_CAST: { // CE_CAST: [opcode, opty, opval] if (Record.size() < 3) return error("Invalid cast constexpr record"); int Opc = getDecodedCastOpcode(Record[0]); if (Opc < 0) { V = UndefValue::get(CurTy); // Unknown cast. } else { + uint8_t Flags = 0; + if (Record.size() >= 4 && (Record[3] & (1 << bitc::NNO_NON_NEG))) + Flags |= NonNegOperator::NonNeg; unsigned OpTyID = Record[1]; Type *OpTy = getTypeByID(OpTyID); if (!OpTy) return error("Invalid cast constexpr record"); - V = BitcodeConstant::create(Alloc, CurTy, Opc, (unsigned)Record[2]); + V = BitcodeConstant::create(Alloc, CurTy, {(uint8_t)Opc, Flags}, + (unsigned)Record[2]); } break; } @@ -4884,17 +4891,17 @@ } break; } - case bitc::FUNC_CODE_INST_CAST: { // CAST: [opval, opty, destty, castopc] + case bitc::FUNC_CODE_INST_CAST: { // CAST: [opval, opty, destty, castopc] unsigned OpNum = 0; Value *Op; unsigned OpTypeID; if (getValueTypePair(Record, OpNum, NextValueNo, Op, OpTypeID, CurBB) || - OpNum+2 != Record.size()) + OpNum + 1 > Record.size()) return error("Invalid record"); - ResTypeID = Record[OpNum]; + ResTypeID = Record[OpNum++]; Type *ResTy = getTypeByID(ResTypeID); - int Opc = getDecodedCastOpcode(Record[OpNum + 1]); + int Opc = getDecodedCastOpcode(Record[OpNum++]); if (Opc == -1 || !ResTy) return error("Invalid record"); Instruction *Temp = nullptr; @@ -4910,6 +4917,9 @@ return error("Invalid cast"); I = CastInst::Create(CastOp, Op, ResTy); } + if (OpNum < Record.size() && (isa(I)) && + (Record[OpNum] & (1 << bitc::NNO_NON_NEG))) + cast(I)->setNonNeg(true); InstructionList.push_back(I); break; } @@ -6401,7 +6411,7 @@ if (Error Err = propagateAttributeTypes(cast(I), ArgTyIDs)) { I->deleteValue(); return Err; - } + } // TODO is nneg flag possible on args? if (FMF.any()) { if (!isa(I)) return error("Fast-math-flags specified for call without " diff --git a/llvm/lib/Bitcode/Writer/BitcodeWriter.cpp b/llvm/lib/Bitcode/Writer/BitcodeWriter.cpp --- a/llvm/lib/Bitcode/Writer/BitcodeWriter.cpp +++ b/llvm/lib/Bitcode/Writer/BitcodeWriter.cpp @@ -114,6 +114,7 @@ CONSTANTS_SETTYPE_ABBREV = bitc::FIRST_APPLICATION_ABBREV, CONSTANTS_INTEGER_ABBREV, CONSTANTS_CE_CAST_Abbrev, + CONSTANTS_CE_CAST_FLAGS_ABBREV, CONSTANTS_NULL_Abbrev, // FUNCTION_BLOCK abbrev id's. @@ -123,6 +124,7 @@ FUNCTION_INST_BINOP_ABBREV, FUNCTION_INST_BINOP_FLAGS_ABBREV, FUNCTION_INST_CAST_ABBREV, + FUNCTION_INST_CAST_FLAGS_ABBREV, FUNCTION_INST_RET_VOID_ABBREV, FUNCTION_INST_RET_VAL_ABBREV, FUNCTION_INST_UNREACHABLE_ABBREV, @@ -1551,8 +1553,11 @@ Flags |= bitc::AllowContract; if (FPMO->hasApproxFunc()) Flags |= bitc::ApproxFunc; + } else if (const auto *NNO = dyn_cast(V)) { + if (NNO->hasNonNeg()) { + Flags |= 1 << bitc::NNO_NON_NEG; + } } - return Flags; } @@ -2647,6 +2652,11 @@ Record.push_back(VE.getTypeID(C->getOperand(0)->getType())); Record.push_back(VE.getValueID(C->getOperand(0))); AbbrevToUse = CONSTANTS_CE_CAST_Abbrev; + uint64_t Flags = getOptimizationFlags(CE); + if (Flags != 0) { + AbbrevToUse = CONSTANTS_CE_CAST_FLAGS_ABBREV; + Record.push_back(Flags); + } } else { assert(CE->getNumOperands() == 2 && "Unknown constant expr!"); Code = bitc::CST_CODE_CE_BINOP; @@ -2833,6 +2843,12 @@ AbbrevToUse = FUNCTION_INST_CAST_ABBREV; Vals.push_back(VE.getTypeID(I.getType())); Vals.push_back(getEncodedCastOpcode(I.getOpcode())); + uint64_t Flags = getOptimizationFlags(&I); + if (Flags != 0) { + if (AbbrevToUse == FUNCTION_INST_CAST_ABBREV) + AbbrevToUse = FUNCTION_INST_CAST_FLAGS_ABBREV; + Vals.push_back(Flags); + } } else { assert(isa(I) && "Unknown instruction!"); Code = bitc::FUNC_CODE_INST_BINOP; @@ -3568,6 +3584,17 @@ CONSTANTS_INTEGER_ABBREV) llvm_unreachable("Unexpected abbrev ordering!"); } + { // CE_CAST abbrev for CONSTANTS_BLOCK. + auto Abbv = std::make_shared(); + Abbv->Add(BitCodeAbbrevOp(bitc::CST_CODE_CE_CAST)); + Abbv->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 4)); // cast opc + Abbv->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, // typeid + VE.computeBitsRequiredForTypeIndicies())); + Abbv->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::VBR, 8)); // value id + if (Stream.EmitBlockInfoAbbrev(bitc::CONSTANTS_BLOCK_ID, Abbv) != + CONSTANTS_CE_CAST_Abbrev) + llvm_unreachable("Unexpected abbrev ordering!"); + } { // CE_CAST abbrev for CONSTANTS_BLOCK. auto Abbv = std::make_shared(); @@ -3576,9 +3603,10 @@ Abbv->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, // typeid VE.computeBitsRequiredForTypeIndicies())); Abbv->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::VBR, 8)); // value id + Abbv->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 8)); // flags if (Stream.EmitBlockInfoAbbrev(bitc::CONSTANTS_BLOCK_ID, Abbv) != - CONSTANTS_CE_CAST_Abbrev) + CONSTANTS_CE_CAST_FLAGS_ABBREV) llvm_unreachable("Unexpected abbrev ordering!"); } { // NULL abbrev for CONSTANTS_BLOCK. @@ -3654,6 +3682,18 @@ FUNCTION_INST_CAST_ABBREV) llvm_unreachable("Unexpected abbrev ordering!"); } + { // INST_CAST FLAGS abbrev for FUNCTION_BLOCK. + auto Abbv = std::make_shared(); + Abbv->Add(BitCodeAbbrevOp(bitc::FUNC_CODE_INST_CAST)); + Abbv->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::VBR, 6)); // OpVal + Abbv->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, // dest ty + VE.computeBitsRequiredForTypeIndicies())); + Abbv->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 4)); // opc + Abbv->Add(BitCodeAbbrevOp(BitCodeAbbrevOp::Fixed, 8)); // flags + if (Stream.EmitBlockInfoAbbrev(bitc::FUNCTION_BLOCK_ID, Abbv) != + FUNCTION_INST_CAST_FLAGS_ABBREV) + llvm_unreachable("Unexpected abbrev ordering!"); + } { // INST_RET abbrev for FUNCTION_BLOCK. auto Abbv = std::make_shared(); diff --git a/llvm/lib/IR/AsmWriter.cpp b/llvm/lib/IR/AsmWriter.cpp --- a/llvm/lib/IR/AsmWriter.cpp +++ b/llvm/lib/IR/AsmWriter.cpp @@ -1343,6 +1343,9 @@ } else if (const GEPOperator *GEP = dyn_cast(U)) { if (GEP->isInBounds()) Out << " inbounds"; + } else if (const NonNegOperator *NNEG = dyn_cast(U)) { + if (NNEG->hasNonNeg()) + Out << " nneg"; } } diff --git a/llvm/lib/IR/Constants.cpp b/llvm/lib/IR/Constants.cpp --- a/llvm/lib/IR/Constants.cpp +++ b/llvm/lib/IR/Constants.cpp @@ -1953,7 +1953,8 @@ /// This is a utility function to handle folding of casts and lookup of the /// cast in the ExprConstants map. It is used by the various get* methods below. static Constant *getFoldedCast(Instruction::CastOps opc, Constant *C, Type *Ty, - bool OnlyIfReduced = false) { + bool OnlyIfReduced = false, + bool NonNeg = false) { assert(Ty->isFirstClassType() && "Cannot cast to an aggregate type!"); // Fold a few common cases if (Constant *FC = ConstantFoldCastInstruction(opc, C, Ty)) @@ -1965,7 +1966,7 @@ LLVMContextImpl *pImpl = Ty->getContext().pImpl; // Look up the constant in the table first to ensure uniqueness. - ConstantExprKeyType Key(opc, C); + ConstantExprKeyType Key(opc, C, 0, NonNeg); return pImpl->ExprConstants.getOrCreate(Ty, Key); } @@ -2116,7 +2117,8 @@ return getFoldedCast(Instruction::SExt, C, Ty, OnlyIfReduced); } -Constant *ConstantExpr::getZExt(Constant *C, Type *Ty, bool OnlyIfReduced) { +Constant *ConstantExpr::getZExt(Constant *C, Type *Ty, bool OnlyIfReduced, + bool NonNeg) { #ifndef NDEBUG bool fromVec = isa(C->getType()); bool toVec = isa(Ty); @@ -2127,7 +2129,7 @@ assert(C->getType()->getScalarSizeInBits() < Ty->getScalarSizeInBits()&& "SrcTy must be smaller than DestTy for ZExt!"); - return getFoldedCast(Instruction::ZExt, C, Ty, OnlyIfReduced); + return getFoldedCast(Instruction::ZExt, C, Ty, OnlyIfReduced, NonNeg); } Constant *ConstantExpr::getFPTrunc(Constant *C, Type *Ty, bool OnlyIfReduced) { diff --git a/llvm/lib/IR/Instruction.cpp b/llvm/lib/IR/Instruction.cpp --- a/llvm/lib/IR/Instruction.cpp +++ b/llvm/lib/IR/Instruction.cpp @@ -170,6 +170,10 @@ cast(this)->setIsExact(b); } +void Instruction::setNonNeg(bool b) { + cast(this)->setNonNeg(b); +} + bool Instruction::hasNoUnsignedWrap() const { return cast(this)->hasNoUnsignedWrap(); } @@ -178,6 +182,10 @@ return cast(this)->hasNoSignedWrap(); } +bool Instruction::hasNonNeg() const { + return cast(this)->hasNonNeg(); +} + bool Instruction::hasPoisonGeneratingFlags() const { return cast(this)->hasPoisonGeneratingFlags(); } @@ -202,6 +210,10 @@ case Instruction::GetElementPtr: cast(this)->setIsInBounds(false); break; + + case Instruction::ZExt: + cast(this)->setNonNeg(false); + break; } if (isa(this)) { setHasNoNaNs(false); @@ -367,6 +379,10 @@ if (auto *SrcGEP = dyn_cast(V)) if (auto *DestGEP = dyn_cast(this)) DestGEP->setIsInBounds(SrcGEP->isInBounds() || DestGEP->isInBounds()); + + if (auto *NNO = dyn_cast(V)) + if(isa(this)) + setNonNeg(NNO->hasNonNeg()); } void Instruction::andIRFlags(const Value *V) { @@ -392,6 +408,10 @@ if (auto *SrcGEP = dyn_cast(V)) if (auto *DestGEP = dyn_cast(this)) DestGEP->setIsInBounds(SrcGEP->isInBounds() && DestGEP->isInBounds()); + + if (auto *NNO = dyn_cast(V)) + if(isa(this)) + setNonNeg(hasNonNeg() && NNO->hasNonNeg()); } const char *Instruction::getOpcodeName(unsigned OpCode) { diff --git a/llvm/lib/IR/Operator.cpp b/llvm/lib/IR/Operator.cpp --- a/llvm/lib/IR/Operator.cpp +++ b/llvm/lib/IR/Operator.cpp @@ -37,6 +37,8 @@ // Note: inrange exists on constexpr only return GEP->isInBounds() || GEP->getInRangeIndex() != std::nullopt; } + case Instruction::ZExt: + return cast(this)->hasNonNeg(); default: if (const auto *FP = dyn_cast(this)) return FP->hasNoNaNs() || FP->hasNoInfs(); diff --git a/llvm/test/Assembler/flags.ll b/llvm/test/Assembler/flags.ll --- a/llvm/test/Assembler/flags.ll +++ b/llvm/test/Assembler/flags.ll @@ -259,4 +259,10 @@ ; CHECK: ret i64 mul nuw (i64 ptrtoint (ptr @addr to i64), i64 91) ret i64 mul nuw (i64 ptrtoint (ptr @addr to i64), i64 91) } +define i64 @test_zext(i32 %a) { +; CHECK: %res = zext nneg i32 %a to i64 + %res = zext nneg i32 %a to i64 + ret i64 %res +} + diff --git a/llvm/test/Bitcode/constantsTest.3.2.ll b/llvm/test/Bitcode/constantsTest.3.2.ll --- a/llvm/test/Bitcode/constantsTest.3.2.ll +++ b/llvm/test/Bitcode/constantsTest.3.2.ll @@ -121,4 +121,4 @@ insertvalue { i32, float } { i32 1, float 2.0 }, i32 0, 0 ret void -} \ No newline at end of file +} diff --git a/llvm/test/Bitcode/flags.ll b/llvm/test/Bitcode/flags.ll --- a/llvm/test/Bitcode/flags.ll +++ b/llvm/test/Bitcode/flags.ll @@ -16,6 +16,8 @@ %s = add nsw i32 %a, 0 ; [#uses=0] %us = add nuw nsw i32 %a, 0 ; [#uses=0] %z = add i32 %a, 0 ; [#uses=0] + %hh = zext i32 %a to i64 ; [#uses=0] + %ff = zext nneg i32 %a to i64 unreachable first: ; preds = %entry @@ -24,5 +26,7 @@ %ss = add nsw i32 %a, 0 ; [#uses=0] %uuss = add nuw nsw i32 %a, 0 ; [#uses=0] %zz = add i32 %a, 0 ; [#uses=0] + %ll = zext i32 %a to i64 ; [#uses=0] + %kk = zext nneg i32 %a to i64 ; [#uses=0] br label %second }