HDMI ——CEC 协议详解以及待机唤醒 实现

本文讲解的是基于HDMI CEC的待机唤醒方案的设计。

目录

cec基本介绍

CEC协议时序:

CEC数据帧

cec待机唤醒介绍

待机唤醒的处理流程和实现

cec基本介绍

如今常见的高清视频接口有HDMI,VGA,DP和DVI。HDMI(High-Definition Multimedia Interface)为当今主流的多媒体高速数字接口,下图为最常见的线缆引脚分布图。其中,CEC(Consumer Electronics Control)信号通过13引脚传输,作为HDMI接口的一部分。CEC总线作为控制信号被分离出来,使得在不增加数据占用宽带的情况下完成高速复杂的通信要求。

CEC 是一套完整的单总线协议,电子设备可以借助CEC信号让使用可控制HDMI接口上所连接的装置,比如单独播放,系统待机,可以实现由单一遥控器控制所有HDMI连接的装置。最多15个设备,允许HDMI设备在没有用户干扰情况下互相命令控制。

CEC 是与其他HDMI信号分开的电信号。这允许设备在睡眠模式下禁止其高速电路,但是可以被CEC唤醒。他是一个单独的共享总线,直接连接在设备上的所有HDMI端口间,可以流过所有完全断电的设备。

总线是开路集电极线,有点像IIC,被动上拉至3.3V,设备拉低进行数据传输。

与IIC相似之处:低速串行总线;采用无源上拉的集电极开路;速度受分布电容影响;接收器可以将发送的1位转换为0:发送1比特并观察是否转换为0以查看是否丢失;面向字节都附加一个应答位;特殊的启动信息

与IIC不同之处:单线并不是两根线;以固定时序发送比特;低速串行总线(417bit/s);四个地址位;定义了动态地址分配协议;标头包括发起者和收件人地址;没有特殊停止信号,每个字节附加一个消息结束标志;没有读操作,通过获取请求获取响应帧,所有数据均从数据发送;每个设备都必须能够作为主设备传输数据;地址后字节数据有详细规定说明。

CEC协议时序:

bit Timing

每个位从线拉低(下降沿)开始,保持时间表示位值,之后拉高,直至后续位开始

正常数据位长为2.40.35ms。保持低电平0.6±0.2ms为逻辑1;保持低电平1.5±0.2ms表示逻辑0。接收器在下降沿后1.05±0.2ms对线路进行采样,然后在下降沿1.9±0.15ms开始观察下一位。

接收者可以将传输的传输的1bt转换为0通过在下降沿后0.35s拉低总线并保持直到表示逻辑0的电平时间。这个通常用于确认传输。

每个帧都有起始位,通过拉低总线3.7±0.2ms,然后允许上升,总持续时间为4.5±0.2ms。在观察总线空闲之后,任何设备都可以发送起始位。(通常5位时间,但成功后立即传输7位时间,以促进总线的公平共享,以及传输失败和重传之间的3位时间。)

对于单接收消息,应答位类似于C:以1位发送,接收器将其下拉至0以确认该位字节。(根据下面给出的波形可以看出,ack位的波形为逻辑1,但是为什么逻辑分析仪解析的值为0呢?是因为接收器将其下拉至0 以确认该位字节)

对于广播消息,应答位被反转:仍然作为1位发送,但被拒绝该字节的任何接收器下拉到0位。每个CEC帧的第一个字节包含4位源和目标地址头。如果寻址目标存在,则它确认该字节。由除标题之外的任何内容组成的帧是pig,它只检查另个设备的存在。

地址15(1111B)用于广播地址(作为目的地)和未注册的设备(作为源),它们尚未选择不同的地址。一些设备不需要接收非广播的消息,因此可以永久使用地址15。需要接收寻址消息的设备需要自己的地址。设备通过pig它获取地址,如果ping未被确认,则设备声明它。如果确认ping,则设备尝试另个地址。

第二个字节是操作码,它指定要执行的操作,以及后续数据字节的数量及含义。

CEC数据帧

cec帧结构 = 起始位+引导块+数据块

Start(bit)+ Header Block + Data Block 1(opcode block) + Data Block 2 (operand blocks)

注:

Block定义:Data(8 bit) + EOM(1 bit) + ACK(1 bit)

Header Block定义:Initiator(4 bit) + Destination(4 bit) + EOM(1 bit) + ACK(1 bit)

