99精品伊人亚洲|最近国产中文炮友|九草在线视频支援|AV网站大全最新|美女黄片免费观看|国产精品资源视频|精彩无码视频一区|91大神在线后入|伊人终合在线播放|久草综合久久中文

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內(nèi)不再提示

基于RT-Thread實現(xiàn)can通訊例程講解

嵌入式應用開發(fā) ? 來源:嵌入式應用開發(fā) ? 作者:嵌入式應用開發(fā) ? 2022-07-07 15:10 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

首先粘貼出官方提供的can例程代碼,根據(jù)需要我將部分內(nèi)容做了修改,供大家參考!

can_sample中包換以下函數(shù): 接收數(shù)據(jù)回調(diào)函數(shù) static rt_err_t can_rx_call(rt_device_t dev, rt_size_t size)

定義can接收線程:static void can_rx_thread(void *parameter)

查找can設備:rt_device_t rt_device_find(const char* name);

打開can設備:rt_err_t rt_device_open(rt_device_t dev, rt_uint16_t oflags);

發(fā)生can數(shù)據(jù):rt_size_t rt_device_write(rt_device_t dev, rt_off_t pos, const void* buffer, rt_size_t size);

在int can_sample(int argc, char *argv[])中調(diào)用can接收線程

其中對于描述是有兩個段代碼可供參考

