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

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

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

RyanMqtt移植指南

冬至子 ? 來(lái)源:Ryan_CW ? 作者:Ryan_CW ? 2023-10-09 10:24 ? 次閱讀

測(cè)試環(huán)境:stm32F401RCT6、RT-Thread版本: v4.1.0、RT-Thread Studio版本: 2.2.6、網(wǎng)絡(luò)硬件使用ec800m移植at_socket使用sal框架。

1、移植介紹
RyanMqtt 庫(kù)希望應(yīng)用程序?yàn)橐韵?a target="_blank">接口提供實(shí)現(xiàn):

system 接口
RyanMqtt 需要 RTOS 支持,必須實(shí)現(xiàn)如下接口才可以保證 mqtt 客戶端的正常運(yùn)行

1.jpg

network 接口
RyanMqtt 依賴于底層傳輸接口 API,必須實(shí)現(xiàn)該接口 API 才能在網(wǎng)絡(luò)上發(fā)送和接收數(shù)據(jù)包

MQTT 協(xié)議要求基礎(chǔ)傳輸層能夠提供有序的、可靠的、雙向傳輸(從客戶端到服務(wù)端 和從服務(wù)端到客戶端)的字節(jié)流

1.jpg

time 接口
RyanMqtt 依靠函數(shù)生成毫秒時(shí)間戳,用于計(jì)算持續(xù)時(shí)間和超時(shí),內(nèi)部已經(jīng)做了數(shù)值溢出處理

1.jpg

2、開(kāi)始移植
得益于RT-Thread驅(qū)動(dòng)應(yīng)用層分離的思想和SAL框架,platform/rtthread的適配層可以適應(yīng)任何RT-Thread代碼,所以我們就不拿RT-Thread來(lái)移植了。

使用FreeRTOS內(nèi)核來(lái)移植,使用CMSIS-RTOS V2兼容層。

system 接口
系統(tǒng)接口,需要移植RTOS的接口。為方便管理類型使用平臺(tái)結(jié)構(gòu)體,修改platformSystem.h里面的結(jié)構(gòu)體
就是線程和互斥鎖

typedef struct
{
    osThreadId_t thread;
} platformThread_t;
typedef struct
{
    osMutexId_t mutex;
} platformMutex_t;

再來(lái)實(shí)現(xiàn)platformSystem.c里面的函數(shù)定義
注意里面的 platformPrint 函數(shù),由于FreeRTOS沒(méi)有官方的打印接口。記得修改為你的打印接口

