1 /* Subroutines needed for unwinding stack frames for exception handling. */
2 /* Copyright (C) 1997, 1998, 1999, 2000, 2001 Free Software Foundation, Inc.
3 Contributed by Jason Merrill <jason@cygnus.com>.
5 This file is part of GNU CC.
7 GNU CC is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 2, or (at your option)
12 In addition to the permissions in the GNU General Public License, the
13 Free Software Foundation gives you unlimited permission to link the
14 compiled version of this file into combinations with other programs,
15 and to distribute those combinations without any restriction coming
16 from the use of this file. (The General Public License restrictions
17 do apply in other respects; for example, they cover modification of
18 the file, and distribution when not linked into a combine
21 GNU CC is distributed in the hope that it will be useful,
22 but WITHOUT ANY WARRANTY; without even the implied warranty of
23 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
24 GNU General Public License for more details.
26 You should have received a copy of the GNU General Public License
27 along with GNU CC; see the file COPYING. If not, write to
28 the Free Software Foundation, 59 Temple Place - Suite 330,
29 Boston, MA 02111-1307, USA. */
33 #include "unwind-dw2-fde.h"
36 static struct object
*objects
;
38 #ifdef __GTHREAD_MUTEX_INIT
39 static __gthread_mutex_t object_mutex
= __GTHREAD_MUTEX_INIT
;
41 static __gthread_mutex_t object_mutex
;
44 #ifdef __GTHREAD_MUTEX_INIT_FUNCTION
46 init_object_mutex (void)
48 __GTHREAD_MUTEX_INIT_FUNCTION (&object_mutex
);
52 init_object_mutex_once (void)
54 static __gthread_once_t once
= __GTHREAD_ONCE_INIT
;
55 __gthread_once (&once
, init_object_mutex
);
58 #define init_object_mutex_once()
61 /* Called from crtbegin.o to register the unwind info for an object. */
64 __register_frame_info (void *begin
, struct object
*ob
)
66 ob
->pc_begin
= ob
->pc_end
= 0;
67 ob
->fde_begin
= begin
;
71 init_object_mutex_once ();
72 __gthread_mutex_lock (&object_mutex
);
77 __gthread_mutex_unlock (&object_mutex
);
81 __register_frame (void *begin
)
83 struct object
*ob
= (struct object
*) malloc (sizeof (struct object
));
84 __register_frame_info (begin
, ob
);
87 /* Similar, but BEGIN is actually a pointer to a table of unwind entries
88 for different translation units. Called from the file generated by
92 __register_frame_info_table (void *begin
, struct object
*ob
)
94 ob
->pc_begin
= ob
->pc_end
= 0;
95 ob
->fde_begin
= begin
;
96 ob
->fde_array
= begin
;
99 init_object_mutex_once ();
100 __gthread_mutex_lock (&object_mutex
);
105 __gthread_mutex_unlock (&object_mutex
);
109 __register_frame_table (void *begin
)
111 struct object
*ob
= (struct object
*) malloc (sizeof (struct object
));
112 __register_frame_info_table (begin
, ob
);
115 /* Called from crtbegin.o to deregister the unwind info for an object. */
118 __deregister_frame_info (void *begin
)
122 init_object_mutex_once ();
123 __gthread_mutex_lock (&object_mutex
);
128 if ((*p
)->fde_begin
== begin
)
130 struct object
*ob
= *p
;
133 /* If we've run init_frame for this object, free the FDE array. */
134 if (ob
->fde_array
&& ob
->fde_array
!= begin
)
135 free (ob
->fde_array
);
137 __gthread_mutex_unlock (&object_mutex
);
143 __gthread_mutex_unlock (&object_mutex
);
148 __deregister_frame (void *begin
)
150 free (__deregister_frame_info (begin
));
154 /* Sorting an array of FDEs by address.
155 (Ideally we would have the linker sort the FDEs so we don't have to do
156 it at run time. But the linkers are not yet prepared for this.) */
158 /* This is a special mix of insertion sort and heap sort, optimized for
159 the data sets that actually occur. They look like
160 101 102 103 127 128 105 108 110 190 111 115 119 125 160 126 129 130.
161 I.e. a linearly increasing sequence (coming from functions in the text
162 section), with additionally a few unordered elements (coming from functions
163 in gnu_linkonce sections) whose values are higher than the values in the
164 surrounding linear sequence (but not necessarily higher than the values
165 at the end of the linear sequence!).
166 The worst-case total run time is O(N) + O(n log (n)), where N is the
167 total number of FDEs and n is the number of erratic ones. */
169 typedef struct fde_vector
175 typedef struct fde_accumulator
182 fde_compare (fde
*x
, fde
*y
)
184 return (saddr
)x
->pc_begin
- (saddr
)y
->pc_begin
;
188 start_fde_sort (fde_accumulator
*accu
, size_t count
)
190 accu
->linear
.array
= count
? (fde
**) malloc (sizeof (fde
*) * count
) : NULL
;
191 accu
->erratic
.array
= accu
->linear
.array
?
192 (fde
**) malloc (sizeof (fde
*) * count
) : NULL
;
193 accu
->linear
.count
= 0;
194 accu
->erratic
.count
= 0;
196 return accu
->linear
.array
!= NULL
;
200 fde_insert (fde_accumulator
*accu
, fde
*this_fde
)
202 if (accu
->linear
.array
)
203 accu
->linear
.array
[accu
->linear
.count
++] = this_fde
;
206 /* Split LINEAR into a linear sequence with low values and an erratic
207 sequence with high values, put the linear one (of longest possible
208 length) into LINEAR and the erratic one into ERRATIC. This is O(N).
210 Because the longest linear sequence we are trying to locate within the
211 incoming LINEAR array can be interspersed with (high valued) erratic
212 entries. We construct a chain indicating the sequenced entries.
213 To avoid having to allocate this chain, we overlay it onto the space of
214 the ERRATIC array during construction. A final pass iterates over the
215 chain to determine what should be placed in the ERRATIC array, and
216 what is the linear sequence. This overlay is safe from aliasing. */
218 fde_split (fde_vector
*linear
, fde_vector
*erratic
)
221 size_t count
= linear
->count
;
222 fde
**chain_end
= &marker
;
225 /* This should optimize out, but it is wise to make sure this assumption
226 is correct. Should these have different sizes, we cannot cast between
227 them and the overlaying onto ERRATIC will not work. */
228 if (sizeof (fde
*) != sizeof (fde
**))
231 for (i
= 0; i
< count
; i
++)
235 for (probe
= chain_end
;
236 probe
!= &marker
&& fde_compare (linear
->array
[i
], *probe
) < 0;
239 chain_end
= (fde
**)erratic
->array
[probe
- linear
->array
];
240 erratic
->array
[probe
- linear
->array
] = NULL
;
242 erratic
->array
[i
] = (fde
*)chain_end
;
243 chain_end
= &linear
->array
[i
];
246 /* Each entry in LINEAR which is part of the linear sequence we have
247 discovered will correspond to a non-NULL entry in the chain we built in
248 the ERRATIC array. */
249 for (i
= j
= k
= 0; i
< count
; i
++)
250 if (erratic
->array
[i
])
251 linear
->array
[j
++] = linear
->array
[i
];
253 erratic
->array
[k
++] = linear
->array
[i
];
258 /* This is O(n log(n)). BSD/OS defines heapsort in stdlib.h, so we must
259 use a name that does not conflict. */
261 frame_heapsort (fde_vector
*erratic
)
263 /* For a description of this algorithm, see:
264 Samuel P. Harbison, Guy L. Steele Jr.: C, a reference manual, 2nd ed.,
266 fde
** a
= erratic
->array
;
267 /* A portion of the array is called a "heap" if for all i>=0:
268 If i and 2i+1 are valid indices, then a[i] >= a[2i+1].
269 If i and 2i+2 are valid indices, then a[i] >= a[2i+2]. */
270 #define SWAP(x,y) do { fde * tmp = x; x = y; y = tmp; } while (0)
271 size_t n
= erratic
->count
;
277 /* Invariant: a[m..n-1] is a heap. */
279 for (i
= m
; 2*i
+1 < n
; )
282 && fde_compare (a
[2*i
+2], a
[2*i
+1]) > 0
283 && fde_compare (a
[2*i
+2], a
[i
]) > 0)
285 SWAP (a
[i
], a
[2*i
+2]);
288 else if (fde_compare (a
[2*i
+1], a
[i
]) > 0)
290 SWAP (a
[i
], a
[2*i
+1]);
299 /* Invariant: a[0..n-1] is a heap. */
302 for (i
= 0; 2*i
+1 < n
; )
305 && fde_compare (a
[2*i
+2], a
[2*i
+1]) > 0
306 && fde_compare (a
[2*i
+2], a
[i
]) > 0)
308 SWAP (a
[i
], a
[2*i
+2]);
311 else if (fde_compare (a
[2*i
+1], a
[i
]) > 0)
313 SWAP (a
[i
], a
[2*i
+1]);
323 /* Merge V1 and V2, both sorted, and put the result into V1. */
325 fde_merge (fde_vector
*v1
, const fde_vector
*v2
)
336 fde2
= v2
->array
[i2
];
337 while (i1
> 0 && fde_compare (v1
->array
[i1
-1], fde2
) > 0)
339 v1
->array
[i1
+i2
] = v1
->array
[i1
-1];
342 v1
->array
[i1
+i2
] = fde2
;
344 v1
->count
+= v2
->count
;
349 end_fde_sort (fde_accumulator
*accu
, size_t count
)
351 if (accu
->linear
.array
&& accu
->linear
.count
!= count
)
354 if (accu
->erratic
.array
)
356 fde_split (&accu
->linear
, &accu
->erratic
);
357 if (accu
->linear
.count
+ accu
->erratic
.count
!= count
)
359 frame_heapsort (&accu
->erratic
);
360 fde_merge (&accu
->linear
, &accu
->erratic
);
361 free (accu
->erratic
.array
);
365 /* We've not managed to malloc an erratic array, so heap sort in the
367 frame_heapsort (&accu
->linear
);
369 return accu
->linear
.array
;
374 count_fdes (fde
*this_fde
)
378 for (count
= 0; this_fde
->length
!= 0; this_fde
= next_fde (this_fde
))
379 /* Skip CIEs and omitted link-once FDE entries. */
380 if (this_fde
->CIE_delta
!= 0 && this_fde
->pc_begin
!= 0)
387 add_fdes (fde
*this_fde
, fde_accumulator
*accu
, void **beg_ptr
, void **end_ptr
)
389 void *pc_begin
= *beg_ptr
;
390 void *pc_end
= *end_ptr
;
392 for (; this_fde
->length
!= 0; this_fde
= next_fde (this_fde
))
394 /* Skip CIEs and linked once FDE entries. */
395 if (this_fde
->CIE_delta
== 0 || this_fde
->pc_begin
== 0)
398 fde_insert (accu
, this_fde
);
400 if (this_fde
->pc_begin
< pc_begin
)
401 pc_begin
= this_fde
->pc_begin
;
402 if (this_fde
->pc_begin
+ this_fde
->pc_range
> pc_end
)
403 pc_end
= this_fde
->pc_begin
+ this_fde
->pc_range
;
411 search_fdes (fde
*this_fde
, void *pc
)
413 for (; this_fde
->length
!= 0; this_fde
= next_fde (this_fde
))
415 /* Skip CIEs and linked once FDE entries. */
416 if (this_fde
->CIE_delta
== 0 || this_fde
->pc_begin
== 0)
419 if ((uaddr
)((char *)pc
- (char *)this_fde
->pc_begin
) < this_fde
->pc_range
)
425 /* Set up a sorted array of pointers to FDEs for a loaded object. We
426 count up the entries before allocating the array because it's likely to
427 be faster. We can be called multiple times, should we have failed to
428 allocate a sorted fde array on a previous occasion. */
431 frame_init (struct object
* ob
)
434 fde_accumulator accu
;
435 void *pc_begin
, *pc_end
;
440 else if (ob
->fde_array
)
442 fde
**p
= ob
->fde_array
;
443 for (count
= 0; *p
; ++p
)
444 count
+= count_fdes (*p
);
447 count
= count_fdes (ob
->fde_begin
);
450 if (!start_fde_sort (&accu
, count
) && ob
->pc_begin
)
453 pc_begin
= (void*)(uaddr
)-1;
458 fde
**p
= ob
->fde_array
;
460 add_fdes (*p
, &accu
, &pc_begin
, &pc_end
);
463 add_fdes (ob
->fde_begin
, &accu
, &pc_begin
, &pc_end
);
464 array
= end_fde_sort (&accu
, count
);
466 ob
->fde_array
= array
;
467 ob
->pc_begin
= pc_begin
;
472 _Unwind_Find_FDE (void *pc
, struct dwarf_eh_bases
*bases ATTRIBUTE_UNUSED
)
477 init_object_mutex_once ();
478 __gthread_mutex_lock (&object_mutex
);
480 /* Linear search through the objects, to find the one containing the pc. */
481 for (ob
= objects
; ob
; ob
= ob
->next
)
483 if (ob
->pc_begin
== 0)
485 if (pc
>= ob
->pc_begin
&& pc
< ob
->pc_end
)
491 __gthread_mutex_unlock (&object_mutex
);
495 if (!ob
->fde_array
|| (void *)ob
->fde_array
== (void *)ob
->fde_begin
)
498 if (ob
->fde_array
&& (void *)ob
->fde_array
!= (void *)ob
->fde_begin
)
500 __gthread_mutex_unlock (&object_mutex
);
502 /* Standard binary search algorithm. */
503 for (lo
= 0, hi
= ob
->count
; lo
< hi
; )
505 size_t i
= (lo
+ hi
) / 2;
506 fde
*f
= ob
->fde_array
[i
];
508 if (pc
< f
->pc_begin
)
510 else if (pc
>= f
->pc_begin
+ f
->pc_range
)
518 /* Long slow labourious linear search, cos we've no memory. */
523 fde
**p
= ob
->fde_array
;
527 f
= search_fdes (*p
, pc
);
535 f
= search_fdes (ob
->fde_begin
, pc
);
537 __gthread_mutex_unlock (&object_mutex
);