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

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

3天內不再提示

如何通過IPC通信玩轉傳感器數(shù)據?

RTThread物聯(lián)網操作系統(tǒng) ? 來源:RTThread物聯(lián)網操作系統(tǒng) ? 作者:RTThread物聯(lián)網操作 ? 2021-01-15 13:39 ? 次閱讀

1、rt-smart的第一個應用程序,imx6ull用戶態(tài)點燈

簡介

首先糾正一下上一篇文章中,在我的倉庫中,1月11日的代碼會出現(xiàn)系統(tǒng)崩潰。原因在于我的驅動中內存物理地址映射到虛擬地址的操作有問題,我已經把這個bug解決了,如果有興趣,歡迎拉取最新的代碼。

這一篇來介紹我在rt-smart的第二個應用。這個應用將加入rt-smart與rt-thread區(qū)別之處--進程間的通信。

功能主要是在用戶態(tài)讀取傳感器數(shù)據,傳感器是100ASK_imx6ull板載的ap3216c,它是采用I2C總線進行通信。

為啥這次會先對接I2C呢?因為接下來想把屏幕在rt-smart跑起來,但是屏幕的觸摸芯片采用I2C,所以就把他先跑起來。

目前屏幕已經在rt-thread上跑起來,但是在rt-smart沒有跑起來,目前在研究LCD的緩存是一個什么樣一個形式。

100ask_imx6ull驅動對接情況:

rt-threadrt-smart

GPIO√√

I2C√√

lcd√×

100ask_imx6ull的rtt倉庫:

rt-thread的倉庫:https://gitee.com/RiceChen0/imx6ull_rt_rthread

rt-smart的廠庫:https://gitee.com/RiceChen0/imx6ull_rt_smart

環(huán)境

100ask_imx6ull開發(fā)板。

兩條micro USB線。

電源。

windows電腦一臺。

I2C驅動適配

在imx6ull中,我適配的是硬件I2C,imx6ull有4組I2C接口。軟件I2C后續(xù)不會進行適配,因為在這顆芯片上,軟件I2C的必要性不大。

如果你要了解RT-Thread的I2C設備驅動框架,可以看一下我之前的文章《rt-thread驅動框架分析》-i2c驅動

在上一篇文章中《rt-smart的第一個應用程序,imx6ull用戶態(tài)點燈》講到,rt-smart不能直接使用物理地址訪問硬件,而需要采用虛擬地址。所以需要進行地址映射(rtt提供的API:rt_hw_kernel_phys_to_virt)。

首先需要查看imx6ull的芯片手冊,需要將I2C相關的物理地址找到。為了不要重復造輪子,定義了一個結構體:struct i2c_addr_config,并把4組I2C相關的地址作為一個表格。如下:

#define I2C1_SCL_MUX_BASE 0x020E00B4U

#define I2C2_SCL_MUX_BASE 0x020E00BCU

#define I2C3_SCL_MUX_BASE 0x020E00E4U

#define I2C4_SCL_MUX_BASE 0x020E00ECU

#define I2C1_SCL_CFG_BASE 0x020E0340U

#define I2C2_SCL_CFG_BASE 0x020E0348U

#define I2C3_SCL_CFG_BASE 0x020E0370U

#define I2C4_SCL_CFG_BASE 0x020E0378U

#define I2C1_SCL_INPUT_BASE 0x020E05A4U

#define I2C2_SCL_INPUT_BASE 0x020E05ACU

#define I2C3_SCL_INPUT_BASE 0x020E05B4U

#define I2C4_SCL_INPUT_BASE 0x020E05BCU

#define I2C1_SDA_MUX_BASE 0x020E00B8U

#define I2C2_SDA_MUX_BASE 0x020E00C0U

#define I2C3_SDA_MUX_BASE 0x020E00E8U

#define I2C4_SDA_MUX_BASE 0x020E00F0U

#define I2C1_SDA_CFG_BASE 0x020E0344U

#define I2C2_SDA_CFG_BASE 0x020E034CU

#define I2C3_SDA_CFG_BASE 0x020E0374U

#define I2C4_SDA_CFG_BASE 0x020E037CU

