https://gcc.gnu.org/bugzilla/show_bug.cgi?id=64670
--- Comment #9 from Tobias Burnus <burnus at gcc dot gnu.org> --- (In reply to Jason Merrill from comment #8) > (In reply to Tobias Burnus from comment #5) > > Updated test case (part 1/2): [main file] compile with g++ -fsanitize=vptr > so the #pragma implementation doesn't match. If I change it to be "test2.h" > I get the typeinfo for LogListener. I have to admit that I am lost - is there a good explanation of #pragma implementation/interface somewhere? In the real code, one has something like the following, which I tried to mimic in the test case. If I understood your comment correctly, I should use in "log.c" not only 'implementation "log.h"' but additionally(?) 'implementation "LogListener.h"' - i.e. using 'pragma implementation' for LogListener.h's classes only in LogListener.h is not sufficient? ----- LogListener.h ------ // Complete file except for "..." bits and #include lines. #pragma interface class LogListener { public: virtual ~LogListener() {} virtual void listen( const Cstring& input) = 0; // virtual ... = 0; // some more } class AccumulatingLogListener : public LogListener {...}; ----- LogListener.cc ------ // matches approx. attachment 34628 // As it is short, that's really the complete file #pragma implementation #include "LogListener.h" void AccumulatingLogListener::listen( const Cstring& input) { m_buffer += input; } ----- log.h ------ #pragma interface class LogListener; class Log_Handler {....}; ----- log.c ------ // matches approx. attachment 34627 #pragma implementation "log.h" #include "log.h" #include "LogListener.h" // implementation of Log_Handler, which causes -fsanitize=vptr to access // "typeinfo for name LogListener", which is "U". ---------------------------------------------------- The real-world LogListener.o has (full output as it is short): 000000000000006d t _GLOBAL__sub_D_00099_0__ZN23AccumulatingLogListener6listenERK7Cstring 0000000000000082 t _GLOBAL__sub_I_00099_1__ZN23AccumulatingLogListener6listenERK7Cstring 0000000000000000 W LogListener::LogListener() 0000000000000000 W LogListener::LogListener() 0000000000000000 W LogListener::~LogListener() 0000000000000000 W LogListener::~LogListener() 0000000000000000 W LogListener::~LogListener() 0000000000000000 T AccumulatingLogListener::listen(Cstring const&) 0000000000000000 W AccumulatingLogListener::AccumulatingLogListener(int) 0000000000000000 W AccumulatingLogListener::AccumulatingLogListener(int) 0000000000000000 W AccumulatingLogListener::~AccumulatingLogListener() 0000000000000000 W AccumulatingLogListener::~AccumulatingLogListener() 0000000000000000 W AccumulatingLogListener::~AccumulatingLogListener() 0000000000000000 W AccumulatingLogListener::getContent() const 0000000000000000 W AccumulatingLogListener::eatsMessages() const 0000000000000000 W AccumulatingLogListener::isAccumulatingLogListerner() const 0000000000000000 V typeinfo for LogListener 0000000000000000 V typeinfo for AccumulatingLogListener 0000000000000000 V typeinfo name for LogListener 0000000000000000 V typeinfo name for AccumulatingLogListener 0000000000000000 V vtable for LogListener 0000000000000000 V vtable for AccumulatingLogListener