| /* Definitions for Rust 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 RUST_EXP_H | 
 | #define RUST_EXP_H | 
 |  | 
 | #include "expop.h" | 
 |  | 
 | extern struct value *eval_op_rust_complement (struct type *expect_type, | 
 | 					      struct expression *exp, | 
 | 					      enum noside noside, | 
 | 					      enum exp_opcode opcode, | 
 | 					      struct value *value); | 
 | extern struct value *eval_op_rust_array (struct type *expect_type, | 
 | 					 struct expression *exp, | 
 | 					 enum noside noside, | 
 | 					 enum exp_opcode opcode, | 
 | 					 struct value *ncopies, | 
 | 					 struct value *elt); | 
 | extern struct value *rust_subscript (struct type *expect_type, | 
 | 				     struct expression *exp, | 
 | 				     enum noside noside, bool for_addr, | 
 | 				     struct value *lhs, struct value *rhs); | 
 | extern struct value *rust_range (struct type *expect_type, | 
 | 				 struct expression *exp, | 
 | 				 enum noside noside, enum range_flag kind, | 
 | 				 struct value *low, struct value *high); | 
 |  | 
 | namespace expr | 
 | { | 
 |  | 
 | using rust_unop_compl_operation = unop_operation<UNOP_COMPLEMENT, | 
 | 						  eval_op_rust_complement>; | 
 | using rust_array_operation = binop_operation<OP_RUST_ARRAY, | 
 | 					     eval_op_rust_array>; | 
 |  | 
 | /* The Rust indirection operation.  */ | 
 | class rust_unop_ind_operation | 
 |   : public unop_ind_operation | 
 | { | 
 | public: | 
 |  | 
 |   using unop_ind_operation::unop_ind_operation; | 
 |  | 
 |   value *evaluate (struct type *expect_type, | 
 | 		   struct expression *exp, | 
 | 		   enum noside noside) override; | 
 | }; | 
 |  | 
 | /* Subscript operator for Rust.  */ | 
 | class rust_subscript_operation | 
 |   : public tuple_holding_operation<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 | 
 |   { | 
 |     value *arg1 = std::get<0> (m_storage)->evaluate (nullptr, exp, noside); | 
 |     value *arg2 = std::get<1> (m_storage)->evaluate (nullptr, exp, noside); | 
 |     return rust_subscript (expect_type, exp, noside, false, arg1, arg2); | 
 |   } | 
 |  | 
 |   value *slice (struct type *expect_type, | 
 | 		struct expression *exp, | 
 | 		enum noside noside) | 
 |   { | 
 |     value *arg1 = std::get<0> (m_storage)->evaluate (nullptr, exp, noside); | 
 |     value *arg2 = std::get<1> (m_storage)->evaluate (nullptr, exp, noside); | 
 |     return rust_subscript (expect_type, exp, noside, true, arg1, arg2); | 
 |   } | 
 |  | 
 |   enum exp_opcode opcode () const override | 
 |   { return BINOP_SUBSCRIPT; } | 
 | }; | 
 |  | 
 | class rust_unop_addr_operation | 
 |   : public tuple_holding_operation<operation_up> | 
 | { | 
 | public: | 
 |  | 
 |   using tuple_holding_operation::tuple_holding_operation; | 
 |  | 
 |   value *evaluate (struct type *expect_type, | 
 | 		   struct expression *exp, | 
 | 		   enum noside noside) override | 
 |   { | 
 |     operation *oper = std::get<0> (m_storage).get (); | 
 |     rust_subscript_operation *sub_op | 
 |       = dynamic_cast<rust_subscript_operation *> (oper); | 
 |     if (sub_op != nullptr) | 
 |       return sub_op->slice (expect_type, exp, noside); | 
 |     return oper->evaluate_for_address (exp, noside); | 
 |   } | 
 |  | 
 |   enum exp_opcode opcode () const override | 
 |   { return UNOP_ADDR; } | 
 | }; | 
 |  | 
 | /* The Rust range operators.  */ | 
 | class rust_range_operation | 
 |   : public tuple_holding_operation<enum range_flag, 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 | 
 |   { | 
 |     auto kind = std::get<0> (m_storage); | 
 |     value *low = nullptr; | 
 |     if (std::get<1> (m_storage) != nullptr) | 
 |       low = std::get<1> (m_storage)->evaluate (nullptr, exp, noside); | 
 |     value *high = nullptr; | 
 |     if (std::get<2> (m_storage) != nullptr) | 
 |       high = std::get<2> (m_storage)->evaluate (nullptr, exp, noside); | 
 |     return rust_range (expect_type, exp, noside, kind, low, high); | 
 |   } | 
 |  | 
 |   enum exp_opcode opcode () const override | 
 |   { return OP_RANGE; } | 
 | }; | 
 |  | 
 | /* Tuple field reference (using an integer).  */ | 
 | class rust_struct_anon | 
 |   : public tuple_holding_operation<int, 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 STRUCTOP_ANONYMOUS; } | 
 | }; | 
 |  | 
 | /* Structure (or union or enum) field reference.  */ | 
 | class rust_structop | 
 |   : 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; | 
 |  | 
 |   value *evaluate_funcall (struct type *expect_type, | 
 | 			   struct expression *exp, | 
 | 			   enum noside noside, | 
 | 			   const std::vector<operation_up> &args) override; | 
 |  | 
 |   enum exp_opcode opcode () const override | 
 |   { return STRUCTOP_STRUCT; } | 
 | }; | 
 |  | 
 | /* Rust aggregate initialization.  */ | 
 | class rust_aggregate_operation | 
 |   : public tuple_holding_operation<struct type *, operation_up, | 
 | 				   std::vector<std::pair<std::string, | 
 | 							 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_AGGREGATE; } | 
 | }; | 
 |  | 
 | /* Rust parenthesized operation.  This is needed to distinguish | 
 |    between 'obj.f()', which is a method call, and '(obj.f)()', which | 
 |    is a call of a function-valued field 'f'.  */ | 
 | class rust_parenthesized_operation | 
 |   : public tuple_holding_operation<operation_up> | 
 | { | 
 | public: | 
 |  | 
 |   explicit rust_parenthesized_operation (operation_up op) | 
 |     : tuple_holding_operation (std::move (op)) | 
 |   { | 
 |   } | 
 |  | 
 |   value *evaluate (struct type *expect_type, | 
 | 		   struct expression *exp, | 
 | 		   enum noside noside) override | 
 |   { | 
 |     return std::get<0> (m_storage)->evaluate (expect_type, exp, noside); | 
 |   } | 
 |  | 
 |   enum exp_opcode opcode () const override | 
 |   { | 
 |     /* A lie but this isn't worth introducing a new opcode for.  */ | 
 |     return UNOP_PLUS; | 
 |   } | 
 | }; | 
 |  | 
 | } /* namespace expr */ | 
 |  | 
 | #endif /* RUST_EXP_H */ |