#define I2C1_SDA_INPUT_BASE 0x020E05A8U

#define I2C2_SDA_INPUT_BASE 0x020E05B0U

#define I2C3_SDA_INPUT_BASE 0x020E05B8U

#define I2C4_SDA_INPUT_BASE 0x020E05C0U

struct i2c_addr_config

{

I2C_Type *i2c;

size_t i2c_scl_mux_base;

size_t i2c_scl_config_base;

size_t i2c_scl_input_base;

size_t i2c_sda_mux_base;

size_t i2c_sda_config_base;

size_t i2c_sda_input_base

};

static struct i2c_addr_config addr_config[] =

{

{I2C1, I2C1_SCL_MUX_BASE, I2C1_SCL_CFG_BASE, I2C1_SCL_INPUT_BASE, I2C1_SDA_MUX_BASE, I2C1_SDA_CFG_BASE, I2C1_SDA_INPUT_BASE},

{I2C2, I2C2_SCL_MUX_BASE, I2C2_SCL_CFG_BASE, I2C2_SCL_INPUT_BASE, I2C2_SDA_MUX_BASE, I2C2_SDA_CFG_BASE, I2C2_SDA_INPUT_BASE},

{I2C3, I2C3_SCL_MUX_BASE, I2C3_SCL_CFG_BASE, I2C3_SCL_INPUT_BASE, I2C3_SDA_MUX_BASE, I2C3_SDA_CFG_BASE, I2C3_SDA_INPUT_BASE},

{I2C4, I2C4_SCL_MUX_BASE, I2C4_SCL_CFG_BASE, I2C4_SCL_INPUT_BASE, I2C4_SDA_MUX_BASE, I2C4_SDA_CFG_BASE, I2C4_SDA_INPUT_BASE},

};

將物理地址轉為虛擬地址,代碼如下:

for(i = 0; i 《 sizeof(addr_config) / sizeof(addr_config[0]); i++)

{

addr_config[i].i2c = (I2C_Type *)rt_hw_kernel_phys_to_virt((void*)(addr_config[i].i2c), 0x1000);

addr_config[i].i2c_scl_mux_base = (size_t)rt_hw_kernel_phys_to_virt((void*)(addr_config[i].i2c_scl_mux_base), 0x1000);

addr_config[i].i2c_scl_config_base = (size_t)rt_hw_kernel_phys_to_virt((void*)(addr_config[i].i2c_scl_config_base), 0x1000);

addr_config[i].i2c_scl_input_base = (size_t)rt_hw_kernel_phys_to_virt((void*)(addr_config[i].i2c_scl_input_base), 0x1000);

addr_config[i].i2c_sda_mux_base = (size_t)rt_hw_kernel_phys_to_virt((void*)(addr_config[i].i2c_sda_mux_base), 0x1000);

addr_config[i].i2c_sda_config_base = (size_t)rt_hw_kernel_phys_to_virt((void*)(addr_config[i].i2c_sda_config_base), 0x1000);

addr_config[i].i2c_sda_input_base = (size_t)rt_hw_kernel_phys_to_virt((void*)(addr_config[i].i2c_sda_input_base), 0x1000);

}

在imx6ull中,I2C需要的步驟,引腳初始化為I2C,然后I2C總線初始化便可以了。

目前imx6ull上,rt-thread和rt-smart都適配I2C,所以可以先看一下rt-thread的倉庫,然后再看rt-smart的倉庫,可能更加理解它的區(qū)別。

I2C的應用:

100ask_imx6ull中,板載有ap3216c傳感器,掛載在I2C1總線上。而且RT-Thread中有相應的軟件包,對接了RT-Thread的傳感器設備框架,這給我驗證代碼提供便攜。不過要在用戶態(tài)中使用該軟件包,還需要做一點操作,需要注冊該傳感器設備。

int ap3216c_test()

{

struct rt_sensor_config cfg;

cfg.intf.dev_name = “i2c1”;

cfg.mode = RT_SENSOR_MODE_POLLING;

rt_hw_ap3216c_init(“ap”, &cfg);

return RT_EOK;

}

INIT_DEVICE_EXPORT(ap3216c_test);

