Mirror of the gdb-patches mailing list
 help / color / mirror / Atom feed
From: Simon Marchi via Gdb-patches <gdb-patches@sourceware.org>
To: Zoran Zaric <Zoran.Zaric@amd.com>, gdb-patches@sourceware.org
Subject: Re: [PATCH 03/43] Move frame context info to dwarf_expr_context
Date: Mon, 26 Apr 2021 22:19:14 -0400	[thread overview]
Message-ID: <7f00dde3-c7dc-f839-bf8d-035f18d65ce1@polymtl.ca> (raw)
In-Reply-To: <20210301144620.103016-4-Zoran.Zaric@amd.com>

Hi Zoran,

This patch was already ok'ed by Tom, but I have some more suggestions
below.  Nothing that compromises the general idea of the patch though.

On 2021-03-01 9:45 a.m., Zoran Zaric via Gdb-patches wrote:
> From: Zoran Zaric <zoran.zaric@amd.com>
> 
> Following 15 patches in this patch series is cleaning up the design of
> the DWARF expression evaluator (dwarf_expr_context) to make future
> extensions of that evaluator easier and cleaner to implement.
> 
> There are three subclasses of the dwarf_expr_context class
> (dwarf_expr_executor, dwarf_evaluate_loc_desc and
> evaluate_for_locexpr_baton). Here is a short description of each class:
> 
> - dwarf_expr_executor is evaluating a DWARF expression in a context
>   of a Call Frame Information. The overridden methods of this subclass
>   report an error if a specific DWARF operation, represented by that
>   method, is not allowed in a CFI context. The source code of this
>   subclass lacks the support for composite as well as implicit pointer
>   location description.
> 
> - dwarf_evaluate_loc_desc can evaluate any expression with no
>   restrictions. All of the methods that this subclass overrides are
>   actually doing what they are intended to do. This subclass contains
>   a full support for all location description types.
> 
> - evaluate_for_locexpr_baton subclass is a specialization of the
>   dwarf_evaluate_loc_desc subclass and it’s function is to add
>   support for passed in buffers. This seems to be a way to go around
>   the fact that DWARF standard lacks a bit offset support for memory
>   location descriptions as well as using any location description for
>   the push object address functionality.
> 
> It all comes down to this question: what is a function of a DWARF
> expression evaluator?
> 
> Is it to evaluate the expression in a given context or to check the
> correctness of that expression in that context?
> 
> Currently, the only reason why there is a dwarf_expr_executor subclass
> is to report an invalid DWARF expression in a context of a CFI, but is
> that what the evaluator is supposed to do considering that the evaluator
> is not tied to a given DWARF version?
> 
> There are more and more vendor and GNU extensions that are not part of
> the DWARF standard, so is it that impossible to expect that some of the
> extensions could actually lift the previously imposed restrictions of
> the CFI context? Not to mention that every new DWARF version is lifting
> some restrictions anyway.
> 
> The thing that makes more sense for an evaluator to do, is to take the
> context of an evaluation and checks the requirements of every operation
> evaluated against that context. With this approach, the evaluator would
> report an error only if parts of the context, necessary for the
> evaluation, are missing.
> 
> If this approach is taken, then the unification of the
> dwarf_evaluate_loc_desc, dwarf_expr_executor and dwarf_expr_context
> is the next logical step. This makes a design of the DWARF expression
> evaluator cleaner and allows more flexibility when supporting future
> vendor and GNU extensions.
> 
> Additional benefit here is that now all evaluators have access to all
> location description types, which means that a vendor extended CFI
> rules could support composite location description as well. This also
> means that a new evaluator interface can be changed to return a single
> struct value (that describes the result of the evaluation) instead of
> a caller poking around the dwarf_expr_context internal data for answers
> (like it is done currently).
> 
> This patch starts the merging proccess by moving the frame context

proccess -> process

> @@ -56,6 +57,31 @@ dwarf_gdbarch_types_init (struct gdbarch *gdbarch)
>    return types;
>  }
>  
> +/* Ensure that a FRAME is defined, throw an exception otherwise.
> +
> +   Throwing NOT_AVAILABLE_ERROR error so that a client can chose
> +   to react differently if the evaluation ended because there
> +   was a missing context information.  */
> +
> +static void
> +ensure_have_frame (struct frame_info *frame, const char *op_name)
> +{
> +  if (frame == nullptr)
> +    throw_error (NOT_AVAILABLE_ERROR,
> +		 _("%s evaluation requires a frame."), op_name);
> +}

I don't remember if we discussed about that or not, so I'll ask:
"available" in GDB terminology usually refers to the tracing
functionality.  While tracing, you can choose which registers and what
part of the memory to collect when hitting a tracepoint.  During later
analysis, if you try for example to print the value of a variable that
requires some information (register or memory) that you didn't collect,
we'll say that this register or memory (and therefore the variable's
value) is not available / unavailable.

