[PATCH] D47108: Add -fforce-emit-vtables
Krzysztof Pszeniczny via Phabricator via llvm-commits
llvm-commits at lists.llvm.org
Mon May 21 05:12:30 PDT 2018
amharc requested changes to this revision.
amharc added a comment.
This revision now requires changes to proceed.
This is not sound: sometimes the forcefully emitted vtable is incorrect due to destructor aliasing. This happens e.g. in the Bullet benchmark from the llvm test suite. A simplified example follows.
For a translation unit:
struct Base {
virtual int foo() { return 42; }
virtual ~Base() { }
};
struct Derived : Base {
Derived();
};
int foo() {
Derived *der = new Derived();
return der->foo();
}
it emits:
; Function Attrs: nounwind
declare dso_local void @_ZN7DerivedD1Ev(%struct.Derived*) unnamed_addr #6
; Function Attrs: nounwind
declare dso_local void @_ZN7DerivedD0Ev(%struct.Derived*) unnamed_addr #6
@_ZTV7Derived = linkonce_odr dso_local unnamed_addr constant { [5 x i8*] } { [5 x i8*] [i8* null, i8* bitcast ({ i8*, i8*, i8* }* @_ZTI7Derived to i8*), i8* bitcast (i32 (%struct.Base*)* @_ZN4Base3fooEv to i8*), i8* bitcast (void (%struct.Derived*)* @_ZN7DerivedD1Ev to i8*), i8* bitcast (void (%struct.Derived*)* @_ZN7DerivedD0Ev to i8*)] }, comdat, align 8
whereas for TU:
struct Base {
virtual int foo() { return 42; }
virtual ~Base() { }
};
struct Derived : Base {
Derived();
};
Derived::Derived() { }
which emits the whole vtable (even without this patch):
@_ZTV7Derived = linkonce_odr dso_local unnamed_addr constant { [5 x i8*] } { [5 x i8*] [i8* null, i8* bitcast ({ i8*, i8*, i8* }* @_ZTI7Derived to i8*), i8* bitcast (i32 (%struct.Base*)* @_ZN4Base3fooEv to i8*), i8* bitcast (void (%struct.Base*)* @_ZN4BaseD2Ev to i8*), i8* bitcast (void (%struct.Derived*)* @_ZN7DerivedD0Ev to i8*)] }, comdat, align 8
Please note that the complete object destructor `@_ZN7DerivedD1Ev` has been RAUWed with the base object destructor `@_ZN4BaseD2Ev`.
Since this triggers the RAUW case (as opposed to the Alias case) in `ItaniumCXXABI::emitCXXStructor`, the complete object destructor is **not** emitted at all.
Because `Derived` lacks a key function, the vtable is `linkonce_odr` and any of those definitions can be picked. Hence, this code can stop linking at the whim of the linker. :(
Repository:
rL LLVM
https://reviews.llvm.org/D47108
More information about the llvm-commits
mailing list