387 / 151 / 16
Регистрация: 12.05.2011
Сообщений: 450
|
|||||||||||||||||||||
1 | |||||||||||||||||||||
проблемы с созданием "правильного" core dump'а01.01.2012, 13:42. Показов 3395. Ответов 8
Метки нет (Все метки)
Суть проблемы
При попытке создать с помощью gdb дамп памяти программы, которая в этот момент выполняет системный вызов, я получаю весьма бесполезный core, в котором, насколько я понимаю, не содержится информации, относящейся к моей программе, а содержится лишь часть, относящаяся к системному вызову. Не по теме: Вообще, я в матчасти не очень силен и, возможно, порю несусветную чушь. Потому ниже привожу подробное описание того, что я делаю, что я получаю и чем я не доволен. подробное описание под катом
Берем простенькую программку, которая будет засыпать после выполнения system("sleep 1000").
Теперь откроем сохраненный дамп и поглядим еще раз.
Толку от такого дампа я не вижу и меня это огорчает. Вопросы
PostScriptum
Если в приведенной выше программе модифицировать func1 (убрать системный вызов):
1
|
01.01.2012, 13:42 | |
Ответы с готовыми решениями:
8
Проблемы с созданием объектов Проблемы с созданием объекта Проблемы с созданием Ln логарифма Проблемы с созданием RAID-5 |
01.01.2012, 14:32 | 2 |
Сообщение было отмечено как решение
Решение
Для порядку ещё сделай оригинальную (общую) постановку задачи. Что конкретно ты хочешь сделать?
Добавлено через 7 минут Сделел немного на своём примере, но приниципальной сути не изменилось. После загрузки коры получил почти нормальный стек: Код
(gdb) bt #0 0x00db7422 in __kernel_vsyscall () #1 0x008e1cc0 in ?? () #2 0x008e1b10 in ?? () #3 0xbfa9b284 in ?? () #4 0x080483f5 in func (x=100) at t.c:5 #5 0x0804840c in main () at t.c:10 Код
(gdb) bt #0 0x00db7422 in __kernel_vsyscall () #1 0x008e1cc0 in nanosleep () from /lib/tls/i686/cmov/libc.so.6 #2 0x008e1b10 in sleep () from /lib/tls/i686/cmov/libc.so.6 #3 0x080483f5 in func (x=100) at t.c:5 #4 0x0804840c in main () at t.c:10 Вопросительные знаки на месте функций из динамических библиотек - это нормальное явление и вот почему. 1. Если запустить программу непосредственно из gdb, то gdb умеет перехватывать ситуации загрузки динамической библиотеки в память в момент запуска отлаживаемого приложения. И поэтому при печати стека gdb знает, какие адреса, соответствующие динамическим библиотекам, к какой конкретно библиотеке относятся (и из этого уже может вычислить конкретную функцию динамической библиотеки). 2. Если gdb приаттачивается к запущенному процессу, то я не знаю, откуда он выцепляет информацию о загруженных динамических библиотеках, но технически я в этом проблем не вижу: в каталоге /proc имеется некоторая дополнительная инфа о работающем процессе, из которой в совокупности с данными динамического загрузчика что-то и как-то можно расковырять. 3. Когда gdb поднимает файл core, то информацию взять уже неоткуда. Файл core содержит лишь образ памяти, занимаемый отлаживаемым процессом. А вся (или частично) информация про динамические библиотки берётся извне по отношению к отлаживаемому процессу: в пункте 1 это делается непосредственно через специальный hook (реализованый в ld.so именно для работы отладчика), в пункте 2 это делается на основании информации из /proc плюс какие-то специальные данные в ld.so (тоже сделанные для работы отладчика), но данные в /proc имеются только в том случае, если процесс жив. Когда процесса уже нет в живых, то эти данные уже утеряны. В итоге в core содержится только образ памяти, в котором нет информации о том, где какая динамическая библиотека. Информация же о самом бинарнике берётся из файла бинарника (т.к. core исследуется не самостоятельно, а в контексте бинарника, который параллельно с core'ой подаётся в отладчик)
3
|
387 / 151 / 16
Регистрация: 12.05.2011
Сообщений: 450
|
|
01.01.2012, 14:37 [ТС] | 3 |
озадачился я этими изысканиями по мотивам твоей темы Встроенная в программу печать стека при помощи внешнего отладчика (спасибо за весьма любопытную статью, кстати).
Идея в том, чтобы помимо распечатки стека бектрейсом еще и автоматически создавать core dump, который дальше можно было бы при необходимости использовать для более детального анализа. (В обсуждении к вышеупомянутой теме я описывал возникшую у меня проблему, но тот пост остался не замечен) Поскольку там основная программа на время работы внешнего отладчика также приостанавливается системным вызовом wait, я тоже получаю неинформативный дамп.
0
|
01.01.2012, 14:46 | 4 | ||||||||||
Повторил с точностью до буквы твой вариант - увидел такую же засаду как и у тебя. Странно. Надо думать
Добавлено через 3 минуты Если в твоём коде заменить
Добавлено через 2 минуты Это такая проблема настройки форумного движка. Твой второй пост просто склеился с первым и такие вещи зачастую оказываются незамеченными в случаях, когда в этот момент другие перечитывают страницу форума
0
|
387 / 151 / 16
Регистрация: 12.05.2011
Сообщений: 450
|
|
01.01.2012, 14:52 [ТС] | 5 |
0
|
01.01.2012, 16:44 | 6 |
У меня стек нарисовался такой:
Код
(gdb) bt #0 0x005bf422 in __kernel_vsyscall () #1 0x006e572d in ?? () #2 0x004ac3c0 in ?? () from /lib/ld-linux.so.2 #3 0x08048a61 in gdb_PrintStackGDB () at t.c:67 Backtrace stopped: frame did not save the PC Добавлено через 8 минут Этот же пример на sparc-linux отрабатывает корректно: Код
(gdb) core core.from.gdb Reading symbols from /usr/lib/libstdc++.so.6...done. Loaded symbols for /usr/lib/libstdc++.so.6 Reading symbols from /lib/libm.so.6...done. Loaded symbols for /lib/libm.so.6 Reading symbols from /lib/libgcc_s.so.1...done. Loaded symbols for /lib/libgcc_s.so.1 Reading symbols from /lib/libc.so.6...done. Loaded symbols for /lib/libc.so.6 Reading symbols from /lib/ld-linux.so.2...done. Loaded symbols for /lib/ld-linux.so.2 Core was generated by `/home/<user>/a.out'. [New process 20095] #0 0xf7cbcf04 in wait () from /lib/libc.so.6 (gdb) bt #0 0xf7cbcf04 in wait () from /lib/libc.so.6 #1 0x00010be8 in gdb_PrintStackGDB () at t.c:67 #2 0x00010e0c in func1 (x=0xffb7d8cc) at t.c:95 #3 0x00010e54 in func2 (j=Cannot access memory at address 0x7 ) at t.c:102 #4 0x00010eb0 in main (argc=1, argv=0xffb7d9d4) at t.c:117 Добавлено через 3 минуты Есть некое предположение того, что gdb в случае i386 скинул образ памяти не основного процесса, а того, что получился после fork'а (т.е. ошибка gdb). Но как экспериментально подтвердить или опровергнуть - не знаю Добавлено через 13 минут В случае с i386 пока нашёл лекарство-затычку. Если линковать статически, то после поднятия core стек рисуется нормальный
1
|
387 / 151 / 16
Регистрация: 12.05.2011
Сообщений: 450
|
|
22.01.2012, 18:56 [ТС] | 7 |
проблема чудесным образом разрешилась при апгрейде компилятора (gcc) с версии 4.4.3 до версии 4.6.2
0
|
02.10.2012, 15:14 | 9 |
Для истории: https://www.cyberforum.ru/faq/... ost3506089
Т.е. проблема скорее всего была в том, что динамическая glibc собрана с дополнительной опцией -fomit-frame-pointer (или что-то типа того), а потому становится невозможна полноценная работа размотки стека. А статическая библиотека собрана без этой опции
0
|
02.10.2012, 15:14 | |
02.10.2012, 15:14 | |
Помогаю со студенческими работами здесь
9
Проблемы с созданием вектора Проблемы с созданием словаря на C++ Проблемы с созданием сервиса Проблемы с созданием процедуры Проблемы с созданием процедуры Проблемы с созданием формы Проблемы с созданием класса Искать еще темы с ответами Или воспользуйтесь поиском по форуму: |