9249e6a1e2
Changes #[derive(Copy, Clone)] to use a faster impl of Clone when both derives are present, and there are no generics in the type. The faster impl is simply returning *self (which works because the type is also Copy). See the comments in libsyntax_ext/deriving/clone.rs for more details. There are a few types which are Copy but not Clone, in violation of the definition of Copy. These include large arrays and tuples. The very existence of these types is arguably a bug, but in order for this optimization not to change the applicability of #[derive(Copy, Clone)], the faster Clone impl also injects calls to a new function, core::clone::assert_receiver_is_clone, to verify that all members are actually Clone. This is not a breaking change, because pursuant to RFC 1521, any type that implements Copy should not do any observable work in its Clone impl.
81 lines
1.4 KiB
Rust
81 lines
1.4 KiB
Rust
// Copyright 2014 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.
|
|
|
|
use std::ops::Deref;
|
|
|
|
struct Root {
|
|
jsref: JSRef
|
|
}
|
|
|
|
impl Deref for Root {
|
|
type Target = JSRef;
|
|
|
|
fn deref<'a>(&'a self) -> &'a JSRef {
|
|
&self.jsref
|
|
}
|
|
}
|
|
|
|
#[derive(Copy, Clone)]
|
|
struct JSRef {
|
|
node: *const Node
|
|
}
|
|
|
|
impl Deref for JSRef {
|
|
type Target = Node;
|
|
|
|
fn deref<'a>(&'a self) -> &'a Node {
|
|
self.get()
|
|
}
|
|
}
|
|
|
|
trait INode {
|
|
fn RemoveChild(&self);
|
|
}
|
|
|
|
impl INode for JSRef {
|
|
fn RemoveChild(&self) {
|
|
self.get().RemoveChild(0)
|
|
}
|
|
}
|
|
|
|
impl JSRef {
|
|
fn AddChild(&self) {
|
|
self.get().AddChild(0);
|
|
}
|
|
|
|
fn get<'a>(&'a self) -> &'a Node {
|
|
unsafe {
|
|
&*self.node
|
|
}
|
|
}
|
|
}
|
|
|
|
struct Node;
|
|
|
|
impl Node {
|
|
fn RemoveChild(&self, _a: usize) {
|
|
}
|
|
|
|
fn AddChild(&self, _a: usize) {
|
|
}
|
|
}
|
|
|
|
fn main() {
|
|
let n = Node;
|
|
let jsref = JSRef { node: &n };
|
|
let root = Root { jsref: jsref };
|
|
|
|
root.AddChild();
|
|
jsref.AddChild();
|
|
|
|
root.RemoveChild();
|
|
jsref.RemoveChild();
|
|
}
|