Home | History | Annotate | Line # | Download | only in gdb.base
      1 # Copyright 2020-2024 Free Software Foundation, Inc.
      2 
      3 # This program is free software; you can redistribute it and/or modify
      4 # it under the terms of the GNU General Public License as published by
      5 # the Free Software Foundation; either version 3 of the License, or
      6 # (at your option) any later version.
      7 #
      8 # This program is distributed in the hope that it will be useful,
      9 # but WITHOUT ANY WARRANTY; without even the implied warranty of
     10 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
     11 # GNU General Public License for more details.
     12 #
     13 # You should have received a copy of the GNU General Public License
     14 # along with this program.  If not, see <http://www.gnu.org/licenses/>.
     15 
     16 # Test defining a conditional breakpoint that applies to multiple
     17 # locations with different contexts (e.g. different set of local vars).
     18 
     19 standard_testfile .cc
     20 
     21 if {[prepare_for_testing "failed to prepare" ${binfile} ${srcfile}]} {
     22     return
     23 }
     24 
     25 set warning "warning: failed to validate condition"
     26 set fill "\[^\r\n\]*"
     27 
     28 # Location indices are determined by their address, which depends on
     29 # how the compiler emits the code.  We keep a map from the location
     30 # index to the location context (i.e. A, Base, or C), so that we can
     31 # write tests without hard-coding location indices.
     32 set loc_name(1) ""
     33 set loc_name(2) ""
     34 set loc_name(3) ""
     35 # And, for convenience, a reverse map from the name to the index.
     36 set loc_index(A)    0
     37 set loc_index(Base) 0
     38 set loc_index(C)    0
     39 
     40 # Find breakpoint location contexts.
     41 
     42 proc find_location_contexts {} {
     43     global loc_name loc_index bpnum1 fill
     44     global decimal hex gdb_prompt
     45 
     46     gdb_test_multiple "info breakpoint $bpnum1" "find_location_indices" {
     47 	-re "stop only if ${fill}\r\n" {
     48 	    exp_continue
     49 	}
     50 	-re "^${bpnum1}\.($decimal) ${fill} ${hex} in (A|Base|C)::func${fill}\r\n" {
     51 	    set index $expect_out(1,string)
     52 	    set name $expect_out(2,string)
     53 	    set loc_name($index) $name
     54 	    set loc_index($name) $index
     55 	    exp_continue
     56 	}
     57 	-re "$gdb_prompt $" {
     58 	    verbose -log "Loc names: $loc_name(1), $loc_name(2), $loc_name(3)"
     59 	    gdb_assert { ![string equal $loc_name(1) $loc_name(2)] }
     60 	    gdb_assert { ![string equal $loc_name(1) $loc_name(3)] }
     61 	    gdb_assert { ![string equal $loc_name(2) $loc_name(3)] }
     62 	    gdb_assert { [string length $loc_name(1)] > 0 }
     63 	    gdb_assert { [string length $loc_name(2)] > 0 }
     64 	    gdb_assert { [string length $loc_name(3)] > 0 }
     65 	}
     66     }
     67 }
     68 
     69 # Test the breakpoint location enabled states.  STATES is a list of
     70 # location states.  We assume STATES to contain the state for A, Base,
     71 # and C, and in this order.  E.g. {N* y n} for 'N*' at A::func, 'y' at
     72 # B::func, and 'n' at C::func, respectively.
     73 
     74 proc check_bp_locations {bpnum states cond {msg ""}} {
     75     global loc_name fill
     76 
     77     # Map location names to location states.
     78     set loc_states(A)    [lindex $states 0]
     79     set loc_states(Base) [lindex $states 1]
     80     set loc_states(C)    [lindex $states 2]
     81 
     82     if {$cond == ""} {
     83 	set cond_info ""
     84     } else {
     85 	set bp_hit_info "${fill}(\r\n${fill}breakpoint already hit 1 time)?"
     86 	set cond_info "\r\n${fill}stop only if ${cond}${bp_hit_info}"
     87     }
     88 
     89     set expected [multi_line \
     90 	"Num${fill}" \
     91 	"${bpnum}${fill}breakpoint${fill}keep y${fill}MULTIPLE${fill}${cond_info}" \
     92 	"${bpnum}.1${fill} $loc_states($loc_name(1)) ${fill}" \
     93 	"${bpnum}.2${fill} $loc_states($loc_name(2)) ${fill}" \
     94 	"${bpnum}.3${fill} $loc_states($loc_name(3)) ${fill}"]
     95 
     96     if {[lsearch $states N*] >= 0} {
     97 	append expected "\r\n\\(\\*\\): Breakpoint condition is invalid at this location."
     98     }
     99 
    100     gdb_test "info break $bpnum" $expected "check bp $bpnum $msg"
    101 }
    102 
    103 # Scenario 1: Define breakpoints conditionally, using the "break N if
    104 # cond" syntax.  Run the program, check that we hit those locations
    105 # only.
    106 
    107 with_test_prefix "scenario 1" {
    108     # Define the conditional breakpoints.  Two locations (Base::func
    109     # and C::func) should be disabled.  We do not test location
    110     # indices strictly at this moment, because we don't know them,
    111     # yet.  We have strict location index tests below.
    112     gdb_test "break func if a == 10" \
    113 	[multi_line \
    114 	     "${warning} at location $decimal, disabling:" \
    115 	     "  No symbol \"a\" in current context." \
    116 	     "${warning} at location $decimal, disabling:" \
    117 	     "  No symbol \"a\" in current context." \
    118 	     "Breakpoint $decimal at $fill .3 locations."] \
    119 	"define bp with condition a == 10"
    120     set bpnum1 [get_integer_valueof "\$bpnum" 0 "get bpnum1"]
    121 
    122     gdb_test "break func if c == 30" \
    123 	[multi_line \
    124 	     ".*${warning} at location $decimal, disabling:" \
    125 	     "  No symbol \"c\" in current context." \
    126 	     ".*${warning} at location $decimal, disabling:" \
    127 	     "  No symbol \"c\" in current context." \
    128 	     ".*Breakpoint $decimal at $fill .3 locations."] \
    129 	"define bp with condition c == 30"
    130     set bpnum2 [get_integer_valueof "\$bpnum" 0 "get bpnum2"]
    131 
    132     find_location_contexts
    133 
    134     with_test_prefix "before run" {
    135 	check_bp_locations $bpnum1 {y N* N*} "a == 10"
    136 	check_bp_locations $bpnum2 {N* N* y} "c == 30"
    137     }
    138 
    139     # Do not use runto_main, it deletes all breakpoints.
    140     gdb_run_cmd
    141 
    142     # Check our conditional breakpoints.
    143     gdb_test "" ".*Breakpoint $bkptno_num_re, A::func .*" \
    144 	"run until A::func"
    145     gdb_test "print a" " = 10"
    146 
    147     gdb_test "continue" "Continuing.*Breakpoint $bkptno_num_re, C::func .*" \
    148 	"run until C::func"
    149     gdb_test "print c" " = 30"
    150 
    151     # No more hits!
    152     gdb_continue_to_end
    153 
    154     with_test_prefix "after run" {
    155 	check_bp_locations $bpnum1 {y N* N*} "a == 10"
    156 	check_bp_locations $bpnum2 {N* N* y} "c == 30"
    157     }
    158 }
    159 
    160 # Start GDB with two breakpoints and define the conditions separately.
    161 
    162 proc setup_bps {} {
    163     global srcfile binfile srcfile2 decimal
    164     global bpnum1 bpnum2 bp_location warning loc_index
    165 
    166     clean_restart ${binfile}
    167 
    168     # Define the breakpoints.
    169     gdb_breakpoint "func"
    170     set bpnum1 [get_integer_valueof "\$bpnum" 0 "get bpnum1"]
    171 
    172     gdb_breakpoint "func"
    173     set bpnum2 [get_integer_valueof "\$bpnum" 0 "get bpnum2"]
    174 
    175     # Defining a condition on 'a' disables 2 locations.
    176     set locs [lsort -integer "$loc_index(Base) $loc_index(C)"]
    177     gdb_test "cond $bpnum1 a == 10" \
    178 	[multi_line \
    179 	     "$warning at location ${bpnum1}.[lindex $locs 0], disabling:" \
    180 	     "  No symbol \"a\" in current context." \
    181 	     "$warning at location ${bpnum1}.[lindex $locs 1], disabling:" \
    182 	     "  No symbol \"a\" in current context."]
    183 
    184     # Defining a condition on 'c' disables 2 locations.
    185     set locs [lsort -integer "$loc_index(Base) $loc_index(A)"]
    186     gdb_test "cond $bpnum2 c == 30" \
    187 	[multi_line \
    188 	     "$warning at location ${bpnum2}.[lindex $locs 0], disabling:" \
    189 	     "  No symbol \"c\" in current context." \
    190 	     "$warning at location ${bpnum2}.[lindex $locs 1], disabling:" \
    191 	     "  No symbol \"c\" in current context."]
    192 }
    193 
    194 # Scenario 2: Define breakpoints unconditionally, and then define
    195 # conditions using the "cond N <cond>" syntax.  Expect that the
    196 # locations where <cond> is not evaluatable are disabled.  Run the
    197 # program, check that we hit the enabled locations only.
    198 
    199 with_test_prefix "scenario 2" {
    200     setup_bps
    201 
    202     with_test_prefix "before run" {
    203 	check_bp_locations $bpnum1 {y N* N*} "a == 10"
    204 	check_bp_locations $bpnum2 {N* N* y} "c == 30"
    205     }
    206 
    207     # Do not use runto_main, it deletes all breakpoints.
    208     gdb_run_cmd
    209 
    210     # Check that we hit enabled locations only.
    211     gdb_test "" ".*Breakpoint $bkptno_num_re, A::func .*" \
    212 	"run until A::func"
    213     gdb_test "print a" " = 10"
    214 
    215     gdb_test "continue" "Continuing.*Breakpoint $bkptno_num_re, C::func .*" \
    216 	"run until C::func"
    217     gdb_test "print c" " = 30"
    218 
    219     # No more hits!
    220     gdb_continue_to_end
    221 
    222     with_test_prefix "after run" {
    223 	check_bp_locations $bpnum1 {y N* N*} "a == 10"
    224 	check_bp_locations $bpnum2 {N* N* y} "c == 30"
    225     }
    226 }
    227 
    228 # Scenario 3: Apply misc. checks on the already-defined breakpoints.
    229 
    230 with_test_prefix "scenario 3" {
    231     setup_bps
    232 
    233     set locs [lsort -integer "$loc_index(Base) $loc_index(A)"]
    234     gdb_test "cond $bpnum1 c == 30" \
    235 	[multi_line \
    236 	     "${warning} at location ${bpnum1}.[lindex $locs 0], disabling:" \
    237 	     "  No symbol \"c\" in current context." \
    238 	     "${warning} at location ${bpnum1}.[lindex $locs 1], disabling:" \
    239 	     "  No symbol \"c\" in current context." \
    240 	     "Breakpoint ${bpnum1}'s condition is now valid at location $loc_index(C), enabling."] \
    241 	"change the condition of bp 1"
    242     check_bp_locations $bpnum1 {N* N* y} "c == 30" "after changing the condition"
    243 
    244     gdb_test "cond $bpnum1" \
    245 	[multi_line \
    246 	     "Breakpoint ${bpnum1}'s condition is now valid at location [lindex $locs 0], enabling." \
    247 	     "Breakpoint ${bpnum1}'s condition is now valid at location [lindex $locs 1], enabling." \
    248 	     "Breakpoint ${bpnum1} now unconditional."] \
    249 	"reset the condition of bp 1"
    250     check_bp_locations $bpnum1 {y y y} "" "after resetting the condition"
    251 
    252     gdb_test_no_output "disable ${bpnum2}.$loc_index(A)"
    253     check_bp_locations $bpnum2 {N* N* y} "c == 30" "after disabling loc for A"
    254 
    255     gdb_test "cond $bpnum2" ".*" "reset the condition of bp 2"
    256     check_bp_locations $bpnum2 {n y y} "" "loc for A should remain disabled"
    257 
    258     gdb_test_no_output "disable ${bpnum2}.$loc_index(C)"
    259     check_bp_locations $bpnum2 {n y n} "" "after disabling loc for C"
    260 
    261     gdb_test "cond $bpnum2 c == 30" \
    262 	[multi_line \
    263 	     "${warning} at location ${bpnum2}.$loc_index(Base), disabling:" \
    264 	     "  No symbol \"c\" in current context."] \
    265 	"re-define a condition"
    266     check_bp_locations $bpnum2 {N* N* n} "c == 30" "loc for C should remain disabled"
    267 
    268     gdb_test "enable ${bpnum2}.$loc_index(Base)" \
    269 	"Breakpoint ${bpnum2}'s condition is invalid at location $loc_index(Base), cannot enable." \
    270 	"reject enabling a location that is disabled-by-cond"
    271     check_bp_locations $bpnum2 {N* N* n} "c == 30" "after enable attempt"
    272 
    273     gdb_test "cond $bpnum2 garbage" \
    274 	"No symbol \"garbage\" in current context." \
    275 	"reject condition if bad for all locations"
    276 
    277     gdb_test_no_output "delete $bpnum1"
    278 
    279     # Do not use runto_main, it deletes all breakpoints.
    280     gdb_breakpoint "main"
    281     gdb_run_cmd
    282     gdb_test "" ".*reakpoint .*, main .*${srcfile}.*" "start"
    283 
    284     # The second BP's locations are all disabled.  No more hits!
    285     gdb_continue_to_end
    286 }
    287 
    288 # Scenario 4: Test the '-force'/'-force-condition' flag.
    289 
    290 with_test_prefix "force" {
    291     clean_restart ${binfile}
    292 
    293     gdb_breakpoint "func"
    294     # Pick a condition that is invalid at every location.
    295     set bpnum1 [get_integer_valueof "\$bpnum" 0 "get bpnum1"]
    296     gdb_test "cond -force $bpnum1 foo" \
    297 	[multi_line \
    298 	     "${warning} at location ${bpnum1}.1, disabling:" \
    299 	     "  No symbol \"foo\" in current context." \
    300 	     "${warning} at location ${bpnum1}.2, disabling:" \
    301 	     "  No symbol \"foo\" in current context." \
    302 	     "${warning} at location ${bpnum1}.3, disabling:" \
    303 	     "  No symbol \"foo\" in current context."] \
    304 	"force the condition of bp 1"
    305     check_bp_locations $bpnum1 {N* N* N*} "foo" "after forcing the condition"
    306 
    307     # Now with the 'break' command.
    308     gdb_breakpoint "func -force-condition if baz"
    309     set bpnum2 [get_integer_valueof "\$bpnum" 0 "get bpnum2"]
    310     check_bp_locations $bpnum2 {N* N* N*} "baz" "set using the break command"
    311 }
    312