Skip to content

Commit 9e27895

Browse files
committed
Auto merge of #30364 - luqmana:mir-calls, r=nikomatsakis
2 parents 5dd29cc + a6b861b commit 9e27895

File tree

2 files changed

+164
-24
lines changed

2 files changed

+164
-24
lines changed

src/librustc_trans/trans/mir/block.rs

+64-24
Original file line numberDiff line numberDiff line change
@@ -9,14 +9,18 @@
99
// except according to those terms.
1010

1111
use llvm::BasicBlockRef;
12+
use middle::infer;
13+
use middle::ty;
1214
use rustc::mir::repr as mir;
1315
use trans::adt;
1416
use trans::base;
1517
use trans::build;
16-
use trans::common::Block;
18+
use trans::common::{self, Block};
1719
use trans::debuginfo::DebugLoc;
20+
use trans::type_of;
1821

1922
use super::MirContext;
23+
use super::operand::OperandValue::{FatPtr, Immediate, Ref};
2024

2125
impl<'bcx, 'tcx> MirContext<'bcx, 'tcx> {
2226
pub fn trans_block(&mut self, bb: mir::BasicBlock) {
@@ -101,29 +105,65 @@ impl<'bcx, 'tcx> MirContext<'bcx, 'tcx> {
101105
base::build_return_block(bcx.fcx, bcx, return_ty, DebugLoc::None);
102106
}
103107

104-
mir::Terminator::Call { .. } => {
105-
unimplemented!()
106-
//let llbb = unimplemented!(); // self.make_landing_pad(panic_bb);
107-
//
108-
//let tr_dest = self.trans_lvalue(bcx, &data.destination);
109-
//
110-
//// Create the callee. This will always be a fn
111-
//// ptr and hence a kind of scalar.
112-
//let callee = self.trans_operand(bcx, &data.func);
113-
//
114-
//// Process the arguments.
115-
//
116-
//let args = unimplemented!();
117-
//
118-
//callee::trans_call_inner(bcx,
119-
// DebugLoc::None,
120-
// |bcx, _| Callee {
121-
// bcx: bcx,
122-
// data: CalleeData::Fn(callee.llval),
123-
// ty: callee.ty,
124-
// },
125-
// args,
126-
// Some(Dest::SaveIn(tr_dest.llval)));
108+
mir::Terminator::Call { ref data, targets } => {
109+
// The location we'll write the result of the call into.
110+
let call_dest = self.trans_lvalue(bcx, &data.destination);
111+
112+
// Create the callee. This will always be a fn
113+
// ptr and hence a kind of scalar.
114+
let callee = self.trans_operand(bcx, &data.func);
115+
let ret_ty = if let ty::TyBareFn(_, ref f) = callee.ty.sty {
116+
let sig = bcx.tcx().erase_late_bound_regions(&f.sig);
117+
let sig = infer::normalize_associated_type(bcx.tcx(), &sig);
118+
sig.output
119+
} else {
120+
panic!("trans_block: expected TyBareFn as callee");
121+
};
122+
123+
// The arguments we'll be passing
124+
let mut llargs = vec![];
125+
126+
// Does the fn use an outptr? If so, that's the first arg.
127+
if let ty::FnConverging(ret_ty) = ret_ty {
128+
if type_of::return_uses_outptr(bcx.ccx(), ret_ty) {
129+
llargs.push(call_dest.llval);
130+
}
131+
}
132+
133+
// Process the rest of the args.
134+
for arg in &data.args {
135+
let arg_op = self.trans_operand(bcx, arg);
136+
match arg_op.val {
137+
Ref(llval) | Immediate(llval) => llargs.push(llval),
138+
FatPtr(base, extra) => {
139+
// The two words in a fat ptr are passed separately
140+
llargs.push(base);
141+
llargs.push(extra);
142+
}
143+
}
144+
}
145+
146+
// FIXME: Handle panics
147+
//let panic_bb = self.llblock(targets.1);
148+
//self.make_landing_pad(panic_bb);
149+
150+
// Do the actual call.
151+
let (llret, b) = base::invoke(bcx,
152+
callee.immediate(),
153+
&llargs[..],
154+
callee.ty,
155+
DebugLoc::None);
156+
bcx = b;
157+
158+
// Copy the return value into the destination.
159+
if let ty::FnConverging(ret_ty) = ret_ty {
160+
if !type_of::return_uses_outptr(bcx.ccx(), ret_ty) &&
161+
!common::type_is_zero_size(bcx.ccx(), ret_ty) {
162+
base::store_ty(bcx, llret, call_dest.llval, ret_ty);
163+
}
164+
}
165+
166+
build::Br(bcx, self.llblock(targets.0), DebugLoc::None)
127167
}
128168
}
129169
}

src/test/run-pass/mir_trans_calls.rs

+100
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,100 @@
1+
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
#![feature(rustc_attrs)]
12+
13+
#[rustc_mir]
14+
fn test1(a: isize, b: (i32, i32), c: &[i32]) -> (isize, (i32, i32), &[i32]) {
15+
// Test passing a number of arguments including a fat pointer.
16+
// Also returning via an out pointer
17+
fn callee(a: isize, b: (i32, i32), c: &[i32]) -> (isize, (i32, i32), &[i32]) {
18+
(a, b, c)
19+
}
20+
callee(a, b, c)
21+
}
22+
23+
#[rustc_mir]
24+
fn test2(a: isize) -> isize {
25+
// Test passing a single argument.
26+
// Not using out pointer.
27+
fn callee(a: isize) -> isize {
28+
a
29+
}
30+
callee(a)
31+
}
32+
33+
struct Foo;
34+
impl Foo {
35+
fn inherent_method(&self, a: isize) -> isize { a }
36+
}
37+
38+
#[rustc_mir]
39+
fn test3(x: &Foo, a: isize) -> isize {
40+
// Test calling inherent method
41+
x.inherent_method(a)
42+
}
43+
44+
trait Bar {
45+
fn extension_method(&self, a: isize) -> isize { a }
46+
}
47+
impl Bar for Foo {}
48+
49+
#[rustc_mir]
50+
fn test4(x: &Foo, a: isize) -> isize {
51+
// Test calling extension method
52+
x.extension_method(a)
53+
}
54+
55+
#[rustc_mir]
56+
fn test5(x: &Bar, a: isize) -> isize {
57+
// Test calling method on trait object
58+
x.extension_method(a)
59+
}
60+
61+
#[rustc_mir]
62+
fn test6<T: Bar>(x: &T, a: isize) -> isize {
63+
// Test calling extension method on generic callee
64+
x.extension_method(a)
65+
}
66+
67+
trait One<T = Self> {
68+
fn one() -> T;
69+
}
70+
impl One for isize {
71+
fn one() -> isize { 1 }
72+
}
73+
74+
#[rustc_mir]
75+
fn test7() -> isize {
76+
// Test calling trait static method
77+
<isize as One>::one()
78+
}
79+
80+
struct Two;
81+
impl Two {
82+
fn two() -> isize { 2 }
83+
}
84+
85+
#[rustc_mir]
86+
fn test8() -> isize {
87+
// Test calling impl static method
88+
Two::two()
89+
}
90+
91+
fn main() {
92+
assert_eq!(test1(1, (2, 3), &[4, 5, 6]), (1, (2, 3), &[4, 5, 6][..]));
93+
assert_eq!(test2(98), 98);
94+
assert_eq!(test3(&Foo, 42), 42);
95+
assert_eq!(test4(&Foo, 970), 970);
96+
assert_eq!(test5(&Foo, 8576), 8576);
97+
assert_eq!(test6(&Foo, 12367), 12367);
98+
assert_eq!(test7(), 1);
99+
assert_eq!(test8(), 2);
100+
}

0 commit comments

Comments
 (0)