然后編譯燒錄,通過list_device就可以看到相對應的設備(pr_ap和li_ap),如下:

29be56fa-56f1-11eb-8b86-12bb97331649.png

RT_Thread的傳感器框架很貼心,提供了測試命令(sensor_polling li_ap),這樣就可以初步驗證傳感器是否正常工作,通過驗證,傳感器和I2C適配都能正常工作:

29dbdc34-56f1-11eb-8b86-12bb97331649.png

上面的驗證都是在內核態(tài)中測試的,而這篇文章的目的是要在用戶態(tài)中讀取傳感器數(shù)據,為了進一步了解rt-smart和RT-Thread的區(qū)別,我這個應用采用進程通信(IPC)做了例子,該例子將上一篇文章例子結合起來:

有兩個進程, 進程1和進程2

進程1,通過接收等待進程2讀取的傳感器數(shù)據是否超標的狀態(tài),來進行閃燈。

進程2,通過讀取ap3216c傳感器光強度數(shù)據,判斷是否超過50lux,如果超過則通知進程1進行閃燈提示。

IPC通信,詳情可以查看官網:https://www.rt-thread.org/document/site/rt-smart/architecture/architecture/。

進程1,等待接收通道發(fā)來的“warning”信息,然后進行閃燈操作:

int main(int argc, char **argv)

{

struct rt_device_pin_mode pin_mode;

struct rt_device_pin_status pin_status;

int server_ch;

int shmid;

struct rt_channel_msg msg_text;

char *str;

printf(“RiceChen rt-smart first app

”);

/* create the IPC channel for ‘server’ */

server_ch = rt_channel_open(“server”, O_CREAT);

if (server_ch == -1) {

printf(“Error: rt_channel_open: fail to create the IPC channel for server!

”);

return -1;

}

printf(“

server: wait on the IPC channel: %d

”, server_ch);

pin_dev = rt_device_find(“pin”);

if(pin_dev == RT_NULL)

{

printf(“not find pin device

”);

return RT_ERROR;

}

rt_device_open(pin_dev, RT_DEVICE_OFLAG_RDWR);

pin_mode.pin = LED_PIN;

pin_mode.mode = 0; //OUTPUT

rt_device_control(pin_dev, 0, (void *)&pin_mode);

pin_status.pin = LED_PIN;

while(1)

{

rt_channel_recv(server_ch, &msg_text); //接收通道信息

shmid = (int)msg_text.u.d;

if (shmid 《 0 || ?。╯tr = (char *)lwp_shmat(shmid, NULL)))

{

msg_text.u.d = (void *)-1;

printf(“server: receive an invalid shared-memory page.

”);

rt_channel_reply(server_ch, &msg_text); /* send back -1 */

continue;

}

if(strcmp(str, “warning”) == 0) //判斷是否接收到“warning”信息

{

printf(“l(fā)ight warning.

”);

pin_status.status = 1;

rt_device_write(pin_dev, 0, (void *)&pin_status, sizeof(pin_status));

rt_thread_mdelay(200);

pin_status.status = 0;

rt_device_write(pin_dev, 0, (void *)&pin_status, sizeof(pin_status));

rt_thread_mdelay(200);

pin_status.status = 1;

rt_device_write(pin_dev, 0, (void *)&pin_status, sizeof(pin_status));

rt_thread_mdelay(200);

pin_status.status = 0;

rt_device_write(pin_dev, 0, (void *)&pin_status, sizeof(pin_status));

rt_thread_mdelay(200);

}

lwp_shmdt(str);

msg_text.type = RT_CHANNEL_RAW;

msg_text.u.d = (void *)1;

rt_channel_reply(server_ch, &msg_text);

}

return 0;

}

進程2,間隔兩面讀取一次傳感器光強度數(shù)據,然后判斷是否操作50lux,超過則通過通道通知進程1進行閃燈:

int main(int argc, char **argv)

