Мы запускаем uclibc Linux на ARM 9. Проблемой является uclibc, не поддерживает след. Когда дамп ядра происходит, я не могу захватить стек вызовов.
У кого-либо есть хорошее решение для этого?
Например, существующее портирование следа для uclibc или какой-либо хороший метод захватить стек вызовов, когда дамп ядра происходит (uclibc+ARM+Linux)?
Обновление:
Похоже, что патч был создан для поддержки backtrace ()
в uclibc для x86 и ARM (XScale), и он использует Символ __libc_stack_end
.
Исходный ответ:
Я работал над проектом, в котором версия glibc, которую мы использовали, не предоставляла функциональную backtrace ()
для нашего процессора ARM, поэтому мы разработали свой собственный вне glibc, используя символ __ libc_stack_end
. Ниже приведен получившийся код. Возможно, вы сможете использовать его для написания функции uclibc backtrace ()
.
extern void * __libc_stack_end;
struct backtrace_frame_t
{
void * fp;
void * sp;
void * lr;
void * pc;
};
int backtrace(void ** array, int size)
{
void * top_frame_p;
void * current_frame_p;
struct backtrace_frame_t * frame_p;
int frame_count;
top_frame_p = __builtin_frame_address(0);
current_frame_p = top_frame_p;
frame_p = (struct backtrace_frame_t*)((void**)(current_frame_p)-3);
frame_count = 0;
if (__builtin_return_address(0) != frame_p->lr)
{
fprintf(stderr, "backtrace error: __builtin_return_address(0) != frame_p->lr\n");
return frame_count;
}
if (current_frame_p != NULL
&& current_frame_p > (void*)&frame_count
&& current_frame_p < __libc_stack_end)
{
while (frame_count < size
&& current_frame_p != NULL
&& current_frame_p > (void*)&frame_count
&& current_frame_p < __libc_stack_end)
{
frame_p = (struct backtrace_frame_t*)((void**)(current_frame_p)-3);
array[frame_count] = frame_p->lr;
frame_count++;
current_frame_p = frame_p->fp;
}
}
return frame_count;
}
Примечание: символ __ libc_stack_end
больше не экспортируется в более поздних версиях glibc, и я не уверен в существовании этого или подобного символа в uclibc.