|  | /* 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 (lhs->type (), 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 */ |