mirror of
https://github.com/rust-lang/rust.git
synced 2025-01-24 13:43:04 +00:00
libsyntax: Allow +
to separate trait bounds from objects.
RFC #27. After a snapshot, the old syntax will be removed. This can break some code that looked like `foo as &Trait:Send`. Now you will need to write `foo as (&Trait+Send)`. Closes #12778. [breaking-change]
This commit is contained in:
parent
03ec8e5cc9
commit
9b9ef44233
@ -724,6 +724,7 @@ pub fn ast_ty_to_ty<AC:AstConv, RS:RegionScope>(
|
|||||||
.collect();
|
.collect();
|
||||||
ty::mk_tup(tcx, flds)
|
ty::mk_tup(tcx, flds)
|
||||||
}
|
}
|
||||||
|
ast::TyParen(ref typ) => ast_ty_to_ty(this, rscope, &**typ),
|
||||||
ast::TyBareFn(ref bf) => {
|
ast::TyBareFn(ref bf) => {
|
||||||
if bf.decl.variadic && bf.abi != abi::C {
|
if bf.decl.variadic && bf.abi != abi::C {
|
||||||
tcx.sess.span_err(ast_ty.span,
|
tcx.sess.span_err(ast_ty.span,
|
||||||
|
@ -1140,6 +1140,7 @@ impl<'a> Rebuilder<'a> {
|
|||||||
}
|
}
|
||||||
ast::TyTup(new_tys)
|
ast::TyTup(new_tys)
|
||||||
}
|
}
|
||||||
|
ast::TyParen(ref typ) => ast::TyParen(build_to(*typ, to)),
|
||||||
ref other => other.clone()
|
ref other => other.clone()
|
||||||
};
|
};
|
||||||
box(GC) ast::Ty { id: from.id, node: new_node, span: from.span }
|
box(GC) ast::Ty { id: from.id, node: new_node, span: from.span }
|
||||||
|
@ -60,7 +60,7 @@ impl<T: Clean<U>, U> Clean<VecPerParamSpace<U>> for VecPerParamSpace<T> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: Clean<U>, U> Clean<U> for Gc<T> {
|
impl<T: 'static + Clean<U>, U> Clean<U> for Gc<T> {
|
||||||
fn clean(&self) -> U {
|
fn clean(&self) -> U {
|
||||||
(**self).clean()
|
(**self).clean()
|
||||||
}
|
}
|
||||||
@ -1198,6 +1198,7 @@ impl Clean<Type> for ast::Ty {
|
|||||||
TyClosure(ref c, region) => Closure(box c.clean(), region.clean()),
|
TyClosure(ref c, region) => Closure(box c.clean(), region.clean()),
|
||||||
TyProc(ref c) => Proc(box c.clean()),
|
TyProc(ref c) => Proc(box c.clean()),
|
||||||
TyBareFn(ref barefn) => BareFunction(box barefn.clean()),
|
TyBareFn(ref barefn) => BareFunction(box barefn.clean()),
|
||||||
|
TyParen(ref ty) => ty.clean(),
|
||||||
TyBot => Bottom,
|
TyBot => Bottom,
|
||||||
ref x => fail!("Unimplemented type {:?}", x),
|
ref x => fail!("Unimplemented type {:?}", x),
|
||||||
}
|
}
|
||||||
|
@ -784,6 +784,8 @@ pub enum Ty_ {
|
|||||||
TyUnboxedFn(Gc<UnboxedFnTy>),
|
TyUnboxedFn(Gc<UnboxedFnTy>),
|
||||||
TyTup(Vec<P<Ty>> ),
|
TyTup(Vec<P<Ty>> ),
|
||||||
TyPath(Path, Option<OwnedSlice<TyParamBound>>, NodeId), // for #7264; see above
|
TyPath(Path, Option<OwnedSlice<TyParamBound>>, NodeId), // for #7264; see above
|
||||||
|
// No-op; kept solely so that we can pretty-print faithfully
|
||||||
|
TyParen(P<Ty>),
|
||||||
TyTypeof(Gc<Expr>),
|
TyTypeof(Gc<Expr>),
|
||||||
// TyInfer means the type should be inferred instead of it having been
|
// TyInfer means the type should be inferred instead of it having been
|
||||||
// specified. This can appear anywhere in a type.
|
// specified. This can appear anywhere in a type.
|
||||||
|
@ -737,6 +737,7 @@ pub fn get_inner_tys(ty: P<Ty>) -> Vec<P<Ty>> {
|
|||||||
| ast::TyUniq(ty)
|
| ast::TyUniq(ty)
|
||||||
| ast::TyFixedLengthVec(ty, _) => vec!(ty),
|
| ast::TyFixedLengthVec(ty, _) => vec!(ty),
|
||||||
ast::TyTup(ref tys) => tys.clone(),
|
ast::TyTup(ref tys) => tys.clone(),
|
||||||
|
ast::TyParen(ty) => get_inner_tys(ty),
|
||||||
_ => Vec::new()
|
_ => Vec::new()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -192,6 +192,7 @@ pub trait Folder {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
TyTup(ref tys) => TyTup(tys.iter().map(|&ty| self.fold_ty(ty)).collect()),
|
TyTup(ref tys) => TyTup(tys.iter().map(|&ty| self.fold_ty(ty)).collect()),
|
||||||
|
TyParen(ref ty) => TyParen(self.fold_ty(*ty)),
|
||||||
TyPath(ref path, ref bounds, id) => {
|
TyPath(ref path, ref bounds, id) => {
|
||||||
let id = self.new_id(id);
|
let id = self.new_id(id);
|
||||||
TyPath(self.fold_path(path),
|
TyPath(self.fold_path(path),
|
||||||
|
@ -51,7 +51,7 @@ use ast::{TokenTree, TraitMethod, TraitRef, TTDelim, TTSeq, TTTok};
|
|||||||
use ast::{TTNonterminal, TupleVariantKind, Ty, Ty_, TyBot, TyBox};
|
use ast::{TTNonterminal, TupleVariantKind, Ty, Ty_, TyBot, TyBox};
|
||||||
use ast::{TypeField, TyFixedLengthVec, TyClosure, TyProc, TyBareFn};
|
use ast::{TypeField, TyFixedLengthVec, TyClosure, TyProc, TyBareFn};
|
||||||
use ast::{TyTypeof, TyInfer, TypeMethod};
|
use ast::{TyTypeof, TyInfer, TypeMethod};
|
||||||
use ast::{TyNil, TyParam, TyParamBound, TyPath, TyPtr, TyRptr};
|
use ast::{TyNil, TyParam, TyParamBound, TyParen, TyPath, TyPtr, TyRptr};
|
||||||
use ast::{TyTup, TyU32, TyUnboxedFn, TyUniq, TyVec, UnUniq};
|
use ast::{TyTup, TyU32, TyUnboxedFn, TyUniq, TyVec, UnUniq};
|
||||||
use ast::{UnboxedFnTy, UnboxedFnTyParamBound, UnnamedField, UnsafeBlock};
|
use ast::{UnboxedFnTy, UnboxedFnTyParamBound, UnnamedField, UnsafeBlock};
|
||||||
use ast::{UnsafeFn, ViewItem, ViewItem_, ViewItemExternCrate, ViewItemUse};
|
use ast::{UnsafeFn, ViewItem, ViewItem_, ViewItemExternCrate, ViewItemUse};
|
||||||
@ -105,7 +105,7 @@ pub enum PathParsingMode {
|
|||||||
/// the type parameters; e.g. `foo::bar::<'a>::Baz::<T>`
|
/// the type parameters; e.g. `foo::bar::<'a>::Baz::<T>`
|
||||||
LifetimeAndTypesWithColons,
|
LifetimeAndTypesWithColons,
|
||||||
/// A path with a lifetime and type parameters with bounds before the last
|
/// A path with a lifetime and type parameters with bounds before the last
|
||||||
/// set of type parameters only; e.g. `foo::bar<'a>::Baz:X+Y<T>` This
|
/// set of type parameters only; e.g. `foo::bar<'a>::Baz+X+Y<T>` This
|
||||||
/// form does not use extra double colons.
|
/// form does not use extra double colons.
|
||||||
LifetimeAndTypesAndBounds,
|
LifetimeAndTypesAndBounds,
|
||||||
}
|
}
|
||||||
@ -1015,7 +1015,14 @@ impl<'a> Parser<'a> {
|
|||||||
};
|
};
|
||||||
|
|
||||||
let (inputs, variadic) = self.parse_fn_args(false, false);
|
let (inputs, variadic) = self.parse_fn_args(false, false);
|
||||||
let (_, bounds) = self.parse_optional_ty_param_bounds(false);
|
let bounds = {
|
||||||
|
if self.eat(&token::COLON) {
|
||||||
|
let (_, bounds) = self.parse_ty_param_bounds(false);
|
||||||
|
Some(bounds)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
};
|
||||||
let (ret_style, ret_ty) = self.parse_ret_ty();
|
let (ret_style, ret_ty) = self.parse_ret_ty();
|
||||||
let decl = P(FnDecl {
|
let decl = P(FnDecl {
|
||||||
inputs: inputs,
|
inputs: inputs,
|
||||||
@ -1083,7 +1090,14 @@ impl<'a> Parser<'a> {
|
|||||||
(is_unboxed, inputs)
|
(is_unboxed, inputs)
|
||||||
};
|
};
|
||||||
|
|
||||||
let (region, bounds) = self.parse_optional_ty_param_bounds(true);
|
let (region, bounds) = {
|
||||||
|
if self.eat(&token::COLON) {
|
||||||
|
let (region, bounds) = self.parse_ty_param_bounds(true);
|
||||||
|
(region, Some(bounds))
|
||||||
|
} else {
|
||||||
|
(None, None)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
let (return_style, output) = self.parse_ret_ty();
|
let (return_style, output) = self.parse_ret_ty();
|
||||||
let decl = P(FnDecl {
|
let decl = P(FnDecl {
|
||||||
@ -1227,7 +1241,7 @@ impl<'a> Parser<'a> {
|
|||||||
// parse a possibly mutable type
|
// parse a possibly mutable type
|
||||||
pub fn parse_mt(&mut self) -> MutTy {
|
pub fn parse_mt(&mut self) -> MutTy {
|
||||||
let mutbl = self.parse_mutability();
|
let mutbl = self.parse_mutability();
|
||||||
let t = self.parse_ty(false);
|
let t = self.parse_ty(true);
|
||||||
MutTy { ty: t, mutbl: mutbl }
|
MutTy { ty: t, mutbl: mutbl }
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1238,7 +1252,7 @@ impl<'a> Parser<'a> {
|
|||||||
let mutbl = self.parse_mutability();
|
let mutbl = self.parse_mutability();
|
||||||
let id = self.parse_ident();
|
let id = self.parse_ident();
|
||||||
self.expect(&token::COLON);
|
self.expect(&token::COLON);
|
||||||
let ty = self.parse_ty(false);
|
let ty = self.parse_ty(true);
|
||||||
let hi = ty.span.hi;
|
let hi = ty.span.hi;
|
||||||
ast::TypeField {
|
ast::TypeField {
|
||||||
ident: id,
|
ident: id,
|
||||||
@ -1261,7 +1275,7 @@ impl<'a> Parser<'a> {
|
|||||||
})
|
})
|
||||||
)
|
)
|
||||||
} else {
|
} else {
|
||||||
(Return, self.parse_ty(false))
|
(Return, self.parse_ty(true))
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
let pos = self.span.lo;
|
let pos = self.span.lo;
|
||||||
@ -1276,10 +1290,11 @@ impl<'a> Parser<'a> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// parse a type.
|
/// Parse a type.
|
||||||
// Useless second parameter for compatibility with quasiquote macros.
|
///
|
||||||
// Bleh!
|
/// The second parameter specifies whether the `+` binary operator is
|
||||||
pub fn parse_ty(&mut self, _: bool) -> P<Ty> {
|
/// allowed in the type grammar.
|
||||||
|
pub fn parse_ty(&mut self, plus_allowed: bool) -> P<Ty> {
|
||||||
maybe_whole!(no_clone self, NtTy);
|
maybe_whole!(no_clone self, NtTy);
|
||||||
|
|
||||||
let lo = self.span.lo;
|
let lo = self.span.lo;
|
||||||
@ -1293,12 +1308,12 @@ impl<'a> Parser<'a> {
|
|||||||
// (t) is a parenthesized ty
|
// (t) is a parenthesized ty
|
||||||
// (t,) is the type of a tuple with only one field,
|
// (t,) is the type of a tuple with only one field,
|
||||||
// of type t
|
// of type t
|
||||||
let mut ts = vec!(self.parse_ty(false));
|
let mut ts = vec!(self.parse_ty(true));
|
||||||
let mut one_tuple = false;
|
let mut one_tuple = false;
|
||||||
while self.token == token::COMMA {
|
while self.token == token::COMMA {
|
||||||
self.bump();
|
self.bump();
|
||||||
if self.token != token::RPAREN {
|
if self.token != token::RPAREN {
|
||||||
ts.push(self.parse_ty(false));
|
ts.push(self.parse_ty(true));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
one_tuple = true;
|
one_tuple = true;
|
||||||
@ -1307,17 +1322,17 @@ impl<'a> Parser<'a> {
|
|||||||
|
|
||||||
if ts.len() == 1 && !one_tuple {
|
if ts.len() == 1 && !one_tuple {
|
||||||
self.expect(&token::RPAREN);
|
self.expect(&token::RPAREN);
|
||||||
return *ts.get(0)
|
TyParen(*ts.get(0))
|
||||||
}
|
} else {
|
||||||
|
|
||||||
let t = TyTup(ts);
|
let t = TyTup(ts);
|
||||||
self.expect(&token::RPAREN);
|
self.expect(&token::RPAREN);
|
||||||
t
|
t
|
||||||
}
|
}
|
||||||
|
}
|
||||||
} else if self.token == token::AT {
|
} else if self.token == token::AT {
|
||||||
// MANAGED POINTER
|
// MANAGED POINTER
|
||||||
self.bump();
|
self.bump();
|
||||||
TyBox(self.parse_ty(false))
|
TyBox(self.parse_ty(plus_allowed))
|
||||||
} else if self.token == token::TILDE {
|
} else if self.token == token::TILDE {
|
||||||
// OWNED POINTER
|
// OWNED POINTER
|
||||||
self.bump();
|
self.bump();
|
||||||
@ -1326,7 +1341,7 @@ impl<'a> Parser<'a> {
|
|||||||
self.obsolete(self.last_span, ObsoleteOwnedVector),
|
self.obsolete(self.last_span, ObsoleteOwnedVector),
|
||||||
_ => self.obsolete(self.last_span, ObsoleteOwnedType),
|
_ => self.obsolete(self.last_span, ObsoleteOwnedType),
|
||||||
};
|
};
|
||||||
TyUniq(self.parse_ty(false))
|
TyUniq(self.parse_ty(true))
|
||||||
} else if self.token == token::BINOP(token::STAR) {
|
} else if self.token == token::BINOP(token::STAR) {
|
||||||
// STAR POINTER (bare pointer?)
|
// STAR POINTER (bare pointer?)
|
||||||
self.bump();
|
self.bump();
|
||||||
@ -1334,7 +1349,7 @@ impl<'a> Parser<'a> {
|
|||||||
} else if self.token == token::LBRACKET {
|
} else if self.token == token::LBRACKET {
|
||||||
// VECTOR
|
// VECTOR
|
||||||
self.expect(&token::LBRACKET);
|
self.expect(&token::LBRACKET);
|
||||||
let t = self.parse_ty(false);
|
let t = self.parse_ty(true);
|
||||||
|
|
||||||
// Parse the `, ..e` in `[ int, ..e ]`
|
// Parse the `, ..e` in `[ int, ..e ]`
|
||||||
// where `e` is a const expression
|
// where `e` is a const expression
|
||||||
@ -1377,10 +1392,15 @@ impl<'a> Parser<'a> {
|
|||||||
} else if self.token == token::MOD_SEP
|
} else if self.token == token::MOD_SEP
|
||||||
|| is_ident_or_path(&self.token) {
|
|| is_ident_or_path(&self.token) {
|
||||||
// NAMED TYPE
|
// NAMED TYPE
|
||||||
|
let mode = if plus_allowed {
|
||||||
|
LifetimeAndTypesAndBounds
|
||||||
|
} else {
|
||||||
|
LifetimeAndTypesWithoutColons
|
||||||
|
};
|
||||||
let PathAndBounds {
|
let PathAndBounds {
|
||||||
path,
|
path,
|
||||||
bounds
|
bounds
|
||||||
} = self.parse_path(LifetimeAndTypesAndBounds);
|
} = self.parse_path(mode);
|
||||||
TyPath(path, bounds, ast::DUMMY_NODE_ID)
|
TyPath(path, bounds, ast::DUMMY_NODE_ID)
|
||||||
} else if self.eat(&token::UNDERSCORE) {
|
} else if self.eat(&token::UNDERSCORE) {
|
||||||
// TYPE TO BE INFERRED
|
// TYPE TO BE INFERRED
|
||||||
@ -1438,7 +1458,7 @@ impl<'a> Parser<'a> {
|
|||||||
special_idents::invalid)
|
special_idents::invalid)
|
||||||
};
|
};
|
||||||
|
|
||||||
let t = self.parse_ty(false);
|
let t = self.parse_ty(true);
|
||||||
|
|
||||||
Arg {
|
Arg {
|
||||||
ty: t,
|
ty: t,
|
||||||
@ -1456,7 +1476,7 @@ impl<'a> Parser<'a> {
|
|||||||
pub fn parse_fn_block_arg(&mut self) -> Arg {
|
pub fn parse_fn_block_arg(&mut self) -> Arg {
|
||||||
let pat = self.parse_pat();
|
let pat = self.parse_pat();
|
||||||
let t = if self.eat(&token::COLON) {
|
let t = if self.eat(&token::COLON) {
|
||||||
self.parse_ty(false)
|
self.parse_ty(true)
|
||||||
} else {
|
} else {
|
||||||
P(Ty {
|
P(Ty {
|
||||||
id: ast::DUMMY_NODE_ID,
|
id: ast::DUMMY_NODE_ID,
|
||||||
@ -1611,9 +1631,19 @@ impl<'a> Parser<'a> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Next, parse a colon and bounded type parameters, if applicable.
|
// Next, parse a plus and bounded type parameters, if applicable.
|
||||||
|
//
|
||||||
|
// NOTE(stage0, pcwalton): Remove `token::COLON` after a snapshot.
|
||||||
let bounds = if mode == LifetimeAndTypesAndBounds {
|
let bounds = if mode == LifetimeAndTypesAndBounds {
|
||||||
let (_, bounds) = self.parse_optional_ty_param_bounds(false);
|
let bounds = {
|
||||||
|
if self.eat(&token::BINOP(token::PLUS)) ||
|
||||||
|
self.eat(&token::COLON) {
|
||||||
|
let (_, bounds) = self.parse_ty_param_bounds(false);
|
||||||
|
Some(bounds)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
};
|
||||||
bounds
|
bounds
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
@ -2438,7 +2468,7 @@ impl<'a> Parser<'a> {
|
|||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
if as_prec > min_prec && self.eat_keyword(keywords::As) {
|
if as_prec > min_prec && self.eat_keyword(keywords::As) {
|
||||||
let rhs = self.parse_ty(true);
|
let rhs = self.parse_ty(false);
|
||||||
let _as = self.mk_expr(lhs.span.lo,
|
let _as = self.mk_expr(lhs.span.lo,
|
||||||
rhs.span.hi,
|
rhs.span.hi,
|
||||||
ExprCast(lhs, rhs));
|
ExprCast(lhs, rhs));
|
||||||
@ -3067,7 +3097,9 @@ impl<'a> Parser<'a> {
|
|||||||
node: TyInfer,
|
node: TyInfer,
|
||||||
span: mk_sp(lo, lo),
|
span: mk_sp(lo, lo),
|
||||||
});
|
});
|
||||||
if self.eat(&token::COLON) { ty = self.parse_ty(false); }
|
if self.eat(&token::COLON) {
|
||||||
|
ty = self.parse_ty(true);
|
||||||
|
}
|
||||||
let init = self.parse_initializer();
|
let init = self.parse_initializer();
|
||||||
box(GC) ast::Local {
|
box(GC) ast::Local {
|
||||||
ty: ty,
|
ty: ty,
|
||||||
@ -3095,7 +3127,7 @@ impl<'a> Parser<'a> {
|
|||||||
}
|
}
|
||||||
let name = self.parse_ident();
|
let name = self.parse_ident();
|
||||||
self.expect(&token::COLON);
|
self.expect(&token::COLON);
|
||||||
let ty = self.parse_ty(false);
|
let ty = self.parse_ty(true);
|
||||||
spanned(lo, self.last_span.hi, ast::StructField_ {
|
spanned(lo, self.last_span.hi, ast::StructField_ {
|
||||||
kind: NamedField(name, pr),
|
kind: NamedField(name, pr),
|
||||||
id: ast::DUMMY_NODE_ID,
|
id: ast::DUMMY_NODE_ID,
|
||||||
@ -3427,7 +3459,7 @@ impl<'a> Parser<'a> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// matches optbounds = ( ( : ( boundseq )? )? )
|
// matches bounds = ( boundseq )?
|
||||||
// where boundseq = ( bound + boundseq ) | bound
|
// where boundseq = ( bound + boundseq ) | bound
|
||||||
// and bound = 'static | ty
|
// and bound = 'static | ty
|
||||||
// Returns "None" if there's no colon (e.g. "T");
|
// Returns "None" if there's no colon (e.g. "T");
|
||||||
@ -3439,13 +3471,9 @@ impl<'a> Parser<'a> {
|
|||||||
// AST doesn't support arbitrary lifetimes in bounds on type parameters. In
|
// AST doesn't support arbitrary lifetimes in bounds on type parameters. In
|
||||||
// the future, this flag should be removed, and the return value of this
|
// the future, this flag should be removed, and the return value of this
|
||||||
// function should be Option<~[TyParamBound]>
|
// function should be Option<~[TyParamBound]>
|
||||||
fn parse_optional_ty_param_bounds(&mut self, allow_any_lifetime: bool)
|
fn parse_ty_param_bounds(&mut self, allow_any_lifetime: bool)
|
||||||
-> (Option<ast::Lifetime>, Option<OwnedSlice<TyParamBound>>)
|
-> (Option<ast::Lifetime>,
|
||||||
{
|
OwnedSlice<TyParamBound>) {
|
||||||
if !self.eat(&token::COLON) {
|
|
||||||
return (None, None);
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut ret_lifetime = None;
|
let mut ret_lifetime = None;
|
||||||
let mut result = vec!();
|
let mut result = vec!();
|
||||||
loop {
|
loop {
|
||||||
@ -3489,7 +3517,7 @@ impl<'a> Parser<'a> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (ret_lifetime, Some(OwnedSlice::from_vec(result)));
|
return (ret_lifetime, OwnedSlice::from_vec(result));
|
||||||
}
|
}
|
||||||
|
|
||||||
// matches typaram = type? IDENT optbounds ( EQ ty )?
|
// matches typaram = type? IDENT optbounds ( EQ ty )?
|
||||||
@ -3497,13 +3525,20 @@ impl<'a> Parser<'a> {
|
|||||||
let sized = self.parse_sized();
|
let sized = self.parse_sized();
|
||||||
let span = self.span;
|
let span = self.span;
|
||||||
let ident = self.parse_ident();
|
let ident = self.parse_ident();
|
||||||
let (_, opt_bounds) = self.parse_optional_ty_param_bounds(false);
|
let opt_bounds = {
|
||||||
|
if self.eat(&token::COLON) {
|
||||||
|
let (_, bounds) = self.parse_ty_param_bounds(false);
|
||||||
|
Some(bounds)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
};
|
||||||
// For typarams we don't care about the difference b/w "<T>" and "<T:>".
|
// For typarams we don't care about the difference b/w "<T>" and "<T:>".
|
||||||
let bounds = opt_bounds.unwrap_or_default();
|
let bounds = opt_bounds.unwrap_or_default();
|
||||||
|
|
||||||
let default = if self.token == token::EQ {
|
let default = if self.token == token::EQ {
|
||||||
self.bump();
|
self.bump();
|
||||||
Some(self.parse_ty(false))
|
Some(self.parse_ty(true))
|
||||||
}
|
}
|
||||||
else { None };
|
else { None };
|
||||||
|
|
||||||
@ -3548,7 +3583,7 @@ impl<'a> Parser<'a> {
|
|||||||
Some(token::COMMA),
|
Some(token::COMMA),
|
||||||
|p| {
|
|p| {
|
||||||
p.forbid_lifetime();
|
p.forbid_lifetime();
|
||||||
p.parse_ty(false)
|
p.parse_ty(true)
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
(lifetimes, result.into_vec())
|
(lifetimes, result.into_vec())
|
||||||
@ -3804,7 +3839,7 @@ impl<'a> Parser<'a> {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
let output = if self.eat(&token::RARROW) {
|
let output = if self.eat(&token::RARROW) {
|
||||||
self.parse_ty(false)
|
self.parse_ty(true)
|
||||||
} else {
|
} else {
|
||||||
P(Ty {
|
P(Ty {
|
||||||
id: ast::DUMMY_NODE_ID,
|
id: ast::DUMMY_NODE_ID,
|
||||||
@ -3830,7 +3865,7 @@ impl<'a> Parser<'a> {
|
|||||||
|p| p.parse_fn_block_arg());
|
|p| p.parse_fn_block_arg());
|
||||||
|
|
||||||
let output = if self.eat(&token::RARROW) {
|
let output = if self.eat(&token::RARROW) {
|
||||||
self.parse_ty(false)
|
self.parse_ty(true)
|
||||||
} else {
|
} else {
|
||||||
P(Ty {
|
P(Ty {
|
||||||
id: ast::DUMMY_NODE_ID,
|
id: ast::DUMMY_NODE_ID,
|
||||||
@ -3942,7 +3977,7 @@ impl<'a> Parser<'a> {
|
|||||||
let could_be_trait = self.token != token::LPAREN;
|
let could_be_trait = self.token != token::LPAREN;
|
||||||
|
|
||||||
// Parse the trait.
|
// Parse the trait.
|
||||||
let mut ty = self.parse_ty(false);
|
let mut ty = self.parse_ty(true);
|
||||||
|
|
||||||
// Parse traits, if necessary.
|
// Parse traits, if necessary.
|
||||||
let opt_trait = if could_be_trait && self.eat_keyword(keywords::For) {
|
let opt_trait = if could_be_trait && self.eat_keyword(keywords::For) {
|
||||||
@ -3965,7 +4000,7 @@ impl<'a> Parser<'a> {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
ty = self.parse_ty(false);
|
ty = self.parse_ty(true);
|
||||||
opt_trait_ref
|
opt_trait_ref
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
@ -4008,7 +4043,7 @@ impl<'a> Parser<'a> {
|
|||||||
let generics = self.parse_generics();
|
let generics = self.parse_generics();
|
||||||
|
|
||||||
let super_struct = if self.eat(&token::COLON) {
|
let super_struct = if self.eat(&token::COLON) {
|
||||||
let ty = self.parse_ty(false);
|
let ty = self.parse_ty(true);
|
||||||
match ty.node {
|
match ty.node {
|
||||||
TyPath(_, None, _) => {
|
TyPath(_, None, _) => {
|
||||||
Some(ty)
|
Some(ty)
|
||||||
@ -4051,7 +4086,7 @@ impl<'a> Parser<'a> {
|
|||||||
let struct_field_ = ast::StructField_ {
|
let struct_field_ = ast::StructField_ {
|
||||||
kind: UnnamedField(p.parse_visibility()),
|
kind: UnnamedField(p.parse_visibility()),
|
||||||
id: ast::DUMMY_NODE_ID,
|
id: ast::DUMMY_NODE_ID,
|
||||||
ty: p.parse_ty(false),
|
ty: p.parse_ty(true),
|
||||||
attrs: attrs,
|
attrs: attrs,
|
||||||
};
|
};
|
||||||
spanned(lo, p.span.hi, struct_field_)
|
spanned(lo, p.span.hi, struct_field_)
|
||||||
@ -4205,7 +4240,7 @@ impl<'a> Parser<'a> {
|
|||||||
let m = if self.eat_keyword(keywords::Mut) {MutMutable} else {MutImmutable};
|
let m = if self.eat_keyword(keywords::Mut) {MutMutable} else {MutImmutable};
|
||||||
let id = self.parse_ident();
|
let id = self.parse_ident();
|
||||||
self.expect(&token::COLON);
|
self.expect(&token::COLON);
|
||||||
let ty = self.parse_ty(false);
|
let ty = self.parse_ty(true);
|
||||||
self.expect(&token::EQ);
|
self.expect(&token::EQ);
|
||||||
let e = self.parse_expr();
|
let e = self.parse_expr();
|
||||||
self.commit_expr_expecting(e, token::SEMI);
|
self.commit_expr_expecting(e, token::SEMI);
|
||||||
@ -4386,7 +4421,7 @@ impl<'a> Parser<'a> {
|
|||||||
|
|
||||||
let ident = self.parse_ident();
|
let ident = self.parse_ident();
|
||||||
self.expect(&token::COLON);
|
self.expect(&token::COLON);
|
||||||
let ty = self.parse_ty(false);
|
let ty = self.parse_ty(true);
|
||||||
let hi = self.span.hi;
|
let hi = self.span.hi;
|
||||||
self.expect(&token::SEMI);
|
self.expect(&token::SEMI);
|
||||||
box(GC) ast::ForeignItem {
|
box(GC) ast::ForeignItem {
|
||||||
@ -4514,7 +4549,7 @@ impl<'a> Parser<'a> {
|
|||||||
let ident = self.parse_ident();
|
let ident = self.parse_ident();
|
||||||
let tps = self.parse_generics();
|
let tps = self.parse_generics();
|
||||||
self.expect(&token::EQ);
|
self.expect(&token::EQ);
|
||||||
let ty = self.parse_ty(false);
|
let ty = self.parse_ty(true);
|
||||||
self.expect(&token::SEMI);
|
self.expect(&token::SEMI);
|
||||||
(ident, ItemTy(ty, tps), None)
|
(ident, ItemTy(ty, tps), None)
|
||||||
}
|
}
|
||||||
@ -4562,7 +4597,7 @@ impl<'a> Parser<'a> {
|
|||||||
&token::LPAREN,
|
&token::LPAREN,
|
||||||
&token::RPAREN,
|
&token::RPAREN,
|
||||||
seq_sep_trailing_disallowed(token::COMMA),
|
seq_sep_trailing_disallowed(token::COMMA),
|
||||||
|p| p.parse_ty(false)
|
|p| p.parse_ty(true)
|
||||||
);
|
);
|
||||||
for ty in arg_tys.move_iter() {
|
for ty in arg_tys.move_iter() {
|
||||||
args.push(ast::VariantArg {
|
args.push(ast::VariantArg {
|
||||||
|
@ -495,6 +495,11 @@ impl<'a> State<'a> {
|
|||||||
}
|
}
|
||||||
try!(self.pclose());
|
try!(self.pclose());
|
||||||
}
|
}
|
||||||
|
ast::TyParen(ref typ) => {
|
||||||
|
try!(self.popen());
|
||||||
|
try!(self.print_type(&**typ));
|
||||||
|
try!(self.pclose());
|
||||||
|
}
|
||||||
ast::TyBareFn(f) => {
|
ast::TyBareFn(f) => {
|
||||||
let generics = ast::Generics {
|
let generics = ast::Generics {
|
||||||
lifetimes: f.lifetimes.clone(),
|
lifetimes: f.lifetimes.clone(),
|
||||||
@ -1673,7 +1678,7 @@ impl<'a> State<'a> {
|
|||||||
|
|
||||||
match *opt_bounds {
|
match *opt_bounds {
|
||||||
None => Ok(()),
|
None => Ok(()),
|
||||||
Some(ref bounds) => self.print_bounds(&None, bounds, true),
|
Some(ref bounds) => self.print_bounds(&None, bounds, true, true),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1932,9 +1937,16 @@ impl<'a> State<'a> {
|
|||||||
pub fn print_bounds(&mut self,
|
pub fn print_bounds(&mut self,
|
||||||
region: &Option<ast::Lifetime>,
|
region: &Option<ast::Lifetime>,
|
||||||
bounds: &OwnedSlice<ast::TyParamBound>,
|
bounds: &OwnedSlice<ast::TyParamBound>,
|
||||||
print_colon_anyway: bool) -> IoResult<()> {
|
print_colon_anyway: bool,
|
||||||
|
print_plus_before_bounds: bool)
|
||||||
|
-> IoResult<()> {
|
||||||
|
let separator = if print_plus_before_bounds {
|
||||||
|
"+"
|
||||||
|
} else {
|
||||||
|
":"
|
||||||
|
};
|
||||||
if !bounds.is_empty() || region.is_some() {
|
if !bounds.is_empty() || region.is_some() {
|
||||||
try!(word(&mut self.s, ":"));
|
try!(word(&mut self.s, separator));
|
||||||
let mut first = true;
|
let mut first = true;
|
||||||
match *region {
|
match *region {
|
||||||
Some(ref lt) => {
|
Some(ref lt) => {
|
||||||
@ -1976,7 +1988,7 @@ impl<'a> State<'a> {
|
|||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
} else if print_colon_anyway {
|
} else if print_colon_anyway {
|
||||||
word(&mut self.s, ":")
|
word(&mut self.s, separator)
|
||||||
} else {
|
} else {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@ -2011,7 +2023,10 @@ impl<'a> State<'a> {
|
|||||||
try!(s.word_space("type"));
|
try!(s.word_space("type"));
|
||||||
}
|
}
|
||||||
try!(s.print_ident(param.ident));
|
try!(s.print_ident(param.ident));
|
||||||
try!(s.print_bounds(&None, ¶m.bounds, false));
|
try!(s.print_bounds(&None,
|
||||||
|
¶m.bounds,
|
||||||
|
false,
|
||||||
|
false));
|
||||||
match param.default {
|
match param.default {
|
||||||
Some(ref default) => {
|
Some(ref default) => {
|
||||||
try!(space(&mut s.s));
|
try!(space(&mut s.s));
|
||||||
@ -2214,7 +2229,7 @@ impl<'a> State<'a> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
opt_bounds.as_ref().map(|bounds| {
|
opt_bounds.as_ref().map(|bounds| {
|
||||||
self.print_bounds(opt_region, bounds, true)
|
self.print_bounds(opt_region, bounds, true, false)
|
||||||
});
|
});
|
||||||
|
|
||||||
try!(self.maybe_print_comment(decl.output.span.lo));
|
try!(self.maybe_print_comment(decl.output.span.lo));
|
||||||
|
@ -336,7 +336,7 @@ pub fn skip_ty<E, V: Visitor<E>>(_: &mut V, _: &Ty, _: E) {
|
|||||||
|
|
||||||
pub fn walk_ty<E: Clone, V: Visitor<E>>(visitor: &mut V, typ: &Ty, env: E) {
|
pub fn walk_ty<E: Clone, V: Visitor<E>>(visitor: &mut V, typ: &Ty, env: E) {
|
||||||
match typ.node {
|
match typ.node {
|
||||||
TyUniq(ty) | TyVec(ty) | TyBox(ty) => {
|
TyUniq(ty) | TyVec(ty) | TyBox(ty) | TyParen(ty) => {
|
||||||
visitor.visit_ty(&*ty, env)
|
visitor.visit_ty(&*ty, env)
|
||||||
}
|
}
|
||||||
TyPtr(ref mutable_type) => {
|
TyPtr(ref mutable_type) => {
|
||||||
|
@ -27,7 +27,7 @@ impl Drop for Foo {
|
|||||||
|
|
||||||
#[plugin_registrar]
|
#[plugin_registrar]
|
||||||
pub fn registrar(_: &mut Registry) {
|
pub fn registrar(_: &mut Registry) {
|
||||||
local_data_key!(foo: Box<Any:Send>);
|
local_data_key!(foo: Box<Any+Send>);
|
||||||
foo.replace(Some(box Foo { foo: 10 } as Box<Any:Send>));
|
foo.replace(Some(box Foo { foo: 10 } as Box<Any+Send>));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -31,7 +31,7 @@ struct A {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
let a = A {v: box B{v: None} as Box<Foo:Send>};
|
let a = A {v: box B{v: None} as Box<Foo+Send>};
|
||||||
//~^ ERROR cannot pack type `~B`, which does not fulfill `Send`
|
//~^ ERROR cannot pack type `~B`, which does not fulfill `Send`
|
||||||
let v = Rc::new(RefCell::new(a));
|
let v = Rc::new(RefCell::new(a));
|
||||||
let w = v.clone();
|
let w = v.clone();
|
||||||
|
@ -45,15 +45,15 @@ fn test<'a,T,U:Copy>(_: &'a int) {
|
|||||||
|
|
||||||
// borrowed object types are generally ok
|
// borrowed object types are generally ok
|
||||||
assert_copy::<&'a Dummy>();
|
assert_copy::<&'a Dummy>();
|
||||||
assert_copy::<&'a Dummy:Copy>();
|
assert_copy::<&'a Dummy+Copy>();
|
||||||
assert_copy::<&'static Dummy:Copy>();
|
assert_copy::<&'static Dummy+Copy>();
|
||||||
|
|
||||||
// owned object types are not ok
|
// owned object types are not ok
|
||||||
assert_copy::<Box<Dummy>>(); //~ ERROR does not fulfill
|
assert_copy::<Box<Dummy>>(); //~ ERROR does not fulfill
|
||||||
assert_copy::<Box<Dummy:Copy>>(); //~ ERROR does not fulfill
|
assert_copy::<Box<Dummy+Copy>>(); //~ ERROR does not fulfill
|
||||||
|
|
||||||
// mutable object types are not ok
|
// mutable object types are not ok
|
||||||
assert_copy::<&'a mut Dummy:Copy>(); //~ ERROR does not fulfill
|
assert_copy::<&'a mut Dummy+Copy>(); //~ ERROR does not fulfill
|
||||||
|
|
||||||
// closures are like an `&mut` object
|
// closures are like an `&mut` object
|
||||||
assert_copy::<||>(); //~ ERROR does not fulfill
|
assert_copy::<||>(); //~ ERROR does not fulfill
|
||||||
|
@ -39,17 +39,17 @@ fn test<'a,T,U:Send>(_: &'a int) {
|
|||||||
// careful with object types, who knows what they close over...
|
// careful with object types, who knows what they close over...
|
||||||
assert_send::<&'static Dummy>(); //~ ERROR does not fulfill `Send`
|
assert_send::<&'static Dummy>(); //~ ERROR does not fulfill `Send`
|
||||||
assert_send::<&'a Dummy>(); //~ ERROR does not fulfill `Send`
|
assert_send::<&'a Dummy>(); //~ ERROR does not fulfill `Send`
|
||||||
assert_send::<&'a Dummy:Send>(); //~ ERROR does not fulfill `Send`
|
assert_send::<&'a Dummy+Send>(); //~ ERROR does not fulfill `Send`
|
||||||
assert_send::<Box<Dummy:>>(); //~ ERROR does not fulfill `Send`
|
assert_send::<Box<Dummy+>>(); //~ ERROR does not fulfill `Send`
|
||||||
|
|
||||||
// ...unless they are properly bounded
|
// ...unless they are properly bounded
|
||||||
assert_send::<&'static Dummy:Send>();
|
assert_send::<&'static Dummy+Send>();
|
||||||
assert_send::<Box<Dummy:Send>>();
|
assert_send::<Box<Dummy+Send>>();
|
||||||
|
|
||||||
// but closure and object types can have lifetime bounds which make
|
// but closure and object types can have lifetime bounds which make
|
||||||
// them not ok (FIXME #5121)
|
// them not ok (FIXME #5121)
|
||||||
// assert_send::<proc:'a()>(); // ERROR does not fulfill `Send`
|
// assert_send::<proc:'a()>(); // ERROR does not fulfill `Send`
|
||||||
// assert_send::<Box<Dummy:'a>>(); // ERROR does not fulfill `Send`
|
// assert_send::<Box<Dummy+'a>>(); // ERROR does not fulfill `Send`
|
||||||
|
|
||||||
// unsafe ptrs are ok unless they point at unsendable things
|
// unsafe ptrs are ok unless they point at unsendable things
|
||||||
assert_send::<*int>();
|
assert_send::<*int>();
|
||||||
|
@ -12,14 +12,14 @@
|
|||||||
trait Foo {
|
trait Foo {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn a(_x: Box<Foo:Send>) {
|
fn a(_x: Box<Foo+Send>) {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn c(x: Box<Foo:Share+Send>) {
|
fn c(x: Box<Foo+Share+Send>) {
|
||||||
a(x);
|
a(x);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn d(x: Box<Foo:>) {
|
fn d(x: Box<Foo+>) {
|
||||||
a(x); //~ ERROR found no bounds
|
a(x); //~ ERROR found no bounds
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -13,17 +13,17 @@
|
|||||||
|
|
||||||
trait Foo {}
|
trait Foo {}
|
||||||
|
|
||||||
fn a(_x: Box<Foo:Send>) {
|
fn a(_x: Box<Foo+Send>) {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn b(_x: &'static Foo) { // should be same as &'static Foo:'static
|
fn b(_x: &'static Foo) { // should be same as &'static Foo+'static
|
||||||
}
|
}
|
||||||
|
|
||||||
fn c(x: Box<Foo:Share>) {
|
fn c(x: Box<Foo+Share>) {
|
||||||
a(x); //~ ERROR expected bounds `Send`
|
a(x); //~ ERROR expected bounds `Send`
|
||||||
}
|
}
|
||||||
|
|
||||||
fn d(x: &'static Foo:Share) {
|
fn d(x: &'static Foo+Share) {
|
||||||
b(x); //~ ERROR expected bounds `'static`
|
b(x); //~ ERROR expected bounds `'static`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -12,5 +12,5 @@
|
|||||||
|
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
fail!(box 413 as Box<::std::any::Any:Send>);
|
fail!(box 413 as Box<::std::any::Any+Send>);
|
||||||
}
|
}
|
||||||
|
@ -33,7 +33,7 @@ fn f<A:Clone + 'static>(a: A, b: u16) -> Box<Invokable<A>:> {
|
|||||||
box Invoker {
|
box Invoker {
|
||||||
a: a,
|
a: a,
|
||||||
b: b,
|
b: b,
|
||||||
} as Box<Invokable<A>>:
|
} as (Box<Invokable<A>>+)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn main() {
|
pub fn main() {
|
||||||
|
18
src/test/run-pass/as-precedence.rs
Normal file
18
src/test/run-pass/as-precedence.rs
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
// 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.
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
assert_eq!(3 as uint * 3, 9);
|
||||||
|
assert_eq!(3 as (uint) * 3, 9);
|
||||||
|
assert_eq!(3 as (uint) / 3, 1);
|
||||||
|
assert_eq!(3 as uint + 3, 6);
|
||||||
|
assert_eq!(3 as (uint) + 3, 6);
|
||||||
|
}
|
||||||
|
|
@ -41,7 +41,7 @@ fn main() {
|
|||||||
let (tx, rx) = channel();
|
let (tx, rx) = channel();
|
||||||
let (mut r, w) = (ChanReader::new(rx), ChanWriter::new(tx));
|
let (mut r, w) = (ChanReader::new(rx), ChanWriter::new(tx));
|
||||||
spawn(proc() {
|
spawn(proc() {
|
||||||
set_logger(box MyWriter(w) as Box<Logger:Send>);
|
set_logger(box MyWriter(w) as Box<Logger+Send>);
|
||||||
debug!("debug");
|
debug!("debug");
|
||||||
info!("info");
|
info!("info");
|
||||||
});
|
});
|
||||||
|
@ -33,11 +33,11 @@ impl<A:Clone> Invokable<A> for Invoker<A> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn f<A:Clone + 'static>(a: A, b: u16) -> Box<Invokable<A>:> {
|
fn f<A:Clone + 'static>(a: A, b: u16) -> Box<Invokable<A>+> {
|
||||||
box Invoker {
|
box Invoker {
|
||||||
a: a,
|
a: a,
|
||||||
b: b,
|
b: b,
|
||||||
} as Box<Invokable<A>>:
|
} as (Box<Invokable<A>>+)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn main() {
|
pub fn main() {
|
||||||
|
@ -12,21 +12,21 @@
|
|||||||
trait Foo {
|
trait Foo {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn a(_x: Box<Foo:>) {
|
fn a(_x: Box<Foo+>) {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn b(_x: Box<Foo:Send>) {
|
fn b(_x: Box<Foo+Send>) {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn c(x: Box<Foo:Share+Send>) {
|
fn c(x: Box<Foo+Share+Send>) {
|
||||||
a(x);
|
a(x);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn d(x: Box<Foo:Send>) {
|
fn d(x: Box<Foo+Send>) {
|
||||||
b(x);
|
b(x);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn e(x: Box<Foo>) { // sugar for Box<Foo:Owned>
|
fn e(x: Box<Foo>) { // sugar for Box<Foo+Owned>
|
||||||
a(x);
|
a(x);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,10 +71,10 @@ pub fn main() {
|
|||||||
swim_speed: 998,
|
swim_speed: 998,
|
||||||
name: "alec_guinness".to_string(),
|
name: "alec_guinness".to_string(),
|
||||||
};
|
};
|
||||||
let arc = Arc::new(vec!(box catte as Box<Pet:Share+Send>,
|
let arc = Arc::new(vec!(box catte as Box<Pet+Share+Send>,
|
||||||
box dogge1 as Box<Pet:Share+Send>,
|
box dogge1 as Box<Pet+Share+Send>,
|
||||||
box fishe as Box<Pet:Share+Send>,
|
box fishe as Box<Pet+Share+Send>,
|
||||||
box dogge2 as Box<Pet:Share+Send>));
|
box dogge2 as Box<Pet+Share+Send>));
|
||||||
let (tx1, rx1) = channel();
|
let (tx1, rx1) = channel();
|
||||||
let arc1 = arc.clone();
|
let arc1 = arc.clone();
|
||||||
task::spawn(proc() { check_legs(arc1); tx1.send(()); });
|
task::spawn(proc() { check_legs(arc1); tx1.send(()); });
|
||||||
@ -89,21 +89,21 @@ pub fn main() {
|
|||||||
rx3.recv();
|
rx3.recv();
|
||||||
}
|
}
|
||||||
|
|
||||||
fn check_legs(arc: Arc<Vec<Box<Pet:Share+Send>>>) {
|
fn check_legs(arc: Arc<Vec<Box<Pet+Share+Send>>>) {
|
||||||
let mut legs = 0;
|
let mut legs = 0;
|
||||||
for pet in arc.iter() {
|
for pet in arc.iter() {
|
||||||
legs += pet.num_legs();
|
legs += pet.num_legs();
|
||||||
}
|
}
|
||||||
assert!(legs == 12);
|
assert!(legs == 12);
|
||||||
}
|
}
|
||||||
fn check_names(arc: Arc<Vec<Box<Pet:Share+Send>>>) {
|
fn check_names(arc: Arc<Vec<Box<Pet+Share+Send>>>) {
|
||||||
for pet in arc.iter() {
|
for pet in arc.iter() {
|
||||||
pet.name(|name| {
|
pet.name(|name| {
|
||||||
assert!(name[0] == 'a' as u8 && name[1] == 'l' as u8);
|
assert!(name[0] == 'a' as u8 && name[1] == 'l' as u8);
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fn check_pedigree(arc: Arc<Vec<Box<Pet:Share+Send>>>) {
|
fn check_pedigree(arc: Arc<Vec<Box<Pet+Share+Send>>>) {
|
||||||
for pet in arc.iter() {
|
for pet in arc.iter() {
|
||||||
assert!(pet.of_good_pedigree());
|
assert!(pet.of_good_pedigree());
|
||||||
}
|
}
|
||||||
|
@ -20,7 +20,7 @@ struct Tree(@RefCell<TreeR>);
|
|||||||
struct TreeR {
|
struct TreeR {
|
||||||
left: Option<Tree>,
|
left: Option<Tree>,
|
||||||
right: Option<Tree>,
|
right: Option<Tree>,
|
||||||
val: Box<to_str:Send>
|
val: Box<to_str+Send>
|
||||||
}
|
}
|
||||||
|
|
||||||
trait to_str {
|
trait to_str {
|
||||||
@ -57,10 +57,10 @@ fn foo<T:to_str>(x: T) -> String { x.to_str_() }
|
|||||||
pub fn main() {
|
pub fn main() {
|
||||||
let t1 = Tree(@RefCell::new(TreeR{left: None,
|
let t1 = Tree(@RefCell::new(TreeR{left: None,
|
||||||
right: None,
|
right: None,
|
||||||
val: box 1 as Box<to_str:Send>}));
|
val: box 1 as Box<to_str+Send>}));
|
||||||
let t2 = Tree(@RefCell::new(TreeR{left: Some(t1),
|
let t2 = Tree(@RefCell::new(TreeR{left: Some(t1),
|
||||||
right: Some(t1),
|
right: Some(t1),
|
||||||
val: box 2 as Box<to_str:Send>}));
|
val: box 2 as Box<to_str+Send>}));
|
||||||
let expected =
|
let expected =
|
||||||
"[2, some([1, none, none]), some([1, none, none])]".to_string();
|
"[2, some([1, none, none]), some([1, none, none])]".to_string();
|
||||||
assert!(t2.to_str_() == expected);
|
assert!(t2.to_str_() == expected);
|
||||||
|
@ -10,8 +10,8 @@
|
|||||||
|
|
||||||
// This is an interesting test case. We have a trait (Bar) that is
|
// This is an interesting test case. We have a trait (Bar) that is
|
||||||
// implemented for a `Box<Foo>` object (note: no bounds). And then we
|
// implemented for a `Box<Foo>` object (note: no bounds). And then we
|
||||||
// have a `Box<Foo:Send>` object. The impl for `Box<Foo>` is applicable
|
// have a `Box<Foo+Send>` object. The impl for `Box<Foo>` is applicable
|
||||||
// to `Box<Foo:Send>` because:
|
// to `Box<Foo+Send>` because:
|
||||||
//
|
//
|
||||||
// 1. The trait Bar is contravariant w/r/t Self because `Self` appears
|
// 1. The trait Bar is contravariant w/r/t Self because `Self` appears
|
||||||
// only in argument position.
|
// only in argument position.
|
||||||
@ -30,7 +30,7 @@ impl Bar for Box<Foo> { fn dummy(&self) { } }
|
|||||||
fn wants_bar<B:Bar>(b: &B) { }
|
fn wants_bar<B:Bar>(b: &B) { }
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
let x: Box<Foo:Send> = (box SFoo);
|
let x: Box<Foo+Send> = (box SFoo);
|
||||||
wants_bar(&x);
|
wants_bar(&x);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user