{

rt_device_t ap3216c_dev;

struct rt_sensor_data sensor_data;

int res;

int server_ch;

char warning_msg[256] = { 0 };

size_t len = 0;

/* channel messages to send and return back */

struct rt_channel_msg ch_msg, ch_msg_ret;

printf(“RiceChen rt-smart second app

”);

/* open the IPC channel created by ‘pong’ */

server_ch = rt_channel_open(“server”, 0);

if (server_ch == -1)

{

printf(“Error: rt_channel_open: could not find the ‘server’ channel!

”);

return -1;

}

ap3216c_dev = rt_device_find(SENSOR_NAME);

if (ap3216c_dev == RT_NULL)

{

rt_kprintf(“Can‘t find device:%s”, SENSOR_NAME);

return -1;

}

if (rt_device_open(ap3216c_dev, RT_DEVICE_FLAG_RDWR) != RT_EOK)

{

rt_kprintf(“open device failed!”);

return -1;

}

rt_device_control(ap3216c_dev, RT_SENSOR_CTRL_SET_ODR, (void *)100);

while(1)

{

res = rt_device_read(ap3216c_dev, 0, &sensor_data, 1); //讀取傳感器數(shù)值

if (res != 1)

{

rt_kprintf(“read data failed!size is %d

”, res);

}

else

{

rt_kprintf(“l(fā)ight:%5d lux, timestamp:%5d

”, sensor_data.light, sensor_data.timestamp);

}

if(sensor_data.light 》 50) //判斷閾值

{

ch_msg.type = RT_CHANNEL_RAW;

snprintf(warning_msg, 255, “%s”, “warning”);

len = strlen(warning_msg) + 1;

warning_msg[len] = ’‘;

int shmid = prepare_data(warning_msg, len);

if (shmid 《 0)

{

printf(“clent: fail to prepare the clent message.

”);

continue;

}

ch_msg.u.d = (void *)shmid;

rt_channel_send_recv(server_ch, &ch_msg, &ch_msg_ret); //發(fā)送警報信息

lwp_shmrm(shmid);

}

rt_thread_mdelay(2000);

}

rt_device_close(ap3216c_dev);

rt_channel_close(server_ch);

return 0;

}

演示

原文標題:rt-smart用戶態(tài)通過IPC通信玩轉傳感器數(shù)據

文章出處:【微信公眾號:RTThread物聯(lián)網操作系統(tǒng)】歡迎添加關注!文章轉載請注明出處。

責任編輯:haq

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

    關注

    2541

    文章

    49961

    瀏覽量

    747507
  • 通信
    +關注

    關注

    18

    文章

    5880

    瀏覽量

    135316
  • 操作系統(tǒng)

    關注

    37

    文章

    6545

    瀏覽量

    122743
  • IPC
    IPC
    +關注

    關注

    3

    文章

    333

    瀏覽量

    51620

原文標題:rt-smart用戶態(tài)通過IPC通信玩轉傳感器數(shù)據