所有的引导块和数据块都是10bit。

块结构:

帧结构:

逻辑分析仪采集解析的一条完整的cec消息:

cec待机唤醒介绍

当一个CEC设备连接到CEC网络中时,其会通过ping的方式获取到自身的逻辑地址。各个设备类型的逻辑地址如下所示,其中总共有16个逻辑地址,而有些设备具有多个逻辑地址,如录音设备就有1、2、9三个不同的逻辑地址。

待机唤醒的场景如下:

其中,playback device 可以是碟机等设备,当TV处于待机状态时,用户可以使用碟机的遥控器或者开机键唤醒碟机后,碟机会发送或消息来唤醒TV,当source需要将输出显示在TV上时,source必须同时发送和消息。其中,消息的操作码为0x04,,的操作码为0x82。

待机唤醒的处理流程和实现

这里只配置接收端,在接收到cec开机信号后唤醒即可,所以下面对接受流程进行说明:

cec初始化配置:

本次基于极海APM32F107RC 芯片,接收CEC信号的GPIO口为:PB0

使用外部中断 和 定时器(TMR4)外设计数的方法 实现CEC信号的接收采集

所以初始化配置需要使能GPIO口,配置定时器和外部中断

void apm_eint0_rising_config(void) //配置外部中断为上升沿触发
{
    
    EINT_Config_T EINT_InitStructure2; //上升沿触发结构体
    EINT_InitStructure2.line = EINT_LINE_0;
    EINT_InitStructure2.mode = EINT_MODE_INTERRUPT;
    EINT_InitStructure2.trigger = EINT_TRIGGER_RISING;
    EINT_InitStructure2.lineCmd = ENABLE;
    EINT_Config(&EINT_InitStructure2);
}
void apm_eint0_falling_config(void)//配置外部中断为下降沿触发
{
    EINT_Config_T EINT_InitStructure1; //下降沿触发结构体
    EINT_InitStructure1.line = EINT_LINE_0;
    EINT_InitStructure1.mode = EINT_MODE_INTERRUPT;
    EINT_InitStructure1.trigger = EINT_TRIGGER_FALLING;
    EINT_InitStructure1.lineCmd = ENABLE;
    EINT_Config(&EINT_InitStructure1);
}
//cec 初始化
void apm_cec_init(void)
{
    GPIO_Config_T gpioConfig;

    TMR_BaseConfig_T baseConfig;
    
    //使能时钟
    RCM_EnableAPB1PeriphClock(RCM_APB1_PERIPH_TMR4);
    RCM_EnableAPB2PeriphClock(RCM_APB2_PERIPH_GPIOB);
    
    
    //初始化GPIO_PB0,浮空输入
    gpioConfig.pin = GPIO_PIN_0;
    gpioConfig.mode = GPIO_MODE_IN_FLOATING;
    gpioConfig.speed = GPIO_SPEED_50MHz;
    GPIO_Config(GPIOB, &gpioConfig);
    
    GPIO_ConfigEINTLine(GPIO_PORT_SOURCE_B,GPIO_PIN_SOURCE_0);//设置IO口与中断的映射关系
    
    EINT_ClearStatusFlag(EINT_LINE_0);
    
    apm_eint0_falling_config();//外部中断初始为下降沿触发
    
    NVIC_EnableIRQRequest(EINT0_IRQn, 2, 1);
    
    //初始化定时器
    baseConfig.period        = 65000;//设定计数器自动重装值,65ms更新
    baseConfig.division      = 71;//预分频器,1M的计数频率,1us加1
    baseConfig.clockDivision = TMR_CLOCK_DIV_1;//设置时钟分割
    baseConfig.countMode     = TMR_COUNTER_MODE_UP;//TIM向上计数模式
    TMR_ConfigTimeBase(TMR4,&baseConfig);
    
    TMR_ClearIntFlag(TMR4,TMR_INT_UPDATE);
    
    TMR_EnableInterrupt(TMR4,TMR_INT_UPDATE);//允许更新中断 

    //使能中断分组及优先级
    NVIC_EnableIRQRequest(TMR4_IRQn,2,2);
    
    TMR_Enable(TMR4);//使能定时器3

}

cec信号接收处理:

当触发外部中断并且设备处于待机状态时,根据cec波形以及时序要求解析CEC信号,通过判断opcode数据位的值 从而判断是否唤醒。

