blob: 5eb482db5c4c00e6fa7ef07a27295afd5461cc7f [file] [log] [blame]
/* Copyright 2020 The TensorFlow 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 "tensorflow/lite/delegates/gpu/metal/kernels/winograd.h"
#import <XCTest/XCTest.h>
#include <vector>
#include "tensorflow/lite/delegates/gpu/common/operations.h"
#include "tensorflow/lite/delegates/gpu/common/shape.h"
#include "tensorflow/lite/delegates/gpu/common/status.h"
#include "tensorflow/lite/delegates/gpu/common/tasks/winograd_test_util.h"
#include "tensorflow/lite/delegates/gpu/common/tensor.h"
#include "tensorflow/lite/delegates/gpu/common/util.h"
#include "tensorflow/lite/delegates/gpu/common/winograd_util.h"
#include "tensorflow/lite/delegates/gpu/metal/kernels/test_util.h"
@interface WinogradTest : XCTestCase
@end
@implementation WinogradTest {
tflite::gpu::metal::MetalExecutionEnvironment exec_env_;
}
namespace tflite {
namespace gpu {
namespace metal {
std::vector<TensorStorageType> GetSupportedStorages() {
return {TensorStorageType::BUFFER, TensorStorageType::IMAGE_BUFFER};
}
absl::Status Winograd4x4To36Test(TestExecutionEnvironment* env) {
TensorFloat32 src_tensor;
src_tensor.shape = BHWC(1, 4, 4, 1);
src_tensor.data.resize(16);
for (int i = 0; i < 16; ++i) {
src_tensor.data[i] = sin(i);
}
TensorFloat32 dst_ref;
dst_ref.shape = BHWC(1, 36, 1, 1);
dst_ref.data.resize(36, 0.0f);
auto b_t = BtMatrixForWinograd4x4To6x6();
// Bt * Src * B
// 1: temp = Src * B
std::vector<float> temp(36, 0.0f);
for (int y = 0; y < 6; ++y) {
for (int x = 0; x < 6; ++x) {
float sum = 0.0f;
for (int i = 0; i < 6; ++i) {
if (y < 1 || y > 4 || i < 1 || i > 4) continue;
const int index = src_tensor.shape.LinearIndex({0, y - 1, i - 1, 0});
sum += src_tensor.data[index] * b_t[x * 6 + i];
}
temp[y * 6 + x] = sum;
}
}
// 2: ref = Bt * temp
for (int y = 0; y < 6; ++y) {
for (int x = 0; x < 6; ++x) {
float sum = 0.0f;
for (int i = 0; i < 6; ++i) {
sum += b_t[y * 6 + i] * temp[i * 6 + x];
}
const int index = dst_ref.shape.LinearIndex({0, y * 6 + x, 0, 0});
dst_ref.data[index] = sum;
}
}
for (auto storage : GetSupportedStorages()) {
for (auto precision : env->GetSupportedPrecisions()) {
float eps;
if (precision == CalculationsPrecision::F32) {
eps = 1e-5f * (env->GetGpuInfo().IsRoundToNearestSupported() ? 1.0f : 4.0f);
} else {
eps = 1e-2f * (env->GetGpuInfo().IsRoundToNearestSupported() ? 1.0f : 4.0f);
}
OperationDef op_def;
op_def.precision = precision;
auto data_type = DeduceDataTypeFromPrecision(precision);
op_def.src_tensors.push_back({data_type, storage, Layout::HWC});
op_def.dst_tensors.push_back({data_type, storage, Layout::HWC});
TensorFloat32 dst_tensor;
Winograd4x4To36Attributes attr;
attr.padding.prepended = tflite::gpu::HW(1, 1);
attr.padding.appended = tflite::gpu::HW(1, 1);
Winograd4x4To36 operation = CreateWinograd4x4To36(op_def, attr);
RETURN_IF_ERROR(env->ExecuteGPUOperation(
src_tensor, absl::make_unique<Winograd4x4To36>(std::move(operation)), BHWC(1, 36, 1, 1),
&dst_tensor));
RETURN_IF_ERROR(PointWiseNear(dst_ref.data, dst_tensor.data, eps));
}
}
return absl::OkStatus();
}
absl::Status Winograd4x4To36TileX6Test(TestExecutionEnvironment* env) {
TensorFloat32 src_tensor;
src_tensor.shape = BHWC(1, 4, 4, 1);
src_tensor.data.resize(16);
for (int i = 0; i < 16; ++i) {
src_tensor.data[i] = sin(i);
}
TensorFloat32 dst_ref;
dst_ref.shape = BHWC(1, 36, 1, 1);
dst_ref.data.resize(36, 0.0f);
auto b_t = BtMatrixForWinograd4x4To6x6();
// Bt * Src * B
// 1: temp = Src * B
std::vector<float> temp(36, 0.0f);
for (int y = 0; y < 6; ++y) {
for (int x = 0; x < 6; ++x) {
float sum = 0.0f;
for (int i = 0; i < 6; ++i) {
if (y < 1 || y > 4 || i < 1 || i > 4) continue;
const int index = src_tensor.shape.LinearIndex({0, y - 1, i - 1, 0});
sum += src_tensor.data[index] * b_t[x * 6 + i];
}
temp[y * 6 + x] = sum;
}
}
// 2: ref = Bt * temp
for (int y = 0; y < 6; ++y) {
for (int x = 0; x < 6; ++x) {
float sum = 0.0f;
for (int i = 0; i < 6; ++i) {
sum += b_t[y * 6 + i] * temp[i * 6 + x];
}
const int index = dst_ref.shape.LinearIndex({0, y * 6 + x, 0, 0});
dst_ref.data[index] = sum;
}
}
for (auto storage : GetSupportedStorages()) {
for (auto precision : env->GetSupportedPrecisions()) {
float eps;
if (precision == CalculationsPrecision::F32) {
eps = 1e-5f * (env->GetGpuInfo().IsRoundToNearestSupported() ? 1.0f : 4.0f);
} else {
eps = 1e-2f * (env->GetGpuInfo().IsRoundToNearestSupported() ? 1.0f : 4.0f);
}
OperationDef op_def;
op_def.precision = precision;
auto data_type = DeduceDataTypeFromPrecision(precision);
op_def.src_tensors.push_back({data_type, storage, Layout::HWC});
op_def.dst_tensors.push_back({data_type, storage, Layout::HWC});
TensorFloat32 dst_tensor;
Winograd4x4To36Attributes attr;
attr.padding.prepended = tflite::gpu::HW(1, 1);
attr.padding.appended = tflite::gpu::HW(1, 1);
Winograd4x4To36TileX6 operation = CreateWinograd4x4To36TileX6(op_def, attr);
RETURN_IF_ERROR(env->ExecuteGPUOperation(
src_tensor, absl::make_unique<Winograd4x4To36TileX6>(std::move(operation)),
BHWC(1, 36, 1, 1), &dst_tensor));
RETURN_IF_ERROR(PointWiseNear(dst_ref.data, dst_tensor.data, eps));
}
}
return absl::OkStatus();
}
absl::Status Winograd36To4x4Test(TestExecutionEnvironment* env) {
TensorFloat32 src_tensor;
src_tensor.shape = BHWC(1, 36, 1, 1);
src_tensor.data.resize(36);
for (int i = 0; i < 36; ++i) {
src_tensor.data[i] = sin(i);
}
::tflite::gpu::Tensor<Linear, DataType::FLOAT32> biases;
biases.shape = Linear(1);
biases.data.resize(biases.shape.DimensionsProduct());
for (int i = 0; i < biases.data.size(); ++i) {
biases.data[i] = 0.0f;
}
TensorFloat32 dst_ref;
dst_ref.shape = BHWC(1, 4, 4, 1);
dst_ref.data.resize(16, 0.0f);
auto a_t = AtMatrixForWinograd4x4To6x6();
tflite::gpu::metal::Winograd36To4x4Attributes attr;
attr.output_shape = dst_ref.shape;
attr.biases = biases;
// At * Src * A
// 1: temp = Src * A
std::vector<float> temp(24, 0.0f);
for (int y = 0; y < 6; ++y) {
for (int x = 0; x < 4; ++x) {
float sum = 0.0f;
for (int i = 0; i < 6; ++i) {
const int index = src_tensor.shape.LinearIndex({0, y * 6 + i, 0, 0});
sum += src_tensor.data[index] * a_t[x * 6 + i];
}
temp[y * 4 + x] = sum;
}
}
// 2: ref = At * temp
for (int y = 0; y < 4; ++y) {
for (int x = 0; x < 4; ++x) {
float sum = 0.0f;
for (int i = 0; i < 6; ++i) {
sum += a_t[y * 6 + i] * temp[i * 4 + x];
}
const int index = dst_ref.shape.LinearIndex({0, y, x, 0});
dst_ref.data[index] = sum;
}
}
for (auto storage : GetSupportedStorages()) {
for (auto precision : env->GetSupportedPrecisions()) {
float eps;
if (precision == CalculationsPrecision::F32) {
eps = 1e-5f * (env->GetGpuInfo().IsRoundToNearestSupported() ? 1.0f : 4.0f);
} else {
eps = 1e-2f * (env->GetGpuInfo().IsRoundToNearestSupported() ? 1.0f : 4.0f);
}
OperationDef op_def;
op_def.precision = precision;
auto data_type = DeduceDataTypeFromPrecision(precision);
op_def.src_tensors.push_back({data_type, storage, Layout::HWC});
op_def.dst_tensors.push_back({data_type, storage, Layout::HWC});
TensorFloat32 dst_tensor;
Winograd36To4x4 operation = CreateWinograd36To4x4(op_def, attr);
RETURN_IF_ERROR(env->ExecuteGPUOperation(
src_tensor, absl::make_unique<Winograd36To4x4>(std::move(operation)), BHWC(1, 4, 4, 1),
&dst_tensor));
RETURN_IF_ERROR(PointWiseNear(dst_ref.data, dst_tensor.data, eps));
}
}
return absl::OkStatus();
}
absl::Status Winograd36To4x4Tile4x1Test(TestExecutionEnvironment* env) {
TensorFloat32 src_tensor;
src_tensor.shape = BHWC(1, 36, 1, 1);
src_tensor.data.resize(36);
for (int i = 0; i < 36; ++i) {
src_tensor.data[i] = sin(i);
}
::tflite::gpu::Tensor<Linear, DataType::FLOAT32> biases;
biases.shape = Linear(1);
biases.data.resize(biases.shape.DimensionsProduct());
for (int i = 0; i < biases.data.size(); ++i) {
biases.data[i] = 0.0f;
}
TensorFloat32 dst_ref;
dst_ref.shape = BHWC(1, 4, 4, 1);
dst_ref.data.resize(16, 0.0f);
auto a_t = AtMatrixForWinograd4x4To6x6();
tflite::gpu::metal::Winograd36To4x4Attributes attr;
attr.output_shape = dst_ref.shape;
attr.biases = biases;
// At * Src * A
// 1: temp = Src * A
std::vector<float> temp(24, 0.0f);
for (int y = 0; y < 6; ++y) {
for (int x = 0; x < 4; ++x) {
float sum = 0.0f;
for (int i = 0; i < 6; ++i) {
const int index = src_tensor.shape.LinearIndex({0, y * 6 + i, 0, 0});
sum += src_tensor.data[index] * a_t[x * 6 + i];
}
temp[y * 4 + x] = sum;
}
}
// 2: ref = At * temp
for (int y = 0; y < 4; ++y) {
for (int x = 0; x < 4; ++x) {
float sum = 0.0f;
for (int i = 0; i < 6; ++i) {
sum += a_t[y * 6 + i] * temp[i * 4 + x];
}
const int index = dst_ref.shape.LinearIndex({0, y, x, 0});
dst_ref.data[index] = sum;
}
}
for (auto storage : GetSupportedStorages()) {
for (auto precision : env->GetSupportedPrecisions()) {
float eps;
if (precision == CalculationsPrecision::F32) {
eps = 1e-5f * (env->GetGpuInfo().IsRoundToNearestSupported() ? 1.0f : 4.0f);
} else {
eps = 1e-2f * (env->GetGpuInfo().IsRoundToNearestSupported() ? 1.0f : 4.0f);
}
OperationDef op_def;
op_def.precision = precision;
auto data_type = DeduceDataTypeFromPrecision(precision);
op_def.src_tensors.push_back({data_type, storage, Layout::HWC});
op_def.dst_tensors.push_back({data_type, storage, Layout::HWC});
TensorFloat32 dst_tensor;
Winograd36To4x4Tile4x1 operation = CreateWinograd36To4x4Tile4x1(op_def, attr);
RETURN_IF_ERROR(env->ExecuteGPUOperation(
src_tensor, absl::make_unique<Winograd36To4x4Tile4x1>(std::move(operation)),
BHWC(1, 4, 4, 1), &dst_tensor));
RETURN_IF_ERROR(PointWiseNear(dst_ref.data, dst_tensor.data, eps));
}
}
return absl::OkStatus();
}
} // namespace metal
} // namespace gpu
} // namespace tflite
- (void)testWinograd4x4To36Metal {
auto status = tflite::gpu::metal::Winograd4x4To36Test(&exec_env_);
XCTAssertTrue(status.ok(), @"%s", std::string(status.message()).c_str());
}
- (void)testWinograd4x4To36TileX6Metal {
auto status = tflite::gpu::metal::Winograd4x4To36TileX6Test(&exec_env_);
XCTAssertTrue(status.ok(), @"%s", std::string(status.message()).c_str());
}
- (void)testWinograd36To4x4Metal {
auto status = tflite::gpu::metal::Winograd36To4x4Test(&exec_env_);
XCTAssertTrue(status.ok(), @"%s", std::string(status.message()).c_str());
}
- (void)testWinograd36To4x4Tile4x1Metal {
auto status = tflite::gpu::metal::Winograd36To4x4Tile4x1Test(&exec_env_);
XCTAssertTrue(status.ok(), @"%s", std::string(status.message()).c_str());
}
- (void)testWinograd4x4To36TileX6 {
auto status = tflite::gpu::Winograd4x4To36TileX6Test(&exec_env_);
XCTAssertTrue(status.ok(), @"%s", std::string(status.message()).c_str());
}
- (void)testWinograd36To4x4Tile4x1 {
auto status = tflite::gpu::Winograd36To4x4Tile4x1Test(&exec_env_);
XCTAssertTrue(status.ok(), @"%s", std::string(status.message()).c_str());
}
@end