[原创] 【匠芯创D133CBS】485的应用

yangjiaxu   2024-10-13 12:27 楼主
今天与大家来分享【匠芯创D133CBS】的485通讯的应用,因为我想设计一个HMI显示屏功能,现在市面上的HMI显示屏一般是通过网口或者是串口进行通讯的,但是如果实际应用方面HMI显示屏的应用数量比较多,其实是可以通过485实现通信的,这样不仅可以方便现场施工布线,也可以降低应用架构的成本。
本次通过D133CBS开发板背面的电路布局设计来看,是有485通信电路的,并且根据电路原理图得知,该电路采用的是自适应2线485通讯,因此我们可以通过外接485实现与D133CBS开发板进行数据交互。
wd_122641z44rizei44h48ses.png
图1 D13x7寸开发板(D133CBS)的背面,485电路部分
接下来就是编写程序了,根据官方提供的demo可以看出,如何应用和调用串口通讯,这里将官方的串口例程修改为uart1路径,之后在里面加入自己需要的应用函数即可。修改的文件路径如下luban-lite\bsp\examples\test-uart。
wd_122641wjc0o0jbk2cpp2ba.png
图2 串口通讯例程位置
代码:
//uart.c

#include <getopt.h>
#include <string.h>
#include <rtthread.h>
#include <aic_core.h>
//#include "beep.h"

#define SAMPLE_UART_NAME "uart1"
#define TIMEOUT_NONE     -4096

static struct rt_semaphore rx_sem;
static rt_device_t serial;
static char str_send[] = "12";
uint8_t g_recv_max = 128;
static int g_uart_test_result = 1;
static int g_exit = 0;
static int g_timeout = TIMEOUT_NONE;

static rt_timer_t  uart1_rx_timeout = RT_NULL;

char str_receive[128 + 1] = {0};

typedef struct
{
    uint8_t buf[128];
    uint8_t len;
    uint8_t rx_over;
} uart_rx_dat_t;

uart_rx_dat_t uart1_rx_dat = { 0 };


static void uart1_rx_timeout_cb(void *p)
{
    rt_timer_stop(uart1_rx_timeout);
    rt_kprintf("read_length: %d \n", uart1_rx_dat.len);
    uart1_rx_dat.rx_over = RT_TRUE;
}

rt_err_t uart1_input(rt_device_t dev, rt_size_t size)
{
    char ch;
    //rt_sem_release(&rx_sem);
    if(rt_device_read(serial, -1, &ch, 1)==1)
    {
       uart1_rx_dat.buf[uart1_rx_dat.len++] = ch; 
       rt_timer_start(uart1_rx_timeout);
    }
    return RT_EOK;
}

void serial_uart1_thread_entry(void *parameter)
{
    uint8_t jx=0;
    char ch;
    int ret = 0;
    
    int cnt = 0;
    static int timeout_time = 0;
    static uint8_t rx_one_byte_flag = 1;

    while (1) 
    {
        if(uart1_rx_dat.rx_over == RT_TRUE)
        {
            uart1_rx_dat.rx_over = RT_FALSE;
            rt_kprintf("read_dat: ");
            for(jx=0; jx<uart1_rx_dat.len; jx++)
            {
                rt_kprintf("%02x  ", uart1_rx_dat.buf[jx]);        
            }
            rt_kprintf("\n");   
             uart1_rx_dat.len=0;
            if( uart1_rx_dat.buf[0]==0xff &  uart1_rx_dat.buf[1]==0x2a)
            {
                if(uart1_rx_dat.buf[2]==0x01)
                {
                   //beep_on();
                   rt_device_write(serial, 0, &uart1_rx_dat, 4);
                   
                }
                else if(uart1_rx_dat.buf[2]==0x00)
                {
                    //beep_off();
                    rt_device_write(serial, 0, &uart1_rx_dat, 4);
                }
            }  
        }
        rt_thread_mdelay(5);
    }
}

int test_uart_485(int argc, char *argv[])
{
    int c = 0;
    rt_err_t ret = RT_EOK;
    static rt_uint8_t open_cnt = 0;

    //查找串口设备
    serial = rt_device_find(SAMPLE_UART_NAME);
    if (!serial)
    {
        rt_kprintf("find %s failed!\n", SAMPLE_UART_NAME);
        return -RT_ERROR;
    }

    //初始化信号量
    rt_sem_init(&rx_sem, "rx_sem", 0, RT_IPC_FLAG_FIFO);

    //打开串口设备
    ret = rt_device_open(serial, RT_DEVICE_OFLAG_RDWR | RT_DEVICE_FLAG_INT_RX);
    if (ret != RT_EOK)
    {
        rt_kprintf("open %s failed : %d !\n", SAMPLE_UART_NAME, ret);
        return -RT_ERROR;
    }

    //设置接收回调函数
    rt_device_set_rx_indicate(serial, uart1_input);

    //rt_device_write(serial, 0, str_send, (sizeof(str_send) - 1));

    //创建 serial 线程
    rt_thread_t thread = rt_thread_create("serial", serial_uart1_thread_entry, RT_NULL, 1024*2, 26, 10);
    if (thread != RT_NULL) {
        rt_thread_startup(thread);
    } else {
        rt_device_close(serial);
        return -RT_ERROR;
    }

    //创建接收超时定时器
    uart1_rx_timeout = rt_timer_create("serial_timer", uart1_rx_timeout_cb, RT_NULL, 50, RT_TIMER_FLAG_PERIODIC);

    return ret;
}

MSH_CMD_EXPORT(test_uart_485,  Uart Test);

 

编写好代码之后需要进行硬件me配置,进入me之后,选择uart1 使能,配置其波特率,数据位停止位等等,并且要配置其为rs485 compact io。
wd_122641apsx7rrx2jsb787c.png
图3 开发板在me环境下,配置uart1
wd_122641as4ig5s2gu87dg8g.png
图4 开发板在me环境下,配置为485 compact io
配置好之后记得保存,这里就可以正常编译了,编译完成之后,在调试串口中输入“aicupg”或者是按住uboot重新上电均可使开发板进入到烧录模式,这时我们选择烧录工具进行将刚刚编译好的固件进行烧录即可。
wd_122641c0hw3kcy2hcqcc2s.jpg
图5 代码编译成功
烧录成功之后,就可以在调试串口界面输入test_uart_485命令,这个命令可以自己通过应用函数修改“MSH_CMD_EXPORT(test_uart_485, Uart Test);”
输入之后即可进入到485通讯功能。这里的应用我简单的写的是开发板收到485的数据之后,利用串口1再返回去,因此在串口调试助手工具中可以看到发送与接收内容相同
wd_122641z1huc77qd766p7yz.jpg
图6 485测试界面
至此,关于485通讯部分已经调试通过了,在这里要感谢坛友@TL-LED,感谢坛友的点拨可以快速实现485通讯的调试工作,通过在调试485通讯的过程中遇到了不少麻烦,首先是与正常的单片机裸机通讯不同,这个仅使用C文件就可以,并且每一个C文件相当于一个线程,也就是每个文件相当于一个单片机进行工作。这种感觉非常奇妙,也让我从单线程的裸机通讯逐步学习并了解多线程的通讯方式,接下来我将尝试实现显示功能,敬请期待~
 

本帖最后由 yangjiaxu 于 2024-10-13 12:31 编辑

回复评论 (1)

非常感谢分享,受益匪浅,期待下次分享

点赞  2024-10-13 14:56
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 京公网安备 11010802033920号
    写回复