OpenCV处理工业相机Bayer格式数据
创作时间:
作者:
@小白创作中心
OpenCV处理工业相机Bayer格式数据
引用
CSDN
1.
https://blog.csdn.net/winyio8/article/details/136471510
工业相机为了传输效率和数据的真实性,很多都是用Raw格式,比如从某宝购买的这台JHEM203GC的相机,就只支持Bayer RG8, Bayer RG10数据。本文将详细介绍如何使用OpenCV处理这种Bayer格式数据,包括图像转换、Mat格式转换以及一步转换等方法。
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,这种格式保存为图片或者显示时可能会出现红蓝对调的情况。
int Bayer2BGR(void *handle, unsigned char* pData, MV_FRAME_OUT_INFO_EX* pstImageInfo, unsigned char **pBgrData)
{
*pBgrData = 0;
if (NULL == pData)
{
return MV_E_PARAMETER;
}
unsigned int nDataSizeForBGR = pstImageInfo->nWidth * pstImageInfo->nHeight * 3;
unsigned char* pDataForBGR = (unsigned char*)malloc(nDataSizeForBGR);
if (NULL == pDataForBGR)
{
return MV_E_BUFOVER;
}
// ch:像素格式转换 | en:Convert pixel format
MV_CC_PIXEL_CONVERT_PARAM stConvertParam = { 0 };
memset(&stConvertParam, 0, sizeof(MV_CC_PIXEL_CONVERT_PARAM));
stConvertParam.nWidth = pstImageInfo->nWidth; //ch:图像宽 | en:image width
stConvertParam.nHeight = pstImageInfo->nHeight; //ch:图像高 | en:image height
stConvertParam.pSrcData = pData; //ch:输入数据缓存 | en:input data buffer
stConvertParam.nSrcDataLen = pstImageInfo->nFrameLen; //ch:输入数据大小 | en:input data size
stConvertParam.enSrcPixelType = pstImageInfo->enPixelType; //ch:输入像素格式 | en:input pixel format
stConvertParam.enDstPixelType = PixelType_Gvsp_BGR8_Packed; //ch:输出像素格式 | en:output pixel format
stConvertParam.pDstBuffer = pDataForBGR; //ch:输出数据缓存 | en:output data buffer
stConvertParam.nDstBufferSize = nDataSizeForBGR; //ch:输出缓存大小 | en:output buffer size
int nRet = MV_CC_ConvertPixelType(handle, &stConvertParam);
if (MV_OK != nRet)
{
free(pDataForBGR);
return MV_E_UNKNOW;
}
*pBgrData = pDataForBGR;
return MV_OK;
}
3 将buffer转成Mat格式
采集的图像为直接数组,需要转换成Mat格式来调用OpenCV做进一步的处理。
bool Convert2Mat(void* handle, MV_FRAME_OUT_INFO_EX* pstImageInfo, unsigned char **pData)
{
cv::Mat srcImage;
if ( pstImageInfo->enPixelType == PixelType_Gvsp_Mono8 )
{
srcImage = cv::Mat(pstImageInfo->nHeight, pstImageInfo->nWidth, CV_8UC1, *pData);
}
else if ( pstImageInfo->enPixelType == PixelType_Gvsp_RGB8_Packed )
{
RGB2BGR(*pData, pstImageInfo->nWidth, pstImageInfo->nHeight);
srcImage = cv::Mat(pstImageInfo->nHeight, pstImageInfo->nWidth, CV_8UC3, *pData);
}
else
{
unsigned char *pBgrData = 0;
Bayer2BGR(handle, *pData, pstImageInfo, &pBgrData);
if (pBgrData)
{
srcImage = cv::Mat(pstImageInfo->nHeight, pstImageInfo->nWidth, CV_8UC3, pBgrData);
free(*pData);
*pData = pBgrData;
}
}
if ( NULL == srcImage.data )
{
return false;
}
#ifdef SAVE
//save converted image in a local file
try {
#if defined (VC9_COMPILE)
cvSaveImage("MatImage.bmp", &(IplImage(srcImage)));
#else
cv::imwrite("MatImage.bmp", srcImage);
#endif
}
catch (cv::Exception& ex) {
fprintf(stderr, "Exception saving image to bmp format: %s\n", ex.what());
}
#endif
//show converted image
try {
#if !defined (VC9_COMPILE)
cvShowImage("CAM", &(IplImage(srcImage)));
cvWaitKey(20);
#else
cv::imshow("CAM", srcImage);
cv::waitKey(20);
#endif
}
catch (cv::Exception& ex) {
fprintf(stderr, "Exception saving image to bmp format: %s\n", ex.what());
}
srcImage.release();
return true;
}
4 cvtColor一步转换
使用相机SDK的好处是集成简单,但对于本身需要使用Opencv的工程来说也可以直接使用Opencv的cvtColor, 按照原有的bayer格式选择参数,即可一步把RAW图像转为RGB图像。
cv::Mat bayerImg = cv::Mat(stImageInfo.nHeight, stImageInfo.nWidth, CV_8UC1, pData);
cv::Mat bgrImg;
if(stImageInfo.enPixelType == PixelType_Gvsp_BayerBG8) cv::cvtColor(bayerImg, bgrImg, cv::COLOR_BayerBG2RGB);
else if(stImageInfo.enPixelType == PixelType_Gvsp_BayerRG8) cv::cvtColor(bayerImg, bgrImg, cv::COLOR_BayerRG2RGB);
else if (stImageInfo.enPixelType == PixelType_Gvsp_BayerGB8) cv::cvtColor(bayerImg, bgrImg, cv::COLOR_BayerGB2RGB);
else if (stImageInfo.enPixelType == PixelType_Gvsp_BayerGR8) cv::cvtColor(bayerImg, bgrImg, cv::COLOR_BayerGR2RGB);
cv::imshow("CAM", bgrImg);
cv::waitKey(20);
5 结果和参考代码
RAW图像,看上去是黑白的实际上是马赛克图像
转换后的RGB图像
热门推荐
明朝武器与清朝武器的较量:一场跨越时空的对决
三亚必打卡:亚特兰蒂斯+椰子鸡,冬日度假的完美选择
从古诗到现代剧:《一不小心爱上你》里的“晴”字意境
王羲之手札:书法中的节奏韵律之美
从甲骨文到楷书:一个字的书法演变史
探访云南普洱太阳河森林公园
北标园——科普探索者的乐园丨跨越回归线,开启科学之旅
《哪吒》《封神》双雄争霸:春节档票房大战背后的制作与营销密码
保护胰脏,从这十种食物开始
费萨尔:沙特现代化进程的关键推动者
成都东站到天府广场地铁攻略新鲜出炉!
成都阴天出行指南:这些小贴士你get了吗?
明朝火器技术是中国古代的巅峰?真实情况让人大跌眼镜
如何与室友和平度过大学四年 | 新学期非暴力沟通实用技巧指南
云南落水村:优美泸沽湖风光 浓郁摩梭民俗风情
开学季起床难!4个建议+2种快速唤醒法,让孩子轻松迈出被窝
温州到淄博自驾游全攻略:14小时玩转两地!
中秋探秘淄博:临淄齐国故城&稷下学宫
南京一日游:约拍师带你捕捉最美瞬间
秋日打卡南京瞻园,感受“金陵第一园”
南京一日游:打卡历史名胜,感受古都魅力
肠梗阻的四大症状是什么?
中华草龟:一位自然界的“智慧老人”
新手养龟全攻略:从环境到饮食,一文掌握中华草龟的科学饲养方法
打造草龟的理想家园:从习性到环境的全方位指南
草龟:自然界的生态守护者
揭秘中华草龟:百岁寿星的生存智慧
背经典免门票,这种“玩法”为啥火了?
三道堰景区:成都近郊的AAAA级水乡古镇
周末打卡泸州美食:黄粑、白糕、豆汤面!