Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

rustc: Implement the Drop trait #3925

Closed
wants to merge 1 commit into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions src/libcore/core.rs
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,8 @@ pub use to_str::ToStr;
#[cfg(notest)]
pub use ops::{Const, Copy, Send, Owned};
#[cfg(notest)]
pub use ops::{Drop};
#[cfg(notest)]
pub use ops::{Add, Sub, Mul, Div, Modulo, Neg, BitAnd, BitOr, BitXor};
#[cfg(notest)]
pub use ops::{Shl, Shr, Index};
Expand All @@ -38,6 +40,8 @@ extern mod coreops(name = "core", vers = "0.5");
#[cfg(test)]
pub use coreops::ops::{Const, Copy, Send, Owned};
#[cfg(test)]
pub use coreops::ops::{Drop};
#[cfg(test)]
pub use coreops::ops::{Add, Sub, Mul, Div, Modulo, Neg, BitAnd, BitOr};
#[cfg(test)]
pub use coreops::ops::{BitXor};
Expand Down
5 changes: 5 additions & 0 deletions src/libcore/ops.rs
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,11 @@ pub trait Owned {
// Empty.
}

#[lang="drop"]
pub trait Drop {
fn finalize(); // XXX: Rename to "drop"? --pcwalton
}

#[lang="add"]
pub trait Add<RHS,Result> {
pure fn add(rhs: &RHS) -> Result;
Expand Down
6 changes: 6 additions & 0 deletions src/rustc/middle/lang_items.rs
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,8 @@ struct LanguageItems {
mut send_trait: Option<def_id>,
mut owned_trait: Option<def_id>,

mut drop_trait: Option<def_id>,

mut add_trait: Option<def_id>,
mut sub_trait: Option<def_id>,
mut mul_trait: Option<def_id>,
Expand Down Expand Up @@ -59,6 +61,8 @@ mod language_items {
send_trait: None,
owned_trait: None,

drop_trait: None,

add_trait: None,
sub_trait: None,
mul_trait: None,
Expand Down Expand Up @@ -94,6 +98,8 @@ fn LanguageItemCollector(crate: @crate, session: Session,
item_refs.insert(~"send", &mut items.send_trait);
item_refs.insert(~"owned", &mut items.owned_trait);

item_refs.insert(~"drop", &mut items.drop_trait);

item_refs.insert(~"add", &mut items.add_trait);
item_refs.insert(~"sub", &mut items.sub_trait);
item_refs.insert(~"mul", &mut items.mul_trait);
Expand Down
20 changes: 18 additions & 2 deletions src/rustc/middle/ty.rs
Original file line number Diff line number Diff line change
Expand Up @@ -406,7 +406,16 @@ type ctxt =
// A mapping from the def ID of an impl to the IDs of the derived
// methods within it.
automatically_derived_methods_for_impl:
HashMap<ast::def_id, @~[ast::def_id]>
HashMap<ast::def_id, @~[ast::def_id]>,

// A mapping from the def ID of an enum or struct type to the def ID
// of the method that implements its destructor. If the type is not
// present in this map, it does not have a destructor. This map is
// populated during the coherence phase of typechecking.
destructor_for_type: HashMap<ast::def_id, ast::def_id>,

// A method will be in this list if and only if it is a destructor.
destructors: HashMap<ast::def_id, ()>
};

enum tbox_flag {
Expand Down Expand Up @@ -977,7 +986,9 @@ fn mk_ctxt(s: session::Session,
deriving_struct_methods: HashMap(),
deriving_enum_methods: HashMap(),
automatically_derived_methods: HashMap(),
automatically_derived_methods_for_impl: HashMap()}
automatically_derived_methods_for_impl: HashMap(),
destructor_for_type: HashMap(),
destructors: HashMap()}
}


Expand Down Expand Up @@ -3627,6 +3638,11 @@ fn item_path_str(cx: ctxt, id: ast::def_id) -> ~str {
/* If class_id names a class with a dtor, return Some(the dtor's id).
Otherwise return none. */
fn ty_dtor(cx: ctxt, class_id: def_id) -> Option<def_id> {
match cx.destructor_for_type.find(class_id) {
Some(method_def_id) => return Some(method_def_id),
None => {} // Continue.
}

if is_local(class_id) {
match cx.items.find(class_id.node) {
Some(ast_map::node_item(@{
Expand Down
20 changes: 20 additions & 0 deletions src/rustc/middle/typeck/check/method.rs
Original file line number Diff line number Diff line change
Expand Up @@ -774,6 +774,7 @@ impl LookupContext {
let fty = self.fn_ty_from_origin(&candidate.origin);

self.enforce_trait_instance_limitations(fty, candidate);
self.enforce_drop_trait_limitations(candidate);

// before we only checked whether self_ty could be a subtype
// of rcvr_ty; now we actually make it so (this may cause
Expand Down Expand Up @@ -858,6 +859,25 @@ impl LookupContext {
}
}

fn enforce_drop_trait_limitations(&self, candidate: &Candidate) {
// No code can call the finalize method explicitly.
let bad;
match candidate.origin {
method_static(method_id) | method_self(method_id, _) => {
bad = self.tcx().destructors.contains_key(method_id);
}
method_param({trait_id: trait_id, _}) |
method_trait(trait_id, _, _) => {
bad = self.tcx().destructor_for_type.contains_key(trait_id);
}
}

if bad {
self.tcx().sess.span_err(self.expr.span,
~"explicit call to destructor");
}
}

fn is_relevant(&self, self_ty: ty::t, candidate: &Candidate) -> bool {
debug!("is_relevant(self_ty=%s, candidate=%s)",
self.ty_to_str(self_ty), self.cand_to_str(candidate));
Expand Down
57 changes: 57 additions & 0 deletions src/rustc/middle/typeck/coherence.rs
Original file line number Diff line number Diff line change
Expand Up @@ -228,6 +228,11 @@ impl CoherenceChecker {
// coherence checks, because we ensure by construction that no errors
// can happen at link time.
self.add_external_crates();

// Populate the table of destructors. It might seem a bit strange to
// do this here, but it's actually the most convenient place, since
// the coherence tables contain the trait -> type mappings.
self.populate_destructor_table();
}

fn check_implementation(item: @item, associated_traits: ~[@trait_ref]) {
Expand Down Expand Up @@ -913,6 +918,58 @@ impl CoherenceChecker {
}
}
}

//
// Destructors
//

fn populate_destructor_table() {
let coherence_info = &self.crate_context.coherence_info;
let tcx = self.crate_context.tcx;
let drop_trait = tcx.lang_items.drop_trait.get();
let impls_opt = coherence_info.extension_methods.find(drop_trait);

let impls;
match impls_opt {
None => return, // No types with (new-style) destructors present.
Some(found_impls) => impls = found_impls
}

for impls.each |impl_info| {
if impl_info.methods.len() < 1 {
// We'll error out later. For now, just don't ICE.
loop;
}
let method_def_id = impl_info.methods[0].did;

let self_type = self.get_self_type_for_implementation(*impl_info);
match ty::get(self_type.ty).sty {
ty::ty_class(type_def_id, _) => {
tcx.destructor_for_type.insert(type_def_id, method_def_id);
tcx.destructors.insert(method_def_id, ());
}
_ => {
// Destructors only work on nominal types.
if impl_info.did.crate == ast::local_crate {
match tcx.items.find(impl_info.did.node) {
Some(ast_map::node_item(@item, _)) => {
tcx.sess.span_err(item.span,
~"the Drop trait may only \
be implemented on \
structures");
}
_ => {
tcx.sess.bug(~"didn't find impl in ast map");
}
}
} else {
tcx.sess.bug(~"found external impl of Drop trait on \
something other than a struct");
}
}
}
}
}
}

fn check_coherence(crate_context: @crate_ctxt, crate: @crate) {
Expand Down
12 changes: 12 additions & 0 deletions src/test/compile-fail/drop-on-non-struct.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
type Foo = @[u8];

impl Foo : Drop { //~ ERROR the Drop trait may only be implemented
fn finalize() {
io::println("kaboom");
}
}

fn main() {
}


15 changes: 15 additions & 0 deletions src/test/compile-fail/explicit-call-to-dtor.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
struct Foo {
x: int
}

impl Foo : Drop {
fn finalize() {
io::println("kaboom");
}
}

fn main() {
let x = Foo { x: 3 };
x.finalize(); //~ ERROR explicit call to destructor
}

25 changes: 25 additions & 0 deletions src/test/compile-fail/explicit-call-to-supertrait-dtor.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
struct Foo {
x: int
}

trait Bar : Drop {
fn blah();
}

impl Foo : Drop {
fn finalize() {
io::println("kaboom");
}
}

impl Foo : Bar {
fn blah() {
self.finalize(); //~ ERROR explicit call to destructor
}
}

fn main() {
let x = Foo { x: 3 };
}


14 changes: 14 additions & 0 deletions src/test/run-pass/drop-trait.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
struct Foo {
x: int
}

impl Foo : Drop {
fn finalize() {
io::println("bye");
}
}

fn main() {
let x: Foo = Foo { x: 3 };
}