2009-06-10 2 views
5

Мне нужно получить информацию о стеке моего приложения C в определенных точках. Я прочитал документацию и обыскал в Сети, но до сих пор не могу понять, как я могу это сделать. Можете ли вы указать на простое объяснение процесса? Или, что еще лучше, к примеру разматывания стека. Мне это нужно для HP-UX (Itanium) и Linux.Stack unwinding на HP-UX и Linux

ответ

4

Заканчивать линукс/stacktrace.h

Вот ссылка на API:

http://www.cs.cmu.edu/afs/cs/Web/People/tekkotsu/dox/StackTrace_8h.html

Должно работать на всех ядрах Linux

Вот альтернативный пример С из

http://www.linuxjournal.com/article/6391

#include <stdio.h> 
#include <signal.h> 
#include <execinfo.h> 

void show_stackframe() { 
    void *trace[16]; 
    char **messages = (char **)NULL; 
    int i, trace_size = 0; 

    trace_size = backtrace(trace, 16); 
    messages = backtrace_symbols(trace, trace_size); 
    printf("[bt] Execution path:\n"); 
    for (i=0; i<trace_size; ++i) 
    printf("[bt] %s\n", messages[i]); 
} 


int func_low(int p1, int p2) { 

    p1 = p1 - p2; 
    show_stackframe(); 

    return 2*p1; 
} 

int func_high(int p1, int p2) { 

    p1 = p1 + p2; 
    show_stackframe(); 

    return 2*p1; 
} 


int test(int p1) { 
    int res; 

    if (p1<10) 
    res = 5+func_low(p1, 2*p1); 
    else 
    res = 5+func_high(p1, 2*p1); 
    return res; 
} 



int main() { 

    printf("First call: %d\n\n", test(27)); 
    printf("Second call: %d\n", test(4)); 

} 
+0

Я не знал, что API существует; как полезно! – Jamie

+0

Не помогает вам HP-UX tho;) – DaveR

+0

@dave, nit-picker: P –

3

Вы хотите посмотреть на libunwind - это кросс-платформенная библиотека разработана первоначально HP для разматывания стека Itanium следы (которые являются особенно сложными); но впоследствии был расширен до многих других платформ; включая как x86-Linux, так и Itanium-HPUX.

Из справочной страницы libunwind (3); Ниже приведен пример использования libunwind написать типичную функцию 'шоу' трассировку:

#define UNW_LOCAL_ONLY 
#include <libunwind.h> 

void show_backtrace (void) { 
    unw_cursor_t cursor; unw_context_t uc; 
    unw_word_t ip, sp; 

    unw_getcontext(&uc); 
    unw_init_local(&cursor, &uc); 
    while (unw_step(&cursor) > 0) { 
    unw_get_reg(&cursor, UNW_REG_IP, &ip); 
    unw_get_reg(&cursor, UNW_REG_SP, &sp); 
    printf ("ip = %lx, sp = %lx\n", (long) ip, (long) sp); 
    } 
} 
Смежные вопросы