diff options
author | Tom Tromey <tom@tromey.com> | 2021-03-08 07:27:57 -0700 |
---|---|---|
committer | Tom Tromey <tom@tromey.com> | 2021-03-08 07:28:28 -0700 |
commit | 5019124b1ddc3839bd8ccba08819b11c0151e8a9 (patch) | |
tree | e83820e9ee3c77bdff1892dfbc7b2664c26d0df6 | |
parent | 2bc9b40ce16a109a22320589d2cfb9fced5fb769 (diff) | |
download | gdb-5019124b1ddc3839bd8ccba08819b11c0151e8a9.zip gdb-5019124b1ddc3839bd8ccba08819b11c0151e8a9.tar.gz gdb-5019124b1ddc3839bd8ccba08819b11c0151e8a9.tar.bz2 |
Implement the "&&" and "||" operators
This implements the "&&" and "||" operators.
gdb/ChangeLog
2021-03-08 Tom Tromey <tom@tromey.com>
* expop.h (class logical_and_operation)
(class logical_or_operation): New.
* eval.c (logical_and_operation::evaluate)
(logical_or_operation::evaluate): New methods.
* ax-gdb.c (logical_and_operation::do_generate_ax)
(logical_or_operation::do_generate_ax): New methods.
-rw-r--r-- | gdb/ChangeLog | 9 | ||||
-rw-r--r-- | gdb/ax-gdb.c | 56 | ||||
-rw-r--r-- | gdb/eval.c | 63 | ||||
-rw-r--r-- | gdb/expop.h | 48 |
4 files changed, 176 insertions, 0 deletions
diff --git a/gdb/ChangeLog b/gdb/ChangeLog index fcacb6d..f91477a 100644 --- a/gdb/ChangeLog +++ b/gdb/ChangeLog @@ -1,5 +1,14 @@ 2021-03-08 Tom Tromey <tom@tromey.com> + * expop.h (class logical_and_operation) + (class logical_or_operation): New. + * eval.c (logical_and_operation::evaluate) + (logical_or_operation::evaluate): New methods. + * ax-gdb.c (logical_and_operation::do_generate_ax) + (logical_or_operation::do_generate_ax): New methods. + +2021-03-08 Tom Tromey <tom@tromey.com> + * m2-lang.c (eval_op_m2_high, eval_op_m2_subscript): No longer static. * m2-exp.h: New file. diff --git a/gdb/ax-gdb.c b/gdb/ax-gdb.c index bdf4a6d..8ec3dd7 100644 --- a/gdb/ax-gdb.c +++ b/gdb/ax-gdb.c @@ -2693,6 +2693,62 @@ var_value_operation::do_generate_ax (struct expression *exp, } } +void +logical_and_operation::do_generate_ax (struct expression *exp, + struct agent_expr *ax, + struct axs_value *value, + struct type *cast_type) +{ + struct axs_value value1, value2; + int if1, go1, if2, go2, end; + + /* Generate the obvious sequence of tests and jumps. */ + std::get<0> (m_storage)->generate_ax (exp, ax, &value1); + gen_usual_unary (ax, &value1); + if1 = ax_goto (ax, aop_if_goto); + go1 = ax_goto (ax, aop_goto); + ax_label (ax, if1, ax->len); + std::get<1> (m_storage)->generate_ax (exp, ax, &value2); + gen_usual_unary (ax, &value2); + if2 = ax_goto (ax, aop_if_goto); + go2 = ax_goto (ax, aop_goto); + ax_label (ax, if2, ax->len); + ax_const_l (ax, 1); + end = ax_goto (ax, aop_goto); + ax_label (ax, go1, ax->len); + ax_label (ax, go2, ax->len); + ax_const_l (ax, 0); + ax_label (ax, end, ax->len); + value->kind = axs_rvalue; + value->type = builtin_type (ax->gdbarch)->builtin_int; +} + +void +logical_or_operation::do_generate_ax (struct expression *exp, + struct agent_expr *ax, + struct axs_value *value, + struct type *cast_type) +{ + struct axs_value value1, value2; + int if1, if2, end; + + /* Generate the obvious sequence of tests and jumps. */ + std::get<0> (m_storage)->generate_ax (exp, ax, &value1); + gen_usual_unary (ax, &value1); + if1 = ax_goto (ax, aop_if_goto); + std::get<1> (m_storage)->generate_ax (exp, ax, &value2); + gen_usual_unary (ax, &value2); + if2 = ax_goto (ax, aop_if_goto); + ax_const_l (ax, 0); + end = ax_goto (ax, aop_goto); + ax_label (ax, if1, ax->len); + ax_label (ax, if2, ax->len); + ax_const_l (ax, 1); + ax_label (ax, end, ax->len); + value->kind = axs_rvalue; + value->type = builtin_type (ax->gdbarch)->builtin_int; +} + } /* This handles the middle-to-right-side of code generation for binary @@ -2486,6 +2486,69 @@ multi_subscript_operation::evaluate (struct type *expect_type, gdb::make_array_view (argvec, values.size ())); } +value * +logical_and_operation::evaluate (struct type *expect_type, + struct expression *exp, + enum noside noside) +{ + value *arg1 = std::get<0> (m_storage)->evaluate (nullptr, exp, noside); + if (noside == EVAL_SKIP) + return eval_skip_value (exp); + + value *arg2 = std::get<1> (m_storage)->evaluate (nullptr, exp, + EVAL_AVOID_SIDE_EFFECTS); + + if (binop_user_defined_p (BINOP_LOGICAL_AND, arg1, arg2)) + { + arg2 = std::get<1> (m_storage)->evaluate (nullptr, exp, noside); + return value_x_binop (arg1, arg2, BINOP_LOGICAL_AND, OP_NULL, noside); + } + else + { + int tem = value_logical_not (arg1); + if (!tem) + { + arg2 = std::get<1> (m_storage)->evaluate (nullptr, exp, noside); + tem = value_logical_not (arg2); + } + struct type *type = language_bool_type (exp->language_defn, + exp->gdbarch); + return value_from_longest (type, !tem); + } +} + +value * +logical_or_operation::evaluate (struct type *expect_type, + struct expression *exp, + enum noside noside) +{ + value *arg1 = std::get<0> (m_storage)->evaluate (nullptr, exp, noside); + if (noside == EVAL_SKIP) + return eval_skip_value (exp); + + value *arg2 = std::get<1> (m_storage)->evaluate (nullptr, exp, + EVAL_AVOID_SIDE_EFFECTS); + + if (binop_user_defined_p (BINOP_LOGICAL_OR, arg1, arg2)) + { + arg2 = std::get<1> (m_storage)->evaluate (nullptr, exp, noside); + return value_x_binop (arg1, arg2, BINOP_LOGICAL_OR, OP_NULL, noside); + } + else + { + int tem = value_logical_not (arg1); + if (tem) + { + arg2 = std::get<1> (m_storage)->evaluate (nullptr, exp, noside); + tem = value_logical_not (arg2); + } + + struct type *type = language_bool_type (exp->language_defn, + exp->gdbarch); + return value_from_longest (type, !tem); + } +} + } struct value * diff --git a/gdb/expop.h b/gdb/expop.h index 6b1a875..7850c45 100644 --- a/gdb/expop.h +++ b/gdb/expop.h @@ -1982,6 +1982,54 @@ public: { return MULTI_SUBSCRIPT; } }; +/* The "&&" operator. */ +class logical_and_operation + : public maybe_constant_operation<operation_up, operation_up> +{ +public: + + using maybe_constant_operation::maybe_constant_operation; + + value *evaluate (struct type *expect_type, + struct expression *exp, + enum noside noside) override; + + enum exp_opcode opcode () const override + { return BINOP_LOGICAL_AND; } + +protected: + + void do_generate_ax (struct expression *exp, + struct agent_expr *ax, + struct axs_value *value, + struct type *cast_type) + override; +}; + +/* The "||" operator. */ +class logical_or_operation + : public maybe_constant_operation<operation_up, operation_up> +{ +public: + + using maybe_constant_operation::maybe_constant_operation; + + value *evaluate (struct type *expect_type, + struct expression *exp, + enum noside noside) override; + + enum exp_opcode opcode () const override + { return BINOP_LOGICAL_OR; } + +protected: + + void do_generate_ax (struct expression *exp, + struct agent_expr *ax, + struct axs_value *value, + struct type *cast_type) + override; +}; + } /* namespace expr */ #endif /* EXPOP_H */ |