This is a second-hand question from an OS development site, but it made me curious since I couldn't find a decent explanation anywhere.
When compiling and linking a free-standing C++ program using gcc, sometimes a linker error like this occurs:
out/kernel.o:(.eh_frame+0x11): undefined reference to `__gxx_personality_v0'
This is apparently because this symbol is defined in libstdc++, which is missing in a free-standing environment. Fixing the problem simply requires defining this symbol somewhere:
void *__gxx_personality_v0;
Which is nice, but I don't like things that just magically work... So the question is, what is the purpose of this symbol?
Exception handling is included in free standing implementations.
The reason of this is that you possibly use
gcc
to compile your code. If you compile with the option-###
you will notice it is missing the linker-option-lstdc++
when it invokes the linker process . Compiling withg++
will include that library, and thus the symbols defined in it.The answers above are correct: it is used in exception handling. The manual for GCC version 6 has more information (which is no longer present in the version 7 manual). The error can arise when linking an external function that - unknown to GCC - throws Java exceptions.
It is used in the stack unwiding tables, which you can see for instance in the assembly output of my answer to another question. As mentioned on that answer, its use is defined by the Itanium C++ ABI, where it is called the Personality Routine.
The reason it "works" by defining it as a global NULL void pointer is probably because nothing is throwing an exception. When something tries to throw an exception, then you will see it misbehave.
Of course, if nothing is using exceptions, you can disable them with
-fno-exceptions
(and if nothing is using RTTI, you can also add-fno-rtti
). If you are using them, you have to (as other answers already noted) link withg++
instead ofgcc
, which will add-lstdc++
for you.For those who mainly came here because they saw the error message but do not entirely understand all the rest like me. This is might not be your solution if you actually want to make a "free-standing C++ program".
What helped me solve all entry point errors was to put in the "libstdc++-6.dll" from your compiler folder like MinGW next to your compiled exe.
This also solved entry point issues from placing a std::string.
It's part of the exception handling. The gcc EH mechanism allows to mix various EH models, and a personality routine is invoked to determine if an exception match, what finalization to invoke, etc. This specific personality routine is for C++ exception handling (as opposed to, say, gcj/Java exception handling).
A quick grep of the
libstd++
code base revealed the following two usages of__gx_personality_v0
:In libsupc++/unwind-cxx.h
In libsupc++/eh_personality.cc
(Note: it's actually a little more complicated than that; there's some conditional compilation that can change some details).
So, as long as your code isn't actually using exception handling, defining the symbol as
void*
won't affect anything, but as soon as it does, you're going to crash -__gxx_personality_v0
is a function, not some global object, so trying to call the function is going to jump to address 0 and cause a segfault.