610 lines
20 KiB
C++
610 lines
20 KiB
C++
/*
|
|
* Copyright (C) 2019 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 "HalCamera.h"
|
|
|
|
#include "Enumerator.h"
|
|
#include "VirtualCamera.h"
|
|
|
|
#include <android-base/file.h>
|
|
#include <android-base/logging.h>
|
|
#include <android-base/strings.h>
|
|
|
|
using ::android::hardware::Return;
|
|
using ::android::hardware::Void;
|
|
|
|
namespace android::automotive::evs::V1_1::implementation {
|
|
|
|
// TODO(changyeon):
|
|
// We need to hook up death monitoring to detect stream death so we can attempt a reconnect
|
|
|
|
using ::android::base::StringAppendF;
|
|
using ::android::base::WriteStringToFd;
|
|
|
|
HalCamera::~HalCamera() {
|
|
// Reports the usage statistics before the destruction
|
|
// EvsUsageStatsReported atom is defined in
|
|
// frameworks/proto_logging/stats/atoms.proto
|
|
mUsageStats->writeStats();
|
|
}
|
|
|
|
sp<VirtualCamera> HalCamera::makeVirtualCamera() {
|
|
// Create the client camera interface object
|
|
std::vector<sp<HalCamera>> sourceCameras;
|
|
sourceCameras.reserve(1);
|
|
sourceCameras.emplace_back(this);
|
|
sp<VirtualCamera> client = new VirtualCamera(sourceCameras);
|
|
if (client == nullptr) {
|
|
LOG(ERROR) << "Failed to create client camera object";
|
|
return nullptr;
|
|
}
|
|
|
|
if (!ownVirtualCamera(client)) {
|
|
LOG(ERROR) << "Failed to own a client camera object";
|
|
client = nullptr;
|
|
}
|
|
|
|
return client;
|
|
}
|
|
|
|
bool HalCamera::ownVirtualCamera(sp<VirtualCamera> virtualCamera) {
|
|
if (virtualCamera == nullptr) {
|
|
LOG(ERROR) << "Failed to create virtualCamera camera object";
|
|
return false;
|
|
}
|
|
|
|
// Make sure we have enough buffers available for all our clients
|
|
if (!changeFramesInFlight(virtualCamera->getAllowedBuffers())) {
|
|
// Gah! We couldn't get enough buffers, so we can't support this virtualCamera
|
|
// Null the pointer, dropping our reference, thus destroying the virtualCamera object
|
|
return false;
|
|
}
|
|
|
|
// Add this virtualCamera to our ownership list via weak pointer
|
|
mClients.emplace_back(virtualCamera);
|
|
|
|
// Update statistics
|
|
mUsageStats->updateNumClients(mClients.size());
|
|
|
|
return true;
|
|
}
|
|
|
|
void HalCamera::disownVirtualCamera(sp<VirtualCamera> virtualCamera) {
|
|
// Ignore calls with null pointers
|
|
if (virtualCamera == nullptr) {
|
|
LOG(WARNING) << "Ignoring disownVirtualCamera call with null pointer";
|
|
return;
|
|
}
|
|
|
|
// Remove the virtual camera from our client list
|
|
const auto clientCount = mClients.size();
|
|
mClients.remove(virtualCamera);
|
|
if (clientCount != mClients.size() + 1) {
|
|
LOG(WARNING) << "Couldn't find camera in our client list to remove it; "
|
|
<< "this client may be removed already.";
|
|
}
|
|
|
|
// Recompute the number of buffers required with the target camera removed from the list
|
|
if (!changeFramesInFlight(0)) {
|
|
LOG(ERROR) << "Error when trying to reduce the in flight buffer count";
|
|
}
|
|
|
|
// Update statistics
|
|
mUsageStats->updateNumClients(mClients.size());
|
|
}
|
|
|
|
void HalCamera::disownVirtualCamera(const VirtualCamera* clientToDisown) {
|
|
// Ignore calls with null pointers
|
|
if (clientToDisown == nullptr) {
|
|
LOG(WARNING) << "Ignoring disownVirtualCamera call with null pointer";
|
|
return;
|
|
}
|
|
|
|
// Remove the virtual camera from our client list
|
|
const auto clientCount = mClients.size();
|
|
mClients.remove_if(
|
|
[&clientToDisown](wp<VirtualCamera>& client) { return client == clientToDisown; });
|
|
if (clientCount == mClients.size()) {
|
|
LOG(WARNING) << "Couldn't find camera in our client list to remove it; "
|
|
<< "this client may be removed already.";
|
|
}
|
|
|
|
// Recompute the number of buffers required with the target camera removed from the list
|
|
if (!changeFramesInFlight(0)) {
|
|
LOG(ERROR) << "Error when trying to reduce the in flight buffer count";
|
|
}
|
|
|
|
// Update statistics
|
|
mUsageStats->updateNumClients(mClients.size());
|
|
}
|
|
|
|
bool HalCamera::changeFramesInFlight(int delta) {
|
|
// Walk all our clients and count their currently required frames
|
|
unsigned bufferCount = 0;
|
|
for (auto&& client : mClients) {
|
|
sp<VirtualCamera> virtCam = client.promote();
|
|
if (virtCam != nullptr) {
|
|
bufferCount += virtCam->getAllowedBuffers();
|
|
}
|
|
}
|
|
|
|
// Add the requested delta
|
|
bufferCount += delta;
|
|
|
|
// Never drop below 1 buffer -- even if all client cameras get closed
|
|
if (bufferCount < 1) {
|
|
bufferCount = 1;
|
|
}
|
|
|
|
// Ask the hardware for the resulting buffer count
|
|
Return<EvsResult> result = mHwCamera->setMaxFramesInFlight(bufferCount);
|
|
bool success = (result.isOk() && result == EvsResult::OK);
|
|
|
|
// Update the size of our array of outstanding frame records
|
|
if (success) {
|
|
std::vector<FrameRecord> newRecords;
|
|
newRecords.reserve(bufferCount);
|
|
|
|
// Copy and compact the old records that are still active
|
|
for (const auto& rec : mFrames) {
|
|
if (rec.refCount > 0) {
|
|
newRecords.emplace_back(rec);
|
|
}
|
|
}
|
|
if (newRecords.size() > (unsigned)bufferCount) {
|
|
LOG(WARNING) << "We found more frames in use than requested.";
|
|
}
|
|
|
|
mFrames.swap(newRecords);
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
bool HalCamera::changeFramesInFlight(const hidl_vec<BufferDesc_1_1>& buffers, int* delta) {
|
|
// Return immediately if a list is empty.
|
|
if (buffers.size() < 1) {
|
|
LOG(DEBUG) << "No external buffers to add.";
|
|
return true;
|
|
}
|
|
|
|
// Walk all our clients and count their currently required frames
|
|
auto bufferCount = 0;
|
|
for (auto&& client : mClients) {
|
|
sp<VirtualCamera> virtCam = client.promote();
|
|
if (virtCam != nullptr) {
|
|
bufferCount += virtCam->getAllowedBuffers();
|
|
}
|
|
}
|
|
|
|
EvsResult status = EvsResult::OK;
|
|
// Ask the hardware for the resulting buffer count
|
|
mHwCamera->importExternalBuffers(buffers, [&](auto result, auto added) {
|
|
status = result;
|
|
*delta = added;
|
|
});
|
|
if (status != EvsResult::OK) {
|
|
LOG(ERROR) << "Failed to add external capture buffers.";
|
|
return false;
|
|
}
|
|
|
|
bufferCount += *delta;
|
|
|
|
// Update the size of our array of outstanding frame records
|
|
std::vector<FrameRecord> newRecords;
|
|
newRecords.reserve(bufferCount);
|
|
|
|
// Copy and compact the old records that are still active
|
|
for (const auto& rec : mFrames) {
|
|
if (rec.refCount > 0) {
|
|
newRecords.emplace_back(rec);
|
|
}
|
|
}
|
|
|
|
if (newRecords.size() > (unsigned)bufferCount) {
|
|
LOG(WARNING) << "We found more frames in use than requested.";
|
|
}
|
|
|
|
mFrames.swap(newRecords);
|
|
|
|
return true;
|
|
}
|
|
|
|
void HalCamera::requestNewFrame(sp<VirtualCamera> client, const int64_t lastTimestamp) {
|
|
FrameRequest req;
|
|
req.client = client;
|
|
req.timestamp = lastTimestamp;
|
|
|
|
std::lock_guard<std::mutex> lock(mFrameMutex);
|
|
mNextRequests->push_back(req);
|
|
}
|
|
|
|
Return<EvsResult> HalCamera::clientStreamStarting() {
|
|
Return<EvsResult> result = EvsResult::OK;
|
|
|
|
if (mStreamState == STOPPED) {
|
|
mStreamState = RUNNING;
|
|
result = mHwCamera->startVideoStream(this);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
void HalCamera::cancelCaptureRequestFromClientLocked(std::deque<struct FrameRequest>* requests,
|
|
const VirtualCamera* client) {
|
|
auto it = requests->begin();
|
|
while (it != requests->end()) {
|
|
if (it->client == client) {
|
|
requests->erase(it);
|
|
return;
|
|
}
|
|
++it;
|
|
}
|
|
}
|
|
|
|
void HalCamera::clientStreamEnding(const VirtualCamera* client) {
|
|
{
|
|
std::lock_guard<std::mutex> lock(mFrameMutex);
|
|
cancelCaptureRequestFromClientLocked(mNextRequests, client);
|
|
cancelCaptureRequestFromClientLocked(mCurrentRequests, client);
|
|
}
|
|
|
|
// Do we still have a running client?
|
|
bool stillRunning = false;
|
|
for (auto&& client : mClients) {
|
|
sp<VirtualCamera> virtCam = client.promote();
|
|
if (virtCam != nullptr) {
|
|
stillRunning |= virtCam->isStreaming();
|
|
}
|
|
}
|
|
|
|
// If not, then stop the hardware stream
|
|
if (!stillRunning) {
|
|
mStreamState = STOPPING;
|
|
mHwCamera->stopVideoStream();
|
|
}
|
|
}
|
|
|
|
Return<void> HalCamera::doneWithFrame(const BufferDesc_1_0& buffer) {
|
|
// Find this frame in our list of outstanding frames
|
|
unsigned i;
|
|
for (i = 0; i < mFrames.size(); i++) {
|
|
if (mFrames[i].frameId == buffer.bufferId) {
|
|
break;
|
|
}
|
|
}
|
|
if (i == mFrames.size()) {
|
|
LOG(ERROR) << "We got a frame back with an ID we don't recognize!";
|
|
} else {
|
|
// Are there still clients using this buffer?
|
|
mFrames[i].refCount--;
|
|
if (mFrames[i].refCount <= 0) {
|
|
// Since all our clients are done with this buffer, return it to the device layer
|
|
mHwCamera->doneWithFrame(buffer);
|
|
|
|
// Counts a returned buffer
|
|
mUsageStats->framesReturned();
|
|
}
|
|
}
|
|
|
|
return Void();
|
|
}
|
|
|
|
Return<void> HalCamera::doneWithFrame(const BufferDesc_1_1& buffer) {
|
|
// Find this frame in our list of outstanding frames
|
|
unsigned i;
|
|
for (i = 0; i < mFrames.size(); i++) {
|
|
if (mFrames[i].frameId == buffer.bufferId) {
|
|
break;
|
|
}
|
|
}
|
|
if (i == mFrames.size()) {
|
|
LOG(ERROR) << "We got a frame back with an ID we don't recognize!";
|
|
} else {
|
|
// Are there still clients using this buffer?
|
|
mFrames[i].refCount--;
|
|
if (mFrames[i].refCount <= 0) {
|
|
// Since all our clients are done with this buffer, return it to the device layer
|
|
hardware::hidl_vec<BufferDesc_1_1> returnedBuffers;
|
|
returnedBuffers.resize(1);
|
|
returnedBuffers[0] = buffer;
|
|
mHwCamera->doneWithFrame_1_1(returnedBuffers);
|
|
|
|
// Counts a returned buffer
|
|
mUsageStats->framesReturned(returnedBuffers);
|
|
}
|
|
}
|
|
|
|
return Void();
|
|
}
|
|
|
|
// Methods from ::android::hardware::automotive::evs::V1_0::IEvsCameraStream follow.
|
|
Return<void> HalCamera::deliverFrame(const BufferDesc_1_0& buffer) {
|
|
/* Frames are delivered via deliverFrame_1_1 callback for clients that implement
|
|
* IEvsCameraStream v1.1 interfaces and therefore this method must not be
|
|
* used.
|
|
*/
|
|
LOG(INFO) << "A delivered frame from EVS v1.0 HW module is rejected.";
|
|
mHwCamera->doneWithFrame(buffer);
|
|
|
|
// Reports a received and returned buffer
|
|
mUsageStats->framesReceived();
|
|
mUsageStats->framesReturned();
|
|
|
|
return Void();
|
|
}
|
|
|
|
// Methods from ::android::hardware::automotive::evs::V1_1::IEvsCameraStream follow.
|
|
Return<void> HalCamera::deliverFrame_1_1(const hardware::hidl_vec<BufferDesc_1_1>& buffer) {
|
|
LOG(VERBOSE) << "Received a frame";
|
|
// Frames are being forwarded to v1.1 clients only who requested new frame.
|
|
const auto timestamp = buffer[0].timestamp;
|
|
// TODO(b/145750636): For now, we are using a approximately half of 1 seconds / 30 frames = 33ms
|
|
// but this must be derived from current framerate.
|
|
constexpr int64_t kThreshold = 16 * 1e+3; // ms
|
|
unsigned frameDeliveriesV1 = 0;
|
|
{
|
|
// Handle frame requests from v1.1 clients
|
|
std::lock_guard<std::mutex> lock(mFrameMutex);
|
|
std::swap(mCurrentRequests, mNextRequests);
|
|
while (!mCurrentRequests->empty()) {
|
|
auto req = mCurrentRequests->front();
|
|
mCurrentRequests->pop_front();
|
|
sp<VirtualCamera> vCam = req.client.promote();
|
|
if (vCam == nullptr) {
|
|
// Ignore a client already dead.
|
|
continue;
|
|
} else if (timestamp - req.timestamp < kThreshold) {
|
|
// Skip current frame because it arrives too soon.
|
|
LOG(DEBUG) << "Skips a frame from " << getId();
|
|
mNextRequests->push_back(req);
|
|
|
|
// Reports a skipped frame
|
|
mUsageStats->framesSkippedToSync();
|
|
} else if (vCam != nullptr) {
|
|
if (!vCam->deliverFrame(buffer[0])) {
|
|
LOG(WARNING) << getId() << " failed to forward the buffer to " << vCam.get();
|
|
} else {
|
|
LOG(ERROR) << getId() << " forwarded the buffer #" << buffer[0].bufferId
|
|
<< " to " << vCam.get() << " from " << this;
|
|
++frameDeliveriesV1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Reports the number of received buffers
|
|
mUsageStats->framesReceived(buffer);
|
|
|
|
// Frames are being forwarded to active v1.0 clients and v1.1 clients if we
|
|
// failed to create a timeline.
|
|
unsigned frameDeliveries = 0;
|
|
for (auto&& client : mClients) {
|
|
sp<VirtualCamera> vCam = client.promote();
|
|
if (vCam == nullptr || vCam->getVersion() > 0) {
|
|
continue;
|
|
}
|
|
|
|
if (vCam->deliverFrame(buffer[0])) {
|
|
++frameDeliveries;
|
|
}
|
|
}
|
|
|
|
frameDeliveries += frameDeliveriesV1;
|
|
if (frameDeliveries < 1) {
|
|
// If none of our clients could accept the frame, then return it
|
|
// right away.
|
|
LOG(INFO) << "Trivially rejecting frame (" << buffer[0].bufferId << ") from " << getId()
|
|
<< " with no acceptance";
|
|
mHwCamera->doneWithFrame_1_1(buffer);
|
|
|
|
// Reports a returned buffer
|
|
mUsageStats->framesReturned(buffer);
|
|
} else {
|
|
// Add an entry for this frame in our tracking list.
|
|
unsigned i;
|
|
for (i = 0; i < mFrames.size(); ++i) {
|
|
if (mFrames[i].refCount == 0) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (i == mFrames.size()) {
|
|
mFrames.emplace_back(buffer[0].bufferId);
|
|
} else {
|
|
mFrames[i].frameId = buffer[0].bufferId;
|
|
}
|
|
mFrames[i].refCount = frameDeliveries;
|
|
}
|
|
|
|
return Void();
|
|
}
|
|
|
|
Return<void> HalCamera::notify(const EvsEventDesc& event) {
|
|
LOG(DEBUG) << "Received an event id: " << static_cast<int32_t>(event.aType);
|
|
if (event.aType == EvsEventType::STREAM_STOPPED) {
|
|
// This event happens only when there is no more active client.
|
|
if (mStreamState != STOPPING) {
|
|
LOG(WARNING) << "Stream stopped unexpectedly";
|
|
}
|
|
|
|
mStreamState = STOPPED;
|
|
}
|
|
|
|
// Forward all other events to the clients
|
|
for (auto&& client : mClients) {
|
|
sp<VirtualCamera> vCam = client.promote();
|
|
if (vCam != nullptr) {
|
|
if (!vCam->notify(event)) {
|
|
LOG(INFO) << "Failed to forward an event";
|
|
}
|
|
}
|
|
}
|
|
|
|
return Void();
|
|
}
|
|
|
|
Return<EvsResult> HalCamera::setMaster(sp<VirtualCamera> virtualCamera) {
|
|
if (mPrimaryClient == nullptr) {
|
|
LOG(DEBUG) << __FUNCTION__ << ": " << virtualCamera.get() << " becomes a primary client.";
|
|
mPrimaryClient = virtualCamera;
|
|
return EvsResult::OK;
|
|
} else {
|
|
LOG(INFO) << "This camera already has a primary client.";
|
|
return EvsResult::OWNERSHIP_LOST;
|
|
}
|
|
}
|
|
|
|
Return<EvsResult> HalCamera::forceMaster(sp<VirtualCamera> virtualCamera) {
|
|
sp<VirtualCamera> prevPrimary = mPrimaryClient.promote();
|
|
if (prevPrimary == virtualCamera) {
|
|
LOG(DEBUG) << "Client " << virtualCamera.get() << " is already a primary client";
|
|
} else {
|
|
mPrimaryClient = virtualCamera;
|
|
if (prevPrimary != nullptr) {
|
|
LOG(INFO) << "High priority client " << virtualCamera.get()
|
|
<< " steals a primary role from " << prevPrimary.get();
|
|
|
|
/* Notify a previous primary client the loss of a primary role */
|
|
EvsEventDesc event;
|
|
event.aType = EvsEventType::MASTER_RELEASED;
|
|
if (!prevPrimary->notify(event)) {
|
|
LOG(ERROR) << "Fail to deliver a primary role lost notification";
|
|
}
|
|
}
|
|
}
|
|
|
|
return EvsResult::OK;
|
|
}
|
|
|
|
Return<EvsResult> HalCamera::unsetMaster(const VirtualCamera* virtualCamera) {
|
|
if (mPrimaryClient.promote() != virtualCamera) {
|
|
return EvsResult::INVALID_ARG;
|
|
} else {
|
|
LOG(INFO) << "Unset a primary camera client";
|
|
mPrimaryClient = nullptr;
|
|
|
|
/* Notify other clients that a primary role becomes available. */
|
|
EvsEventDesc event;
|
|
event.aType = EvsEventType::MASTER_RELEASED;
|
|
auto cbResult = this->notify(event);
|
|
if (!cbResult.isOk()) {
|
|
LOG(ERROR) << "Fail to deliver a parameter change notification";
|
|
}
|
|
|
|
return EvsResult::OK;
|
|
}
|
|
}
|
|
|
|
Return<EvsResult> HalCamera::setParameter(sp<VirtualCamera> virtualCamera, CameraParam id,
|
|
int32_t* value) {
|
|
EvsResult result = EvsResult::INVALID_ARG;
|
|
if (virtualCamera == mPrimaryClient.promote()) {
|
|
mHwCamera->setIntParameter(id, *value, [&result, value](auto status, auto readValue) {
|
|
result = status;
|
|
*value = readValue[0];
|
|
});
|
|
|
|
if (result == EvsResult::OK) {
|
|
/* Notify a parameter change */
|
|
EvsEventDesc event;
|
|
event.aType = EvsEventType::PARAMETER_CHANGED;
|
|
event.payload[0] = static_cast<uint32_t>(id);
|
|
event.payload[1] = static_cast<uint32_t>(*value);
|
|
auto cbResult = this->notify(event);
|
|
if (!cbResult.isOk()) {
|
|
LOG(ERROR) << "Fail to deliver a parameter change notification";
|
|
}
|
|
}
|
|
} else {
|
|
LOG(WARNING) << "A parameter change request from the non-primary client is declined.";
|
|
|
|
/* Read a current value of a requested camera parameter */
|
|
getParameter(id, value);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
Return<EvsResult> HalCamera::getParameter(CameraParam id, int32_t* value) {
|
|
EvsResult result = EvsResult::OK;
|
|
mHwCamera->getIntParameter(id, [&result, value](auto status, auto readValue) {
|
|
result = status;
|
|
if (result == EvsResult::OK) {
|
|
*value = readValue[0];
|
|
}
|
|
});
|
|
|
|
return result;
|
|
}
|
|
|
|
CameraUsageStatsRecord HalCamera::getStats() const {
|
|
return mUsageStats->snapshot();
|
|
}
|
|
|
|
Stream HalCamera::getStreamConfiguration() const {
|
|
return mStreamConfig;
|
|
}
|
|
|
|
std::string HalCamera::toString(const char* indent) const {
|
|
std::string buffer;
|
|
|
|
const auto timeElapsedMs = android::uptimeMillis() - mTimeCreatedMs;
|
|
StringAppendF(&buffer, "%sCreated: @%" PRId64 " (elapsed %" PRId64 " ms)\n", indent,
|
|
mTimeCreatedMs, timeElapsedMs);
|
|
|
|
std::string double_indent(indent);
|
|
double_indent += indent;
|
|
buffer += CameraUsageStats::toString(getStats(), double_indent.c_str());
|
|
for (auto&& client : mClients) {
|
|
auto handle = client.promote();
|
|
if (!handle) {
|
|
continue;
|
|
}
|
|
|
|
StringAppendF(&buffer, "%sClient %p\n", indent, handle.get());
|
|
buffer += handle->toString(double_indent.c_str());
|
|
}
|
|
|
|
StringAppendF(&buffer, "%sPrimary client: %p\n", indent, mPrimaryClient.promote().get());
|
|
|
|
buffer += HalCamera::toString(mStreamConfig, indent);
|
|
|
|
return buffer;
|
|
}
|
|
|
|
std::string HalCamera::toString(Stream configuration, const char* indent) {
|
|
std::string streamInfo;
|
|
std::string double_indent(indent);
|
|
double_indent += indent;
|
|
StringAppendF(&streamInfo,
|
|
"%sActive Stream Configuration\n"
|
|
"%sid: %d\n"
|
|
"%swidth: %d\n"
|
|
"%sheight: %d\n"
|
|
"%sformat: 0x%X\n"
|
|
"%susage: 0x%" PRIx64 "\n"
|
|
"%srotation: 0x%X\n\n",
|
|
indent, double_indent.c_str(), configuration.id, double_indent.c_str(),
|
|
configuration.width, double_indent.c_str(), configuration.height,
|
|
double_indent.c_str(), configuration.format, double_indent.c_str(),
|
|
configuration.usage, double_indent.c_str(), configuration.rotation);
|
|
|
|
return streamInfo;
|
|
}
|
|
|
|
} // namespace android::automotive::evs::V1_1::implementation
|