diff options
author | Tom Tromey <tromey@redhat.com> | 2011-12-09 18:40:46 +0000 |
---|---|---|
committer | Tom Tromey <tromey@redhat.com> | 2011-12-09 18:40:46 +0000 |
commit | 8a2c437bd981a866a9bc84161d5c4d399551febd (patch) | |
tree | 0191f55a003f8b883298b85df9993d572de258ba /gdb/breakpoint.c | |
parent | 27949e7316463f8b3ebf9f1bb33096e14f61cdbc (diff) | |
download | gdb-8a2c437bd981a866a9bc84161d5c4d399551febd.zip gdb-8a2c437bd981a866a9bc84161d5c4d399551febd.tar.gz gdb-8a2c437bd981a866a9bc84161d5c4d399551febd.tar.bz2 |
* breakpoint.c (compare_breakpoints): New function.
(clear_command): Remove duplicate breakpoints. Properly clean
up.
Diffstat (limited to 'gdb/breakpoint.c')
-rw-r--r-- | gdb/breakpoint.c | 44 |
1 files changed, 43 insertions, 1 deletions
diff --git a/gdb/breakpoint.c b/gdb/breakpoint.c index d9d5bbe..6f4f2d6 100644 --- a/gdb/breakpoint.c +++ b/gdb/breakpoint.c @@ -10028,18 +10028,41 @@ tcatch_command (char *arg, int from_tty) error (_("Catch requires an event name.")); } +/* A qsort comparison function that sorts breakpoints in order. */ + +static int +compare_breakpoints (const void *a, const void *b) +{ + const breakpoint_p *ba = a; + uintptr_t ua = (uintptr_t) *ba; + const breakpoint_p *bb = b; + uintptr_t ub = (uintptr_t) *bb; + + if ((*ba)->number < (*bb)->number) + return -1; + else if ((*ba)->number > (*bb)->number) + return 1; + + /* Now sort by address, in case we see, e..g, two breakpoints with + the number 0. */ + if (ua < ub) + return -1; + return ub > ub ? 1 : 0; +} + /* Delete breakpoints by address or line. */ static void clear_command (char *arg, int from_tty) { - struct breakpoint *b; + struct breakpoint *b, *prev; VEC(breakpoint_p) *found = 0; int ix; int default_match; struct symtabs_and_lines sals; struct symtab_and_line sal; int i; + struct cleanup *cleanups = make_cleanup (null_cleanup, NULL); if (arg) { @@ -10090,6 +10113,7 @@ clear_command (char *arg, int from_tty) breakpoint. */ found = NULL; + make_cleanup (VEC_cleanup (breakpoint_p), &found); for (i = 0; i < sals.nelts; i++) { /* If exact pc given, clear bpts at that pc. @@ -10143,6 +10167,7 @@ clear_command (char *arg, int from_tty) VEC_safe_push(breakpoint_p, found, b); } } + /* Now go thru the 'found' chain and delete them. */ if (VEC_empty(breakpoint_p, found)) { @@ -10152,6 +10177,21 @@ clear_command (char *arg, int from_tty) error (_("No breakpoint at this line.")); } + /* Remove duplicates from the vec. */ + qsort (VEC_address (breakpoint_p, found), + VEC_length (breakpoint_p, found), + sizeof (breakpoint_p), + compare_breakpoints); + prev = VEC_index (breakpoint_p, found, 0); + for (ix = 1; VEC_iterate (breakpoint_p, found, ix, b); ++ix) + { + if (b == prev) + { + VEC_ordered_remove (breakpoint_p, found, ix); + --ix; + } + } + if (VEC_length(breakpoint_p, found) > 1) from_tty = 1; /* Always report if deleted more than one. */ if (from_tty) @@ -10171,6 +10211,8 @@ clear_command (char *arg, int from_tty) } if (from_tty) putchar_unfiltered ('\n'); + + do_cleanups (cleanups); } /* Delete breakpoint in BS if they are `delete' breakpoints and |