// Copyright 2012 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 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. //! Logic relating to rooting and write guards for managed values //! (`@` and `@mut`). This code is primarily for use by datum; //! it exists in its own module both to keep datum.rs bite-sized //! and for each in debugging (e.g., so you can use //! `RUST_LOG=rustc::middle::trans::write_guard`). use lib::llvm::ValueRef; use middle::borrowck::{RootInfo, root_map_key, DynaImm, DynaMut}; use middle::lang_items::CheckNotBorrowedFnLangItem; use middle::lang_items::{BorrowAsImmFnLangItem, BorrowAsMutFnLangItem}; use middle::lang_items::{RecordBorrowFnLangItem, UnrecordBorrowFnLangItem}; use middle::lang_items::ReturnToMutFnLangItem; use middle::trans::base::*; use middle::trans::build::*; use middle::trans::callee; use middle::trans::common::*; use middle::trans::datum::*; use middle::trans::expr; use middle::ty; use syntax::codemap::Span; use syntax::ast; use middle::trans::type_::Type; pub fn root_and_write_guard(datum: &Datum, mut bcx: @Block, span: Span, expr_id: ast::NodeId, derefs: uint) -> @Block { let key = root_map_key { id: expr_id, derefs: derefs }; debug!("write_guard::root_and_write_guard(key={:?})", key); // root the autoderef'd value, if necessary: // // (Note: root'd values are always boxes) let ccx = bcx.ccx(); bcx = { let root_map = ccx.maps.root_map.borrow(); match root_map.get().find(&key) { None => bcx, Some(&root_info) => root(datum, bcx, span, key, root_info) } }; // Perform the write guard, if necessary. // // (Note: write-guarded values are always boxes) let write_guard_map = ccx.maps.write_guard_map.borrow(); if write_guard_map.get().contains(&key) { perform_write_guard(datum, bcx, span) } else { bcx } } pub fn return_to_mut(mut bcx: @Block, root_key: root_map_key, frozen_val_ref: ValueRef, bits_val_ref: ValueRef, filename_val: ValueRef, line_val: ValueRef) -> @Block { debug!("write_guard::return_to_mut(root_key={:?}, {}, {}, {})", root_key, bcx.to_str(), bcx.val_to_str(frozen_val_ref), bcx.val_to_str(bits_val_ref)); let box_ptr = Load(bcx, PointerCast(bcx, frozen_val_ref, Type::i8p().ptr_to())); let bits_val = Load(bcx, bits_val_ref); if bcx.tcx().sess.debug_borrows() { bcx = callee::trans_lang_call( bcx, langcall(bcx, None, "unborrow", UnrecordBorrowFnLangItem), [ box_ptr, bits_val, filename_val, line_val ], Some(expr::Ignore)).bcx; } callee::trans_lang_call( bcx, langcall(bcx, None, "unborrow", ReturnToMutFnLangItem), [ box_ptr, bits_val, filename_val, line_val ], Some(expr::Ignore) ).bcx } fn root(datum: &Datum, mut bcx: @Block, span: Span, root_key: root_map_key, root_info: RootInfo) -> @Block { //! In some cases, borrowck will decide that an @T/@[]/@str //! value must be rooted for the program to be safe. In that //! case, we will call this function, which will stash a copy //! away until we exit the scope `scope_id`. debug!("write_guard::root(root_key={:?}, root_info={:?}, datum={:?})", root_key, root_info, datum.to_str(bcx.ccx())); // First, root the datum. Note that we must zero this value, // because sometimes we root on one path but not another. // See e.g. #4904. let scratch = scratch_datum(bcx, datum.ty, "__write_guard", true); datum.copy_to_datum(bcx, INIT, scratch); let cleanup_bcx = find_bcx_for_scope(bcx, root_info.scope); add_clean_temp_mem_in_scope(cleanup_bcx, root_info.scope, scratch.val, scratch.ty); // Now, consider also freezing it. match root_info.freeze { None => {} Some(freeze_kind) => { let (filename, line) = filename_and_line_num_from_span(bcx, span); // in this case, we don't have to zero, because // scratch.val will be NULL should the cleanup get // called without the freezing actually occurring, and // return_to_mut checks for this condition. let scratch_bits = scratch_datum(bcx, ty::mk_uint(), "__write_guard_bits", false); let freeze_item = match freeze_kind { DynaImm => BorrowAsImmFnLangItem, DynaMut => BorrowAsMutFnLangItem, }; let box_ptr = Load(bcx, PointerCast(bcx, scratch.val, Type::i8p().ptr_to())); let llresult = unpack_result!(bcx, callee::trans_lang_call( bcx, langcall(bcx, Some(span), "freeze", freeze_item), [ box_ptr, filename, line ], Some(expr::SaveIn(scratch_bits.val)))); if bcx.tcx().sess.debug_borrows() { bcx = callee::trans_lang_call( bcx, langcall(bcx, Some(span), "freeze", RecordBorrowFnLangItem), [ box_ptr, llresult, filename, line ], Some(expr::Ignore)).bcx; } add_clean_return_to_mut(cleanup_bcx, root_info.scope, root_key, scratch.val, scratch_bits.val, filename, line); } } bcx } fn perform_write_guard(datum: &Datum, bcx: @Block, span: Span) -> @Block { debug!("perform_write_guard"); let llval = datum.to_value_llval(bcx); let (filename, line) = filename_and_line_num_from_span(bcx, span); callee::trans_lang_call( bcx, langcall(bcx, Some(span), "write guard", CheckNotBorrowedFnLangItem), [PointerCast(bcx, llval, Type::i8p()), filename, line], Some(expr::Ignore)).bcx }