1//! `i686`'s Streaming SIMD Extensions 4.1 (SSE4.1)
2
3use crate::{core_arch::x86::*, intrinsics::simd::*, mem::transmute};
4
5#[cfg(test)]
6use stdarch_test::assert_instr;
7
8/// Extracts an 64-bit integer from `a` selected with `IMM1`
9///
10/// [Intel's documentation](https://www.intel.com/content/www/us/en/docs/intrinsics-guide/index.html#text=_mm_extract_epi64)
11#[inline]
12#[target_feature(enable = "sse4.1")]
13#[cfg_attr(all(test, not(target_os = "windows")), assert_instr(pextrq, IMM1 = 1))]
14#[rustc_legacy_const_generics(1)]
15#[stable(feature = "simd_x86", since = "1.27.0")]
16pub unsafe fn _mm_extract_epi64<const IMM1: i32>(a: __m128i) -> i64 {
17 static_assert_uimm_bits!(IMM1, 1);
18 simd_extract!(a.as_i64x2(), IMM1 as u32)
19}
20
21/// Returns a copy of `a` with the 64-bit integer from `i` inserted at a
22/// location specified by `IMM1`.
23///
24/// [Intel's documentation](https://www.intel.com/content/www/us/en/docs/intrinsics-guide/index.html#text=_mm_insert_epi64)
25#[inline]
26#[target_feature(enable = "sse4.1")]
27#[cfg_attr(test, assert_instr(pinsrq, IMM1 = 0))]
28#[rustc_legacy_const_generics(2)]
29#[stable(feature = "simd_x86", since = "1.27.0")]
30pub unsafe fn _mm_insert_epi64<const IMM1: i32>(a: __m128i, i: i64) -> __m128i {
31 static_assert_uimm_bits!(IMM1, 1);
32 transmute(src:simd_insert!(a.as_i64x2(), IMM1 as u32, i))
33}
34
35#[cfg(test)]
36mod tests {
37 use crate::core_arch::arch::x86_64::*;
38 use stdarch_test::simd_test;
39
40 #[simd_test(enable = "sse4.1")]
41 unsafe fn test_mm_extract_epi64() {
42 let a = _mm_setr_epi64x(0, 1);
43 let r = _mm_extract_epi64::<1>(a);
44 assert_eq!(r, 1);
45 let r = _mm_extract_epi64::<0>(a);
46 assert_eq!(r, 0);
47 }
48
49 #[simd_test(enable = "sse4.1")]
50 unsafe fn test_mm_insert_epi64() {
51 let a = _mm_set1_epi64x(0);
52 let e = _mm_setr_epi64x(0, 32);
53 let r = _mm_insert_epi64::<1>(a, 32);
54 assert_eq_m128i(r, e);
55 let e = _mm_setr_epi64x(32, 0);
56 let r = _mm_insert_epi64::<0>(a, 32);
57 assert_eq_m128i(r, e);
58 }
59}
60