当前位置:   article > 正文

OpenCV处理工业相机Bayer格式数据_opencv bayer

opencv bayer

        工业相机为了传输效率和数据的真实性,很多都是用Raw格式,比如从某宝购买的这台JHEM203GC的相机,就只支持Bayer RG8, Bayer RG10数据。

1 Bayer图像

        Bayer图像是彩色Bayer传感器产生的图像数据, 这种传感器每个像素只记录了红,绿,蓝一个分量的数据,如下图所示,按照排列方式分为RGGB,BGGR, GRBG, GBRG四种排列。

        对相机返回的这种Bayer数据,如果需要显示为正常的彩色图像,需要将每个像素丢失的信息插补回来,一般是利用相邻和附近的像素信息进行插值,有许多不同的算法得到的效果也不尽相同。

2 图像转RGB或者BGR

       如调用MVS SDK,可以通过MV_CC_ConvertPixelType实现,代码如下,转换的格式通常为PixelType_Gvsp_BGR8_Packed,就是BGR排列,如果是RGB排列,使用PixelType_Gvsp_RGB8_Packed,这种格式保存为图片或者显示时可能会出现红蓝对调的情况。

  1. int Bayer2BGR(void *handle, unsigned char* pData, MV_FRAME_OUT_INFO_EX* pstImageInfo, unsigned char **pBgrData)
  2. {
  3. *pBgrData = 0;
  4. if (NULL == pData)
  5. {
  6. return MV_E_PARAMETER;
  7. }
  8. unsigned int nDataSizeForBGR = pstImageInfo->nWidth * pstImageInfo->nHeight * 3;
  9. unsigned char* pDataForBGR = (unsigned char*)malloc(nDataSizeForBGR);
  10. if (NULL == pDataForBGR)
  11. {
  12. return MV_E_BUFOVER;
  13. }
  14. // ch:像素格式转换 | en:Convert pixel format
  15. MV_CC_PIXEL_CONVERT_PARAM stConvertParam = { 0 };
  16. memset(&stConvertParam, 0, sizeof(MV_CC_PIXEL_CONVERT_PARAM));
  17. stConvertParam.nWidth = pstImageInfo->nWidth; //ch:图像宽 | en:image width
  18. stConvertParam.nHeight = pstImageInfo->nHeight; //ch:图像高 | en:image height
  19. stConvertParam.pSrcData = pData; //ch:输入数据缓存 | en:input data buffer
  20. stConvertParam.nSrcDataLen = pstImageInfo->nFrameLen; //ch:输入数据大小 | en:input data size
  21. stConvertParam.enSrcPixelType = pstImageInfo->enPixelType; //ch:输入像素格式 | en:input pixel format
  22. stConvertParam.enDstPixelType = PixelType_Gvsp_BGR8_Packed; //ch:输出像素格式 | en:output pixel format
  23. stConvertParam.pDstBuffer = pDataForBGR; //ch:输出数据缓存 | en:output data buffer
  24. stConvertParam.nDstBufferSize = nDataSizeForBGR; //ch:输出缓存大小 | en:output buffer size
  25. int nRet = MV_CC_ConvertPixelType(handle, &stConvertParam);
  26. if (MV_OK != nRet)
  27. {
  28. free(pDataForBGR);
  29. return MV_E_UNKNOW;
  30. }
  31. *pBgrData = pDataForBGR;
  32. return MV_OK;
  33. }

3 将buffer转成Mat格式

        采集的图像为直接数组,需要转换成Mat格式来调用OpenCV做进一步的处理。

  1. // convert data stream in Mat format
  2. bool Convert2Mat(void* handle, MV_FRAME_OUT_INFO_EX* pstImageInfo, unsigned char **pData)
  3. {
  4. cv::Mat srcImage;
  5. if ( pstImageInfo->enPixelType == PixelType_Gvsp_Mono8 )
  6. {
  7. srcImage = cv::Mat(pstImageInfo->nHeight, pstImageInfo->nWidth, CV_8UC1, *pData);
  8. }
  9. else if ( pstImageInfo->enPixelType == PixelType_Gvsp_RGB8_Packed )
  10. {
  11. RGB2BGR(*pData, pstImageInfo->nWidth, pstImageInfo->nHeight);
  12. srcImage = cv::Mat(pstImageInfo->nHeight, pstImageInfo->nWidth, CV_8UC3, *pData);
  13. }
  14. else
  15. {
  16. unsigned char *pBgrData = 0;
  17. Bayer2BGR(handle, *pData, pstImageInfo, &pBgrData);
  18. if (pBgrData)
  19. {
  20. srcImage = cv::Mat(pstImageInfo->nHeight, pstImageInfo->nWidth, CV_8UC3, pBgrData);
  21. free(*pData);
  22. *pData = pBgrData;
  23. }
  24. }
  25. if ( NULL == srcImage.data )
  26. {
  27. return false;
  28. }
  29. #ifdef SAVE
  30. //save converted image in a local file
  31. try {
  32. #if defined (VC9_COMPILE)
  33. cvSaveImage("MatImage.bmp", &(IplImage(srcImage)));
  34. #else
  35. cv::imwrite("MatImage.bmp", srcImage);
  36. #endif
  37. }
  38. catch (cv::Exception& ex) {
  39. fprintf(stderr, "Exception saving image to bmp format: %s\n", ex.what());
  40. }
  41. #endif
  42. //show converted image
  43. try {
  44. #if !defined (VC9_COMPILE)
  45. cvShowImage("CAM", &(IplImage(srcImage)));
  46. cvWaitKey(20);
  47. #else
  48. cv::imshow("CAM", srcImage);
  49. cv::waitKey(20);
  50. #endif
  51. }
  52. catch (cv::Exception& ex) {
  53. fprintf(stderr, "Exception saving image to bmp format: %s\n", ex.what());
  54. }
  55. srcImage.release();
  56. return true;
  57. }

 4 cvtColor一步转换

        使用相机SDK的好处是集成简单,但对于本身需要使用Opencv的工程来说也可以直接使用Opencv的cvtColor, 按照原有的bayer格式选择参数,即可一步把RAW图像转为RGB图像。

  1. cv::Mat bayerImg = cv::Mat(stImageInfo.nHeight, stImageInfo.nWidth, CV_8UC1, pData);
  2. cv::Mat bgrImg;
  3. if(stImageInfo.enPixelType == PixelType_Gvsp_BayerBG8) cv::cvtColor(bayerImg, bgrImg, cv::COLOR_BayerBG2RGB);
  4. else if(stImageInfo.enPixelType == PixelType_Gvsp_BayerRG8) cv::cvtColor(bayerImg, bgrImg, cv::COLOR_BayerRG2RGB);
  5. else if (stImageInfo.enPixelType == PixelType_Gvsp_BayerGB8) cv::cvtColor(bayerImg, bgrImg, cv::COLOR_BayerGB2RGB);
  6. else if (stImageInfo.enPixelType == PixelType_Gvsp_BayerGR8) cv::cvtColor(bayerImg, bgrImg, cv::COLOR_BayerGR2RGB);
  7. cv::imshow("CAM", bgrImg);
  8. cv::waitKey(20);

5 结果和参考代码

        RAW图像,看上去是黑白的实际上是马赛克图像

       转换后的RGB图像

        参考代码:https://www.jinghangtech.com/media/ug/UG007_OpenCV.zip

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/寸_铁/article/detail/839695
推荐阅读
相关标签
  

闽ICP备14008679号