Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit bff0ff4

Browse files
committedFeb 6, 2025
derive(PartialEq): Also derive StructuralPartialEq
gcc/rust/ChangeLog: * expand/rust-derive-partial-eq.cc: Adapt signatures to generate two impls. * expand/rust-derive-partial-eq.h: Likewise. * expand/rust-derive.cc (DeriveVisitor::derive): Adapt to multiple item generation. gcc/testsuite/ChangeLog: * rust/compile/derive-eq-invalid.rs: * rust/compile/derive-partialeq1.rs:
1 parent 4d8b1bc commit bff0ff4

File tree

6 files changed

+43
-22
lines changed

6 files changed

+43
-22
lines changed
 

‎gcc/rust/expand/rust-derive-partial-eq.cc

+23-18
Original file line numberDiff line numberDiff line change
@@ -27,35 +27,40 @@
2727

2828
namespace Rust {
2929
namespace AST {
30-
DerivePartialEq::DerivePartialEq (location_t loc)
31-
: DeriveVisitor (loc), expanded (nullptr)
32-
{}
30+
DerivePartialEq::DerivePartialEq (location_t loc) : DeriveVisitor (loc) {}
3331

34-
std::unique_ptr<AST::Item>
32+
std::vector<std::unique_ptr<AST::Item>>
3533
DerivePartialEq::go (Item &item)
3634
{
3735
item.accept_vis (*this);
3836

39-
rust_assert (expanded);
40-
4137
return std::move (expanded);
4238
}
4339

44-
std::unique_ptr<Item>
45-
DerivePartialEq::partial_eq_impl (
40+
std::vector<std::unique_ptr<Item>>
41+
DerivePartialEq::partialeq_impls (
4642
std::unique_ptr<AssociatedItem> &&eq_fn, std::string name,
4743
const std::vector<std::unique_ptr<GenericParam>> &type_generics)
4844
{
4945
auto eq = builder.type_path (LangItem::Kind::EQ);
46+
auto speq = builder.type_path (LangItem::Kind::STRUCTURAL_PEQ);
5047

5148
auto trait_items = vec (std::move (eq_fn));
5249

53-
auto generics
50+
// no extra bound on StructuralPeq
51+
auto peq_generics
5452
= setup_impl_generics (name, type_generics, builder.trait_bound (eq));
53+
auto speq_generics = setup_impl_generics (name, type_generics);
54+
55+
auto peq = builder.trait_impl (eq, std::move (peq_generics.self_type),
56+
std::move (trait_items),
57+
std::move (peq_generics.impl));
58+
59+
auto structural_peq
60+
= builder.trait_impl (speq, std::move (speq_generics.self_type), {},
61+
std::move (speq_generics.impl));
5562

56-
return builder.trait_impl (eq, std::move (generics.self_type),
57-
std::move (trait_items),
58-
std::move (generics.impl));
63+
return vec (std::move (peq), std::move (structural_peq));
5964
}
6065

6166
std::unique_ptr<AssociatedItem>
@@ -137,7 +142,7 @@ DerivePartialEq::visit_tuple (TupleStruct &item)
137142
auto fn = eq_fn (build_eq_expression (std::move (fields)), type_name);
138143

139144
expanded
140-
= partial_eq_impl (std::move (fn), type_name, item.get_generic_params ());
145+
= partialeq_impls (std::move (fn), type_name, item.get_generic_params ());
141146
}
142147

143148
void
@@ -153,7 +158,7 @@ DerivePartialEq::visit_struct (StructStruct &item)
153158
auto fn = eq_fn (build_eq_expression (std::move (fields)), type_name);
154159

155160
expanded
156-
= partial_eq_impl (std::move (fn), type_name, item.get_generic_params ());
161+
= partialeq_impls (std::move (fn), type_name, item.get_generic_params ());
157162
}
158163

159164
MatchCase
@@ -250,11 +255,12 @@ void
250255
DerivePartialEq::visit_enum (Enum &item)
251256
{
252257
auto cases = std::vector<MatchCase> ();
258+
auto type_name = item.get_identifier ().as_string ();
253259

254260
for (auto &variant : item.get_variants ())
255261
{
256262
auto variant_path
257-
= builder.variant_path (item.get_identifier ().as_string (),
263+
= builder.variant_path (type_name,
258264
variant->get_identifier ().as_string ());
259265

260266
switch (variant->get_enum_item_kind ())
@@ -290,11 +296,10 @@ DerivePartialEq::visit_enum (Enum &item)
290296
builder.identifier ("other"))),
291297
std::move (cases));
292298

293-
auto fn = eq_fn (std::move (match), item.get_identifier ().as_string ());
299+
auto fn = eq_fn (std::move (match), type_name);
294300

295301
expanded
296-
= partial_eq_impl (std::move (fn), item.get_identifier ().as_string (),
297-
item.get_generic_params ());
302+
= partialeq_impls (std::move (fn), type_name, item.get_generic_params ());
298303
}
299304