/*
 * 程序清單:這是一個 CAN 設備使用例程
 * 例程導出了 can_sample 命令到控制終端
 * 命令調(diào)用格式:can_sample can1
 * 命令解釋:命令第二個參數(shù)是要使用的 CAN 設備名稱,為空則使用默認的 CAN 設備
 * 程序功能:通過 CAN 設備發(fā)送一幀,并創(chuàng)建一個線程接收數(shù)據(jù)然后打印輸出。
*/
#include 
#include "rtdevice.h"
#define CAN_DEV_NAME       "can1"      /* CAN 設備名稱 */
static struct rt_semaphore rx_sem;     /* 用于接收消息的信號量 */
static rt_device_t can_dev;            /* CAN 設備句柄 */
/* 接收數(shù)據(jù)回調(diào)函數(shù) */
static rt_err_t can_rx_call(rt_device_t dev, rt_size_t size)
{
    /* CAN 接收到數(shù)據(jù)后產(chǎn)生中斷,調(diào)用此回調(diào)函數(shù),然后發(fā)送接收信號量 */
    rt_sem_release(&rx_sem);
    return RT_EOK;
}
static void can_rx_thread(void *parameter)
{
    int i;
    rt_err_t res;
    struct rt_can_msg rxmsg = {0};
    /* 設置接收回調(diào)函數(shù) */
    rt_device_set_rx_indicate(can_dev, can_rx_call);

    while (1)
    {
        /* hdr 值為 - 1,表示直接從 uselist 鏈表讀取數(shù)據(jù) */
        rxmsg.hdr = -1;
        /* 阻塞等待接收信號量 */
        rt_sem_take(&rx_sem, RT_WAITING_FOREVER);
        /* 從 CAN 讀取一幀數(shù)據(jù) */
        rt_device_read(can_dev, 0, &rxmsg, sizeof(rxmsg));
        /* 打印數(shù)據(jù) ID 及內(nèi)容 */
        rt_kprintf("ID:%x", rxmsg.id);
        for (i = 0; i < 8; i++)
        {
            rt_kprintf("%2x", rxmsg.data[i]);
        }
        rt_kprintf("\n");
    }
}
int can_sample(int argc, char *argv[])
{
    struct rt_can_msg msg = {0};
    rt_err_t res;
    rt_size_t  size;
    rt_thread_t thread;
    char can_name[RT_NAME_MAX];
    if (argc == 2)
    {
        rt_strncpy(can_name, argv[1], RT_NAME_MAX);
    }
    else
    {
        rt_strncpy(can_name, CAN_DEV_NAME, RT_NAME_MAX);
    }
    /* 查找 CAN 設備 */
    can_dev = rt_device_find(can_name);
    if (!can_dev)
    {
        rt_kprintf("find %s failed!\n", can_name);
        return RT_ERROR;
    }
    /* 初始化 CAN 接收信號量 */
    rt_sem_init(&rx_sem, "rx_sem", 0, RT_IPC_FLAG_FIFO);
    /* 以中斷接收及發(fā)送方式打開 CAN 設備 */
    res = rt_device_open(can_dev, RT_DEVICE_FLAG_INT_TX | RT_DEVICE_FLAG_INT_RX);
    RT_ASSERT(res == RT_EOK);
    /* 創(chuàng)建數(shù)據(jù)接收線程 */
    thread = rt_thread_create("can_rx", can_rx_thread, RT_NULL, 1024, 25, 10);
    if (thread != RT_NULL)
    {
        rt_thread_startup(thread);
    }
    else
    {
        rt_kprintf("create can_rx thread failed!\n");
    }
    msg.id = 0x78;              /* ID 為 0x78 */
    msg.ide = RT_CAN_STDID;     /* 標準格式 */
    msg.rtr = RT_CAN_DTR;       /* 數(shù)據(jù)幀 */
    msg.len = 8;                /* 數(shù)據(jù)長度為 8 */
    /* 待發(fā)送的 8 字節(jié)數(shù)據(jù) */
    msg.data[0] = 0x00;
    msg.data[1] = 0x11;
    msg.data[2] = 0x22;
    msg.data[3] = 0x33;
    msg.data[4] = 0x44;
    msg.data[5] = 0x55;
    msg.data[6] = 0x66;
    msg.data[7] = 0x77;
    /* 發(fā)送一幀 CAN 數(shù)據(jù) */
    size = rt_device_write(can_dev, 0, &msg, sizeof(msg));
    if (size == 0)
    {
        rt_kprintf("can dev write data failed!\n");
    }
    return res;
}
void can_send_test(void)
{
    struct rt_can_msg msg = {0};
    rt_size_t  size;
    static rt_uint8_t num = 0;
    msg.id = 0x78;              /* ID 為 0x78 */
    msg.ide = RT_CAN_STDID;     /* 標準格式 */
    msg.rtr = RT_CAN_DTR;       /* 數(shù)據(jù)幀 */
    msg.len = 8;                /* 數(shù)據(jù)長度為 8 */
    /* 待發(fā)送的 8 字節(jié)數(shù)據(jù) */
    msg.data[0] = 0x00;
    msg.data[1] = num++;     //can發(fā)送數(shù)據(jù)隨意更改
    msg.data[2] = 0x22;
    msg.data[3] = 0x33;
    msg.data[4] = num++;
    msg.data[5] = 0x55;
    msg.data[6] = 0x66;
    msg.data[7] = 0x77;
    /* 發(fā)送一幀 CAN 數(shù)據(jù) */
    size = rt_device_write(can_dev, 0, &msg, sizeof(msg));
    if (size == 0)
    {
        rt_kprintf("can dev write data failed!\n");
    }
}
/* 導出到 msh 命令列表中 */
MSH_CMD_EXPORT(can_sample, can device sample);
MSH_CMD_EXPORT(can_send_test, can send test);

另外一個版本,實現(xiàn)的功能是一樣的

/*
 * 程序清單:這是一個 CAN 設備使用例程
 * 例程導出了 can_sample 命令到控制終端
 * 命令調(diào)用格式:can_sample can1
 * 命令解釋:命令第二個參數(shù)是要使用的 CAN 設備名稱,為空則使用默認的 CAN 設備
 * 程序功能:通過 CAN 設備發(fā)送一幀,并創(chuàng)建一個線程接收數(shù)據(jù)然后打印輸出。
*/

#include 
#include "rtdevice.h"

#define CAN_DEV_NAME       "can1"      /* CAN 設備名稱 */

static struct rt_semaphore rx_sem;     /* 用于接收消息的信號量 */
static rt_device_t can_dev;            /* CAN 設備句柄 */