#include "platformSystem.h"
/**

  • @brief 申請(qǐng)內(nèi)存
  • @param size
  • @return void*
    /
    void platformMemoryMalloc(size_t size)
    {
    return pvPortMalloc(size);
    }
    /
  • @brief 釋放內(nèi)存
  • @param ptr
    /
    void platformMemoryFree(void ptr)
    {
    vPortFree(ptr);
    }
    /
  • @brief ms延時(shí)
  • @param ms
    /
    void platformDelay(uint32_t ms)
    {
    osDelay(ms);
    }
    /
    *
  • @brief 打印字符串函數(shù),可通過(guò)串口打印出去
  • @param str
  • @param strLen
    /
    void platformPrint(char str, uint16_t strLen)
    {
    }
    /
  • @brief 初始化并運(yùn)行線程
  • @param userData
  • @param platformThread
  • @param name
  • @param entry
  • @param param
  • @param stackSize
  • @param priority
  • @return RyanMqttError_e
    */
    RyanMqttError_e platformThreadInit(void *userData,
    platformThread_t *platformThread,
    const char *name,
    void (*entry)(void ),
    void const param,
    uint32_t stackSize,
    uint32_t priority)
    {
    const osThreadAttr_t myTask02_attributes = {
    .name = name,
    .stack_size = stackSize,
    .priority = (osPriority_t)priority,
    };
    platformThread->thread = osThreadNew(entry, param, &myTask02_attributes);
    if (NULL == platformThread->thread)
    return RyanMqttNoRescourceError;
    return RyanMqttSuccessError;
    }
    /
  • @brief 銷(xiāo)毀自身線程
  • @param userData
  • @param platformThread
  • @return RyanMqttError_e
    */
    RyanMqttError_e platformThreadDestroy(void userData, platformThread_t platformThread)
    {
    osThreadTerminate(platformThread->thread);
    return RyanMqttSuccessError;
    }
    /
  • @brief 開(kāi)啟線程
  • @param userData
  • @param platformThread
  • @return RyanMqttError_e
    */
    RyanMqttError_e platformThreadStart(void userData, platformThread_t platformThread)
    {
    osThreadResume(platformThread->thread);
    return RyanMqttSuccessError;
    }
    /
  • @brief 掛起線程
  • @param userData
  • @param platformThread
  • @return RyanMqttError_e
    */
    RyanMqttError_e platformThreadStop(void userData, platformThread_t platformThread)
    {
    osThreadSuspend(platformThread->thread);
    return RyanMqttSuccessError;
    }
    /
  • @brief 互斥鎖初始化
  • @param userData
  • @param platformMutex
  • @return RyanMqttError_e
    */
    RyanMqttError_e platformMutexInit(void userData, platformMutex_t platformMutex)
    {
    const osMutexAttr_t myMutex01_attributes = {
    .name = "mqttMutex"};
    platformMutex->mutex = osMutexNew(&myMutex01_attributes);
    return RyanMqttSuccessError;
    }
    /
  • @brief 銷(xiāo)毀互斥鎖
  • @param userData
  • @param platformMutex
  • @return RyanMqttError_e
    */
    RyanMqttError_e platformMutexDestroy(void userData, platformMutex_t platformMutex)
    {
    osMutexDelete(platformMutex->mutex);
    return RyanMqttSuccessError;
    }
    /
  • @brief 阻塞獲取互斥鎖
  • @param userData
  • @param platformMutex
  • @return RyanMqttError_e
    */
    RyanMqttError_e platformMutexLock(void userData, platformMutex_t platformMutex)
    {
    osMutexAcquire(platformMutex->mutex, osWaitForever);
    return RyanMqttSuccessError;
    }
    /
  • @brief 釋放互斥鎖
  • @param userData
  • @param platformMutex
  • @return RyanMqttError_e
    */
    RyanMqttError_e platformMutexUnLock(void userData, platformMutex_t platformMutex)
    {
    osMutexRelease(platformMutex->mutex);
    return RyanMqttSuccessError;
    }
    /
  • @brief 進(jìn)入臨界區(qū) / 關(guān)中斷

/
void platformCriticalEnter(void)
{
osKernelLock();
}
/
*

  • @brief 退出臨界區(qū) / 開(kāi)中斷

*/
void platformCriticalExit(void)
{
osKernelUnlock();
}
time 接口
time接口,只需要提供一個(gè)ms時(shí)間戳就行,直接修改函數(shù),這里使用FreeRTOS的心跳。

uint32_t platformUptimeMs(void)
{
if (1000 == osKernelGetTickFreq())
return (uint32_t)osKernelGetTickCount();
else
{
uint32_t tick = 0;
tick = osKernelGetTickCount() * 1000;
return (uint32_t)((tick + osKernelGetTickCount() - 1) / osKernelGetTickCount());
}
}
network 接口
MQTT 協(xié)議要求基礎(chǔ)傳輸層能夠提供有序的、可靠的、雙向傳輸(從客戶端到服務(wù)端 和從服務(wù)端到客戶端)的字節(jié)流
由于FreeRTOS沒(méi)有規(guī)定標(biāo)準(zhǔn)的網(wǎng)絡(luò)層,你可以選擇 FreeRTOS-Plus-TCP / FreeRTOS-Cellular-Interface/ lwip / W5500等網(wǎng)絡(luò)方法,幾乎RT-Thread支持的你也可以在FreeRTOS倉(cāng)庫(kù)找到。

這里以lwip為例,使用socket接口來(lái)實(shí)現(xiàn),網(wǎng)絡(luò)阻塞發(fā)送和接收使用 SO_SNDTIMEO 和 SO_RCVTIMEO 來(lái)實(shí)現(xiàn),你也可以選擇select / poll / epoll等方式。

修改 platformNetwork_t 結(jié)構(gòu)體以支持 socket

typedef struct
{
    int socket;
} platformNetwork_t;

接著實(shí)現(xiàn)platformNetwork.c里面的函數(shù)

