p_read_frame = read_frame_yuv;/*函数指针赋值*/
/*
得到帧的总数
这儿传的参数,就是上面open_file_yuv结束前传出的那个文件指针
这儿通过文件的总长度/每帧尺寸来计算得到总帧数,可以看到,文件必须是yuv420的,如果是其它格式或者包含音频的,是不能用此函数计算的
*/
int get_frame_total_yuv( hnd_t handle )
{
yuv_input_t *h = handle;
int i_frame_total = 0;
if( !fseek( h->fh, 0, SEEK_END ) )//重定位流(数据流/文件)上的文件内部位置指针;函数设置文件指针stream的位置。如果执行成功,stream将指向以fromwhere(偏移起始位置:文件头0,当前位置1,文件尾2)为基准,偏移offset(指针偏移量)个字节的位置。如果执行失败(比如offset超过文件自身大小),则不改变stream指向的位置。
{
uint64_t i_size = ftell( h->fh );//函数 ftell() 用于得到文件位置指针当前位置相对于文件首的偏移字节数。在随机方式存取文件时,由于文件位置频繁的前后移动,程序不容易确定文件的当前位置。调用函数ftell()就能非常容易地确定文件的当前位置
fseek( h->fh, 0, SEEK_SET );//重定位流(数据流/文件)上的文件内部位置指针,注意:不是定位文件指针,文件指针指向文件/流。位置指针指向文件内部的字节位置,随着文件的读取会移动,文件指针如果不重新赋值将不会改变指向别的文件
//其中fseek中第一个参数为文件指针,第二个参数为偏移量,起始位置,SEEK_END 表示文件尾,SEEK_SET 表示文件头。
i_frame_total = (int)(i_size / ( h->width * h->height * 3 / 2 ));//计算总的帧数, 这里乘以1.5是因为一个编码单位是一个亮度块加2个色度块,大小上等于1.5个亮度块
}
return i_frame_total;
}
/*
读取帧,yuv格式的
这里支持的yuv存储格式为:先存一帧的全部亮度值,再存一帧的全部Cb值,再存一帧的全部Cr值,下一帧也是如此;其它存储方式,这儿不支持
*/
int read_frame_yuv( x264_picture_t *p_pic, hnd_t handle, int i_frame )
{
yuv_input_t *h = handle;
if( i_frame != h->next_frame )
if( fseek( h->fh, (uint64_t)i_frame * h->width * h->height * 3 / 2, SEEK_SET ) )//定位文件
return -1;
if( fread( p_pic->img.plane[0], 1, h->width * h->height, h->fh ) <= 0 //fread:从一个流中读数据//读Y亮度
|| fread( p_pic->img.plane[1], 1, h->width * h->height / 4, h->fh ) <= 0//读Cb
|| fread( p_pic->img.plane[2], 1, h->width * h->height / 4, h->fh ) <= 0 )//读Cr
return -1;
h->next_frame = i_frame+1;//记住下一帧是第几帧,存到了yuv_input_t结构的最后一个字段里
return 0;
}
/*
关闭YUV文件
*/
int close_file_yuv(hnd_t handle)
{
yuv_input_t *h = handle;
if( !h || !h->fh )
return 0;
return fclose(h->fh);
}
评论