这个月 20 号准备去参加 RT-Thread 一年一度的 RDC 开发者大会,顺便会带上我们公司的产品,这个产品就用到了大彩串口屏,所以昨天我也写了一篇表驱动法在大彩串口屏上的应用,文章如下:

 

【12 月】大彩串口屏 RT-Thread Nano STM32 表驱动法产品应用开发

 

接下来我会做一个产品级的基于大彩串口屏的开源项目,用的大彩串口屏型号是:

 

DC80480F070_6111_ON,128M,如下,这是一个 7 寸屏幕,分辨率 800*480;当然价格也是超级便宜的了,入手价也就 180 块钱,今年屏疯狂涨价,这个价格已经很良心了。

 

 

 

 

近年来,RTOS 在嵌入式系统设计中的主导地位也越来越明确,越来越多的工程师选用 RTOS 来完成产品功能的开发;从最熟悉不过的ucos,到后来的freertosrt-threadTencentos tiny等等,以使用者的角度,我在产品开发上用过的 RTOS 非常多;但最后得出一个结论,只要通一个,其它则一通百通;正因为 RTOS 种类越来越多,所以 ARM 公司推出了 CMSIS-RTOS,为统一操作系统、降低嵌入式门槛而发布的操作系统标准软件接口,CMSIS-RTOS 的作用用通俗的话来讲就是:劳资不管你是什么 RTOS,你只需要学习我的 CMSIS-RTOS 怎么用就可以了,但前提是你要把那些 RTOS 的接口适配到 CMSIS-RTOS 上,然后你就可以抛弃那些含义相同,写法不同的 RTOS API,通通都可以不用它们,只用 CMSIS-RTOS 的 API 接口即可!

 

CMSIS-RTOS 架构图如下:

 

 

详情学习可以参考世伟兄之前在腾讯实习的时候周末写的文章:

 

RTOS 内功修炼记(八)— CMSIS RTOS API,内核通用 API 接口

 

1、串口屏是什么?

串口屏,在百度百科上是这么来解释的:

 

一套由单片机或 PLC 带控制器的显示方案,显示方案中的通讯部分由串口通讯,UART 串口或者 SPI 串口等;它由显示驱动板、外壳、LCD 液晶显 示屏三部分构成。接收用户单片机串口发送过来的指令,完成在 LCD 上绘图的所有操作。

 

 

1.1、大彩串口屏的数据收发接口

1.1.1、大彩串口屏数据接收处理

收的部分昨天的文章已经介绍过了:

 

【12 月】大彩串口屏 RT-Thread Nano STM32 表驱动法产品应用开发

 

是通过一种类似消息机制的队列来进行实现,然后将队列里的数据进行拼接加工后满足大彩科技定义的一种协议指令集,所以中断服务函数实现如下,这样就可以持续的来接收串口屏回复的指令:

 

/**
  * @brief This function handles USART2 global interrupt.
  */
void USART2_IRQHandler(void)
{
    /* USER CODE BEGIN USART2_IRQn 0 */
    uint32_t i ;
    uint32_t uart2_dma_rxlen ;
    /*进入中断调用*/
    rt_interrupt_enter();
    if(__HAL_UART_GET_IT_SOURCE(&huart2, UART_IT_IDLE) != RESET)
    {
        __HAL_UART_CLEAR_IDLEFLAG(&huart2);
        HAL_UART_DMAStop(&huart2);
        uart2_dma_rxlen = HMI_LCD_U2_BUFFER_SIZE - (__HAL_DMA_GET_COUNTER(huart2.hdmarx));
    
        for(i = 0; i < uart2_dma_rxlen; i++)
        {
            queue_push(HMI_LCD_Handler.HMI_LCD_U2_Buffer[i]);
        }

        __HAL_UART_ENABLE_IT(&huart2, UART_IT_IDLE);
        HAL_UART_Receive_DMA(&huart2, HMI_LCD_Handler.HMI_LCD_U2_Buffer, HMI_LCD_U2_BUFFER_SIZE);
    }

    /* USER CODE END USART2_IRQn 0 */
    HAL_UART_IRQHandler(&huart2);
    /* USER CODE BEGIN USART2_IRQn 1 */
   /*离开中断调用*/
   rt_interrupt_leave();
  /* USER CODE END USART2_IRQn 1 */
}

 

