diff options
| author | Dan Robertson <dan@dlrobertson.com> | 2018-12-05 02:44:08 +0000 |
|---|---|---|
| committer | Dan Robertson <dan@dlrobertson.com> | 2018-12-07 22:54:56 +0000 |
| commit | 3dfd8f7a64067d8b348ba597db10a06e2eccb773 (patch) | |
| tree | 26939b48c8c82ebd9e0dd970fb81a001b42f0ff0 /src/librustc_codegen_llvm | |
| parent | fc84f5f837a3e1b9b9bc992dd603d3d968502288 (diff) | |
| download | rust-3dfd8f7a64067d8b348ba597db10a06e2eccb773.tar.gz rust-3dfd8f7a64067d8b348ba597db10a06e2eccb773.zip | |
codegen: Fix va_list - aaarch64 iOS/Windows
According to the Apple developer docs: > The type va_list is an alias for char * rather than for the struct > type specified in the generic PCS. The current implementation uses the generic Aarch64 structure for VaList for Aarch64 iOS. Windows always uses the char * variant of the va_list.
Diffstat (limited to 'src/librustc_codegen_llvm')
| -rw-r--r-- | src/librustc_codegen_llvm/va_arg.rs | 36 |
1 files changed, 26 insertions, 10 deletions
diff --git a/src/librustc_codegen_llvm/va_arg.rs b/src/librustc_codegen_llvm/va_arg.rs index fbc3e6f06d1..1e5bb03ddb1 100644 --- a/src/librustc_codegen_llvm/va_arg.rs +++ b/src/librustc_codegen_llvm/va_arg.rs @@ -105,13 +105,30 @@ pub(super) fn emit_va_arg( ) -> &'ll Value { // Determine the va_arg implementation to use. The LLVM va_arg instruction // is lacking in some instances, so we should only use it as a fallback. + let target = &bx.cx.tcx.sess.target.target; let arch = &bx.cx.tcx.sess.target.target.arch; - match (&**arch, - bx.cx.tcx.sess.target.target.options.is_like_windows) { + match (&**arch, target.options.is_like_windows) { + // Windows x86 ("x86", true) => { emit_ptr_va_arg(bx, addr, target_ty, false, Align::from_bytes(4).unwrap(), false) } + // Generic x86 + ("x86", _) => { + emit_ptr_va_arg(bx, addr, target_ty, false, + Align::from_bytes(4).unwrap(), true) + } + // Windows Aarch64 + ("aarch4", true) => { + emit_ptr_va_arg(bx, addr, target_ty, false, + Align::from_bytes(8).unwrap(), false) + } + // iOS Aarch64 + ("aarch4", _) if target.target_os == "ios" => { + emit_ptr_va_arg(bx, addr, target_ty, false, + Align::from_bytes(8).unwrap(), true) + } + // Windows x86_64 ("x86_64", true) => { let target_ty_size = bx.cx.size_of(target_ty).bytes(); let indirect = if target_ty_size > 8 || !target_ty_size.is_power_of_two() { @@ -122,15 +139,14 @@ pub(super) fn emit_va_arg( emit_ptr_va_arg(bx, addr, target_ty, indirect, Align::from_bytes(8).unwrap(), false) } - ("x86", false) => { - emit_ptr_va_arg(bx, addr, target_ty, false, - Align::from_bytes(4).unwrap(), true) - } + // For all other architecture/OS combinations fall back to using + // the LLVM va_arg instruction. + // https://llvm.org/docs/LangRef.html#va-arg-instruction _ => { - let va_list = if (bx.tcx().sess.target.target.arch == "aarch64" || - bx.tcx().sess.target.target.arch == "x86_64" || - bx.tcx().sess.target.target.arch == "powerpc") && - !bx.tcx().sess.target.target.options.is_like_windows { + let va_list = if (target.arch == "aarch64" || + target.arch == "x86_64" || + target.arch == "powerpc") && + !target.options.is_like_windows { bx.load(addr.immediate(), bx.tcx().data_layout.pointer_align.abi) } else { addr.immediate() |
