[libcamera-devel] [PATCH v2 7/7] android: Plumb all sensitivity-related controls
Paul Elder
paul.elder at ideasonboard.com
Fri Oct 1 12:33:25 CEST 2021
Plumb through the HAL the following three controls:
- ANDROID_SENSOR_INFO_SENSITIVITY_RANGE (static)
- ANDROID_SENSOR_MAX_ANALOG_SENSITIVITY (static)
- ANDROID_SENSOR_SENSITIVITY (request, result)
Also add a minISO to the HAL config, and add appropriate capability
detection.
The sensitivity range comes from:
- min: hardcode to 100, or take minISO from the HAL config
- max: set to minISO * max(AnalogueGain) * max(DigitalGain)
The max analog sensitivity comes from:
- hardcode to minISO * max(AnalogueGain)
The request sensitivity is mapped to requested ISO / minISO, and split
between analog and digital gain.
The result sensitivity is mapped from minISO * analog gain * digital
gain.
Bug: https://bugs.libcamera.org/show_bug.cgi?id=47
Signed-off-by: Paul Elder <paul.elder at ideasonboard.com>
---
No changes in v2
- I'm moving this into the AE-related series, because this is AE-related
---
src/android/camera_capabilities.cpp | 61 ++++++++++++++++++++---
src/android/camera_capabilities.h | 3 +-
src/android/camera_device.cpp | 76 ++++++++++++++++++++++++++++-
src/android/camera_device.h | 1 +
src/android/camera_hal_config.cpp | 10 +++-
src/android/camera_hal_config.h | 1 +
6 files changed, 142 insertions(+), 10 deletions(-)
diff --git a/src/android/camera_capabilities.cpp b/src/android/camera_capabilities.cpp
index 3fed3f83..8e84a736 100644
--- a/src/android/camera_capabilities.cpp
+++ b/src/android/camera_capabilities.cpp
@@ -233,6 +233,15 @@ bool CameraCapabilities::validateManualSensorCapability()
return false;
}
+ /*
+ * Checking the sensitivity range is sufficient, as it also covers the
+ * max analog sensitivity and the sensor sensitivity request/result key
+ */
+ if (!staticMetadata_->hasEntry(ANDROID_SENSOR_INFO_SENSITIVITY_RANGE)) {
+ LOG(HAL, Info) << noMode << "missing sensitivity range";
+ return false;
+ }
+
if (!staticMetadata_->hasEntry(ANDROID_SENSOR_INFO_EXPOSURE_TIME_RANGE)) {
LOG(HAL, Info) << noMode << "missing exposure time range";
return false;
@@ -401,11 +410,12 @@ void CameraCapabilities::computeHwLevel(
}
int CameraCapabilities::initialize(std::shared_ptr<Camera> camera,
- int orientation, int facing)
+ int orientation, int facing, int minISO)
{
camera_ = camera;
orientation_ = orientation;
facing_ = facing;
+ minISO_ = minISO;
rawStreamAvailable_ = false;
/* Acquire the camera and initialize available stream configurations. */
@@ -721,7 +731,6 @@ int CameraCapabilities::initializeStaticMetadata()
ANDROID_SENSOR_INFO_MAX_FRAME_DURATION,
ANDROID_SENSOR_INFO_PHYSICAL_SIZE,
ANDROID_SENSOR_INFO_PIXEL_ARRAY_SIZE,
- ANDROID_SENSOR_INFO_SENSITIVITY_RANGE,
ANDROID_SENSOR_INFO_TIMESTAMP_SOURCE,
ANDROID_SENSOR_ORIENTATION,
ANDROID_STATISTICS_INFO_AVAILABLE_FACE_DETECT_MODES,
@@ -1077,11 +1086,49 @@ int CameraCapabilities::initializeStaticMetadata()
data);
}
- int32_t sensitivityRange[] = {
- 32, 2400,
- };
- staticMetadata_->addEntry(ANDROID_SENSOR_INFO_SENSITIVITY_RANGE,
- sensitivityRange);
+ /*
+ * The following three controls are only necessary for FULL
+ * (specifically the manual sensor capability):
+ * - ANDROID_SENSOR_INFO_SENSITIVITY_RANGE (static)
+ * - ANDROID_SENSOR_MAX_ANALOG_SENSITIVITY (static)
+ * - ANDROID_SENSOR_SENSITIVITY (request, result)
+ *
+ * The first and second depend on analog gain, and the third depend on
+ * both, so if analog gain is not available then we can cross out all
+ * of them. We'll default to 100 minimum sensitivity if it isn't
+ * specified in the HAL configuration file.
+ *
+ * Digital gain is optional; if it's unavailable then the max analog
+ * sensitivity will be equal to the max sensitivity range.
+ *
+ * The minimum sensitivity must be at most 100, while the maximum must
+ * be at least 800.
+ */
+ const auto &analogGainInfo = controlsInfo.find(&controls::AnalogueGain);
+ if (analogGainInfo != controlsInfo.end()) {
+ float maxAnalogGain = analogGainInfo->second.max().get<float>();
+
+ const auto &digitalGainInfo = controlsInfo.find(&controls::DigitalGain);
+ float maxDigitalGain = digitalGainInfo == controlsInfo.end() ?
+ 1.0f :
+ digitalGainInfo->second.max().get<float>();
+
+ int32_t maxISO = minISO_ * maxAnalogGain * maxDigitalGain;
+ int32_t sensitivityRange[] = {
+ minISO_,
+ maxISO < 800 ? 800 : maxISO,
+ };
+ staticMetadata_->addEntry(ANDROID_SENSOR_INFO_SENSITIVITY_RANGE,
+ sensitivityRange);
+
+ staticMetadata_->addEntry(ANDROID_SENSOR_MAX_ANALOG_SENSITIVITY,
+ minISO_ * maxAnalogGain);
+
+ availableCharacteristicsKeys_.insert(ANDROID_SENSOR_INFO_SENSITIVITY_RANGE);
+ availableCharacteristicsKeys_.insert(ANDROID_SENSOR_MAX_ANALOG_SENSITIVITY);
+ availableRequestKeys_.insert(ANDROID_SENSOR_SENSITIVITY);
+ availableResultKeys_.insert(ANDROID_SENSOR_SENSITIVITY);
+ }
/* Report the color filter arrangement if the camera reports it. */
if (properties.contains(properties::draft::ColorFilterArrangement)) {
diff --git a/src/android/camera_capabilities.h b/src/android/camera_capabilities.h
index a1259699..b1eee37f 100644
--- a/src/android/camera_capabilities.h
+++ b/src/android/camera_capabilities.h
@@ -26,7 +26,7 @@ public:
CameraCapabilities() = default;
int initialize(std::shared_ptr<libcamera::Camera> camera,
- int orientation, int facing);
+ int orientation, int facing, int minISO);
CameraMetadata *staticMetadata() const { return staticMetadata_.get(); }
libcamera::PixelFormat toPixelFormat(int format) const;
@@ -68,6 +68,7 @@ private:
int facing_;
int orientation_;
+ int minISO_;
bool rawStreamAvailable_;
camera_metadata_enum_android_info_supported_hardware_level hwLevel_;
std::set<camera_metadata_enum_android_request_available_capabilities> capabilities_;
diff --git a/src/android/camera_device.cpp b/src/android/camera_device.cpp
index d5027ec5..8854a960 100644
--- a/src/android/camera_device.cpp
+++ b/src/android/camera_device.cpp
@@ -393,7 +393,15 @@ int CameraDevice::initialize(const CameraConfigData *cameraConfigData)
orientation_ = 0;
}
- return capabilities_.initialize(camera_, orientation_, facing_);
+ if (cameraConfigData && cameraConfigData->minISO != -1) {
+ minISO_ = cameraConfigData->minISO;
+ } else {
+ LOG(HAL, Error)
+ << "Minimum ISO not in configuration file. Using 100.";
+ minISO_ = 100;
+ }
+
+ return capabilities_.initialize(camera_, orientation_, facing_, minISO_);
}
/*
@@ -908,6 +916,51 @@ int CameraDevice::processControls(Camera3RequestDescriptor *descriptor)
}
}
+ if (settings.getEntry(ANDROID_SENSOR_SENSITIVITY, &entry)) {
+ const auto &info = camera_->controls().find(&controls::AnalogueGain);
+ if (info != camera_->controls().end()) {
+ const auto &dInfo =
+ camera_->controls().find(&controls::DigitalGain);
+ bool digitalGainAvailable =
+ dInfo == camera_->controls().end();
+
+ float maxAnalogGain = info->second.max().get<float>();
+ float maxDigitalGain = digitalGainAvailable ?
+ dInfo->second.max().get<float>() : 1.0f;
+
+ /* target ISO / min ISO = gain */
+ float gain = *entry.data.i32 / minISO_;
+
+ /*
+ * Max out analog gain before applying digital gain, if
+ * digital gain is available.
+ */
+ bool setDigital = false;
+ if (gain <= maxAnalogGain) {
+ lastAnalogueGain_ = gain;
+ } else if (gain <= maxAnalogGain * maxDigitalGain) {
+ lastAnalogueGain_ = maxAnalogGain;
+ if (digitalGainAvailable) {
+ setDigital = true;
+ lastDigitalGain_ = gain / maxAnalogGain;
+ }
+ } else {
+ lastAnalogueGain_ = maxAnalogGain;
+ if (digitalGainAvailable) {
+ setDigital = true;
+ lastDigitalGain_ = maxDigitalGain;
+ }
+ }
+
+ /* Don't disable libcamera's internal AeMode::Lock */
+ if (aeMode != AeMode::Lock)
+ controls.set(controls::AnalogueGain, lastAnalogueGain_);
+
+ if (setDigital && (aeMode != AeMode::Lock))
+ controls.set(controls::DigitalGain, lastDigitalGain_);
+ }
+ }
+
/* Trigger libcamera's locked -> manual state change */
if (aeMode == AeMode::Manual && !settings.hasEntry(ANDROID_SENSOR_EXPOSURE_TIME)) {
const auto &info = camera_->controls().find(&controls::ExposureTime);
@@ -915,6 +968,17 @@ int CameraDevice::processControls(Camera3RequestDescriptor *descriptor)
controls.set(controls::ExposureTime, lastExposureTime_);
}
+ /* Trigger libcamera's locked -> manual state change */
+ if (aeMode == AeMode::Manual && !settings.hasEntry(ANDROID_SENSOR_SENSITIVITY)) {
+ const auto &aInfo = camera_->controls().find(&controls::AnalogueGain);
+ if (aInfo != camera_->controls().end())
+ controls.set(controls::AnalogueGain, lastAnalogueGain_);
+
+ const auto &dInfo = camera_->controls().find(&controls::DigitalGain);
+ if (dInfo != camera_->controls().end())
+ controls.set(controls::DigitalGain, lastDigitalGain_);
+ }
+
return 0;
}
@@ -1548,6 +1612,16 @@ CameraDevice::getResultMetadata(const Camera3RequestDescriptor &descriptor) cons
resultMetadata->addEntry(ANDROID_SENSOR_EXPOSURE_TIME, exposure);
}
+ if (metadata.contains(controls::AnalogueGain) &&
+ settings.hasEntry(ANDROID_SENSOR_SENSITIVITY)) {
+ bool hasDigitalGain = metadata.contains(controls::DigitalGain);
+ float analogGain = metadata.get(controls::AnalogueGain);
+ float digitalGain = hasDigitalGain ? metadata.get(controls::DigitalGain) : 1.0f;
+
+ resultMetadata->addEntry(ANDROID_SENSOR_SENSITIVITY,
+ minISO_ * analogGain * digitalGain);
+ }
+
if (metadata.contains(controls::FrameDuration)) {
int64_t duration = metadata.get(controls::FrameDuration) * 1000;
resultMetadata->addEntry(ANDROID_SENSOR_FRAME_DURATION,
diff --git a/src/android/camera_device.h b/src/android/camera_device.h
index f693cdbc..e119dfd9 100644
--- a/src/android/camera_device.h
+++ b/src/android/camera_device.h
@@ -154,6 +154,7 @@ private:
int facing_;
int orientation_;
+ int minISO_;
/* Track the last-set android AE controls */
bool aeOn_;
diff --git a/src/android/camera_hal_config.cpp b/src/android/camera_hal_config.cpp
index aa90dac7..863cdff0 100644
--- a/src/android/camera_hal_config.cpp
+++ b/src/android/camera_hal_config.cpp
@@ -180,6 +180,13 @@ int CameraHalConfig::Private::parseCameraConfigData(const std::string &cameraId)
return -EINVAL;
}
cameraConfigData.rotation = ret;
+ } else if (key == "minISO") {
+ ret = std::stoi(value);
+ if (ret < 0 || ret > 100) {
+ LOG(HALConfig, Error)
+ << "Invalid minimum ISO: " << value;
+ return -EINVAL;
+ }
} else {
LOG(HALConfig, Error)
<< "Unknown key: " << key;
@@ -384,7 +391,8 @@ int CameraHalConfig::parseConfigurationFile()
const CameraConfigData &camera = c.second;
LOG(HALConfig, Debug) << "'" << cameraId << "' "
<< "(" << camera.facing << ")["
- << camera.rotation << "]";
+ << camera.rotation << "], "
+ << "minISO: " << camera.minISO;
}
return 0;
diff --git a/src/android/camera_hal_config.h b/src/android/camera_hal_config.h
index a79d5d6c..afa02a9b 100644
--- a/src/android/camera_hal_config.h
+++ b/src/android/camera_hal_config.h
@@ -15,6 +15,7 @@
struct CameraConfigData {
int facing = -1;
int rotation = -1;
+ int minISO = -1;
};
class CameraHalConfig final : public libcamera::Extensible
--
2.27.0
More information about the libcamera-devel
mailing list