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

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

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

RT-Thread SPI作為從模式接收數(shù)據(jù)的使用方法

冬至子 ? 來源:DCUU_8834 ? 作者:DCUU_8834 ? 2023-10-17 14:45 ? 次閱讀

最近遇到了如下需求:

MCU作為主控芯片通過SPI與藍牙芯片連接。
藍牙芯片會時不時向MCU發(fā)送大量定長的數(shù)據(jù)包。
這種情況下,如果MCU的SPI接口采用主模式,通過查詢的方式詢問藍牙芯片是否有數(shù)據(jù)要發(fā)送,就會非常占用資源,并且遇到突發(fā)大量數(shù)據(jù)也可能會來不及處理。

比較好的一種方法是,MCU采用從模式的SPI。藍牙芯片無腦向MCU吐數(shù)據(jù)。如果主控MCU的SPI時鐘最大頻率大于藍牙芯片的SPI最大頻率,此方法可以跑到藍牙芯片SPI的傳輸極限。

大體思路:

初始化SPI為從模式,并為SPI_CS引腳注冊中斷函數(shù),下降沿觸發(fā)
在中斷函數(shù)中,啟動SPI的接收。
SPI接收完成后,做其他處理,比如解析,轉(zhuǎn)發(fā)等

代碼實現(xiàn)

下面是如何實現(xiàn),平臺采用了STM32F1系列芯片,啟用SPI DMA傳輸,RT-Thread 4.0.2,SPI約定為Slave,MODE3,MSB,CS active low。一次傳輸長度為package_length。

使用內(nèi)存池+郵箱的緩沖方式,當然也可以使用消息隊列,根據(jù)自己的喜好。此處對中斷做了底半處理。

初始化
SPI初始化
static struct rt_spi_device spi_device; //
static struct stm32_hw_spi_cs spi_cs; //中斷引腳
static int spi_init(void)
{
rt_pin_mode(CS_PIN, PIN_MODE_INPUT_PULLUP);
/
attach the device to spi bus
/
spi_device = (struct rt_spi_device *)rt_malloc(sizeof(struct rt_spi_device));
RT_ASSERT(uwb_device != RT_NULL);
spi_cs = (struct stm32_hw_spi_cs )rt_malloc(sizeof(struct stm32_hw_spi_cs));
RT_ASSERT(uwb_spi_cs != RT_NULL);
spi_cs->GPIOx = GPIOA;
spi_cs->GPIO_Pin = 4;
result = rt_spi_bus_attach_device(uwb_device, "spi10", "spi1", (void )uwb_spi_cs);
if (result != RT_EOK)
{
LOG_E("%s attach to %s faild, %dn", "spi10", "spi1", result);
return result;
}
LOG_D("%s attach to %s done", UWB_SPI_NAME, UWB_SPI_BUS);
/
get SPI bus /
spi_device->bus->owner = spi_device;
/
configure SPI device
/
{
struct rt_spi_configuration cfg;
cfg.data_width = 8;
cfg.mode = RT_SPI_SLAVE | RT_SPI_MODE_3 | RT_SPI_MSB;
cfg.max_hz = 8 * 1000 * 1000;
rt_spi_configure(spi_device, &cfg);
}
if (rt_device_open((rt_device_t)spi_device, RT_DEVICE_FLAG_DMA_RX) != RT_EOK)
{
LOG_E("open UWB SPI device %s error.", "spi10");
return -RT_ERROR;
}
return RT_OK;
}
!?。∽⒁?,這里需要修改一下rt_spi_configure函數(shù)中的宏定義RT_SPI_MODE_MASK,從

(RT_SPI_CPHA | RT_SPI_CPOL | RT_SPI_MSB)
改為

(RT_SPI_SLAVE | RT_SPI_CPHA | RT_SPI_CPOL | RT_SPI_MSB)

否則無法將SPI接口配置為從模式,調(diào)用rt_spi_revice_message會崩潰。

初始化信號量,郵箱和內(nèi)存池
/* create RX semaphore /
spi_start_sem = rt_sem_create("spi1_start", 0, RT_IPC_FLAG_FIFO);
/
create RX mp /
spi_mp = rt_mp_create("spi_mp", SPI_MB_LEN, RT_ALIGN(sizeof(rt_uint8_t), sizeof(intptr_t)) * package_length);
/
create RX mailbox /
rt_mb_init(&spi_mb, "UWB_mb", &spi_mb_pool[0], sizeof(spi_mb_pool) / 4, RT_IPC_FLAG_FIFO);
為CS引腳綁定中斷函數(shù)
rt_pin_attach_irq(4, PIN_IRQ_MODE_FALLING, (void (
)(void *))spi_cs_isr, RT_NULL);

此時,可以先不使能中斷,可以等待系統(tǒng)所有初始工作完成后,由其他線程使能中斷,以啟動SPI接收。

到此,初始化的工作就完成了。接下來,要進行數(shù)據(jù)的接收工作,為此我們需要創(chuàng)建一些其他的函數(shù)。

數(shù)據(jù)的接收

首先我們需要創(chuàng)建一個中斷函數(shù),這個中斷函數(shù)通過CS引腳的下降沿觸發(fā),用來通知系統(tǒng)開始接收數(shù)據(jù)。

static void spi_cs_isr(void)
{
/* enter interrupt /
rt_interrupt_enter();
rt_sem_release(spi_start_sem);
/
leave interrupt */
rt_interrupt_leave();
}

然后,我們還需要:一個線程用來啟動DMA接收;一個中斷函數(shù)用于通知系統(tǒng)DMA接收已經(jīng)完成。

使用DMA方式的好處是,全部的SPI接收過程可以交給DMA。這種非阻塞的方式使得,系統(tǒng)在這個時候可以搞搞其他事情(相當于雙線程???)。在SPI大量傳輸數(shù)據(jù)時尤其好用。

static uint8_t *rx_buff = RT_NULL;
static void spi_start_thread_entry(void *parameter)
{
struct rt_spi_message spi_msg;
spi_msg.send_buf = RT_NULL;
spi_msg.length = uwb_package_length;
spi_msg.cs_take = 0;
spi_msg.cs_take = 0;
spi_msg.next = RT_NULL;
while (1)
{
if (rt_sem_take(spi_start_sem, RT_WAITING_FOREVER) == RT_EOK)
{
rx_buff = rt_mp_alloc(spi_mp, RT_WAITING_NO);
if (rx_buff != RT_NULL)
{
rt_spi_revice_message(spi_device, &spi_msg);
}
}
}
}

這里使用了RT_WAITING_NO的方式來申請空間,如果沒有申請到(緩沖區(qū)已滿),就拋棄這條數(shù)據(jù)。

使用rt_spi_revice_message函數(shù)來啟動DMA接收,并且約定了接收的長度固定為package_length。

DMA接收完成函數(shù)

void HAL_SPI_RxCpltCallback(SPI_HandleTypeDef *hspi)
{
if (hspi->Instance == SPI1)
{
if (rx_buff != RT_NULL)
{
rt_mb_send(&spi_mb, (rt_uint32_t)rx_buff); //發(fā)送郵件
}
}
}

最后,還需要一個線程用于處理接收到的數(shù)據(jù)

static void spi_dma_clp_thread_entry(void *parameter)
{
rt_uint8_t *net_tx_buff = RT_NULL;
while (1)
{
if (rt_mb_recv(&uwb_mb, (rt_ubase_t *)&net_tx_buff, RT_WAITING_FOREVER) == RT_EOK)
{
//TODO
//data process
}
rt_mp_free(net_tx_buff);
net_tx_buff = RT_NULL;
}
}

到此,基于RT-Thread的SPI從接收就基本完成了。這些只是一個大體的思路,也可以使用自己喜歡的方式,或者添加其他的功能。如果大家有更好的思路,歡迎分享出來。

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

    關注

    8

    文章

    1175

    瀏覽量

    53340
  • 中斷處理
    +關注

    關注

    0

    文章

    94

    瀏覽量

    10943
  • SPI接口
    +關注

    關注

    0

    文章

    258

    瀏覽量

    34293
  • RT-Thread
    +關注

    關注

    31

    文章

    1251

    瀏覽量

    39738
  • MCU芯片
    +關注

    關注

    3

    文章

    245

    瀏覽量

    11321
收藏 人收藏

    評論

    相關推薦

    2024 RT-Thread全球巡回 線下培訓火熱來襲!

    親愛的RT-Thread社區(qū)成員們:我們非常高興地宣布,2024年RT-Thread全球開發(fā)者線下培訓即將拉開帷幕!24年全球巡回培訓將覆蓋超10座城市及國家,為開發(fā)者提供一個深入學習RT-Thread嵌入式開發(fā)的絕佳機會。
    的頭像 發(fā)表于 08-07 08:35 ?727次閱讀
    2024 <b class='flag-5'>RT-Thread</b>全球巡回 線下培訓火熱來襲!

    RT-Thread 新里程碑達成——GitHub Star 破萬!

    RT-Thread實時操作系統(tǒng)開源項目在GitHub上的star數(shù)量突破一萬!截止發(fā)文,RT-Thread作為實時操作系統(tǒng)在業(yè)界Star數(shù)量排名第一!倉庫地址:https://github.com
    的頭像 發(fā)表于 07-04 08:35 ?344次閱讀
    <b class='flag-5'>RT-Thread</b> 新里程碑達成——GitHub Star 破萬!

    6月6日杭州站RT-Thread線下workshop,探索RT-Thread混合部署新模式!

    6月6日下午我們將在杭州舉辦RT-Thread混合部署線下workshop,在瑞芯微RK3568平臺上實現(xiàn)同時運行RT-Thread和linux,本次workshop邀請到RT-Thread資深
    的頭像 發(fā)表于 05-28 08:35 ?355次閱讀
    6月6日杭州站<b class='flag-5'>RT-Thread</b>線下workshop,探索<b class='flag-5'>RT-Thread</b>混合部署新<b class='flag-5'>模式</b>!

    5月16日南京站RT-Thread線下workshop,探索RT-Thread混合部署新模式!

    5月16日下午我們將在南京舉辦RT-Thread混合部署線下workshop,在瑞芯微RK3568平臺上實現(xiàn)同時運行RT-Thread和linux,本次workshop邀請到RT-Thread資深
    的頭像 發(fā)表于 05-01 08:35 ?280次閱讀
    5月16日南京站<b class='flag-5'>RT-Thread</b>線下workshop,探索<b class='flag-5'>RT-Thread</b>混合部署新<b class='flag-5'>模式</b>!

    RT-Thread混合部署Workshop北京站來啦!

    4月25日,下午我們將在北京舉辦RT-Thread混合部署線下workshop,在瑞芯微RK3568平臺上實現(xiàn)同時運行RT-Thread和linux,本次workshop邀請到RT-Thread資深
    的頭像 發(fā)表于 04-19 08:34 ?365次閱讀
    <b class='flag-5'>RT-Thread</b>混合部署Workshop北京站來啦!

    4月25日北京站RT-Thread線下workshop,探索RT-Thread混合部署新模式

    4月25日,下午我們將在北京舉辦RT-Thread混合部署線下workshop,在瑞芯微RK3568平臺上實現(xiàn)同時運行RT-Thread和linux,本次workshop邀請到RT-Thread資深
    的頭像 發(fā)表于 04-16 08:35 ?353次閱讀
    4月25日北京站<b class='flag-5'>RT-Thread</b>線下workshop,探索<b class='flag-5'>RT-Thread</b>混合部署新<b class='flag-5'>模式</b>

    【4月10日-深圳-workshop】RT-Thread帶你探索混合部署新模式

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

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

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

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

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

    就在本周四!探索RT-Thread混合部署新模式!

    3月21日(本周四)我們將在上海張江舉辦RT-Thread混合部署線下workshop,在瑞芯微RK3568平臺上的實現(xiàn)同時運行RT-Thread和linux,本次培訓邀請到RT-Thread資深
    的頭像 發(fā)表于 03-20 08:34 ?456次閱讀
    就在本周四!探索<b class='flag-5'>RT-Thread</b>混合部署新<b class='flag-5'>模式</b>!

    恩智浦半導體正式加入RT-Thread全球合作伙伴計劃!

    前不久,恩智浦半導體正式加入RT-Thread全球合作伙伴計劃,成為RT-Thread高級會員合作伙伴。同時,RT-Thread現(xiàn)已成為恩智浦注冊合作伙伴(RT-Thread| 簡介合
    的頭像 發(fā)表于 03-14 10:40 ?527次閱讀
    恩智浦半導體正式加入<b class='flag-5'>RT-Thread</b>全球合作伙伴計劃!

    RT-Thread設備驅(qū)動開發(fā)指南》基礎篇--以先楫bsp的hwtimer設備為例

    一、概述(一)RT-Thread設備驅(qū)動《RT-Thread設備驅(qū)動開發(fā)指南》書籍是RT-thread官方出品撰寫,系統(tǒng)講解RT-threadIO設備驅(qū)動開發(fā)
    的頭像 發(fā)表于 02-24 08:16 ?1206次閱讀
    《<b class='flag-5'>RT-Thread</b>設備驅(qū)動開發(fā)指南》基礎篇--以先楫bsp的hwtimer設備為例

    RT-Thread設備驅(qū)動開發(fā)指南基礎篇—以先楫bsp的hwtimer設備為例

    RT-Thread設備驅(qū)動開發(fā)指南》書籍是RT-thread官方出品撰寫,系統(tǒng)講解RT-thread IO設備驅(qū)動開發(fā)方法,三方面進行講解
    的頭像 發(fā)表于 02-20 16:01 ?1508次閱讀
    <b class='flag-5'>RT-Thread</b>設備驅(qū)動開發(fā)指南基礎篇—以先楫bsp的hwtimer設備為例

    RT-Thread Nano入門:串口接收與消息隊列

    本文主要介紹怎么用RT-Thread Nano的消息隊列方式實現(xiàn)串口數(shù)據(jù)接收,結合串口接收中斷和空閑中斷,接收上位機發(fā)來的一幀
    的頭像 發(fā)表于 11-22 11:07 ?3422次閱讀
    <b class='flag-5'>RT-Thread</b> Nano入門:串口<b class='flag-5'>接收</b>與消息隊列

    i.MX RT1170:VGLite移植RT-Thread Nano過程講解(下)

    上篇介紹了如何移植 RT-Thread Nano 內(nèi)核與 Finsh 控制臺到 RT1170。本篇繼續(xù)介紹如何將 NXP 官方的 VGLite API 移植到 RT-Thread Nano 上。
    的頭像 發(fā)表于 11-09 11:22 ?824次閱讀