|
9 | 9 | // except according to those terms.
|
10 | 10 |
|
11 | 11 | use llvm::BasicBlockRef;
|
| 12 | +use middle::infer; |
| 13 | +use middle::ty; |
12 | 14 | use rustc::mir::repr as mir;
|
13 | 15 | use trans::adt;
|
14 | 16 | use trans::base;
|
15 | 17 | use trans::build;
|
16 |
| -use trans::common::Block; |
| 18 | +use trans::common::{self, Block}; |
17 | 19 | use trans::debuginfo::DebugLoc;
|
| 20 | +use trans::type_of; |
18 | 21 |
|
19 | 22 | use super::MirContext;
|
| 23 | +use super::operand::OperandValue::{FatPtr, Immediate, Ref}; |
20 | 24 |
|
21 | 25 | impl<'bcx, 'tcx> MirContext<'bcx, 'tcx> {
|
22 | 26 | pub fn trans_block(&mut self, bb: mir::BasicBlock) {
|
@@ -101,29 +105,65 @@ impl<'bcx, 'tcx> MirContext<'bcx, 'tcx> {
|
101 | 105 | base::build_return_block(bcx.fcx, bcx, return_ty, DebugLoc::None);
|
102 | 106 | }
|
103 | 107 |
|
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) |
127 | 167 | }
|
128 | 168 | }
|
129 | 169 | }
|
|
0 commit comments