| /* |
| * Copyright (c) 2022 Samsung Electronics Co., Ltd. |
| * All Rights Reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions are met: |
| * |
| * - Redistributions of source code must retain the above copyright notice, |
| * this list of conditions and the following disclaimer. |
| * |
| * - Redistributions in binary form must reproduce the above copyright notice, |
| * this list of conditions and the following disclaimer in the documentation |
| * and/or other materials provided with the distribution. |
| * |
| * - Neither the name of the copyright owner, nor the names of its contributors |
| * may be used to endorse or promote products derived from this software |
| * without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
| * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
| * ARE DISCLAIMED.IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE |
| * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| * CONSEQUENTIAL DAMAGES(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
| * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
| * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
| * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE |
| * POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #include "oapv_sad_avx.h" |
| |
| #if X86_SSE |
| |
| /* SAD ***********************************************************************/ |
| static int sad_16b_avx_8x8(int w, int h, void* src1, void* src2, int s_src1, int s_src2) |
| { |
| s16* s1 = (s16*)src1; |
| s16* s2 = (s16*)src2; |
| __m256i zero_vector = _mm256_setzero_si256(); |
| __m256i s1_vector, s2_vector, diff_vector, diff_abs1, diff_abs2; |
| // Because we are working with 16 elements at a time, stride is multiplied by 2. |
| s16 s1_stride = 2 * s_src1; |
| s16 s2_stride = 2 * s_src2; |
| { // Row 0 and Row 1 |
| // Load Row 0 and Row 1 data into registers. |
| s1_vector = _mm256_loadu_si256((const __m256i*)(s1)); |
| s1 += s1_stride; |
| s2_vector = _mm256_loadu_si256((const __m256i*)(s2)); |
| s2 += s2_stride; |
| // Calculate absolute difference between two rows. |
| diff_vector = _mm256_sub_epi16(s1_vector, s2_vector); |
| diff_abs1 = _mm256_abs_epi16(diff_vector); |
| } |
| { // Row 2 and Row 3 |
| s1_vector = _mm256_loadu_si256((const __m256i*)(s1)); |
| s1 += s1_stride; |
| s2_vector = _mm256_loadu_si256((const __m256i*)(s2)); |
| s2 += s2_stride; |
| diff_vector = _mm256_sub_epi16(s1_vector, s2_vector); |
| diff_abs2 = _mm256_abs_epi16(diff_vector); |
| } |
| // Add absolute differences to running total. |
| __m256i sum = _mm256_add_epi16(diff_abs1, diff_abs2); |
| { // Row 4 and Row 5 |
| s1_vector = _mm256_loadu_si256((const __m256i*)(s1)); |
| s1 += s1_stride; |
| s2_vector = _mm256_loadu_si256((const __m256i*)(s2)); |
| s2 += s2_stride; |
| diff_vector = _mm256_sub_epi16(s1_vector, s2_vector); |
| diff_abs2 = _mm256_abs_epi16(diff_vector); |
| sum = _mm256_add_epi16(sum, diff_abs2); |
| } |
| { // Row 6 and Row 7 |
| s1_vector = _mm256_loadu_si256((const __m256i*)(s1)); |
| s2_vector = _mm256_loadu_si256((const __m256i*)(s2)); |
| diff_vector = _mm256_sub_epi16(s1_vector, s2_vector); |
| diff_abs2 = _mm256_abs_epi16(diff_vector); |
| sum = _mm256_add_epi16(sum, diff_abs2); |
| } |
| // Convert 16-bit integers to 32-bit integers for summation. |
| __m128i sum_low = _mm256_extracti128_si256(sum, 0); |
| __m128i sum_high = _mm256_extracti128_si256(sum, 1); |
| __m256i sum_low_32 = _mm256_cvtepi16_epi32(sum_low); |
| __m256i sum_high_32 = _mm256_cvtepi16_epi32(sum_high); |
| // Sum up all the values in the array to get final SAD value. |
| sum = _mm256_add_epi32(sum_low_32, sum_high_32); |
| __m256i sum_hadd = _mm256_hadd_epi32(sum, zero_vector); // Horizontal add with zeros |
| sum = _mm256_hadd_epi32(sum_hadd, zero_vector); // Horizontal add with zeros |
| int sum1 = _mm256_extract_epi32(sum, 0); |
| int sum2 = _mm256_extract_epi32(sum, 4); |
| int sad = sum1 + sum2; |
| return sad; |
| } |
| |
| const oapv_fn_sad_t oapv_tbl_fn_sad_16b_avx[2] = |
| { |
| sad_16b_avx_8x8, |
| NULL |
| }; |
| |
| /* SSD ***********************************************************************/ |
| static s64 ssd_16b_avx_8x8(int w, int h, void* src1, void* src2, int s_src1, int s_src2) |
| { |
| s16* s1 = (s16*)src1; |
| s16* s2 = (s16*)src2; |
| __m256i s1_vector, s2_vector, diff_vector, sq_vector1, sq_vector2; |
| s64 sum_arr[4]; |
| // Because we are working with 16 elements at a time, stride is multiplied by 2. |
| s16 s1_stride = 2 * s_src1; |
| s16 s2_stride = 2 * s_src2; |
| s64 ssd = 0; |
| { // Row 0 and Row 1 |
| // Load Row 0 and Row 1 data into registers. |
| s1_vector = _mm256_loadu_si256((const __m256i*)(s1)); |
| s1 += s1_stride; |
| s2_vector = _mm256_loadu_si256((const __m256i*)(s2)); |
| s2 += s2_stride; |
| // Calculate squared difference between two rows. |
| diff_vector = _mm256_sub_epi16(s1_vector, s2_vector); |
| sq_vector1 = _mm256_madd_epi16(diff_vector, diff_vector); |
| } |
| { // Row 2 and Row 3 |
| s1_vector = _mm256_loadu_si256((const __m256i*)(s1)); |
| s1 += s1_stride; |
| s2_vector = _mm256_loadu_si256((const __m256i*)(s2)); |
| s2 += s2_stride; |
| diff_vector = _mm256_sub_epi16(s1_vector, s2_vector); |
| sq_vector2 = _mm256_madd_epi16(diff_vector, diff_vector); |
| } |
| // Add squared differences to running total. |
| __m256i sum = _mm256_add_epi32(sq_vector1, sq_vector2); |
| { // Row 4 and Row 5 |
| s1_vector = _mm256_loadu_si256((const __m256i*)(s1)); |
| s1 += s1_stride; |
| s2_vector = _mm256_loadu_si256((const __m256i*)(s2)); |
| s2 += s2_stride; |
| diff_vector = _mm256_sub_epi16(s1_vector, s2_vector); |
| sq_vector2 = _mm256_madd_epi16(diff_vector, diff_vector); |
| sum = _mm256_add_epi32(sum, sq_vector2); |
| } |
| { // Row 6 and Row 7 |
| s1_vector = _mm256_loadu_si256((const __m256i*)(s1)); |
| s2_vector = _mm256_loadu_si256((const __m256i*)(s2)); |
| diff_vector = _mm256_sub_epi16(s1_vector, s2_vector); |
| sq_vector2 = _mm256_madd_epi16(diff_vector, diff_vector); |
| sum = _mm256_add_epi32(sum, sq_vector2); |
| } |
| // Convert 16-bit integers to 32-bit integers for summation. |
| __m128i sum_low = _mm256_extracti128_si256(sum, 0); |
| __m128i sum_high = _mm256_extracti128_si256(sum, 1); |
| __m256i sum_low_64 = _mm256_cvtepi32_epi64(sum_low); |
| __m256i sum_high_64 = _mm256_cvtepi32_epi64(sum_high); |
| // Sum up all the values in the array to get final SSD value. |
| sum = _mm256_add_epi64(sum_low_64, sum_high_64); |
| _mm256_storeu_si256((__m256i*)sum_arr, sum); // store in array for summation. |
| ssd = sum_arr[0] + sum_arr[1] + sum_arr[2] + sum_arr[3]; |
| return ssd; |
| } |
| |
| const oapv_fn_ssd_t oapv_tbl_fn_ssd_16b_avx[2] = |
| { |
| ssd_16b_avx_8x8, |
| NULL |
| }; |
| |
| /* DIFF ***********************************************************************/ |
| static void diff_16b_avx_8x8(int w, int h, void* src1, void* src2, int s_src1, int s_src2, int s_diff, s16 *diff) |
| { |
| s16* s1 = (s16*)src1; |
| s16* s2 = (s16*)src2; |
| __m256i s1_vector, s2_vector, diff_vector; |
| // Because we are working with 16 elements at a time, stride is multiplied by 2. |
| s16 s1_stride = 2 * s_src1; |
| s16 s2_stride = 2 * s_src2; |
| s16 diff_stride = 2 * s_diff; |
| { // Row 0 and Row 1 |
| // Load Row 0 and Row 1 data into registers. |
| s1_vector = _mm256_loadu_si256((const __m256i*)(s1)); |
| s1 += s1_stride; |
| s2_vector = _mm256_loadu_si256((const __m256i*)(s2)); |
| s2 += s2_stride; |
| // Calculate difference between two rows and store it in diff buffer. |
| diff_vector = _mm256_sub_epi16(s1_vector, s2_vector); |
| _mm256_storeu_si256((__m256i*)diff, diff_vector); |
| diff += diff_stride; |
| } |
| { // Row 2 and Row 3 |
| s1_vector = _mm256_loadu_si256((const __m256i*)(s1)); |
| s1 += s1_stride; |
| s2_vector = _mm256_loadu_si256((const __m256i*)(s2)); |
| s2 += s2_stride; |
| diff_vector = _mm256_sub_epi16(s1_vector, s2_vector); |
| _mm256_storeu_si256((__m256i*)diff, diff_vector); |
| diff += diff_stride; |
| } |
| { // Row 4 and Row 5 |
| s1_vector = _mm256_loadu_si256((const __m256i*)(s1)); |
| s1 += s1_stride; |
| s2_vector = _mm256_loadu_si256((const __m256i*)(s2)); |
| s2 += s2_stride; |
| diff_vector = _mm256_sub_epi16(s1_vector, s2_vector); |
| _mm256_storeu_si256((__m256i*)diff, diff_vector); |
| diff += diff_stride; |
| } |
| { // Row 6 and Row 7 |
| s1_vector = _mm256_loadu_si256((const __m256i*)(s1)); |
| s2_vector = _mm256_loadu_si256((const __m256i*)(s2)); |
| diff_vector = _mm256_sub_epi16(s1_vector, s2_vector); |
| _mm256_storeu_si256((__m256i*)diff, diff_vector); |
| } |
| } |
| |
| const oapv_fn_diff_t oapv_tbl_fn_diff_16b_avx[2] = |
| { |
| diff_16b_avx_8x8, |
| NULL |
| }; |
| #endif |