2012-12-03 16:48:01 -08:00
|
|
|
// 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 <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.
|
|
|
|
|
2013-01-07 14:16:52 -08:00
|
|
|
|
2012-09-04 11:54:36 -07:00
|
|
|
use back::abi;
|
2013-02-25 14:11:21 -05:00
|
|
|
use lib;
|
2013-06-16 15:45:48 +12:00
|
|
|
use lib::llvm::{llvm, ValueRef};
|
2013-07-15 20:42:13 -07:00
|
|
|
use middle::lang_items::StrDupUniqFnLangItem;
|
2013-02-25 14:11:21 -05:00
|
|
|
use middle::trans::base;
|
|
|
|
use middle::trans::base::*;
|
2012-12-13 13:05:22 -08:00
|
|
|
use middle::trans::build::*;
|
2013-02-25 14:11:21 -05:00
|
|
|
use middle::trans::callee;
|
2012-12-13 13:05:22 -08:00
|
|
|
use middle::trans::common::*;
|
|
|
|
use middle::trans::datum::*;
|
|
|
|
use middle::trans::expr::{Dest, Ignore, SaveIn};
|
2012-12-23 17:41:37 -05:00
|
|
|
use middle::trans::expr;
|
|
|
|
use middle::trans::glue;
|
2013-10-29 23:34:43 -07:00
|
|
|
use middle::trans::machine::{llsize_of, nonzero_llsize_of, llsize_of_alloc};
|
2012-12-23 17:41:37 -05:00
|
|
|
use middle::trans::type_of;
|
|
|
|
use middle::ty;
|
2012-12-13 13:05:22 -08:00
|
|
|
use util::common::indenter;
|
2012-09-04 11:54:36 -07:00
|
|
|
use util::ppaux::ty_to_str;
|
2012-12-13 13:05:22 -08:00
|
|
|
|
2013-06-16 22:52:44 +12:00
|
|
|
use middle::trans::type_::Type;
|
|
|
|
|
2013-06-28 18:32:26 -04:00
|
|
|
use std::option::None;
|
2012-12-13 13:05:22 -08:00
|
|
|
use syntax::ast;
|
2013-02-25 14:11:21 -05:00
|
|
|
use syntax::codemap;
|
2011-08-22 11:48:00 -07:00
|
|
|
|
2012-06-14 17:34:06 -07:00
|
|
|
// Boxed vector types are in some sense currently a "shorthand" for a box
|
|
|
|
// containing an unboxed vector. This expands a boxed vector type into such an
|
|
|
|
// expanded type. It doesn't respect mutability, but that doesn't matter at
|
|
|
|
// this point.
|
2013-01-30 11:46:19 -08:00
|
|
|
pub fn expand_boxed_vec_ty(tcx: ty::ctxt, t: ty::t) -> ty::t {
|
2012-06-14 17:34:06 -07:00
|
|
|
let unit_ty = ty::sequence_element_type(tcx, t);
|
|
|
|
let unboxed_vec_ty = ty::mk_mut_unboxed_vec(tcx, unit_ty);
|
2012-09-11 16:20:31 -07:00
|
|
|
match ty::get(t).sty {
|
2012-08-03 19:59:04 -07:00
|
|
|
ty::ty_estr(ty::vstore_uniq) | ty::ty_evec(_, ty::vstore_uniq) => {
|
2013-07-15 14:23:42 -04:00
|
|
|
ty::mk_imm_uniq(tcx, unboxed_vec_ty)
|
2012-06-14 17:34:06 -07:00
|
|
|
}
|
2012-08-03 19:59:04 -07:00
|
|
|
ty::ty_estr(ty::vstore_box) | ty::ty_evec(_, ty::vstore_box) => {
|
2012-06-14 17:34:06 -07:00
|
|
|
ty::mk_imm_box(tcx, unboxed_vec_ty)
|
|
|
|
}
|
2013-05-19 01:07:44 -04:00
|
|
|
_ => tcx.sess.bug("non boxed-vec type \
|
|
|
|
in tvec::expand_boxed_vec_ty")
|
2012-06-14 17:34:06 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn get_fill(bcx: @Block, vptr: ValueRef) -> ValueRef {
|
2013-06-17 16:23:24 +12:00
|
|
|
let _icx = push_ctxt("tvec::get_fill");
|
2012-08-27 12:16:37 -07:00
|
|
|
Load(bcx, GEPi(bcx, vptr, [0u, abi::vec_elt_fill]))
|
2011-08-25 10:18:02 +02:00
|
|
|
}
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn set_fill(bcx: @Block, vptr: ValueRef, fill: ValueRef) {
|
2012-08-27 12:16:37 -07:00
|
|
|
Store(bcx, fill, GEPi(bcx, vptr, [0u, abi::vec_elt_fill]));
|
2012-03-22 16:02:34 +01:00
|
|
|
}
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn get_alloc(bcx: @Block, vptr: ValueRef) -> ValueRef {
|
2012-08-27 12:16:37 -07:00
|
|
|
Load(bcx, GEPi(bcx, vptr, [0u, abi::vec_elt_alloc]))
|
2012-03-22 16:02:34 +01:00
|
|
|
}
|
2012-05-21 18:36:52 -07:00
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn get_bodyptr(bcx: @Block, vptr: ValueRef, t: ty::t) -> ValueRef {
|
2013-11-05 14:50:33 -05:00
|
|
|
if ty::type_contents(bcx.tcx(), t).owns_managed() {
|
2013-07-15 14:23:42 -04:00
|
|
|
GEPi(bcx, vptr, [0u, abi::box_field_body])
|
|
|
|
} else {
|
|
|
|
vptr
|
|
|
|
}
|
2012-05-21 18:36:52 -07:00
|
|
|
}
|
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn get_dataptr(bcx: @Block, vptr: ValueRef) -> ValueRef {
|
2013-06-17 16:23:24 +12:00
|
|
|
let _icx = push_ctxt("tvec::get_dataptr");
|
2012-08-27 12:16:37 -07:00
|
|
|
GEPi(bcx, vptr, [0u, abi::vec_elt_elems, 0u])
|
2011-08-25 10:18:02 +02:00
|
|
|
}
|
2011-08-22 11:48:00 -07:00
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn pointer_add_byte(bcx: @Block, ptr: ValueRef, bytes: ValueRef) -> ValueRef {
|
2013-10-16 12:04:51 -04:00
|
|
|
let _icx = push_ctxt("tvec::pointer_add_byte");
|
2011-08-25 10:18:02 +02:00
|
|
|
let old_ty = val_ty(ptr);
|
2013-06-16 15:45:48 +12:00
|
|
|
let bptr = PointerCast(bcx, ptr, Type::i8p());
|
2013-05-19 01:07:44 -04:00
|
|
|
return PointerCast(bcx, InBoundsGEP(bcx, bptr, [bytes]), old_ty);
|
2011-08-25 10:18:02 +02:00
|
|
|
}
|
2011-08-22 11:48:00 -07:00
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn alloc_raw(bcx: @Block, unit_ty: ty::t,
|
2013-01-30 11:46:19 -08:00
|
|
|
fill: ValueRef, alloc: ValueRef, heap: heap) -> Result {
|
2013-06-17 16:23:24 +12:00
|
|
|
let _icx = push_ctxt("tvec::alloc_uniq");
|
2012-02-21 14:20:18 +01:00
|
|
|
let ccx = bcx.ccx();
|
2012-06-12 14:54:02 -07:00
|
|
|
|
|
|
|
let vecbodyty = ty::mk_mut_unboxed_vec(bcx.tcx(), unit_ty);
|
|
|
|
let vecsize = Add(bcx, alloc, llsize_of(ccx, ccx.opaque_vec_type));
|
|
|
|
|
2013-07-15 14:23:42 -04:00
|
|
|
if heap == heap_exchange {
|
|
|
|
let Result { bcx: bcx, val: val } = malloc_raw_dyn(bcx, vecbodyty, heap_exchange, vecsize);
|
|
|
|
Store(bcx, fill, GEPi(bcx, val, [0u, abi::vec_elt_fill]));
|
|
|
|
Store(bcx, alloc, GEPi(bcx, val, [0u, abi::vec_elt_alloc]));
|
|
|
|
return rslt(bcx, val);
|
2013-07-02 19:24:58 -04:00
|
|
|
} else {
|
2013-12-11 17:04:50 -08:00
|
|
|
let base::MallocResult {bcx, smart_ptr: bx, body} =
|
2013-07-15 14:23:42 -04:00
|
|
|
base::malloc_general_dyn(bcx, vecbodyty, heap, vecsize);
|
|
|
|
Store(bcx, fill, GEPi(bcx, body, [0u, abi::vec_elt_fill]));
|
|
|
|
Store(bcx, alloc, GEPi(bcx, body, [0u, abi::vec_elt_alloc]));
|
|
|
|
base::maybe_set_managed_unique_rc(bcx, bx, heap);
|
|
|
|
return rslt(bcx, bx);
|
2013-07-02 19:24:58 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn alloc_uniq_raw(bcx: @Block, unit_ty: ty::t,
|
2013-01-30 11:46:19 -08:00
|
|
|
fill: ValueRef, alloc: ValueRef) -> Result {
|
2013-07-15 14:23:42 -04:00
|
|
|
alloc_raw(bcx, unit_ty, fill, alloc, base::heap_for_unique(bcx, unit_ty))
|
2012-06-14 17:34:06 -07:00
|
|
|
}
|
2011-08-29 22:30:18 +02:00
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn alloc_vec(bcx: @Block,
|
2013-01-30 11:46:19 -08:00
|
|
|
unit_ty: ty::t,
|
|
|
|
elts: uint,
|
|
|
|
heap: heap)
|
|
|
|
-> Result {
|
2013-06-17 16:23:24 +12:00
|
|
|
let _icx = push_ctxt("tvec::alloc_uniq");
|
2012-02-21 14:20:18 +01:00
|
|
|
let ccx = bcx.ccx();
|
2012-05-21 18:36:52 -07:00
|
|
|
let llunitty = type_of::type_of(ccx, unit_ty);
|
2013-01-08 00:24:43 -08:00
|
|
|
let unit_sz = nonzero_llsize_of(ccx, llunitty);
|
2011-08-25 10:18:02 +02:00
|
|
|
|
2011-10-14 20:38:24 -07:00
|
|
|
let fill = Mul(bcx, C_uint(ccx, elts), unit_sz);
|
2012-03-12 10:05:15 +01:00
|
|
|
let alloc = if elts < 4u { Mul(bcx, C_int(ccx, 4), unit_sz) }
|
|
|
|
else { fill };
|
2012-08-28 15:54:45 -07:00
|
|
|
let Result {bcx: bcx, val: vptr} =
|
|
|
|
alloc_raw(bcx, unit_ty, fill, alloc, heap);
|
|
|
|
return rslt(bcx, vptr);
|
2011-08-25 10:18:02 +02:00
|
|
|
}
|
2011-08-29 22:30:18 +02:00
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn make_drop_glue_unboxed(bcx: @Block, vptr: ValueRef, vec_ty: ty::t) ->
|
|
|
|
@Block {
|
2013-06-17 16:23:24 +12:00
|
|
|
let _icx = push_ctxt("tvec::make_drop_glue_unboxed");
|
2013-06-04 21:43:41 -07:00
|
|
|
let tcx = bcx.tcx();
|
|
|
|
let unit_ty = ty::sequence_element_type(tcx, vec_ty);
|
2012-06-14 17:34:06 -07:00
|
|
|
if ty::type_needs_drop(tcx, unit_ty) {
|
2012-08-28 15:54:45 -07:00
|
|
|
iter_vec_unboxed(bcx, vptr, vec_ty, glue::drop_ty)
|
2012-06-14 17:34:06 -07:00
|
|
|
} else { bcx }
|
2011-08-22 13:30:53 -07:00
|
|
|
}
|
|
|
|
|
2013-01-30 11:46:19 -08:00
|
|
|
pub struct VecTypes {
|
2012-09-07 14:50:47 -07:00
|
|
|
vec_ty: ty::t,
|
|
|
|
unit_ty: ty::t,
|
2013-06-16 15:45:48 +12:00
|
|
|
llunit_ty: Type,
|
2013-10-29 23:34:43 -07:00
|
|
|
llunit_size: ValueRef,
|
|
|
|
llunit_alloc_size: uint
|
2012-08-03 18:48:17 -07:00
|
|
|
}
|
|
|
|
|
2013-05-31 15:17:22 -07:00
|
|
|
impl VecTypes {
|
2013-06-13 19:19:50 +12:00
|
|
|
pub fn to_str(&self, ccx: &CrateContext) -> ~str {
|
2013-10-29 23:34:43 -07:00
|
|
|
format!("VecTypes \\{vec_ty={}, unit_ty={}, llunit_ty={}, llunit_size={}, \
|
|
|
|
llunit_alloc_size={}\\}",
|
2012-08-28 15:54:45 -07:00
|
|
|
ty_to_str(ccx.tcx, self.vec_ty),
|
|
|
|
ty_to_str(ccx.tcx, self.unit_ty),
|
2013-06-15 15:16:03 +12:00
|
|
|
ccx.tn.type_to_str(self.llunit_ty),
|
2013-10-29 23:34:43 -07:00
|
|
|
ccx.tn.val_to_str(self.llunit_size),
|
|
|
|
self.llunit_alloc_size)
|
2012-08-28 15:54:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn trans_fixed_vstore(bcx: @Block,
|
2013-09-30 19:37:17 +02:00
|
|
|
vstore_expr: &ast::Expr,
|
2013-09-02 03:45:37 +02:00
|
|
|
content_expr: &ast::Expr,
|
2013-01-30 11:46:19 -08:00
|
|
|
dest: expr::Dest)
|
2013-12-18 14:54:42 -08:00
|
|
|
-> @Block {
|
2012-08-28 15:54:45 -07:00
|
|
|
//!
|
|
|
|
//
|
2012-10-10 00:28:04 -04:00
|
|
|
// [...] allocates a fixed-size array and moves it around "by value".
|
2012-08-28 15:54:45 -07:00
|
|
|
// In this case, it means that the caller has already given us a location
|
|
|
|
// to store the array of the suitable size, so all we have to do is
|
|
|
|
// generate the content.
|
|
|
|
|
2013-10-21 13:08:31 -07:00
|
|
|
debug!("trans_fixed_vstore(vstore_expr={}, dest={:?})",
|
2012-08-28 15:54:45 -07:00
|
|
|
bcx.expr_to_str(vstore_expr), dest.to_str(bcx.ccx()));
|
|
|
|
let _indenter = indenter();
|
|
|
|
|
|
|
|
let vt = vec_types_from_expr(bcx, vstore_expr);
|
|
|
|
|
|
|
|
return match dest {
|
|
|
|
Ignore => write_content(bcx, &vt, vstore_expr, content_expr, dest),
|
|
|
|
SaveIn(lldest) => {
|
|
|
|
// lldest will have type *[T x N], but we want the type *T,
|
|
|
|
// so use GEP to convert:
|
|
|
|
let lldest = GEPi(bcx, lldest, [0, 0]);
|
|
|
|
write_content(bcx, &vt, vstore_expr, content_expr, SaveIn(lldest))
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn trans_slice_vstore(bcx: @Block,
|
2013-09-30 19:37:17 +02:00
|
|
|
vstore_expr: &ast::Expr,
|
|
|
|
content_expr: &ast::Expr,
|
2013-01-30 11:46:19 -08:00
|
|
|
dest: expr::Dest)
|
2013-12-18 14:54:42 -08:00
|
|
|
-> @Block {
|
2012-08-28 15:54:45 -07:00
|
|
|
//!
|
|
|
|
//
|
|
|
|
// &[...] allocates memory on the stack and writes the values into it,
|
|
|
|
// returning a slice (pair of ptr, len). &"..." is similar except that
|
|
|
|
// the memory can be statically allocated.
|
|
|
|
|
2012-03-15 09:47:03 -04:00
|
|
|
let ccx = bcx.ccx();
|
2012-08-03 18:48:17 -07:00
|
|
|
|
2013-10-21 13:08:31 -07:00
|
|
|
debug!("trans_slice_vstore(vstore_expr={}, dest={})",
|
2012-08-28 15:54:45 -07:00
|
|
|
bcx.expr_to_str(vstore_expr), dest.to_str(ccx));
|
|
|
|
let _indenter = indenter();
|
|
|
|
|
|
|
|
// Handle the &"..." case:
|
|
|
|
match content_expr.node {
|
2013-10-08 02:49:10 +02:00
|
|
|
ast::ExprLit(@codemap::Spanned {node: ast::lit_str(s, _), span: _}) => {
|
2012-08-28 15:54:45 -07:00
|
|
|
return trans_lit_str(bcx, content_expr, s, dest);
|
2011-09-27 10:50:18 +02:00
|
|
|
}
|
2012-08-28 15:54:45 -07:00
|
|
|
_ => {}
|
2011-09-27 10:50:18 +02:00
|
|
|
}
|
2012-04-19 15:42:02 -07:00
|
|
|
|
2012-08-28 15:54:45 -07:00
|
|
|
// Handle the &[...] case:
|
|
|
|
let vt = vec_types_from_expr(bcx, vstore_expr);
|
|
|
|
let count = elements_required(bcx, content_expr);
|
2013-10-21 13:08:31 -07:00
|
|
|
debug!("vt={}, count={:?}", vt.to_str(ccx), count);
|
2012-08-28 15:54:45 -07:00
|
|
|
|
|
|
|
// Make a fixed-length backing array and allocate it on the stack.
|
|
|
|
let llcount = C_uint(ccx, count);
|
|
|
|
let llfixed = base::arrayalloca(bcx, vt.llunit_ty, llcount);
|
|
|
|
|
|
|
|
// Arrange for the backing array to be cleaned up.
|
|
|
|
let fixed_ty = ty::mk_evec(bcx.tcx(),
|
2013-09-02 03:45:37 +02:00
|
|
|
ty::mt {ty: vt.unit_ty, mutbl: ast::MutMutable},
|
2012-08-28 15:54:45 -07:00
|
|
|
ty::vstore_fixed(count));
|
2013-06-16 15:45:48 +12:00
|
|
|
let llfixed_ty = type_of::type_of(bcx.ccx(), fixed_ty).ptr_to();
|
2012-08-28 15:54:45 -07:00
|
|
|
let llfixed_casted = BitCast(bcx, llfixed, llfixed_ty);
|
|
|
|
add_clean(bcx, llfixed_casted, fixed_ty);
|
|
|
|
|
|
|
|
// Generate the content into the backing array.
|
|
|
|
let bcx = write_content(bcx, &vt, vstore_expr,
|
|
|
|
content_expr, SaveIn(llfixed));
|
|
|
|
|
|
|
|
// Finally, create the slice pair itself.
|
|
|
|
match dest {
|
|
|
|
Ignore => {}
|
|
|
|
SaveIn(lldest) => {
|
|
|
|
Store(bcx, llfixed, GEPi(bcx, lldest, [0u, abi::slice_elt_base]));
|
2013-10-15 00:37:32 -04:00
|
|
|
Store(bcx, llcount, GEPi(bcx, lldest, [0u, abi::slice_elt_len]));
|
2012-08-28 15:54:45 -07:00
|
|
|
}
|
2012-08-03 18:48:17 -07:00
|
|
|
}
|
|
|
|
|
2012-08-28 15:54:45 -07:00
|
|
|
return bcx;
|
|
|
|
}
|
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn trans_lit_str(bcx: @Block,
|
2013-09-30 19:37:17 +02:00
|
|
|
lit_expr: &ast::Expr,
|
2013-06-13 03:02:55 +10:00
|
|
|
str_lit: @str,
|
2013-01-30 11:46:19 -08:00
|
|
|
dest: Dest)
|
2013-12-18 14:54:42 -08:00
|
|
|
-> @Block {
|
2012-08-28 15:54:45 -07:00
|
|
|
//!
|
|
|
|
//
|
|
|
|
// Literal strings translate to slices into static memory. This is
|
|
|
|
// different from trans_slice_vstore() above because it does need to copy
|
|
|
|
// the content anywhere.
|
|
|
|
|
2013-10-21 13:08:31 -07:00
|
|
|
debug!("trans_lit_str(lit_expr={}, dest={})",
|
2012-08-28 15:54:45 -07:00
|
|
|
bcx.expr_to_str(lit_expr),
|
|
|
|
dest.to_str(bcx.ccx()));
|
|
|
|
let _indenter = indenter();
|
|
|
|
|
|
|
|
match dest {
|
|
|
|
Ignore => bcx,
|
|
|
|
SaveIn(lldest) => {
|
2013-01-10 21:23:07 -08:00
|
|
|
unsafe {
|
2013-08-14 23:40:28 -07:00
|
|
|
let bytes = str_lit.len();
|
2013-01-10 21:23:07 -08:00
|
|
|
let llbytes = C_uint(bcx.ccx(), bytes);
|
2013-02-10 16:33:16 -08:00
|
|
|
let llcstr = C_cstr(bcx.ccx(), str_lit);
|
2013-06-16 15:45:48 +12:00
|
|
|
let llcstr = llvm::LLVMConstPointerCast(llcstr, Type::i8p().to_ref());
|
|
|
|
Store(bcx, llcstr,
|
2013-01-10 21:23:07 -08:00
|
|
|
GEPi(bcx, lldest, [0u, abi::slice_elt_base]));
|
2013-06-16 15:45:48 +12:00
|
|
|
Store(bcx, llbytes,
|
2013-01-10 21:23:07 -08:00
|
|
|
GEPi(bcx, lldest, [0u, abi::slice_elt_len]));
|
|
|
|
bcx
|
|
|
|
}
|
2012-08-28 15:54:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn trans_uniq_or_managed_vstore(bcx: @Block, heap: heap, vstore_expr: &ast::Expr,
|
2013-09-02 03:45:37 +02:00
|
|
|
content_expr: &ast::Expr) -> DatumBlock {
|
2012-08-28 15:54:45 -07:00
|
|
|
//!
|
|
|
|
//
|
|
|
|
// @[...] or ~[...] (also @"..." or ~"...") allocate boxes in the
|
|
|
|
// appropriate heap and write the array elements into them.
|
|
|
|
|
2013-10-21 13:08:31 -07:00
|
|
|
debug!("trans_uniq_or_managed_vstore(vstore_expr={}, heap={:?})",
|
2012-08-28 15:54:45 -07:00
|
|
|
bcx.expr_to_str(vstore_expr), heap);
|
|
|
|
let _indenter = indenter();
|
|
|
|
|
2013-01-29 19:59:52 -08:00
|
|
|
// Handle ~"".
|
|
|
|
match heap {
|
2013-07-15 14:23:42 -04:00
|
|
|
heap_exchange => {
|
2013-01-29 19:59:52 -08:00
|
|
|
match content_expr.node {
|
2013-09-02 03:45:37 +02:00
|
|
|
ast::ExprLit(@codemap::Spanned {
|
2013-10-08 02:49:10 +02:00
|
|
|
node: ast::lit_str(s, _), span
|
2013-01-29 19:59:52 -08:00
|
|
|
}) => {
|
2013-02-10 16:33:16 -08:00
|
|
|
let llptrval = C_cstr(bcx.ccx(), s);
|
2013-06-16 15:45:48 +12:00
|
|
|
let llptrval = PointerCast(bcx, llptrval, Type::i8p());
|
2013-01-29 19:59:52 -08:00
|
|
|
let llsizeval = C_uint(bcx.ccx(), s.len());
|
|
|
|
let typ = ty::mk_estr(bcx.tcx(), ty::vstore_uniq);
|
2013-06-20 15:21:37 -04:00
|
|
|
let lldestval = scratch_datum(bcx, typ, "", false);
|
2013-07-15 20:42:13 -07:00
|
|
|
let alloc_fn = langcall(bcx, Some(span), "",
|
|
|
|
StrDupUniqFnLangItem);
|
2013-02-27 18:34:04 -08:00
|
|
|
let bcx = callee::trans_lang_call(
|
2013-01-29 19:59:52 -08:00
|
|
|
bcx,
|
2013-07-15 20:42:13 -07:00
|
|
|
alloc_fn,
|
2013-05-19 01:07:44 -04:00
|
|
|
[ llptrval, llsizeval ],
|
2013-07-08 08:12:01 +02:00
|
|
|
Some(expr::SaveIn(lldestval.to_ref_llval(bcx)))).bcx;
|
2013-02-25 14:11:21 -05:00
|
|
|
return DatumBlock {
|
2013-01-29 19:59:52 -08:00
|
|
|
bcx: bcx,
|
|
|
|
datum: lldestval
|
|
|
|
};
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
}
|
2013-10-21 13:08:31 -07:00
|
|
|
heap_exchange_closure => fail!("vectors use exchange_alloc"),
|
2013-02-20 15:02:21 -08:00
|
|
|
heap_managed | heap_managed_unique => {}
|
2013-01-29 19:59:52 -08:00
|
|
|
}
|
|
|
|
|
2012-08-28 15:54:45 -07:00
|
|
|
let vt = vec_types_from_expr(bcx, vstore_expr);
|
|
|
|
let count = elements_required(bcx, content_expr);
|
|
|
|
|
|
|
|
let Result {bcx, val} = alloc_vec(bcx, vt.unit_ty, count, heap);
|
2013-02-20 15:02:21 -08:00
|
|
|
|
2012-08-28 15:54:45 -07:00
|
|
|
add_clean_free(bcx, val, heap);
|
2013-07-15 14:23:42 -04:00
|
|
|
let dataptr = get_dataptr(bcx, get_bodyptr(bcx, val, vt.vec_ty));
|
2012-08-28 15:54:45 -07:00
|
|
|
|
2013-10-21 13:08:31 -07:00
|
|
|
debug!("alloc_vec() returned val={}, dataptr={}",
|
2013-06-15 15:16:03 +12:00
|
|
|
bcx.val_to_str(val), bcx.val_to_str(dataptr));
|
2012-08-28 15:54:45 -07:00
|
|
|
|
|
|
|
let bcx = write_content(bcx, &vt, vstore_expr,
|
|
|
|
content_expr, SaveIn(dataptr));
|
|
|
|
|
|
|
|
revoke_clean(bcx, val);
|
2012-04-19 15:42:02 -07:00
|
|
|
|
2012-08-28 15:54:45 -07:00
|
|
|
return immediate_rvalue_bcx(bcx, val, vt.vec_ty);
|
|
|
|
}
|
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn write_content(bcx: @Block,
|
2013-01-30 11:46:19 -08:00
|
|
|
vt: &VecTypes,
|
2013-09-30 19:37:17 +02:00
|
|
|
vstore_expr: &ast::Expr,
|
2013-09-02 03:45:37 +02:00
|
|
|
content_expr: &ast::Expr,
|
2013-01-30 11:46:19 -08:00
|
|
|
dest: Dest)
|
2013-12-18 14:54:42 -08:00
|
|
|
-> @Block {
|
2013-06-17 16:23:24 +12:00
|
|
|
let _icx = push_ctxt("tvec::write_content");
|
2012-08-28 15:54:45 -07:00
|
|
|
let mut bcx = bcx;
|
|
|
|
|
2013-10-21 13:08:31 -07:00
|
|
|
debug!("write_content(vt={}, dest={}, vstore_expr={:?})",
|
2012-08-28 15:54:45 -07:00
|
|
|
vt.to_str(bcx.ccx()),
|
|
|
|
dest.to_str(bcx.ccx()),
|
|
|
|
bcx.expr_to_str(vstore_expr));
|
|
|
|
let _indenter = indenter();
|
|
|
|
|
2013-03-20 01:17:42 -04:00
|
|
|
match content_expr.node {
|
2013-11-28 12:22:53 -08:00
|
|
|
ast::ExprLit(@codemap::Spanned { node: ast::lit_str(s, _), .. }) => {
|
2012-08-28 15:54:45 -07:00
|
|
|
match dest {
|
|
|
|
Ignore => {
|
|
|
|
return bcx;
|
|
|
|
}
|
|
|
|
SaveIn(lldest) => {
|
2013-08-04 13:22:56 -07:00
|
|
|
let bytes = s.len();
|
2012-08-28 15:54:45 -07:00
|
|
|
let llbytes = C_uint(bcx.ccx(), bytes);
|
2013-02-10 16:33:16 -08:00
|
|
|
let llcstr = C_cstr(bcx.ccx(), s);
|
2013-05-25 09:52:25 -04:00
|
|
|
base::call_memcpy(bcx, lldest, llcstr, llbytes, 1);
|
2012-08-28 15:54:45 -07:00
|
|
|
return bcx;
|
|
|
|
}
|
2012-08-03 18:48:17 -07:00
|
|
|
}
|
|
|
|
}
|
2013-09-02 03:45:37 +02:00
|
|
|
ast::ExprVec(ref elements, _) => {
|
2012-08-28 15:54:45 -07:00
|
|
|
match dest {
|
|
|
|
Ignore => {
|
2013-08-03 12:45:23 -04:00
|
|
|
for element in elements.iter() {
|
2012-09-19 16:55:01 -07:00
|
|
|
bcx = expr::trans_into(bcx, *element, Ignore);
|
2012-08-28 15:54:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SaveIn(lldest) => {
|
|
|
|
let mut temp_cleanups = ~[];
|
2013-08-03 12:45:23 -04:00
|
|
|
for (i, element) in elements.iter().enumerate() {
|
2012-08-28 15:54:45 -07:00
|
|
|
let lleltptr = GEPi(bcx, lldest, [i]);
|
2013-10-21 13:08:31 -07:00
|
|
|
debug!("writing index {:?} with lleltptr={:?}",
|
2013-06-15 15:16:03 +12:00
|
|
|
i, bcx.val_to_str(lleltptr));
|
2012-09-21 18:43:30 -07:00
|
|
|
bcx = expr::trans_into(bcx, *element,
|
2012-08-28 15:54:45 -07:00
|
|
|
SaveIn(lleltptr));
|
|
|
|
add_clean_temp_mem(bcx, lleltptr, vt.unit_ty);
|
2012-09-26 17:33:34 -07:00
|
|
|
temp_cleanups.push(lleltptr);
|
2012-08-28 15:54:45 -07:00
|
|
|
}
|
2013-08-03 12:45:23 -04:00
|
|
|
for cleanup in temp_cleanups.iter() {
|
2012-09-18 21:41:37 -07:00
|
|
|
revoke_clean(bcx, *cleanup);
|
2012-08-28 15:54:45 -07:00
|
|
|
}
|
2012-08-03 18:48:17 -07:00
|
|
|
}
|
|
|
|
}
|
2012-08-28 15:54:45 -07:00
|
|
|
return bcx;
|
|
|
|
}
|
2013-09-02 03:45:37 +02:00
|
|
|
ast::ExprRepeat(element, count_expr, _) => {
|
2012-08-28 15:54:45 -07:00
|
|
|
match dest {
|
|
|
|
Ignore => {
|
|
|
|
return expr::trans_into(bcx, element, Ignore);
|
|
|
|
}
|
|
|
|
SaveIn(lldest) => {
|
2013-07-16 04:27:54 -04:00
|
|
|
let count = ty::eval_repeat_count(&bcx.tcx(), count_expr);
|
2012-08-28 15:54:45 -07:00
|
|
|
if count == 0 {
|
|
|
|
return bcx;
|
|
|
|
}
|
|
|
|
|
2013-03-25 15:46:10 -07:00
|
|
|
// Some cleanup would be required in the case in which failure happens
|
|
|
|
// during a copy. But given that copy constructors are not overridable,
|
|
|
|
// this can only happen as a result of OOM. So we just skip out on the
|
|
|
|
// cleanup since things would *probably* be broken at that point anyways.
|
|
|
|
|
2013-03-24 00:59:44 -07:00
|
|
|
let elem = unpack_datum!(bcx, {
|
2012-08-28 15:54:45 -07:00
|
|
|
expr::trans_to_datum(bcx, element)
|
|
|
|
});
|
|
|
|
|
2013-10-29 23:34:43 -07:00
|
|
|
iter_vec_loop(bcx, lldest, vt,
|
|
|
|
C_uint(bcx.ccx(), count), |set_bcx, lleltptr, _| {
|
|
|
|
elem.copy_to(set_bcx, INIT, lleltptr)
|
|
|
|
})
|
2012-08-28 15:54:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
bcx.tcx().sess.span_bug(content_expr.span,
|
2013-05-03 01:28:53 +09:00
|
|
|
"Unexpected evec content");
|
2012-08-03 18:48:17 -07:00
|
|
|
}
|
2011-08-22 11:48:00 -07:00
|
|
|
}
|
2012-08-28 15:54:45 -07:00
|
|
|
}
|
2012-04-19 15:42:02 -07:00
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn vec_types_from_expr(bcx: @Block, vec_expr: &ast::Expr) -> VecTypes {
|
2012-08-28 15:54:45 -07:00
|
|
|
let vec_ty = node_id_type(bcx, vec_expr.id);
|
|
|
|
vec_types(bcx, vec_ty)
|
|
|
|
}
|
2012-04-19 15:42:02 -07:00
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn vec_types(bcx: @Block, vec_ty: ty::t) -> VecTypes {
|
2012-08-28 15:54:45 -07:00
|
|
|
let ccx = bcx.ccx();
|
|
|
|
let unit_ty = ty::sequence_element_type(bcx.tcx(), vec_ty);
|
|
|
|
let llunit_ty = type_of::type_of(ccx, unit_ty);
|
2013-01-08 00:24:43 -08:00
|
|
|
let llunit_size = nonzero_llsize_of(ccx, llunit_ty);
|
2013-10-29 23:34:43 -07:00
|
|
|
let llunit_alloc_size = llsize_of_alloc(ccx, llunit_ty);
|
2013-01-08 00:24:43 -08:00
|
|
|
|
2012-08-28 15:54:45 -07:00
|
|
|
VecTypes {vec_ty: vec_ty,
|
|
|
|
unit_ty: unit_ty,
|
|
|
|
llunit_ty: llunit_ty,
|
2013-10-29 23:34:43 -07:00
|
|
|
llunit_size: llunit_size,
|
|
|
|
llunit_alloc_size: llunit_alloc_size}
|
2011-08-22 11:48:00 -07:00
|
|
|
}
|
2011-09-27 20:21:44 +02:00
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn elements_required(bcx: @Block, content_expr: &ast::Expr) -> uint {
|
2012-08-28 15:54:45 -07:00
|
|
|
//! Figure out the number of elements we need to store this content
|
|
|
|
|
2013-03-20 01:17:42 -04:00
|
|
|
match content_expr.node {
|
2013-11-28 12:22:53 -08:00
|
|
|
ast::ExprLit(@codemap::Spanned { node: ast::lit_str(s, _), .. }) => {
|
2013-08-04 13:22:56 -07:00
|
|
|
s.len()
|
2012-12-27 14:36:00 -05:00
|
|
|
},
|
2013-09-02 03:45:37 +02:00
|
|
|
ast::ExprVec(ref es, _) => es.len(),
|
|
|
|
ast::ExprRepeat(_, count_expr, _) => {
|
2013-07-16 04:27:54 -04:00
|
|
|
ty::eval_repeat_count(&bcx.tcx(), count_expr)
|
2012-08-28 15:54:45 -07:00
|
|
|
}
|
|
|
|
_ => bcx.tcx().sess.span_bug(content_expr.span,
|
2013-05-03 01:28:53 +09:00
|
|
|
"Unexpected evec content")
|
2012-04-09 17:32:49 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn get_base_and_byte_len(bcx: @Block, llval: ValueRef,
|
2013-01-30 11:46:19 -08:00
|
|
|
vec_ty: ty::t) -> (ValueRef, ValueRef) {
|
2012-08-28 15:54:45 -07:00
|
|
|
//!
|
|
|
|
//
|
|
|
|
// Converts a vector into the slice pair. The vector should be stored in
|
|
|
|
// `llval` which should be either immediate or by-ref as appropriate for
|
|
|
|
// the vector type. If you have a datum, you would probably prefer to
|
2013-10-16 11:59:35 -04:00
|
|
|
// call `Datum::get_base_and_byte_len()` which will handle any conversions for
|
2012-08-28 15:54:45 -07:00
|
|
|
// you.
|
2012-04-17 15:07:38 -07:00
|
|
|
|
2012-08-28 15:54:45 -07:00
|
|
|
let ccx = bcx.ccx();
|
|
|
|
let vt = vec_types(bcx, vec_ty);
|
2012-04-19 15:42:02 -07:00
|
|
|
|
2012-09-11 16:20:31 -07:00
|
|
|
let vstore = match ty::get(vt.vec_ty).sty {
|
2012-08-03 19:59:04 -07:00
|
|
|
ty::ty_estr(vst) | ty::ty_evec(_, vst) => vst,
|
|
|
|
_ => ty::vstore_uniq
|
2012-04-17 15:07:38 -07:00
|
|
|
};
|
|
|
|
|
2012-08-06 12:34:08 -07:00
|
|
|
match vstore {
|
2012-08-28 15:54:45 -07:00
|
|
|
ty::vstore_fixed(n) => {
|
|
|
|
let base = GEPi(bcx, llval, [0u, 0u]);
|
|
|
|
let len = Mul(bcx, C_uint(ccx, n), vt.llunit_size);
|
|
|
|
(base, len)
|
|
|
|
}
|
|
|
|
ty::vstore_slice(_) => {
|
|
|
|
let base = Load(bcx, GEPi(bcx, llval, [0u, abi::slice_elt_base]));
|
2013-10-15 00:37:32 -04:00
|
|
|
let count = Load(bcx, GEPi(bcx, llval, [0u, abi::slice_elt_len]));
|
|
|
|
let len = Mul(bcx, count, vt.llunit_size);
|
2012-08-28 15:54:45 -07:00
|
|
|
(base, len)
|
|
|
|
}
|
|
|
|
ty::vstore_uniq | ty::vstore_box => {
|
2013-07-15 14:23:42 -04:00
|
|
|
let body = get_bodyptr(bcx, llval, vec_ty);
|
2013-02-25 14:11:21 -05:00
|
|
|
(get_dataptr(bcx, body), get_fill(bcx, body))
|
2012-08-28 15:54:45 -07:00
|
|
|
}
|
2012-04-17 15:07:38 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn get_base_and_len(bcx: @Block, llval: ValueRef, vec_ty: ty::t) -> (ValueRef, ValueRef) {
|
2013-10-16 12:04:51 -04:00
|
|
|
//!
|
|
|
|
//
|
|
|
|
// Converts a vector into the slice pair. The vector should be stored in
|
|
|
|
// `llval` which should be either immediate or by-ref as appropriate for
|
|
|
|
// the vector type. If you have a datum, you would probably prefer to
|
|
|
|
// call `Datum::get_base_and_len()` which will handle any conversions for
|
|
|
|
// you.
|
|
|
|
|
|
|
|
let ccx = bcx.ccx();
|
|
|
|
let vt = vec_types(bcx, vec_ty);
|
|
|
|
|
|
|
|
let vstore = match ty::get(vt.vec_ty).sty {
|
|
|
|
ty::ty_estr(vst) | ty::ty_evec(_, vst) => vst,
|
|
|
|
_ => ty::vstore_uniq
|
|
|
|
};
|
|
|
|
|
|
|
|
match vstore {
|
|
|
|
ty::vstore_fixed(n) => {
|
|
|
|
let base = GEPi(bcx, llval, [0u, 0u]);
|
|
|
|
(base, C_uint(ccx, n))
|
|
|
|
}
|
|
|
|
ty::vstore_slice(_) => {
|
|
|
|
let base = Load(bcx, GEPi(bcx, llval, [0u, abi::slice_elt_base]));
|
|
|
|
let count = Load(bcx, GEPi(bcx, llval, [0u, abi::slice_elt_len]));
|
|
|
|
(base, count)
|
|
|
|
}
|
|
|
|
ty::vstore_uniq | ty::vstore_box => {
|
|
|
|
let body = get_bodyptr(bcx, llval, vec_ty);
|
|
|
|
(get_dataptr(bcx, body), UDiv(bcx, get_fill(bcx, body), vt.llunit_size))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub type iter_vec_block<'a> = 'a |@Block, ValueRef, ty::t|
|
|
|
|
-> @Block;
|
2011-08-22 11:48:00 -07:00
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn iter_vec_loop(bcx: @Block,
|
2013-10-29 23:34:43 -07:00
|
|
|
data_ptr: ValueRef,
|
|
|
|
vt: &VecTypes,
|
|
|
|
count: ValueRef,
|
|
|
|
f: iter_vec_block
|
2013-12-18 14:54:42 -08:00
|
|
|
) -> @Block {
|
2013-10-29 23:34:43 -07:00
|
|
|
let _icx = push_ctxt("tvec::iter_vec_loop");
|
|
|
|
|
|
|
|
let next_bcx = sub_block(bcx, "iter_vec_loop: while next");
|
|
|
|
let loop_bcx = loop_scope_block(bcx, next_bcx, None, "iter_vec_loop", None);
|
|
|
|
let cond_bcx = scope_block(loop_bcx, None, "iter_vec_loop: loop cond");
|
|
|
|
let body_bcx = scope_block(loop_bcx, None, "iter_vec_loop: body: main");
|
|
|
|
let inc_bcx = scope_block(loop_bcx, None, "iter_vec_loop: loop inc");
|
|
|
|
Br(bcx, loop_bcx.llbb);
|
|
|
|
|
|
|
|
let loop_counter = {
|
|
|
|
// i = 0
|
|
|
|
let i = alloca(loop_bcx, bcx.ccx().int_type, "__i");
|
|
|
|
Store(loop_bcx, C_uint(bcx.ccx(), 0), i);
|
|
|
|
|
|
|
|
Br(loop_bcx, cond_bcx.llbb);
|
|
|
|
i
|
|
|
|
};
|
|
|
|
|
|
|
|
{ // i < count
|
|
|
|
let lhs = Load(cond_bcx, loop_counter);
|
|
|
|
let rhs = count;
|
|
|
|
let cond_val = ICmp(cond_bcx, lib::llvm::IntULT, lhs, rhs);
|
|
|
|
|
|
|
|
CondBr(cond_bcx, cond_val, body_bcx.llbb, next_bcx.llbb);
|
|
|
|
}
|
|
|
|
|
|
|
|
{ // loop body
|
|
|
|
let i = Load(body_bcx, loop_counter);
|
|
|
|
let lleltptr = if vt.llunit_alloc_size == 0 {
|
|
|
|
data_ptr
|
|
|
|
} else {
|
|
|
|
InBoundsGEP(body_bcx, data_ptr, [i])
|
|
|
|
};
|
|
|
|
let body_bcx = f(body_bcx, lleltptr, vt.unit_ty);
|
|
|
|
|
|
|
|
Br(body_bcx, inc_bcx.llbb);
|
|
|
|
}
|
|
|
|
|
|
|
|
{ // i += 1
|
|
|
|
let i = Load(inc_bcx, loop_counter);
|
|
|
|
let plusone = Add(inc_bcx, i, C_uint(bcx.ccx(), 1));
|
|
|
|
Store(inc_bcx, plusone, loop_counter);
|
|
|
|
|
|
|
|
Br(inc_bcx, cond_bcx.llbb);
|
|
|
|
}
|
|
|
|
|
|
|
|
next_bcx
|
|
|
|
}
|
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn iter_vec_raw(bcx: @Block, data_ptr: ValueRef, vec_ty: ty::t,
|
|
|
|
fill: ValueRef, f: iter_vec_block) -> @Block {
|
2013-06-17 16:23:24 +12:00
|
|
|
let _icx = push_ctxt("tvec::iter_vec_raw");
|
2012-04-19 15:42:02 -07:00
|
|
|
|
2013-10-29 23:34:43 -07:00
|
|
|
let vt = vec_types(bcx, vec_ty);
|
|
|
|
if (vt.llunit_alloc_size == 0) {
|
|
|
|
// Special-case vectors with elements of size 0 so they don't go out of bounds (#9890)
|
|
|
|
iter_vec_loop(bcx, data_ptr, &vt, fill, f)
|
|
|
|
} else {
|
|
|
|
// Calculate the last pointer address we want to handle.
|
|
|
|
// FIXME (#3729): Optimize this when the size of the unit type is
|
|
|
|
// statically known to not use pointer casts, which tend to confuse
|
|
|
|
// LLVM.
|
|
|
|
let data_end_ptr = pointer_add_byte(bcx, data_ptr, fill);
|
|
|
|
|
|
|
|
// Now perform the iteration.
|
|
|
|
let header_bcx = base::sub_block(bcx, "iter_vec_loop_header");
|
|
|
|
Br(bcx, header_bcx.llbb);
|
|
|
|
let data_ptr =
|
|
|
|
Phi(header_bcx, val_ty(data_ptr), [data_ptr], [bcx.llbb]);
|
|
|
|
let not_yet_at_end =
|
|
|
|
ICmp(header_bcx, lib::llvm::IntULT, data_ptr, data_end_ptr);
|
|
|
|
let body_bcx = base::sub_block(header_bcx, "iter_vec_loop_body");
|
|
|
|
let next_bcx = base::sub_block(header_bcx, "iter_vec_next");
|
|
|
|
CondBr(header_bcx, not_yet_at_end, body_bcx.llbb, next_bcx.llbb);
|
|
|
|
let body_bcx = f(body_bcx, data_ptr, vt.unit_ty);
|
|
|
|
AddIncomingToPhi(data_ptr, InBoundsGEP(body_bcx, data_ptr,
|
|
|
|
[C_int(bcx.ccx(), 1)]),
|
|
|
|
body_bcx.llbb);
|
|
|
|
Br(body_bcx, header_bcx.llbb);
|
|
|
|
next_bcx
|
2012-04-19 15:42:02 -07:00
|
|
|
|
2013-10-29 23:34:43 -07:00
|
|
|
}
|
2012-04-19 15:42:02 -07:00
|
|
|
}
|
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn iter_vec_uniq(bcx: @Block, vptr: ValueRef, vec_ty: ty::t,
|
|
|
|
fill: ValueRef, f: iter_vec_block) -> @Block {
|
2013-06-17 16:23:24 +12:00
|
|
|
let _icx = push_ctxt("tvec::iter_vec_uniq");
|
2013-07-15 14:23:42 -04:00
|
|
|
let data_ptr = get_dataptr(bcx, get_bodyptr(bcx, vptr, vec_ty));
|
2012-04-19 15:42:02 -07:00
|
|
|
iter_vec_raw(bcx, data_ptr, vec_ty, fill, f)
|
2011-08-22 11:48:00 -07:00
|
|
|
}
|
2011-08-25 10:18:02 +02:00
|
|
|
|
2013-12-18 14:54:42 -08:00
|
|
|
pub fn iter_vec_unboxed(bcx: @Block, body_ptr: ValueRef, vec_ty: ty::t,
|
|
|
|
f: iter_vec_block) -> @Block {
|
2013-06-17 16:23:24 +12:00
|
|
|
let _icx = push_ctxt("tvec::iter_vec_unboxed");
|
2012-06-14 17:34:06 -07:00
|
|
|
let fill = get_fill(bcx, body_ptr);
|
|
|
|
let dataptr = get_dataptr(bcx, body_ptr);
|
2012-08-01 17:30:05 -07:00
|
|
|
return iter_vec_raw(bcx, dataptr, vec_ty, fill, f);
|
2011-08-29 16:14:24 +02:00
|
|
|
}
|