blob: 13a7fae5fcee16316ba557625ff1a25b349c9a02 [file] [log] [blame]
//! Implements portable vector shuffles with immediate indices.
// FIXME: comprehensive tests
// https://github.com/rust-lang-nursery/packed_simd/issues/20
/// Shuffles vector elements.
///
/// This macro returns a new vector that contains a shuffle of the elements in
/// one (`shuffle!(vec, [indices...])`) or two (`shuffle!(vec0, vec1,
/// [indices...])`) input vectors.
///
/// The type of `vec0` and `vec1` must be equal, and the element type of the
/// resulting vector is the element type of the input vector.
///
/// The number of `indices` must be a power-of-two in range `[0, 64)`, since
/// currently, the largest vector supported by the library has 64 lanes. The
/// length of the resulting vector equals the number of indices provided.
///
/// The indices must be in range `[0, M * N)` where `M` is the number of input
/// vectors (`1` or `2`) and `N` is the number of lanes of the input vectors.
/// The indices `i` in range `[0, N)` refer to the `i`-th element of `vec0`,
/// while the indices in range `[N, 2*N)` refer to the `i - N`-th element of
/// `vec1`.
///
/// # Examples
///
/// Shuffling elements of two vectors:
///
/// ```
/// # #[macro_use]
/// # extern crate packed_simd;
/// # use packed_simd::*;
/// # fn main() {
/// // Shuffle allows reordering the elements:
/// let x = i32x4::new(1, 2, 3, 4);
/// let y = i32x4::new(5, 6, 7, 8);
/// let r = shuffle!(x, y, [4, 0, 5, 1]);
/// assert_eq!(r, i32x4::new(5, 1, 6, 2));
///
/// // The resulting vector can als be smaller than the input:
/// let r = shuffle!(x, y, [1, 6]);
/// assert_eq!(r, i32x2::new(2, 7));
///
/// // Or larger:
/// let r = shuffle!(x, y, [1, 3, 4, 2, 1, 7, 2, 2]);
/// assert_eq!(r, i32x8::new(2, 4, 5, 3, 2, 8, 3, 3));
/// // At most 2 * the number of lanes in the input vector.
/// # }
/// ```
///
/// Shuffling elements of one vector:
///
/// ```
/// # #[macro_use]
/// # extern crate packed_simd;
/// # use packed_simd::*;
/// # fn main() {
/// // Shuffle allows reordering the elements of a vector:
/// let x = i32x4::new(1, 2, 3, 4);
/// let r = shuffle!(x, [2, 1, 3, 0]);
/// assert_eq!(r, i32x4::new(3, 2, 4, 1));
///
/// // The resulting vector can be smaller than the input:
/// let r = shuffle!(x, [1, 3]);
/// assert_eq!(r, i32x2::new(2, 4));
///
/// // Equal:
/// let r = shuffle!(x, [1, 3, 2, 0]);
/// assert_eq!(r, i32x4::new(2, 4, 3, 1));
///
/// // Or larger:
/// let r = shuffle!(x, [1, 3, 2, 2, 1, 3, 2, 2]);
/// assert_eq!(r, i32x8::new(2, 4, 3, 3, 2, 4, 3, 3));
/// // At most 2 * the number of lanes in the input vector.
/// # }
/// ```
#[macro_export]
macro_rules! shuffle {
($vec0:expr, $vec1:expr, [$l0:expr, $l1:expr]) => {{
#[allow(unused_unsafe)]
unsafe {
$crate::Simd($crate::__shuffle_vector2(
$vec0.0,
$vec1.0,
[$l0, $l1],
))
}
}};
($vec0:expr, $vec1:expr, [$l0:expr, $l1:expr, $l2:expr, $l3:expr]) => {{
#[allow(unused_unsafe)]
unsafe {
$crate::Simd($crate::__shuffle_vector4(
$vec0.0,
$vec1.0,
[$l0, $l1, $l2, $l3],
))
}
}};
($vec0:expr, $vec1:expr,
[$l0:expr, $l1:expr, $l2:expr, $l3:expr,
$l4:expr, $l5:expr, $l6:expr, $l7:expr]) => {{
#[allow(unused_unsafe)]
unsafe {
$crate::Simd($crate::__shuffle_vector8(
$vec0.0,
$vec1.0,
[$l0, $l1, $l2, $l3, $l4, $l5, $l6, $l7],
))
}
}};
($vec0:expr, $vec1:expr,
[$l0:expr, $l1:expr, $l2:expr, $l3:expr,
$l4:expr, $l5:expr, $l6:expr, $l7:expr,
$l8:expr, $l9:expr, $l10:expr, $l11:expr,
$l12:expr, $l13:expr, $l14:expr, $l15:expr]) => {{
#[allow(unused_unsafe)]
unsafe {
$crate::Simd($crate::__shuffle_vector16(
$vec0.0,
$vec1.0,
[
$l0, $l1, $l2, $l3, $l4, $l5, $l6, $l7, $l8, $l9, $l10,
$l11, $l12, $l13, $l14, $l15,
],
))
}
}};
($vec0:expr, $vec1:expr,
[$l0:expr, $l1:expr, $l2:expr, $l3:expr,
$l4:expr, $l5:expr, $l6:expr, $l7:expr,
$l8:expr, $l9:expr, $l10:expr, $l11:expr,
$l12:expr, $l13:expr, $l14:expr, $l15:expr,
$l16:expr, $l17:expr, $l18:expr, $l19:expr,
$l20:expr, $l21:expr, $l22:expr, $l23:expr,
$l24:expr, $l25:expr, $l26:expr, $l27:expr,
$l28:expr, $l29:expr, $l30:expr, $l31:expr]) => {{
#[allow(unused_unsafe)]
unsafe {
$crate::Simd($crate::__shuffle_vector32(
$vec0.0,
$vec1.0,
[
$l0, $l1, $l2, $l3, $l4, $l5, $l6, $l7, $l8, $l9, $l10,
$l11, $l12, $l13, $l14, $l15, $l16, $l17, $l18, $l19,
$l20, $l21, $l22, $l23, $l24, $l25, $l26, $l27, $l28,
$l29, $l30, $l31,
],
))
}
}};
($vec0:expr, $vec1:expr,
[$l0:expr, $l1:expr, $l2:expr, $l3:expr,
$l4:expr, $l5:expr, $l6:expr, $l7:expr,
$l8:expr, $l9:expr, $l10:expr, $l11:expr,
$l12:expr, $l13:expr, $l14:expr, $l15:expr,
$l16:expr, $l17:expr, $l18:expr, $l19:expr,
$l20:expr, $l21:expr, $l22:expr, $l23:expr,
$l24:expr, $l25:expr, $l26:expr, $l27:expr,
$l28:expr, $l29:expr, $l30:expr, $l31:expr,
$l32:expr, $l33:expr, $l34:expr, $l35:expr,
$l36:expr, $l37:expr, $l38:expr, $l39:expr,
$l40:expr, $l41:expr, $l42:expr, $l43:expr,
$l44:expr, $l45:expr, $l46:expr, $l47:expr,
$l48:expr, $l49:expr, $l50:expr, $l51:expr,
$l52:expr, $l53:expr, $l54:expr, $l55:expr,
$l56:expr, $l57:expr, $l58:expr, $l59:expr,
$l60:expr, $l61:expr, $l62:expr, $l63:expr]) => {{
#[allow(unused_unsafe)]
unsafe {
$crate::Simd($crate::__shuffle_vector64(
$vec0.0,
$vec1.0,
[
$l0, $l1, $l2, $l3, $l4, $l5, $l6, $l7, $l8, $l9, $l10,
$l11, $l12, $l13, $l14, $l15, $l16, $l17, $l18, $l19,
$l20, $l21, $l22, $l23, $l24, $l25, $l26, $l27, $l28,
$l29, $l30, $l31, $l32, $l33, $l34, $l35, $l36, $l37,
$l38, $l39, $l40, $l41, $l42, $l43, $l44, $l45, $l46,
$l47, $l48, $l49, $l50, $l51, $l52, $l53, $l54, $l55,
$l56, $l57, $l58, $l59, $l60, $l61, $l62, $l63,
],
))
}
}};
($vec:expr, [$($l:expr),*]) => {
match $vec {
v => shuffle!(v, v, [$($l),*])
}
};
}