//接收一个cec  帧
void apm_cec_receive_frame(void)
{
    if(EINT_ReadIntFlag(EINT_LINE_0)) //外部中断
    {
        if(apm_get_power_status() == POWER_OFF_STATUS)  //当前状态不是开机
        {
        if(CEC_State == CEC_STATE_IDLE)
        {  
            //如果波形状态为空闲,开启定时器3计数
            TMR_ConfigCounter(TMR4,0);
            TMR_Enable(TMR4);
            cec_previous_time = TMR_ReadCounter(TMR4);
            //SKG_INFO("cec_previous_time: %d\r\n",cec_previous_time);
            apm_eint0_rising_config(); //上升沿触发
            updatecnt = 0;
            CEC_State = CEC_STATE_START; //波形为起始码接收状态
        }
        else
        {
            cec_current_time = TMR_ReadCounter(TMR4); //读取当前捕获值
            //SKG_INFO("cec_current_time: %d\r\n",cec_current_time);
            if(updatecnt == 0)
            {
                //计算当前时间    //时间间隔=当前时间-前一时间
                cec_interval_time = abs(cec_current_time - cec_previous_time);
            }
            else
            {
                cec_interval_time = abs(0x186A0 * updatecnt - cec_previous_time);
                cec_interval_time += cec_current_time;
            }
            cec_previous_time = cec_current_time;//将当前时间设置为前一个时间
            updatecnt = 0; //计数器的范围返回0状态
                
            //SKG_INFO("cec_interval_time: %d\r\n",cec_interval_time);
                
            if(CEC_State == CEC_STATE_START)
            {
                //低电平时间在3.5ms —— 3.9ms之间
                if((cec_interval_time > 3500) && (cec_interval_time < 3900))
                {
                    CEC_State = CEC_STATE_STARTEND; //起始结束接收状态
                }
                else
                {
                    
                    //SKG_INFO("debug 0 cec_interval_time: %d\r\n",cec_interval_time);
                    CEC_State = CEC_STATE_IDLE;
                }
                    apm_eint0_falling_config(); //下降沿触发
                }
                else if(CEC_State == CEC_STATE_STARTEND)
                {
                    //起始结束的低电平区间4ms --- 1.2ms
                    if((cec_interval_time > 300) && (cec_interval_time < 1300))
                    {
                        cec_byte = 0;
                        CEC_State = CEC_STATE_HEADERLOW; //header block 数据位的低电平
                        apm_eint0_rising_config(); //上升沿触发
                    }
                    else
                    {
                        //SKG_INFO("debug 1 cec_interval_time:%d\r\n",cec_interval_time);
                        //SKG_INFO("AAAAAAAAAAAAAAAAAA\r\n");
                        CEC_State = CEC_STATE_IDLE;
                        apm_eint0_falling_config();  //如果波形采集异常,中断回到初始状态下降沿触发
                    }
                    
                }
                else if(CEC_State == CEC_STATE_HEADERLOW)
                {
                    //逻辑1    低电平300-900
                    if((cec_interval_time > 300) && (cec_interval_time < 900))
                    {
                        headerdata <<= 1;
                        headerdata |= 1;
                        cec_byte++;
                        CEC_State = CEC_STATE_HEADERHIGH; //header block  数据位高电平
                    }
                    //逻辑0    低电平1200-1800 
                    else if((cec_interval_time > 1200) && (cec_interval_time < 1800))
                    {
                        headerdata <<= 1;
                        cec_byte++;
                        CEC_State = CEC_STATE_HEADERHIGH;
                    }
                    else
                    {
                        CEC_State = CEC_STATE_IDLE;
                        //SKG_INFO("debug 2 cec_interval_time:%d\r\n",cec_interval_time);
                    }

                    apm_eint0_falling_config();//下降沿触发
                }
                else if(CEC_State == CEC_STATE_HEADERHIGH)
                {
                    //如果上一次低电平的时间+当前高电平的时间在2.05ms-2.75ms,说明data bit传输正确
                    if(2050 < (cec_pre_interval_time + cec_interval_time) < 2750)
                    {
                        CEC_State = CEC_STATE_HEADERLOW;
                    }
                    else
                    {
                        //SKG_INFO("debug 3 cec_interval_time:%d\r\n",cec_interval_time);
                        CEC_State = CEC_STATE_IDLE;
                    }
                    if(cec_byte == 10) //header block 字节接收完成
                    {
                        //head_src = headerdata >> 6;
                        head_src = headerdata >> 2;
                        //head_src = headerdata;
                        headerdata = 0;
                        CEC_State = CEC_STATE_CODELOW; //data block  数据位低电平接收
                    }
                    apm_eint0_rising_config(); //上升沿触发
                }
                else if(CEC_State == CEC_STATE_CODELOW)
                {
                    //逻辑1    低电平300-900
                    if((cec_interval_time > 300) && (cec_interval_time < 900))
                    {
                        codedata <<= 1;
                        codedata |= 1;
                        cec_byte++;
                        CEC_State = CEC_STATE_CODEHIGH; //data block  数据位高电平
                    }
                    //逻辑0    低电平1200-1800
                    else if((cec_interval_time > 1200) && (cec_interval_time < 1800))
                    {
                        codedata <<= 1;
                        cec_byte++;
                        CEC_State = CEC_STATE_CODEHIGH;
                    }
                    else
                    {
                        //SKG_INFO("debug 4 cec_interval_time:%d\r\n",cec_interval_time);
                        CEC_State = CEC_STATE_IDLE;
                    }
                    
                    apm_eint0_falling_config();//下降沿触发
                }
                else if(CEC_State == CEC_STATE_CODEHIGH)
                {
                    //如果上一次低电平的时间+当前高电平的时间在2.05ms-2.75ms,说明data bit传输正确
                    if(2050 < (cec_pre_interval_time + cec_interval_time) < 2750)
                    {
                        CEC_State = CEC_STATE_CODELOW;
                    }
                    else
                    {
                        //SKG_INFO("debug 5 cec_interval_time:%d\r\n",cec_interval_time);
                        CEC_State = CEC_STATE_IDLE;
                    }
                                        if(cec_byte == 18)
                    {
                        opcode = codedata;
                        codedata = 0;
                        CEC_State = CEC_STATE_IDLE;
                    }
                    apm_eint0_rising_config(); //上升沿触发
                }
            }
            cec_pre_interval_time = cec_interval_time;
        }
    }
}
                    

