/* Copyright (c) 2018 PaddlePaddle Authors. All Rights Reserved. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ #include "framework/cl/cl_image_converter.h" namespace paddle_mobile { namespace framework { const DDim &CLImageConverterDefault::InitImageDimInfoWith( const DDim &tensor_dim) { size_t new_dims[] = {1, 1, 1, 1}; for (int j = 0; j < tensor_dim.size(); ++j) { new_dims[4 - tensor_dim.size() + j] = tensor_dim[j]; } size_t N, C, H, W; N = new_dims[0]; C = new_dims[1]; H = new_dims[2]; W = new_dims[3]; size_t width = W * ((C + 3) / 4); size_t height = H * N; return make_ddim({width, height}); } void CLImageConverterDefault::NCHWToImage(float *nchw, half_t *image, const DDim &tensor_dim) { size_t new_dims[] = {1, 1, 1, 1}; for (int j = 0; j < tensor_dim.size(); ++j) { new_dims[4 - tensor_dim.size() + j] = tensor_dim[j]; } size_t N, C, H, W; N = new_dims[0]; C = new_dims[1]; H = new_dims[2]; W = new_dims[3]; DDim in_image_dim = InitImageDimInfoWith(tensor_dim); DLOG << " tensor dim " << tensor_dim; DLOG << " image dim " << in_image_dim; size_t width = in_image_dim[0]; size_t height = in_image_dim[1]; int w_block = width / W; float *p = nchw; size_t i0 = 0; for (int n = 0; n < N; n++) { for (int c = 0; c < w_block * 4; c++) { size_t i1 = i0 + (c / 4) * W; for (int h = 0; h < H; h++) { size_t i2 = (i1 << 2) + c % 4; for (int w = 0; w < W; w++) { if (c < C) { // int x = (n * width * H + h * width + (c / 4) * W + w) * 4 + // (c % 4); image[i2] = Float2Half(*p); i2 += 4; p++; } else { image[i2] = 0.0; i2 += 4; } } i1 += width; } } i0 += width * H; } } void CLImageConverterDefault::ImageToNCHW(half_t *image, float *tensor, const DDim &image_dim, const DDim &tensor_dim) { size_t new_dims[] = {1, 1, 1, 1}; for (int j = 0; j < tensor_dim.size(); ++j) { new_dims[4 - tensor_dim.size() + j] = tensor_dim[j]; } size_t N, C, H, W; N = new_dims[0]; C = new_dims[1]; H = new_dims[2]; W = new_dims[3]; int width = image_dim[0]; int height = image_dim[0]; float *p = tensor; size_t i0 = 0; for (int n = 0; n < N; n++) { for (int c = 0; c < C; c++) { size_t i1 = i0 + (c / 4) * W; for (int h = 0; h < H; h++) { size_t i2 = (i1 << 2) + c % 4; for (int w = 0; w < W; w++) { *p = Half2Float(image[i2]); i2 += 4; p++; } i1 += width; } } i0 += width * H; } } const DDim &CLImageConverterFolder::InitImageDimInfoWith( const DDim &tensor_dim) { if (tensor_dim.size() <= 2) { int tdim[2] = {1, 1}; if (tensor_dim.size() == 1) { tdim[1] = tensor_dim[0]; } else { tdim[0] = tensor_dim[0]; tdim[1] = tensor_dim[1]; } int width = (tdim[1] + 3) / 4; int height = tdim[0]; width_of_one_block_ = width; height_of_one_block_ = height; c_block_ = 1; return make_ddim({width, height}); } else { size_t new_dims[] = {1, 1, 1, 1}; for (int j = 0; j < tensor_dim.size(); ++j) { new_dims[4 - tensor_dim.size() + j] = tensor_dim[j]; } size_t N, C, H, W; N = new_dims[0]; C = new_dims[1]; H = new_dims[2]; W = new_dims[3]; size_t width = W * ((C + 3) / 4); size_t height = H * N; width_of_one_block_ = W; height_of_one_block_ = H; c_block_ = width / W; return make_ddim({width, height}); } } void CLImageConverterFolder::NCHWToImage(float *tensor, half_t *image, const DDim &tensor_dim) { PADDLE_MOBILE_ENFORCE(tensor_dim.size() <= 4 && tensor_dim.size() > 0, "tensor dim is not support "); if (tensor_dim.size() > 2) { CLImageConverterDefault default_converter; default_converter.NCHWToImage(tensor, image, tensor_dim); } else { int tdim[2] = {1, 1}; if (tensor_dim.size() == 1) { tdim[1] = tensor_dim[0]; } else { tdim[0] = tensor_dim[0]; tdim[1] = tensor_dim[1]; } DDim image_dim = InitImageDimInfoWith(tensor_dim); int width = image_dim[0]; for (int h = 0; h < tdim[0]; h++) { for (int w = 0; w < tdim[1]; w++) { image[(h * width + w / 4) * 4 + (w % 4)] = Float2Half(tensor[h * tdim[1] + w]); } } } } void CLImageConverterFolder::ImageToNCHW(half_t *image, float *tensor, const DDim &image_dim, const DDim &tensor_dim) { if (tensor_dim.size() > 2) { CLImageConverterDefault default_converter; default_converter.ImageToNCHW(image, tensor, image_dim, tensor_dim); } else { int width = image_dim[0]; int height = image_dim[1]; int H, W; if (tensor_dim.size() == 2) { H = tensor_dim[0]; W = tensor_dim[1]; } else if (tensor_dim.size() == 1) { H = 1; W = tensor_dim[0]; } float *p = tensor; for (int h = 0; h < H; h++) { for (int w = 0; w < W; w++) { p[h * W + w] = Half2Float(image[(h * width + w / 4) * 4 + (w % 4)]); } } } } const DDim &CLImageConverterNWBlock::InitImageDimInfoWith( const DDim &tensor_dim) { PADDLE_MOBILE_ENFORCE(tensor_dim.size() == 4, " tensor dim is not 4"); size_t N, C, H, W; N = tensor_dim[0]; C = tensor_dim[1]; H = tensor_dim[2]; W = tensor_dim[3]; size_t width = W * ((N + 3) / 4); size_t height = C * H; return make_ddim({width, height}); } void CLImageConverterNWBlock::NCHWToImage(float *tensor, half_t *image, const DDim &tensor_dim) { PADDLE_MOBILE_ENFORCE(tensor_dim.size() == 4, " tensor dim is not 4"); auto image_dim = InitImageDimInfoWith(tensor_dim); float *p = tensor; int N = tensor_dim[0]; int C = tensor_dim[1]; int H = tensor_dim[2]; int W = tensor_dim[3]; int width = image_dim[0]; int height = image_dim[1]; int block = image_dim[0] / tensor_dim[3]; for (int n = 0; n < block * 4; n++) { for (int c = 0; c < C; c++) { for (int h = 0; h < H; ++h) { for (int w = 0; w < W; ++w) { int index = 4 * c * (width * H) + 4 * h * width + 4 * W * (n / 4) + w * 4 + n % 4; if (n < N) { image[index] = Float2Half(*p); p++; } else { image[index] = 0.0; } if (index >= (width * height * 4)) { DLOG << " index out of range "; } } } } } DLOG << " init done"; } void CLImageConverterNWBlock::ImageToNCHW(half_t *image, float *tensor, const DDim &image_dim, const DDim &tensor_dim) { PADDLE_MOBILE_ENFORCE(tensor_dim.size() == 4, " tensor dim is not 4"); float *p = tensor; int N = tensor_dim[0]; int C = tensor_dim[1]; int H = tensor_dim[2]; int W = tensor_dim[3]; int width = image_dim[0]; int height = image_dim[1]; int block = image_dim[0] / tensor_dim[3]; for (int n = 0; n < N; n++) { for (int c = 0; c < C; c++) { for (int h = 0; h < H; ++h) { for (int w = 0; w < W; ++w) { int index = 4 * c * (width * H) + 4 * h * width + 4 * W * (n / 4) + w * 4 + n % 4; *p = Half2Float(image[index]); p++; if (index >= (width * height * 4)) { DLOG << " index out of range "; } } } } } DLOG << " init done"; } const DDim &CLImageConverterDWBlock::InitImageDimInfoWith( const DDim &tensor_dim) { PADDLE_MOBILE_ENFORCE(tensor_dim.size() == 4, " tensor dim is not 4"); size_t N, C, H, W; N = tensor_dim[0]; C = tensor_dim[1]; H = tensor_dim[2]; W = tensor_dim[3]; size_t width = W * ((N + 3) / 4); size_t height = C * H; return make_ddim({width, height}); } void CLImageConverterDWBlock::NCHWToImage(float *tensor, half_t *image, const DDim &tensor_dim) { size_t new_dims[] = {1, 1, 1, 1}; for (int j = 0; j < tensor_dim.size(); ++j) { new_dims[4 - tensor_dim.size() + j] = tensor_dim[j]; } size_t N, C, H, W; N = new_dims[1]; C = new_dims[0]; H = new_dims[2]; W = new_dims[3]; DDim in_image_dim = InitImageDimInfoWith(tensor_dim); DLOG << " tensor dim " << tensor_dim; DLOG << " image dim " << in_image_dim; size_t width = in_image_dim[0]; size_t height = in_image_dim[1]; int w_block = width / W; float *p = tensor; size_t i0 = 0; for (int n = 0; n < N; n++) { for (int c = 0; c < w_block * 4; c++) { size_t i1 = i0 + (c / 4) * W; for (int h = 0; h < H; h++) { size_t i2 = (i1 << 2) + c % 4; for (int w = 0; w < W; w++) { if (c < C) { // int x = (n * width * H + h * width + (c / 4) * W + w) * 4 + // (c % 4); image[i2] = Float2Half(*p); i2 += 4; p++; } else { image[i2] = 0.0; i2 += 4; } } i1 += width; } } i0 += width * H; } } void CLImageConverterDWBlock::ImageToNCHW(half_t *image, float *tensor, const DDim &image_dim, const DDim &tensor_dim) { PADDLE_MOBILE_ENFORCE(tensor_dim.size() == 4, " tensor dim is not 4"); float *p = tensor; int N = tensor_dim[1]; int C = tensor_dim[0]; int H = tensor_dim[2]; int W = tensor_dim[3]; int width = image_dim[0]; int height = image_dim[0]; size_t i0 = 0; for (int n = 0; n < N; n++) { for (int c = 0; c < C; c++) { size_t i1 = i0 + (c / 4) * W; for (int h = 0; h < H; h++) { size_t i2 = (i1 << 2) + c % 4; for (int w = 0; w < W; w++) { *p = Half2Float(image[i2]); i2 += 4; p++; } i1 += width; } } i0 += width * H; } } const DDim &CLImageConverterNormal::InitImageDimInfoWith( const DDim &tensor_dim) { size_t new_dims[] = {1, 1, 1, 1}; for (int j = 0; j < tensor_dim.size(); ++j) { new_dims[4 - tensor_dim.size() + j] = tensor_dim[j]; } size_t N, C, H, W; N = new_dims[0]; C = new_dims[1]; H = new_dims[2]; W = new_dims[3]; size_t width = W * ((C + 3) / 4); size_t height = H * N; width_of_one_block_ = W; height_of_one_block_ = H; c_block_ = width / W; return make_ddim({width, height}); } void CLImageConverterNormal::NCHWToImage(float *tensor, half_t *image, const DDim &tensor_dim) { PADDLE_MOBILE_ENFORCE(tensor_dim.size() <= 4 && tensor_dim.size() > 0, "tensor dim is not support "); CLImageConverterDefault default_converter; default_converter.NCHWToImage(tensor, image, tensor_dim); } void CLImageConverterNormal::ImageToNCHW(half_t *image, float *tensor, const DDim &image_dim, const DDim &tensor_dim) { CLImageConverterDefault default_converter; default_converter.ImageToNCHW(image, tensor, image_dim, tensor_dim); } } // namespace framework } // namespace paddle_mobile