2016-12-14 19:13:15 -08:00
|
|
|
/*
|
|
|
|
|
* 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.
|
|
|
|
|
*/
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
#define LOG_TAG "CamPrvdr@2.4-legacy"
|
2017-08-22 16:15:09 -07:00
|
|
|
//#define LOG_NDEBUG 0
|
2016-12-14 19:13:15 -08:00
|
|
|
#include <android/log.h>
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
#include "LegacyCameraProviderImpl_2_4.h"
|
2017-01-23 17:27:26 -08:00
|
|
|
#include "CameraDevice_1_0.h"
|
2017-08-22 16:15:09 -07:00
|
|
|
#include "CameraDevice_3_3.h"
|
2017-11-13 16:03:44 +00:00
|
|
|
#include "CameraDevice_3_4.h"
|
2018-08-17 13:52:40 -07:00
|
|
|
#include "CameraDevice_3_5.h"
|
2019-01-18 17:32:06 -08:00
|
|
|
#include "CameraProvider_2_4.h"
|
2017-08-22 16:15:09 -07:00
|
|
|
#include <cutils/properties.h>
|
2019-01-18 17:32:06 -08:00
|
|
|
#include <regex>
|
2016-12-14 19:13:15 -08:00
|
|
|
#include <string.h>
|
|
|
|
|
#include <utils/Trace.h>
|
|
|
|
|
|
|
|
|
|
namespace android {
|
|
|
|
|
namespace hardware {
|
|
|
|
|
namespace camera {
|
|
|
|
|
namespace provider {
|
|
|
|
|
namespace V2_4 {
|
|
|
|
|
namespace implementation {
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
template struct CameraProvider<LegacyCameraProviderImpl_2_4>;
|
|
|
|
|
|
2016-12-14 19:13:15 -08:00
|
|
|
namespace {
|
|
|
|
|
// "device@<version>/legacy/<id>"
|
|
|
|
|
const std::regex kDeviceNameRE("device@([0-9]+\\.[0-9]+)/legacy/(.+)");
|
2017-11-13 16:03:44 +00:00
|
|
|
const char *kHAL3_4 = "3.4";
|
2018-08-17 13:52:40 -07:00
|
|
|
const char *kHAL3_5 = "3.5";
|
2016-12-14 19:13:15 -08:00
|
|
|
const int kMaxCameraDeviceNameLen = 128;
|
|
|
|
|
const int kMaxCameraIdLen = 16;
|
|
|
|
|
|
2017-04-04 20:02:25 -07:00
|
|
|
bool matchDeviceName(const hidl_string& deviceName, std::string* deviceVersion,
|
|
|
|
|
std::string* cameraId) {
|
|
|
|
|
std::string deviceNameStd(deviceName.c_str());
|
|
|
|
|
std::smatch sm;
|
|
|
|
|
if (std::regex_match(deviceNameStd, sm, kDeviceNameRE)) {
|
|
|
|
|
if (deviceVersion != nullptr) {
|
|
|
|
|
*deviceVersion = sm[1];
|
|
|
|
|
}
|
|
|
|
|
if (cameraId != nullptr) {
|
|
|
|
|
*cameraId = sm[2];
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2016-12-14 19:13:15 -08:00
|
|
|
} // anonymous namespace
|
|
|
|
|
|
|
|
|
|
using ::android::hardware::camera::common::V1_0::CameraMetadataType;
|
|
|
|
|
using ::android::hardware::camera::common::V1_0::Status;
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
void LegacyCameraProviderImpl_2_4::addDeviceNames(int camera_id, CameraDeviceStatus status, bool cam_new)
|
Camera: Fix hotplug
According to the camera_module_t::get_number_of_cameras() description
in camera_common.h, it should only return the number of "fixed"
cameras, i.e. cameras, facing to the front and to the back. Any
cameras, with the facing value of "external," i.e. externally connected
cameras, should not be reported by it, instead they should later be
reported, using the
camera_module_callbacks_t::camera_device_status_change() callback.
However, this doesn't work. When a camera is reported this way, it is
ignored as unknown. Fix this by adding a new camera, when its status
change is reported.
Change-Id: I18874ed005f477e32fb723c00c16b67a81cde6ce
Signed-off-by: Guennadi Liakhovetski <guennadi.liakhovetski@intel.com>
2017-11-28 09:28:56 +01:00
|
|
|
{
|
|
|
|
|
char cameraId[kMaxCameraIdLen];
|
|
|
|
|
snprintf(cameraId, sizeof(cameraId), "%d", camera_id);
|
|
|
|
|
std::string cameraIdStr(cameraId);
|
|
|
|
|
|
|
|
|
|
mCameraIds.add(cameraIdStr);
|
|
|
|
|
|
|
|
|
|
// initialize mCameraDeviceNames and mOpenLegacySupported
|
|
|
|
|
mOpenLegacySupported[cameraIdStr] = false;
|
|
|
|
|
int deviceVersion = mModule->getDeviceVersion(camera_id);
|
|
|
|
|
auto deviceNamePair = std::make_pair(cameraIdStr,
|
|
|
|
|
getHidlDeviceName(cameraIdStr, deviceVersion));
|
|
|
|
|
mCameraDeviceNames.add(deviceNamePair);
|
|
|
|
|
if (cam_new) {
|
|
|
|
|
mCallbacks->cameraDeviceStatusChange(deviceNamePair.second, status);
|
|
|
|
|
}
|
|
|
|
|
if (deviceVersion >= CAMERA_DEVICE_API_VERSION_3_2 &&
|
|
|
|
|
mModule->isOpenLegacyDefined()) {
|
|
|
|
|
// try open_legacy to see if it actually works
|
|
|
|
|
struct hw_device_t* halDev = nullptr;
|
|
|
|
|
int ret = mModule->openLegacy(cameraId, CAMERA_DEVICE_API_VERSION_1_0, &halDev);
|
|
|
|
|
if (ret == 0) {
|
|
|
|
|
mOpenLegacySupported[cameraIdStr] = true;
|
|
|
|
|
halDev->close(halDev);
|
|
|
|
|
deviceNamePair = std::make_pair(cameraIdStr,
|
|
|
|
|
getHidlDeviceName(cameraIdStr, CAMERA_DEVICE_API_VERSION_1_0));
|
|
|
|
|
mCameraDeviceNames.add(deviceNamePair);
|
|
|
|
|
if (cam_new) {
|
|
|
|
|
mCallbacks->cameraDeviceStatusChange(deviceNamePair.second, status);
|
|
|
|
|
}
|
|
|
|
|
} else if (ret == -EBUSY || ret == -EUSERS) {
|
|
|
|
|
// Looks like this provider instance is not initialized during
|
|
|
|
|
// system startup and there are other camera users already.
|
|
|
|
|
// Not a good sign but not fatal.
|
|
|
|
|
ALOGW("%s: open_legacy try failed!", __FUNCTION__);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
void LegacyCameraProviderImpl_2_4::removeDeviceNames(int camera_id)
|
2017-12-07 10:59:35 +01:00
|
|
|
{
|
|
|
|
|
std::string cameraIdStr = std::to_string(camera_id);
|
|
|
|
|
|
|
|
|
|
mCameraIds.remove(cameraIdStr);
|
|
|
|
|
|
|
|
|
|
int deviceVersion = mModule->getDeviceVersion(camera_id);
|
|
|
|
|
auto deviceNamePair = std::make_pair(cameraIdStr,
|
|
|
|
|
getHidlDeviceName(cameraIdStr, deviceVersion));
|
|
|
|
|
mCameraDeviceNames.remove(deviceNamePair);
|
|
|
|
|
mCallbacks->cameraDeviceStatusChange(deviceNamePair.second, CameraDeviceStatus::NOT_PRESENT);
|
|
|
|
|
if (deviceVersion >= CAMERA_DEVICE_API_VERSION_3_2 &&
|
|
|
|
|
mModule->isOpenLegacyDefined() && mOpenLegacySupported[cameraIdStr]) {
|
|
|
|
|
|
|
|
|
|
deviceNamePair = std::make_pair(cameraIdStr,
|
|
|
|
|
getHidlDeviceName(cameraIdStr, CAMERA_DEVICE_API_VERSION_1_0));
|
|
|
|
|
mCameraDeviceNames.remove(deviceNamePair);
|
|
|
|
|
mCallbacks->cameraDeviceStatusChange(deviceNamePair.second,
|
|
|
|
|
CameraDeviceStatus::NOT_PRESENT);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mModule->removeCamera(camera_id);
|
|
|
|
|
}
|
|
|
|
|
|
2016-12-14 19:13:15 -08:00
|
|
|
/**
|
|
|
|
|
* static callback forwarding methods from HAL to instance
|
|
|
|
|
*/
|
2019-01-18 17:32:06 -08:00
|
|
|
void LegacyCameraProviderImpl_2_4::sCameraDeviceStatusChange(
|
2016-12-14 19:13:15 -08:00
|
|
|
const struct camera_module_callbacks* callbacks,
|
|
|
|
|
int camera_id,
|
|
|
|
|
int new_status) {
|
2019-01-18 17:32:06 -08:00
|
|
|
LegacyCameraProviderImpl_2_4* cp = const_cast<LegacyCameraProviderImpl_2_4*>(
|
|
|
|
|
static_cast<const LegacyCameraProviderImpl_2_4*>(callbacks));
|
2016-12-14 19:13:15 -08:00
|
|
|
if (cp == nullptr) {
|
|
|
|
|
ALOGE("%s: callback ops is null", __FUNCTION__);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Mutex::Autolock _l(cp->mCbLock);
|
|
|
|
|
char cameraId[kMaxCameraIdLen];
|
|
|
|
|
snprintf(cameraId, sizeof(cameraId), "%d", camera_id);
|
|
|
|
|
std::string cameraIdStr(cameraId);
|
|
|
|
|
cp->mCameraStatusMap[cameraIdStr] = (camera_device_status_t) new_status;
|
2019-01-16 19:11:16 +08:00
|
|
|
|
|
|
|
|
if (cp->mCallbacks == nullptr) {
|
|
|
|
|
// For camera connected before mCallbacks is set, the corresponding
|
|
|
|
|
// addDeviceNames() would be called later in setCallbacks().
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool found = false;
|
|
|
|
|
CameraDeviceStatus status = (CameraDeviceStatus)new_status;
|
|
|
|
|
for (auto const& deviceNamePair : cp->mCameraDeviceNames) {
|
|
|
|
|
if (cameraIdStr.compare(deviceNamePair.first) == 0) {
|
|
|
|
|
cp->mCallbacks->cameraDeviceStatusChange(deviceNamePair.second, status);
|
|
|
|
|
found = true;
|
2016-12-14 19:13:15 -08:00
|
|
|
}
|
2019-01-16 19:11:16 +08:00
|
|
|
}
|
Camera: Fix hotplug
According to the camera_module_t::get_number_of_cameras() description
in camera_common.h, it should only return the number of "fixed"
cameras, i.e. cameras, facing to the front and to the back. Any
cameras, with the facing value of "external," i.e. externally connected
cameras, should not be reported by it, instead they should later be
reported, using the
camera_module_callbacks_t::camera_device_status_change() callback.
However, this doesn't work. When a camera is reported this way, it is
ignored as unknown. Fix this by adding a new camera, when its status
change is reported.
Change-Id: I18874ed005f477e32fb723c00c16b67a81cde6ce
Signed-off-by: Guennadi Liakhovetski <guennadi.liakhovetski@intel.com>
2017-11-28 09:28:56 +01:00
|
|
|
|
2019-01-16 19:11:16 +08:00
|
|
|
switch (status) {
|
2017-12-07 10:59:35 +01:00
|
|
|
case CameraDeviceStatus::PRESENT:
|
|
|
|
|
case CameraDeviceStatus::ENUMERATING:
|
|
|
|
|
if (!found) {
|
|
|
|
|
cp->addDeviceNames(camera_id, status, true);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case CameraDeviceStatus::NOT_PRESENT:
|
|
|
|
|
if (found) {
|
|
|
|
|
cp->removeDeviceNames(camera_id);
|
|
|
|
|
}
|
2016-12-14 19:13:15 -08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
void LegacyCameraProviderImpl_2_4::sTorchModeStatusChange(
|
2016-12-14 19:13:15 -08:00
|
|
|
const struct camera_module_callbacks* callbacks,
|
|
|
|
|
const char* camera_id,
|
|
|
|
|
int new_status) {
|
2019-01-18 17:32:06 -08:00
|
|
|
LegacyCameraProviderImpl_2_4* cp = const_cast<LegacyCameraProviderImpl_2_4*>(
|
|
|
|
|
static_cast<const LegacyCameraProviderImpl_2_4*>(callbacks));
|
2016-12-14 19:13:15 -08:00
|
|
|
|
|
|
|
|
if (cp == nullptr) {
|
|
|
|
|
ALOGE("%s: callback ops is null", __FUNCTION__);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Mutex::Autolock _l(cp->mCbLock);
|
|
|
|
|
if (cp->mCallbacks != nullptr) {
|
|
|
|
|
std::string cameraIdStr(camera_id);
|
|
|
|
|
TorchModeStatus status = (TorchModeStatus) new_status;
|
|
|
|
|
for (auto const& deviceNamePair : cp->mCameraDeviceNames) {
|
2016-12-22 14:55:02 -08:00
|
|
|
if (cameraIdStr.compare(deviceNamePair.first) == 0) {
|
2016-12-14 19:13:15 -08:00
|
|
|
cp->mCallbacks->torchModeStatusChange(
|
|
|
|
|
deviceNamePair.second, status);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
Status LegacyCameraProviderImpl_2_4::getHidlStatus(int status) {
|
2016-12-14 19:13:15 -08:00
|
|
|
switch (status) {
|
|
|
|
|
case 0: return Status::OK;
|
|
|
|
|
case -ENODEV: return Status::INTERNAL_ERROR;
|
|
|
|
|
case -EINVAL: return Status::ILLEGAL_ARGUMENT;
|
|
|
|
|
default:
|
|
|
|
|
ALOGE("%s: unknown HAL status code %d", __FUNCTION__, status);
|
|
|
|
|
return Status::INTERNAL_ERROR;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
std::string LegacyCameraProviderImpl_2_4::getLegacyCameraId(const hidl_string& deviceName) {
|
2017-04-04 20:02:25 -07:00
|
|
|
std::string cameraId;
|
|
|
|
|
matchDeviceName(deviceName, nullptr, &cameraId);
|
|
|
|
|
return cameraId;
|
2016-12-14 19:13:15 -08:00
|
|
|
}
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
std::string LegacyCameraProviderImpl_2_4::getHidlDeviceName(
|
2016-12-14 19:13:15 -08:00
|
|
|
std::string cameraId, int deviceVersion) {
|
|
|
|
|
// Maybe consider create a version check method and SortedVec to speed up?
|
|
|
|
|
if (deviceVersion != CAMERA_DEVICE_API_VERSION_1_0 &&
|
|
|
|
|
deviceVersion != CAMERA_DEVICE_API_VERSION_3_2 &&
|
|
|
|
|
deviceVersion != CAMERA_DEVICE_API_VERSION_3_3 &&
|
2017-11-13 16:03:44 +00:00
|
|
|
deviceVersion != CAMERA_DEVICE_API_VERSION_3_4 &&
|
2018-09-20 15:06:13 -07:00
|
|
|
deviceVersion != CAMERA_DEVICE_API_VERSION_3_5 &&
|
|
|
|
|
deviceVersion != CAMERA_DEVICE_API_VERSION_3_6) {
|
2016-12-14 19:13:15 -08:00
|
|
|
return hidl_string("");
|
|
|
|
|
}
|
2018-09-20 15:06:13 -07:00
|
|
|
|
|
|
|
|
// Supported combinations:
|
|
|
|
|
// CAMERA_DEVICE_API_VERSION_1_0 -> ICameraDevice@1.0
|
|
|
|
|
// CAMERA_DEVICE_API_VERSION_3_[2-4] -> ICameraDevice@[3.2|3.3]
|
|
|
|
|
// CAMERA_DEVICE_API_VERSION_3_5 + CAMERA_MODULE_API_VERSION_2_4 -> ICameraDevice@3.4
|
|
|
|
|
// CAMERA_DEVICE_API_VERSION_3_[5-6] + CAMERA_MODULE_API_VERSION_2_5 -> ICameraDevice@3.5
|
2017-08-22 16:15:09 -07:00
|
|
|
bool isV1 = deviceVersion == CAMERA_DEVICE_API_VERSION_1_0;
|
|
|
|
|
int versionMajor = isV1 ? 1 : 3;
|
|
|
|
|
int versionMinor = isV1 ? 0 : mPreferredHal3MinorVersion;
|
2017-11-13 16:03:44 +00:00
|
|
|
if (deviceVersion == CAMERA_DEVICE_API_VERSION_3_5) {
|
2018-08-17 13:52:40 -07:00
|
|
|
if (mModule->getModuleApiVersion() == CAMERA_MODULE_API_VERSION_2_5) {
|
|
|
|
|
versionMinor = 5;
|
|
|
|
|
} else {
|
|
|
|
|
versionMinor = 4;
|
|
|
|
|
}
|
2018-09-20 15:06:13 -07:00
|
|
|
} else if (deviceVersion == CAMERA_DEVICE_API_VERSION_3_6) {
|
|
|
|
|
versionMinor = 5;
|
2017-11-13 16:03:44 +00:00
|
|
|
}
|
2016-12-14 19:13:15 -08:00
|
|
|
char deviceName[kMaxCameraDeviceNameLen];
|
2017-08-22 16:15:09 -07:00
|
|
|
snprintf(deviceName, sizeof(deviceName), "device@%d.%d/legacy/%s",
|
|
|
|
|
versionMajor, versionMinor, cameraId.c_str());
|
2016-12-14 19:13:15 -08:00
|
|
|
return deviceName;
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
LegacyCameraProviderImpl_2_4::LegacyCameraProviderImpl_2_4() :
|
2016-12-14 19:13:15 -08:00
|
|
|
camera_module_callbacks_t({sCameraDeviceStatusChange,
|
|
|
|
|
sTorchModeStatusChange}) {
|
|
|
|
|
mInitFailed = initialize();
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
LegacyCameraProviderImpl_2_4::~LegacyCameraProviderImpl_2_4() {}
|
2016-12-14 19:13:15 -08:00
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
bool LegacyCameraProviderImpl_2_4::initialize() {
|
2016-12-14 19:13:15 -08:00
|
|
|
camera_module_t *rawModule;
|
|
|
|
|
int err = hw_get_module(CAMERA_HARDWARE_MODULE_ID,
|
|
|
|
|
(const hw_module_t **)&rawModule);
|
|
|
|
|
if (err < 0) {
|
|
|
|
|
ALOGE("Could not load camera HAL module: %d (%s)", err, strerror(-err));
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mModule = new CameraModule(rawModule);
|
|
|
|
|
err = mModule->init();
|
|
|
|
|
if (err != OK) {
|
2017-01-31 16:00:20 -08:00
|
|
|
ALOGE("Could not initialize camera HAL module: %d (%s)", err, strerror(-err));
|
2016-12-14 19:13:15 -08:00
|
|
|
mModule.clear();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
ALOGI("Loaded \"%s\" camera module", mModule->getModuleName());
|
|
|
|
|
|
2017-03-15 18:26:39 -07:00
|
|
|
// Setup vendor tags here so HAL can setup vendor keys in camera characteristics
|
|
|
|
|
VendorTagDescriptor::clearGlobalVendorTagDescriptor();
|
|
|
|
|
if (!setUpVendorTags()) {
|
|
|
|
|
ALOGE("%s: Vendor tag setup failed, will not be available.", __FUNCTION__);
|
|
|
|
|
}
|
|
|
|
|
|
2017-01-31 16:00:20 -08:00
|
|
|
// Setup callback now because we are going to try openLegacy next
|
|
|
|
|
err = mModule->setCallbacks(this);
|
|
|
|
|
if (err != OK) {
|
|
|
|
|
ALOGE("Could not set camera module callback: %d (%s)", err, strerror(-err));
|
|
|
|
|
mModule.clear();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-12 17:50:42 +09:00
|
|
|
mPreferredHal3MinorVersion =
|
|
|
|
|
property_get_int32("ro.vendor.camera.wrapper.hal3TrebleMinorVersion", 3);
|
2017-08-22 16:15:09 -07:00
|
|
|
ALOGV("Preferred HAL 3 minor version is %d", mPreferredHal3MinorVersion);
|
|
|
|
|
switch(mPreferredHal3MinorVersion) {
|
|
|
|
|
case 2:
|
|
|
|
|
case 3:
|
|
|
|
|
// OK
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
ALOGW("Unknown minor camera device HAL version %d in property "
|
2017-11-13 16:03:44 +00:00
|
|
|
"'camera.wrapper.hal3TrebleMinorVersion', defaulting to 3",
|
|
|
|
|
mPreferredHal3MinorVersion);
|
2017-08-22 16:15:09 -07:00
|
|
|
mPreferredHal3MinorVersion = 3;
|
|
|
|
|
}
|
|
|
|
|
|
2016-12-14 19:13:15 -08:00
|
|
|
mNumberOfLegacyCameras = mModule->getNumberOfCameras();
|
|
|
|
|
for (int i = 0; i < mNumberOfLegacyCameras; i++) {
|
2017-04-10 16:12:55 +01:00
|
|
|
struct camera_info info;
|
|
|
|
|
auto rc = mModule->getCameraInfo(i, &info);
|
|
|
|
|
if (rc != NO_ERROR) {
|
|
|
|
|
ALOGE("%s: Camera info query failed!", __func__);
|
|
|
|
|
mModule.clear();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (checkCameraVersion(i, info) != OK) {
|
|
|
|
|
ALOGE("%s: Camera version check failed!", __func__);
|
|
|
|
|
mModule.clear();
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2016-12-14 19:13:15 -08:00
|
|
|
char cameraId[kMaxCameraIdLen];
|
|
|
|
|
snprintf(cameraId, sizeof(cameraId), "%d", i);
|
|
|
|
|
std::string cameraIdStr(cameraId);
|
|
|
|
|
mCameraStatusMap[cameraIdStr] = CAMERA_DEVICE_STATUS_PRESENT;
|
Camera: Fix hotplug
According to the camera_module_t::get_number_of_cameras() description
in camera_common.h, it should only return the number of "fixed"
cameras, i.e. cameras, facing to the front and to the back. Any
cameras, with the facing value of "external," i.e. externally connected
cameras, should not be reported by it, instead they should later be
reported, using the
camera_module_callbacks_t::camera_device_status_change() callback.
However, this doesn't work. When a camera is reported this way, it is
ignored as unknown. Fix this by adding a new camera, when its status
change is reported.
Change-Id: I18874ed005f477e32fb723c00c16b67a81cde6ce
Signed-off-by: Guennadi Liakhovetski <guennadi.liakhovetski@intel.com>
2017-11-28 09:28:56 +01:00
|
|
|
|
|
|
|
|
addDeviceNames(i);
|
2016-12-14 19:13:15 -08:00
|
|
|
}
|
|
|
|
|
|
2017-02-09 19:45:31 -08:00
|
|
|
return false; // mInitFailed
|
2016-12-14 19:13:15 -08:00
|
|
|
}
|
|
|
|
|
|
2017-04-10 16:12:55 +01:00
|
|
|
/**
|
|
|
|
|
* Check that the device HAL version is still in supported.
|
|
|
|
|
*/
|
2019-01-18 17:32:06 -08:00
|
|
|
int LegacyCameraProviderImpl_2_4::checkCameraVersion(int id, camera_info info) {
|
2017-04-10 16:12:55 +01:00
|
|
|
if (mModule == nullptr) {
|
|
|
|
|
return NO_INIT;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// device_version undefined in CAMERA_MODULE_API_VERSION_1_0,
|
|
|
|
|
// All CAMERA_MODULE_API_VERSION_1_0 devices are backward-compatible
|
2018-09-20 15:06:13 -07:00
|
|
|
uint16_t moduleVersion = mModule->getModuleApiVersion();
|
|
|
|
|
if (moduleVersion >= CAMERA_MODULE_API_VERSION_2_0) {
|
2017-04-10 16:12:55 +01:00
|
|
|
// Verify the device version is in the supported range
|
|
|
|
|
switch (info.device_version) {
|
|
|
|
|
case CAMERA_DEVICE_API_VERSION_1_0:
|
|
|
|
|
case CAMERA_DEVICE_API_VERSION_3_2:
|
|
|
|
|
case CAMERA_DEVICE_API_VERSION_3_3:
|
|
|
|
|
case CAMERA_DEVICE_API_VERSION_3_4:
|
2017-11-13 16:03:44 +00:00
|
|
|
case CAMERA_DEVICE_API_VERSION_3_5:
|
2017-04-10 16:12:55 +01:00
|
|
|
// in support
|
|
|
|
|
break;
|
2018-09-20 15:06:13 -07:00
|
|
|
case CAMERA_DEVICE_API_VERSION_3_6:
|
|
|
|
|
/**
|
|
|
|
|
* ICameraDevice@3.5 contains APIs from both
|
|
|
|
|
* CAMERA_DEVICE_API_VERSION_3_6 and CAMERA_MODULE_API_VERSION_2_5
|
|
|
|
|
* so we require HALs to uprev both for simplified supported combinations.
|
|
|
|
|
* HAL can still opt in individual new APIs indepedently.
|
|
|
|
|
*/
|
|
|
|
|
if (moduleVersion < CAMERA_MODULE_API_VERSION_2_5) {
|
|
|
|
|
ALOGE("%s: Device %d has unsupported version combination:"
|
|
|
|
|
"HAL version %x and module version %x",
|
|
|
|
|
__FUNCTION__, id, info.device_version, moduleVersion);
|
|
|
|
|
return NO_INIT;
|
|
|
|
|
}
|
|
|
|
|
break;
|
2017-04-10 16:12:55 +01:00
|
|
|
case CAMERA_DEVICE_API_VERSION_2_0:
|
|
|
|
|
case CAMERA_DEVICE_API_VERSION_2_1:
|
|
|
|
|
case CAMERA_DEVICE_API_VERSION_3_0:
|
|
|
|
|
case CAMERA_DEVICE_API_VERSION_3_1:
|
|
|
|
|
// no longer supported
|
|
|
|
|
default:
|
|
|
|
|
ALOGE("%s: Device %d has HAL version %x, which is not supported",
|
|
|
|
|
__FUNCTION__, id, info.device_version);
|
|
|
|
|
return NO_INIT;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
bool LegacyCameraProviderImpl_2_4::setUpVendorTags() {
|
2016-12-14 19:13:15 -08:00
|
|
|
ATRACE_CALL();
|
|
|
|
|
vendor_tag_ops_t vOps = vendor_tag_ops_t();
|
|
|
|
|
|
|
|
|
|
// Check if vendor operations have been implemented
|
|
|
|
|
if (!mModule->isVendorTagDefined()) {
|
|
|
|
|
ALOGI("%s: No vendor tags defined for this device.", __FUNCTION__);
|
2017-02-09 19:45:31 -08:00
|
|
|
return true;
|
2016-12-14 19:13:15 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mModule->getVendorTagOps(&vOps);
|
|
|
|
|
|
|
|
|
|
// Ensure all vendor operations are present
|
|
|
|
|
if (vOps.get_tag_count == nullptr || vOps.get_all_tags == nullptr ||
|
|
|
|
|
vOps.get_section_name == nullptr || vOps.get_tag_name == nullptr ||
|
|
|
|
|
vOps.get_tag_type == nullptr) {
|
|
|
|
|
ALOGE("%s: Vendor tag operations not fully defined. Ignoring definitions."
|
|
|
|
|
, __FUNCTION__);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Read all vendor tag definitions into a descriptor
|
|
|
|
|
sp<VendorTagDescriptor> desc;
|
|
|
|
|
status_t res;
|
|
|
|
|
if ((res = VendorTagDescriptor::createDescriptorFromOps(&vOps, /*out*/desc))
|
|
|
|
|
!= OK) {
|
|
|
|
|
ALOGE("%s: Could not generate descriptor from vendor tag operations,"
|
|
|
|
|
"received error %s (%d). Camera clients will not be able to use"
|
|
|
|
|
"vendor tags", __FUNCTION__, strerror(res), res);
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set the global descriptor to use with camera metadata
|
|
|
|
|
VendorTagDescriptor::setAsGlobalVendorTagDescriptor(desc);
|
|
|
|
|
const SortedVector<String8>* sectionNames = desc->getAllSectionNames();
|
|
|
|
|
size_t numSections = sectionNames->size();
|
|
|
|
|
std::vector<std::vector<VendorTag>> tagsBySection(numSections);
|
|
|
|
|
int tagCount = desc->getTagCount();
|
|
|
|
|
std::vector<uint32_t> tags(tagCount);
|
|
|
|
|
desc->getTagArray(tags.data());
|
|
|
|
|
for (int i = 0; i < tagCount; i++) {
|
|
|
|
|
VendorTag vt;
|
|
|
|
|
vt.tagId = tags[i];
|
|
|
|
|
vt.tagName = desc->getTagName(tags[i]);
|
|
|
|
|
vt.tagType = (CameraMetadataType) desc->getTagType(tags[i]);
|
|
|
|
|
ssize_t sectionIdx = desc->getSectionIndex(tags[i]);
|
|
|
|
|
tagsBySection[sectionIdx].push_back(vt);
|
|
|
|
|
}
|
|
|
|
|
mVendorTagSections.resize(numSections);
|
|
|
|
|
for (size_t s = 0; s < numSections; s++) {
|
2023-08-23 18:46:52 +00:00
|
|
|
mVendorTagSections[s].sectionName = (*sectionNames)[s].c_str();
|
2016-12-14 19:13:15 -08:00
|
|
|
mVendorTagSections[s].tags = tagsBySection[s];
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Methods from ::android::hardware::camera::provider::V2_4::ICameraProvider follow.
|
2019-01-18 17:32:06 -08:00
|
|
|
Return<Status> LegacyCameraProviderImpl_2_4::setCallback(
|
|
|
|
|
const sp<ICameraProviderCallback>& callback) {
|
2023-09-12 14:10:40 +08:00
|
|
|
if (callback == nullptr) {
|
|
|
|
|
return Status::ILLEGAL_ARGUMENT;
|
|
|
|
|
}
|
2017-01-31 16:00:20 -08:00
|
|
|
Mutex::Autolock _l(mCbLock);
|
|
|
|
|
mCallbacks = callback;
|
2019-01-16 19:11:16 +08:00
|
|
|
// Add and report all presenting external cameras.
|
|
|
|
|
for (auto const& statusPair : mCameraStatusMap) {
|
|
|
|
|
int id = std::stoi(statusPair.first);
|
|
|
|
|
auto status = static_cast<CameraDeviceStatus>(statusPair.second);
|
|
|
|
|
if (id >= mNumberOfLegacyCameras && status != CameraDeviceStatus::NOT_PRESENT) {
|
|
|
|
|
addDeviceNames(id, status, true);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-01-31 16:00:20 -08:00
|
|
|
return Status::OK;
|
2016-12-14 19:13:15 -08:00
|
|
|
}
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
Return<void> LegacyCameraProviderImpl_2_4::getVendorTags(
|
|
|
|
|
ICameraProvider::getVendorTags_cb _hidl_cb) {
|
2016-12-14 19:13:15 -08:00
|
|
|
_hidl_cb(Status::OK, mVendorTagSections);
|
|
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
Return<void> LegacyCameraProviderImpl_2_4::getCameraIdList(
|
|
|
|
|
ICameraProvider::getCameraIdList_cb _hidl_cb) {
|
2016-12-14 19:13:15 -08:00
|
|
|
std::vector<hidl_string> deviceNameList;
|
|
|
|
|
for (auto const& deviceNamePair : mCameraDeviceNames) {
|
2019-01-16 19:11:16 +08:00
|
|
|
if (std::stoi(deviceNamePair.first) >= mNumberOfLegacyCameras) {
|
|
|
|
|
// External camera devices must be reported through the device status change callback,
|
|
|
|
|
// not in this list.
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2016-12-14 19:13:15 -08:00
|
|
|
if (mCameraStatusMap[deviceNamePair.first] == CAMERA_DEVICE_STATUS_PRESENT) {
|
|
|
|
|
deviceNameList.push_back(deviceNamePair.second);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
hidl_vec<hidl_string> hidlDeviceNameList(deviceNameList);
|
2016-12-22 14:55:02 -08:00
|
|
|
_hidl_cb(Status::OK, hidlDeviceNameList);
|
2016-12-14 19:13:15 -08:00
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
Return<void> LegacyCameraProviderImpl_2_4::isSetTorchModeSupported(
|
|
|
|
|
ICameraProvider::isSetTorchModeSupported_cb _hidl_cb) {
|
2016-12-14 19:13:15 -08:00
|
|
|
bool support = mModule->isSetTorchModeSupported();
|
|
|
|
|
_hidl_cb (Status::OK, support);
|
|
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
Return<void> LegacyCameraProviderImpl_2_4::getCameraDeviceInterface_V1_x(
|
|
|
|
|
const hidl_string& cameraDeviceName,
|
|
|
|
|
ICameraProvider::getCameraDeviceInterface_V1_x_cb _hidl_cb) {
|
2017-04-04 20:02:25 -07:00
|
|
|
std::string cameraId, deviceVersion;
|
|
|
|
|
bool match = matchDeviceName(cameraDeviceName, &deviceVersion, &cameraId);
|
2017-01-23 17:27:26 -08:00
|
|
|
if (!match) {
|
|
|
|
|
_hidl_cb(Status::ILLEGAL_ARGUMENT, nullptr);
|
|
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::string deviceName(cameraDeviceName.c_str());
|
|
|
|
|
ssize_t index = mCameraDeviceNames.indexOf(std::make_pair(cameraId, deviceName));
|
|
|
|
|
if (index == NAME_NOT_FOUND) { // Either an illegal name or a device version mismatch
|
|
|
|
|
Status status = Status::OK;
|
|
|
|
|
ssize_t idx = mCameraIds.indexOf(cameraId);
|
|
|
|
|
if (idx == NAME_NOT_FOUND) {
|
|
|
|
|
ALOGE("%s: cannot find camera %s!", __FUNCTION__, cameraId.c_str());
|
|
|
|
|
status = Status::ILLEGAL_ARGUMENT;
|
|
|
|
|
} else { // invalid version
|
|
|
|
|
ALOGE("%s: camera device %s does not support version %s!",
|
|
|
|
|
__FUNCTION__, cameraId.c_str(), deviceVersion.c_str());
|
|
|
|
|
status = Status::OPERATION_NOT_SUPPORTED;
|
|
|
|
|
}
|
|
|
|
|
_hidl_cb(status, nullptr);
|
|
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (mCameraStatusMap.count(cameraId) == 0 ||
|
|
|
|
|
mCameraStatusMap[cameraId] != CAMERA_DEVICE_STATUS_PRESENT) {
|
|
|
|
|
_hidl_cb(Status::ILLEGAL_ARGUMENT, nullptr);
|
|
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sp<android::hardware::camera::device::V1_0::implementation::CameraDevice> device =
|
|
|
|
|
new android::hardware::camera::device::V1_0::implementation::CameraDevice(
|
|
|
|
|
mModule, cameraId, mCameraDeviceNames);
|
|
|
|
|
|
|
|
|
|
if (device == nullptr) {
|
|
|
|
|
ALOGE("%s: cannot allocate camera device for id %s", __FUNCTION__, cameraId.c_str());
|
|
|
|
|
_hidl_cb(Status::INTERNAL_ERROR, nullptr);
|
|
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (device->isInitFailed()) {
|
|
|
|
|
ALOGE("%s: camera device %s init failed!", __FUNCTION__, cameraId.c_str());
|
|
|
|
|
device = nullptr;
|
|
|
|
|
_hidl_cb(Status::INTERNAL_ERROR, nullptr);
|
|
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_hidl_cb (Status::OK, device);
|
2016-12-14 19:13:15 -08:00
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
|
2019-01-18 17:32:06 -08:00
|
|
|
Return<void> LegacyCameraProviderImpl_2_4::getCameraDeviceInterface_V3_x(
|
|
|
|
|
const hidl_string& cameraDeviceName,
|
|
|
|
|
ICameraProvider::getCameraDeviceInterface_V3_x_cb _hidl_cb) {
|
2017-04-04 20:02:25 -07:00
|
|
|
std::string cameraId, deviceVersion;
|
|
|
|
|
bool match = matchDeviceName(cameraDeviceName, &deviceVersion, &cameraId);
|
2016-12-14 19:13:15 -08:00
|
|
|
if (!match) {
|
|
|
|
|
_hidl_cb(Status::ILLEGAL_ARGUMENT, nullptr);
|
|
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::string deviceName(cameraDeviceName.c_str());
|
|
|
|
|
ssize_t index = mCameraDeviceNames.indexOf(std::make_pair(cameraId, deviceName));
|
|
|
|
|
if (index == NAME_NOT_FOUND) { // Either an illegal name or a device version mismatch
|
|
|
|
|
Status status = Status::OK;
|
|
|
|
|
ssize_t idx = mCameraIds.indexOf(cameraId);
|
|
|
|
|
if (idx == NAME_NOT_FOUND) {
|
2016-12-22 14:55:02 -08:00
|
|
|
ALOGE("%s: cannot find camera %s!", __FUNCTION__, cameraId.c_str());
|
2016-12-14 19:13:15 -08:00
|
|
|
status = Status::ILLEGAL_ARGUMENT;
|
|
|
|
|
} else { // invalid version
|
2016-12-22 14:55:02 -08:00
|
|
|
ALOGE("%s: camera device %s does not support version %s!",
|
|
|
|
|
__FUNCTION__, cameraId.c_str(), deviceVersion.c_str());
|
2016-12-14 19:13:15 -08:00
|
|
|
status = Status::OPERATION_NOT_SUPPORTED;
|
|
|
|
|
}
|
|
|
|
|
_hidl_cb(status, nullptr);
|
|
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (mCameraStatusMap.count(cameraId) == 0 ||
|
|
|
|
|
mCameraStatusMap[cameraId] != CAMERA_DEVICE_STATUS_PRESENT) {
|
|
|
|
|
_hidl_cb(Status::ILLEGAL_ARGUMENT, nullptr);
|
|
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
|
2018-08-17 13:52:40 -07:00
|
|
|
sp<android::hardware::camera::device::V3_2::implementation::CameraDevice> deviceImpl;
|
2018-09-20 15:06:13 -07:00
|
|
|
|
|
|
|
|
// ICameraDevice 3.4 or upper
|
2018-08-17 13:52:40 -07:00
|
|
|
if (deviceVersion >= kHAL3_4) {
|
2018-09-20 15:06:13 -07:00
|
|
|
ALOGV("Constructing v3.4+ camera device");
|
2018-08-17 13:52:40 -07:00
|
|
|
if (deviceVersion == kHAL3_4) {
|
|
|
|
|
deviceImpl = new android::hardware::camera::device::V3_4::implementation::CameraDevice(
|
|
|
|
|
mModule, cameraId, mCameraDeviceNames);
|
|
|
|
|
} else if (deviceVersion == kHAL3_5) {
|
|
|
|
|
deviceImpl = new android::hardware::camera::device::V3_5::implementation::CameraDevice(
|
2017-11-13 16:03:44 +00:00
|
|
|
mModule, cameraId, mCameraDeviceNames);
|
2018-08-17 13:52:40 -07:00
|
|
|
}
|
2017-11-13 16:03:44 +00:00
|
|
|
if (deviceImpl == nullptr || deviceImpl->isInitFailed()) {
|
|
|
|
|
ALOGE("%s: camera device %s init failed!", __FUNCTION__, cameraId.c_str());
|
|
|
|
|
_hidl_cb(Status::INTERNAL_ERROR, nullptr);
|
|
|
|
|
return Void();
|
|
|
|
|
}
|
2018-08-17 13:52:40 -07:00
|
|
|
IF_ALOGV() {
|
|
|
|
|
deviceImpl->getInterface()->interfaceChain([](
|
|
|
|
|
::android::hardware::hidl_vec<::android::hardware::hidl_string> interfaceChain) {
|
|
|
|
|
ALOGV("Device interface chain:");
|
|
|
|
|
for (auto iface : interfaceChain) {
|
|
|
|
|
ALOGV(" %s", iface.c_str());
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
_hidl_cb (Status::OK, deviceImpl->getInterface());
|
2017-11-13 16:03:44 +00:00
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-06 15:38:34 -07:00
|
|
|
// ICameraDevice 3.2 and 3.3
|
2017-08-22 16:15:09 -07:00
|
|
|
// Since some Treble HAL revisions can map to the same legacy HAL version(s), we default
|
|
|
|
|
// to the newest possible Treble HAL revision, but allow for override if needed via
|
|
|
|
|
// system property.
|
|
|
|
|
switch (mPreferredHal3MinorVersion) {
|
|
|
|
|
case 2: { // Map legacy camera device v3 HAL to Treble camera device HAL v3.2
|
|
|
|
|
ALOGV("Constructing v3.2 camera device");
|
2018-08-17 13:52:40 -07:00
|
|
|
deviceImpl = new android::hardware::camera::device::V3_2::implementation::CameraDevice(
|
2016-12-14 19:13:15 -08:00
|
|
|
mModule, cameraId, mCameraDeviceNames);
|
2017-08-22 16:15:09 -07:00
|
|
|
if (deviceImpl == nullptr || deviceImpl->isInitFailed()) {
|
|
|
|
|
ALOGE("%s: camera device %s init failed!", __FUNCTION__, cameraId.c_str());
|
|
|
|
|
_hidl_cb(Status::INTERNAL_ERROR, nullptr);
|
|
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case 3: { // Map legacy camera device v3 HAL to Treble camera device HAL v3.3
|
|
|
|
|
ALOGV("Constructing v3.3 camera device");
|
2018-08-17 13:52:40 -07:00
|
|
|
deviceImpl = new android::hardware::camera::device::V3_3::implementation::CameraDevice(
|
2017-08-22 16:15:09 -07:00
|
|
|
mModule, cameraId, mCameraDeviceNames);
|
|
|
|
|
if (deviceImpl == nullptr || deviceImpl->isInitFailed()) {
|
|
|
|
|
ALOGE("%s: camera device %s init failed!", __FUNCTION__, cameraId.c_str());
|
|
|
|
|
_hidl_cb(Status::INTERNAL_ERROR, nullptr);
|
|
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
ALOGE("%s: Unknown HAL minor version %d!", __FUNCTION__, mPreferredHal3MinorVersion);
|
|
|
|
|
_hidl_cb(Status::INTERNAL_ERROR, nullptr);
|
|
|
|
|
return Void();
|
2016-12-14 19:13:15 -08:00
|
|
|
}
|
2018-08-17 13:52:40 -07:00
|
|
|
|
|
|
|
|
_hidl_cb (Status::OK, deviceImpl->getInterface());
|
2016-12-14 19:13:15 -08:00
|
|
|
return Void();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace implementation
|
|
|
|
|
} // namespace V2_4
|
|
|
|
|
} // namespace provider
|
|
|
|
|
} // namespace camera
|
|
|
|
|
} // namespace hardware
|
|
|
|
|
} // namespace android
|