* breakpoint.c: #include "hashtab.h".

(ambiguous_names_p): New fn.
	(update_breakpoint_locations): When restoring bp enable status, don't
	compare function names if any functions have same name.
	* Makefile.in (breakpoint.o): Add hashtab.h dependency.

	* gdb.cp/mb-inline.exp: New.
	* gdb.cp/mb-inline.h: New.
	* gdb.cp/mb-inline1.cc: New.
	* gdb.cp/mb-inline2.cc: New.
This commit is contained in:
Doug Evans 2008-02-08 00:42:07 +00:00
parent b29242315e
commit 776592bf17
7 changed files with 271 additions and 8 deletions

View file

@ -1,3 +1,11 @@
2008-02-07 Doug Evans <dje@google.com>
* breakpoint.c: #include "hashtab.h".
(ambiguous_names_p): New fn.
(update_breakpoint_locations): When restoring bp enable status, don't
compare function names if any functions have same name.
* Makefile.in (breakpoint.o): Add hashtab.h dependency.
2008-02-07 Joel Brobecker <brobecker@adacore.com>
* ada-lang.c (symbol_completion_add): Make SV parameter a VEC**

View file

@ -1950,7 +1950,7 @@ breakpoint.o: breakpoint.c $(defs_h) $(symtab_h) $(frame_h) $(breakpoint_h) \
$(objfiles_h) $(source_h) $(linespec_h) $(completer_h) $(gdb_h) \
$(ui_out_h) $(cli_script_h) $(gdb_assert_h) $(block_h) $(solib_h) \
$(solist_h) $(observer_h) $(exceptions_h) $(gdb_events_h) \
$(mi_common_h) $(memattr_h) $(ada_lang_h) $(top_h)
$(mi_common_h) $(memattr_h) $(ada_lang_h) $(top_h) $(hashtab_h)
bsd-kvm.o: bsd-kvm.c $(defs_h) $(cli_cmds_h) $(command_h) $(frame_h) \
$(regcache_h) $(target_h) $(value_h) $(gdbcore_h) $(gdb_assert_h) \
$(readline_h) $(bsd_kvm_h)

View file

