21583568a8
Test: add `USE_CAMERA_V4L2_HAL := true` to hardware/libhardware/modules/camera/3_4/Android.mk and build mmm hardware/libhardware/modules/camera/3_4 Change-Id: I464761128a09d90415ee6ac1ef1244391edb0b54 Signed-off-by: Sergii Piatakov <sergii.piatakov@globallogic.com>
159 lines
4.5 KiB
C++
159 lines
4.5 KiB
C++
/*
|
|
* Copyright 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.
|
|
*/
|
|
|
|
// #define LOG_NDEBUG 0
|
|
#define LOG_TAG "RequestTracker"
|
|
|
|
#include "request_tracker.h"
|
|
|
|
#include <cutils/log.h>
|
|
|
|
namespace default_camera_hal {
|
|
|
|
RequestTracker::RequestTracker() {}
|
|
|
|
RequestTracker::~RequestTracker() {}
|
|
|
|
void RequestTracker::SetStreamConfiguration(
|
|
const camera3_stream_configuration_t& config) {
|
|
// Clear the old configuration.
|
|
ClearStreamConfiguration();
|
|
// Add an entry to the buffer tracking map for each configured stream.
|
|
for (size_t i = 0; i < config.num_streams; ++i) {
|
|
buffers_in_flight_.emplace(config.streams[i], 0);
|
|
}
|
|
}
|
|
|
|
void RequestTracker::ClearStreamConfiguration() {
|
|
// The keys of the in flight buffer map are the configured streams.
|
|
buffers_in_flight_.clear();
|
|
}
|
|
|
|
// Helper: get the streams used by a request.
|
|
std::set<camera3_stream_t*> RequestStreams(const CaptureRequest& request) {
|
|
std::set<camera3_stream_t*> result;
|
|
if (request.input_buffer) {
|
|
result.insert(request.input_buffer->stream);
|
|
}
|
|
for (const auto& output_buffer : request.output_buffers) {
|
|
result.insert(output_buffer.stream);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
bool RequestTracker::Add(std::shared_ptr<CaptureRequest> request) {
|
|
if (!CanAddRequest(*request)) {
|
|
return false;
|
|
}
|
|
|
|
// Add to the count for each stream used.
|
|
for (const auto stream : RequestStreams(*request)) {
|
|
++buffers_in_flight_[stream];
|
|
}
|
|
|
|
// Store the request.
|
|
frames_in_flight_[request->frame_number] = request;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool RequestTracker::Remove(std::shared_ptr<CaptureRequest> request) {
|
|
if (!request) {
|
|
return false;
|
|
}
|
|
|
|
// Get the request.
|
|
const auto frame_number_request =
|
|
frames_in_flight_.find(request->frame_number);
|
|
if (frame_number_request == frames_in_flight_.end()) {
|
|
ALOGE("%s: Frame %u is not in flight.", __func__, request->frame_number);
|
|
return false;
|
|
} else if (request != frame_number_request->second) {
|
|
ALOGE(
|
|
"%s: Request for frame %u cannot be removed: "
|
|
"does not matched the stored request.",
|
|
__func__,
|
|
request->frame_number);
|
|
return false;
|
|
}
|
|
|
|
frames_in_flight_.erase(frame_number_request);
|
|
|
|
// Decrement the counts of used streams.
|
|
for (const auto stream : RequestStreams(*request)) {
|
|
--buffers_in_flight_[stream];
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void RequestTracker::Clear(
|
|
std::set<std::shared_ptr<CaptureRequest>>* requests) {
|
|
// If desired, extract all the currently in-flight requests.
|
|
if (requests) {
|
|
for (auto& frame_number_request : frames_in_flight_) {
|
|
requests->insert(frame_number_request.second);
|
|
}
|
|
}
|
|
|
|
// Clear out all tracking.
|
|
frames_in_flight_.clear();
|
|
// Maintain the configuration, but reset counts.
|
|
for (auto& stream_count : buffers_in_flight_) {
|
|
stream_count.second = 0;
|
|
}
|
|
}
|
|
|
|
bool RequestTracker::CanAddRequest(const CaptureRequest& request) const {
|
|
// Check that it's not a duplicate.
|
|
if (frames_in_flight_.count(request.frame_number) > 0) {
|
|
ALOGE("%s: Already tracking a request with frame number %d.",
|
|
__func__,
|
|
request.frame_number);
|
|
return false;
|
|
}
|
|
|
|
// Check that each stream has space
|
|
// (which implicitly checks if it is configured).
|
|
for (const auto stream : RequestStreams(request)) {
|
|
if (StreamFull(stream)) {
|
|
ALOGE("%s: Stream %p is full.", __func__, stream);
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool RequestTracker::StreamFull(const camera3_stream_t* handle) const {
|
|
const auto it = buffers_in_flight_.find(handle);
|
|
if (it == buffers_in_flight_.end()) {
|
|
// Unconfigured streams are implicitly full.
|
|
ALOGV("%s: Stream %p is not a configured stream.", __func__, handle);
|
|
return true;
|
|
} else {
|
|
return it->second >= it->first->max_buffers;
|
|
}
|
|
}
|
|
|
|
bool RequestTracker::InFlight(uint32_t frame_number) const {
|
|
return frames_in_flight_.count(frame_number) > 0;
|
|
}
|
|
|
|
bool RequestTracker::Empty() const {
|
|
return frames_in_flight_.empty();
|
|
}
|
|
|
|
} // namespace default_camera_hal
|