/****************************************************************************** * * Copyright (C) 2016 Google, Inc. * * 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 <gmock/gmock.h> #include <gtest/gtest.h> #include <base/logging.h> #include "osi/include/leaky_bonded_queue.h" namespace testing { using system_bt_osi::LeakyBondedQueue; #define ITEM_EQ(a, b) \ do { \ EXPECT_EQ(a, b); \ EXPECT_EQ(a->index, b->index); \ } while (0) class Item { public: Item(int i) { index = i; } virtual ~Item() {} int index; }; class MockItem : public Item { public: MockItem(int i) : Item(i) {} ~MockItem() { Destruct(); } MOCK_METHOD0(Destruct, void()); }; TEST(LeakyBondedQueueTest, TestEnqueueDequeue) { MockItem* item1 = new MockItem(1); MockItem* item2 = new MockItem(2); MockItem* item3 = new MockItem(3); MockItem* item4 = new MockItem(4); LeakyBondedQueue<MockItem>* queue = new LeakyBondedQueue<MockItem>(3); EXPECT_EQ(queue->Capacity(), static_cast<size_t>(3)); EXPECT_EQ(queue->Length(), static_cast<size_t>(0)); queue->Enqueue(item1); EXPECT_EQ(queue->Length(), static_cast<size_t>(1)); queue->Enqueue(item2); EXPECT_EQ(queue->Length(), static_cast<size_t>(2)); queue->Enqueue(item3); EXPECT_EQ(queue->Length(), static_cast<size_t>(3)); EXPECT_CALL(*item1, Destruct()).Times(1); queue->Enqueue(item4); EXPECT_EQ(queue->Length(), static_cast<size_t>(3)); MockItem* item2_2 = queue->Dequeue(); MockItem* item3_3 = queue->Dequeue(); MockItem* item4_4 = queue->Dequeue(); EXPECT_THAT(item2_2, NotNull()); ITEM_EQ(item2_2, item2); EXPECT_THAT(item3_3, NotNull()); ITEM_EQ(item3_3, item3); EXPECT_THAT(item4_4, NotNull()); ITEM_EQ(item4_4, item4); LOG(INFO) << "All done release items"; EXPECT_CALL(*item2_2, Destruct()).Times(1); delete item2_2; EXPECT_CALL(*item3_3, Destruct()).Times(1); delete item3_3; EXPECT_CALL(*item4_4, Destruct()).Times(1); delete item4_4; delete queue; } TEST(LeakyBondedQueueTest, TestEnqueueDequeue2) { MockItem* item1 = new MockItem(1); MockItem* item2 = new MockItem(2); MockItem* item3 = new MockItem(3); MockItem* item4 = new MockItem(4); LeakyBondedQueue<MockItem>* queue = new LeakyBondedQueue<MockItem>(2); EXPECT_EQ(queue->Capacity(), static_cast<size_t>(2)); EXPECT_EQ(queue->Length(), static_cast<size_t>(0)); queue->Enqueue(item1); EXPECT_EQ(queue->Length(), static_cast<size_t>(1)); MockItem* item1_1 = queue->Dequeue(); ITEM_EQ(item1, item1_1); EXPECT_EQ(queue->Length(), static_cast<size_t>(0)); queue->Enqueue(item2); EXPECT_EQ(queue->Length(), static_cast<size_t>(1)); queue->Enqueue(item3); EXPECT_EQ(queue->Length(), static_cast<size_t>(2)); EXPECT_CALL(*item2, Destruct()).Times(1); queue->Enqueue(item4); EXPECT_EQ(queue->Length(), static_cast<size_t>(2)); EXPECT_CALL(*item3, Destruct()).Times(1); queue->Enqueue(item1); EXPECT_EQ(queue->Length(), static_cast<size_t>(2)); MockItem* item4_4_4 = queue->Dequeue(); MockItem* item1_1_1 = queue->Dequeue(); ITEM_EQ(item4_4_4, item4); ITEM_EQ(item1_1_1, item1); EXPECT_CALL(*item1_1_1, Destruct()).Times(1); delete item1_1_1; EXPECT_CALL(*item4_4_4, Destruct()).Times(1); delete item4_4_4; delete queue; } TEST(LeakyBondedQueueTest, TestEnqueuePop) { MockItem* item1 = new MockItem(1); MockItem* item2 = new MockItem(2); MockItem* item3 = new MockItem(3); MockItem* item4 = new MockItem(4); LeakyBondedQueue<MockItem>* queue = new LeakyBondedQueue<MockItem>(2); EXPECT_EQ(queue->Capacity(), static_cast<size_t>(2)); EXPECT_EQ(queue->Length(), static_cast<size_t>(0)); queue->Enqueue(item1); EXPECT_EQ(queue->Length(), static_cast<size_t>(1)); MockItem* item1_1 = queue->Dequeue(); ITEM_EQ(item1, item1_1); EXPECT_EQ(queue->Length(), static_cast<size_t>(0)); queue->Enqueue(item2); EXPECT_EQ(queue->Length(), static_cast<size_t>(1)); queue->Enqueue(item3); EXPECT_EQ(queue->Length(), static_cast<size_t>(2)); MockItem* item2_2 = queue->EnqueueWithPop(item4); EXPECT_THAT(item2_2, NotNull()); ITEM_EQ(item2_2, item2); EXPECT_CALL(*item2, Destruct()).Times(1); delete item2_2; EXPECT_EQ(queue->Length(), static_cast<size_t>(2)); MockItem* item3_3 = queue->EnqueueWithPop(item1); EXPECT_THAT(item3_3, NotNull()); ITEM_EQ(item3_3, item3); EXPECT_CALL(*item3, Destruct()).Times(1); delete item3_3; EXPECT_EQ(queue->Length(), static_cast<size_t>(2)); MockItem* item4_4_4 = queue->Dequeue(); MockItem* item1_1_1 = queue->Dequeue(); ITEM_EQ(item4_4_4, item4); ITEM_EQ(item1_1_1, item1); EXPECT_CALL(*item1_1_1, Destruct()).Times(1); delete item1_1_1; EXPECT_CALL(*item4_4_4, Destruct()).Times(1); delete item4_4_4; delete queue; } TEST(LeakyBondedQueueTest, TestQueueClear) { MockItem* item1 = new MockItem(1); MockItem* item2 = new MockItem(2); MockItem* item3 = new MockItem(3); MockItem* item4 = new MockItem(4); LeakyBondedQueue<MockItem>* queue = new LeakyBondedQueue<MockItem>(2); EXPECT_EQ(queue->Capacity(), static_cast<size_t>(2)); EXPECT_EQ(queue->Length(), static_cast<size_t>(0)); queue->Enqueue(item1); EXPECT_EQ(queue->Length(), static_cast<size_t>(1)); MockItem* item1_1 = queue->Dequeue(); ITEM_EQ(item1, item1_1); EXPECT_EQ(queue->Length(), static_cast<size_t>(0)); queue->Enqueue(item2); EXPECT_EQ(queue->Length(), static_cast<size_t>(1)); queue->Enqueue(item3); EXPECT_EQ(queue->Length(), static_cast<size_t>(2)); EXPECT_CALL(*item2, Destruct()).Times(1); queue->Enqueue(item4); EXPECT_EQ(queue->Length(), static_cast<size_t>(2)); EXPECT_CALL(*item3, Destruct()).Times(1); queue->Enqueue(item1); EXPECT_EQ(queue->Length(), static_cast<size_t>(2)); EXPECT_CALL(*item1, Destruct()).Times(1); EXPECT_CALL(*item4, Destruct()).Times(1); queue->Clear(); delete queue; } TEST(LeakyBondedQueueTest, TestQueueFree) { MockItem* item1 = new MockItem(1); MockItem* item2 = new MockItem(2); MockItem* item3 = new MockItem(3); MockItem* item4 = new MockItem(4); LeakyBondedQueue<MockItem>* queue = new LeakyBondedQueue<MockItem>(2); EXPECT_EQ(queue->Capacity(), static_cast<size_t>(2)); EXPECT_EQ(queue->Length(), static_cast<size_t>(0)); queue->Enqueue(item1); EXPECT_EQ(queue->Length(), static_cast<size_t>(1)); MockItem* item1_1 = queue->Dequeue(); ITEM_EQ(item1, item1_1); EXPECT_EQ(queue->Length(), static_cast<size_t>(0)); queue->Enqueue(item2); EXPECT_EQ(queue->Length(), static_cast<size_t>(1)); queue->Enqueue(item3); EXPECT_EQ(queue->Length(), static_cast<size_t>(2)); EXPECT_CALL(*item2, Destruct()).Times(1); queue->Enqueue(item4); EXPECT_EQ(queue->Length(), static_cast<size_t>(2)); EXPECT_CALL(*item3, Destruct()).Times(1); queue->Enqueue(item1); EXPECT_EQ(queue->Length(), static_cast<size_t>(2)); EXPECT_CALL(*item1, Destruct()).Times(1); EXPECT_CALL(*item4, Destruct()).Times(1); delete queue; } TEST(LeakyBondedQueueTest, TestPushNull) { MockItem* item1 = nullptr; LeakyBondedQueue<MockItem>* queue = new LeakyBondedQueue<MockItem>(2); queue->Enqueue(item1); MockItem* item1_1 = queue->Dequeue(); EXPECT_THAT(item1_1, IsNull()); } TEST(LeakyBondedQueueTest, TestPushNullOverflowQueue) { MockItem* item1 = nullptr; MockItem* item2 = nullptr; LeakyBondedQueue<MockItem>* queue = new LeakyBondedQueue<MockItem>(1); queue->Enqueue(item1); queue->Enqueue(item2); MockItem* item2_2 = queue->Dequeue(); EXPECT_THAT(item2_2, IsNull()); } TEST(LeakyBondedQueueTest, TestPushNullDeleteQueue) { MockItem* item1 = nullptr; MockItem* item2 = nullptr; LeakyBondedQueue<MockItem>* queue = new LeakyBondedQueue<MockItem>(2); queue->Enqueue(item1); queue->Enqueue(item2); delete queue; } }