#define rlogEnable 1 // 是否使能日志
#define rlogColorEnable 1 // 是否使能日志顏色
#define rlogLevel (rlogLvlWarning) // 日志打印等級(jí)
#define rlogTag "RyanMqttNet" // 日志tag
#include "platformNetwork.h"
#include "RyanMqttLog.h"
/**

  • @brief 連接mqtt服務(wù)器
  • @param userData
  • @param platformNetwork
  • @param host
  • @param port
  • @return RyanMqttError_e
  • 成功返回RyanMqttSuccessError, 失敗返回錯(cuò)誤信息
    */
    RyanMqttError_e platformNetworkConnect(void *userData, platformNetwork_t *platformNetwork, const char *host, const char port)
    {
    RyanMqttError_e result = RyanMqttSuccessError;
    struct addrinfo addrList = NULL;
    struct addrinfo hints = {
    .ai_family = AF_UNSPEC,
    .ai_socktype = SOCK_STREAM,
    .ai_protocol = IPPROTO_TCP};
    if (getaddrinfo(host, port, &hints, &addrList) != 0)
    {
    result = RyanSocketFailedError;
    goto exit;
    }
    platformNetwork->socket = socket(addrList->ai_family, addrList->ai_socktype, addrList->ai_protocol);
    if (platformNetwork->socket < 0)
    {
    result = RyanSocketFailedError;
    goto exit;
    }
    if (connect(platformNetwork->socket, addrList->ai_addr, addrList->ai_addrlen) != 0)
    {
    platformNetworkClose(userData, platformNetwork);
    result = RyanMqttSocketConnectFailError;
    goto exit;
    }
    exit:
    if (NULL != addrList)
    freeaddrinfo(addrList);
    return result;
    }
    /
  • @brief 非阻塞接收數(shù)據(jù)
  • @param userData
  • @param platformNetwork
  • @param recvBuf
  • @param recvLen
  • @param timeout
  • @return RyanMqttError_e
  • socket錯(cuò)誤返回 RyanSocketFailedError
  • 接收超時(shí)或者接收數(shù)據(jù)長(zhǎng)度不等于期待數(shù)據(jù)接受長(zhǎng)度 RyanMqttRecvPacketTimeOutError
  • 接收成功 RyanMqttSuccessError
    */
    RyanMqttError_e platformNetworkRecvAsync(void *userData, platformNetwork_t *platformNetwork, char recvBuf, int recvLen, int timeout)
    {
    int32_t recvResult = 0;
    int32_t offset = 0;
    int32_t timeOut2 = timeout;
    struct timeval tv = {0};
    platformTimer_t timer = {0};
    if (-1 == platformNetwork->socket)
    return RyanSocketFailedError;
    platformTimerCutdown(&timer, timeout);
    while ((offset < recvLen) && (0 != timeOut2))
    {
    tv.tv_sec = timeOut2 / 1000;
    tv.tv_usec = timeOut2 % 1000 * 1000;
    if (tv.tv_sec <= 0 && tv.tv_usec <= 100)
    {
    tv.tv_sec = 0;
    tv.tv_usec = 100;
    }
    setsockopt(platformNetwork->socket, SOL_SOCKET, SO_RCVTIMEO, (char )&tv, sizeof(struct timeval)); // 設(shè)置錯(cuò)做模式為非阻塞
    recvResult = recv(platformNetwork->socket, recvBuf + offset, recvLen - offset, 0);
    if (recvResult <= 0) // 小于零,表示錯(cuò)誤,個(gè)別錯(cuò)誤不代表socket錯(cuò)誤
    {
    // 下列3種表示沒(méi)問(wèn)題,但需要推出發(fā)送
    if ((errno == EAGAIN || // 套接字已標(biāo)記為非阻塞,而接收操作被阻塞或者接收超時(shí)
    errno == EWOULDBLOCK || // 發(fā)送時(shí)套接字發(fā)送緩沖區(qū)已滿,或接收時(shí)套接字接收緩沖區(qū)為空
    errno == EINTR)) // 操作被信號(hào)中斷
    break;
    return RyanSocketFailedError;
    }
    offset += recvResult;
    timeOut2 = platformTimerRemain(&timer);
    }
    if (offset != recvLen)
    return RyanMqttRecvPacketTimeOutError;
    return RyanMqttSuccessError;
    }
    /
  • @brief 非阻塞發(fā)送數(shù)據(jù)
  • @param userData
  • @param platformNetwork
  • @param sendBuf
  • @param sendLen
  • @param timeout
  • @return RyanMqttError_e
  • socket錯(cuò)誤返回 RyanSocketFailedError
  • 接收超時(shí)或者接收數(shù)據(jù)長(zhǎng)度不等于期待數(shù)據(jù)接受長(zhǎng)度 RyanMqttRecvPacketTimeOutError
  • 接收成功 RyanMqttSuccessError
    */
    RyanMqttError_e platformNetworkSendAsync(void *userData, platformNetwork_t *platformNetwork, char sendBuf, int sendLen, int timeout)
    {
    int32_t sendResult = 0;
    int32_t offset = 0;
    int32_t timeOut2 = timeout;
    struct timeval tv = {0};
    platformTimer_t timer = {0};
    if (-1 == platformNetwork->socket)
    return RyanSocketFailedError;
    platformTimerCutdown(&timer, timeout);
    while ((offset < sendLen) && (0 != timeOut2))
    {
    tv.tv_sec = timeOut2 / 1000;
    tv.tv_usec = timeOut2 % 1000 * 1000;
    if (tv.tv_sec <= 0 && tv.tv_usec <= 100)
    {
    tv.tv_sec = 0;
    tv.tv_usec = 100;
    }
    setsockopt(platformNetwork->socket, SOL_SOCKET, SO_SNDTIMEO, (char )&tv, sizeof(struct timeval)); // 設(shè)置錯(cuò)做模式為非阻塞
    sendResult = send(platformNetwork->socket, sendBuf + offset, sendLen - offset, 0);
    if (sendResult <= 0) // 小于零,表示錯(cuò)誤,個(gè)別錯(cuò)誤不代表socket錯(cuò)誤
    {
    // 下列3種表示沒(méi)問(wèn)題,但需要推出發(fā)送
    if ((errno == EAGAIN || // 套接字已標(biāo)記為非阻塞,而接收操作被阻塞或者接收超時(shí)
    errno == EWOULDBLOCK || // 發(fā)送時(shí)套接字發(fā)送緩沖區(qū)已滿,或接收時(shí)套接字接收緩沖區(qū)為空
    errno == EINTR)) // 操作被信號(hào)中斷
    break;
    return RyanSocketFailedError;
    }
    offset += sendResult;
    timeOut2 = platformTimerRemain(&timer);
    }
    if (offset != sendLen)
    return RyanMqttSendPacketTimeOutError;
    return RyanMqttSuccessError;
    }
    /

    @brief 斷開(kāi)mqtt服務(wù)器連接

