1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
27 #include <sys/frame.h>
28 #include "backtrace.h"
34 #define FRAME_PTR_OFFSET 1
35 #define FRAME_OFFSET 0
36 #define STACK_BIAS 0x7ff
40 #define FRAME_PTR_OFFSET 1
41 #define FRAME_OFFSET 0
46 #elif defined( INTEL )
48 #define FRAME_PTR_OFFSET 3
49 #define FRAME_OFFSET 0
54 #error Unknown Solaris target platform.
56 #endif /* defined SPARC or INTEL */
59 int backtrace( void **buffer
, int max_frames
)
66 /* flush register windows */
71 /* get stack- and framepointer */
74 fpval
= ((long*)(ctx
))[FRAME_PTR_OFFSET
];
75 fp
= (struct frame
*)((char*)(fpval
) + STACK_BIAS
);
77 for (i
= 0; (i
< FRAME_OFFSET
) && (fp
!= 0); i
++)
78 fp
= (struct frame
*)((char*)(fp
->fr_savfp
) + STACK_BIAS
);
80 /* iterate through backtrace */
81 for (i
= 0; (fp
!= 0) && (fp
->fr_savpc
!= 0) && (i
< max_frames
); i
++)
83 /* saved (prev) frame */
84 struct frame
* prev
= (struct frame
*)((char*)(fp
->fr_savfp
) + STACK_BIAS
);
87 *(buffer
++) = (void*)(fp
->fr_savpc
);
89 /* prev frame (w/ stack growing top down) */
90 fp
= (prev
> fp
) ? prev
: 0;
93 /* return number of frames stored */
97 void backtrace_symbols_fd( void **buffer
, int size
, int fd
)
99 FILE *fp
= fdopen( fd
, "w" );
105 for ( pFramePtr
= buffer
; size
> 0 && pFramePtr
&& *pFramePtr
; pFramePtr
++, size
-- )
110 if ( 0 != dladdr( *pFramePtr
, &dli
) )
112 if ( dli
.dli_fname
&& dli
.dli_fbase
)
114 offset
= (ptrdiff_t)*pFramePtr
- (ptrdiff_t)dli
.dli_fbase
;
115 fprintf( fp
, "%s+0x%x", dli
.dli_fname
, offset
);
117 if ( dli
.dli_sname
&& dli
.dli_saddr
)
119 offset
= (ptrdiff_t)*pFramePtr
- (ptrdiff_t)dli
.dli_saddr
;
120 fprintf( fp
, "(%s+0x%x)", dli
.dli_sname
, offset
);
123 fprintf( fp
, "[0x%x]\n", *pFramePtr
);
131 #endif /* defined SOLARIS */
134 #if defined FREEBSD || defined NETBSD || defined OPENBSD || defined(DRAGONFLY)
140 #include "backtrace.h"
142 #define FRAME_PTR_OFFSET 1
143 #define FRAME_OFFSET 0
145 int backtrace( void **buffer
, int max_frames
)
150 /* get stack- and framepointer */
152 fp
= (struct frame
*)(((size_t*)(ctx
))[FRAME_PTR_OFFSET
]);
153 for ( i
=0; (i
<FRAME_OFFSET
) && (fp
!=0); i
++)
155 /* iterate through backtrace */
156 for (i
=0; fp
&& fp
->fr_savpc
&& i
<max_frames
; i
++)
159 *(buffer
++) = (void *)fp
->fr_savpc
;
166 void backtrace_symbols_fd( void **buffer
, int size
, int fd
)
168 FILE *fp
= fdopen( fd
, "w" );
173 for ( pFramePtr
= buffer
; size
> 0 && pFramePtr
&& *pFramePtr
; pFramePtr
++, size
-- )
178 if ( 0 != dladdr( *pFramePtr
, &dli
) )
180 if ( dli
.dli_fname
&& dli
.dli_fbase
)
182 offset
= (ptrdiff_t)*pFramePtr
- (ptrdiff_t)dli
.dli_fbase
;
183 fprintf( fp
, "%s+0x%x", dli
.dli_fname
, offset
);
185 if ( dli
.dli_sname
&& dli
.dli_saddr
)
187 offset
= (ptrdiff_t)*pFramePtr
- (ptrdiff_t)dli
.dli_saddr
;
188 fprintf( fp
, "(%s+0x%x)", dli
.dli_sname
, offset
);
191 fprintf( fp
, "[0x%x]\n", *pFramePtr
);
197 #endif /* defined FREEBSD */
205 #endif /* defined LINUX */
207 #if defined( MACOSX )
211 #include "backtrace.h"
213 typedef unsigned ptrdiff_t;
215 /* glib backtrace is only available on MacOsX 10.5 or higher
216 so we do it on our own */
218 int backtrace( void **buffer
, int max_frames
)
220 void **frame
= (void **)__builtin_frame_address(0);
221 void **bp
= ( void **)(*frame
);
225 for ( i
= 0; bp
&& ip
&& i
< max_frames
; i
++ )
230 bp
= (void**)(bp
[0]);
237 void backtrace_symbols_fd( void **buffer
, int size
, int fd
)
239 FILE *fp
= fdopen( fd
, "w" );
245 for ( pFramePtr
= buffer
; size
> 0 && pFramePtr
&& *pFramePtr
; pFramePtr
++, size
-- )
249 if ( 0 != dladdr( *pFramePtr
, &dli
) )
253 if ( dli
.dli_fname
&& dli
.dli_fbase
)
255 offset
= (ptrdiff_t)*pFramePtr
- (ptrdiff_t)dli
.dli_fbase
;
256 fprintf( fp
, "%s+0x%x", dli
.dli_fname
, offset
);
258 if ( dli
.dli_sname
&& dli
.dli_saddr
)
260 offset
= (ptrdiff_t)*pFramePtr
- (ptrdiff_t)dli
.dli_saddr
;
261 fprintf( fp
, "(%s+0x%x)", dli
.dli_sname
, offset
);
264 fprintf( fp
, "[0x%x]\n", (unsigned int)*pFramePtr
);
272 #endif /* defined MACOSX */
275 int backtrace( void **buffer
, int max_frames
)
280 void backtrace_symbols_fd( void **buffer
, int size
, int fd
)
285 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */