| // Reference: ELF Application Binary Interface s390x Supplement | 
 | // https://github.com/IBM/s390x-abi | 
 |  | 
 | use rustc_abi::{BackendRepr, HasDataLayout, TyAbiInterface}; | 
 |  | 
 | use crate::callconv::{ArgAbi, FnAbi, Reg, RegKind}; | 
 | use crate::spec::HasTargetSpec; | 
 |  | 
 | fn classify_ret<Ty>(ret: &mut ArgAbi<'_, Ty>) { | 
 |     let size = ret.layout.size; | 
 |     if size.bits() <= 128 && matches!(ret.layout.backend_repr, BackendRepr::SimdVector { .. }) { | 
 |         return; | 
 |     } | 
 |     if !ret.layout.is_aggregate() && size.bits() <= 64 { | 
 |         ret.extend_integer_width_to(64); | 
 |     } else { | 
 |         ret.make_indirect(); | 
 |     } | 
 | } | 
 |  | 
 | fn classify_arg<'a, Ty, C>(cx: &C, arg: &mut ArgAbi<'a, Ty>) | 
 | where | 
 |     Ty: TyAbiInterface<'a, C> + Copy, | 
 |     C: HasDataLayout + HasTargetSpec, | 
 | { | 
 |     if !arg.layout.is_sized() { | 
 |         // Not touching this... | 
 |         return; | 
 |     } | 
 |     if arg.is_ignore() { | 
 |         // s390x-unknown-linux-{gnu,musl,uclibc} doesn't ignore ZSTs. | 
 |         if cx.target_spec().os == "linux" | 
 |             && matches!(&*cx.target_spec().env, "gnu" | "musl" | "uclibc") | 
 |             && arg.layout.is_zst() | 
 |         { | 
 |             arg.make_indirect_from_ignore(); | 
 |         } | 
 |         return; | 
 |     } | 
 |  | 
 |     let size = arg.layout.size; | 
 |     if size.bits() <= 128 { | 
 |         if let BackendRepr::SimdVector { .. } = arg.layout.backend_repr { | 
 |             // pass non-wrapped vector types using `PassMode::Direct` | 
 |             return; | 
 |         } | 
 |  | 
 |         if arg.layout.is_single_vector_element(cx, size) { | 
 |             // pass non-transparent wrappers around a vector as `PassMode::Cast` | 
 |             arg.cast_to(Reg { kind: RegKind::Vector, size }); | 
 |             return; | 
 |         } | 
 |     } | 
 |     if !arg.layout.is_aggregate() && size.bits() <= 64 { | 
 |         arg.extend_integer_width_to(64); | 
 |         return; | 
 |     } | 
 |  | 
 |     if arg.layout.is_single_fp_element(cx) { | 
 |         match size.bytes() { | 
 |             4 => arg.cast_to(Reg::f32()), | 
 |             8 => arg.cast_to(Reg::f64()), | 
 |             _ => arg.make_indirect(), | 
 |         } | 
 |     } else { | 
 |         match size.bytes() { | 
 |             1 => arg.cast_to(Reg::i8()), | 
 |             2 => arg.cast_to(Reg::i16()), | 
 |             4 => arg.cast_to(Reg::i32()), | 
 |             8 => arg.cast_to(Reg::i64()), | 
 |             _ => arg.make_indirect(), | 
 |         } | 
 |     } | 
 | } | 
 |  | 
 | pub(crate) fn compute_abi_info<'a, Ty, C>(cx: &C, fn_abi: &mut FnAbi<'a, Ty>) | 
 | where | 
 |     Ty: TyAbiInterface<'a, C> + Copy, | 
 |     C: HasDataLayout + HasTargetSpec, | 
 | { | 
 |     if !fn_abi.ret.is_ignore() { | 
 |         classify_ret(&mut fn_abi.ret); | 
 |     } | 
 |  | 
 |     for arg in fn_abi.args.iter_mut() { | 
 |         classify_arg(cx, arg); | 
 |     } | 
 | } |