@ -21,6 +21,7 @@
#include "defs.h"
#include <ctype.h>
#include "hashtab.h"
#include "symtab.h"
#include "frame.h"
#include "breakpoint.h"
@ -7081,6 +7082,43 @@ all_locations_are_pending (struct bp_location *loc)
return 1;
}
/* Subroutine of update_breakpoint_locations to simplify it.
Return non-zero if multiple fns in list LOC have the same name.
Null names are ignored. */
static int
ambiguous_names_p (struct bp_location *loc)
{
struct bp_location *l;
htab_t htab = htab_create_alloc (13, htab_hash_string,
(int (*) (const void *, const void *)) streq,
NULL, xcalloc, xfree);
for (l = loc; l != NULL; l = l->next)
{
const char **slot;
const char *name = l->function_name;
/* Allow for some names to be NULL, ignore them. */
if (name == NULL)
continue;
slot = (const char **) htab_find_slot (htab, (const void *) name,
INSERT);
/* NOTE: We can assume slot != NULL here because xcalloc never returns
NULL. */
if (*slot != NULL)
{
htab_delete (htab);
return 1;
}
*slot = name;
}
htab_delete (htab);
return 0;
}
static void
update_breakpoint_locations (struct breakpoint *b,
struct symtabs_and_lines sals)
@ -7143,18 +7181,37 @@ update_breakpoint_locations (struct breakpoint *b,
/* If possible, carry over 'disable' status from existing breakpoints. */
{
struct bp_location *e = existing_locations;
/* If there are multiple breakpoints with the same function name,
e.g. for inline functions, comparing function names won't work.
Instead compare pc addresses; this is just a heuristic as things
may have moved, but in practice it gives the correct answer
often enough until a better solution is found. */
int have_ambiguous_names = ambiguous_names_p (b->loc);
for (; e; e = e->next)
{
if (!e->enabled && e->function_name)
{
struct bp_location *l = b->loc;
for (; l; l = l->next)
if (l->function_name
&& strcmp (e->function_name, l->function_name) == 0)
{
l->enabled = 0;
break;
}
if (have_ambiguous_names)
{
for (; l; l = l->next)
if (e->address == l->address)
{
l->enabled = 0;
break;
}
}
else
{
for (; l; l = l->next)
if (l->function_name
&& strcmp (e->function_name, l->function_name) == 0)
{
l->enabled = 0;
break;
}
}
}
}
}

View file

@ -0,0 +1,108 @@
# Copyright 2008 Free Software Foundation, Inc.
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 3 of the License, or
# (at your option) any later version.
#
# This program 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 General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
# This file is part of the gdb testsuite.
# This test verifies that setting breakpoint on line in inline
# function will fire in all instantiations of that function.
if $tracelevel then {
strace $tracelevel
}
if { [skip_cplus_tests] } { continue }
set prms_id 0
set bug_id 0
set testfile "mb-inline"
set hdrfile "${testfile}.h"
set srcfile1 "${testfile}1.cc"
set objfile1 "${testfile}1.o"
set srcfile2 "${testfile}2.cc"
set objfile2 "${testfile}2.o"
set binfile "${objdir}/${subdir}/${testfile}"
if { [gdb_compile "$srcdir/$subdir/$srcfile1" "$objdir/$subdir/$objfile1" object {debug c++}] != "" } {
untested mb-inline.exp
return -1
}
if { [gdb_compile "$srcdir/$subdir/$srcfile2" "$objdir/$subdir/$objfile2" object {debug c++}] != "" } {
untested mb-inline.exp
return -1
}
if { [gdb_compile "$objdir/$subdir/$objfile1 $objdir/$subdir/$objfile2" "${binfile}" executable {debug c++}] != "" } {
untested mb-inline.exp
return -1
}
if [get_compiler_info ${binfile} "c++"] {
return -1
}
gdb_exit
gdb_start
gdb_reinitialize_dir $srcdir/$subdir
gdb_load ${binfile}
set bp_location [gdb_get_line_number "set breakpoint here" $hdrfile]
# Set a breakpoint with multiple locations.
gdb_test "break $hdrfile:$bp_location" \
"Breakpoint.*at.* file .*$hdrfile, line.*\\(2 locations\\).*" \
"set breakpoint"
gdb_run_cmd
gdb_expect {
-re "Breakpoint \[0-9\]+,.*foo \\(i=0\\).*$gdb_prompt $" {
pass "run to breakpoint"
}
-re "$gdb_prompt $" {
fail "run to breakpoint"
}
timeout {
fail "run to breakpoint (timeout)"
}
}
gdb_test "continue" \
".*Breakpoint.*foo \\(i=1\\).*" \
"run to breakpoint 2"
# Try disabling a single location. We also test
# that at least in simple cases, the enable/disable
# state of locations survive "run".
# Early bug would disable 1.1 and enable 1.2 when program is run.
gdb_test "disable 1.2" "" "disabling location: disable"
gdb_run_cmd
gdb_expect {
-re "Breakpoint \[0-9\]+,.*foo \\(i=0\\).*$gdb_prompt $" {
pass "disabling location: run to breakpoint"
}
-re "$gdb_prompt $" {
fail "disabling location: run to breakpoint"
}
timeout {
fail "disabling location: run to breakpoint (timeout)"
}
}
gdb_test "continue" \
".*Program exited normally.*" \
"continue with disabled breakpoint 1.2"

View file

@ -0,0 +1,30 @@
/* Test gdb support for setting multiple file:line breakpoints on static
functions. In practice the functions may be inline fns compiled with -O0.
We avoid using inline here for simplicity's sake.
This testcase is part of GDB, the GNU debugger.
Copyright 2008 Free Software Foundation, Inc.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
static int
foo (int i)
{
return i; // set breakpoint here
}
extern int afn ();
extern int bfn ();

View file

@ -0,0 +1,35 @@
/* Test gdb support for setting file:line breakpoints on inline fns.
This testcase is part of GDB, the GNU debugger.
Copyright 2008 Free Software Foundation, Inc.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "mb-inline.h"
int
afn ()
{
return foo (0);
}
int
main ()
{
int a = afn ();
int b = bfn ();
return a * b;
}

View file

@ -0,0 +1,25 @@
/* This testcase is part of GDB, the GNU debugger.
Copyright 2008 Free Software Foundation, Inc.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "mb-inline.h"
int
bfn ()
{
return foo (1);
}