0fadfc5fb7
Code that collects fields in struct-like patterns used to ignore wildcard patterns like `Foo{_}`. But `enter_defaults` considered struct-like patterns as default in order to overcome this (accoring to my understanding of situation). However such behaviour caused code like this: ``` enum E { Foo{f: int}, Bar } let e = Bar; match e { Foo{f: _f} => { /* do something (1) */ } _ => { /* do something (2) */ } } ``` consider pattern `Foo{f: _f}` as default. That caused inproper behaviour and even segfaults while trying to destruct `Bar` as `Foo{f: _f}`. Issues: #5625 , #5530. This patch fixes `collect_record_or_struct_fields` to split cases of single wildcard struct-like pattern and no struct-like pattern at all. Former case resolved with `enter_rec_or_struct` (and not with `enter_defaults`). Closes #5625. Closes #5530.
48 lines
1.4 KiB
Rust
48 lines
1.4 KiB
Rust
// Copyright 2013 The Rust Project Developers. See the COPYRIGHT
|
|
// file at the top-level directory of this distribution and at
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
// option. This file may not be copied, modified, or distributed
|
|
// except according to those terms.
|
|
|
|
enum Enum {
|
|
Foo { foo: uint },
|
|
Bar { bar: uint }
|
|
}
|
|
|
|
fn fun1(e1: &Enum, e2: &Enum) -> uint {
|
|
match (e1, e2) {
|
|
(&Foo { foo: _ }, &Foo { foo: _ }) => 0,
|
|
(&Foo { foo: _ }, &Bar { bar: _ }) => 1,
|
|
(&Bar { bar: _ }, &Bar { bar: _ }) => 2,
|
|
(&Bar { bar: _ }, &Foo { foo: _ }) => 3,
|
|
}
|
|
}
|
|
|
|
fn fun2(e1: &Enum, e2: &Enum) -> uint {
|
|
match (e1, e2) {
|
|
(&Foo { foo: _ }, &Foo { foo: _ }) => 0,
|
|
(&Foo { foo: _ }, _ ) => 1,
|
|
(&Bar { bar: _ }, &Bar { bar: _ }) => 2,
|
|
(&Bar { bar: _ }, _ ) => 3,
|
|
}
|
|
}
|
|
|
|
pub fn main() {
|
|
let foo = Foo { foo: 1 };
|
|
let bar = Bar { bar: 1 };
|
|
|
|
assert_eq!(fun1(&foo, &foo), 0);
|
|
assert_eq!(fun1(&foo, &bar), 1);
|
|
assert_eq!(fun1(&bar, &bar), 2);
|
|
assert_eq!(fun1(&bar, &foo), 3);
|
|
|
|
assert_eq!(fun2(&foo, &foo), 0);
|
|
assert_eq!(fun2(&foo, &bar), 1); // fun2 returns 0
|
|
assert_eq!(fun2(&bar, &bar), 2);
|
|
assert_eq!(fun2(&bar, &foo), 3); // fun2 returns 2
|
|
}
|