| Index: simd/jcsample-altivec.c
|
| diff --git a/simd/jcsample-altivec.c b/simd/jcsample-altivec.c
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..603492dbb3c999d959377f38ddba09028b2f69ea
|
| --- /dev/null
|
| +++ b/simd/jcsample-altivec.c
|
| @@ -0,0 +1,158 @@
|
| +/*
|
| + * AltiVec optimizations for libjpeg-turbo
|
| + *
|
| + * Copyright (C) 2015, D. R. Commander.
|
| + * All rights reserved.
|
| + * This software is provided 'as-is', without any express or implied
|
| + * warranty. In no event will the authors be held liable for any damages
|
| + * arising from the use of this software.
|
| + *
|
| + * Permission is granted to anyone to use this software for any purpose,
|
| + * including commercial applications, and to alter it and redistribute it
|
| + * freely, subject to the following restrictions:
|
| + *
|
| + * 1. The origin of this software must not be misrepresented; you must not
|
| + * claim that you wrote the original software. If you use this software
|
| + * in a product, an acknowledgment in the product documentation would be
|
| + * appreciated but is not required.
|
| + * 2. Altered source versions must be plainly marked as such, and must not be
|
| + * misrepresented as being the original software.
|
| + * 3. This notice may not be removed or altered from any source distribution.
|
| + */
|
| +
|
| +/* CHROMA DOWNSAMPLING */
|
| +
|
| +#include "jsimd_altivec.h"
|
| +#include "jcsample.h"
|
| +
|
| +
|
| +void
|
| +jsimd_h2v1_downsample_altivec (JDIMENSION image_width, int max_v_samp_factor,
|
| + JDIMENSION v_samp_factor,
|
| + JDIMENSION width_blocks,
|
| + JSAMPARRAY input_data, JSAMPARRAY output_data)
|
| +{
|
| + int outrow, outcol;
|
| + JDIMENSION output_cols = width_blocks * DCTSIZE;
|
| + JSAMPROW inptr, outptr;
|
| +
|
| + __vector unsigned char this0, next0, out;
|
| + __vector unsigned short this0e, this0o, next0e, next0o, outl, outh;
|
| +
|
| + /* Constants */
|
| + __vector unsigned short pw_bias = { __4X2(0, 1) },
|
| + pw_one = { __8X(1) };
|
| + __vector unsigned char even_odd_index =
|
| + {0,2,4,6,8,10,12,14,1,3,5,7,9,11,13,15},
|
| + pb_zero = { __16X(0) };
|
| +
|
| + expand_right_edge(input_data, max_v_samp_factor, image_width,
|
| + output_cols * 2);
|
| +
|
| + for (outrow = 0; outrow < v_samp_factor; outrow++) {
|
| + outptr = output_data[outrow];
|
| + inptr = input_data[outrow];
|
| +
|
| + for (outcol = output_cols; outcol > 0;
|
| + outcol -= 16, inptr += 32, outptr += 16) {
|
| +
|
| + this0 = vec_ld(0, inptr);
|
| + this0 = vec_perm(this0, this0, even_odd_index);
|
| + this0e = (__vector unsigned short)VEC_UNPACKHU(this0);
|
| + this0o = (__vector unsigned short)VEC_UNPACKLU(this0);
|
| + outl = vec_add(this0e, this0o);
|
| + outl = vec_add(outl, pw_bias);
|
| + outl = vec_sr(outl, pw_one);
|
| +
|
| + if (outcol > 8) {
|
| + next0 = vec_ld(16, inptr);
|
| + next0 = vec_perm(next0, next0, even_odd_index);
|
| + next0e = (__vector unsigned short)VEC_UNPACKHU(next0);
|
| + next0o = (__vector unsigned short)VEC_UNPACKLU(next0);
|
| + outh = vec_add(next0e, next0o);
|
| + outh = vec_add(outh, pw_bias);
|
| + outh = vec_sr(outh, pw_one);
|
| + } else
|
| + outh = vec_splat_u16(0);
|
| +
|
| + out = vec_pack(outl, outh);
|
| + vec_st(out, 0, outptr);
|
| + }
|
| + }
|
| +}
|
| +
|
| +
|
| +void
|
| +jsimd_h2v2_downsample_altivec (JDIMENSION image_width, int max_v_samp_factor,
|
| + JDIMENSION v_samp_factor,
|
| + JDIMENSION width_blocks,
|
| + JSAMPARRAY input_data, JSAMPARRAY output_data)
|
| +{
|
| + int inrow, outrow, outcol;
|
| + JDIMENSION output_cols = width_blocks * DCTSIZE;
|
| + JSAMPROW inptr0, inptr1, outptr;
|
| +
|
| + __vector unsigned char this0, next0, this1, next1, out;
|
| + __vector unsigned short this0e, this0o, next0e, next0o, this1e, this1o,
|
| + next1e, next1o, out0l, out0h, out1l, out1h, outl, outh;
|
| +
|
| + /* Constants */
|
| + __vector unsigned short pw_bias = { __4X2(1, 2) },
|
| + pw_two = { __8X(2) };
|
| + __vector unsigned char even_odd_index =
|
| + { 0, 2, 4, 6, 8, 10, 12, 14, 1, 3, 5, 7, 9, 11, 13, 15 },
|
| + pb_zero = { __16X(0) };
|
| +
|
| + expand_right_edge(input_data, max_v_samp_factor, image_width,
|
| + output_cols * 2);
|
| +
|
| + for (inrow = 0, outrow = 0; outrow < v_samp_factor;
|
| + inrow += 2, outrow++) {
|
| +
|
| + inptr0 = input_data[inrow];
|
| + inptr1 = input_data[inrow + 1];
|
| + outptr = output_data[outrow];
|
| +
|
| + for (outcol = output_cols; outcol > 0;
|
| + outcol -= 16, inptr0 += 32, inptr1 += 32, outptr += 16) {
|
| +
|
| + this0 = vec_ld(0, inptr0);
|
| + this0 = vec_perm(this0, this0, even_odd_index);
|
| + this0e = (__vector unsigned short)VEC_UNPACKHU(this0);
|
| + this0o = (__vector unsigned short)VEC_UNPACKLU(this0);
|
| + out0l = vec_add(this0e, this0o);
|
| +
|
| + this1 = vec_ld(0, inptr1);
|
| + this1 = vec_perm(this1, this1, even_odd_index);
|
| + this1e = (__vector unsigned short)VEC_UNPACKHU(this1);
|
| + this1o = (__vector unsigned short)VEC_UNPACKLU(this1);
|
| + out1l = vec_add(this1e, this1o);
|
| +
|
| + outl = vec_add(out0l, out1l);
|
| + outl = vec_add(outl, pw_bias);
|
| + outl = vec_sr(outl, pw_two);
|
| +
|
| + if (outcol > 8) {
|
| + next0 = vec_ld(16, inptr0);
|
| + next0 = vec_perm(next0, next0, even_odd_index);
|
| + next0e = (__vector unsigned short)VEC_UNPACKHU(next0);
|
| + next0o = (__vector unsigned short)VEC_UNPACKLU(next0);
|
| + out0h = vec_add(next0e, next0o);
|
| +
|
| + next1 = vec_ld(16, inptr1);
|
| + next1 = vec_perm(next1, next1, even_odd_index);
|
| + next1e = (__vector unsigned short)VEC_UNPACKHU(next1);
|
| + next1o = (__vector unsigned short)VEC_UNPACKLU(next1);
|
| + out1h = vec_add(next1e, next1o);
|
| +
|
| + outh = vec_add(out0h, out1h);
|
| + outh = vec_add(outh, pw_bias);
|
| + outh = vec_sr(outh, pw_two);
|
| + } else
|
| + outh = vec_splat_u16(0);
|
| +
|
| + out = vec_pack(outl, outh);
|
| + vec_st(out, 0, outptr);
|
| + }
|
| + }
|
| +}
|
|
|