|  | /* Definitions for Fortran 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 FORTRAN_EXP_H | 
|  | #define FORTRAN_EXP_H | 
|  |  | 
|  | #include "expop.h" | 
|  |  | 
|  | extern struct value *eval_op_f_abs (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode opcode, | 
|  | struct value *arg1); | 
|  | extern struct value *eval_op_f_mod (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode opcode, | 
|  | struct value *arg1, struct value *arg2); | 
|  |  | 
|  | /* Implement expression evaluation for Fortran's CEILING intrinsic function | 
|  | called with one argument.  For EXPECT_TYPE, EXP, and NOSIDE see | 
|  | expression::evaluate (in expression.h).  OPCODE will always be | 
|  | FORTRAN_CEILING and ARG1 is the argument passed to CEILING.  */ | 
|  |  | 
|  | extern struct value *eval_op_f_ceil (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode opcode, | 
|  | struct value *arg1); | 
|  |  | 
|  | /* Implement expression evaluation for Fortran's CEILING intrinsic function | 
|  | called with two arguments.  For EXPECT_TYPE, EXP, and NOSIDE see | 
|  | expression::evaluate (in expression.h).  OPCODE will always be | 
|  | FORTRAN_CEILING, ARG1 is the first argument passed to CEILING, and KIND_ARG | 
|  | is the type corresponding to the KIND parameter passed to CEILING.  */ | 
|  |  | 
|  | extern value *eval_op_f_ceil (type *expect_type, expression *exp, | 
|  | noside noside, exp_opcode opcode, value *arg1, | 
|  | type *kind_arg); | 
|  |  | 
|  | /* Implement expression evaluation for Fortran's FLOOR intrinsic function | 
|  | called with one argument.  For EXPECT_TYPE, EXP, and NOSIDE see | 
|  | expression::evaluate (in expression.h).  OPCODE will always be FORTRAN_FLOOR | 
|  | and ARG1 is the argument passed to FLOOR.  */ | 
|  |  | 
|  | extern struct value *eval_op_f_floor (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode opcode, | 
|  | struct value *arg1); | 
|  |  | 
|  | /* Implement expression evaluation for Fortran's FLOOR intrinsic function | 
|  | called with two arguments.  For EXPECT_TYPE, EXP, and NOSIDE see | 
|  | expression::evaluate (in expression.h).  OPCODE will always be | 
|  | FORTRAN_FLOOR, ARG1 is the first argument passed to FLOOR, and KIND_ARG is | 
|  | the type corresponding to the KIND parameter passed to FLOOR.  */ | 
|  |  | 
|  | extern value *eval_op_f_floor (type *expect_type, expression *exp, | 
|  | noside noside, exp_opcode opcode, value *arg1, | 
|  | type *kind_arg); | 
|  |  | 
|  | extern struct value *eval_op_f_modulo (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode opcode, | 
|  | struct value *arg1, struct value *arg2); | 
|  |  | 
|  | /* Implement expression evaluation for Fortran's CMPLX intrinsic function | 
|  | called with one argument.  For EXPECT_TYPE, EXP, and NOSIDE see | 
|  | expression::evaluate (in expression.h). OPCODE will always be | 
|  | FORTRAN_CMPLX and ARG1 is the argument passed to CMPLX if.  */ | 
|  |  | 
|  | extern value *eval_op_f_cmplx (type *expect_type, expression *exp, | 
|  | noside noside, exp_opcode opcode, value *arg1); | 
|  |  | 
|  | /* Implement expression evaluation for Fortran's CMPLX intrinsic function | 
|  | called with two arguments.  For EXPECT_TYPE, EXP, and NOSIDE see | 
|  | expression::evaluate (in expression.h).  OPCODE will always be | 
|  | FORTRAN_CMPLX, ARG1 and ARG2 are the arguments passed to CMPLX.  */ | 
|  |  | 
|  | extern struct value *eval_op_f_cmplx (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode opcode, | 
|  | struct value *arg1, struct value *arg2); | 
|  |  | 
|  | /* Implement expression evaluation for Fortran's CMPLX intrinsic function | 
|  | called with three arguments.  For EXPECT_TYPE, EXP, and NOSIDE see | 
|  | expression::evaluate (in expression.h).  OPCODE will always be | 
|  | FORTRAN_CMPLX, ARG1 and ARG2 are real and imaginary part passed to CMPLX, | 
|  | and KIND_ARG is the type corresponding to the KIND parameter passed to | 
|  | CMPLX.  */ | 
|  |  | 
|  | extern value *eval_op_f_cmplx (type *expect_type, expression *exp, | 
|  | noside noside, exp_opcode opcode, value *arg1, | 
|  | value *arg2, type *kind_arg); | 
|  |  | 
|  | extern struct value *eval_op_f_kind (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode opcode, | 
|  | struct value *arg1); | 
|  | extern struct value *eval_op_f_associated (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode opcode, | 
|  | struct value *arg1); | 
|  | extern struct value *eval_op_f_associated (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode opcode, | 
|  | struct value *arg1, | 
|  | struct value *arg2); | 
|  | extern struct value * eval_op_f_allocated (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode op, | 
|  | struct value *arg1); | 
|  | extern struct value * eval_op_f_loc (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode op, | 
|  | struct value *arg1); | 
|  |  | 
|  | /* Implement the evaluation of UNOP_FORTRAN_RANK.  EXPECTED_TYPE, EXP, and | 
|  | NOSIDE are as for expression::evaluate (see expression.h).  OP will | 
|  | always be UNOP_FORTRAN_RANK, and ARG1 is the argument being passed to | 
|  | the expression.   */ | 
|  |  | 
|  | extern struct value *eval_op_f_rank (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode op, | 
|  | struct value *arg1); | 
|  |  | 
|  | /* Implement expression evaluation for Fortran's SIZE keyword. For | 
|  | EXPECT_TYPE, EXP, and NOSIDE see expression::evaluate (in | 
|  | expression.h).  OPCODE will always for FORTRAN_ARRAY_SIZE.  ARG1 is the | 
|  | value passed to SIZE if it is only passed a single argument.  For the | 
|  | two argument form see the overload of this function below.  */ | 
|  |  | 
|  | extern struct value *eval_op_f_array_size (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode opcode, | 
|  | struct value *arg1); | 
|  |  | 
|  | /* An overload of EVAL_OP_F_ARRAY_SIZE above, this version takes two | 
|  | arguments, representing the two values passed to Fortran's SIZE | 
|  | keyword.  */ | 
|  |  | 
|  | extern struct value *eval_op_f_array_size (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode opcode, | 
|  | struct value *arg1, | 
|  | struct value *arg2); | 
|  |  | 
|  | /* Implement expression evaluation for Fortran's SIZE intrinsic function called | 
|  | with three arguments.  For EXPECT_TYPE, EXP, and NOSIDE see | 
|  | expression::evaluate (in expression.h).  OPCODE will always be | 
|  | FORTRAN_ARRAY_SIZE, ARG1 and ARG2 the first two values passed to SIZE, and | 
|  | KIND_ARG is the type corresponding to the KIND parameter passed to SIZE.  */ | 
|  |  | 
|  | extern value *eval_op_f_array_size (type *expect_type, expression *exp, | 
|  | noside noside, exp_opcode opcode, | 
|  | value *arg1, value *arg2, type *kind_arg); | 
|  |  | 
|  | /* Implement the evaluation of Fortran's SHAPE keyword.  EXPECTED_TYPE, | 
|  | EXP, and NOSIDE are as for expression::evaluate (see expression.h).  OP | 
|  | will always be UNOP_FORTRAN_SHAPE, and ARG1 is the argument being passed | 
|  | to the expression.  */ | 
|  |  | 
|  | extern struct value *eval_op_f_array_shape (struct type *expect_type, | 
|  | struct expression *exp, | 
|  | enum noside noside, | 
|  | enum exp_opcode op, | 
|  | struct value *arg1); | 
|  |  | 
|  | namespace expr | 
|  | { | 
|  |  | 
|  | /* Function prototype for Fortran intrinsic functions taking one argument and | 
|  | one kind argument.  */ | 
|  | typedef value *binary_kind_ftype (type *expect_type, expression *exp, | 
|  | noside noside, exp_opcode op, value *arg1, | 
|  | type *kind_arg); | 
|  |  | 
|  | /* Two-argument operation with the second argument being a kind argument.  */ | 
|  | template<exp_opcode OP, binary_kind_ftype FUNC> | 
|  | class fortran_kind_2arg | 
|  | : public tuple_holding_operation<operation_up, type*> | 
|  | { | 
|  | public: | 
|  |  | 
|  | using tuple_holding_operation::tuple_holding_operation; | 
|  |  | 
|  | value *evaluate (type *expect_type, expression *exp, noside noside) override | 
|  | { | 
|  | value *arg1 = std::get<0> (m_storage)->evaluate (nullptr, exp, noside); | 
|  | type *kind_arg = std::get<1> (m_storage); | 
|  | return FUNC (expect_type, exp, noside, OP, arg1, kind_arg); | 
|  | } | 
|  |  | 
|  | exp_opcode opcode () const override | 
|  | { return OP; } | 
|  | }; | 
|  |  | 
|  | /* Function prototype for Fortran intrinsic functions taking two arguments and | 
|  | one kind argument.  */ | 
|  | typedef value *ternary_kind_ftype (type *expect_type, expression *exp, | 
|  | noside noside, exp_opcode op, value *arg1, | 
|  | value *arg2, type *kind_arg); | 
|  |  | 
|  | /* Three-argument operation with the third argument being a kind argument.  */ | 
|  | template<exp_opcode OP, ternary_kind_ftype FUNC> | 
|  | class fortran_kind_3arg | 
|  | : public tuple_holding_operation<operation_up, operation_up, type *> | 
|  | { | 
|  | public: | 
|  |  | 
|  | using tuple_holding_operation::tuple_holding_operation; | 
|  |  | 
|  | value *evaluate (type *expect_type, expression *exp, 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); | 
|  | type *kind_arg = std::get<2> (m_storage); | 
|  | return FUNC (expect_type, exp, noside, OP, arg1, arg2, kind_arg); | 
|  | } | 
|  |  | 
|  | exp_opcode opcode () const override | 
|  | { return OP; } | 
|  | }; | 
|  |  | 
|  | using fortran_abs_operation = unop_operation<UNOP_ABS, eval_op_f_abs>; | 
|  | using fortran_ceil_operation_1arg = unop_operation<FORTRAN_CEILING, | 
|  | eval_op_f_ceil>; | 
|  | using fortran_ceil_operation_2arg = fortran_kind_2arg<FORTRAN_CEILING, | 
|  | eval_op_f_ceil>; | 
|  | using fortran_floor_operation_1arg = unop_operation<FORTRAN_FLOOR, | 
|  | eval_op_f_floor>; | 
|  | using fortran_floor_operation_2arg = fortran_kind_2arg<FORTRAN_FLOOR, | 
|  | eval_op_f_floor>; | 
|  | using fortran_kind_operation = unop_operation<UNOP_FORTRAN_KIND, | 
|  | eval_op_f_kind>; | 
|  | using fortran_allocated_operation = unop_operation<UNOP_FORTRAN_ALLOCATED, | 
|  | eval_op_f_allocated>; | 
|  | using fortran_loc_operation = unop_operation<UNOP_FORTRAN_LOC, | 
|  | eval_op_f_loc>; | 
|  |  | 
|  | using fortran_mod_operation = binop_operation<BINOP_MOD, eval_op_f_mod>; | 
|  | using fortran_modulo_operation = binop_operation<BINOP_FORTRAN_MODULO, | 
|  | eval_op_f_modulo>; | 
|  | using fortran_associated_1arg = unop_operation<FORTRAN_ASSOCIATED, | 
|  | eval_op_f_associated>; | 
|  | using fortran_associated_2arg = binop_operation<FORTRAN_ASSOCIATED, | 
|  | eval_op_f_associated>; | 
|  | using fortran_rank_operation = unop_operation<UNOP_FORTRAN_RANK, | 
|  | eval_op_f_rank>; | 
|  | using fortran_array_size_1arg = unop_operation<FORTRAN_ARRAY_SIZE, | 
|  | eval_op_f_array_size>; | 
|  | using fortran_array_size_2arg = binop_operation<FORTRAN_ARRAY_SIZE, | 
|  | eval_op_f_array_size>; | 
|  | using fortran_array_size_3arg = fortran_kind_3arg<FORTRAN_ARRAY_SIZE, | 
|  | eval_op_f_array_size>; | 
|  | using fortran_array_shape_operation = unop_operation<UNOP_FORTRAN_SHAPE, | 
|  | eval_op_f_array_shape>; | 
|  | using fortran_cmplx_operation_1arg = unop_operation<FORTRAN_CMPLX, | 
|  | eval_op_f_cmplx>; | 
|  | using fortran_cmplx_operation_2arg = binop_operation<FORTRAN_CMPLX, | 
|  | eval_op_f_cmplx>; | 
|  | using fortran_cmplx_operation_3arg = fortran_kind_3arg<FORTRAN_CMPLX, | 
|  | eval_op_f_cmplx>; | 
|  |  | 
|  | /* OP_RANGE for Fortran.  */ | 
|  | class fortran_range_operation | 
|  | : public tuple_holding_operation<enum range_flag, 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 | 
|  | { | 
|  | error (_("ranges not allowed in this context")); | 
|  | } | 
|  |  | 
|  | range_flag get_flags () const | 
|  | { | 
|  | return std::get<0> (m_storage); | 
|  | } | 
|  |  | 
|  | value *evaluate0 (struct expression *exp, enum noside noside) const | 
|  | { | 
|  | return std::get<1> (m_storage)->evaluate (nullptr, exp, noside); | 
|  | } | 
|  |  | 
|  | value *evaluate1 (struct expression *exp, enum noside noside) const | 
|  | { | 
|  | return std::get<2> (m_storage)->evaluate (nullptr, exp, noside); | 
|  | } | 
|  |  | 
|  | value *evaluate2 (struct expression *exp, enum noside noside) const | 
|  | { | 
|  | return std::get<3> (m_storage)->evaluate (nullptr, exp, noside); | 
|  | } | 
|  |  | 
|  | enum exp_opcode opcode () const override | 
|  | { return OP_RANGE; } | 
|  | }; | 
|  |  | 
|  | /* In F77, functions, substring ops and array subscript operations | 
|  | cannot be disambiguated at parse time.  This operation handles | 
|  | both, deciding which do to at evaluation time.  */ | 
|  | class fortran_undetermined | 
|  | : public tuple_holding_operation<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_F77_UNDETERMINED_ARGLIST; } | 
|  |  | 
|  | private: | 
|  |  | 
|  | value *value_subarray (value *array, struct expression *exp, | 
|  | enum noside noside); | 
|  | }; | 
|  |  | 
|  | /* Single-argument form of Fortran ubound/lbound intrinsics.  */ | 
|  | class fortran_bound_1arg | 
|  | : public tuple_holding_operation<exp_opcode, 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); } | 
|  | }; | 
|  |  | 
|  | /* Two-argument form of Fortran ubound/lbound intrinsics.  */ | 
|  | class fortran_bound_2arg | 
|  | : public tuple_holding_operation<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); } | 
|  | }; | 
|  |  | 
|  | /* Three-argument form of Fortran ubound/lbound intrinsics.  */ | 
|  | class fortran_bound_3arg | 
|  | : public tuple_holding_operation<exp_opcode, operation_up, operation_up, | 
|  | type *> | 
|  | { | 
|  | public: | 
|  |  | 
|  | using tuple_holding_operation::tuple_holding_operation; | 
|  |  | 
|  | value *evaluate (type *expect_type, expression *exp, noside noside) override; | 
|  |  | 
|  | exp_opcode opcode () const override | 
|  | { return std::get<0> (m_storage); } | 
|  | }; | 
|  |  | 
|  | /* Implement STRUCTOP_STRUCT for Fortran.  */ | 
|  | class fortran_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; } | 
|  | }; | 
|  |  | 
|  | } /* namespace expr */ | 
|  |  | 
|  | #endif /* FORTRAN_EXP_H */ |