/* RetroArch - A frontend for libretro. * Copyright (C) 2010-2013 - Hans-Kristian Arntzen * * RetroArch is free software: you can redistribute it and/or modify it under the terms * of the GNU General Public License as published by the Free Software Found- * ation, either version 3 of the License, or (at your option) any later version. * * RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR * PURPOSE. See the GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along with RetroArch. * If not, see . */ #include "pixconv.h" #include #include #include #include #ifdef SCALER_NO_SIMD #undef __SSE2__ #endif #if defined(__SSE2__) #include #endif #if defined(__SSE2_) void conv_rgb565_0rgb1555(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint16_t *input = (const uint16_t*)input_; uint16_t *output = (uint16_t*)output_; int max_width = width - 7; const __m128i hi_mask = _mm_set1_epi16(0x7fe0); const __m128i lo_mask = _mm_set1_epi16(0x1f); for (int h = 0; h < height; h++, output += out_stride >> 1, input += in_stride >> 1) { int w; for (w = 0; w < max_width; w += 8) { const __m128i in = _mm_loadu_si128((const __m128i*)(input + w)); __m128i hi = _mm_and_si128(_mm_slli_epi16(in, 1), hi_mask); __m128i lo = _mm_and_si128(in, lo_mask); _mm_storeu_si128((__m128i*)(output + w), _mm_or_si128(hi, lo)); } for (; w < width; w++) { uint16_t col = input[w]; uint16_t hi = (col >> 1) & 0x7fe0; uint16_t lo = col & 0x1f; output[w] = hi | lo; } } } #else void conv_rgb565_0rgb1555(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint16_t *input = (const uint16_t*)input_; uint16_t *output = (uint16_t*)output_; for (int h = 0; h < height; h++, output += out_stride >> 1, input += in_stride >> 1) { for (int w = 0; w < width; w++) { uint16_t col = input[w]; uint16_t hi = (col >> 1) & 0x7fe0; uint16_t lo = col & 0x1f; output[w] = hi | lo; } } } #endif #if defined(__SSE2__) void conv_0rgb1555_rgb565(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint16_t *input = (const uint16_t*)input_; uint16_t *output = (uint16_t*)output_; int max_width = width - 7; const __m128i hi_mask = _mm_set1_epi16((int16_t)((0x1f << 11) | (0x1f << 6))); const __m128i lo_mask = _mm_set1_epi16(0x1f); const __m128i glow_mask = _mm_set1_epi16(1 << 5); for (int h = 0; h < height; h++, output += out_stride >> 1, input += in_stride >> 1) { int w; for (w = 0; w < max_width; w += 8) { const __m128i in = _mm_loadu_si128((const __m128i*)(input + w)); __m128i rg = _mm_and_si128(_mm_slli_epi16(in, 1), hi_mask); __m128i b = _mm_and_si128(in, lo_mask); __m128i glow = _mm_and_si128(_mm_srli_epi16(in, 4), glow_mask); _mm_storeu_si128((__m128i*)(output + w), _mm_or_si128(rg, _mm_or_si128(b, glow))); } for (; w < width; w++) { uint16_t col = input[w]; uint16_t rg = (col << 1) & ((0x1f << 11) | (0x1f << 6)); uint16_t b = col & 0x1f; uint16_t glow = (col >> 4) & (1 << 5); output[w] = rg | b | glow; } } } #else void conv_0rgb1555_rgb565(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint16_t *input = (const uint16_t*)input_; uint16_t *output = (uint16_t*)output_; for (int h = 0; h < height; h++, output += out_stride >> 1, input += in_stride >> 1) { for (int w = 0; w < width; w++) { uint16_t col = input[w]; uint16_t rg = (col << 1) & ((0x1f << 11) | (0x1f << 6)); uint16_t b = col & 0x1f; uint16_t glow = (col >> 4) & (1 << 5); output[w] = rg | b | glow; } } } #endif #if defined(__SSE2__) void conv_0rgb1555_argb8888(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint16_t *input = (const uint16_t*)input_; uint32_t *output = (uint32_t*)output_; const __m128i pix_mask_r = _mm_set1_epi16(0x1f << 10); const __m128i pix_mask_gb = _mm_set1_epi16(0x1f << 5); const __m128i mul15_mid = _mm_set1_epi16(0x4200); const __m128i mul15_hi = _mm_set1_epi16(0x0210); const __m128i a = _mm_set1_epi16(0x00ff); int max_width = width - 7; for (int h = 0; h < height; h++, output += out_stride >> 2, input += in_stride >> 1) { int w; for (w = 0; w < max_width; w += 8) { const __m128i in = _mm_loadu_si128((const __m128i*)(input + w)); __m128i r = _mm_and_si128(in, pix_mask_r); __m128i g = _mm_and_si128(in, pix_mask_gb); __m128i b = _mm_and_si128(_mm_slli_epi16(in, 5), pix_mask_gb); r = _mm_mulhi_epi16(r, mul15_hi); g = _mm_mulhi_epi16(g, mul15_mid); b = _mm_mulhi_epi16(b, mul15_mid); __m128i res_lo_bg = _mm_unpacklo_epi8(b, g); __m128i res_hi_bg = _mm_unpackhi_epi8(b, g); __m128i res_lo_ra = _mm_unpacklo_epi8(r, a); __m128i res_hi_ra = _mm_unpackhi_epi8(r, a); __m128i res_lo = _mm_or_si128(res_lo_bg, _mm_slli_si128(res_lo_ra, 2)); __m128i res_hi = _mm_or_si128(res_hi_bg, _mm_slli_si128(res_hi_ra, 2)); _mm_storeu_si128((__m128i*)(output + w + 0), res_lo); _mm_storeu_si128((__m128i*)(output + w + 4), res_hi); } for (; w < width; w++) { uint32_t col = input[w]; uint32_t r = (col >> 10) & 0x1f; uint32_t g = (col >> 5) & 0x1f; uint32_t b = (col >> 0) & 0x1f; r = (r << 3) | (r >> 2); g = (g << 3) | (g >> 2); b = (b << 3) | (b >> 2); output[w] = (0xff << 24) | (r << 16) | (g << 8) | (b << 0); } } } #else void conv_0rgb1555_argb8888(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint16_t *input = (const uint16_t*)input_; uint32_t *output = (uint32_t*)output_; for (int h = 0; h < height; h++, output += out_stride >> 2, input += in_stride >> 1) { for (int w = 0; w < width; w++) { uint32_t col = input[w]; uint32_t r = (col >> 10) & 0x1f; uint32_t g = (col >> 5) & 0x1f; uint32_t b = (col >> 0) & 0x1f; r = (r << 3) | (r >> 2); g = (g << 3) | (g >> 2); b = (b << 3) | (b >> 2); output[w] = (0xffu << 24) | (r << 16) | (g << 8) | (b << 0); } } } #endif #if defined(__SSE2__) void conv_rgb565_argb8888(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint16_t *input = (const uint16_t*)input_; uint32_t *output = (uint32_t*)output_; const __m128i pix_mask_r = _mm_set1_epi16(0x1f << 10); const __m128i pix_mask_g = _mm_set1_epi16(0x3f << 5); const __m128i pix_mask_b = _mm_set1_epi16(0x1f << 5); const __m128i mul16_r = _mm_set1_epi16(0x0210); const __m128i mul16_g = _mm_set1_epi16(0x2080); const __m128i mul16_b = _mm_set1_epi16(0x4200); const __m128i a = _mm_set1_epi16(0x00ff); int max_width = width - 7; for (int h = 0; h < height; h++, output += out_stride >> 2, input += in_stride >> 1) { int w; for (w = 0; w < max_width; w += 8) { const __m128i in = _mm_loadu_si128((const __m128i*)(input + w)); __m128i r = _mm_and_si128(_mm_srli_epi16(in, 1), pix_mask_r); __m128i g = _mm_and_si128(in, pix_mask_g); __m128i b = _mm_and_si128(_mm_slli_epi16(in, 5), pix_mask_b); r = _mm_mulhi_epi16(r, mul16_r); g = _mm_mulhi_epi16(g, mul16_g); b = _mm_mulhi_epi16(b, mul16_b); __m128i res_lo_bg = _mm_unpacklo_epi8(b, g); __m128i res_hi_bg = _mm_unpackhi_epi8(b, g); __m128i res_lo_ra = _mm_unpacklo_epi8(r, a); __m128i res_hi_ra = _mm_unpackhi_epi8(r, a); __m128i res_lo = _mm_or_si128(res_lo_bg, _mm_slli_si128(res_lo_ra, 2)); __m128i res_hi = _mm_or_si128(res_hi_bg, _mm_slli_si128(res_hi_ra, 2)); _mm_storeu_si128((__m128i*)(output + w + 0), res_lo); _mm_storeu_si128((__m128i*)(output + w + 4), res_hi); } for (; w < width; w++) { uint32_t col = input[w]; uint32_t r = (col >> 11) & 0x1f; uint32_t g = (col >> 5) & 0x3f; uint32_t b = (col >> 0) & 0x1f; r = (r << 3) | (r >> 2); g = (g << 2) | (g >> 4); b = (b << 3) | (b >> 2); output[w] = (0xff << 24) | (r << 16) | (g << 8) | (b << 0); } } } #else void conv_rgb565_argb8888(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint16_t *input = (const uint16_t*)input_; uint32_t *output = (uint32_t*)output_; for (int h = 0; h < height; h++, output += out_stride >> 2, input += in_stride >> 1) { for (int w = 0; w < width; w++) { uint32_t col = input[w]; uint32_t r = (col >> 11) & 0x1f; uint32_t g = (col >> 5) & 0x3f; uint32_t b = (col >> 0) & 0x1f; r = (r << 3) | (r >> 2); g = (g << 2) | (g >> 4); b = (b << 3) | (b >> 2); output[w] = (0xffu << 24) | (r << 16) | (g << 8) | (b << 0); } } } #endif #if defined(__SSE2__) // :( TODO: Make this saner. static inline void store_bgr24_sse2(void *output, __m128i a, __m128i b, __m128i c, __m128i d) { const __m128i mask_0 = _mm_set_epi32(0, 0, 0, 0x00ffffff); const __m128i mask_1 = _mm_set_epi32(0, 0, 0x00ffffff, 0); const __m128i mask_2 = _mm_set_epi32(0, 0x00ffffff, 0, 0); const __m128i mask_3 = _mm_set_epi32(0x00ffffff, 0, 0, 0); __m128i a0 = _mm_and_si128(a, mask_0); __m128i a1 = _mm_srli_si128(_mm_and_si128(a, mask_1), 1); __m128i a2 = _mm_srli_si128(_mm_and_si128(a, mask_2), 2); __m128i a3 = _mm_srli_si128(_mm_and_si128(a, mask_3), 3); __m128i a4 = _mm_slli_si128(_mm_and_si128(b, mask_0), 12); __m128i a5 = _mm_slli_si128(_mm_and_si128(b, mask_1), 11); __m128i b0 = _mm_srli_si128(_mm_and_si128(b, mask_1), 5); __m128i b1 = _mm_srli_si128(_mm_and_si128(b, mask_2), 6); __m128i b2 = _mm_srli_si128(_mm_and_si128(b, mask_3), 7); __m128i b3 = _mm_slli_si128(_mm_and_si128(c, mask_0), 8); __m128i b4 = _mm_slli_si128(_mm_and_si128(c, mask_1), 7); __m128i b5 = _mm_slli_si128(_mm_and_si128(c, mask_2), 6); __m128i c0 = _mm_srli_si128(_mm_and_si128(c, mask_2), 10); __m128i c1 = _mm_srli_si128(_mm_and_si128(c, mask_3), 11); __m128i c2 = _mm_slli_si128(_mm_and_si128(d, mask_0), 4); __m128i c3 = _mm_slli_si128(_mm_and_si128(d, mask_1), 3); __m128i c4 = _mm_slli_si128(_mm_and_si128(d, mask_2), 2); __m128i c5 = _mm_slli_si128(_mm_and_si128(d, mask_3), 1); __m128i *out = (__m128i*)output; _mm_storeu_si128(out + 0, _mm_or_si128(a0, _mm_or_si128(a1, _mm_or_si128(a2, _mm_or_si128(a3, _mm_or_si128(a4, a5)))))); _mm_storeu_si128(out + 1, _mm_or_si128(b0, _mm_or_si128(b1, _mm_or_si128(b2, _mm_or_si128(b3, _mm_or_si128(b4, b5)))))); _mm_storeu_si128(out + 2, _mm_or_si128(c0, _mm_or_si128(c1, _mm_or_si128(c2, _mm_or_si128(c3, _mm_or_si128(c4, c5)))))); } void conv_0rgb1555_bgr24(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint16_t *input = (const uint16_t*)input_; uint8_t *output = (uint8_t*)output_; const __m128i pix_mask_r = _mm_set1_epi16(0x1f << 10); const __m128i pix_mask_gb = _mm_set1_epi16(0x1f << 5); const __m128i mul15_mid = _mm_set1_epi16(0x4200); const __m128i mul15_hi = _mm_set1_epi16(0x0210); const __m128i a = _mm_set1_epi16(0x00ff); int max_width = width - 15; for (int h = 0; h < height; h++, output += out_stride, input += in_stride >> 1) { uint8_t *out = output; int w; for (w = 0; w < max_width; w += 16, out += 48) { const __m128i in0 = _mm_loadu_si128((const __m128i*)(input + w + 0)); const __m128i in1 = _mm_loadu_si128((const __m128i*)(input + w + 8)); __m128i r0 = _mm_and_si128(in0, pix_mask_r); __m128i r1 = _mm_and_si128(in1, pix_mask_r); __m128i g0 = _mm_and_si128(in0, pix_mask_gb); __m128i g1 = _mm_and_si128(in1, pix_mask_gb); __m128i b0 = _mm_and_si128(_mm_slli_epi16(in0, 5), pix_mask_gb); __m128i b1 = _mm_and_si128(_mm_slli_epi16(in1, 5), pix_mask_gb); r0 = _mm_mulhi_epi16(r0, mul15_hi); r1 = _mm_mulhi_epi16(r1, mul15_hi); g0 = _mm_mulhi_epi16(g0, mul15_mid); g1 = _mm_mulhi_epi16(g1, mul15_mid); b0 = _mm_mulhi_epi16(b0, mul15_mid); b1 = _mm_mulhi_epi16(b1, mul15_mid); __m128i res_lo_bg0 = _mm_unpacklo_epi8(b0, g0); __m128i res_lo_bg1 = _mm_unpacklo_epi8(b1, g1); __m128i res_hi_bg0 = _mm_unpackhi_epi8(b0, g0); __m128i res_hi_bg1 = _mm_unpackhi_epi8(b1, g1); __m128i res_lo_ra0 = _mm_unpacklo_epi8(r0, a); __m128i res_lo_ra1 = _mm_unpacklo_epi8(r1, a); __m128i res_hi_ra0 = _mm_unpackhi_epi8(r0, a); __m128i res_hi_ra1 = _mm_unpackhi_epi8(r1, a); __m128i res_lo0 = _mm_or_si128(res_lo_bg0, _mm_slli_si128(res_lo_ra0, 2)); __m128i res_lo1 = _mm_or_si128(res_lo_bg1, _mm_slli_si128(res_lo_ra1, 2)); __m128i res_hi0 = _mm_or_si128(res_hi_bg0, _mm_slli_si128(res_hi_ra0, 2)); __m128i res_hi1 = _mm_or_si128(res_hi_bg1, _mm_slli_si128(res_hi_ra1, 2)); // Non-POT pixel sizes ftl :( store_bgr24_sse2(out, res_lo0, res_hi0, res_lo1, res_hi1); } for (; w < width; w++) { uint32_t col = input[w]; uint32_t b = (col >> 0) & 0x1f; uint32_t g = (col >> 5) & 0x1f; uint32_t r = (col >> 10) & 0x1f; b = (b << 3) | (b >> 2); g = (g << 3) | (g >> 2); r = (r << 3) | (r >> 2); *out++ = b; *out++ = g; *out++ = r; } } } void conv_rgb565_bgr24(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint16_t *input = (const uint16_t*)input_; uint8_t *output = (uint8_t*)output_; const __m128i pix_mask_r = _mm_set1_epi16(0x1f << 10); const __m128i pix_mask_g = _mm_set1_epi16(0x3f << 5); const __m128i pix_mask_b = _mm_set1_epi16(0x1f << 5); const __m128i mul16_r = _mm_set1_epi16(0x0210); const __m128i mul16_g = _mm_set1_epi16(0x2080); const __m128i mul16_b = _mm_set1_epi16(0x4200); const __m128i a = _mm_set1_epi16(0x00ff); int max_width = width - 15; for (int h = 0; h < height; h++, output += out_stride, input += in_stride >> 1) { uint8_t *out = output; int w; for (w = 0; w < max_width; w += 16, out += 48) { const __m128i in0 = _mm_loadu_si128((const __m128i*)(input + w)); const __m128i in1 = _mm_loadu_si128((const __m128i*)(input + w + 8)); __m128i r0 = _mm_and_si128(_mm_srli_epi16(in0, 1), pix_mask_r); __m128i g0 = _mm_and_si128(in0, pix_mask_g); __m128i b0 = _mm_and_si128(_mm_slli_epi16(in0, 5), pix_mask_b); __m128i r1 = _mm_and_si128(_mm_srli_epi16(in1, 1), pix_mask_r); __m128i g1 = _mm_and_si128(in1, pix_mask_g); __m128i b1 = _mm_and_si128(_mm_slli_epi16(in1, 5), pix_mask_b); r0 = _mm_mulhi_epi16(r0, mul16_r); g0 = _mm_mulhi_epi16(g0, mul16_g); b0 = _mm_mulhi_epi16(b0, mul16_b); r1 = _mm_mulhi_epi16(r1, mul16_r); g1 = _mm_mulhi_epi16(g1, mul16_g); b1 = _mm_mulhi_epi16(b1, mul16_b); __m128i res_lo_bg0 = _mm_unpacklo_epi8(b0, g0); __m128i res_hi_bg0 = _mm_unpackhi_epi8(b0, g0); __m128i res_lo_ra0 = _mm_unpacklo_epi8(r0, a); __m128i res_hi_ra0 = _mm_unpackhi_epi8(r0, a); __m128i res_lo_bg1 = _mm_unpacklo_epi8(b1, g1); __m128i res_hi_bg1 = _mm_unpackhi_epi8(b1, g1); __m128i res_lo_ra1 = _mm_unpacklo_epi8(r1, a); __m128i res_hi_ra1 = _mm_unpackhi_epi8(r1, a); __m128i res_lo0 = _mm_or_si128(res_lo_bg0, _mm_slli_si128(res_lo_ra0, 2)); __m128i res_hi0 = _mm_or_si128(res_hi_bg0, _mm_slli_si128(res_hi_ra0, 2)); __m128i res_lo1 = _mm_or_si128(res_lo_bg1, _mm_slli_si128(res_lo_ra1, 2)); __m128i res_hi1 = _mm_or_si128(res_hi_bg1, _mm_slli_si128(res_hi_ra1, 2)); store_bgr24_sse2(out, res_lo0, res_hi0, res_lo1, res_hi1); } for (; w < width; w++) { uint32_t col = input[w]; uint32_t r = (col >> 11) & 0x1f; uint32_t g = (col >> 5) & 0x3f; uint32_t b = (col >> 0) & 0x1f; r = (r << 3) | (r >> 2); g = (g << 2) | (g >> 4); b = (b << 3) | (b >> 2); *out++ = b; *out++ = g; *out++ = r; } } } #else void conv_0rgb1555_bgr24(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint16_t *input = (const uint16_t*)input_; uint8_t *output = (uint8_t*)output_; for (int h = 0; h < height; h++, output += out_stride, input += in_stride >> 1) { uint8_t *out = output; for (int w = 0; w < width; w++) { uint32_t col = input[w]; uint32_t b = (col >> 0) & 0x1f; uint32_t g = (col >> 5) & 0x1f; uint32_t r = (col >> 10) & 0x1f; b = (b << 3) | (b >> 2); g = (g << 3) | (g >> 2); r = (r << 3) | (r >> 2); *out++ = b; *out++ = g; *out++ = r; } } } void conv_rgb565_bgr24(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint16_t *input = (const uint16_t*)input_; uint8_t *output = (uint8_t*)output_; for (int h = 0; h < height; h++, output += out_stride, input += in_stride >> 1) { uint8_t *out = output; for (int w = 0; w < width; w++) { uint32_t col = input[w]; uint32_t b = (col >> 0) & 0x1f; uint32_t g = (col >> 5) & 0x3f; uint32_t r = (col >> 11) & 0x1f; b = (b << 3) | (b >> 2); g = (g << 2) | (g >> 4); r = (r << 3) | (r >> 2); *out++ = b; *out++ = g; *out++ = r; } } } #endif void conv_bgr24_argb8888(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint8_t *input = (const uint8_t*)input_; uint32_t *output = (uint32_t*)output_; for (int h = 0; h < height; h++, output += out_stride >> 2, input += in_stride) { const uint8_t *inp = input; for (int w = 0; w < width; w++) { uint32_t b = *inp++; uint32_t g = *inp++; uint32_t r = *inp++; output[w] = (0xffu << 24) | (r << 16) | (g << 8) | (b << 0); } } } void conv_argb8888_0rgb1555(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint32_t *input = (const uint32_t*)input_; uint16_t *output = (uint16_t*)output_; for (int h = 0; h < height; h++, output += out_stride >> 1, input += in_stride >> 2) { for (int w = 0; w < width; w++) { uint32_t col = input[w]; uint16_t r = (col >> 19) & 0x1f; uint16_t g = (col >> 11) & 0x1f; uint16_t b = (col >> 3) & 0x1f; output[w] = (r << 10) | (g << 5) | (b << 0); } } } #if defined(__SSE2__) void conv_argb8888_bgr24(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint32_t *input = (const uint32_t*)input_; uint8_t *output = (uint8_t*)output_; int max_width = width - 15; for (int h = 0; h < height; h++, output += out_stride, input += in_stride >> 2) { uint8_t *out = output; int w; for (w = 0; w < max_width; w += 16, out += 48) { store_bgr24_sse2(out, _mm_loadu_si128((const __m128i*)(input + w + 0)), _mm_loadu_si128((const __m128i*)(input + w + 4)), _mm_loadu_si128((const __m128i*)(input + w + 8)), _mm_loadu_si128((const __m128i*)(input + w + 12))); } for (; w < width; w++) { uint32_t col = input[w]; *out++ = (uint8_t)(col >> 0); *out++ = (uint8_t)(col >> 8); *out++ = (uint8_t)(col >> 16); } } } #else void conv_argb8888_bgr24(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint32_t *input = (const uint32_t*)input_; uint8_t *output = (uint8_t*)output_; for (int h = 0; h < height; h++, output += out_stride, input += in_stride >> 2) { uint8_t *out = output; for (int w = 0; w < width; w++) { uint32_t col = input[w]; *out++ = (uint8_t)(col >> 0); *out++ = (uint8_t)(col >> 8); *out++ = (uint8_t)(col >> 16); } } } #endif void conv_argb8888_abgr8888(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { const uint32_t *input = (const uint32_t*)input_; uint32_t *output = (uint32_t*)output_; for (int h = 0; h < height; h++, output += out_stride >> 2, input += in_stride >> 2) { for (int w = 0; w < width; w++) { uint32_t col = input[w]; output[w] = ((col << 16) & 0xff0000) | ((col >> 16) & 0xff) | (col & 0xff00ff00); } } } void conv_copy(void *output_, const void *input_, int width, int height, int out_stride, int in_stride) { int copy_len = abs(out_stride); if (abs(in_stride) < copy_len) copy_len = abs(in_stride); const uint8_t *input = (const uint8_t*)input_; uint8_t *output = (uint8_t*)output_; for (int h = 0; h < height; h++, output += out_stride, input += in_stride) memcpy(output, input, copy_len); }