Skip to content

Commit bcc5486

Browse files
committed
Allow overloading explicit dereferences.
1 parent 52532d1 commit bcc5486

14 files changed

+463
-135
lines changed

src/doc/rust.md

+5-2
Original file line numberDiff line numberDiff line change
@@ -2485,8 +2485,11 @@ before the expression they apply to.
24852485
`*`
24862486
: Dereference. When applied to a [pointer](#pointer-types) it denotes the pointed-to location.
24872487
For pointers to mutable locations, the resulting [lvalue](#lvalues-rvalues-and-temporaries) can be assigned to.
2488-
For [enums](#enumerated-types) that have only a single variant, containing a single parameter,
2489-
the dereference operator accesses this parameter.
2488+
On non-pointer types, it calls calls the `deref` method of the `std::ops::Deref` trait, or the
2489+
`deref_mut` method of the `std::ops::DerefMut` trait (if implemented by the type and required
2490+
for an outer expression that will or could mutate the dereference), and produces the
2491+
result of dereferencing the `&` or `&mut` borrowed pointer returned from the overload method.
2492+
24902493
`!`
24912494
: Logical negation. On the boolean type, this flips between `true` and
24922495
`false`. On integer types, this inverts the individual bits in the

src/librustc/middle/borrowck/mod.rs

+5-2
Original file line numberDiff line numberDiff line change
@@ -921,14 +921,17 @@ impl mc::Typer for TcxTyper {
921921
Ok(ty::node_id_to_type(self.tcx, id))
922922
}
923923

924+
fn node_method_ty(&mut self, id: ast::NodeId) -> Option<ty::t> {
925+
self.method_map.borrow().get().find(&id).map(|method| method.ty)
926+
}
927+
924928
fn adjustment(&mut self, id: ast::NodeId) -> Option<@ty::AutoAdjustment> {
925929
let adjustments = self.tcx.adjustments.borrow();
926930
adjustments.get().find_copy(&id)
927931
}
928932

929933
fn is_method_call(&mut self, id: ast::NodeId) -> bool {
930-
let method_map = self.method_map.borrow();
931-
method_map.get().contains_key(&id)
934+
self.method_map.borrow().get().contains_key(&id)
932935
}
933936

934937
fn temporary_scope(&mut self, id: ast::NodeId) -> Option<ast::NodeId> {

src/librustc/middle/mem_categorization.rs

+8-5
Original file line numberDiff line numberDiff line change
@@ -267,6 +267,7 @@ pub type McResult<T> = Result<T, ()>;
267267
pub trait Typer {
268268
fn tcx(&self) -> ty::ctxt;
269269
fn node_ty(&mut self, id: ast::NodeId) -> McResult<ty::t>;
270+
fn node_method_ty(&mut self, id: ast::NodeId) -> Option<ty::t>;
270271
fn adjustment(&mut self, node_id: ast::NodeId) -> Option<@ty::AutoAdjustment>;
271272
fn is_method_call(&mut self, id: ast::NodeId) -> bool;
272273
fn temporary_scope(&mut self, rvalue_id: ast::NodeId) -> Option<ast::NodeId>;
@@ -433,11 +434,13 @@ impl<TYPER:Typer> MemCategorizationContext<TYPER> {
433434
let expr_ty = if_ok!(self.expr_ty(expr));
434435
match expr.node {
435436
ast::ExprUnary(ast::UnDeref, e_base) => {
436-
if self.typer.is_method_call(expr.id) {
437-
return Ok(self.cat_rvalue_node(expr.id(), expr.span(), expr_ty));
438-
}
439-
440-
let base_cmt = if_ok!(self.cat_expr(e_base));
437+
let base_cmt = match self.typer.node_method_ty(expr.id) {
438+
Some(method_ty) => {
439+
let ref_ty = ty::ty_fn_ret(method_ty);
440+
self.cat_rvalue_node(expr.id(), expr.span(), ref_ty)
441+
}
442+
None => if_ok!(self.cat_expr(e_base))
443+
};
441444
Ok(self.cat_deref(expr, base_cmt, 0))
442445
}
443446

src/librustc/middle/trans/callee.rs

+4-11
Original file line numberDiff line numberDiff line change
@@ -444,14 +444,12 @@ pub fn trans_call<'a>(
444444
call_ex: &ast::Expr,
445445
f: &ast::Expr,
446446
args: CallArgs,
447-
id: ast::NodeId,
448447
dest: expr::Dest)
449448
-> &'a Block<'a> {
450449
let _icx = push_ctxt("trans_call");
451450
trans_call_inner(in_cx,
452451
Some(common::expr_info(call_ex)),
453452
expr_ty(in_cx, f),
454-
node_id_type(in_cx, id),
455453
|cx, _| trans(cx, f),
456454
args,
457455
Some(dest)).bcx
@@ -471,7 +469,6 @@ pub fn trans_method_call<'a>(
471469
bcx,
472470
Some(common::expr_info(call_ex)),
473471
monomorphize_type(bcx, method_ty),
474-
expr_ty(bcx, call_ex),
475472
|cx, arg_cleanup_scope| {
476473
meth::trans_method_callee(cx, call_ex.id, rcvr, arg_cleanup_scope)
477474
},
@@ -490,11 +487,9 @@ pub fn trans_lang_call<'a>(
490487
} else {
491488
csearch::get_type(bcx.ccx().tcx, did).ty
492489
};
493-
let rty = ty::ty_fn_ret(fty);
494490
callee::trans_call_inner(bcx,
495491
None,
496492
fty,
497-
rty,
498493
|bcx, _| {
499494
trans_fn_ref_with_vtables_to_callee(bcx,
500495
did,
@@ -520,12 +515,10 @@ pub fn trans_lang_call_with_type_params<'a>(
520515
fty = csearch::get_type(bcx.tcx(), did).ty;
521516
}
522517

523-
let rty = ty::ty_fn_ret(fty);
524518
return callee::trans_call_inner(
525519
bcx,
526520
None,
527521
fty,
528-
rty,
529522
|bcx, _| {
530523
let callee =
531524
trans_fn_ref_with_vtables_to_callee(bcx, did, 0,
@@ -554,7 +547,6 @@ pub fn trans_call_inner<'a>(
554547
bcx: &'a Block<'a>,
555548
call_info: Option<NodeInfo>,
556549
callee_ty: ty::t,
557-
ret_ty: ty::t,
558550
get_callee: |bcx: &'a Block<'a>,
559551
arg_cleanup_scope: cleanup::ScopeId|
560552
-> Callee<'a>,
@@ -610,9 +602,10 @@ pub fn trans_call_inner<'a>(
610602
}
611603
};
612604

613-
let abi = match ty::get(callee_ty).sty {
614-
ty::ty_bare_fn(ref f) => f.abis,
615-
_ => AbiSet::Rust()
605+
let (abi, ret_ty) = match ty::get(callee_ty).sty {
606+
ty::ty_bare_fn(ref f) => (f.abis, f.sig.output),
607+
ty::ty_closure(ref f) => (AbiSet::Rust(), f.sig.output),
608+
_ => fail!("expected bare rust fn or closure in trans_call_inner")
616609
};
617610
let is_rust_fn =
618611
abi.is_rust() ||

src/librustc/middle/trans/expr.rs

+21-34
Original file line numberDiff line numberDiff line change
@@ -467,10 +467,6 @@ fn trans_datum_unadjusted<'a>(bcx: &'a Block<'a>,
467467

468468
trans_binary(bcx, expr, op, lhs, rhs)
469469
}
470-
ast::ExprUnary(ast::UnDeref, base) => {
471-
let basedatum = unpack_datum!(bcx, trans(bcx, base));
472-
deref_once(bcx, expr, basedatum, 0)
473-
}
474470
ast::ExprUnary(op, x) => {
475471
trans_unary_datum(bcx, expr, op, x)
476472
}
@@ -782,12 +778,7 @@ fn trans_rvalue_dps_unadjusted<'a>(bcx: &'a Block<'a>,
782778
closure::trans_expr_fn(bcx, sigil, decl, body, expr.id, dest)
783779
}
784780
ast::ExprCall(f, ref args) => {
785-
callee::trans_call(bcx,
786-
expr,
787-
f,
788-
callee::ArgExprs(args.as_slice()),
789-
expr.id,
790-
dest)
781+
callee::trans_call(bcx, expr, f, callee::ArgExprs(args.as_slice()), dest)
791782
}
792783
ast::ExprMethodCall(_, _, ref args) => {
793784
callee::trans_method_call(bcx,
@@ -798,18 +789,15 @@ fn trans_rvalue_dps_unadjusted<'a>(bcx: &'a Block<'a>,
798789
}
799790
ast::ExprBinary(_, lhs, rhs) => {
800791
// if not overloaded, would be RvalueDatumExpr
801-
trans_overloaded_op(bcx, expr, lhs,
802-
Some(&*rhs), expr_ty(bcx, expr), dest)
792+
trans_overloaded_op(bcx, expr, lhs, Some(&*rhs), Some(dest)).bcx
803793
}
804794
ast::ExprUnary(_, subexpr) => {
805795
// if not overloaded, would be RvalueDatumExpr
806-
trans_overloaded_op(bcx, expr, subexpr,
807-
None, expr_ty(bcx, expr), dest)
796+
trans_overloaded_op(bcx, expr, subexpr, None, Some(dest)).bcx
808797
}
809798
ast::ExprIndex(base, idx) => {
810799
// if not overloaded, would be RvalueDatumExpr
811-
trans_overloaded_op(bcx, expr, base,
812-
Some(&*idx), expr_ty(bcx, expr), dest)
800+
trans_overloaded_op(bcx, expr, base, Some(&*idx), Some(dest)).bcx
813801
}
814802
ast::ExprCast(val, _) => {
815803
// DPS output mode means this is a trait cast:
@@ -1185,17 +1173,14 @@ fn trans_unary_datum<'a>(
11851173
let mut bcx = bcx;
11861174
let _icx = push_ctxt("trans_unary_datum");
11871175

1188-
// if deref, would be LvalueExpr
1189-
assert!(op != ast::UnDeref);
1190-
1191-
// if overloaded, would be RvalueDpsExpr
1192-
{
1176+
let overloaded = {
11931177
let method_map = bcx.ccx().maps.method_map.borrow();
1194-
assert!(!method_map.get().contains_key(&un_expr.id));
1195-
}
1178+
method_map.get().contains_key(&un_expr.id)
1179+
};
1180+
// if overloaded, would be RvalueDpsExpr
1181+
assert!(!overloaded || op == ast::UnDeref);
11961182

11971183
let un_ty = expr_ty(bcx, un_expr);
1198-
let sub_ty = expr_ty(bcx, sub_expr);
11991184

12001185
return match op {
12011186
ast::UnNot => {
@@ -1226,15 +1211,19 @@ fn trans_unary_datum<'a>(
12261211
immediate_rvalue_bcx(bcx, llneg, un_ty).to_expr_datumblock()
12271212
}
12281213
ast::UnBox => {
1229-
trans_boxed_expr(bcx, un_ty, sub_expr, sub_ty, heap_managed)
1214+
trans_boxed_expr(bcx, un_ty, sub_expr, expr_ty(bcx, sub_expr), heap_managed)
12301215
}
12311216
ast::UnUniq => {
1232-
trans_boxed_expr(bcx, un_ty, sub_expr, sub_ty, heap_exchange)
1217+
trans_boxed_expr(bcx, un_ty, sub_expr, expr_ty(bcx, sub_expr), heap_exchange)
12331218
}
12341219
ast::UnDeref => {
1235-
bcx.sess().bug("deref expressions should have been \
1236-
translated using trans_lvalue(), not \
1237-
trans_unary_datum()")
1220+
if overloaded {
1221+
let r = trans_overloaded_op(bcx, un_expr, sub_expr, None, None);
1222+
DatumBlock(r.bcx, Datum(r.val, un_ty, LvalueExpr))
1223+
} else {
1224+
let datum = unpack_datum!(bcx, trans(bcx, sub_expr));
1225+
deref_once(bcx, un_expr, datum, 0)
1226+
}
12381227
}
12391228
};
12401229
}
@@ -1506,22 +1495,20 @@ fn trans_overloaded_op<'a, 'b>(
15061495
expr: &ast::Expr,
15071496
rcvr: &'b ast::Expr,
15081497
arg: Option<&'b ast::Expr>,
1509-
ret_ty: ty::t,
1510-
dest: Dest)
1511-
-> &'a Block<'a> {
1498+
dest: Option<Dest>)
1499+
-> Result<'a> {
15121500
let method_ty = bcx.ccx().maps.method_map.borrow().get().get(&expr.id).ty;
15131501
callee::trans_call_inner(bcx,
15141502
Some(expr_info(expr)),
15151503
monomorphize_type(bcx, method_ty),
1516-
ret_ty,
15171504
|bcx, arg_cleanup_scope| {
15181505
meth::trans_method_callee(bcx,
15191506
expr.id,
15201507
rcvr,
15211508
arg_cleanup_scope)
15221509
},
15231510
callee::ArgAutorefSecond(rcvr, arg),
1524-
Some(dest)).bcx
1511+
dest)
15251512
}
15261513

15271514
fn int_cast(bcx: &Block,

src/librustc/middle/trans/reflect.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -105,9 +105,8 @@ impl<'a> Reflector<'a> {
105105
for (i, a) in args.iter().enumerate() {
106106
debug!("arg {}: {}", i, bcx.val_to_str(*a));
107107
}
108-
let bool_ty = ty::mk_bool();
109108
let result = unpack_result!(bcx, callee::trans_call_inner(
110-
self.bcx, None, mth_ty, bool_ty,
109+
self.bcx, None, mth_ty,
111110
|bcx, _| meth::trans_trait_callee_from_llval(bcx,
112111
mth_ty,
113112
mth_idx,

src/librustc/middle/ty.rs

+1
Original file line numberDiff line numberDiff line change
@@ -3284,6 +3284,7 @@ pub fn expr_kind(tcx: ctxt,
32843284
// exception, as its result is always unit.
32853285
return match expr.node {
32863286
ast::ExprAssignOp(..) => RvalueStmtExpr,
3287+
ast::ExprUnary(ast::UnDeref, _) => LvalueExpr,
32873288
_ => RvalueDpsExpr
32883289
};
32893290
}

0 commit comments

Comments
 (0)