Spaces:
Running
Running
// Copyright (C) 2018-2022 Intel Corporation | |
// SPDX-License-Identifier: Apache-2.0 | |
// | |
#pragma OPENCL EXTENSION cl_khr_fp16 : enable | |
#pragma OPENCL EXTENSION cl_khr_extended_async_copies : enable | |
__kernel void quantize( | |
__global const half *restrict src_data, | |
__global const half *restrict input_low, | |
__global const half *restrict input_high, | |
__global const half *restrict output_low, | |
__global const half *restrict output_high, | |
__global half *restrict dst_data, | |
int levels, | |
int input_low_size, | |
int input_high_size, | |
int output_low_size, | |
int output_high_size, | |
int W, | |
int H) | |
{ | |
__local half local_src[15 * 1024]; | |
__local half local_dst[15 * 1024]; | |
event_t e1 = async_work_group_copy(local_src, src_data + get_group_id(2) * W * H, W * H, 0); | |
wait_group_events(1, &e1); | |
int c = get_group_id(2); | |
half h_ilow = (input_low_size == 1 ? input_low[0] : input_low[c]); | |
half h_ihigh = (input_high_size == 1 ? input_high[0] : input_high[c]); | |
half h_olow = (output_low_size == 1 ? output_low[0] : output_low[c]); | |
half h_ohigh = (output_high_size == 1 ? output_high[0] : output_high[c]); | |
half const1 = (half)( | |
!(h_ihigh - h_ilow) ? 0.0f : convert_float(levels - 1) / (convert_float(h_ihigh) - convert_float(h_ilow))); | |
half const2 = | |
(half)(!(levels - 1) ? 0.0f : (convert_float(h_ohigh) - convert_float(h_olow)) / convert_float(levels - 1)); | |
__local const half *restrict src = local_src + W * get_local_id(1); | |
__local half *restrict dst = local_dst + W * get_local_id(1); | |
for (int w = 0; w < W / 8; w++) { | |
half8 val = *((__local half8 *)src + w); | |
half8 aux = (val - (half8)h_ilow) * (half8)const1 + (half8)0.5h; | |
aux = (half8){ | |
(half)(short)(aux.s0), | |
(half)(short)(aux.s1), | |
(half)(short)(aux.s2), | |
(half)(short)(aux.s3), | |
(half)(short)(aux.s4), | |
(half)(short)(aux.s5), | |
(half)(short)(aux.s6), | |
(half)(short)(aux.s7)}; | |
aux = aux * (half8)const2 + (half8)h_olow; | |
short8 a; | |
short8 b; | |
a.s0 = (val.s0 <= h_ilow); | |
a.s1 = (val.s1 <= h_ilow); | |
a.s2 = (val.s2 <= h_ilow); | |
a.s3 = (val.s3 <= h_ilow); | |
a.s4 = (val.s4 <= h_ilow); | |
a.s5 = (val.s5 <= h_ilow); | |
a.s6 = (val.s6 <= h_ilow); | |
a.s7 = (val.s7 <= h_ilow); | |
b.s0 = (val.s0 > h_ihigh); | |
b.s1 = (val.s1 > h_ihigh); | |
b.s2 = (val.s2 > h_ihigh); | |
b.s3 = (val.s3 > h_ihigh); | |
b.s4 = (val.s4 > h_ihigh); | |
b.s5 = (val.s5 > h_ihigh); | |
b.s6 = (val.s6 > h_ihigh); | |
b.s7 = (val.s7 > h_ihigh); | |
a = ~(a - (short8)1); | |
b = ~(b - (short8)1); | |
short8 c1 = (~a & b); | |
short8 c2 = (~a & ~b); | |
short8 res = (a & as_short8((half8)h_olow)) | (c1 & as_short8((half8)h_ohigh)) | (c2 & as_short8(aux)); | |
*((__local half8 *)dst + w) = as_half8(res); | |
} | |
for (int w = W & (~0x7); w < W; w++) { | |
half val = src[w]; | |
short a = val <= h_ilow; | |
a = ~(a - 1); | |
short b = val > h_ihigh; | |
b = ~(b - 1); | |
short c1 = (~a & b); | |
short c2 = (~a & ~b); | |
short res = (a & as_short(h_olow)) | (c1 & as_short(h_ohigh)) | |
| (c2 & as_short(((half)(round((val - h_ilow) * const1) * const2) + h_olow))); | |
dst[w] = as_half(res); | |
} | |
barrier(CLK_LOCAL_MEM_FENCE); | |
event_t e2 = async_work_group_copy(dst_data + get_group_id(2) * W * H, local_dst, W * H, 0); | |
wait_group_events(1, &e2); | |
} | |