mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-01-12 12:16:04 +08:00
1d506c26d9
This commit is the result of the following actions: - Running gdb/copyright.py to update all of the copyright headers to include 2024, - Manually updating a few files the copyright.py script told me to update, these files had copyright headers embedded within the file, - Regenerating gdbsupport/Makefile.in to refresh it's copyright date, - Using grep to find other files that still mentioned 2023. If these files were updated last year from 2022 to 2023 then I've updated them this year to 2024. I'm sure I've probably missed some dates. Feel free to fix them up as you spot them.
230 lines
6.4 KiB
C++
230 lines
6.4 KiB
C++
/* Definitions for Rust expressions
|
|
|
|
Copyright (C) 2020-2024 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 */
|