#define THREAD_PRIORITY         25
#define THREAD_STACK_SIZE       512
#define THREAD_TIMESLICE        5

static rt_thread_t tid1 = RT_NULL;

/* 接收數(shù)據(jù)回調(diào)函數(shù) */
static rt_err_t can_rx_call(rt_device_t dev, rt_size_t size) {
    /* CAN 接收到數(shù)據(jù)后產(chǎn)生中斷,調(diào)用此回調(diào)函數(shù),然后發(fā)送接收信號量 */
    rt_sem_release(&rx_sem);

    return RT_EOK;
}

static void can_rx_thread(void *parameter) {
    int i;
    rt_err_t res;
    struct rt_can_msg rxmsg = {0};

    /* 設置接收回調(diào)函數(shù) */
    rt_device_set_rx_indicate(can_dev, can_rx_call);

    while (1) {
        /* hdr 值為 - 1,表示直接從 uselist 鏈表讀取數(shù)據(jù) */
        rxmsg.hdr = -1;
        /* 阻塞等待接收信號量 */
        rt_sem_take(&rx_sem, RT_WAITING_FOREVER);
        /* 從 CAN 讀取一幀數(shù)據(jù) */
        rt_device_read(can_dev, 0, &rxmsg, sizeof(rxmsg));
        /* 打印數(shù)據(jù) ID 及內(nèi)容 */
        rt_kprintf("ID:%x", rxmsg.id);
        for (i = 0; i < 8; i++) {
            rt_kprintf("%2x", rxmsg.data[i]);
        }

        rt_kprintf("\n");
    }
}

/* 線程 1 的入口函數(shù) */
static void thread1_entry(void *parameter) {
    struct rt_can_msg msg = {0};
    unsigned long count = 0;

    msg.id = 0x78;              /* ID 為 0x78 */
    msg.ide = RT_CAN_STDID;     /* 標準格式 */
    msg.rtr = RT_CAN_DTR;       /* 數(shù)據(jù)幀 */
    msg.len = 8;                /* 數(shù)據(jù)長度為 3 */
    /* 待發(fā)送的 3 字節(jié)數(shù)據(jù) */
    msg.data[0] = 0x00;
    msg.data[1] = 0x11;
    msg.data[2] = 0x22;
    msg.data[3] = 0x00;
    msg.data[4] = 0x11;
    msg.data[5] = 0x00;
    msg.data[6] = 0x11;
    msg.data[7] = 0x22;

    rt_kprintf("send %ld \n", ++count);

    while (1) {
        /* 線程 1 采用低優(yōu)先級運行,一直打印計數(shù)值 */
        rt_device_write(can_dev, 0, &msg, sizeof(msg));
        rt_kprintf("send %ld \n", ++count);
        rt_thread_mdelay(500);
    }
}



