2011-03-25 12:19:20 -07:00
|
|
|
// xfail-stage0
|
2011-05-03 18:13:51 -07:00
|
|
|
// xfail-stage1
|
|
|
|
// xfail-stage2
|
2010-06-23 21:03:09 -07:00
|
|
|
// -*- rust -*-
|
|
|
|
|
2010-09-03 16:25:56 -07:00
|
|
|
use std;
|
2011-05-12 17:24:54 +02:00
|
|
|
import std::_str;
|
|
|
|
import std::_vec;
|
2010-09-03 16:25:56 -07:00
|
|
|
|
2010-11-09 14:15:07 -08:00
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
// FIXME: import std::dbg::const_refcount. Currently
|
2010-11-09 14:15:07 -08:00
|
|
|
// cross-crate const references don't work.
|
|
|
|
const uint const_refcount = 0x7bad_face_u;
|
|
|
|
|
2010-08-31 13:01:51 -07:00
|
|
|
fn fast_growth() {
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[int] v = [1,2,3,4,5];
|
|
|
|
v += [6,7,8,9,0];
|
2010-08-31 13:01:51 -07:00
|
|
|
|
2010-06-23 21:03:09 -07:00
|
|
|
log v.(9);
|
2011-05-02 17:47:24 -07:00
|
|
|
assert (v.(0) == 1);
|
|
|
|
assert (v.(7) == 8);
|
|
|
|
assert (v.(9) == 0);
|
2010-06-23 21:03:09 -07:00
|
|
|
}
|
2010-08-31 13:01:51 -07:00
|
|
|
|
|
|
|
fn slow_growth() {
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[int] v = [];
|
2010-08-31 13:01:51 -07:00
|
|
|
let vec[int] u = v;
|
2011-05-16 18:21:22 -07:00
|
|
|
v += [17];
|
2010-08-31 13:01:51 -07:00
|
|
|
|
|
|
|
log v.(0);
|
2011-05-02 17:47:24 -07:00
|
|
|
assert (v.(0) == 17);
|
2010-08-31 13:01:51 -07:00
|
|
|
}
|
|
|
|
|
2010-09-03 16:25:56 -07:00
|
|
|
fn slow_growth2_helper(str s) { // ref up: s
|
|
|
|
|
|
|
|
obj acc(vec[str] v) {
|
2011-05-16 18:21:22 -07:00
|
|
|
fn add(&str s) { v += [s]; }
|
2010-09-03 16:25:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
let str ss = s; // ref up: s
|
|
|
|
let str mumble = "mrghrm"; // ref up: mumble
|
|
|
|
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Within this block, mumble goes into a vec that is referenced
|
|
|
|
* both by the local slot v and the acc's v data field. When we
|
|
|
|
* add(s) on the acc, its v undergoes a slow append (allocate a
|
|
|
|
* new vec, copy over existing elements). Here we're testing to
|
|
|
|
* see that this slow path goes over well. In particular, the
|
|
|
|
* copy of existing elements should increment the ref count of
|
|
|
|
* mumble, the existing str in the originally- shared vec.
|
|
|
|
*/
|
2011-05-16 18:21:22 -07:00
|
|
|
let vec[str] v = [mumble]; // ref up: v, mumble
|
2010-09-03 16:25:56 -07:00
|
|
|
let acc a = acc(v); // ref up: a, v
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
log _vec::refcount[str](v);
|
|
|
|
assert (_vec::refcount[str](v) == 2u);
|
2010-09-03 16:25:56 -07:00
|
|
|
|
|
|
|
a.add(s); // ref up: mumble, s. ref down: v
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
log _vec::refcount[str](v);
|
|
|
|
log _str::refcount(s);
|
|
|
|
log _str::refcount(mumble);
|
2010-09-03 16:25:56 -07:00
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
assert (_vec::refcount[str](v) == 1u);
|
|
|
|
assert (_str::refcount(s) == const_refcount);
|
|
|
|
assert (_str::refcount(mumble) == const_refcount);
|
2010-09-03 16:25:56 -07:00
|
|
|
|
|
|
|
log v.(0);
|
2011-05-12 17:24:54 +02:00
|
|
|
log _vec::len[str](v);
|
|
|
|
assert (_str::eq(v.(0), mumble));
|
|
|
|
assert (_vec::len[str](v) == 1u);
|
2010-09-03 16:25:56 -07:00
|
|
|
} // ref down: a, mumble, s, v
|
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
log _str::refcount(s);
|
|
|
|
log _str::refcount(mumble);
|
2010-09-03 16:25:56 -07:00
|
|
|
|
2011-05-12 17:24:54 +02:00
|
|
|
assert (_str::refcount(s) == const_refcount);
|
|
|
|
assert (_str::refcount(mumble) == const_refcount);
|
2010-09-03 16:25:56 -07:00
|
|
|
|
|
|
|
log mumble;
|
|
|
|
log ss;
|
|
|
|
} // ref down
|
|
|
|
|
|
|
|
fn slow_growth2() {
|
|
|
|
let str s = "hi"; // ref up: s
|
|
|
|
slow_growth2_helper(s);
|
2011-05-12 17:24:54 +02:00
|
|
|
log _str::refcount(s);
|
|
|
|
assert (_str::refcount(s) == const_refcount);
|
2010-09-03 16:25:56 -07:00
|
|
|
}
|
|
|
|
|
2010-08-31 13:01:51 -07:00
|
|
|
fn main() {
|
|
|
|
fast_growth();
|
|
|
|
slow_growth();
|
2010-09-03 16:25:56 -07:00
|
|
|
slow_growth2();
|
2010-08-31 13:01:51 -07:00
|
|
|
}
|