以下是大彩科技提供给开发者的 MCU 例程文档中接收指令集的流程图:

 

 

以使用 RT-Thread 为例,在进入中断前调用:rt_interrupt_enter,在离开中断前调用:rt_interrupt_leave

 

 

以上描述来自 RT-Thread 文档中心。

 

比如TencentOS tiny也提供了一组 API:

 

tos_knl_irq_enter
tos_knl_irq_leave

 

在进入中断处理函数调用tos_knl_irq_enter,在退出前调用tos_knl_irq_leave

 

又比如UCOSIII也提供了一组 API:

 

OSIntEnter();
OSIntExit();

 

在进入中断处理函数调用OSIntEnter,在退出前调用OSIntExit

 

其它的 RTOS 也是类似的,这里就不多做介绍了,有兴趣可以自己测试和研究。

 

1.1.2、大彩串口屏数据发送处理

大彩串口屏提供了hmi_driver.c这个文件,这个文件提供了一系列串口命令驱动的函数,例如设置控件的值等等,这些 操作依赖于以下这些发送接口:

 

#define TX_8(P1) SEND_DATA((P1)&0xFF)                    // 发送单个字节
#define TX_8N(P,N) SendNU8((uint8 *)P,N)                 // 发送 N 个字节
#define TX_16(P1) TX_8((P1)>>8);TX_8(P1)                 // 发送 16 位整数
#define TX_16N(P,N) SendNU16((uint16 *)P,N)              // 发送 N 个 16 位整数
#define TX_32(P1) TX_16((P1)>>16);TX_16((P1)&0xFFFF)     // 发送 32 位整数

 

上面这些接口,最终我们需要提供这样一个发送单个字节的函数:

 

/*! 
*  \brief  发送一个字节
*  \param  c 
*/
void SEND_DATA(uint8 c)
{
    SendChar(c);
}

 

那我们就直接实现SendChar这个函数就行了,以带 RT-Thread 操作系统的 STM32 工程为例,编写如下接口:

 

void SendChar(uint8_t data)
{
    /*调度器上锁*/
    rt_enter_critical();
    HAL_UART_Transmit(&huart2, &data, 1, 1000);
    while(__HAL_UART_GET_FLAG(&huart2, UART_FLAG_TXE) != SET);
    /*调度器解锁*/
    rt_exit_critical();
}

 

这里为什么要加上调度锁呢??假设,你在界面上需要在不同任务里同时调用如下接口:

 

void SetTextValue(u16 screen_id, u16 control_id, u8 *str)
{
    BEGIN_CMD();
    TX_8(0xB1);
    TX_8(0x10);
    TX_16(screen_id);
    TX_16(control_id);
    SendStrings(str);
    END_CMD();
}

 

这个接口是用来在给界面上某个文本控件显示字符串用的;当多个任务同时调用该接口时,这样不就是我们之前谈的打架问题了吗?在多任务系统中,这就是一种潜在的风险,当一个任务在使用某个资源的过程中,还没有完全结束对资源的访问时就被打断了,这样就会出现一些奇奇怪怪的问题,比如之前我用 OLED 结合 RTOS 编程时候也会出现像屏幕花屏的现象,这里我采用的方法是直接在底层的接口函数处加上调度锁,以防止这种情况发生,当然,还有另外一种方法可以实现,那就是互斥锁。

 

至于互斥锁该怎么用,打开各大 RTOS 的 API 参考手册,上面会详细的告诉你如何创建,如何使用,照着做就是了,这里就不多说了。

 

初学 RTOS 会遇到各种各样的坑,以上我提到的这些坑都是初学者碰得最多的,还有一些测试了很久都没有被解决且难以复现的问题;最后都是在不断的调试中找到分析问题的方法和解决技巧,但万变不离其宗,我们要努力去 Get 最基础的操作系统原理,在理论基础知识的支撑上,才能更好的帮我们去分析问题和解决问题。