int can_sample(int argc, char *argv[]) {
    rt_err_t res;
    rt_size_t  size;
    rt_thread_t thread;
    char can_name[RT_NAME_MAX];

    if (argc == 2) {
        rt_strncpy(can_name, argv[1], RT_NAME_MAX);
    } else {
        rt_strncpy(can_name, CAN_DEV_NAME, RT_NAME_MAX);
    }

    /* 查找 CAN 設備 */
    can_dev = rt_device_find(can_name);
    if (!can_dev) {
        rt_kprintf("find %s failed!\n", can_name);
        return RT_ERROR;
    }

    /* 初始化 CAN 接收信號量 */
    rt_sem_init(&rx_sem, "rx_sem", 0, RT_IPC_FLAG_FIFO);

    /* 以中斷接收及發(fā)送方式打開 CAN 設備 */

    res = rt_device_open(can_dev, RT_DEVICE_FLAG_INT_TX | RT_DEVICE_FLAG_INT_RX);
    /* 設置 CAN 的工作模式為正常工作模式 */
    res = rt_device_control(can_dev, RT_CAN_CMD_SET_MODE, (void *)RT_CAN_MODE_NORMAL);
    res = rt_device_control(can_dev, RT_CAN_CMD_SET_BAUD, (void *)CAN500kBaud);
    RT_ASSERT(res == RT_EOK);
    /* 創(chuàng)建數(shù)據(jù)接收線程 */
    thread = rt_thread_create("can_rx", can_rx_thread, RT_NULL, 1024, 25, 10);
    if (thread != RT_NULL) {
        rt_thread_startup(thread);
    } else {
        rt_kprintf("create can_rx thread failed!\n");
    }


    if (size == 0) {
        rt_kprintf("can dev write data failed!\n");
    }

    /* 創(chuàng)建線程 1,名稱是 thread1,入口是 thread1_entry*/
    tid1 = rt_thread_create("thread1",
                            thread1_entry, RT_NULL,
                            THREAD_STACK_SIZE,
                            THREAD_PRIORITY, THREAD_TIMESLICE);

    /* 如果獲得線程控制塊,啟動這個線程 */
    if (tid1 != RT_NULL)
        rt_thread_startup(tid1);
    else
        rt_kprintf("start can send fail\n");

    return res;
}
/* 導出到 msh 命令列表中 */
MSH_CMD_EXPORT(can_sample, can device sample);

代碼編譯完成,下載。串口運行:can_sample,打開創(chuàng)芯科技CAN卡,打開電腦USB CAN工具

2020-12-08_215917.png

打開USB轉CAN,選擇對應波特率,這里配置為1Mbps。

重啟,再次運行can_sample,發(fā)現(xiàn),接收到STM32發(fā)出的CAN數(shù)據(jù)幀。

多次運行 can_send_test,電腦端可以接受數(shù)據(jù)。

審核編輯:湯梓紅
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • CAN
    CAN
    +關注

    關注

    57

    文章

    2918

    瀏覽量

    467677
  • RT-Thread
    +關注

    關注

    32

    文章

    1403

    瀏覽量

    41886
