// Copyright 2012-2015 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 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. //! # Translation of inline assembly. use llvm; use trans::build::*; use trans::callee; use trans::common::*; use trans::cleanup; use trans::cleanup::CleanupMethods; use trans::expr; use trans::type_of; use trans::type_::Type; use syntax::ast; use std::ffi::CString; use libc::{c_uint, c_char}; // Take an inline assembly expression and splat it out via LLVM pub fn trans_inline_asm<'blk, 'tcx>(bcx: Block<'blk, 'tcx>, ia: &ast::InlineAsm) -> Block<'blk, 'tcx> { let fcx = bcx.fcx; let mut bcx = bcx; let mut constraints = Vec::new(); let mut output_types = Vec::new(); let temp_scope = fcx.push_custom_cleanup_scope(); let mut ext_inputs = Vec::new(); let mut ext_constraints = Vec::new(); // Prepare the output operands let outputs = ia.outputs.iter().enumerate().map(|(i, &(ref c, ref out, is_rw))| { constraints.push((*c).clone()); let out_datum = unpack_datum!(bcx, expr::trans(bcx, &**out)); output_types.push(type_of::type_of(bcx.ccx(), out_datum.ty)); let val = out_datum.val; if is_rw { ext_inputs.push(unpack_result!(bcx, { callee::trans_arg_datum(bcx, expr_ty(bcx, &**out), out_datum, cleanup::CustomScope(temp_scope), callee::DontAutorefArg) })); ext_constraints.push(i.to_string()); } val }).collect::>(); // Now the input operands let mut inputs = ia.inputs.iter().map(|&(ref c, ref input)| { constraints.push((*c).clone()); let in_datum = unpack_datum!(bcx, expr::trans(bcx, &**input)); unpack_result!(bcx, { callee::trans_arg_datum(bcx, expr_ty(bcx, &**input), in_datum, cleanup::CustomScope(temp_scope), callee::DontAutorefArg) }) }).collect::>(); inputs.push_all(&ext_inputs[]); // no failure occurred preparing operands, no need to cleanup fcx.pop_custom_cleanup_scope(temp_scope); let mut constraints = constraints.iter() .map(|s| s.to_string()) .chain(ext_constraints.into_iter()) .collect::>() .connect(","); let mut clobbers = ia.clobbers.iter() .map(|s| format!("~{{{}}}", &s)) .collect::>() .connect(","); let more_clobbers = get_clobbers(); if !more_clobbers.is_empty() { if !clobbers.is_empty() { clobbers.push(','); } clobbers.push_str(&more_clobbers[]); } // Add the clobbers to our constraints list if clobbers.len() != 0 && constraints.len() != 0 { constraints.push(','); constraints.push_str(&clobbers[]); } else { constraints.push_str(&clobbers[]); } debug!("Asm Constraints: {}", &constraints[]); let num_outputs = outputs.len(); // Depending on how many outputs we have, the return type is different let output_type = if num_outputs == 0 { Type::void(bcx.ccx()) } else if num_outputs == 1 { output_types[0] } else { Type::struct_(bcx.ccx(), &output_types[], false) }; let dialect = match ia.dialect { ast::AsmAtt => llvm::AD_ATT, ast::AsmIntel => llvm::AD_Intel }; let asm = CString::from_slice(ia.asm.as_bytes()); let constraints = CString::from_slice(constraints.as_bytes()); let r = InlineAsmCall(bcx, asm.as_ptr(), constraints.as_ptr(), &inputs, output_type, ia.volatile, ia.alignstack, dialect); // Again, based on how many outputs we have if num_outputs == 1 { Store(bcx, r, outputs[0]); } else { for (i, o) in outputs.iter().enumerate() { let v = ExtractValue(bcx, r, i); Store(bcx, v, *o); } } // Store expn_id in a metadata node so we can map LLVM errors // back to source locations. See #17552. unsafe { let key = "srcloc"; let kind = llvm::LLVMGetMDKindIDInContext(bcx.ccx().llcx(), key.as_ptr() as *const c_char, key.len() as c_uint); let val: llvm::ValueRef = C_i32(bcx.ccx(), ia.expn_id.to_llvm_cookie()); llvm::LLVMSetMetadata(r, kind, llvm::LLVMMDNodeInContext(bcx.ccx().llcx(), &val, 1)); } return bcx; } // Default per-arch clobbers // Basically what clang does #[cfg(not(any(target_arch = "x86", target_arch = "x86_64")))] fn get_clobbers() -> String { "".to_string() } #[cfg(any(target_arch = "x86", target_arch = "x86_64"))] fn get_clobbers() -> String { "~{dirflag},~{fpsr},~{flags}".to_string() }