色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

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

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

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

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

嵌入式應(yīng)用開發(fā) ? 來源:嵌入式應(yīng)用開發(fā) ? 作者:嵌入式應(yīng)用開發(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設(shè)備:rt_device_t rt_device_find(const char* name);

打開can設(shè)備: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接收線程

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

/*
 * 程序清單:這是一個(gè) CAN 設(shè)備使用例程
 * 例程導(dǎo)出了 can_sample 命令到控制終端
 * 命令調(diào)用格式:can_sample can1
 * 命令解釋:命令第二個(gè)參數(shù)是要使用的 CAN 設(shè)備名稱,為空則使用默認(rèn)的 CAN 設(shè)備
 * 程序功能:通過 CAN 設(shè)備發(fā)送一幀,并創(chuàng)建一個(gè)線程接收數(shù)據(jù)然后打印輸出。
*/
#include 
#include "rtdevice.h"
#define CAN_DEV_NAME       "can1"      /* CAN 設(shè)備名稱 */
static struct rt_semaphore rx_sem;     /* 用于接收消息的信號量 */
static rt_device_t can_dev;            /* CAN 設(shè)備句柄 */
/* 接收數(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};
    /* 設(shè)置接收回調(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 設(shè)備 */
    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 設(shè)備 */
    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;     /* 標(biāo)準(zhǔn)格式 */
    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;     /* 標(biāo)準(zhǔn)格式 */
    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");
    }
}
/* 導(dǎo)出到 msh 命令列表中 */
MSH_CMD_EXPORT(can_sample, can device sample);
MSH_CMD_EXPORT(can_send_test, can send test);

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

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

#include 
#include "rtdevice.h"

#define CAN_DEV_NAME       "can1"      /* CAN 設(shè)備名稱 */

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

#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};

    /* 設(shè)置接收回調(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;     /* 標(biāo)準(zhǔn)格式 */
    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)先級運(yùn)行,一直打印計(jì)數(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 設(shè)備 */
    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 設(shè)備 */

    res = rt_device_open(can_dev, RT_DEVICE_FLAG_INT_TX | RT_DEVICE_FLAG_INT_RX);
    /* 設(shè)置 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);

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

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

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

2020-12-08_215917.png

打開USB轉(zhuǎn)CAN,選擇對應(yīng)波特率,這里配置為1Mbps。

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

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

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

    關(guān)注

    57

    文章

    2756

    瀏覽量

    463893
  • RT-Thread
    +關(guān)注

    關(guān)注

    31

    文章

    1293

    瀏覽量

    40228
收藏 人收藏

    評論

    相關(guān)推薦

    RT-ThreadCAN實(shí)踐

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

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

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

    RT-Thread編程指南

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

    RT-Thread上的CAN總線介紹以及驅(qū)動(dòng)編寫

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

    基于 Keil MDK 移植 RT-Thread Nano

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

    RT-Thread全球技術(shù)大會:Kconfig在RT-Thread中的工作機(jī)制

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

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

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

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

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

    RT-Thread學(xué)習(xí)筆記 RT-Thread的架構(gòu)概述

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

    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 ?9次下載
    <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 設(shè)備

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

    基于RT-Thread Studio學(xué)習(xí)

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

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

    4月10日我們將在深圳福田舉辦RT-Thread混合部署線下workshop,在瑞芯微RK3568平臺上實(shí)現(xiàn)同時(shí)運(yùn)行RT-Thread和linux,本次workshop邀請到RT-Thread
    的頭像 發(fā)表于 03-27 11:36 ?837次閱讀
    4月10日深圳場<b class='flag-5'>RT-Thread</b>線下workshop,探索<b class='flag-5'>RT-Thread</b>混合部署新模式!
    主站蜘蛛池模板: 日本aaaa| 亚洲欧洲精品成人久久曰影片| 超碰免费视频caoporn| 最新老头恋老OLDMAN| 最新色导航| 德国黄色录像| 久久这里只有精品1| 国产女合集小岁9三部| 国产免费播放一区二区三区| 国产在线伊人| 幻女FREE性俄罗斯学生| 国产人在线成免费视频| 免费国产成人手机在线观看| 日韩精品一区VR观看| 一个人看的www视频动漫版| 亚洲精品无码国产爽快A片| 99久久精品久久久| 国产熟妇无码一区二| 久久黄视频| 强奷乱码欧妇女中文字幕熟女| 色www.亚洲免费视频| 日日操夜夜操狠狠操| 又黄又粗又爽免费观看| se01短视频在线观看| 久久国产免费一区二区三区| 无码AV熟妇素人内射V在线| 97人妻中文字幕免费视频| 国产99视频精品一区| 精品久久久久久电影网| 欧美巨大xxxx做受高清| 息与子在线交尾中文字幕| 中文字幕精品在线观看| 大香网伊人久久综合网2020| 么公一夜要了我一八次视频HD| 狼人射综合| 欧美又粗又长又大AAAA片| 伊人影院网| 俄罗斯一级毛片aaaa| 年轻的搜子8中字在线观看| 中文字幕AV亚洲精品影视| 精品无码国产AV一区二区三区|