[v2,01/11,PR,gdb/14441] gdb: gdbtypes: add definitions for rvalue reference type

Message ID 1453229609-20159-2-git-send-email-artemiyv@acm.org
State New, archived
Headers

Commit Message

Artemiy Volkov Jan. 19, 2016, 6:53 p.m. UTC
  This patch introduces preliminal definitions regarding C++0x rvalue references
to the gdb type system. In addition to an enum type_code entry, a field in
struct type and an accessor macro for that which are created similarly to the
lvalue references' counterparts, we also introduce a TYPE_REFERENCE convenience
macro used to check for both kinds of references simultaneously as they are
equivalent in many contexts.

./Changelog:

2016-01-19  Artemiy Volkov  <artemiyv@acm.org>

        * gdb/gdbtypes.h (enum type_code): Add TYPE_CODE_RVALUE_REF
        constant.
        (TYPE_REFERENCE): New macro.
        (struct type): Add rvalue_reference_type field.
        (TYPE_RVALUE_REFERENCE_TYPE): New macro.
---
 gdb/gdbtypes.h | 13 +++++++++++++
 1 file changed, 13 insertions(+)
  

Comments

Keith Seitz Feb. 19, 2016, 6:49 p.m. UTC | #1
On 01/19/2016 10:53 AM, Artemiy Volkov wrote:
> 2016-01-19  Artemiy Volkov  <artemiyv@acm.org>
> 
>         * gdb/gdbtypes.h (enum type_code): Add TYPE_CODE_RVALUE_REF
>         constant.

Nit: I believe the syntax for this would be:

  (enum type_code) <TYPE_CODE_RVALUE_REF>: New constant.

Also, remove the leading "gdb/" from all gdb/ChangeLog entries. [This
appears in all patches.]

>         (TYPE_REFERENCE): New macro.
>         (struct type): Add rvalue_reference_type field.
>         (TYPE_RVALUE_REFERENCE_TYPE): New macro.
> ---
>  gdb/gdbtypes.h | 13 +++++++++++++
>  1 file changed, 13 insertions(+)
> 
> diff --git a/gdb/gdbtypes.h b/gdb/gdbtypes.h
> index e775a1d..52419b4 100644
> --- a/gdb/gdbtypes.h
> +++ b/gdb/gdbtypes.h
> @@ -362,6 +364,12 @@ enum type_instance_flag_value
>  #define TYPE_ATOMIC(t) \
>    (TYPE_INSTANCE_FLAGS (t) & TYPE_INSTANCE_FLAG_ATOMIC)
>  
> +/* * C++ lvalue and rvalue references are equivalent in many contexts,
> +   thus create a convenience macro that checks if a type is either of them. */
> +
> +#define TYPE_REFERENCE(t) \
> +  (TYPE_CODE(t) == TYPE_CODE_REF || TYPE_CODE(t) == TYPE_CODE_RVALUE_REF)
> +

Nit: a single whitespace between TYPE_CODE and '(', TYPE_CODE (t) ...
I think this macro could have a more helpful name. It's very close to
TYPE_REFERENCE_TYPE. How about TYPE_IS_REFERENCE (similarly named to
TYPE_IS_OPAQUE)? Then the comment becomes even clearer: "True if this
type represents either an lvalue or rvalue reference type."

>  /* * Instruction-space delimited type.  This is for Harvard architectures
>     which have separate instruction and data address spaces (and perhaps
>     others).
> @@ -767,6 +775,10 @@ struct type
>  
>    struct type *reference_type;
>  
> +  /* * A C++ rvalue reference type added in C++0x. */
> +
> +  struct type *rvalue_reference_type;
> +

Why is this new field necessary? AFAICT, it is used exactly the same way
as the reference_type field above it, and whether a reference type is an
rvalue type is encoded into the type code.

>    /* * Variant chain.  This points to a type that differs from this
>       one only in qualifiers and length.  Currently, the possible
>       qualifiers are const, volatile, code-space, data-space, and
> @@ -1229,6 +1241,7 @@ extern void allocate_gnat_aux_type (struct type *);
>  #define TYPE_TARGET_TYPE(thistype) TYPE_MAIN_TYPE(thistype)->target_type
>  #define TYPE_POINTER_TYPE(thistype) (thistype)->pointer_type
>  #define TYPE_REFERENCE_TYPE(thistype) (thistype)->reference_type
> +#define TYPE_RVALUE_REFERENCE_TYPE(thistype) (thistype)->rvalue_reference_type
>  #define TYPE_CHAIN(thistype) (thistype)->chain
>  /* * Note that if thistype is a TYPEDEF type, you have to call check_typedef.
>     But check_typedef does set the TYPE_LENGTH of the TYPEDEF type,

If struct type.rvalue_reference_type is superfluous, this is unneeded.

Keith
  
Artemiy Volkov Feb. 23, 2016, 7:03 a.m. UTC | #2
On Fri, Feb 19, 2016 at 10:49:38AM -0800, Keith Seitz wrote:

[snip]

> >  /* * Instruction-space delimited type.  This is for Harvard architectures
> >     which have separate instruction and data address spaces (and perhaps
> >     others).
> > @@ -767,6 +775,10 @@ struct type
> >  
> >    struct type *reference_type;
> >  
> > +  /* * A C++ rvalue reference type added in C++0x. */
> > +
> > +  struct type *rvalue_reference_type;
> > +
> 
> Why is this new field necessary? AFAICT, it is used exactly the same way
> as the reference_type field above it, and whether a reference type is an
> rvalue type is encoded into the type code.