波形的处理流程是  使用状态标志位  的变化来表示接收的字节

typedef enum
{
    CEC_STATE_IDLE=0,        //0  初始为空闲
    CEC_STATE_START,        //1  起始位
    CEC_STATE_STARTEND,     //2     起始位结束
    CEC_STATE_HEADERLOW,    //3  head block  数据位低电平
    CEC_STATE_HEADERHIGH,  //4  head block  数据位高电平
    CEC_STATE_CODELOW,     //5  data block  数据位低电平
    CEC_STATE_CODEHIGH,    //6  data block  数据位高电平
}CEC_State_enum;

处理过程中没有单独解析ack和eom,字节只接收两个,一个header   10位数据,一个opcode  8位数据。

波形处理流程:

第一次进入中断为空闲,开启定时器计数并读取当前计数值,状态改为起始位接收,中断改为上升沿触发

第二次进入中断读取定时器的值,减去前一次计数器的值,就是起始位低电平维持的时间,判断时间是否为在协议范围,把触发改为下降沿,如果在范围内将状态改为起始位结束状态

第三次进入中断 依然是计算出时间的差值,判断高电平维持的时间范围,在范围内则起始位接收完成,将状态改为接收header clock,把中断改为上升沿触发

第四次进入中断  由于上一次是下降沿触发,所以时间的差值就是低电平的持续时间,判断时间的范围得到逻辑0或逻辑1,逻辑1 headerdata  

数据处理后将状态改为HEADERHIGH  ,中断改为下降沿触发

下一次进入中断  时间的差值就是高电平的持续时间,进入HEADERHIGH 条件判断,当前的时间差值+上一次的时间差值就是数据传输的总时间,判断总时间是否在范围,在范围将状态改为HEADERLOW,中断改为上升沿触发,接收处理下一位数据,接收的数据位达到10位,将状态改为接收data clock。Data的接收处理和header相同。接收的数据位达到18位就将数据保存,状态改为空闲。

void apm_cec_ctrl_handle(void)
{
    if(opcode == CEC_VIEW_ON || opcode == ACTIVE_SOURSE)
    {
        if(apm_get_power_status() == POWER_OFF_STATUS)
        {    
            power_on();            
        }

    }
}

在实际调试过程中可以根据需求和现象,调整电平时间判断范围,加入EOM位的判断作为数据传输结束的标志,加大数据的接收范围等。