@param userData
@param platformNetwork
@return RyanMqttError_e
*/
RyanMqttError_e platformNetworkClose(void *userData, platformNetwork_t *platformNetwork)
{
if (platformNetwork->socket >= 0)
{
closesocket(platformNetwork->socket);
platformNetwork->socket = -1;
}
return RyanMqttSuccessError;
}

3、總結(jié)
可以看到,RyanMqtt移植非常簡(jiǎn)單,有專門(mén)的platform層用來(lái)移植。

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

    關(guān)注

    12

    文章

    483

    瀏覽量

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

    關(guān)注

    31

    文章

    1239

    瀏覽量

    39434
  • STM32F401
    +關(guān)注

    關(guān)注

    1

    文章

    16

    瀏覽量

    10435
  • MQTT協(xié)議
    +關(guān)注

    關(guān)注

    0

    文章

    93

    瀏覽量

    5308
  • TCP通信
    +關(guān)注

    關(guān)注

    0

    文章

    146

    瀏覽量

    4184
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    RyanMqtt使用介紹和示例代碼(1)

    此步驟不做過(guò)多解釋,是lwip就用lwip,是at設(shè)備就用at_socket。推薦所有平臺(tái)都使用SAL框架(RyanMqtt軟件包會(huì)自動(dòng)使能)。
    的頭像 發(fā)表于 09-28 10:09 ?1164次閱讀
    <b class='flag-5'>RyanMqtt</b>使用介紹和示例代碼(1)

    RyanMqtt實(shí)現(xiàn)MQTT3.1.1協(xié)議的客戶端

    RyanMqttgithub: https://github.com/Ryan-CW-Code/RyanMqtt此庫(kù)已經(jīng)制作軟件包提pr給RT-Thread/packages,機(jī)器審核已通過(guò),但
    發(fā)表于 12-01 15:19

    emWin移植指南手冊(cè)

    emWin移植指南:EA LPC1788 BSP到Keil MCB1700
    發(fā)表于 12-09 06:55

    RyanMqtt在間隔1s發(fā)送消息時(shí)常就報(bào)錯(cuò)RyanSocketFailedError

    使用RyanMqtt庫(kù)以及例子,在間隔1s發(fā)送消息時(shí)常就報(bào)錯(cuò)RyanSocketFailedError,然后就進(jìn)入重連機(jī)制
    發(fā)表于 08-04 16:48

    ATWILC器件Linux移植指南

    本用戶指南介紹了如何將 ATWILC1000 和 ATWILC3000 Linux 驅(qū)動(dòng)程序移植到另一個(gè)平臺(tái),以及移植驅(qū)動(dòng)程序需要進(jìn)行哪些修改
    發(fā)表于 04-29 10:10 ?8次下載

    TDE移植指南

    電子發(fā)燒友網(wǎng)站提供《TDE移植指南.pdf》資料免費(fèi)下載
    發(fā)表于 09-27 11:09 ?1次下載
    TDE<b class='flag-5'>移植</b><b class='flag-5'>指南</b>

    AJE移植指南

    電子發(fā)燒友網(wǎng)站提供《AJE移植指南.pdf》資料免費(fèi)下載
    發(fā)表于 09-27 11:10 ?1次下載
    AJE<b class='flag-5'>移植</b><b class='flag-5'>指南</b>

    UM-B-097:681 至 683 移植指南

    UM-B-097:681 至 683 移植指南
    發(fā)表于 03-14 20:09 ?0次下載
    UM-B-097:681 至 683 <b class='flag-5'>移植</b><b class='flag-5'>指南</b>

    OpenHarmony富設(shè)備移植指南(1)導(dǎo)言

    OpenHarmony富設(shè)備移植指南導(dǎo)言。在研究學(xué)習(xí)OpenHamony移植的路上,文檔資料的缺失讓我倍感痛苦,如今移植樹(shù)莓派4b以及小米6的成功讓我確信我的
    的頭像 發(fā)表于 02-06 14:04 ?1131次閱讀
    OpenHarmony富設(shè)備<b class='flag-5'>移植</b><b class='flag-5'>指南</b>(1)導(dǎo)言

    OpenHarmony富設(shè)備移植指南(2)從postmarketOS獲取移植資源

    OpenHarmony富設(shè)備移植指南(2)從postmarketOS獲取移植資源
    的頭像 發(fā)表于 02-08 10:58 ?2009次閱讀
    OpenHarmony富設(shè)備<b class='flag-5'>移植</b><b class='flag-5'>指南</b>(2)從postmarketOS獲取<b class='flag-5'>移植</b>資源

    UM-B-097:681 至 683 移植指南

    UM-B-097:681 至 683 移植指南
    發(fā)表于 07-05 20:36 ?0次下載
    UM-B-097:681 至 683 <b class='flag-5'>移植</b><b class='flag-5'>指南</b>

    N76E003系列到MS51系列的移植指南

    N76E003系列到MS51系列的移植指南
    的頭像 發(fā)表于 08-10 16:22 ?972次閱讀
    N76E003系列到MS51系列的<b class='flag-5'>移植</b><b class='flag-5'>指南</b>

    PN7160安卓移植指南

    電子發(fā)燒友網(wǎng)站提供《PN7160安卓移植指南.pdf》資料免費(fèi)下載
    發(fā)表于 08-17 11:40 ?3次下載
    PN7160安卓<b class='flag-5'>移植</b><b class='flag-5'>指南</b>

    Vitis HLS移植指南

    電子發(fā)燒友網(wǎng)站提供《Vitis HLS移植指南.pdf》資料免費(fèi)下載
    發(fā)表于 09-13 09:21 ?0次下載
    Vitis HLS<b class='flag-5'>移植</b><b class='flag-5'>指南</b>

    【鴻蒙】標(biāo)準(zhǔn)系統(tǒng)移植指南

    標(biāo)準(zhǔn)系統(tǒng)移植指南 本文描述了移植一塊開(kāi)發(fā)板的通用步驟,和具體芯片相關(guān)的詳細(xì)移植過(guò)程無(wú)法在此一一列舉。后續(xù)社區(qū)還會(huì)陸續(xù)發(fā)布開(kāi)發(fā)板移植的實(shí)例供開(kāi)
    的頭像 發(fā)表于 02-27 14:36 ?665次閱讀
    【鴻蒙】標(biāo)準(zhǔn)系統(tǒng)<b class='flag-5'>移植</b><b class='flag-5'>指南</b>