diff options
author | David Majnemer <david.majnemer@gmail.com> | 2014-06-20 21:11:00 +0000 |
---|---|---|
committer | David Majnemer <david.majnemer@gmail.com> | 2014-06-20 21:11:00 +0000 |
commit | 5d229785a325e5f4533dd90494f74a278f29927b (patch) | |
tree | cfa936dfaa2a51aa6edd38ef198aa0a9ff8d7b9f /test/CodeGenCXX/microsoft-abi-typeid.cpp | |
parent | 76981cc33cb8d6477a9c65a6104725e3047d05b9 (diff) |
CodeGen: Refactor dynamic_cast and typeid
This refactors the emission of dynamic_cast and typeid expressions so
that ABI specific knowledge lives in appropriate places. There are
quite a few benefits for having the two implementations share a common
core like sharing logic for optimization opportunities.
While we are at it, clean up the tests.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@211402 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'test/CodeGenCXX/microsoft-abi-typeid.cpp')
-rw-r--r-- | test/CodeGenCXX/microsoft-abi-typeid.cpp | 74 |
1 files changed, 32 insertions, 42 deletions
diff --git a/test/CodeGenCXX/microsoft-abi-typeid.cpp b/test/CodeGenCXX/microsoft-abi-typeid.cpp index ffbebec1fa..b8074baa62 100644 --- a/test/CodeGenCXX/microsoft-abi-typeid.cpp +++ b/test/CodeGenCXX/microsoft-abi-typeid.cpp @@ -1,58 +1,48 @@ -// RUN: %clang_cc1 -emit-llvm -O2 -optzns -o - -triple=i386-pc-win32 2>/dev/null %s | FileCheck %s +// RUN: %clang_cc1 -emit-llvm -O1 -o - -triple=i386-pc-win32 %s | FileCheck %s // REQUIRES: asserts -struct type_info { const char* raw_name() const; }; +struct type_info; namespace std { using ::type_info; } -struct V { virtual void f() {}; }; -struct A : virtual V {}; +struct V { virtual void f(); }; +struct A : virtual V { A(); }; -A a; -int b; +extern A a; +extern int b; A* fn(); const std::type_info* test0_typeid() { return &typeid(int); } -// CHECK: define %struct.type_info* @"\01?test0_typeid@@YAPBUtype_info@@XZ"() #0 { -// CHECK-NEXT: entry: -// CHECK-NEXT: ret %struct.type_info* bitcast (%"MSRTTITypeDescriptor\02"* @"\01??_R0H@8" to %struct.type_info*) -// CHECK-NEXT: } +// CHECK-LABEL: define %struct.type_info* @"\01?test0_typeid@@YAPBUtype_info@@XZ"() +// CHECK: ret %struct.type_info* bitcast (%"MSRTTITypeDescriptor\02"* @"\01??_R0H@8" to %struct.type_info*) const std::type_info* test1_typeid() { return &typeid(A); } -// CHECK: define %struct.type_info* @"\01?test1_typeid@@YAPBUtype_info@@XZ"() #0 { -// CHECK-NEXT: entry: -// CHECK-NEXT: ret %struct.type_info* bitcast (%"MSRTTITypeDescriptor\07"* @"\01??_R0?AUA@@@8" to %struct.type_info*) -// CHECK-NEXT: } +// CHECK-LABEL: define %struct.type_info* @"\01?test1_typeid@@YAPBUtype_info@@XZ"() +// CHECK: ret %struct.type_info* bitcast (%"MSRTTITypeDescriptor\07"* @"\01??_R0?AUA@@@8" to %struct.type_info*) const std::type_info* test2_typeid() { return &typeid(&a); } -// CHECK: define %struct.type_info* @"\01?test2_typeid@@YAPBUtype_info@@XZ"() #0 { -// CHECK-NEXT: entry: -// CHECK-NEXT: ret %struct.type_info* bitcast (%"MSRTTITypeDescriptor\07"* @"\01??_R0PAUA@@@8" to %struct.type_info*) -// CHECK-NEXT: } +// CHECK-LABEL: define %struct.type_info* @"\01?test2_typeid@@YAPBUtype_info@@XZ"() +// CHECK: ret %struct.type_info* bitcast (%"MSRTTITypeDescriptor\07"* @"\01??_R0PAUA@@@8" to %struct.type_info*) const std::type_info* test3_typeid() { return &typeid(*fn()); } -// CHECK: define %struct.type_info* @"\01?test3_typeid@@YAPBUtype_info@@XZ"() #1 { -// CHECK-NEXT: entry: -// CHECK-NEXT: %call = tail call %struct.A* @"\01?fn@@YAPAUA@@XZ"() #3 -// CHECK-NEXT: %0 = icmp eq %struct.A* %call, null -// CHECK-NEXT: br i1 %0, label %type_id.call, label %type_id.valid -// CHECK: type_id.valid: ; preds = %entry -// CHECK-NEXT: %1 = bitcast %struct.A* %call to i8* -// CHECK-NEXT: %2 = bitcast %struct.A* %call to i8** -// CHECK-NEXT: %vbtable = load i8** %2, align 4 -// CHECK-NEXT: %3 = getelementptr inbounds i8* %vbtable, i32 4 -// CHECK-NEXT: %4 = bitcast i8* %3 to i32* -// CHECK-NEXT: %vbase_offs = load i32* %4, align 4 -// CHECK-NEXT: %5 = getelementptr inbounds i8* %1, i32 %vbase_offs -// CHECK-NEXT: br label %type_id.call -// CHECK: type_id.call: ; preds = %type_id.valid, %entry -// CHECK-NEXT: %6 = phi i8* [ %5, %type_id.valid ], [ null, %entry ] -// CHECK-NEXT: %7 = tail call i8* @__RTtypeid(i8* %6) #3 -// CHECK-NEXT: %8 = bitcast i8* %7 to %struct.type_info* -// CHECK-NEXT: ret %struct.type_info* %8 -// CHECK-NEXT: } +// CHECK-LABEL: define %struct.type_info* @"\01?test3_typeid@@YAPBUtype_info@@XZ"() +// CHECK: [[CALL:%.*]] = tail call %struct.A* @"\01?fn@@YAPAUA@@XZ"() +// CHECK-NEXT: [[CMP:%.*]] = icmp eq %struct.A* [[CALL]], null +// CHECK-NEXT: br i1 [[CMP]], label %typeid.bad_typeid, label %typeid.end +// CHECK-LABEL: typeid.bad_typeid: +// CHECK-NEXT: tail call i8* @__RTtypeid(i8* null) +// CHECK-NEXT: unreachable +// CHECK-LABEL: typeid.end: +// CHECK-NEXT: [[THIS:%.*]] = bitcast %struct.A* [[CALL]] to i8* +// CHECK-NEXT: [[VBTBLP:%.*]] = bitcast %struct.A* [[CALL]] to i8** +// CHECK-NEXT: [[VBTBL:%.*]] = load i8** [[VBTBLP]], align 4 +// CHECK-NEXT: [[VBSLOT:%.*]] = getelementptr inbounds i8* [[VBTBL]], i32 4 +// CHECK-NEXT: [[VBITCST:%.*]] = bitcast i8* [[VBSLOT]] to i32* +// CHECK-NEXT: [[VBASE_OFFS:%.*]] = load i32* [[VBITCST]], align 4 +// CHECK-NEXT: [[ADJ:%.*]] = getelementptr inbounds i8* [[THIS]], i32 [[VBASE_OFFS]] +// CHECK-NEXT: [[RT:%.*]] = tail call i8* @__RTtypeid(i8* [[ADJ]]) +// CHECK-NEXT: [[RET:%.*]] = bitcast i8* [[RT]] to %struct.type_info* +// CHECK-NEXT: ret %struct.type_info* [[RET]] const std::type_info* test4_typeid() { return &typeid(b); } -// CHECK: define %struct.type_info* @"\01?test4_typeid@@YAPBUtype_info@@XZ"() #0 { -// CHECK-NEXT: entry: -// CHECK-NEXT: ret %struct.type_info* bitcast (%"MSRTTITypeDescriptor\02"* @"\01??_R0H@8" to %struct.type_info*) -// CHECK-NEXT: } +// CHECK: define %struct.type_info* @"\01?test4_typeid@@YAPBUtype_info@@XZ"() +// CHECK: ret %struct.type_info* bitcast (%"MSRTTITypeDescriptor\02"* @"\01??_R0H@8" to %struct.type_info*) |