6310
- 收藏
- 点赞
- 分享
- 举报
一个海思SDK包中保存YUV数据的函数疑问
本帖最后由 dazhazha 于 2016-11-23 12:13 编辑
在海思SDK包中,有个保存YUV数据的函数,如下:
/* sp420 转存为 p420 ; sp422 转存为 p422 */
[code]void hisi_comm_save_one_frame(VIDEO_FRAME_S * pVBuf, FILE *pfd)
{
unsigned int w, h;
char * pVBufVirt_Y;
char * pVBufVirt_C;
char * pMemContent;
unsigned char TmpBuff[MAX_FRM_WIDTH]; //如果这个值太小,图像很大的话存不了
HI_U32 phy_addr,size;
HI_CHAR *pUserPageAddr[2];
PIXEL_FORMAT_E enPixelFormat = pVBuf->enPixelFormat;
HI_U32 u32UvHeight;/* 存为planar 格式时的UV分量的高度 */
if (pVBuf->u32Width > MAX_FRM_WIDTH)
{
printf("Over max frame width: %d, can't support.\n", MAX_FRM_WIDTH);
return;
}
if (PIXEL_FORMAT_YUV_SEMIPLANAR_420 == enPixelFormat)
{
size = (pVBuf->u32Stride[0])*(pVBuf->u32Height)*3/2;
u32UvHeight = pVBuf->u32Height/2;
}
else
{
size = (pVBuf->u32Stride[0])*(pVBuf->u32Height)*2;
u32UvHeight = pVBuf->u32Height;
}
phy_addr = pVBuf->u32PhyAddr[0];
//printf("phy_addr:%x, size:%d\n", phy_addr, size);
pUserPageAddr[0] = (HI_CHAR *) HI_MPI_SYS_Mmap(phy_addr, size);
if (NULL == pUserPageAddr[0])
{
return;
}
printf("stride: %d,%d\n",pVBuf->u32Stride[0],pVBuf->u32Stride[1]);
printf("u32Width: %d,u32Height: %d\n", pVBuf->u32Width, pVBuf->u32Height);
pVBufVirt_Y = pUserPageAddr[0];
pVBufVirt_C = pVBufVirt_Y + (pVBuf->u32Stride[0])*(pVBuf->u32Height);
/* save Y ----------------------------------------------------------------*/
fprintf(stderr, "saving......Y......");
fflush(stderr);
for(h=0; hu32Height; h++)
{
pMemContent = pVBufVirt_Y + h*pVBuf->u32Stride[0];
fwrite(pMemContent, pVBuf->u32Width, 1, pfd);
}
fflush(pfd);
/* save U ----------------------------------------------------------------*/
fprintf(stderr, "U......");
fflush(stderr);
for(h=0; h
{
pMemContent = pVBufVirt_C + h*pVBuf->u32Stride[1];
pMemContent += 1;
for(w=0; wu32Width/2; w++)
{
TmpBuff[w] = *pMemContent;
pMemContent += 2;
}
fwrite(TmpBuff, pVBuf->u32Width/2, 1, pfd);
}
fflush(pfd);
/* save V ----------------------------------------------------------------*/
fprintf(stderr, "V......");
fflush(stderr);
for(h=0; h
{
pMemContent = pVBufVirt_C + h*pVBuf->u32Stride[1];
for(w=0; wu32Width/2; w++)
{
TmpBuff[w] = *pMemContent;
pMemContent += 2;
}
fwrite(TmpBuff, pVBuf->u32Width/2, 1, pfd);
}
fflush(pfd);
fprintf(stderr, "done %d!\n", pVBuf->u32TimeRef);
fflush(stderr);
HI_MPI_SYS_Munmap(pUserPageAddr[0], size);
}[/code]
我看了好久,这里将yuv420sp转化为yuv420p,在保存U分量的时候有个+1操作,即pMemContent += 1;
sp格式是按照uvuv序列存放,这个+1操作,不是将指针指向了V分量吗?我尝试不做+1操作,保存V分量的时候做+1操作,图像又不正确了,这是为什么了?
在海思SDK包中,有个保存YUV数据的函数,如下:
/* sp420 转存为 p420 ; sp422 转存为 p422 */
[code]void hisi_comm_save_one_frame(VIDEO_FRAME_S * pVBuf, FILE *pfd)
{
unsigned int w, h;
char * pVBufVirt_Y;
char * pVBufVirt_C;
char * pMemContent;
unsigned char TmpBuff[MAX_FRM_WIDTH]; //如果这个值太小,图像很大的话存不了
HI_U32 phy_addr,size;
HI_CHAR *pUserPageAddr[2];
PIXEL_FORMAT_E enPixelFormat = pVBuf->enPixelFormat;
HI_U32 u32UvHeight;/* 存为planar 格式时的UV分量的高度 */
if (pVBuf->u32Width > MAX_FRM_WIDTH)
{
printf("Over max frame width: %d, can't support.\n", MAX_FRM_WIDTH);
return;
}
if (PIXEL_FORMAT_YUV_SEMIPLANAR_420 == enPixelFormat)
{
size = (pVBuf->u32Stride[0])*(pVBuf->u32Height)*3/2;
u32UvHeight = pVBuf->u32Height/2;
}
else
{
size = (pVBuf->u32Stride[0])*(pVBuf->u32Height)*2;
u32UvHeight = pVBuf->u32Height;
}
phy_addr = pVBuf->u32PhyAddr[0];
//printf("phy_addr:%x, size:%d\n", phy_addr, size);
pUserPageAddr[0] = (HI_CHAR *) HI_MPI_SYS_Mmap(phy_addr, size);
if (NULL == pUserPageAddr[0])
{
return;
}
printf("stride: %d,%d\n",pVBuf->u32Stride[0],pVBuf->u32Stride[1]);
printf("u32Width: %d,u32Height: %d\n", pVBuf->u32Width, pVBuf->u32Height);
pVBufVirt_Y = pUserPageAddr[0];
pVBufVirt_C = pVBufVirt_Y + (pVBuf->u32Stride[0])*(pVBuf->u32Height);
/* save Y ----------------------------------------------------------------*/
fprintf(stderr, "saving......Y......");
fflush(stderr);
for(h=0; h
{
pMemContent = pVBufVirt_Y + h*pVBuf->u32Stride[0];
fwrite(pMemContent, pVBuf->u32Width, 1, pfd);
}
fflush(pfd);
/* save U ----------------------------------------------------------------*/
fprintf(stderr, "U......");
fflush(stderr);
for(h=0; h
pMemContent = pVBufVirt_C + h*pVBuf->u32Stride[1];
pMemContent += 1;
for(w=0; w
{
TmpBuff[w] = *pMemContent;
pMemContent += 2;
}
fwrite(TmpBuff, pVBuf->u32Width/2, 1, pfd);
}
fflush(pfd);
/* save V ----------------------------------------------------------------*/
fprintf(stderr, "V......");
fflush(stderr);
for(h=0; h
pMemContent = pVBufVirt_C + h*pVBuf->u32Stride[1];
for(w=0; w
{
TmpBuff[w] = *pMemContent;
pMemContent += 2;
}
fwrite(TmpBuff, pVBuf->u32Width/2, 1, pfd);
}
fflush(pfd);
fprintf(stderr, "done %d!\n", pVBuf->u32TimeRef);
fflush(stderr);
HI_MPI_SYS_Munmap(pUserPageAddr[0], size);
}[/code]
我看了好久,这里将yuv420sp转化为yuv420p,在保存U分量的时候有个+1操作,即pMemContent += 1;
sp格式是按照uvuv序列存放,这个+1操作,不是将指针指向了V分量吗?我尝试不做+1操作,保存V分量的时候做+1操作,图像又不正确了,这是为什么了?
我来回答
回答8个
时间排序
认可量排序
认可0
认可0
认可0
认可0
认可0
认可0
认可0
认可0
或将文件直接拖到这里
悬赏:
E币
网盘
* 网盘链接:
* 提取码:
悬赏:
E币
Markdown 语法
- 加粗**内容**
- 斜体*内容*
- 删除线~~内容~~
- 引用> 引用内容
- 代码`代码`
- 代码块```编程语言↵代码```
- 链接[链接标题](url)
- 无序列表- 内容
- 有序列表1. 内容
- 缩进内容
- 图片![alt](url)
相关问答
-
2016-10-28 15:32:24
-
2024-08-24 11:40:04
-
22015-07-10 14:25:59
-
2020-08-29 10:50:04
-
2015-01-19 22:11:42
-
2018-06-19 14:47:12
-
2014-10-13 21:07:27
-
2018-12-26 15:06:23
-
2018-10-20 09:20:11
-
2020-12-04 15:17:24
-
2021-06-19 16:01:53
-
2021-07-09 16:51:42
-
2016-10-27 17:50:12
-
2024-07-17 16:40:20
-
2017-03-08 10:39:30
-
2015-11-11 10:35:58
-
2019-07-30 15:08:20
-
2018-12-22 09:43:19
-
2020-07-27 10:58:01
无更多相似问答 去提问
点击登录
-- 积分
-- E币
提问
—
收益
—
被采纳
—
我要提问
切换马甲
上一页
下一页
悬赏问答
-
5Hi3516CV610 如何使用SD卡升级固件
-
5cat /dev/logmpp 报错 <3>[ vi] [func]:vi_send_frame_node [line]:99 [info]:vi pic queue is full!
-
50如何获取vpss chn的图像修改后发送至vo
-
5FPGA通过Bt1120传YUV422数据过来,vi接收不到数据——3516dv500
-
50SS928 运行PQtools 拼接 推到设备里有一半画面会异常
-
53536AV100的sample_vdec输出到CVBS显示
-
10海思板子mpp怎么在vi阶段改变视频数据尺寸
-
10HI3559AV100 多摄像头同步模式
-
9海思ss928单路摄像头vio中加入opencv处理并显示
-
10EB-RV1126-BC-191板子运行自己编码的程序
举报反馈
举报类型
- 内容涉黄/赌/毒
- 内容侵权/抄袭
- 政治相关
- 涉嫌广告
- 侮辱谩骂
- 其他
详细说明
提醒
你的问题还没有最佳答案,是否结题,结题后将扣除20%的悬赏金
取消
确认
提醒
你的问题还没有最佳答案,是否结题,结题后将根据回答情况扣除相应悬赏金(1回答=1E币)
取消
确认