It is also used with the "record" command / concept, where some things
are recorded during execution to be able to step backwards in time,
that's pretty much the same as tracing.  If you step backwards in time
and try to access an information that wasn't recorded and an exception
is thrown as a result, it will be NOT_AVAILABLE_ERROR.

So I am a bit worried that by using NOT_AVAILABLE_ERROR to say "to
evaluate this expression, you need a frame, but there is no frame in the
current context", we are overloading this exception type / return code.

I can imagine that I could be inspecting a trace, I try to print a
variable whose location expression needs to be evaluated.  Evaluating
the location expression requires accessing some memory or register that
wasn't recorded in my trace, and that results in throwing
NOT_AVAILABLE_ERROR.  That would be a different error than trying to
evaluate an expression that requires a frame in a context that doesn't
have a frame.

So it seems to me like a new exception type would be desirable.  Maybe
INSUFFICIENT_CONTEXT_ERROR?

On an unrelated topic, things like ensure_have_frame always worry me
because I'm scared I'll forget to call it where necessary.  An
alternative could be to make sure we get the frame through a getter that
throws if frame == nullptr.

Basically, a tiny class (could be internal to dwarf_expr_context):

struct frame_safe
{
  frame_safe (frame_info *frame)
    : m_frame (frame)
  {}

  frame_info *frame (const char *op)
  {
    if (m_frame == nullptr)
      ... throw ...

    return m_frame;
  }

private:
  frame_info *m_frame;
};

dwarf_expr_context would have a `frame_safe m_frame` field and a
`frame (const char *op)` getter that just does
`return m_frame->frame (op)`.

This way, it's impossible to get the frame and forgetting an
ensure_have_frame call.

> @@ -857,7 +928,7 @@ dwarf_expr_context::execute_stack_op (const gdb_byte *op_ptr,
>  	    if (this->location == DWARF_VALUE_MEMORY)
>  	      result = fetch_address (0);
>  	    else if (this->location == DWARF_VALUE_REGISTER)
> -	      result = this->read_addr_from_reg (value_as_long (fetch (0)));
> +	      result = read_addr_from_reg (this->frame, value_as_long (fetch (0)));

This line is slightly too long.

> @@ -259,8 +247,23 @@ struct dwarf_expr_context
>    void add_piece (ULONGEST size, ULONGEST offset);
>    void execute_stack_op (const gdb_byte *op_ptr, const gdb_byte *op_end);
>    void pop ();
> +
> +  /* Return a value of type TYPE, stored in register number REGNUM
> +     of the frame associated to the given BATON.

"the given BATON" is a bit confusing, I'm not sure which BATON this
refers to.  Could you try to improve this comment?

> +
> +     REGNUM is a DWARF register number.  */
> +  struct value *get_reg_value (struct type *type, int regnum);
> +
> +  /* Return the location expression for the frame base attribute, in
> +     START and LENGTH.  The result must be live until the current
> +     expression evaluation is complete.  */
> +  void get_frame_base (const gdb_byte **start, size_t *length);
>  };
>  
> +/* Return the value of register number REG (a DWARF register number),
> +   read as an address in a given FRAME.  */
> +CORE_ADDR read_addr_from_reg (struct frame_info *, int);

Can you please add the names to the parameters?  It's a bit confusing
otherwise, since the comment refers to them.

Simon

  reply	other threads:[~2021-04-27  2:19 UTC|newest]