Do you suggest keeping only the lvalue version of the reference type and
then adjust its type code from TYPE_CODE_REF to TYPE_CODE_RVALUE_REF on
lookup_rvalue_reference_type()? It seems somewhat hacky to me. E.g. how
would we be able to create a struct type for a complex type involving a
T&&, such as a typedef of it?

> 
> >    /* * Variant chain.  This points to a type that differs from this
> >       one only in qualifiers and length.  Currently, the possible
> >       qualifiers are const, volatile, code-space, data-space, and
> > @@ -1229,6 +1241,7 @@ extern void allocate_gnat_aux_type (struct type *);
> >  #define TYPE_TARGET_TYPE(thistype) TYPE_MAIN_TYPE(thistype)->target_type
> >  #define TYPE_POINTER_TYPE(thistype) (thistype)->pointer_type
> >  #define TYPE_REFERENCE_TYPE(thistype) (thistype)->reference_type
> > +#define TYPE_RVALUE_REFERENCE_TYPE(thistype) (thistype)->rvalue_reference_type
> >  #define TYPE_CHAIN(thistype) (thistype)->chain
> >  /* * Note that if thistype is a TYPEDEF type, you have to call check_typedef.
> >     But check_typedef does set the TYPE_LENGTH of the TYPEDEF type,
> 
> If struct type.rvalue_reference_type is superfluous, this is unneeded.
> 
> Keith
>
  
Keith Seitz Feb. 25, 2016, 1:22 a.m. UTC | #3
On 02/22/2016 11:03 PM, Artemiy Volkov wrote:
> On Fri, Feb 19, 2016 at 10:49:38AM -0800, Keith Seitz wrote:
> 
> Do you suggest keeping only the lvalue version of the reference type and
> then adjust its type code from TYPE_CODE_REF to TYPE_CODE_RVALUE_REF on
> lookup_rvalue_reference_type()? It seems somewhat hacky to me. E.g. how
> would we be able to create a struct type for a complex type involving a
> T&&, such as a typedef of it?

Bah. No, I am wrong. Please disregard this comment.

Sorry about the confusion.

Keith
  
Artemiy Volkov Feb. 25, 2016, 1:31 a.m. UTC | #4
On Wed, Feb 24, 2016 at 05:22:18PM -0800, Keith Seitz wrote:
> On 02/22/2016 11:03 PM, Artemiy Volkov wrote:
> > On Fri, Feb 19, 2016 at 10:49:38AM -0800, Keith Seitz wrote:
> > 
> > Do you suggest keeping only the lvalue version of the reference type and
> > then adjust its type code from TYPE_CODE_REF to TYPE_CODE_RVALUE_REF on
> > lookup_rvalue_reference_type()? It seems somewhat hacky to me. E.g. how
> > would we be able to create a struct type for a complex type involving a
> > T&&, such as a typedef of it?
> 
> Bah. No, I am wrong. Please disregard this comment.
> 
> Sorry about the confusion.

No problem at all. Correct me if I am wrong, but I think this
invalidates your remarks in 2/11 that refer to this change. Do those 2
hunks in 2/11 look OK now?

> 
> Keith

Thanks,
Artemiy
  
Keith Seitz Feb. 25, 2016, 1:41 a.m. UTC | #5
[bah -- forgot to "reply all"]

On 02/24/2016 05:31 PM, Artemiy Volkov wrote:
> On Wed, Feb 24, 2016 at 05:22:18PM -0800, Keith Seitz wrote:
>> On 02/22/2016 11:03 PM, Artemiy Volkov wrote:
>>> On Fri, Feb 19, 2016 at 10:49:38AM -0800, Keith Seitz wrote:
>>>
>>> Do you suggest keeping only the lvalue version of the reference type and
>>> then adjust its type code from TYPE_CODE_REF to TYPE_CODE_RVALUE_REF on
>>> lookup_rvalue_reference_type()? It seems somewhat hacky to me. E.g. how
>>> would we be able to create a struct type for a complex type involving a
>>> T&&, such as a typedef of it?
>>
>> Bah. No, I am wrong. Please disregard this comment.
>>
>> Sorry about the confusion.
> 
> No problem at all. Correct me if I am wrong, but I think this
> invalidates your remarks in 2/11 that refer to this change. Do those 2
> hunks in 2/11 look OK now?

Yes, ignore my comments in make_reference_type. The only thing of note
in there, then, is that we prefer the usage of explicit NULL checks:

  if (*reftype != NULL)
    ...

instead of

  if (!*reftype)
    ...

Keith
  
Artemiy Volkov Feb. 25, 2016, 1:45 a.m. UTC | #6
On Wed, Feb 24, 2016 at 05:41:09PM -0800, Keith Seitz wrote:
> [bah -- forgot to "reply all"]
> 
> On 02/24/2016 05:31 PM, Artemiy Volkov wrote:
> > On Wed, Feb 24, 2016 at 05:22:18PM -0800, Keith Seitz wrote:
> >> On 02/22/2016 11:03 PM, Artemiy Volkov wrote:
> >>> On Fri, Feb 19, 2016 at 10:49:38AM -0800, Keith Seitz wrote:
> >>>
> >>> Do you suggest keeping only the lvalue version of the reference type and
> >>> then adjust its type code from TYPE_CODE_REF to TYPE_CODE_RVALUE_REF on
> >>> lookup_rvalue_reference_type()? It seems somewhat hacky to me. E.g. how
> >>> would we be able to create a struct type for a complex type involving a
> >>> T&&, such as a typedef of it?
> >>
> >> Bah. No, I am wrong. Please disregard this comment.
> >>
> >> Sorry about the confusion.
> > 
> > No problem at all. Correct me if I am wrong, but I think this
> > invalidates your remarks in 2/11 that refer to this change. Do those 2
> > hunks in 2/11 look OK now?
> 
> Yes, ignore my comments in make_reference_type. The only thing of note
> in there, then, is that we prefer the usage of explicit NULL checks:
> 
>   if (*reftype != NULL)
>     ...
> 
> instead of
> 
>   if (!*reftype)
>     ...

Gotcha. I'll make sure to fix that.

> 
> Keith

Thanks,
Artemiy
  

Patch

diff --git a/gdb/gdbtypes.h b/gdb/gdbtypes.h
index e775a1d..52419b4 100644
--- a/gdb/gdbtypes.h
+++ b/gdb/gdbtypes.h
@@ -160,6 +160,8 @@  enum type_code
 
     TYPE_CODE_REF,		/**< C++ Reference types */
 
