|  | /* Return backtrace of current program state.  Generic version. | 
|  | Copyright (C) 1998-2016 Free Software Foundation, Inc. | 
|  | This file is part of the GNU C Library. | 
|  | Contributed by Ulrich Drepper <drepper@cygnus.com>, 1998. | 
|  |  | 
|  | The GNU C Library is free software; you can redistribute it and/or | 
|  | modify it under the terms of the GNU Lesser General Public | 
|  | License as published by the Free Software Foundation; either | 
|  | version 2.1 of the License, or (at your option) any later version. | 
|  |  | 
|  | The GNU C Library is distributed in the hope that it will be useful, | 
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
|  | Lesser General Public License for more details. | 
|  |  | 
|  | You should have received a copy of the GNU Lesser General Public | 
|  | License along with the GNU C Library; if not, see | 
|  | <http://www.gnu.org/licenses/>.  */ | 
|  |  | 
|  | #include <execinfo.h> | 
|  | #include <signal.h> | 
|  | #include <frame.h> | 
|  | #include <sigcontextinfo.h> | 
|  | #include <ldsodefs.h> | 
|  |  | 
|  | /* This implementation assumes a stack layout that matches the defaults | 
|  | used by gcc's `__builtin_frame_address' and `__builtin_return_address' | 
|  | (FP is the frame pointer register): | 
|  |  | 
|  | +-----------------+     +-----------------+ | 
|  | FP -> | previous FP --------> | previous FP ------>... | 
|  | |                 |     |                 | | 
|  | | return address  |     | return address  | | 
|  | +-----------------+     +-----------------+ | 
|  |  | 
|  | */ | 
|  |  | 
|  | /* Get some notion of the current stack.  Need not be exactly the top | 
|  | of the stack, just something somewhere in the current frame.  */ | 
|  | #ifndef CURRENT_STACK_FRAME | 
|  | # define CURRENT_STACK_FRAME  ({ char __csf; &__csf; }) | 
|  | #endif | 
|  |  | 
|  | /* By default we assume that the stack grows downward.  */ | 
|  | #ifndef INNER_THAN | 
|  | # define INNER_THAN < | 
|  | #endif | 
|  |  | 
|  | /* By default assume the `next' pointer in struct layout points to the | 
|  | next struct layout.  */ | 
|  | #ifndef ADVANCE_STACK_FRAME | 
|  | # define ADVANCE_STACK_FRAME(next) ((struct layout *) (next)) | 
|  | #endif | 
|  |  | 
|  | /* By default, the frame pointer is just what we get from gcc.  */ | 
|  | #ifndef FIRST_FRAME_POINTER | 
|  | # define FIRST_FRAME_POINTER  __builtin_frame_address (0) | 
|  | #endif | 
|  |  | 
|  | int | 
|  | __backtrace (void **array, int size) | 
|  | { | 
|  | struct layout *current; | 
|  | void *top_frame; | 
|  | void *top_stack; | 
|  | int cnt = 0; | 
|  |  | 
|  | top_frame = FIRST_FRAME_POINTER; | 
|  | top_stack = CURRENT_STACK_FRAME; | 
|  |  | 
|  | /* We skip the call to this function, it makes no sense to record it.  */ | 
|  | current = ((struct layout *) top_frame); | 
|  | while (cnt < size) | 
|  | { | 
|  | if ((void *) current INNER_THAN top_stack | 
|  | || !((void *) current INNER_THAN __libc_stack_end)) | 
|  | /* This means the address is out of range.  Note that for the | 
|  | toplevel we see a frame pointer with value NULL which clearly is | 
|  | out of range.  */ | 
|  | break; | 
|  |  | 
|  | array[cnt++] = current->return_address; | 
|  |  | 
|  | current = ADVANCE_STACK_FRAME (current->next); | 
|  | } | 
|  |  | 
|  | return cnt; | 
|  | } | 
|  | weak_alias (__backtrace, backtrace) | 
|  | libc_hidden_def (__backtrace) |