美文网首页C/CPLUS
Android 平台下基于AnativeWindow实现相机预览

Android 平台下基于AnativeWindow实现相机预览

作者: 码农小白two | 来源:发表于2020-10-26 09:33 被阅读0次

1、AnativeWindow 官方解释为:

  • ANativeWindow represents the producer end of an image queue.
  • It is the C counterpart of the android.view.Surface object in Java,
  • and can be converted both ways. Depending on the consumer, images
  • submitted to ANativeWindow can be shown on the display or sent to
  • other consumers, such as video encoders.
    谷歌翻译:ANativeWindow表示图像队列的生产者端。它是Java中android.view.Surface对象的C对应物,并且可以双向转换。 取决于消费者,提交到ANativeWindow的图像可以显示在显示器上或发送给其他消费者,例如视频编码器。
    因此可以理解的是 Java中可以提供一个surface通过NDK的方式利用AnativeWindow的API实现图像的绘制。
    2、常规的相机数据处理并显示的方案:
    ①、利用OpenGL ES 中的shader语言对图像进行处理;(高效也推荐这个方法)
    ②、将camera的数据传到底层 进行识别,上层再利用cavans或者其它去绘制我们想表达的东西,例如:框框、文字等。
    ③、利用AnativeWindow API去实现相机图像数据的处理,这种做法存在一定的局限性,对于一些本身处理比较复杂的算法,图像的显示和绘制会极为的卡顿。但是AnativeWindow常常用于Video的显示,算是Android多媒体中一个比较有用的API。
    3、AnativeWindow显示流程:
    ①、Camera2 通过ImageReader方式得到相机data;
    ②、将所得到的Yuvdata通过JNI传到native层;
    ③、native中将Yuvdata进行解码转成AnativeWindow可以用来绘制的图像格式;
    AnativeWindow所支持的图像数据格式:
enum ANativeWindow_LegacyFormat {
    // NOTE: these values must match the values from graphics/common/x.x/types.hal

    /** Red: 8 bits, Green: 8 bits, Blue: 8 bits, Alpha: 8 bits. **/
    WINDOW_FORMAT_RGBA_8888          = AHARDWAREBUFFER_FORMAT_R8G8B8A8_UNORM,
    /** Red: 8 bits, Green: 8 bits, Blue: 8 bits, Unused: 8 bits. **/
    WINDOW_FORMAT_RGBX_8888          = AHARDWAREBUFFER_FORMAT_R8G8B8X8_UNORM,
    /** Red: 5 bits, Green: 6 bits, Blue: 5 bits. **/
    WINDOW_FORMAT_RGB_565            = AHARDWAREBUFFER_FORMAT_R5G6B5_UNORM,
};

4、关键代码
Java:

private ImageReader.OnImageAvailableListener mOnImageAvailableListener = new ImageReader.OnImageAvailableListener() {

        /*
         *  The following method will be called every time an image is ready
         *  be sure to use method acquireNextImage() and then close(), otherwise, the display may STOP
         */
        @Override
        public void onImageAvailable(ImageReader reader) {
            // get the newest frame
            Image image = reader.acquireNextImage();

            if (image == null) {
                return;
            }
            final byte[] yuvdata=ImageUtil.getBytesFromImageAsType(image,0);
            draw(image.getWidth(),image.getHeight(),yuvdata,surface);
            Log.d("pctest","yuvdata length:"+yuvdata.length+"image height:"+image.getHeight()+"image width"+image.getWidth());
            image.close();
        }
    };

private void startPreview(CameraDevice mCamera) throws CameraAccessException {
        SurfaceTexture texture = mPreviewView.getSurfaceTexture();
        texture.setDefaultBufferSize(mPreviewSize.getWidth(),mPreviewSize.getHeight());
        surface = new Surface(texture);
        try {
            // to set request for PREVIEW
            mPreviewBuilder = mCamera.createCaptureRequest(CameraDevice.TEMPLATE_PREVIEW);
        } catch (CameraAccessException e) {
            e.printStackTrace();
        }
        mImageReader = ImageReader.newInstance(mImageWidth,mImageHeight, ImageFormat.YUV_420_888,2);
        mImageReader.setOnImageAvailableListener(mOnImageAvailableListener,mHandler);
        mPreviewBuilder.addTarget(mImageReader.getSurface());
        List<Surface> outputSurfaces = new ArrayList<>();
        outputSurfaces.add(mImageReader.getSurface());
        mCamera.createCaptureSession(outputSurfaces, mSessionStateCallback, mHandler);
    }

Native:
在对数据处理中利用的 OpenCV 进行图像数据转码,只是为了方便= 。=(PS:转码这是太。。。)

extern "C"
JNIEXPORT void JNICALL
Java_com_example_anativewindow4camera2_MainActivity_draw(JNIEnv *env, jobject thiz, jint width,
                                                         jint height, jbyteArray yuvdata,
                                                         jobject surface) {
    // TODO: implement draw()
    jbyte *data = env->GetByteArrayElements(yuvdata, NULL);
    Mat src(height + height / 2, width, CV_8UC1, data);
    cvtColor(src, src, COLOR_YUV2RGBA_IYUV);
    cv::transpose(src,src);
    cv::flip(src,src,1);
    ANativeWindow * window = ANativeWindow_fromSurface(env, surface);
    ANativeWindow_acquire(window);
    ANativeWindow_Buffer buffer;
    ANativeWindow_setBuffersGeometry(window, src.cols, src.rows, WINDOW_FORMAT_RGBA_8888);
    if (int32_t err = ANativeWindow_lock(window, &buffer, NULL)) {
        LOGE("ANativeWindow_lock failed with error code: %d\n", err);
        ANativeWindow_release(window);
    }
    uint8_t * outPtr = reinterpret_cast<uint8_t *>(buffer.bits);
    int dst_line_size = buffer.stride * 4;
    //一行一行拷贝
    for (int i = 0; i < buffer.height; ++i) {
        //void *memcpy(void *dest, const void *src, size_t n);
        //从源src所指的内存地址的起始位置开始拷贝n个字节到目标dest所指的内存地址的起始位置中
        memcpy(outPtr + i * dst_line_size,
               src.data + i * src.cols * 4, dst_line_size);
    }

    ANativeWindow_unlockAndPost(window);
    ANativeWindow_release(window);
    src.release();
    env->ReleaseByteArrayElements(yuvdata, data, 0);
}

5、记录一下,怕后面忘了。。。

相关文章

网友评论

    本文标题:Android 平台下基于AnativeWindow实现相机预览

    本文链接:https://www.haomeiwen.com/subject/kmaomktx.html