blob: d8cef2b4e6eabbab023cf9efca1aeb7f88fe3c61 [file] [log] [blame]
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +00001/*
Michele Di Giorgiod9eaf612020-07-08 11:12:57 +01002 * Copyright (c) 2017-2020 Arm Limited.
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +00003 *
4 * SPDX-License-Identifier: MIT
5 *
6 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 * of this software and associated documentation files (the "Software"), to
8 * deal in the Software without restriction, including without limitation the
9 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
10 * sell copies of the Software, and to permit persons to whom the Software is
11 * furnished to do so, subject to the following conditions:
12 *
13 * The above copyright notice and this permission notice shall be included in all
14 * copies or substantial portions of the Software.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
22 * SOFTWARE.
23 */
24#include "helpers.h"
25
Michele Di Giorgiocbbed282019-12-20 13:26:08 +000026#if defined(DATA_TYPE) && defined(INITIAL_VALUE)
27#define VEC_TYPE(VEC_SIZE) VEC_DATA_TYPE(DATA_TYPE, VEC_SIZE)
28
Pablo Telloa52e4cf2019-04-01 14:55:18 +010029#if defined(OFFSET_IN1) && defined(OFFSET_OUT) && defined(SCALE_IN1) && defined(SCALE_OUT)
Michele Di Giorgiocbbed282019-12-20 13:26:08 +000030#define VEC_FLOAT(VEC_SIZE) VEC_DATA_TYPE(float, VEC_SIZE)
Michalis Spyrou4335a8c2019-04-05 16:41:30 +010031#define VEC_INT(VEC_SIZE) VEC_DATA_TYPE(int, VEC_SIZE)
Michalis Spyrou4335a8c2019-04-05 16:41:30 +010032#define CONVERT_RTE(x, type) (convert_##type##_rte((x)))
Pablo Telloa52e4cf2019-04-01 14:55:18 +010033#define CONVERT_DOWN(x, type) CONVERT_RTE(x, type)
34#define REQUANTIZE(VEC_SIZE, input, in_offset, out_offset, in_scale, out_scale, res) \
35 { \
36 const VEC_FLOAT(VEC_SIZE) in_f32 = (CONVERT(input, VEC_FLOAT(VEC_SIZE)) - (VEC_FLOAT(VEC_SIZE))((float)in_offset)) * (VEC_FLOAT(VEC_SIZE))((float)in_scale); \
37 const VEC_FLOAT(VEC_SIZE) out_f32 = in_f32 / ((VEC_FLOAT(VEC_SIZE))(float)out_scale) + ((VEC_FLOAT(VEC_SIZE))((float)out_offset)); \
Michele Di Giorgiocbbed282019-12-20 13:26:08 +000038 res = CONVERT_SAT(CONVERT_DOWN(out_f32, VEC_INT(VEC_SIZE)), VEC_TYPE(VEC_SIZE)); \
Pablo Telloa52e4cf2019-04-01 14:55:18 +010039 }
40#endif /* defined(OFFSET_IN1) && defined(OFFSET_OUT) && defined(SCALE_IN1) && defined(SCALE_OUT) */
41
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +000042#if defined(POOL_AVG)
43#define POOL_OP(x, y) ((x) + (y))
44#else /* defined(POOL_AVG) */
45#define POOL_OP(x, y) (max((x), (y)))
46#endif /* defined(POOL_AVG) */
47
48#define DIV_OP(x, y) (x * (1.f / y))
49
50#if defined(POOL_L2)
51#error "L2 pooling is not supported"
52#endif /* defined(POOL_L2) */
53
Isabella Gottardia527e8c2018-01-31 17:49:25 +000054int calculate_avg_scale(const int pool_size_x, const int pool_size_y, const int upper_bound_w, const int upper_bound_h,
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +000055 const int pad_x, const int pad_y, const int stride_x, const int stride_y)
56{
57 int start_x = get_global_id(0) * stride_x - pad_x;
58 int start_y = get_global_id(1) * stride_y - pad_y;
Isabella Gottardia527e8c2018-01-31 17:49:25 +000059 const int end_x = min(start_x + pool_size_x, upper_bound_w);
60 const int end_y = min(start_y + pool_size_y, upper_bound_h);
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +000061#if defined(EXCLUDE_PADDING)
62 start_x = max(0, start_x);
63 start_y = max(0, start_y);
64#endif /* defined(EXCLUDE_PADDING) */
65 return ((end_y - start_y) * (end_x - start_x));
66}
67
Michalis Spyroue74b2012018-04-18 09:49:16 +010068/** Performs a pooling function of pool size equal to N (NCHW)
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +000069 *
Isabella Gottardia527e8c2018-01-31 17:49:25 +000070 * @note Pool sizes must be passed using -DPOOL_SIZE_X and -DPOOL_SIZE_Y e.g. -DPOOL_SIZE_X=13;
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +000071 * @note In case of average pooling the following information must be passed at compile time:
72 * -DPOOL_AVG must be provided otherwise max pooling will be performed.
73 * -DMAX_WIDTH and -DMAX_HEIGHT which are the maximum accessible indeces in x and y dimensions (width + pad)
74 * -DSTRIDE_X and -DSTRIDE_Y which are the steps of the window along the x and y directions
75 * -DPAD_X and -DPAD_Y which are the pooling paddings in x and y dimension
Michele Di Giorgiocbbed282019-12-20 13:26:08 +000076 * @note Input data type must be passed at compile time using -DDAT_TYPE=type, e.g. -DDATA_TYPE=uchar
77 * @note The initial value for the pooling operation must be passed at compile time using -DINITIAL_VALUE e.g. -DINITIAL_VALUE=0
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +000078 *
Michele Di Giorgiocbbed282019-12-20 13:26:08 +000079 * @param[in] input_ptr Pointer to the source image. Supported data types: QASYMM8/QASYMM8_SIGNED
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +000080 * @param[in] input_stride_x Stride of the source image in X dimension (in bytes)
81 * @param[in] input_step_x input_stride_x * number of elements along X processed per workitem(in bytes)
82 * @param[in] input_stride_y Stride of the source image in Y dimension (in bytes)
83 * @param[in] input_step_y input_stride_y * number of elements along Y processed per workitem(in bytes)
84 * @param[in] input_stride_z Stride of the source tensor in Z dimension (in bytes)
85 * @param[in] input_step_z input_stride_z * number of elements along Z processed per workitem(in bytes)
86 * @param[in] input_offset_first_element_in_bytes The offset of the first element in the source image
87 * @param[out] output_ptr Pointer to the destination image. Supported data types: same as @p input_ptr
88 * @param[in] output_stride_x Stride of the destination image in X dimension (in bytes)
89 * @param[in] output_step_x output_stride_x * number of elements along X processed per workitem(in bytes)
90 * @param[in] output_stride_y Stride of the destination image in Y dimension (in bytes)
91 * @param[in] output_step_y output_stride_y * number of elements along Y processed per workitem(in bytes)
92 * @param[in] output_stride_z Stride of the source tensor in Z dimension (in bytes)
93 * @param[in] output_step_z output_stride_z * number of elements along Z processed per workitem(in bytes)
94 * @param[in] output_offset_first_element_in_bytes The offset of the first element in the destination image
95 */
Michalis Spyroue74b2012018-04-18 09:49:16 +010096__kernel void pooling_layer_MxN_quantized_nchw(
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +000097 TENSOR3D_DECLARATION(input),
98 TENSOR3D_DECLARATION(output))
99{
100 // Get pixels pointer
101 Tensor3D input = CONVERT_TO_TENSOR3D_STRUCT(input);
102 Tensor3D output = CONVERT_TO_TENSOR3D_STRUCT(output);
103
Michele Di Giorgiocbbed282019-12-20 13:26:08 +0000104 int8 vdata = INITIAL_VALUE;
105 int sdata = INITIAL_VALUE;
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +0000106
107 // Load data
Isabella Gottardia527e8c2018-01-31 17:49:25 +0000108 for(int y = 0; y < POOL_SIZE_Y; y++)
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +0000109 {
110 int x = 0;
Isabella Gottardia527e8c2018-01-31 17:49:25 +0000111 for(; x <= ((int)POOL_SIZE_X - 8); x += 8)
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +0000112 {
Michele Di Giorgiocbbed282019-12-20 13:26:08 +0000113 VEC_TYPE(8)
114 data = vload8(0, (__global DATA_TYPE *)tensor3D_offset(&input, x, y, 0));
115 int8 data0 = convert_int8(data);
116 vdata = POOL_OP(vdata, data0);
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +0000117 }
118
119 // Leftover
Isabella Gottardia527e8c2018-01-31 17:49:25 +0000120 for(; x < (int)POOL_SIZE_X; ++x)
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +0000121 {
Michele Di Giorgiocbbed282019-12-20 13:26:08 +0000122 DATA_TYPE data = *((__global DATA_TYPE *)tensor3D_offset(&input, x, y, 0));
123 int data0 = convert_int(data);
124 sdata = POOL_OP(sdata, data0);
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +0000125 }
126 }
127
128 // Reduce result
129 int4 reduce4 = POOL_OP(vdata.s0123, vdata.s4567);
130 int2 reduce2 = POOL_OP(reduce4.s01, reduce4.s23);
131 int res = POOL_OP(reduce2.s0, reduce2.s1);
132 res = POOL_OP(res, sdata);
133
134#if defined(POOL_AVG)
Isabella Gottardia527e8c2018-01-31 17:49:25 +0000135 res = round(DIV_OP(res, calculate_avg_scale(POOL_SIZE_X, POOL_SIZE_Y, MAX_WIDTH, MAX_HEIGHT, PAD_X, PAD_Y, STRIDE_X, STRIDE_Y)));
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +0000136#endif /* defined(POOL_AVG) */
137
Michele Di Giorgiocbbed282019-12-20 13:26:08 +0000138 DATA_TYPE result_q8 = CONVERT(res, DATA_TYPE);
Pablo Telloa52e4cf2019-04-01 14:55:18 +0100139
140#if defined(OFFSET_IN1) && defined(OFFSET_OUT) && defined(SCALE_IN1) && defined(SCALE_OUT)
141
Michele Di Giorgiocbbed282019-12-20 13:26:08 +0000142 const float result_f32 = convert_float(result_q8);
Pablo Telloa52e4cf2019-04-01 14:55:18 +0100143 const float input_offset = (float)OFFSET_IN1;
144 const float input_scale = (float)SCALE_IN1;
145 const float scale_out = (float)SCALE_OUT;
146 const float offset_out = (float)OFFSET_OUT;
147 const float in_f32 = (result_f32 - input_offset) * input_scale;
148 const float out_f32 = in_f32 / scale_out + offset_out;
Michele Di Giorgiocbbed282019-12-20 13:26:08 +0000149 result_q8 = CONVERT_SAT(convert_int_rte(out_f32), DATA_TYPE);
Pablo Telloa52e4cf2019-04-01 14:55:18 +0100150
151#endif /* defined(OFFSET_IN1) && defined(OFFSET_OUT) && defined(SCALE_IN1) && defined(SCALE_OUT) */
152
Michele Di Giorgiocbbed282019-12-20 13:26:08 +0000153 *(__global DATA_TYPE *)output.ptr = result_q8;
Anton Lokhmotovaf6204c2017-11-08 09:34:19 +0000154}
Michalis Spyroue74b2012018-04-18 09:49:16 +0100155
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100156#if defined(VEC_SIZE) && defined(VEC_SIZE_LEFTOVER) && defined(SRC_WIDTH) && defined(SRC_HEIGHT) && defined(DST_CHANNELS) && defined(DST_HEIGHT) && defined(DST_BATCH_SIZE) && defined(ACC_DATA_TYPE)
157/** Performs pooling layer of size equal to MxN. This OpenCL kernel can perform the following pooling types:
158 * -# max, -DPOOL_MAX must be passed at compile time
159 * -# average, -DPOOL_AVG must be passed at compile time. If padding has to be expluded, -DEXCLUDE_PADDING should be passed at compile time
Michalis Spyroue74b2012018-04-18 09:49:16 +0100160 *
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100161 * @note Datatype must be passed at compile type using -DDATA_TYPE e.g. -DDATA_TYPE=uchar. Supported data types are QASYMM8/QASYMM8_SIGNED
162 * @note Accumulation data type must be passed at compile time using -DACC_DATA_TYPE e.g. -DACC_DATA_TYPE=int
163 * @note Pool size must be passed at compile time using -DPOOL_SIZE_X and -DPOOL_SIZE_Y. e.g. -DPOOL_SIZE_X=4, -DPOOL_SIZE_Y=4
164 * @note Input tensor width and height must be passed at compile time using -DSRC_WIDTH and -DSRC_HEIGHT
165 * @note Output tensor height, channels and batch size must be passed at compile time using -DDST_HEIGHT, -DDST_CHANNELS and -DDST_BATCH_SIZE
166 * @note Pool strides must be passed at compile time using -DSTRIDE_X and -DSTRIDE_Y which are the steps of the window along the x and y directions
167 * @note Pool pads must be passed at compile time using -DPAD_X and -DPAD_Y
168 * @note Vector size must be passed at compile time using -DVEC_SIZE=size. e.g. -DVEC_SIZE=16
169 * @note Leftover vector size must be passed at compile time using -DVEC_SIZE_LEFTOVER. e.g. -DVEC_SIZE_LEFTOVER=3. It is defined as the remainder between the input's first dimension and VEC_SIZE
Michele Di Giorgiocbbed282019-12-20 13:26:08 +0000170 * @note The initial value for the pooling operation must be passed at compile time using -DINITIAL_VALUE e.g. -DINITIAL_VALUE=0
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100171 * @note If the output has be requantized, -DOFFSET_IN1, -DOFFSET_OUT, -DSCALE_IN1 and -DSCALE_OUT muste be passed at compile time
Michalis Spyroue74b2012018-04-18 09:49:16 +0100172 *
Michele Di Giorgiocbbed282019-12-20 13:26:08 +0000173 * @param[in] input_ptr Pointer to the source image. Supported data types: QASYMM8/QASYMM8_SIGNED
Michalis Spyroue74b2012018-04-18 09:49:16 +0100174 * @param[in] input_stride_x Stride of the source image in X dimension (in bytes)
175 * @param[in] input_step_x input_stride_x * number of elements along X processed per workitem(in bytes)
176 * @param[in] input_stride_y Stride of the source image in Y dimension (in bytes)
177 * @param[in] input_step_y input_stride_y * number of elements along Y processed per workitem(in bytes)
178 * @param[in] input_stride_z Stride of the source tensor in Z dimension (in bytes)
179 * @param[in] input_step_z input_stride_z * number of elements along Z processed per workitem(in bytes)
Georgios Pinitas89d71732018-10-29 20:07:15 +0000180 * @param[in] input_stride_w Stride of the source tensor in W dimension (in bytes)
181 * @param[in] input_step_w input_stride_w * number of elements along W processed per workitem(in bytes)
Michalis Spyroue74b2012018-04-18 09:49:16 +0100182 * @param[in] input_offset_first_element_in_bytes The offset of the first element in the source image
183 * @param[out] output_ptr Pointer to the destination image. Supported data types: same as @p input_ptr
Georgios Pinitas89d71732018-10-29 20:07:15 +0000184 * @param[in] output_stride_x Stride of the destination tensor in X dimension (in bytes)
Michalis Spyroue74b2012018-04-18 09:49:16 +0100185 * @param[in] output_step_x output_stride_x * number of elements along X processed per workitem(in bytes)
Georgios Pinitas89d71732018-10-29 20:07:15 +0000186 * @param[in] output_stride_y Stride of the destination tensor in Y dimension (in bytes)
Michalis Spyroue74b2012018-04-18 09:49:16 +0100187 * @param[in] output_step_y output_stride_y * number of elements along Y processed per workitem(in bytes)
Georgios Pinitas89d71732018-10-29 20:07:15 +0000188 * @param[in] output_stride_z Stride of the destination tensor in Z dimension (in bytes)
Michalis Spyroue74b2012018-04-18 09:49:16 +0100189 * @param[in] output_step_z output_stride_z * number of elements along Z processed per workitem(in bytes)
Georgios Pinitas89d71732018-10-29 20:07:15 +0000190 * @param[in] output_stride_w Stride of the destination tensor in W dimension (in bytes)
191 * @param[in] output_step_w output_stride_w * number of elements along W processed per workitem(in bytes)
Michalis Spyroue74b2012018-04-18 09:49:16 +0100192 * @param[in] output_offset_first_element_in_bytes The offset of the first element in the destination image
193 */
194__kernel void pooling_layer_MxN_quantized_nhwc(
Georgios Pinitas89d71732018-10-29 20:07:15 +0000195 TENSOR4D_DECLARATION(input),
196 TENSOR4D_DECLARATION(output))
Michalis Spyroue74b2012018-04-18 09:49:16 +0100197{
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100198 // Note: If C is not multiple of VEC_SIZE, we shift back of VEC_SIZE_LEFTOVER elements to compute the leftover elements for get_global_id(0) == 0
199 // Note: If C is less than VEC_SIZE, VEC_SIZE should be SHRINKED to the closest smaller VEC_SIZE. This operation is performed on the host side
Giorgio Arena2d1a8352020-10-26 15:04:08 +0000200 int offset_c = max((int)(get_global_id(0) * VEC_SIZE - (VEC_SIZE - VEC_SIZE_LEFTOVER) % VEC_SIZE), 0) * sizeof(DATA_TYPE);
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100201 int idx_out_w = get_global_id(1);
202#if DST_BATCH_SIZE != 1
203 // If batch size != 1, the batch size dimension is collapsed over the height dimension
204 int idx_out_h = get_global_id(2) % DST_HEIGHT;
205 int idx_out_n = get_global_id(2) / DST_HEIGHT;
Giorgio Arena2d1a8352020-10-26 15:04:08 +0000206#else //DST_BATCH_SIZE != 1
207 int idx_out_h = get_global_id(2);
208 int idx_out_n = 0;
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100209#endif // DST_BATCH_SIZE != 1
Michalis Spyroue74b2012018-04-18 09:49:16 +0100210
Giorgio Arena2d1a8352020-10-26 15:04:08 +0000211 int idx_in_w = idx_out_w * STRIDE_X - PAD_X;
212 int idx_in_h = idx_out_h * STRIDE_Y - PAD_Y;
Michalis Spyroue74b2012018-04-18 09:49:16 +0100213
Giorgio Arena2d1a8352020-10-26 15:04:08 +0000214 __global unsigned char *in_base_ptr = input_ptr + input_offset_first_element_in_bytes + offset_c + idx_out_n * input_stride_w;
Michalis Spyroue74b2012018-04-18 09:49:16 +0100215
Giorgio Arena2d1a8352020-10-26 15:04:08 +0000216 __global unsigned char *out_base_ptr = output_ptr + output_offset_first_element_in_bytes + offset_c + idx_out_w * output_stride_y + idx_out_h * output_stride_z + idx_out_n * output_stride_w;
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100217
218 int pool_x_s = max((int)0, -idx_in_w);
219 int pool_x_e = min((int)POOL_SIZE_X, (int)SRC_WIDTH - idx_in_w);
220 int pool_y_s = max((int)0, -idx_in_h);
221 int pool_y_e = min((int)POOL_SIZE_Y, (int)SRC_HEIGHT - idx_in_h);
222
223#if defined(POOL_AVG) && defined(EXCLUDE_PADDING)
224 int filter_size = 0;
225#elif defined(POOL_AVG) && !defined(EXCLUDE_PADDING) // defined(POOL_AVG) && defined(EXCLUDE_PADDING)
226 int filter_size = POOL_SIZE_X * POOL_SIZE_Y;
Giorgio Arena2d1a8352020-10-26 15:04:08 +0000227#endif // defined(POOL_AVG) && !defined(EXCLUDE_PADDING)
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100228
229 VEC_DATA_TYPE(ACC_DATA_TYPE, VEC_SIZE)
230 res0 = INITIAL_VALUE;
231
232 for(int y = pool_y_s; y < pool_y_e; ++y)
Michalis Spyroue74b2012018-04-18 09:49:16 +0100233 {
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100234 for(int x = pool_x_s; x < pool_x_e; ++x)
Michalis Spyroue74b2012018-04-18 09:49:16 +0100235 {
Giorgio Arena2d1a8352020-10-26 15:04:08 +0000236 VEC_DATA_TYPE(DATA_TYPE, VEC_SIZE)
237 data;
238 VEC_DATA_TYPE(ACC_DATA_TYPE, VEC_SIZE)
239 data0;
Georgios Pinitas89d71732018-10-29 20:07:15 +0000240
Giorgio Arena2d1a8352020-10-26 15:04:08 +0000241 data = VLOAD(VEC_SIZE)(0, (__global DATA_TYPE *)(in_base_ptr + (x + idx_in_w) * input_stride_y + (y + idx_in_h) * input_stride_z));
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100242 data0 = CONVERT(data, VEC_DATA_TYPE(ACC_DATA_TYPE, VEC_SIZE));
Georgios Pinitas89d71732018-10-29 20:07:15 +0000243
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100244 res0 = POOL_OP(res0, data0);
245
246#if defined(POOL_AVG) && defined(EXCLUDE_PADDING)
247 filter_size++;
248#endif // defined(POOL_AVG) && defined(EXCLUDE_PADDING)
Michalis Spyroue74b2012018-04-18 09:49:16 +0100249 }
250 }
251
252#if defined(POOL_AVG)
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100253 res0 = (res0 + (VEC_DATA_TYPE(ACC_DATA_TYPE, VEC_SIZE))(filter_size >> 1)) / filter_size;
254#endif // defined(POOL_AVG)
Michalis Spyroue74b2012018-04-18 09:49:16 +0100255
Giorgio Arena2d1a8352020-10-26 15:04:08 +0000256 VEC_DATA_TYPE(DATA_TYPE, VEC_SIZE)
257 out_q0 = CONVERT(res0, VEC_DATA_TYPE(DATA_TYPE, VEC_SIZE));
Pablo Telloa52e4cf2019-04-01 14:55:18 +0100258#if defined(OFFSET_IN1) && defined(OFFSET_OUT) && defined(SCALE_IN1) && defined(SCALE_OUT)
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100259 REQUANTIZE(VEC_SIZE, out_q0, OFFSET_IN1, OFFSET_OUT, SCALE_IN1, SCALE_OUT, out_q0);
Pablo Telloa52e4cf2019-04-01 14:55:18 +0100260#endif /* defined(OFFSET_IN1) && defined(OFFSET_OUT) && defined(SCALE_IN1) && defined(SCALE_OUT) */
261
Michalis Spyroue74b2012018-04-18 09:49:16 +0100262 // Store result
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100263 STORE_VECTOR_SELECT(out_q, DATA_TYPE, out_base_ptr, VEC_SIZE, VEC_SIZE_LEFTOVER, ((VEC_SIZE_LEFTOVER != 0) && get_global_id(0) == 0));
Pablo Telloa52e4cf2019-04-01 14:55:18 +0100264}
Giorgio Arena2d1a8352020-10-26 15:04:08 +0000265#endif // defined(VEC_SIZE) && defined(VEC_SIZE_LEFTOVER) && defined(SRC_WIDTH) && defined(SRC_HEIGHT) && defined(DST_CHANNELS) && defined(DST_HEIGHT) && defined(DST_BATCH_SIZE) && defined(ACC_DATA_TYPE)
Gian Marco Iodice7333e1f2020-10-08 10:25:49 +0100266#endif // defined(DATA_TYPE) && defined(INITIAL_VALUE)