300305
void

‎gcc/rust/expand/rust-derive-partial-eq.h

+7-3
Original file line numberDiff line numberDiff line change
@@ -30,12 +30,16 @@ class DerivePartialEq : DeriveVisitor
3030
public:
3131
DerivePartialEq (location_t loc);
3232

33-
std::unique_ptr<AST::Item> go (Item &item);
33+
std::vector<std::unique_ptr<AST::Item>> go (Item &item);
3434

3535
private:
36-
std::unique_ptr<Item> expanded;
36+
std::vector<std::unique_ptr<Item>> expanded;
3737

38-
std::unique_ptr<Item> partial_eq_impl (
38+
/**
39+
* Generate both an implementation of `PartialEq` and `StructuralPartialEq`
40+
* for the given type
41+
*/
42+
std::vector<std::unique_ptr<Item>> partialeq_impls (
3943
std::unique_ptr<AssociatedItem> &&eq_fn, std::string name,
4044
const std::vector<std::unique_ptr<GenericParam>> &type_generics);
4145

‎gcc/rust/expand/rust-derive.cc

+1-1
Original file line numberDiff line numberDiff line change
@@ -52,7 +52,7 @@ DeriveVisitor::derive (Item &item, const Attribute &attr,
5252
case BuiltinMacro::Eq:
5353
return vec (DeriveEq (attr.get_locus ()).go (item));
5454
case BuiltinMacro::PartialEq:
55-
return vec (DerivePartialEq (attr.get_locus ()).go (item));
55+
return DerivePartialEq (attr.get_locus ()).go (item);
5656
case BuiltinMacro::Ord:
5757
case BuiltinMacro::PartialOrd:
5858
case BuiltinMacro::Hash:

‎gcc/testsuite/rust/compile/derive-eq-invalid.rs

+6
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,12 @@ struct PhantomData<T>;
2121
#[lang = "sized"]
2222
trait Sized {}
2323

24+
#[lang = "structural_peq"]
25+
trait StructuralPartialEq {}
26+
27+
#[lang = "structural_teq"]
28+
trait StructuralEq {}
29+
2430
#[derive(PartialEq)]
2531
struct NotEq;
2632

‎gcc/testsuite/rust/compile/derive-partialeq1.rs

+3
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,9 @@ trait Sized {}
66
#[lang = "copy"]
77
trait Copy {}
88

9+
#[lang = "structural_peq"]
10+
trait StructuralPartialEq {}
11+
912
#[lang = "eq"]
1013
pub trait PartialEq<Rhs: ?Sized = Self> {
1114
/// This method tests for `self` and `other` values to be equal, and is used

‎gcc/testsuite/rust/execute/torture/derive-partialeq1.rs

+3
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,9 @@ pub trait PartialEq<Rhs: ?Sized = Self> {
2525
}
2626
}
2727

28+
#[lang = "structural_peq"]
29+
trait StructuralPartialEq {}
30+
2831
#[derive(PartialEq, Copy)] // { dg-warning "unused name" }
2932
struct Foo;
3033

0 commit comments

Comments
 (0)
Please sign in to comment.