/*
* Copyright (C) 2016 The Android Open Source Project
*
* 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 "request_tracker.h"
#include <gtest/gtest.h>
using testing::Test;
namespace default_camera_hal {
class RequestTrackerTest : public Test {
protected:
void SetUp() {
stream1_.max_buffers = 3;
stream2_.max_buffers = 3;
dut_.reset(new RequestTracker());
streams_ = {&stream1_, &stream2_};
camera3_stream_configuration_t config{
static_cast<uint32_t>(streams_.size()),
streams_.data(),
0,
nullptr};
dut_->SetStreamConfiguration(config);
}
std::shared_ptr<CaptureRequest> GenerateCaptureRequest(
uint32_t frame, std::vector<camera3_stream_t*> streams) {
std::shared_ptr<CaptureRequest> request =
std::make_shared<CaptureRequest>();
// Set the frame number and buffers.
request->frame_number = frame;
for (const auto stream : streams) {
// All we really care about for the buffers is which stream they're for.
camera3_stream_buffer_t buffer{stream, nullptr, 0, -1, -1};
request->output_buffers.push_back(buffer);
}
return request;
}
void AddRequest(uint32_t frame,
std::vector<camera3_stream_t*> streams,
bool expected = true) {
std::shared_ptr<CaptureRequest> request =
GenerateCaptureRequest(frame, streams);
EXPECT_EQ(dut_->CanAddRequest(*request), expected);
if (expected) {
EXPECT_FALSE(dut_->InFlight(frame));
}
EXPECT_EQ(dut_->Add(request), expected);
if (expected) {
EXPECT_TRUE(dut_->InFlight(frame));
}
}
camera3_stream_t stream1_;
camera3_stream_t stream2_;
std::vector<camera3_stream_t*> streams_;
std::shared_ptr<RequestTracker> dut_;
};
TEST_F(RequestTrackerTest, AddValid) {
uint32_t frame = 34;
EXPECT_FALSE(dut_->InFlight(frame));
AddRequest(frame, {&stream1_});
}
TEST_F(RequestTrackerTest, AddInput) {
EXPECT_TRUE(dut_->Empty());
// Add a request
uint32_t frame = 42;
std::shared_ptr<CaptureRequest> expected = GenerateCaptureRequest(frame, {});
// Set the input buffer instead of any outputs.
expected->input_buffer.reset(
new camera3_stream_buffer_t{&stream1_, nullptr, 0, -1, -1});
stream1_.max_buffers = 1;
EXPECT_TRUE(dut_->Add(expected));
EXPECT_TRUE(dut_->InFlight(frame));
// Should have added to the count of buffers for stream 1.
EXPECT_TRUE(dut_->StreamFull(&stream1_));
}
TEST_F(RequestTrackerTest, AddMultipleStreams) {
stream1_.max_buffers = 1;
stream2_.max_buffers = 1;
EXPECT_FALSE(dut_->StreamFull(&stream1_));
EXPECT_FALSE(dut_->StreamFull(&stream2_));
// Add a request using both streams.
AddRequest(99, {&stream1_, &stream2_});
// Should both have been counted.
EXPECT_TRUE(dut_->StreamFull(&stream1_));
EXPECT_TRUE(dut_->StreamFull(&stream2_));
}
TEST_F(RequestTrackerTest, AddUnconfigured) {
camera3_stream_t stream;
// Unconfigured should be considered full.
EXPECT_TRUE(dut_->StreamFull(&stream));
AddRequest(1, {&stream}, false);
}
TEST_F(RequestTrackerTest, AddPastCapacity) {
// Set the limit of stream 2 to 1.
stream2_.max_buffers = 1;
for (size_t i = 0; i < stream1_.max_buffers; ++i) {
EXPECT_FALSE(dut_->StreamFull(&stream1_));
EXPECT_FALSE(dut_->StreamFull(&stream2_));
AddRequest(i, {&stream1_});
}
// Filled up stream 1.
EXPECT_TRUE(dut_->StreamFull(&stream1_));
// Stream 2 should still not be full since nothing was added.
EXPECT_FALSE(dut_->StreamFull(&stream2_));
// Limit has been hit, can't add more.
AddRequest(stream1_.max_buffers, {&stream1_, &stream2_}, false);
EXPECT_TRUE(dut_->StreamFull(&stream1_));
// Should not have added to the count of stream 2.
EXPECT_FALSE(dut_->StreamFull(&stream2_));
}
TEST_F(RequestTrackerTest, AddDuplicate) {
uint32_t frame = 42;
AddRequest(frame, {&stream1_});
// Can't add a duplicate.
AddRequest(frame, {&stream2_}, false);
}
TEST_F(RequestTrackerTest, RemoveValid) {
EXPECT_TRUE(dut_->Empty());
// Add a request
uint32_t frame = 42;
std::shared_ptr<CaptureRequest> request =
GenerateCaptureRequest(frame, {&stream1_});
EXPECT_TRUE(dut_->Add(request));
EXPECT_TRUE(dut_->InFlight(frame));
AddRequest(frame + 1, {&stream1_});
EXPECT_FALSE(dut_->Empty());
// Remove it.
EXPECT_TRUE(dut_->Remove(request));
// Should have removed only the desired request.
EXPECT_FALSE(dut_->Empty());
}
TEST_F(RequestTrackerTest, RemoveInvalidFrame) {
EXPECT_TRUE(dut_->Empty());
// Add a request
uint32_t frame = 42;
AddRequest(frame, {&stream1_});
EXPECT_FALSE(dut_->Empty());
// Try to remove a different one.
uint32_t bad_frame = frame + 1;
std::shared_ptr<CaptureRequest> bad_request =
GenerateCaptureRequest(bad_frame, {&stream1_});
EXPECT_FALSE(dut_->InFlight(bad_frame));
EXPECT_FALSE(dut_->Remove(bad_request));
EXPECT_FALSE(dut_->Empty());
}
TEST_F(RequestTrackerTest, RemoveInvalidData) {
EXPECT_TRUE(dut_->Empty());
// Add a request
uint32_t frame = 42;
AddRequest(frame, {&stream1_});
EXPECT_FALSE(dut_->Empty());
// Try to remove a different one.
// Even though this request looks the same, that fact that it is
// a pointer to a different object means it should fail.
std::shared_ptr<CaptureRequest> bad_request =
GenerateCaptureRequest(frame, {&stream1_});
EXPECT_TRUE(dut_->InFlight(frame));
EXPECT_FALSE(dut_->Remove(bad_request));
EXPECT_FALSE(dut_->Empty());
}
TEST_F(RequestTrackerTest, RemoveNull) {
EXPECT_FALSE(dut_->Remove(nullptr));
}
TEST_F(RequestTrackerTest, ClearRequests) {
// Create some requests.
uint32_t frame1 = 42;
uint32_t frame2 = frame1 + 1;
std::shared_ptr<CaptureRequest> request1 =
GenerateCaptureRequest(frame1, {&stream1_});
std::shared_ptr<CaptureRequest> request2 =
GenerateCaptureRequest(frame2, {&stream2_});
std::set<std::shared_ptr<CaptureRequest>> expected;
expected.insert(request1);
expected.insert(request2);
// Insert them.
EXPECT_TRUE(dut_->Add(request1));
EXPECT_TRUE(dut_->Add(request2));
EXPECT_TRUE(dut_->InFlight(frame1));
EXPECT_TRUE(dut_->InFlight(frame2));
EXPECT_FALSE(dut_->Empty());
std::set<std::shared_ptr<CaptureRequest>> actual;
// Clear them out.
dut_->Clear(&actual);
EXPECT_TRUE(dut_->Empty());
EXPECT_EQ(actual, expected);
// Configuration (max values) should not have been cleared.
EXPECT_TRUE(dut_->Add(request1));
}
TEST_F(RequestTrackerTest, ClearRequestsNoResult) {
// Add some requests.
EXPECT_TRUE(dut_->Empty());
AddRequest(1, {&stream1_});
AddRequest(2, {&stream2_});
EXPECT_FALSE(dut_->Empty());
// Don't bother getting the cleared requests.
dut_->Clear();
EXPECT_TRUE(dut_->Empty());
}
TEST_F(RequestTrackerTest, ClearConfiguration) {
EXPECT_FALSE(dut_->StreamFull(&stream1_));
EXPECT_FALSE(dut_->StreamFull(&stream2_));
// Clear the configuration.
dut_->ClearStreamConfiguration();
// Both streams should be considered full now, since neither is configured.
EXPECT_TRUE(dut_->StreamFull(&stream1_));
EXPECT_TRUE(dut_->StreamFull(&stream2_));
}
} // namespace default_camera_hal