From ac7e16fdf6bbeb079d6b2007adb1ca77cf6d4a51 Mon Sep 17 00:00:00 2001 From: Andrey Kamaev Date: Mon, 25 Jul 2011 12:32:33 +0000 Subject: [PATCH] Fixed camera output format handling --- .../armeabi-v7a/libnative_camera_r2.2.2.so | Bin 121724 -> 121724 bytes .../armeabi-v7a/libnative_camera_r2.3.3.so | Bin 123012 -> 123012 bytes .../lib/armeabi/libnative_camera_r2.2.2.so | Bin 107048 -> 107048 bytes .../lib/armeabi/libnative_camera_r2.3.3.so | Bin 109424 -> 109424 bytes .../camera_wrapper/camera_wrapper.cpp | 2 +- modules/androidcamera/src/camera_activity.cpp | 6 +- .../include/opencv2/highgui/highgui_c.h | 3 +- modules/highgui/src/cap_android.cpp | 99 ++++++++++++------ .../include/opencv2/imgproc/imgproc.hpp | 2 + .../imgproc/include/opencv2/imgproc/types_c.h | 2 + modules/imgproc/src/color.cpp | 52 +++++---- .../opencv/samples/tutorial1/Sample1View.java | 2 +- .../opencv/samples/tutorial4/Sample4View.java | 4 +- 13 files changed, 109 insertions(+), 63 deletions(-) diff --git a/3rdparty/lib/armeabi-v7a/libnative_camera_r2.2.2.so b/3rdparty/lib/armeabi-v7a/libnative_camera_r2.2.2.so index 0a1bc5e12b39303e88e0d6471d57c2f3a9d50baa..b36b97a36a8bcfbfc7b8a8d334802a8f11877e61 100644 GIT binary patch delta 20 ccmeyfjs4Fy_6<&F7>k>o&un)-!)VD50B-OJMgRZ+ delta 20 ccmeyfjs4Fy_6<&F7>%2q&un)-!)VD50Bmpx1^@s6 diff --git a/3rdparty/lib/armeabi-v7a/libnative_camera_r2.3.3.so b/3rdparty/lib/armeabi-v7a/libnative_camera_r2.3.3.so index e518e8d4ec15c38b33f6db0eff42f02c7f266e0d..28c567672b5631b40494319bc9c60af3ed3a3ad6 100644 GIT binary patch delta 20 ccmZoU$lh|0eM8bY#^UDWbK8^8F)9iH0AgAQy#N3J delta 20 ccmZoU$lh|0eM8bYM&st>bK8^8F)9iH0AJb&eE(^b diff --git a/3rdparty/lib/armeabi/libnative_camera_r2.3.3.so b/3rdparty/lib/armeabi/libnative_camera_r2.3.3.so index 56632749160366de8dfb24ba866537ae39cad9bb..af006de71bddb1f137c8456854698b65bb5e2947 100755 GIT binary patch delta 16 YcmexxjqSrVwhgP!F&1xLeeUQT08?fP0ssI2 delta 16 YcmexxjqSrVwhgP!F&b}PeeUQT08!=%!TgetProperty(ANDROID_CAMERA_PROPERTY_SUPPORTED_PREVIEW_SIZES_STRING); + case CV_CAP_PROP_PREVIEW_FORMAT: + return (double)m_activity->getProperty(ANDROID_CAMERA_PROPERTY_PREVIEW_FORMAT_STRING); default: CV_Error( CV_StsOutOfRange, "Failed attempt to GET unsupported camera property." ); break; @@ -318,9 +331,9 @@ bool CvCapture_Android::grabFrame() if (m_dataState == CVCAPTURE_ANDROID_STATE_HAS_NEW_FRAME_UNGRABBED) { //LOGD("CvCapture_Android::grabFrame: get new frame"); //swap current and new frames - unsigned char* tmp = m_frameYUV420i; - m_frameYUV420i = m_frameYUV420inext; - m_frameYUV420inext = tmp; + unsigned char* tmp = m_frameYUV420; + m_frameYUV420 = m_frameYUV420next; + m_frameYUV420next = tmp; //discard cached frames m_hasGray = false; @@ -348,27 +361,39 @@ IplImage* CvCapture_Android::retrieveFrame( int outputType ) { IplImage* image = NULL; - unsigned char *current_frameYUV420i=m_frameYUV420i; + unsigned char *current_frameYUV420=m_frameYUV420; //Attention! all the operations in this function below should occupy less time than the period between two frames from camera - if (NULL != current_frameYUV420i) + if (NULL != current_frameYUV420) { + if (m_frameFormat == noformat) + { + union {double prop; const char* name;} u; + u.prop = getProperty(CV_CAP_PROP_PREVIEW_FORMAT); + if (0 == strcmp(u.name, "yuv420sp")) + m_frameFormat = yuv420sp; + else if (0 == strcmp(u.name, "yuv420i")) + m_frameFormat = yuv420i; + else + m_frameFormat = yuvUnknown; + } + switch(outputType) { case CV_CAP_ANDROID_GREY_FRAME: if (!m_hasGray) - if (!(m_hasGray = convertYUV420i2Grey(m_width, m_height, current_frameYUV420i, m_frameGray.mat))) + if (!(m_hasGray = convertYUV2Grey(m_width, m_height, current_frameYUV420, m_frameGray.mat))) return NULL; image = m_frameGray.getIplImagePtr(); break; case CV_CAP_ANDROID_COLOR_FRAME_BGR: case CV_CAP_ANDROID_COLOR_FRAME_RGB: if (!m_hasColor) - if (!(m_hasColor = convertYUV420i2BGR(m_width, m_height, current_frameYUV420i, m_frameColor.mat, outputType == CV_CAP_ANDROID_COLOR_FRAME_RGB, false))) + if (!(m_hasColor = convertYUV2BGR(m_width, m_height, current_frameYUV420, m_frameColor.mat, outputType == CV_CAP_ANDROID_COLOR_FRAME_RGB, false))) return NULL; image = m_frameColor.getIplImagePtr(); break; case CV_CAP_ANDROID_COLOR_FRAME_BGRA: case CV_CAP_ANDROID_COLOR_FRAME_RGBA: if (!m_hasColor) - if (!(m_hasColor = convertYUV420i2BGR(m_width, m_height, current_frameYUV420i, m_frameColor.mat, outputType == CV_CAP_ANDROID_COLOR_FRAME_RGBA, true))) + if (!(m_hasColor = convertYUV2BGR(m_width, m_height, current_frameYUV420, m_frameColor.mat, outputType == CV_CAP_ANDROID_COLOR_FRAME_RGBA, true))) return NULL; image = m_frameColor.getIplImagePtr(); break; @@ -391,22 +416,22 @@ void CvCapture_Android::setFrame(const void* buffer, int bufferSize) if ( expectedSize != bufferSize) { - LOGE("ERROR reading YUV420i buffer: width=%d, height=%d, size=%d, receivedSize=%d", width, height, expectedSize, bufferSize); + LOGE("ERROR reading YUV buffer: width=%d, height=%d, size=%d, receivedSize=%d", width, height, expectedSize, bufferSize); return; } //allocate memory if needed - prepareCacheForYUV420i(width, height); + prepareCacheForYUV(width, height); //copy data - memcpy(m_frameYUV420inext, buffer, bufferSize); + memcpy(m_frameYUV420next, buffer, bufferSize); //LOGD("CvCapture_Android::setFrame -- memcpy is done"); #if 0 //moved this part of code into grabFrame //swap current and new frames - unsigned char* tmp = m_frameYUV420i; - m_frameYUV420i = m_frameYUV420inext; - m_frameYUV420inext = tmp; + unsigned char* tmp = m_frameYUV420; + m_frameYUV420 = m_frameYUV420next; + m_frameYUV420next = tmp; //discard cached frames m_hasGray = false; @@ -418,30 +443,31 @@ void CvCapture_Android::setFrame(const void* buffer, int bufferSize) } //Attention: this method should be called inside pthread_mutex_lock(m_nextFrameMutex) only -void CvCapture_Android::prepareCacheForYUV420i(int width, int height) +void CvCapture_Android::prepareCacheForYUV(int width, int height) { if (width != m_width || height != m_height) { - LOGD("CvCapture_Android::prepareCacheForYUV420i: Changing size of buffers: from width=%d height=%d to width=%d height=%d", m_width, m_height, width, height); + LOGD("CvCapture_Android::prepareCacheForYUV: Changing size of buffers: from width=%d height=%d to width=%d height=%d", m_width, m_height, width, height); m_width = width; m_height = height; - unsigned char *tmp = m_frameYUV420inext; - m_frameYUV420inext = new unsigned char [width * height * 3 / 2]; + unsigned char *tmp = m_frameYUV420next; + m_frameYUV420next = new unsigned char [width * height * 3 / 2]; if (tmp != NULL) { delete[] tmp; } - tmp = m_frameYUV420i; - m_frameYUV420i = new unsigned char [width * height * 3 / 2]; + tmp = m_frameYUV420; + m_frameYUV420 = new unsigned char [width * height * 3 / 2]; if (tmp != NULL) { delete[] tmp; } } } -bool CvCapture_Android::convertYUV420i2Grey(int width, int height, const unsigned char* yuv, cv::Mat& resmat) +bool CvCapture_Android::convertYUV2Grey(int width, int height, const unsigned char* yuv, cv::Mat& resmat) { if (yuv == 0) return false; + if (m_frameFormat != yuv420sp && m_frameFormat != yuv420i) return false; #define ALWAYS_COPY_GRAY 0 #if ALWAYS_COPY_GRAY resmat.create(height, width, CV_8UC1); @@ -453,14 +479,19 @@ bool CvCapture_Android::convertYUV420i2Grey(int width, int height, const unsigne return !resmat.empty(); } -bool CvCapture_Android::convertYUV420i2BGR(int width, int height, const unsigned char* yuv, cv::Mat& resmat, bool inRGBorder, bool withAlpha) +bool CvCapture_Android::convertYUV2BGR(int width, int height, const unsigned char* yuv, cv::Mat& resmat, bool inRGBorder, bool withAlpha) { if (yuv == 0) return false; + if (m_frameFormat != yuv420sp && m_frameFormat != yuv420i) return false; + CV_Assert(width % 2 == 0 && height % 2 == 0); cv::Mat src(height*3/2, width, CV_8UC1, (void*)yuv); - cv::cvtColor(src, resmat, inRGBorder ? CV_YUV420i2RGB : CV_YUV420i2BGR, withAlpha ? 4 : 3); + if (m_frameFormat == yuv420sp) + cv::cvtColor(src, resmat, inRGBorder ? CV_YUV420sp2RGB : CV_YUV420sp2BGR, withAlpha ? 4 : 3); + else if (m_frameFormat == yuv420i) + cv::cvtColor(src, resmat, inRGBorder ? CV_YUV420i2RGB : CV_YUV420i2BGR, withAlpha ? 4 : 3); return !resmat.empty(); } diff --git a/modules/imgproc/include/opencv2/imgproc/imgproc.hpp b/modules/imgproc/include/opencv2/imgproc/imgproc.hpp index 38528c044..3879b501e 100644 --- a/modules/imgproc/include/opencv2/imgproc/imgproc.hpp +++ b/modules/imgproc/include/opencv2/imgproc/imgproc.hpp @@ -902,6 +902,8 @@ enum COLOR_YUV420i2RGB = 90, COLOR_YUV420i2BGR = 91, + COLOR_YUV420sp2RGB = 92, + COLOR_YUV420sp2BGR = 93, COLOR_COLORCVT_MAX =100 }; diff --git a/modules/imgproc/include/opencv2/imgproc/types_c.h b/modules/imgproc/include/opencv2/imgproc/types_c.h index 7c3b5a37e..5a984fd62 100644 --- a/modules/imgproc/include/opencv2/imgproc/types_c.h +++ b/modules/imgproc/include/opencv2/imgproc/types_c.h @@ -228,6 +228,8 @@ enum CV_YUV420i2RGB = 90, CV_YUV420i2BGR = 91, + CV_YUV420sp2RGB = 92, + CV_YUV420sp2BGR = 93, CV_COLORCVT_MAX =100 }; diff --git a/modules/imgproc/src/color.cpp b/modules/imgproc/src/color.cpp index b9748c8a0..b7e0360c5 100644 --- a/modules/imgproc/src/color.cpp +++ b/modules/imgproc/src/color.cpp @@ -2648,16 +2648,16 @@ static void Bayer2RGB_VNG_8u( const Mat& srcmat, Mat& dstmat, int code ) } } -///////////////////////////////////// YUV420i -> RGB ///////////////////////////////////// +///////////////////////////////////// YUV420 -> RGB ///////////////////////////////////// -template -struct YUV420i2BGR888Invoker +template +struct YUV4202BGR888Invoker { Mat* dst; const uchar* my1, *muv; int width; - YUV420i2BGR888Invoker(Mat& _dst, int _width, const uchar* _y1, const uchar* _uv) + YUV4202BGR888Invoker(Mat& _dst, int _width, const uchar* _y1, const uchar* _uv) : dst(&_dst), my1(_y1), muv(_uv), width(_width) {} void operator()(const BlockedRange& range) const @@ -2676,8 +2676,8 @@ struct YUV420i2BGR888Invoker for(int i = 0; i < width; i+=2,row1+=6,row2+=6) { - int cr = uv[i] - 128; - int cb = uv[i+1] - 128; + int cr = uv[i + SPorI + 0] - 128; + int cb = uv[i - SPorI + 1] - 128; int ruv = 409 * cr + 128; int guv = 128 - 100 * cb - 208 * cr; @@ -2707,14 +2707,14 @@ struct YUV420i2BGR888Invoker } }; -template -struct YUV420i2BGRA8888Invoker +template +struct YUV4202BGRA8888Invoker { Mat* dst; const uchar* my1, *muv; int width; - YUV420i2BGRA8888Invoker(Mat& _dst, int _width, const uchar* _y1, const uchar* _uv) + YUV4202BGRA8888Invoker(Mat& _dst, int _width, const uchar* _y1, const uchar* _uv) : dst(&_dst), my1(_y1), muv(_uv), width(_width) {} void operator()(const BlockedRange& range) const @@ -2733,8 +2733,8 @@ struct YUV420i2BGRA8888Invoker for(int i = 0; i < width; i+=2,row1+=8,row2+=8) { - int cr = uv[i] - 128; - int cb = uv[i+1] - 128; + int cr = uv[i + SPorI + 0] - 128; + int cb = uv[i - SPorI + 1] - 128; int ruv = 409 * cr + 128; int guv = 128 - 100 * cb - 208 * cr; @@ -3112,7 +3112,7 @@ void cv::cvtColor( InputArray _src, OutputArray _dst, int code, int dcn ) Bayer2RGB_VNG_8u(src, dst, code); } break; - case CV_YUV420i2BGR: case CV_YUV420i2RGB: + case CV_YUV420sp2BGR: case CV_YUV420sp2RGB: case CV_YUV420i2BGR: case CV_YUV420i2RGB: { if(dcn <= 0) dcn = 3; CV_Assert( dcn == 3 || dcn == 4 ); @@ -3126,22 +3126,36 @@ void cv::cvtColor( InputArray _src, OutputArray _dst, int code, int dcn ) const uchar* uv = y + dstSz.area(); #ifdef HAVE_TEGRA_OPTIMIZATION - if (!tegra::YUV420i2BGR(y, uv, dst, CV_YUV420i2RGB == code)) + if (!tegra::YUV420i2BGR(y, uv, dst, CV_YUV420sp2RGB == code)) #endif { - if (CV_YUV420i2RGB == code) + if (CV_YUV420sp2RGB == code) { if (dcn == 3) - parallel_for(BlockedRange(0, dstSz.height, 2), YUV420i2BGR888Invoker<2>(dst, dstSz.width, y, uv)); + parallel_for(BlockedRange(0, dstSz.height, 2), YUV4202BGR888Invoker<2,0>(dst, dstSz.width, y, uv)); else - parallel_for(BlockedRange(0, dstSz.height, 2), YUV420i2BGRA8888Invoker<2>(dst, dstSz.width, y, uv)); + parallel_for(BlockedRange(0, dstSz.height, 2), YUV4202BGRA8888Invoker<2,0>(dst, dstSz.width, y, uv)); } - else + else if (CV_YUV420sp2BGR == code) { if (dcn == 3) - parallel_for(BlockedRange(0, dstSz.height, 2), YUV420i2BGR888Invoker<0>(dst, dstSz.width, y, uv)); + parallel_for(BlockedRange(0, dstSz.height, 2), YUV4202BGR888Invoker<0,0>(dst, dstSz.width, y, uv)); else - parallel_for(BlockedRange(0, dstSz.height, 2), YUV420i2BGRA8888Invoker<0>(dst, dstSz.width, y, uv)); + parallel_for(BlockedRange(0, dstSz.height, 2), YUV4202BGRA8888Invoker<0,0>(dst, dstSz.width, y, uv)); + } + else if (CV_YUV420i2RGB == code) + { + if (dcn == 3) + parallel_for(BlockedRange(0, dstSz.height, 2), YUV4202BGR888Invoker<2,1>(dst, dstSz.width, y, uv)); + else + parallel_for(BlockedRange(0, dstSz.height, 2), YUV4202BGRA8888Invoker<2,1>(dst, dstSz.width, y, uv)); + } + else if (CV_YUV420i2BGR == code) + { + if (dcn == 3) + parallel_for(BlockedRange(0, dstSz.height, 2), YUV4202BGR888Invoker<0,1>(dst, dstSz.width, y, uv)); + else + parallel_for(BlockedRange(0, dstSz.height, 2), YUV4202BGRA8888Invoker<0,1>(dst, dstSz.width, y, uv)); } } } diff --git a/samples/android/tutorial-1-addopencv/src/org/opencv/samples/tutorial1/Sample1View.java b/samples/android/tutorial-1-addopencv/src/org/opencv/samples/tutorial1/Sample1View.java index 81d2e754c..3cc1017c7 100644 --- a/samples/android/tutorial-1-addopencv/src/org/opencv/samples/tutorial1/Sample1View.java +++ b/samples/android/tutorial-1-addopencv/src/org/opencv/samples/tutorial1/Sample1View.java @@ -45,7 +45,7 @@ class Sample1View extends SampleViewBase { Imgproc.cvtColor(mGraySubmat, mRgba, Imgproc.COLOR_GRAY2RGBA, 4); break; case Sample1Java.VIEW_MODE_RGBA: - Imgproc.cvtColor(mYuv, mRgba, Imgproc.COLOR_YUV420i2RGB, 4); + Imgproc.cvtColor(mYuv, mRgba, Imgproc.COLOR_YUV420sp2RGB, 4); Core.putText(mRgba, "OpenCV + Android", new Point(10, 100), 3/* CV_FONT_HERSHEY_COMPLEX */, 2, new Scalar(255, 0, 0, 255), 3); break; case Sample1Java.VIEW_MODE_CANNY: diff --git a/samples/android/tutorial-4-mixed/src/org/opencv/samples/tutorial4/Sample4View.java b/samples/android/tutorial-4-mixed/src/org/opencv/samples/tutorial4/Sample4View.java index c589f6df6..ee771839f 100644 --- a/samples/android/tutorial-4-mixed/src/org/opencv/samples/tutorial4/Sample4View.java +++ b/samples/android/tutorial-4-mixed/src/org/opencv/samples/tutorial4/Sample4View.java @@ -42,14 +42,14 @@ class Sample4View extends SampleViewBase { Imgproc.cvtColor(mGraySubmat, mRgba, Imgproc.COLOR_GRAY2RGBA, 4); break; case Sample4Mixed.VIEW_MODE_RGBA: - Imgproc.cvtColor(mYuv, mRgba, Imgproc.COLOR_YUV420i2RGB, 4); + Imgproc.cvtColor(mYuv, mRgba, Imgproc.COLOR_YUV420sp2RGB, 4); break; case Sample4Mixed.VIEW_MODE_CANNY: Imgproc.Canny(mGraySubmat, mIntermediateMat, 80, 100); Imgproc.cvtColor(mIntermediateMat, mRgba, Imgproc.COLOR_GRAY2BGRA, 4); break; case Sample4Mixed.VIEW_MODE_FEATURES: - Imgproc.cvtColor(mYuv, mRgba, Imgproc.COLOR_YUV420i2RGB, 4); + Imgproc.cvtColor(mYuv, mRgba, Imgproc.COLOR_YUV420sp2RGB, 4); FindFeatures(mGraySubmat.getNativeObjAddr(), mRgba.getNativeObjAddr()); break; }