‘壹’ 视频编码-CRF模式
Constant Rate Factor(CRF)是一种码率控制模式,最早是X264提出来的,后面的X265和libvpx也都有类似的模式。这种码控模式可以达到基本恒定的质量,不过码率会随着内容的不同而波动。如果使用ffmpeg,可以通过下面的命令来设置crf:
x264和x265里面crf的设置范围是0-51,crf越高, 质量越差。CRF越低,质量越好。一般CRF低于18的时候,人眼感知不到明显的质量差异。
CQP也就是Constant QP,也就是对于同样的帧类型(I帧,P帧,B帧),使用相同的QP进行量化。编码器的失真是由量化引起的,相同的量化参数可以保证对于同样的残差信号可以实现大概相同的损失。
而CRF则是Constant Quality,这里的Quality是感知到的quality。CRF会对不同复杂度的帧使用不同的QP进行编码,从而达到一个恒定的感知到的质量。那么感知到的质量和信号层面的质量有什么差异?信号层面的质量可以使用PSNR来衡量,而感知到的质量和人眼的特性有关。一般来说,对于高复杂度(草地)或者剧烈运动的场景,细节的损失人眼是很难感知到的,而对于平坦区域的细节损失人眼很容易感知到(如平坦区域的马赛克或者banding)。
举一个例子,Constant QP在QP=18的时候编码,会把每帧的QP都设置为18(实际中不同类型的帧可能会有不同,暂时忽略);而CRF则可能对于复杂的帧,使用QP=20来编码,对于简单的帧,使用QP=18来编码。
CRF是保证质量的码控模式,对于最终的码率大小没有保证,在实际应用场景里面,很多时候需要一个最大码率的限制,此时就可以使用CRF+VBV的模式。这种模式一般需要设置三个参数,对应的参数名和意义如下:
crf : 期望的质量,越低越好
bufsize : 一个slide window的bit buffer,编码器需要保证这个buffer没有overflow。
maxrate : 最大的local bitrate 。
举个例子:
怎么设置bufsize的大小?bufsize一般是和maxrate的成一定的比例关系,譬如2x, 1x, 5x等。bufsize越大,对码控的约束越小,允许的码率波动越大。bufsize越小,对码控的约束越大,允许的码率波动就越小。所有这个值的设定取决于应用对于码率波动大小的要求。
‘贰’ 如何用psnr模型计算相关系数
1、相关系数就用命令corrcoef
min(min(corrcoef(x1, x2))) 就是x1,x2之间的相关系数。
比如
t = (1:0.1:100)';
w = 2*pi;
x1=sin(w*t)+randn(size(t));
x2=cos(w*t)+randn(size(t));
x3=sin(w*t)+randn(size(t));
x1_x2 = min(min(corrcoef(x1, x2)))
x1_x3 = min(min(corrcoef(x1, x3)))
2、用corrcoef函数
设a1,b1,c1,d1 ,a2,b2,c2,d2 分别为f(x)和g(x)的系数
x=[a1,b1,c1,d1];
y=[a2,b2,c2,d2];
z=corrcoef(x,y)
‘叁’ RC522读卡芯片怎么用
什么意思?#include
#define RST 1 << 8 // P0.7控制RC522复位,低电平复位/*
*********************************************************************************************************
** 函数名称 :RC522_RST()
** 函数功能 :复位RC522
** 调试说明 :需将跳线JP5和RST连接。
*********************************************************************************************************
*/int RC522_RST (void)
{ //PINSEL0 = 0x00000000; // 设置管脚连接GPIO
IO0DIR = RST; // 设置RST控制口为输出
IO0SET = RST; // P0.7置高
DelayNS(10); // 延时
IO0CLR = RST; // P0.7置低
DelayNS(3);
IO0SET = RST;
return 0;
}/*
********************************************************************************************************
** 函数名称:ReadRawRC()
** 函数功能:读MF522寄存器
** 参数说明:Address[IN]:寄存器地址
** 返 回 值:读出的值
********************************************************************************************************
*/unsigned char ReadRawRC(unsigned char Address)
{
unsigned char ucAddr;
unsigned char ucResult = 0;
ucAddr = ((Address<<1)&0x7E)|0x80;
ucResult = MSPI_SendData (ucAddr);
return ucResult;
}
/*
*******************************************************************************************************
** 函数名称:WriteRawRC()
** 函数功能:写MF522寄存器
** 参数说明:Address[IN]:寄存器地址
** value[IN]:写入的值
** 返 回 值:无
*******************************************************************************************************
*/void WriteRawRC(unsigned char Address,unsigned char value)
{
unsigned char ucAddr;
ucAddr = ((Address<<1)&0x7E);
MSPI_SendData(ucAddr);
MSPI_SendData(value);
return 0;}/*
*******************************************************************************************************
** 函数名称:SetBitMask()
** 函数功能:置MF522寄存器位
** 参数说明:reg[IN]:寄存器地址
** mask[IN]:置位值
** 返 回 值:无
*******************************************************************************************************
*/void SetBitMask(unsigned char reg,unsigned char mask)
{
char tmp = 0x0;
tmp = ReadRawRC(reg);
WriteRawRC(reg,tmp | mask);
return 0;
}/*
*******************************************************************************************************
** 函数名称:ClearBitMask()
** 函数功能:清MF522寄存器位
** 参数说明:reg[IN]:寄存器地址
** mask[IN]:清位值
** 返 回 值:无
*******************************************************************************************************
*/void ClearBitMask(unsigned char reg,unsigned char mask)
{
char tmp = 0x0;
tmp ReadRawRC(reg);
WriteRawRC(reg,tmp & ~mask);
return 0;
}/*
*******************************************************************************************************
** 函数名称:PcdAntennaOn()
** 函数功能:开启天线
** 函数说明:每次开启或关闭天线发射之间至少有1ms的间隔
** 返 回 值:无
*******************************************************************************************************
*/void PcdAntennaOn()
{
unsigned char i;
i = ReadRawRC(TxControlReg);
if (!(i & 0x03))
{
SetBitMask(TxControlReg,0x03);
}return 0;
}/*
*******************************************************************************************************
** 函数名称:PcdAntennaOff()
** 函数功能:关闭天线
** 函数说明:每次开启或关闭天线发射之间至少有1ms的间隔
** 返 回 值:无
*******************************************************************************************************
*/void PcdAntennaOff()
{
ClearBitMask(TxControlReg,0x03);
}/*
*******************************************************************************************************
** 函数名称:PcdComMF522()
** 函数功能:通过RC522和ISO14443卡通讯
** 函数说明:Command[IN]:RC522命令字
** pIndata[IN]:通过RC522发送到卡片的数据
** InLenByte[IN]:发送数据的字节长度
** pOutData[OUT]:接收到的卡片返回数据
** *pOutLenBit[OUT]:返回数据的位长度
** 返 回 值:成功返回 MI_OK
*******************************************************************************************************
*/char PcdComMF522(unsigned char Command,
unsigned char *pInData,
unsigned char InLenByte,
unsigned char *pOutData,
unsigned int *pOutLenBit)
{
char status = MI_ERR;
unsigned char irqEn = 0x00;
unsigned char waitFor = 0x00;
unsigned char lastBits;
unsigned char n;
unsigned int i;
switch (Command)
{
case PCD_AUTHENT:
irqEn = 0x12;
waitFor = 0x10;
break;
case PCD_TRANSCEIVE:
irqEn = 0x77;
waitFor = 0x30;
break;
default:
break;
}
WriteRawRC(ComIEnReg,irqEn|0x80);
ClearBitMask(ComIrqReg,0x80);
WriteRawRC(CommandReg,PCD_IDLE);
SetBitMask(FIFOLevelReg,0x80);
for (i=0; i<InLenByte; i++)
{ WriteRawRC(FIFODataReg, pInData[i]); }
WriteRawRC(CommandReg, Command);
if (Command == PCD_TRANSCEIVE)
{ SetBitMask(BitFramingReg,0x80); }
i = 600;//根据时钟频率调整,操作M1卡最大等待时间25ms
do
{
n = ReadRawRC(ComIrqReg);
i--;
}
while ((i!=0) && !(n&0x01) && !(n&waitFor));
ClearBitMask(BitFramingReg,0x80);
if (i!=0)
{
if(!(ReadRawRC(ErrorReg)&0x1B))
{
status = MI_OK;
if (n & irqEn & 0x01)
{ status = MI_NOTAGERR; }
if (Command == PCD_TRANSCEIVE)
{
n = ReadRawRC(FIFOLevelReg);
lastBits = ReadRawRC(ControlReg) & 0x07;
if (lastBits)
{ *pOutLenBit = (n-1)*8 + lastBits; }
else
{ *pOutLenBit = n*8; }
if (n == 0)
{ n = 1; }
if (n > MAXRLEN)
{ n = MAXRLEN; }
for (i=0; i<n; i++)
{ pOutData[i] = ReadRawRC(FIFODataReg); }
}
}
else
{ status = MI_ERR; }
}
SetBitMask(ControlReg,0x80); // stop timer now
WriteRawRC(CommandReg,PCD_IDLE);
return status;
}2007-8-11 22:33:00 jianjian1981
等级:初中二年级
文章:51
积分:1461
门派:无门无派
注册:2006年4月19日第 2 楼 --------------------------------------------------------------------------------
/*
*******************************************************************************************************
** 函数名称:PcdRequest()
** 函数功能:寻卡
** 函数说明:req_code[IN]:寻卡方式
** 0x52 = 寻感应区内所有符合14443A标准的卡
** 0x26 = 寻未进入休眠状态的卡
** pTagType[OUT]:卡片类型代码
** 0x4400 = Mifare_UltraLight
** 0x0400 = Mifare_One(S50)
** 0x0200 = Mifare_One(S70)
** 0x0800 = Mifare_Pro(X)
** 0x4403 = Mifare_DESFire
** 返 回 值:成功返回MI_OK
*******************************************************************************************************
*/char PcdRequest(unsigned char req_code,unsigned char *pTagType)
{
char status;
unsigned int unLen;
unsigned char ucComMF522Buf[MAXRLEN]; ClearBitMask(Status2Reg,0x08);
WriteRawRC(BitFramingReg,0x07);
SetBitMask(TxControlReg,0x03);
ucComMF522Buf[0] = req_code; status = PcdComMF522(PCD_TRANSCEIVE,ucComMF522Buf,1,ucComMF522Buf,&unLen);
if ((status == MI_OK) && (unLen == 0x10))
{
*pTagType = ucComMF522Buf[0];
*(pTagType+1) = ucComMF522Buf[1];
}
else
{ status = MI_ERR; }
return status;
}/*
*******************************************************************************************************
** 函数名称:PcdAnticoll()
** 函数功能:防冲撞
** 函数说明:pSnr[OUT]:卡片序列号,4字节
** 返 回 值:成功返回MI_OK
*******************************************************************************************************
*/char PcdAnticoll(unsigned char *pSnr)
{
char status;
unsigned char i,snr_check=0;
unsigned int unLen;
unsigned char ucComMF522Buf[MAXRLEN];
ClearBitMask(Status2Reg,0x08);
WriteRawRC(BitFramingReg,0x00);
ClearBitMask(CollReg,0x80);
ucComMF522Buf[0] = PICC_ANTICOLL1;
ucComMF522Buf[1] = 0x20; status = PcdComMF522(PCD_TRANSCEIVE,ucComMF522Buf,2,ucComMF522Buf,&unLen); if (status == MI_OK)
{
for (i=0; i<4; i++)
{
*(pSnr+i) = ucComMF522Buf[i];
snr_check ^= ucComMF522Buf[i];
}
if (snr_check != ucComMF522Buf[i])
{ status = MI_ERR; }
}
SetBitMask(CollReg,0x80);
return status;
}/*
*******************************************************************************************************
** 函数名称:PcdSelect()
** 函数功能:选定卡片
** 函数说明:pSnr[IN]:卡片序列号,4字节
** 返 回 值:成功返回MI_OK
*******************************************************************************************************
*/char PcdSelect(unsigned char *pSnr)
{
char status;
unsigned char i;
unsigned int unLen;
unsigned char ucComMF522Buf[MAXRLEN];
ucComMF522Buf[0] = PICC_ANTICOLL1;
ucComMF522Buf[1] = 0x70;
ucComMF522Buf[6] = 0;
for (i=0; i<4; i++)
{
ucComMF522Buf[i+2] = *(pSnr+i);
ucComMF522Buf[6] ^= *(pSnr+i);
}
CalulateCRC(ucComMF522Buf,7,&ucComMF522Buf[7]);
ClearBitMask(Status2Reg,0x08); status = PcdComMF522(PCD_TRANSCEIVE,ucComMF522Buf,9,ucComMF522Buf,&unLen);
if ((status == MI_OK) && (unLen == 0x18))
{ status = MI_OK; }
else
{ status = MI_ERR; } return status;
}
‘肆’ 我想用ffmpeg命令行转换一个视频文件,视、音频编码格式不变,只是将画面旋转90度,请问命令行该如何写
在你的转码命令里面添加上-vf rotate=PI/2或者-vf rotate=3*PI/2,例如
ffmpeg -i E:\ffmpeg\test.mp4 -y -vf rotate=PI/2 E:\ffmpeg\dest1.mp4