来自Android camera2 API的图像数据在Galaxy S5上翻转并压缩

Ste*_* G. 8 android android-camera galaxy android-5.0-lollipop

我正在实现一个应用程序,它使用来自相机的实时图像进行实时图像处理.使用现已弃用的android.hardware.Camera,它有限制地工作; 为了提高灵活性和性能,我想使用新的android.hardware.camera2 API.然而,我无法获取原始图像数据进行处理.这是三星Galaxy S5.(不幸的是,我没有其他Lollipop设备可以方便地在其他硬件上进行测试).

我得到了整体框架(灵感来自'HdrViewFinder'和'Camera2Basic'样本),并通过SurfaceTexture和GLSurfaceView在屏幕上绘制实时图像.但是,我还需要访问图像数据(灰度级很好,至少目前为止),以进行自定义图像处理.根据StreamConfigurationMap.isOutputSupportedFor(类)的文档,直接获取图像数据的推荐表面是ImageReader(正确吗?).

所以我将捕获请求设置为:

mSurfaceTexture.setDefaultBufferSize(640, 480);
mSurface = new Surface(surfaceTexture);
...
mImageReader = ImageReader.newInstance(640, 480, format, 2);
...
List<Surface> surfaces = new ArrayList<Surface>();
surfaces.add(mSurface);
surfaces.add(mImageReader.getSurface());
...
mCameraDevice.createCaptureSession(surfaces, mCameraSessionListener, mCameraHandler);
Run Code Online (Sandbox Code Playgroud)

在ImageReader的onImageAvailable回调中,我按如下方式访问数据:

Image img = reader.acquireLatestImage();
ByteBuffer grayscalePixelsDirectByteBuffer = img.getPlanes()[0].getBuffer();
Run Code Online (Sandbox Code Playgroud)

...虽然(如上所述)实时图像预览正在运行,但我得到的数据(或者我得到它的方式)有问题.根据

mCameraInfo.get(CameraCharacteristics.SCALER_STREAM_CONFIGURATION_MAP).getOutputFormats();
Run Code Online (Sandbox Code Playgroud)

...应支持以下ImageFormats:NV21,JPEG,YV12,YUV_420_888.我已经尝试了所有(插入上面的"格式"),所有都支持设置的分辨率getOutputSizes(format),但没有一个给出所需的结果:

  • NV21:ImageReader.newInstance抛出java.lang.IllegalArgumentException:不支持NV21格式
  • JPEG:这确实有效,但对于实时应用程序来说,对每个帧进行JPEG编码和解码似乎没有意义......
  • YV12和YUV_420_888:这是最奇怪的结果 - 我可以看到获得灰度图像,但是它是垂直翻转的(是的,翻转,没有旋转!)并且显着地被压扁(水平缩放,但不是垂直缩放).

我在这里错过了什么?是什么导致图像被翻转和压扁?如何获得几何正确的灰度缓冲区?我应该使用不同类型的表面(而不是ImageReader)吗?

任何提示赞赏.

Ste*_* G. 5

我找到了一个解释(虽然不一定是一个令人满意的解决方案):事实证明传感器阵列的纵横比是16:9(通过发现mCameraInfo.get(CameraCharacteristics.SENSOR_INFO_ACTIVE_ARRAY_SIZE);).

至少在请求YV12/YUV_420_888时,拖缆似乎不以任何方式裁剪图像,而是非均匀地缩放图像以达到所请求的帧尺寸.在请求16:9格式时,图像具有正确的比例(不幸的是,其中只有两个高分辨率格式).对我来说似乎有点奇怪 - 在请求JPEG,或使用等效的旧相机API函数或静止图像时似乎不会发生这种情况; 我不确定非均匀缩放的帧会有什么用处.

我觉得这不是一个真正令人满意的解决方案,因为这意味着你不能依赖输出格式列表,而是必须首先找到传感器大小,找到具有相同宽高比的格式,然后自己对图像进行下采样(如所须)...

我不知道这是否是这里的预期结果或S5的"特征".意见或建议仍然欢迎.