mirror of
https://github.com/espressif/binutils-gdb.git
synced 2025-05-31 18:20:12 +08:00
Fix c++/16675 -- sizeof reference type should give the size of
the referent, not the size of the actual reference variable.
This commit is contained in:
@ -1,3 +1,10 @@
|
|||||||
|
2014-04-11 Keith Seitz <keiths@redhat.com>
|
||||||
|
|
||||||
|
PR c++/16675
|
||||||
|
* c-exp.y (exp : SIZEOF '(' type ')'): Handle reference types.
|
||||||
|
* eval.c (evaluate_subexp_for_sizeof): Refactor and handle
|
||||||
|
reference types.
|
||||||
|
|
||||||
2014-04-11 Sanimir Agovic <sanimir.agovic@intel.com>
|
2014-04-11 Sanimir Agovic <sanimir.agovic@intel.com>
|
||||||
|
|
||||||
* eval.c (evaluate_subexp_for_sizeof): Add enum noside argument.
|
* eval.c (evaluate_subexp_for_sizeof): Add enum noside argument.
|
||||||
|
14
gdb/c-exp.y
14
gdb/c-exp.y
@ -787,14 +787,22 @@ exp : SELECTOR '(' name ')'
|
|||||||
;
|
;
|
||||||
|
|
||||||
exp : SIZEOF '(' type ')' %prec UNARY
|
exp : SIZEOF '(' type ')' %prec UNARY
|
||||||
{ write_exp_elt_opcode (pstate, OP_LONG);
|
{ struct type *type = $3;
|
||||||
|
write_exp_elt_opcode (pstate, OP_LONG);
|
||||||
write_exp_elt_type (pstate, lookup_signed_typename
|
write_exp_elt_type (pstate, lookup_signed_typename
|
||||||
(parse_language (pstate),
|
(parse_language (pstate),
|
||||||
parse_gdbarch (pstate),
|
parse_gdbarch (pstate),
|
||||||
"int"));
|
"int"));
|
||||||
CHECK_TYPEDEF ($3);
|
CHECK_TYPEDEF (type);
|
||||||
|
|
||||||
|
/* $5.3.3/2 of the C++ Standard (n3290 draft)
|
||||||
|
says of sizeof: "When applied to a reference
|
||||||
|
or a reference type, the result is the size of
|
||||||
|
the referenced type." */
|
||||||
|
if (TYPE_CODE (type) == TYPE_CODE_REF)
|
||||||
|
type = check_typedef (TYPE_TARGET_TYPE (type));
|
||||||
write_exp_elt_longcst (pstate,
|
write_exp_elt_longcst (pstate,
|
||||||
(LONGEST) TYPE_LENGTH ($3));
|
(LONGEST) TYPE_LENGTH (type));
|
||||||
write_exp_elt_opcode (pstate, OP_LONG); }
|
write_exp_elt_opcode (pstate, OP_LONG); }
|
||||||
;
|
;
|
||||||
|
|
||||||
|
33
gdb/eval.c
33
gdb/eval.c
@ -3030,21 +3030,22 @@ evaluate_subexp_for_sizeof (struct expression *exp, int *pos,
|
|||||||
&& TYPE_CODE (type) != TYPE_CODE_REF
|
&& TYPE_CODE (type) != TYPE_CODE_REF
|
||||||
&& TYPE_CODE (type) != TYPE_CODE_ARRAY)
|
&& TYPE_CODE (type) != TYPE_CODE_ARRAY)
|
||||||
error (_("Attempt to take contents of a non-pointer value."));
|
error (_("Attempt to take contents of a non-pointer value."));
|
||||||
type = check_typedef (TYPE_TARGET_TYPE (type));
|
|
||||||
if (is_dynamic_type (type))
|
if (is_dynamic_type (type))
|
||||||
type = value_type (value_ind (val));
|
type = value_type (value_ind (val));
|
||||||
return value_from_longest (size_type, (LONGEST) TYPE_LENGTH (type));
|
else
|
||||||
|
type = TYPE_TARGET_TYPE (type);
|
||||||
|
break;
|
||||||
|
|
||||||
case UNOP_MEMVAL:
|
case UNOP_MEMVAL:
|
||||||
(*pos) += 3;
|
(*pos) += 3;
|
||||||
type = check_typedef (exp->elts[pc + 1].type);
|
type = exp->elts[pc + 1].type;
|
||||||
return value_from_longest (size_type, (LONGEST) TYPE_LENGTH (type));
|
break;
|
||||||
|
|
||||||
case UNOP_MEMVAL_TYPE:
|
case UNOP_MEMVAL_TYPE:
|
||||||
(*pos) += 1;
|
(*pos) += 1;
|
||||||
val = evaluate_subexp (NULL, exp, pos, EVAL_AVOID_SIDE_EFFECTS);
|
val = evaluate_subexp (NULL, exp, pos, EVAL_AVOID_SIDE_EFFECTS);
|
||||||
type = check_typedef (value_type (val));
|
type = value_type (val);
|
||||||
return value_from_longest (size_type, (LONGEST) TYPE_LENGTH (type));
|
break;
|
||||||
|
|
||||||
case OP_VAR_VALUE:
|
case OP_VAR_VALUE:
|
||||||
type = check_typedef (SYMBOL_TYPE (exp->elts[pc + 2].symbol));
|
type = check_typedef (SYMBOL_TYPE (exp->elts[pc + 2].symbol));
|
||||||
@ -3055,8 +3056,7 @@ evaluate_subexp_for_sizeof (struct expression *exp, int *pos,
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
(*pos) += 4;
|
(*pos) += 4;
|
||||||
return
|
break;
|
||||||
value_from_longest (size_type, (LONGEST) TYPE_LENGTH (type));
|
|
||||||
|
|
||||||
/* Deal with the special case if NOSIDE is EVAL_NORMAL and the resulting
|
/* Deal with the special case if NOSIDE is EVAL_NORMAL and the resulting
|
||||||
type of the subscript is a variable length array type. In this case we
|
type of the subscript is a variable length array type. In this case we
|
||||||
@ -3080,8 +3080,8 @@ evaluate_subexp_for_sizeof (struct expression *exp, int *pos,
|
|||||||
if (TYPE_RANGE_DATA (type)->flag_bound_evaluated)
|
if (TYPE_RANGE_DATA (type)->flag_bound_evaluated)
|
||||||
{
|
{
|
||||||
val = evaluate_subexp (NULL_TYPE, exp, pos, EVAL_NORMAL);
|
val = evaluate_subexp (NULL_TYPE, exp, pos, EVAL_NORMAL);
|
||||||
return value_from_longest
|
type = value_type (val);
|
||||||
(size_type, (LONGEST) TYPE_LENGTH (value_type (val)));
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -3091,9 +3091,18 @@ evaluate_subexp_for_sizeof (struct expression *exp, int *pos,
|
|||||||
|
|
||||||
default:
|
default:
|
||||||
val = evaluate_subexp (NULL_TYPE, exp, pos, EVAL_AVOID_SIDE_EFFECTS);
|
val = evaluate_subexp (NULL_TYPE, exp, pos, EVAL_AVOID_SIDE_EFFECTS);
|
||||||
return value_from_longest (size_type,
|
type = value_type (val);
|
||||||
(LONGEST) TYPE_LENGTH (value_type (val)));
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* $5.3.3/2 of the C++ Standard (n3290 draft) says of sizeof:
|
||||||
|
"When applied to a reference or a reference type, the result is
|
||||||
|
the size of the referenced type." */
|
||||||
|
CHECK_TYPEDEF (type);
|
||||||
|
if (exp->language_defn->la_language == language_cplus
|
||||||
|
&& TYPE_CODE (type) == TYPE_CODE_REF)
|
||||||
|
type = check_typedef (TYPE_TARGET_TYPE (type));
|
||||||
|
return value_from_longest (size_type, (LONGEST) TYPE_LENGTH (type));
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Parse a type expression in the string [P..P+LENGTH). */
|
/* Parse a type expression in the string [P..P+LENGTH). */
|
||||||
|
@ -1,3 +1,9 @@
|
|||||||
|
2014-04-11 Keith Seitz <keiths@redhat.com>
|
||||||
|
|
||||||
|
PR c++/16675
|
||||||
|
* gdb.cp/cpsizeof.exp: New file.
|
||||||
|
* gdb.cp/cpsizeof.cc: New file.
|
||||||
|
|
||||||
2014-04-11 Sanimir Agovic <sanimir.agovic@intel.com>
|
2014-04-11 Sanimir Agovic <sanimir.agovic@intel.com>
|
||||||
|
|
||||||
* mi-vla-c99.exp: New file.
|
* mi-vla-c99.exp: New file.
|
||||||
|
71
gdb/testsuite/gdb.cp/cpsizeof.cc
Normal file
71
gdb/testsuite/gdb.cp/cpsizeof.cc
Normal file
@ -0,0 +1,71 @@
|
|||||||
|
/* This testcase is part of GDB, the GNU debugger.
|
||||||
|
|
||||||
|
Copyright 2014 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/>. */
|
||||||
|
|
||||||
|
struct Class
|
||||||
|
{
|
||||||
|
int a;
|
||||||
|
char b;
|
||||||
|
long c;
|
||||||
|
|
||||||
|
Class () : a (1), b ('2'), c (3) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
union Union
|
||||||
|
{
|
||||||
|
Class *kp;
|
||||||
|
char a;
|
||||||
|
int b;
|
||||||
|
long c;
|
||||||
|
};
|
||||||
|
|
||||||
|
enum Enum { A, B, C, D };
|
||||||
|
|
||||||
|
typedef unsigned char a4[4];
|
||||||
|
typedef unsigned char a8[8];
|
||||||
|
typedef unsigned char a12[12];
|
||||||
|
typedef Class c4[4];
|
||||||
|
typedef Union u8[8];
|
||||||
|
typedef Enum e12[12];
|
||||||
|
|
||||||
|
#define T(N) \
|
||||||
|
N N ## obj; \
|
||||||
|
N& N ## _ref = N ## obj; \
|
||||||
|
N* N ## p = &(N ## obj); \
|
||||||
|
N*& N ## p_ref = N ## p; \
|
||||||
|
int size_ ## N = sizeof (N ## _ref); \
|
||||||
|
int size_ ## N ## p = sizeof (N ## p_ref); \
|
||||||
|
|
||||||
|
int
|
||||||
|
main (void)
|
||||||
|
{
|
||||||
|
T (char);
|
||||||
|
T (int);
|
||||||
|
T (long);
|
||||||
|
T (float);
|
||||||
|
T (double);
|
||||||
|
T (a4);
|
||||||
|
T (a8);
|
||||||
|
T (a12);
|
||||||
|
T (Class);
|
||||||
|
T (Union);
|
||||||
|
T (Enum);
|
||||||
|
T (c4);
|
||||||
|
T (u8);
|
||||||
|
T (e12);
|
||||||
|
|
||||||
|
return 0; /* break here */
|
||||||
|
}
|
40
gdb/testsuite/gdb.cp/cpsizeof.exp
Normal file
40
gdb/testsuite/gdb.cp/cpsizeof.exp
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
# sizeof() tests [c++/16675]
|
||||||
|
# Copyright 2014 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/>.
|
||||||
|
|
||||||
|
standard_testfile .cc
|
||||||
|
|
||||||
|
if {[skip_cplus_tests]} { continue }
|
||||||
|
|
||||||
|
if {[prepare_for_testing ${testfile}.exp $testfile $srcfile {debug c++}] } {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
|
||||||
|
if {![runto_main]} {
|
||||||
|
perror "could not run to main"
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
gdb_breakpoint [gdb_get_line_number "break here"]
|
||||||
|
gdb_continue_to_breakpoint "break here"
|
||||||
|
|
||||||
|
# Compare sizeof from the compiler and gdb. Do this once with the actual
|
||||||
|
# type name and once with a reference variable.
|
||||||
|
foreach v {char int long float double a4 a8 a12 Class Union Enum c4 u8 e12} {
|
||||||
|
gdb_test "print size_$v == sizeof (${v}&)" "= true"
|
||||||
|
gdb_test "print size_$v == sizeof (${v}_ref)" "= true"
|
||||||
|
gdb_test "print size_${v}p == sizeof (${v}*&)" "= true"
|
||||||
|
gdb_test "print size_${v}p == sizeof (${v}p_ref)" "= true"
|
||||||
|
}
|
Reference in New Issue
Block a user