aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--gdb/ChangeLog11
-rw-r--r--gdb/breakpoint.c107
2 files changed, 76 insertions, 42 deletions
diff --git a/gdb/ChangeLog b/gdb/ChangeLog
index cc27fa9..dda1c03 100644
--- a/gdb/ChangeLog
+++ b/gdb/ChangeLog
@@ -1,3 +1,14 @@
+2008-01-29 Vladimir Prus <vladimir@codesourcery.com>
+
+ Don't reset watchpoint block on solib load.
+
+ * breakpoint.c (insert_bp_location): For watchpoints,
+ recompute condition.
+ (breakpoint_re_set_one): Instead of recomputing value
+ and condition for watchpoints, just reset value and
+ let insert_breakpoints/insert_bp_location recompute it.
+ Don't do anything about disabled watchpoint.
+
2008-01-29 Pierre Muller <muller@ics.u-strasbg.fr>
* valarith.c (value_binop): Handle unsigned integer
diff --git a/gdb/breakpoint.c b/gdb/breakpoint.c
index 9551bd5..ecc2478 100644
--- a/gdb/breakpoint.c
+++ b/gdb/breakpoint.c
@@ -1107,6 +1107,18 @@ Note: automatically using hardware breakpoints for read-only addresses.\n"));
}
}
}
+
+ if (bpt->owner->cond_string != NULL)
+ {
+ char *s = bpt->owner->cond_string;
+ if (bpt->cond)
+ {
+ xfree (bpt->cond);
+ bpt->cond = NULL;
+ }
+ bpt->cond = parse_exp_1 (&s, bpt->owner->exp_valid_block, 0);
+ }
+
/* Failure to insert a watchpoint on any memory value in the
value chain brings us here. */
if (!bpt->inserted)
@@ -7228,53 +7240,64 @@ breakpoint_re_set_one (void *bint)
case bp_hardware_watchpoint:
case bp_read_watchpoint:
case bp_access_watchpoint:
- innermost_block = NULL;
- /* The issue arises of what context to evaluate this in. The
- same one as when it was set, but what does that mean when
- symbols have been re-read? We could save the filename and
- functionname, but if the context is more local than that, the
- best we could do would be something like how many levels deep
- and which index at that particular level, but that's going to
- be less stable than filenames or function names. */
-
- /* So for now, just use a global context. */
- if (b->exp)
- {
- xfree (b->exp);
- /* Avoid re-freeing b->exp if an error during the call to
- parse_expression. */
- b->exp = NULL;
- }
- b->exp = parse_expression (b->exp_string);
- b->exp_valid_block = innermost_block;
- mark = value_mark ();
- if (b->val)
- {
- value_free (b->val);
- /* Avoid re-freeing b->val if an error during the call to
- evaluate_expression. */
- b->val = NULL;
- }
- b->val = evaluate_expression (b->exp);
- release_value (b->val);
- if (value_lazy (b->val) && breakpoint_enabled (b))
- value_fetch_lazy (b->val);
+ /* Watchpoint can be either on expression using entirely global variables,
+ or it can be on local variables.
+
+ Watchpoints of the first kind are never auto-deleted, and even persist
+ across program restarts. Since they can use variables from shared
+ libraries, we need to reparse expression as libraries are loaded
+ and unloaded.
+
+ Watchpoints on local variables can also change meaning as result
+ of solib event. For example, if a watchpoint uses both a local and
+ a global variables in expression, it's a local watchpoint, but
+ unloading of a shared library will make the expression invalid.
+ This is not a very common use case, but we still re-evaluate
+ expression, to avoid surprises to the user.
+
+ Note that for local watchpoints, we re-evaluate it only if
+ watchpoints frame id is still valid. If it's not, it means
+ the watchpoint is out of scope and will be deleted soon. In fact,
+ I'm not sure we'll ever be called in this case.
+
+ If a local watchpoint's frame id is still valid, then
+ b->exp_valid_block is likewise valid, and we can safely use it.
+
+ Don't do anything about disabled watchpoints, since they will
+ be reevaluated again when enabled. */
- if (b->cond_string != NULL)
+ if (!breakpoint_enabled (b))
+ break;
+
+ if (b->exp_valid_block == NULL
+ || frame_find_by_id (b->watchpoint_frame) != NULL)
{
- s = b->cond_string;
- if (b->loc->cond)
+ if (b->exp)
{
- xfree (b->loc->cond);
- /* Avoid re-freeing b->exp if an error during the call
- to parse_exp_1. */
- b->loc->cond = NULL;
+ xfree (b->exp);
+ b->exp = NULL;
}
- b->loc->cond = parse_exp_1 (&s, (struct block *) 0, 0);
+ s = b->exp_string;
+ b->exp = parse_exp_1 (&s, b->exp_valid_block, 0);
+
+ /* Since we reparsed expression, we need to update the
+ value. I'm not aware of any way a single solib load or unload
+ can change a valid value into different valid value, but:
+ - even if the value is no longer valid, we have to record
+ this fact, so that when it becomes valid we reports this
+ as value change
+ - unloaded followed by load can change the value for sure.
+
+ We set value to NULL, and insert_breakpoints will
+ update the value. */
+ if (b->val)
+ value_free (b->val);
+ b->val = NULL;
+
+ /* Loading of new shared library change the meaning of
+ watchpoint condition. However, insert_bp_location will
+ recompute watchpoint condition anyway, nothing to do here. */
}
- if (breakpoint_enabled (b))
- mention (b);
- value_free_to_mark (mark);
break;
/* We needn't really do anything to reset these, since the mask
that requests them is unaffected by e.g., new libraries being