+    TYPE_CODE_RVALUE_REF,	/**< C++ rvalue reference types */
+
     TYPE_CODE_CHAR,		/**< *real* character type */
 
     /* * Boolean type.  0 is false, 1 is true, and other values are
@@ -362,6 +364,12 @@  enum type_instance_flag_value
 #define TYPE_ATOMIC(t) \
   (TYPE_INSTANCE_FLAGS (t) & TYPE_INSTANCE_FLAG_ATOMIC)
 
+/* * C++ lvalue and rvalue references are equivalent in many contexts,
+   thus create a convenience macro that checks if a type is either of them. */
+
+#define TYPE_REFERENCE(t) \
+  (TYPE_CODE(t) == TYPE_CODE_REF || TYPE_CODE(t) == TYPE_CODE_RVALUE_REF)
+
 /* * Instruction-space delimited type.  This is for Harvard architectures
    which have separate instruction and data address spaces (and perhaps
    others).
@@ -767,6 +775,10 @@  struct type
 
   struct type *reference_type;
 
+  /* * A C++ rvalue reference type added in C++0x. */
+
+  struct type *rvalue_reference_type;
+
   /* * Variant chain.  This points to a type that differs from this
      one only in qualifiers and length.  Currently, the possible
      qualifiers are const, volatile, code-space, data-space, and
@@ -1229,6 +1241,7 @@  extern void allocate_gnat_aux_type (struct type *);
 #define TYPE_TARGET_TYPE(thistype) TYPE_MAIN_TYPE(thistype)->target_type
 #define TYPE_POINTER_TYPE(thistype) (thistype)->pointer_type
 #define TYPE_REFERENCE_TYPE(thistype) (thistype)->reference_type
+#define TYPE_RVALUE_REFERENCE_TYPE(thistype) (thistype)->rvalue_reference_type
 #define TYPE_CHAIN(thistype) (thistype)->chain
 /* * Note that if thistype is a TYPEDEF type, you have to call check_typedef.
    But check_typedef does set the TYPE_LENGTH of the TYPEDEF type,