[Comm] Знатокам gcc (3.2, C++)

Sergey Vlasov =?iso-8859-1?q?vsu_=CE=C1_altlinux=2Eru?=
Пн Окт 28 17:04:05 MSK 2002


On Mon, Oct 28, 2002 at 19:50:52 +0600, Alexey Morozov wrote:
> В задаче имеется:
> 1. библиотека, написанная на C (libpopt)
> 2. код на C++, ее использующий
> 
> Постановка задачи:
> 
> имеется C'шная функция, принимающая в качестве аргумента коллбэк и дергающая
> его (при определенных условиях):
> /* ------------------- test_lib.c ------------ */
> void some_func(void (*callback_func)(void))
> {
>    callback_func();
> }
> /* ------------------ end of test_lib.c ---------- */
> 
> С++-ный код коллбэка и функции, инициирующей вызов some_func:
> /* ------------------- test.cc --------------- */
> #include <iostream>
> using namespace std;
> struct A {};
> void callback_func(void)
> {
>     throw A();
> }
> 
> extern "C" {
>     void some_func(void (*callback)(void));
> }
> int main(void)
> {
>    try {
>        some_func(callback_func);
>    } catch (...) {
>        cerr << "An exception caught" << endl;
>    }
> }
> /* ----------------- end of test.cc --------------*/
> 
> Проблема: происходит аварийное завершение указанной программы (abort),
> как если бы брошенный exception не соответствовал throw-спецификации.
> 
> Вопрос: как бы это обойти, и таки поймать в main брошенный в коллбэке
> эксепшн?
> 
> Повторю, все происходит на gcc-3.2.1, менять библиотечную функцию по
> условию задачи нельзя.

Похоже, никак. Новый gcc производит разбор стека при поиске обработчика по
записям таблицы, генерируемой при компиляции (по адресам возврата). Для
кода на C эта таблица не генерируется, поэтому при обнаружении в цепочке
вызовов адреса из такого участка кода происходит облом (как раз в виде
вызова unexpected()).

Варианты:

1) Избавиться в этом месте от exception вообще и работать старым дедовским
способом - по кодам возврата.  Если что-то бросается из глубины - ловить
внутри callback-a, не допуская распространения в код на C.

2) info gcc рекомендует в подобных случаях компилировать код на C с опцией
-fexceptions - тогда работает (но, разумеется, память, выделенная в
C-коде, не освобождается - будет утечка).

3) Можно задействовать setjmp/longjmp - но тогда утечка будет уже в
C++-частях.



Подробная информация о списке рассылки community