1 // Copyright (C) 2018-2019 Intel Corporation
2 // SPDX-License-Identifier: Apache-2.0
5 #include <gtest/gtest.h>
6 #include <gmock/gmock-spec-builders.h>
8 #include "mock_allocator.hpp"
10 #include <ie_blob_proxy.hpp>
12 using namespace ::testing;
14 using namespace InferenceEngine;
15 using namespace InferenceEngine::details;
17 #define MAKE_SHORT(l, h) h * 0x100 + l
19 class BlobProxyTests: public ::testing::Test {
21 virtual void TearDown() {
24 virtual void SetUp() {
27 shared_ptr<MockAllocator> createMockAllocator() {
28 return shared_ptr<MockAllocator>(new MockAllocator());
36 TEST_F(BlobProxyTests, convertByteBlobToFloat) {
38 float test_array[size] = {2.2f, 3.5f, 1.1f, 0.0f};
39 TBlob<uint8_t>::Ptr b(new TBlob<uint8_t>(Precision::U8, C, {size * sizeof(float)}));
41 uint8_t *sPtr = (uint8_t *) test_array;
42 uint8_t *dPtr = b->data();
43 ASSERT_EQ(b->size(), size * sizeof(float));
44 for (size_t i = 0; i < b->size(); i++) {
49 TEST_F(BlobProxyTests, shouldNotDeAllocate)
51 SizeVector v = {1, 2, 3};
52 auto allocator = createMockAllocator();
54 TBlobProxy<float> proxy(Precision::FP32, C, TBlob<float>(Precision::FP32, CHW, v, dynamic_pointer_cast<IAllocator>(allocator)), 2, {2});
56 EXPECT_EQ(((Blob&)proxy).deallocate(), false);
60 TEST_F(BlobProxyTests, canAccessProxyBlobUsingBaseMethod)
62 SizeVector v = {1, 2, 3};
63 auto allocator = createMockAllocator();
65 float data[] = {5,6,7,8,9,10};
67 EXPECT_CALL(*allocator.get(), alloc(1 * 2 * 3 * sizeof(float))).WillRepeatedly(Return((void*)1));
68 EXPECT_CALL(*allocator.get(), lock(_,LOCK_FOR_WRITE)).WillRepeatedly(Return(data));
69 EXPECT_CALL(*allocator.get(), unlock(_)).Times(1);
70 EXPECT_CALL(*allocator.get(), free(_)).Times(1);
72 TBlob<float> blob(Precision::FP32, CHW, v, dynamic_pointer_cast<IAllocator>(allocator));
75 TBlobProxy<float> proxy(Precision::FP32, C, move(blob), 2, {2});
77 auto proxyBuffer = proxy.buffer();
78 float *ptr = (float*)(void*)proxyBuffer;
79 ASSERT_EQ(ptr[2] , 9);
82 TEST_F(BlobProxyTests, canAccessProxyBlobUsingHelpers)
84 SizeVector v = {1, 2, 3};
85 auto allocator = createMockAllocator();
87 float data[] = {5,6,7, 8, 9, 10};
89 EXPECT_CALL(*allocator.get(), alloc(1 * 2 * 3 * sizeof(float))).WillRepeatedly(Return((void*)1));
90 EXPECT_CALL(*allocator.get(), lock(_,LOCK_FOR_WRITE)).WillOnce(Return(data));
91 EXPECT_CALL(*allocator.get(), lock(_,LOCK_FOR_READ)).WillOnce(Return(data));
92 EXPECT_CALL(*allocator.get(), unlock(_)).Times(2);
93 EXPECT_CALL(*allocator.get(), free(_)).Times(1);
95 TBlob<float> blob(Precision::FP32, CHW, v, dynamic_pointer_cast<IAllocator>(allocator));
98 TBlobProxy<float> proxy(Precision::FP32, C, std::move(blob), 2, {2});
100 auto proxyData = proxy.data();
101 float *ptr = (float * )&proxyData[0];
102 ASSERT_EQ(ptr[2] , 9);
104 auto readOnly = proxy.readOnly();
105 ptr = (float * )&readOnly[0];
106 ASSERT_EQ(ptr[2] , 9);
109 TEST_F(BlobProxyTests, canCreateProxyBlobFromDifferentBaseBlobType)
111 SizeVector v = {1, 2, 3};
112 auto allocator = createMockAllocator();
114 uint8_t data[] = {5, 6, 7, 8, 9, 10};
116 EXPECT_CALL(*allocator.get(), alloc(1 * 2 * 3 * sizeof(uint8_t))).WillRepeatedly(Return((void*)1));
117 EXPECT_CALL(*allocator.get(), lock(_,LOCK_FOR_READ)).WillOnce(Return(data));
118 EXPECT_CALL(*allocator.get(), unlock(_)).Times(1);
119 EXPECT_CALL(*allocator.get(), free(_)).Times(1);
121 TBlob<uint8_t > blob(Precision::U8, CHW, v, dynamic_pointer_cast<IAllocator>(allocator));
124 Blob::Ptr spBlob (&blob, [](Blob*){
128 TBlobProxy<short> proxy(Precision::I16, C, spBlob, 0, {3});
130 auto readOnly = proxy.readOnly();
131 const short * ptr = readOnly;
132 ASSERT_EQ(ptr[0] , MAKE_SHORT(data[0], data[1]));
133 ASSERT_EQ(ptr[1] , MAKE_SHORT(data[2], data[3]));
134 ASSERT_EQ(ptr[2] , MAKE_SHORT(data[4], data[5]));
137 TEST_F(BlobProxyTests, canNotCreateBlobWithOffsetOfSizeOutOfOriginal) {
138 SizeVector v = {1, 1, 3};
139 auto allocator = createMockAllocator();
141 EXPECT_CALL(*allocator.get(), alloc(1 * 1 * 3 * sizeof(float))).WillRepeatedly(Return((void*)1));
142 EXPECT_CALL(*allocator.get(), free(_)).Times(1);
144 TBlob<float> blob(Precision::FP32, CHW, v, dynamic_pointer_cast<IAllocator>(allocator));
147 Blob::Ptr spBlob (&blob, [](Blob*){
151 EXPECT_THROW(TBlobProxy<float>(Precision::FP32, C, spBlob, 0, {4}), InferenceEngineException);
152 EXPECT_THROW(TBlobProxy<float>(Precision::FP32, C, spBlob, 3, {1}), InferenceEngineException);
153 EXPECT_THROW(TBlobProxy<float>(Precision::FP32, C, spBlob, 2, {2}), InferenceEngineException);
157 TEST_F(BlobProxyTests, canAccessBothArraysAfterProxying)
159 SizeVector v = {1, 2, 4};
160 auto allocator = createMockAllocator();
162 uint8_t data[] = {5, 6, 7, 8, 9, 10, 11, 12};
164 EXPECT_CALL(*allocator.get(), alloc(1 * 2 * 4 * sizeof(uint8_t))).WillRepeatedly(Return((void*)1));
165 EXPECT_CALL(*allocator.get(), lock(_,LOCK_FOR_READ)).Times(2).WillRepeatedly(Return(data));
166 EXPECT_CALL(*allocator.get(), unlock(_)).Times(2);
167 EXPECT_CALL(*allocator.get(), free(_)).Times(1);
169 TBlob<uint8_t> blob(Precision::U8, CHW, v, dynamic_pointer_cast<IAllocator>(allocator));
172 Blob::Ptr spBlob (&blob, [](Blob*){
176 TBlobProxy<short> proxy(Precision::I16, C, spBlob, 2, {3});
178 auto readOnly = proxy.readOnly();
179 short * ptr = (short * )&readOnly[0];
180 ASSERT_EQ(ptr[0] , MAKE_SHORT(data[2], data[3]));
181 ASSERT_EQ(ptr[1] , MAKE_SHORT(data[4], data[5]));
183 auto origBuffer = blob.readOnly();
184 const uint8_t* origPtr = origBuffer;
186 ASSERT_EQ(origPtr[0] , 5);
187 ASSERT_EQ(origPtr[1] , 6);
191 TEST_F(BlobProxyTests, convertTwoByteBlobToFloat) {
193 float test_array[size] = {2.2f, 3.5f, 1.1f, 0.0f};
194 TBlob<uint16_t>::Ptr b(new TBlob<uint16_t>(Precision::U16, C, {size*sizeof(float) / sizeof(uint16_t)}));
196 uint16_t *sPtr = (uint16_t *) test_array;
197 uint16_t *dPtr = b->data();
198 ASSERT_EQ(b->byteSize(), size*sizeof(float));
199 ASSERT_EQ(b->size(), size*sizeof(float)/sizeof(uint16_t));
200 for(size_t i = 0; i < b->size(); i++) {
204 TBlobProxy<float>::Ptr proxy(new TBlobProxy<float>(Precision::FP32, C, b, sizeof(float) / sizeof(uint16_t), {size - 1}));
205 ASSERT_NEAR(3.5f, proxy->data()[0], 0.0001f);
206 ASSERT_NEAR(1.1f, proxy->data()[1], 0.0001f);
207 ASSERT_NEAR(0.0f, proxy->data()[2], 0.0001f);
208 ASSERT_EQ(size - 1, proxy->size());
209 ASSERT_EQ(size*sizeof(float) - sizeof(float), proxy->byteSize());
212 TEST_F(BlobProxyTests, throwsIfSmallProxyObjectSize) {
213 TBlob<float>::Ptr b(new TBlob<float>(Precision::FP32, C));
214 b->set({ 1.0f, 2.0f, 3.0f });
216 TBlobProxy<uint8_t> proxy(Precision::U8, C, b, 0, { b->byteSize() + 1 });
217 FAIL() << "Should have failed by now: proxy size is larger than blob size";
219 catch (InferenceEngine::details::InferenceEngineException ex) {};
222 TEST_F(BlobProxyTests, canReturnConstantData) {
223 TBlob<float>::Ptr b(new TBlob<float>(Precision::FP32, C));
224 b->set({ 1.0f, 2.0f, 3.0f });
225 TBlobProxy<uint8_t> const proxy(Precision::U8, C, b, 0, { b->byteSize() });
226 ASSERT_NE(proxy.cbuffer().as<const void*>(), nullptr);
229 TEST_F(BlobProxyTests, canIterateOverData) {
230 TBlob<uint8_t>::Ptr b(new TBlob<uint8_t >(Precision::FP32, C));
232 TBlobProxy<uint8_t> proxy(Precision::U8, C, b, 1, { 2 });
233 vector<uint8_t > u8buffer;
234 for (auto & element : proxy) {
235 u8buffer.push_back(element);
237 ASSERT_EQ(2, u8buffer.size());
238 ASSERT_EQ(2, u8buffer[0]);
239 ASSERT_EQ(3, u8buffer[1]);
242 TEST_F(BlobProxyTests, canIterateOverReadOnly) {
243 TBlob<uint8_t>::Ptr b(new TBlob<uint8_t >(Precision::FP32, C));
245 TBlobProxy<uint8_t> const proxy(Precision::U8, C, b, 1, { 2 });
246 vector<uint8_t > u8buffer;
247 for (auto element : proxy) {
248 u8buffer.push_back(element);
250 ASSERT_EQ(2, u8buffer.size());
251 ASSERT_EQ(2, u8buffer[0]);
252 ASSERT_EQ(3, u8buffer[1]);