文章出處:【微信號:RTThread,微信公眾號:RTThread物聯(lián)網操作系統(tǒng)】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    傳感器數(shù)據怎么傳到云平臺

    傳感器數(shù)據傳到云平臺,通??梢?b class='flag-5'>通過以下幾種方式實現(xiàn): 1. 直接連接 網絡接口 :傳感器可以通過以太網、Wi-Fi等網絡接口直接連接到云服
    的頭像 發(fā)表于 09-07 10:59 ?332次閱讀

    車載傳感器網絡是什么意思啊

    車載傳感器網絡(Vehicle Sensor Networks,VSN)是指在車輛上部署的傳感器網絡,用于收集車輛運行狀態(tài)、環(huán)境信息、交通狀況等數(shù)據,并通過無線
    的頭像 發(fā)表于 09-07 09:32 ?157次閱讀

    lidar傳感器和激光測距傳感器的區(qū)別

    基于激光雷達技術的傳感器,它通過發(fā)射激光脈沖并接收反射回來的激光信號來測量距離。LiDAR傳感器可以同時測量多個點的距離,從而生成三維空間的點云數(shù)據。 激光測距
    的頭像 發(fā)表于 08-29 15:56 ?247次閱讀

    傳感器數(shù)據采集平臺是什么

    傳感器數(shù)據采集平臺是一種系統(tǒng),用于收集來自各種傳感器數(shù)據顯示和處理。這些傳感器可以監(jiān)測環(huán)境條件、物理參數(shù)、生物特征等眾多類型的信息。
    的頭像 發(fā)表于 06-13 09:26 ?347次閱讀

    請問傳感器怎么和STM32實現(xiàn)modbus rtu通信?

    第一次接觸Modbus,傳感器是采用Modbus協(xié)議的485信號,我用STM32做主機,想把傳感器采集的數(shù)據顯示到STM32觸摸屏上,要怎么實現(xiàn)modbus通信啊,有木有具體的例程可以
    發(fā)表于 05-17 08:57

    HarmonyOS跨進程通信IPC與RPC通信開發(fā)

    一、IPC與RPC通信概述 基本概念 IPC(Inter-Process Communication)與RPC(Remote Procedure Call)用于實現(xiàn)跨進程通信,不同的是
    的頭像 發(fā)表于 02-02 17:47 ?1094次閱讀
    HarmonyOS跨進程<b class='flag-5'>通信</b>—<b class='flag-5'>IPC</b>與RPC<b class='flag-5'>通信</b>開發(fā)

    如何通過仿真動圖理解各種傳感器的原理呢?

    如何通過仿真動圖理解各種傳感器的原理呢?
    發(fā)表于 01-15 10:39 ?400次閱讀
    如何<b class='flag-5'>通過</b>仿真動圖理解各種<b class='flag-5'>傳感器</b>的原理呢?

    stm32和傳感器進行通信,為什么返回的數(shù)據基本是0呢?

    本人使用的是stm32和傳感器進行通信,再設置好SPI的基礎設置后,去讀取x軸的陀螺儀角速度,為什么返回的數(shù)據基本是0呢?
    發(fā)表于 01-02 06:10

    傳感器現(xiàn)場總線及無線傳輸原理

    傳感器現(xiàn)場總線和無線傳輸技術是現(xiàn)代傳感器應用中常用的兩種數(shù)據傳輸方式。傳感器現(xiàn)場總線是一種通過串行通信
    的頭像 發(fā)表于 12-19 17:50 ?802次閱讀

    傳感器數(shù)據融合算法python代碼

    算法的原理和應用,并給出一些使用Python編程語言實現(xiàn)傳感器數(shù)據融合算法的示例代碼。 首先,讓我們來了解傳感器數(shù)據融合算法的原理。傳感器
    的頭像 發(fā)表于 12-15 10:28 ?1173次閱讀

    無線傳感器如何聯(lián)網?無線傳感器如何通信

    無線傳感器如何聯(lián)網?無線傳感器如何通信? 無線傳感器的聯(lián)網和通信是指利用無線技術將傳感器節(jié)點連接
    的頭像 發(fā)表于 12-15 10:10 ?1866次閱讀

    什么是無線物聯(lián)網傳感器,為什么無線物聯(lián)網傳感器有用呢?

    的采集、傳輸、存儲和分析的設備。無線物聯(lián)網傳感器可以通過感知環(huán)境中的各種物理量,如溫度、濕度、光照強度、壓力、加速度等,將這些數(shù)據通過無線通信技術傳輸?shù)皆破脚_或者應用程序中進行分析和利
    的頭像 發(fā)表于 12-15 10:10 ?755次閱讀

    HarmonyOS傳感器開發(fā)指南

    Framework:主要實現(xiàn)傳感器的訂閱管理,數(shù)據通道的創(chuàng)建、銷毀、訂閱與取消訂閱,實現(xiàn)與 SensorService 的通信。 ● Sensor Service:主要實現(xiàn) HD_IDL 層
    發(fā)表于 11-28 15:36

    傳感器網絡通信基礎知識

    電子發(fā)燒友網站提供《傳感器網絡通信基礎知識.pdf》資料免費下載
    發(fā)表于 11-15 11:50 ?0次下載
    <b class='flag-5'>傳感器</b>網絡<b class='flag-5'>通信</b>基礎知識

    傳感器中的數(shù)據怎么傳出來的?

    例如,我打算通過一個壓力傳感器(ot500型)得到數(shù)據,用什么方式接受這個數(shù)據?越詳細越好,謝謝。?
    發(fā)表于 10-26 07:27