Tak, odpowiedź zależy od kompilatora.
Szybki eksperyment z moim kompilatorem ( g++ 4.4.3
) ujawnia, że jego biblioteka malloc
uruchomieniowa najpierw próbuje zapamiętać wyjątek i, w przypadku niepowodzenia, próbuje przydzielić miejsce w „buforze awaryjnym” obejmującym cały proces, który żyje w segmencie danych. Jeśli to nie zadziała, dzwoni std::terminate()
.
Wydawałoby się, że głównym celem bufora awaryjnego jest możliwość wyrzucenia std::bad_alloc
po tym, jak procesowi zabraknie miejsca na stercie (w takim przypadku malloc
wywołanie zakończy się niepowodzeniem).
Odpowiednią funkcją jest __cxa_allocate_exception
:
extern "C" void *
__cxxabiv1::__cxa_allocate_exception(std::size_t thrown_size) throw()
{
void *ret;
thrown_size += sizeof (__cxa_refcounted_exception);
ret = malloc (thrown_size);
if (! ret)
{
__gnu_cxx::__scoped_lock sentry(emergency_mutex);
bitmask_type used = emergency_used;
unsigned int which = 0;
if (thrown_size > EMERGENCY_OBJ_SIZE)
goto failed;
while (used & 1)
{
used >>= 1;
if (++which >= EMERGENCY_OBJ_COUNT)
goto failed;
}
emergency_used |= (bitmask_type)1 << which;
ret = &emergency_buffer[which][0];
failed:;
if (!ret)
std::terminate ();
}
__cxa_eh_globals *globals = __cxa_get_globals ();
globals->uncaughtExceptions += 1;
memset (ret, 0, sizeof (__cxa_refcounted_exception));
return (void *)((char *)ret + sizeof (__cxa_refcounted_exception));
}
Nie wiem, jak typowy jest ten schemat.
raise
), zniszcz go. Problem z alokacją sterty polega na tym, co zrobić, jeśli próba przydzielenia wyjątku nie powiedzie się? Musiałbyś przerwać (tak jak w przypadku przepełnienia stosu, jeśli umieszczenie go na stosie „nie powiedzie się” z powodu braku miejsca). W przeciwieństwie do Java implementacja nie może zamiast tego zgłosić innego wyjątku.