/* * 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