如何使用 JNI 在 C 中获取原始 Android 相机缓冲区?
我一直在彻底搜索 Google 和 StackOverflow,但找不到这个.也许我错过了一些明显的东西.谢谢!
I've been searching Google and StackOverflow exhaustively and cannot find this. Maybe I'm missing something obvious. Thanks!
(这是因为预览回调的Java实现[即使有缓冲区]效率太低.)
(This is because the Java implementation of the preview callback [even with buffer] is too inefficient.)
推荐答案
我对主题做了一点调查.这个介绍(来自p.277,中文)帮助很大.
I made a little investigation on topic. This presentation (from p.277, Chinese) helped a lot.
正如其他人提到的,您可以使用 Camera.setPreviewCallback
方法获取缓冲区.
以下是它的发生方式(详细版本):
As others mentioned, you can get a buffer using a Camera.setPreviewCallback
method.
Here's how it happens there (a verbose version):
- 用户调用
Camera.startPreview()
这是一个原生函数. android_hardware_Camera_startPreview
调用C++Camera
类的startPreview
方法.Camera
调用ICamera
接口的startPreview
方法ICamera
对远程客户端进行IPC
调用.- 调用
CameraService
类的setCameraMode
方法. CameraService
设置一个窗口显示预览并调用CameraHardwareInterface
类的startPreview
方法.- 后者尝试在特定
camera_device_t
设备上调用start_preview
方法.
我没有进一步查找,但它应该会调用驱动程序. - 当图片到达时,调用
CameraService
的dataCallback
. - 将数据传递给客户端的
handlePreviewData
方法. - 客户端要么复制缓冲区,要么直接将其发送到
ICameraClient
. ICameraClient
通过IPC
将其发送到Camera
.Camera
调用已注册的监听器并将缓冲区传递给JNI
.- 它调用 Java 类中的回调.如果用户使用
Camera.addCallbackBuffer
提供了一个缓冲区,则它首先复制到缓冲区. - 最后,Java 类
Camera
处理消息并调用Camera.PreviewCallback
的onPreviewFrame
方法.
- User calls
Camera.startPreview()
which is a native function. android_hardware_Camera_startPreview
callsstartPreview
method of C++Camera
class.Camera
calls astartPreview
method ofICamera
interfaceICamera
makes anIPC
call to remote client.- It calls a
setCameraMode
method ofCameraService
class. CameraService
sets a window to display a preview and calls astartPreview
method ofCameraHardwareInterface
class.- The latter tries to call a
start_preview
method on particularcamera_device_t
device.
I didn't looked up further but it should perform a call to the driver. - When image arrives,
dataCallback
ofCameraService
is invoked. - It passes data to
handlePreviewData
method of client. - Client either copies the buffer or sends it directly to the
ICameraClient
. ICameraClient
sends it overIPC
to theCamera
.Camera
calls a registered listener and passes buffer toJNI
.- It invokes a callback in Java class. If user provided a buffer with
Camera.addCallbackBuffer
then it copies to the buffer first. - Finally Java class
Camera
handles the message and invokes aonPreviewFrame
method ofCamera.PreviewCallback
.
如您所见,调用了 2 个 IPC
调用,并且在步骤 10、11 中至少复制了两次缓冲区.camera_device_t
返回的原始缓冲区的第一个实例是托管在另一个进程中,由于 CameraService
中的安全检查,您无法访问它.
As you can see 2 IPC
calls were invoked and buffer was copied at least twice on steps 10, 11. First instance of raw buffer which is returned by camera_device_t
is hosted in another process and you cannot access it due to security checks in CameraService
.
但是,当您使用 Camera.setPreviewTexture
或 Camera.setPreviewDisplay
设置预览表面时,它会直接传递给相机设备并实时刷新,而无需上面的所有链条.正如它的文档所说:
However, when you set a preview surface using either Camera.setPreviewTexture
or Camera.setPreviewDisplay
it is be passed directly to the camera device and refreshed in realtime without participation of all the chain above. As it's documentation says:
处理一个由屏幕合成器管理的原始缓冲区.
Handle onto a raw buffer that is being managed by the screen compositor.
Java 类 Surface
有一个方法来检索它的内容:
Java class Surface
has a method to retrieve it's contents:
public static native Bitmap screenshot(int width, int height, int minLayer, int maxLayer);
但是这个 API 是隐藏的.参见 这个问题 了解使用方法.
But this API is hidden. See i.e. this question for a way to use it.
相关文章