зеркало из https://github.com/microsoft/clang-1.git
If the base type of a member call is a record type we don't need to emit a virtual call.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@83816 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Родитель
7a9474e025
Коммит
3b89f3fa61
|
@ -230,10 +230,13 @@ RValue CodeGenFunction::EmitCXXMemberCallExpr(const CXXMemberCallExpr *CE) {
|
|||
// C++ [class.virtual]p12:
|
||||
// Explicit qualification with the scope operator (5.1) suppresses the
|
||||
// virtual call mechanism.
|
||||
//
|
||||
// We also don't emit a virtual call if the base expression has a record type
|
||||
// because then we know what the type is.
|
||||
llvm::Value *Callee;
|
||||
if (MD->isVirtual() && !ME->hasQualifier())
|
||||
// FIXME: push getCanonicalDecl as a conversion using the static type system (CanCXXMethodDecl).
|
||||
Callee = BuildVirtualCall(MD->getCanonicalDecl(), This, Ty);
|
||||
if (MD->isVirtual() && !ME->hasQualifier() &&
|
||||
!ME->getBase()->getType()->isRecordType())
|
||||
Callee = BuildVirtualCall(MD, This, Ty);
|
||||
else if (const CXXDestructorDecl *Destructor
|
||||
= dyn_cast<CXXDestructorDecl>(MD))
|
||||
Callee = CGM.GetAddrOfFunction(GlobalDecl(Destructor, Dtor_Complete), Ty);
|
||||
|
@ -795,8 +798,6 @@ CodeGenFunction::GetVirtualCXXBaseClassOffset(llvm::Value *This,
|
|||
llvm::Value *
|
||||
CodeGenFunction::BuildVirtualCall(const CXXMethodDecl *MD, llvm::Value *&This,
|
||||
const llvm::Type *Ty) {
|
||||
// FIXME: If we know the dynamic type, we don't have to do a virtual dispatch.
|
||||
|
||||
int64_t Index = CGM.getVtableInfo().getMethodVtableIndex(MD);
|
||||
|
||||
Ty = llvm::PointerType::get(Ty, 0);
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
// RUN: clang-cc -emit-llvm-only %s
|
||||
// RUN: clang-cc %s -emit-llvm -o - | FileCheck %s
|
||||
|
||||
// PR5021
|
||||
struct A {
|
||||
|
@ -8,3 +8,10 @@ struct A {
|
|||
void f(A *a) {
|
||||
a->f('c');
|
||||
}
|
||||
|
||||
void f(A a) {
|
||||
// This should not be a virtual function call.
|
||||
|
||||
// CHECK: call void @_ZN1A1fEc
|
||||
a.f('c');
|
||||
}
|
Загрузка…
Ссылка в новой задаче