| /* Definitions for C expressions |
| |
| Copyright (C) 2020-2023 Free Software Foundation, Inc. |
| |
| This file is part of GDB. |
| |
| 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/>. */ |
| |
| #ifndef C_EXP_H |
| #define C_EXP_H |
| |
| #include "expop.h" |
| #include "objc-lang.h" |
| |
| extern struct value *eval_op_objc_selector (struct type *expect_type, |
| struct expression *exp, |
| enum noside noside, |
| const char *sel); |
| extern struct value *opencl_value_cast (struct type *type, struct value *arg); |
| extern struct value * |
| eval_opencl_assign (struct type *expect_type, struct expression *exp, |
| enum noside noside, enum exp_opcode op, struct value *arg1, |
| struct value *arg2); |
| extern struct value *opencl_relop (struct type *expect_type, |
| struct expression *exp, enum noside noside, |
| enum exp_opcode op, struct value *arg1, |
| struct value *arg2); |
| extern struct value * |
| opencl_logical_not (struct type *expect_type, struct expression *exp, |
| enum noside noside, enum exp_opcode op, struct value *arg); |
| |
| namespace expr |
| { |
| |
| class c_string_operation |
| : public tuple_holding_operation<enum c_string_type_values, |
| std::vector<std::string>> |
| { |
| public: |
| |
| using tuple_holding_operation::tuple_holding_operation; |
| |
| value *evaluate (struct type *expect_type, struct expression *exp, |
| enum noside noside) override; |
| |
| enum exp_opcode opcode () const override { return OP_STRING; } |
| }; |
| |
| class objc_nsstring_operation : public tuple_holding_operation<std::string> |
| { |
| public: |
| |
| using tuple_holding_operation::tuple_holding_operation; |
| |
| value *evaluate (struct type *expect_type, struct expression *exp, |
| enum noside noside) override |
| { |
| const std::string &str = std::get<0> (m_storage); |
| return value_nsstring (exp->gdbarch, str.c_str (), str.size () + 1); |
| } |
| |
| enum exp_opcode opcode () const override { return OP_OBJC_NSSTRING; } |
| }; |
| |
| class objc_selector_operation : public tuple_holding_operation<std::string> |
| { |
| public: |
| |
| using tuple_holding_operation::tuple_holding_operation; |
| |
| value *evaluate (struct type *expect_type, struct expression *exp, |
| enum noside noside) override |
| { |
| return eval_op_objc_selector (expect_type, exp, noside, |
| std::get<0> (m_storage).c_str ()); |
| } |
| |
| enum exp_opcode opcode () const override { return OP_OBJC_SELECTOR; } |
| }; |
| |
| /* An Objective C message call. */ |
| class objc_msgcall_operation |
| : public tuple_holding_operation<CORE_ADDR, operation_up, |
| std::vector<operation_up>> |
| { |
| public: |
| |
| using tuple_holding_operation::tuple_holding_operation; |
| |
| value *evaluate (struct type *expect_type, struct expression *exp, |
| enum noside noside) override; |
| |
| enum exp_opcode opcode () const override { return OP_OBJC_MSGCALL; } |
| }; |
| |
| using opencl_cast_type_operation |
| = cxx_cast_operation<UNOP_CAST_TYPE, opencl_value_cast>; |
| |
| /* Binary operations, as needed for OpenCL. */ |
| template<enum exp_opcode OP, binary_ftype FUNC, |
| typename BASE = maybe_constant_operation<operation_up, operation_up>> |
| class opencl_binop_operation : public BASE |
| { |
| public: |
| |
| using BASE::BASE; |
| |
| value *evaluate (struct type *expect_type, struct expression *exp, |
| enum noside noside) override |
| { |
| value *lhs |
| = std::get<0> (this->m_storage)->evaluate (nullptr, exp, noside); |
| value *rhs = std::get<1> (this->m_storage) |
| ->evaluate (value_type (lhs), exp, noside); |
| return FUNC (expect_type, exp, noside, OP, lhs, rhs); |
| } |
| |
| enum exp_opcode opcode () const override { return OP; } |
| }; |
| |
| using opencl_assign_operation |
| = opencl_binop_operation<BINOP_ASSIGN, eval_opencl_assign, assign_operation>; |
| using opencl_equal_operation |
| = opencl_binop_operation<BINOP_EQUAL, opencl_relop>; |
| using opencl_notequal_operation |
| = opencl_binop_operation<BINOP_NOTEQUAL, opencl_relop>; |
| using opencl_less_operation = opencl_binop_operation<BINOP_LESS, opencl_relop>; |
| using opencl_gtr_operation = opencl_binop_operation<BINOP_GTR, opencl_relop>; |
| using opencl_geq_operation = opencl_binop_operation<BINOP_GEQ, opencl_relop>; |
| using opencl_leq_operation = opencl_binop_operation<BINOP_LEQ, opencl_relop>; |
| |
| using opencl_not_operation |
| = unop_operation<UNOP_LOGICAL_NOT, opencl_logical_not>; |
| |
| /* STRUCTOP_STRUCT implementation for OpenCL. */ |
| class opencl_structop_operation : public structop_base_operation |
| { |
| public: |
| |
| using structop_base_operation::structop_base_operation; |
| |
| value *evaluate (struct type *expect_type, struct expression *exp, |
| enum noside noside) override; |
| |
| enum exp_opcode opcode () const override { return STRUCTOP_STRUCT; } |
| }; |
| |
| /* This handles the "&&" and "||" operations for OpenCL. */ |
| class opencl_logical_binop_operation |
| : public tuple_holding_operation<enum exp_opcode, operation_up, operation_up> |
| { |
| public: |
| |
| using tuple_holding_operation::tuple_holding_operation; |
| |
| value *evaluate (struct type *expect_type, struct expression *exp, |
| enum noside noside) override; |
| |
| enum exp_opcode opcode () const override { return std::get<0> (m_storage); } |
| }; |
| |
| /* The ?: ternary operator for OpenCL. */ |
| class opencl_ternop_cond_operation |
| : public tuple_holding_operation<operation_up, operation_up, operation_up> |
| { |
| public: |
| |
| using tuple_holding_operation::tuple_holding_operation; |
| |
| value *evaluate (struct type *expect_type, struct expression *exp, |
| enum noside noside) override; |
| |
| enum exp_opcode opcode () const override { return TERNOP_COND; } |
| }; |
| |
| } /* namespace expr */ |
| |
| #endif /* C_EXP_H */ |