-
Notifications
You must be signed in to change notification settings - Fork 5.6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Exposure Control for other UVC Webcams #1300
Comments
The issue likely stems from how setExposure() maps values to UVC properties, you could try,
VideoCapture capture = new VideoCapture(0); // Adjust index for your camera
capture.set(Videoio.CAP_PROP_EXPOSURE, -1 * desiredExposure); // Negate the value as needed
|
Hi there - author of EasyOpenCV here - a couple notes: Using OpenCV to open a video capture device and set the exposure will not work. OpenCV has absolutely no support for external webcams on Android, and the FTC SDK Vision Portal relies entirely upon EasyOpenCV to bridge that gap. EasyOpenCV in turn relies on the FTC SDK's heavily modified copy of I couldn't seem to find the actual source code for the ArduCam SDK that @torinriley mentioned (I only found .deb binaries in the GitHub repo), but even if you could find the source, I can say with a high degree of confidence that it would not work on Android without extensive modifications. Without having hardware in hand, it would be extremely difficult to try to debug this, as cameras tend to be the wild west and just because a camera is "UVC compatible" does not necessarily mean it is "well behaved". @ftc19743 You mentioned wanting to try to bypass the ExposureControl class and set UVC properties directly. I don't know of a super easy way to do that off the top of my head, but if you want to trace the callstack in a debugger and try to see if that yields any clues, here's the general way we get from calling for example
One other thing to keep in mind is that the UVC standard also supports relative exposure (see for instance |
Thanks as always for the insights! We can see in our log a bit of the call stack you referenced above.
This led us to think that maybe we could poke values into the camera registers with setExposure() in the SDK and then read the exposure value back out of the camera to see what is actually stored there.
We tried it and realized that the SDK is caching the exposure value somewhere and just returns that:
So the follow on question: Is there perhaps a simple way to trick the SDK into calling down through the libuvc library to get the camera's currently stored value for the exposure property and returning/logging that? |
Here's an OpMode that uses reflection to access the UvcDeviceHandle object. Please note that this is absolutely not safe to use on a competition robot, do not use this code for any purposes beyond debugging attempts. package org.firstinspires.ftc.teamcode;
import com.qualcomm.robotcore.eventloop.opmode.LinearOpMode;
import com.qualcomm.robotcore.eventloop.opmode.TeleOp;
import org.firstinspires.ftc.robotcore.external.hardware.camera.WebcamName;
import org.firstinspires.ftc.robotcore.external.hardware.camera.controls.ExposureControl;
import org.firstinspires.ftc.robotcore.internal.camera.delegating.CachingExposureControl;
import org.firstinspires.ftc.robotcore.internal.camera.libuvc.api.UvcApiExposureControl;
import org.firstinspires.ftc.robotcore.internal.camera.libuvc.nativeobject.UvcDeviceHandle;
import org.firstinspires.ftc.vision.VisionPortal;
import java.lang.reflect.Field;
@TeleOp
public class UvcControlDirectAccess extends LinearOpMode
{
public void runOpMode()
{
VisionPortal portal = VisionPortal.easyCreateWithDefaults(
hardwareMap.get(WebcamName.class, "Webcam 1"));
UvcDeviceHandle uvcDeviceHandle = null;
while (!isStopRequested())
{
if (portal.getCameraState() == VisionPortal.CameraState.STREAMING)
{
CachingExposureControl cachingExposureControl = (CachingExposureControl) portal.getCameraControl(ExposureControl.class);
if (cachingExposureControl == null)
{
throw new RuntimeException("Failed to get exposure control");
}
try
{
Field f = CachingExposureControl.class.getDeclaredField("delegatedExposureControl");
f.setAccessible(true);
UvcApiExposureControl uvcApiExposureControl = (UvcApiExposureControl) f.get(cachingExposureControl);
Field f2 = UvcApiExposureControl.class.getDeclaredField("uvcDeviceHandle");
f2.setAccessible(true);
uvcDeviceHandle = (UvcDeviceHandle) f2.get(uvcApiExposureControl);
}
catch (Exception e)
{
throw new RuntimeException("Failed to reflect");
}
uvcDeviceHandle.setExposureMode(ExposureControl.Mode.Manual);
uvcDeviceHandle.setAePriority(false);
break;
}
telemetry.addData("Camera State", portal.getCameraState());
telemetry.update();
}
while (!isStopRequested())
{
long requestedExposure = (uvcDeviceHandle.getMinExposure() + uvcDeviceHandle.getMaxExposure()) / 2;
uvcDeviceHandle.setExposure(requestedExposure);
telemetry.addData("Min exposure (ns)", uvcDeviceHandle.getMinExposure());
telemetry.addData("Max exposure (ns)", uvcDeviceHandle.getMaxExposure());
telemetry.addData("Current exposure (ns)", uvcDeviceHandle.getExposure());
telemetry.addData("Requested Exposure (ns)", requestedExposure);
telemetry.update();
}
}
} |
Thanks, very interesting to see how to get down to the lower levels. This confirmed that the values passed to setExposure are indeed finding their way down to the camera so that uvcDeviceHandle.getExposure is returning them (translated to ns). Of course it does not solve our mystery of why "increasing" the exposure time on the camera results in a darker image. In any event, we were able to find an exposure time that results in solid CV with negligible blur even at high speeds. So all is well for now. As you pointed out, these UVC web cameras are under documented. :) |
If it's any comfort, it's not just your mystery! Here are a couple of points from the long tutorial on Camera Controls at ftc-docs:
|
Just stating the obvious here, possibly for others...... the Sample opmode ConceptAprilTagOptimizeExposure lets you interactively play with gain and exposure on a camera to observe the effects in real-time... (connect to the RC with ScrCpy or direct HDMI) |
The team is now trying to use the captureTimeNanos parameter that is passed to processFrame() to help coordinate their CV with actuator movements. They have not been able to determine what this capture time parameter is relative to. After converting to milliseconds it doesn't seem to be directly related to the system clock on the control hub. Does anyone know for certain what this parameter is? (This is only tangentially related to the original post on this thread, but maybe this is a thread for the more esoteric aspects of the Vision library? :) Thx! |
We are using an ArduCam B0454 UVC compliant webcam this season and have enjoyed the enhanced VisionPortal and CameraControls (thank you!).
We are setting the exposure manually and noticing that there does not seem to be a clear mapping from the numbers we are passing in via exposureControl.setExposure(exposure, TimeUnit.MILLISECONDS) and the camera's behavior. For example, a value of 1 for "exposure" in the previous call results in a fairly typically lighted image while a value of 2 reduces the brightness considerably (implying a shorter exposure time). This trend continues as the exposure value increases. getMinExposure(TimeUnit.MILLISECONDS) and getMaxExposure(TimeUnit.MILLISECONDS) return a range of 1-204.
We have this data from the manufacturer: https://docs.arducam.com/UVC-Camera/Adjust-the-minimum-exposure-time/ Its tempting to think that the "exposure" value used in setExposure() is being passed directly to the CAP_PROP_EXPOSURE property but with the sign flipped. However, we are fairly confident that the actual exposure time with an exposure value of "1" is considerably less than the 500ms the linked document suggests.
Is there a way to address this, or perhaps a way via the SDK to bypass ExposureControl and set UVC properties on the webcam directly?
The text was updated successfully, but these errors were encountered: