1a07aba5dSTimm Baeder //===--- Function.h - Bytecode function for the VM --------------*- C++ -*-===// 2a07aba5dSTimm Baeder // 3a07aba5dSTimm Baeder // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4a07aba5dSTimm Baeder // See https://llvm.org/LICENSE.txt for license information. 5a07aba5dSTimm Baeder // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6a07aba5dSTimm Baeder // 7a07aba5dSTimm Baeder //===----------------------------------------------------------------------===// 8a07aba5dSTimm Baeder 9a07aba5dSTimm Baeder #include "Function.h" 10a07aba5dSTimm Baeder #include "Program.h" 11a07aba5dSTimm Baeder #include "clang/AST/Decl.h" 12a07aba5dSTimm Baeder #include "clang/AST/DeclCXX.h" 13a07aba5dSTimm Baeder #include "clang/Basic/Builtins.h" 14a07aba5dSTimm Baeder 15a07aba5dSTimm Baeder using namespace clang; 16a07aba5dSTimm Baeder using namespace clang::interp; 17a07aba5dSTimm Baeder 18ca148b21STimm Baeder Function::Function(Program &P, FunctionDeclTy Source, unsigned ArgSize, 19a07aba5dSTimm Baeder llvm::SmallVectorImpl<PrimType> &&ParamTypes, 20a07aba5dSTimm Baeder llvm::DenseMap<unsigned, ParamDescriptor> &&Params, 21a07aba5dSTimm Baeder llvm::SmallVectorImpl<unsigned> &&ParamOffsets, 223745a2e8STimm Baeder bool HasThisPointer, bool HasRVO, unsigned BuiltinID) 23ca148b21STimm Baeder : P(P), Source(Source), ArgSize(ArgSize), ParamTypes(std::move(ParamTypes)), 24a07aba5dSTimm Baeder Params(std::move(Params)), ParamOffsets(std::move(ParamOffsets)), 253745a2e8STimm Baeder HasThisPointer(HasThisPointer), HasRVO(HasRVO), BuiltinID(BuiltinID) { 26ca148b21STimm Baeder if (const auto *F = Source.dyn_cast<const FunctionDecl *>()) 27ca148b21STimm Baeder Variadic = F->isVariadic(); 28ca148b21STimm Baeder } 29a07aba5dSTimm Baeder 30a07aba5dSTimm Baeder Function::ParamDescriptor Function::getParamDescriptor(unsigned Offset) const { 31a07aba5dSTimm Baeder auto It = Params.find(Offset); 32a07aba5dSTimm Baeder assert(It != Params.end() && "Invalid parameter offset"); 33a07aba5dSTimm Baeder return It->second; 34a07aba5dSTimm Baeder } 35a07aba5dSTimm Baeder 36a07aba5dSTimm Baeder SourceInfo Function::getSource(CodePtr PC) const { 37a07aba5dSTimm Baeder assert(PC >= getCodeBegin() && "PC does not belong to this function"); 38a07aba5dSTimm Baeder assert(PC <= getCodeEnd() && "PC Does not belong to this function"); 39a07aba5dSTimm Baeder assert(hasBody() && "Function has no body"); 40a07aba5dSTimm Baeder unsigned Offset = PC - getCodeBegin(); 41a07aba5dSTimm Baeder using Elem = std::pair<unsigned, SourceInfo>; 42a07aba5dSTimm Baeder auto It = llvm::lower_bound(SrcMap, Elem{Offset, {}}, llvm::less_first()); 43a07aba5dSTimm Baeder if (It == SrcMap.end()) 44a07aba5dSTimm Baeder return SrcMap.back().second; 45a07aba5dSTimm Baeder return It->second; 46a07aba5dSTimm Baeder } 47a07aba5dSTimm Baeder 48a07aba5dSTimm Baeder bool Function::isVirtual() const { 49ca148b21STimm Baeder if (const auto *M = dyn_cast_if_present<CXXMethodDecl>( 50ca148b21STimm Baeder Source.dyn_cast<const FunctionDecl *>())) 51a07aba5dSTimm Baeder return M->isVirtual(); 52a07aba5dSTimm Baeder return false; 53a07aba5dSTimm Baeder } 543745a2e8STimm Baeder 553745a2e8STimm Baeder /// Unevaluated builtins don't get their arguments put on the stack 563745a2e8STimm Baeder /// automatically. They instead operate on the AST of their Call 573745a2e8STimm Baeder /// Expression. 583745a2e8STimm Baeder /// Similar information is available via ASTContext::BuiltinInfo, 593745a2e8STimm Baeder /// but that is not correct for our use cases. 603745a2e8STimm Baeder static bool isUnevaluatedBuiltin(unsigned BuiltinID) { 613745a2e8STimm Baeder return BuiltinID == Builtin::BI__builtin_classify_type || 623745a2e8STimm Baeder BuiltinID == Builtin::BI__builtin_os_log_format_buffer_size || 63*f838d6b1STimm Baeder BuiltinID == Builtin::BI__builtin_constant_p || 64*f838d6b1STimm Baeder BuiltinID == Builtin::BI__noop; 653745a2e8STimm Baeder } 663745a2e8STimm Baeder 673745a2e8STimm Baeder bool Function::isUnevaluatedBuiltin() const { 683745a2e8STimm Baeder return ::isUnevaluatedBuiltin(BuiltinID); 693745a2e8STimm Baeder } 70