收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    RT-ThreadCAN實踐

    開箱測試RT-Thread官方已完成了對英飛凌XMC7200EVK的移植,通過shell可以看到做好了uart3的console。本文將介紹如何進行RT-ThreadCan移植。接下來我們要完成CAN_FD的驅動移植,并正常啟動
    的頭像 發(fā)表于 11-13 01:03 ?2098次閱讀
    <b class='flag-5'>RT-Thread</b>上<b class='flag-5'>CAN</b>實踐

    【原創(chuàng)精選】RT-Thread征文精選技術文章合集

    開發(fā)板的詳細步驟例程stm32裸機RTthread開始創(chuàng)建線程詳解基于標準庫的keil移植到RT-thread例程
    發(fā)表于 07-26 14:56

    RT-Thread編程指南

    RT-Thread編程指南——RT-Thread開發(fā)組(2015-03-31)。RT-Thread做為國內(nèi)有較大影響力的開源實時操作系統(tǒng),本文是RT-Thread實時操作系統(tǒng)的編程指南
    發(fā)表于 11-26 16:06 ?0次下載

    RT-Thread上的CAN總線介紹以及驅動編寫

    昨晚很榮幸邀請到李工在RT-Thread微信群進行RT-Thread上的CAN驅動和應用講座。小編整理了講座內(nèi)容,特發(fā)出講義以供享用。
    的頭像 發(fā)表于 09-25 10:16 ?2.5w次閱讀

    基于 Keil MDK 移植 RT-Thread Nano

    本文介紹如何基于 Keil MDK 移植 RT-Thread Nano ,并以一個 stm32f103 的基礎工程作為示例進行講解。 RT-Thread Nano 已集成在 Keil MD...
    發(fā)表于 01-26 17:04 ?17次下載
    基于 Keil MDK 移植 <b class='flag-5'>RT-Thread</b> Nano

    RT-Thread全球技術大會:Kconfig在RT-Thread中的工作機制

    RT-Thread全球技術大會:Kconfig在RT-Thread中的工作機制 ? ? ? ? ? ? ? 審核編輯:彭靜
    的頭像 發(fā)表于 05-27 14:49 ?1859次閱讀
    <b class='flag-5'>RT-Thread</b>全球技術大會:Kconfig在<b class='flag-5'>RT-Thread</b>中的工作機制

    RT-Thread全球技術大會:RT-Thread測試用例集合案例

    RT-Thread全球技術大會:RT-Thread測試用例集合案例 ? ? ? ? ? 審核編輯:彭靜
    的頭像 發(fā)表于 05-27 16:34 ?2392次閱讀
    <b class='flag-5'>RT-Thread</b>全球技術大會:<b class='flag-5'>RT-Thread</b>測試用例集合案例

    RT-Thread全球技術大會:RT-Thread對POSIX的實現(xiàn)情況介紹

    RT-Thread全球技術大會:RT-Thread對POSIX的實現(xiàn)情況介紹 ? ? ? ? ? ? 審核編輯:彭靜
    的頭像 發(fā)表于 05-27 16:52 ?2150次閱讀
    <b class='flag-5'>RT-Thread</b>全球技術大會:<b class='flag-5'>RT-Thread</b>對POSIX的<b class='flag-5'>實現(xiàn)</b>情況介紹

    RT-Thread學習筆記 RT-Thread的架構概述

    RT-Thread 簡介 作為一名 RTOS 的初學者,也許你對 RT-Thread 還比較陌生。然而,隨著你的深入接觸,你會逐漸發(fā)現(xiàn) RT-Thread 的魅力和它相較于其他同類型 RTOS
    的頭像 發(fā)表于 07-09 11:27 ?5021次閱讀
    <b class='flag-5'>RT-Thread</b>學習筆記 <b class='flag-5'>RT-Thread</b>的架構概述

    RT-Thread文檔_RT-Thread 簡介

    RT-Thread文檔_RT-Thread 簡介
    發(fā)表于 02-22 18:22 ?5次下載
    <b class='flag-5'>RT-Thread</b>文檔_<b class='flag-5'>RT-Thread</b> 簡介

    RT-Thread文檔_RT-Thread 潘多拉 STM32L475 上手指南

    RT-Thread文檔_RT-Thread 潘多拉 STM32L475 上手指南
    發(fā)表于 02-22 18:23 ?10次下載
    <b class='flag-5'>RT-Thread</b>文檔_<b class='flag-5'>RT-Thread</b> 潘多拉 STM32L475 上手指南

    RT-Thread文檔_RT-Thread SMP 介紹與移植

    RT-Thread文檔_RT-Thread SMP 介紹與移植
    發(fā)表于 02-22 18:31 ?9次下載
    <b class='flag-5'>RT-Thread</b>文檔_<b class='flag-5'>RT-Thread</b> SMP 介紹與移植

    RT-Thread文檔_CAN 設備

    RT-Thread文檔_CAN 設備
    發(fā)表于 02-22 18:34 ?0次下載
    <b class='flag-5'>RT-Thread</b>文檔_<b class='flag-5'>CAN</b> 設備

    基于RT-Thread Studio學習

    前期準備:從官網(wǎng)下載 RT-Thread Studio,弄個賬號登陸,開啟rt-thread學習之旅。
    的頭像 發(fā)表于 05-15 11:00 ?5242次閱讀
    基于<b class='flag-5'>RT-Thread</b> Studio學習

    4月10日深圳場RT-Thread線下workshop,探索RT-Thread混合部署新模式!

    4月10日我們將在深圳福田舉辦RT-Thread混合部署線下workshop,在瑞芯微RK3568平臺上實現(xiàn)同時運行RT-Thread和linux,本次workshop邀請到RT-Thread
    的頭像 發(fā)表于 03-27 11:36 ?1234次閱讀
    4月10日深圳場<b class='flag-5'>RT-Thread</b>線下workshop,探索<b class='flag-5'>RT-Thread</b>混合部署新模式!