Thread overview: 86+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2021-03-01 14:45 [PATCH 00/43 V2] Allow location description on the DWARF stack Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 01/43] Replace the symbol needs evaluator with a parser Zoran Zaric via Gdb-patches
2021-04-27  1:20   ` Simon Marchi via Gdb-patches
2021-04-28 10:17     ` Zoran Zaric via Gdb-patches
2021-04-28 14:08       ` Simon Marchi via Gdb-patches
2021-04-28 15:02         ` Zoran Zaric via Gdb-patches
2021-04-28 15:31         ` Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 02/43] Cleanup of the dwarf_expr_context constructor Zoran Zaric via Gdb-patches
2021-04-27  1:23   ` Simon Marchi via Gdb-patches
2021-04-28 10:19     ` Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 03/43] Move frame context info to dwarf_expr_context Zoran Zaric via Gdb-patches
2021-04-27  2:19   ` Simon Marchi via Gdb-patches [this message]
2021-04-28 10:51     ` Zoran Zaric via Gdb-patches
2021-04-28 14:14       ` Simon Marchi via Gdb-patches
2021-04-28 15:55         ` Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 04/43] Remove get_frame_cfa from dwarf_expr_context Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 05/43] Move compilation unit info to dwarf_expr_context Zoran Zaric via Gdb-patches
2021-04-27  2:58   ` Simon Marchi via Gdb-patches
2021-04-28 11:28     ` Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 06/43] Move dwarf_call " Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 07/43] Move get_object_address " Zoran Zaric via Gdb-patches
2021-04-27  3:12   ` Simon Marchi via Gdb-patches
2021-04-28 11:34     ` Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 08/43] Move read_mem " Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 09/43] Move push_dwarf_reg_entry_value to expr.c Zoran Zaric via Gdb-patches
2021-04-27  3:56   ` Simon Marchi via Gdb-patches
2021-04-28 11:36     ` Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 10/43] Inline get_reg_value method of dwarf_expr_context Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 11/43] Remove empty frame and full evaluators Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 12/43] Merge evaluate_for_locexpr_baton evaluator Zoran Zaric via Gdb-patches
2021-04-28  1:33   ` Simon Marchi via Gdb-patches
2021-04-28 11:39     ` Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 13/43] Move piece_closure and its support to expr.c Zoran Zaric via Gdb-patches
2021-04-28  1:56   ` Simon Marchi via Gdb-patches
2021-04-28 11:40     ` Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 14/43] Make value_copy also copy the stack data member Zoran Zaric via Gdb-patches
2021-04-28  2:01   ` Simon Marchi via Gdb-patches
2021-04-28 11:43     ` Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 15/43] Make DWARF evaluator return a single struct value Zoran Zaric via Gdb-patches
2021-04-28  2:21   ` Simon Marchi via Gdb-patches
2021-04-28 11:47     ` Zoran Zaric via Gdb-patches
2021-04-28 14:24       ` Simon Marchi via Gdb-patches
2021-03-01 14:45 ` [PATCH 16/43] Simplify dwarf_expr_context class interface Zoran Zaric via Gdb-patches
2021-04-28  2:45   ` Simon Marchi via Gdb-patches
2021-04-28 13:15     ` Zoran Zaric via Gdb-patches
2021-04-28 14:41       ` Simon Marchi via Gdb-patches
2021-04-28 15:39         ` Zoran Zaric via Gdb-patches
2021-04-28 19:19           ` Simon Marchi via Gdb-patches
2021-04-29 15:49       ` Simon Marchi via Gdb-patches
2021-04-29 15:55         ` Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 17/43] Add as_lval argument to expression evaluator Zoran Zaric via Gdb-patches
2021-04-28  3:04   ` Simon Marchi via Gdb-patches
2021-04-28 13:16     ` Zoran Zaric via Gdb-patches
2021-04-28  3:30   ` Simon Marchi via Gdb-patches
2021-03-01 14:45 ` [PATCH 18/43] Add new register access interface to expr.c Zoran Zaric via Gdb-patches
2021-03-08 23:52   ` Lancelot SIX via Gdb-patches
2021-04-28  3:25   ` Simon Marchi via Gdb-patches
2021-04-28 13:29     ` Zoran Zaric via Gdb-patches
2021-04-28 14:48       ` Simon Marchi via Gdb-patches
2021-04-28 15:42         ` Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 19/43] Add new memory " Zoran Zaric via Gdb-patches
2021-04-30 21:24   ` Simon Marchi via Gdb-patches
2021-03-01 14:45 ` [PATCH 20/43] Add new classes that model DWARF stack element Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 21/43] Add to_location method to DWARF entry classes Zoran Zaric via Gdb-patches
2021-03-01 14:45 ` [PATCH 22/43] Add to_value " Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 23/43] Add read method to location description classes Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 24/43] Add write " Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 25/43] Add deref " Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 26/43] Add read_from_gdb_value method to dwarf_location Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 27/43] Add write_to_gdb_value " Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 28/43] Add is_implicit_ptr_at " Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 29/43] Add indirect_implicit_ptr to dwarf_location class Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 30/43] Add new computed struct value callback interface Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 31/43] Add to_gdb_value method to DWARF entry class Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 32/43] Change DWARF stack to use new dwarf_entry classes Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 33/43] Remove old computed struct value callbacks Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 34/43] Comments cleanup between expr.h and expr.c Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 35/43] Remove dwarf_expr_context from expr.h interface Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 36/43] Move read_addr_from_reg function to frame.c Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 37/43] Add frame info check to DW_OP_reg operations Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 38/43] Remove DWARF expression composition check Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 39/43] Change back the symbol needs to use the evaluator Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 40/43] Add support for any location description in CFI Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 41/43] Add DWARF operations for byte and bit offset Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 42/43] Add support for DW_OP_LLVM_undefined operation Zoran Zaric via Gdb-patches
2021-03-01 14:46 ` [PATCH 43/43] Add support for nested composite locations Zoran Zaric via Gdb-patches

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=7f00dde3-c7dc-f839-bf8d-035f18d65ce1@polymtl.ca \
    --to=gdb-patches@sourceware.org \
    --cc=Zoran.Zaric@amd.com \
    --cc=simon.marchi@polymtl.ca \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox