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

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

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

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

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

最近遇到了如下需求:

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

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

大體思路:

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

代碼實(shí)現(xiàn)

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

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

初始化
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會(huì)崩潰。

初始化信號(hào)量,郵箱和內(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);

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

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

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

首先我們需要?jiǎng)?chuàng)建一個(gè)中斷函數(shù),這個(gè)中斷函數(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();
}

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

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

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ǐng)空間,如果沒有申請(qǐng)到(緩沖區(qū)已滿),就拋棄這條數(shù)據(jù)。

使用rt_spi_revice_message函數(shù)來啟動(dòng)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ā)送郵件
}
}
}

最后,還需要一個(gè)線程用于處理接收到的數(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從接收就基本完成了。這些只是一個(gè)大體的思路,也可以使用自己喜歡的方式,或者添加其他的功能。如果大家有更好的思路,歡迎分享出來。

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

    關(guān)注

    8

    文章

    1178

    瀏覽量

    53377
  • 中斷處理
    +關(guān)注

    關(guān)注

    0

    文章

    94

    瀏覽量

    10947
  • SPI接口
    +關(guān)注

    關(guān)注

    0

    文章

    258

    瀏覽量

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

    關(guān)注

    31

    文章

    1261

    瀏覽量

    39839
  • MCU芯片
    +關(guān)注

    關(guān)注

    3

    文章

    246

    瀏覽量

    11355
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    【S32K146 RT-thread】之 SPI驅(qū)動(dòng)適配

    概述RT-Thread對(duì)SPI總線的驅(qū)動(dòng),抽象出了spibus的設(shè)備驅(qū)動(dòng),我們基于S32K146的硬件學(xué)習(xí)spibus設(shè)備驅(qū)動(dòng)。
    的頭像 發(fā)表于 11-01 08:11 ?128次閱讀
    【S32K146 <b class='flag-5'>RT-thread</b>】之 <b class='flag-5'>SPI</b>驅(qū)動(dòng)適配

    2024 RT-Thread全球巡回 線下培訓(xùn)火熱來襲!

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

    RT-Thread 新里程碑達(dá)成——GitHub Star 破萬!

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

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

    6月6日下午我們將在杭州舉辦RT-Thread混合部署線下workshop,在瑞芯微RK3568平臺(tái)上實(shí)現(xiàn)同時(shí)運(yùn)行RT-Thread和linux,本次workshop邀請(qǐng)到RT-Thread資深
    的頭像 發(fā)表于 05-28 08:35 ?388次閱讀
    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平臺(tái)上實(shí)現(xiàn)同時(shí)運(yùn)行RT-Thread和linux,本次workshop邀請(qǐng)到RT-Thread資深
    的頭像 發(fā)表于 05-01 08:35 ?302次閱讀
    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平臺(tái)上實(shí)現(xiàn)同時(shí)運(yùn)行RT-Thread和linux,本次workshop邀請(qǐng)到RT-Thread資深
    的頭像 發(fā)表于 04-19 08:34 ?384次閱讀
    <b class='flag-5'>RT-Thread</b>混合部署Workshop北京站來啦!

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

    4月25日,下午我們將在北京舉辦RT-Thread混合部署線下workshop,在瑞芯微RK3568平臺(tái)上實(shí)現(xiàn)同時(shí)運(yùn)行RT-Thread和linux,本次workshop邀請(qǐng)到RT-Thread資深
    的頭像 發(fā)表于 04-16 08:35 ?382次閱讀
    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平臺(tái)上實(shí)現(xiàn)同時(shí)運(yùn)行RT-Thread和linux,本次workshop邀請(qǐng)到RT-Thread資深
    的頭像 發(fā)表于 04-04 08:34 ?285次閱讀
    【4月10日-深圳-workshop】<b class='flag-5'>RT-Thread</b>帶你探索混合部署新<b class='flag-5'>模式</b>

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

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

    4月10日深圳場(chǎng)RT-Thread線下workshop,探索RT-Thread混合部署新模式

    4月10日我們將在深圳福田舉辦RT-Thread混合部署線下workshop,在瑞芯微RK3568平臺(tái)上實(shí)現(xiàn)同時(shí)運(yùn)行RT-Thread和linux,本次workshop邀請(qǐng)到RT-Thread資深
    的頭像 發(fā)表于 03-27 08:34 ?464次閱讀
    4月10日深圳場(chǎng)<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平臺(tái)上的實(shí)現(xiàn)同時(shí)運(yùn)行RT-Thread和linux,本次培訓(xùn)邀請(qǐng)到RT-Thread資深
    的頭像 發(fā)表于 03-20 08:34 ?485次閱讀
    就在本周四!探索<b class='flag-5'>RT-Thread</b>混合部署新<b class='flag-5'>模式</b>!

    恩智浦半導(dǎo)體正式加入RT-Thread全球合作伙伴計(jì)劃!

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

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

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

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

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

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

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