基于stm32mp157的嵌入式linux+qt项目实战物联网毕业设计选题之智慧医疗项目

news2024/11/24 17:00:28

stm32mp157开发板FS-MP1A是华清远见自主研发的一款高品质、高性价比的Linux+单片机二合一的嵌入式教学级开发板。开发板搭载ST的STM32MP157高性能微处理器,集成2个Cortex-A7核和1个Cortex-M4 核,A7核上可以跑Linux操作系统,M4核上可以跑FreeRTOS、RT-Thread等实时操作系统。开发板搭配仿真器、显示屏、摄像头、资源扩展板等丰富的扩展模块,可拓展物联网、人工智能等相关技术学习,还可以拓展丰富的项目实战,非常贴合企业当下开发需求,是一款嵌入式Linux入门进阶必备开发板!

可学习技术:嵌入式Linux应用/系统/驱动开发、ARM裸机开发、Qt界面编程、STM32单片机、FreeRTOS、人工智能机器视觉等。其中ARM Cortex-A7裸机开发课程是华清远见独有特色课程,可关注:https://www.bilibili.com/video/BV1Xe4y1i7vm/,持续更新中。

可实战项目:14个Linux+Qt综合项目案例,6个MP1A物联网拓展项目

关注公众号“华清远见在线实验室”,回复“mp157项目”,即可领取。

Linux+Qt综合项目案例:华清远见stm32mp157开发板优势特色部分,包括音乐播放器、智慧家庭、智能工业电表、智能出行助手、智能猫眼、环境监测、智能安防、智能语音识别等10余个项目案例,涉及家居、医疗、农业多种应用方向,在案例中使用了多种物联网和嵌入式技术,包括OT开发、linux应用开发、linux驱动开发、物联网云端接入、MQTT协议、json字符串等知识点。

基于Linux+Qt的智慧医疗项目

项目简介:

使用心率/脉搏传感器采集数据,通过算法计算得到心率和血氧。心率并不像环境光及接近传感器那样数据比较简单,并不能通过读取一个文件获得当前的心率或血氧,而是通过一系列的数据计算出来的,这样 sys 文件系统的接口就不能完全适用了,所以需要结合系统调用以字符设备的方式获取对应数据,计算出心率和血氧。

关键文件说明:

/sys/bus/iio/devices/iio:device2/scan_elements/(in_intensity_ir_en/in_intensity_red_en): 通道使能,通过写 1 或 0 使能或关闭通道。

/sys/bus/iio/devices/iio:device2/scan_elements/(in_intensity_ir_type/in_intensity_red_type): 通道数据类型。读取格式为:be:u18/32>>8,be 表示数据为大端存储(bigendian)、u 表示 数据为无符号数,18/32>>8 表示 32 位数据左移 8 位后有效数据为 18 位。

/sys/bus/iio/devices/iio:device2/scan_elements/(in_intensity_ir_index/in_intensity_red_index): 读取数据为当前通道的序号,red 为 0 通道,ir 为 1 通道

/sys/bus/iio/devices/iio:device2/buffer/length: 写入数据用来设置缓存区样本数量。

/sys/bus/iio/devices/iio:device2/buffer/enable: 写入 1 或 0 用来使能设备采样或停止设备采样。

开发平台:

华清远见stm32mp157开发板豪华套餐(开发板+仿真器+五寸屏+摄像头+资源扩展板+tf卡+读卡器)

源码分析

数据采集线程

threadgetheart.h

该文件主要对数据采集用到的函数进行声明。

#ifndef THREADGETHEART_H

#define THREADGETHEART_H

#include <QThread>

#define true1

#define false0

#define FS100

#define BUFFER_SIZE(FS*5)//#define BUFFER_SIZE (FS* 2)

#define HR_FIFO_SIZE7

#define MA4_SIZE4// DO NOT CHANGE

#define HAMMING_SIZE5// DO NOT CHANGE

#define min(x,y)((x)<(y)?(x):(y))QT_BEGIN_NAMESPACEconst unsigned short auw_hamm[31]={41,276,512,276,41};const unsigned char uch_spo2_table[184]={95,95,95,96,96,96,97,97,97,97,97,98,98,98,98,98,99,99,99,99,99,99,99,99,100,100,100,100,100,100,100,100,100,100,100,100,100,100,100,100,100,100,100,100,99,99,99,99,99,99,99,99,98,98,98,98,98,98,97,97,97,97,96,96,96,96,95,95,95,94,94,94,93,93,93,92,92,92,91,91,90,90,89,89,89,88,88,87,87,86,86,85,85,84,84,83,82,82,81,81,80,80,79,78,78,77,76,76,75,74,74,73,72,72,71,70,69,69,68,67,66,66,65,64,63,62,62,61,60,59,58,57,56,56,55,54,53,52,51,50,49,48,47,46,45,44,43,42,41,40,39,38,37,36,35,34,33,31,30,29,28,27,26,25,23,22,21,20,19,17,16,15,14,12,11,10,9,7,6,5,3,2,1};static int an_dx[BUFFER_SIZE-MA4_SIZE];static int an_x[BUFFER_SIZE];//irstatic int an_y[BUFFER_SIZE];//redclassThreadgetheart:public QThread

{Q_OBJECTpublic:Threadgetheart();

 virtual voidrun();private:

 int write_sys_int(char *filename, int data);

 int enable_disable_all_channels(int enable);

 int enable_disable_buffer(int enable);

 int set_buffer_len(int len);voidmaxim_heart_rate_and_oxygen_saturation(unsigned int *pun_ir_buffer,

int n_ir_buffer_length, unsigned int *pun_red_buffer, int *pn_spo2, int

*pch_spo2_valid,

 int *pn_heart_rate, int *pch_hr_valid);voidmaxim_find_peaks( int *pn_locs, int *pn_npks, int *pn_x, int n_size, int

n_min_height, int n_min_distance, int n_max_num );voidmaxim_peaks_above_min_height( int *pn_locs, int *pn_npks, int *pn_x,

int n_size, int n_min_height );voidmaxim_remove_close_peaks( int *pn_locs, int *pn_npks, int *pn_x, int

n_min_distance );voidmaxim_sort_ascend( int *pn_x, int n_size );voidmaxim_sort_indices_descend( int *pn_x, int *pn_indx, int n_size);voidget_heart_rate();

signals:voidinforsignal(int SpO2,int heart_rate);};

#endif // THREADGETHEART_H

threadgethreat.cpp

这里只把线程的 run 函数展示出来,run 函数里调用的一些其他成员函数因为代码量太大就不展示了,可以参考源码。这些成员函数都是对采集的数据进行处理,转换成心率和血氧

void Threadgetheart::run(){get_heart_rate();}void Threadgetheart::get_heart_rate(){

 int ret =0;

 int data[2];

 char device_path[128]={0};

 int fd =0;

 int i, j;

int read_size;

 int SpO2;

 int heart_rate;

 int hr_valid;

 int spo2_valid;

 int tmp_val =0;sprintf(device_name,"%s","iio:device2");enable_disable_all_channels(1);set_buffer_len(4);enable_disable_buffer(1);sprintf(device_path,"/dev/%s", device_name);

 fd =open(device_path,O_RDONLY|O_NONBLOCK);if(fd <0){/* TODO: If it isn't there make the node */

 goto error;}for(j =0; j <DATA_BUF_SIZE; j++){

 struct pollfd pfd ={.fd = fd,.events =POLLIN,};

 ret =poll(&pfd,1,-1);if(ret <0){

 ret =-errno;

 goto error;}elseif(ret ==0){printf("continue\n");continue;}

 read_size =read(fd, data,8);if(read_size <0){if(errno ==EAGAIN){qDebug()<<"nothing available\n";continue;}else{break;}}

 tmp_val =be32toh(data[0]);

 aun_red_buf[j]=(tmp_val >>DATA_SHIFT)&DATA_MASK;

 tmp_val =be32toh(data[1]);

 aun_ir_buf[j]=(tmp_val >>DATA_SHIFT)&DATA_MASK;}maxim_heart_rate_and_oxygen_saturation(aun_ir_buf,DATA_BUF_SIZE,

aun_red_buf,&SpO2,&spo2_valid,&heart_rate,&hr_valid);while(1){for(i =100; i <DATA_BUF_SIZE; i++){

 aun_red_buf[i -100]= aun_red_buf[i];

 aun_ir_buf[i -100]= aun_ir_buf[i];}

 int count =0;for(j =DATA_BUF_SIZE-100; j <DATA_BUF_SIZE; j++){

 struct pollfd pfd ={.fd = fd,.events =POLLIN,};

 ret =poll(&pfd,1,-1);if(ret <0){

 ret =-errno;

 goto error;}elseif(ret ==0){qDebug("continue\n");continue;}

 read_size =read(fd, data,8);if(read_size <0){if(errno ==EAGAIN){qDebug("nothing available\n");continue;}else{break;}}

 tmp_val =be32toh(data[0]);

 aun_red_buf[j]=(tmp_val >>DATA_SHIFT)&DATA_MASK;

 tmp_val =be32toh(data[1]);

 aun_ir_buf[j]=(tmp_val >>DATA_SHIFT)&DATA_MASK;

 count++;}maxim_heart_rate_and_oxygen_saturation(aun_ir_buf,DATA_BUF_SIZE,

aun_red_buf,&SpO2,&spo2_valid,&heart_rate,&hr_valid);if(spo2_valid&&hr_valid){qDebug("SpO2: %d\t", SpO2);qDebug("heart rate: %d\n", heart_rate);

 emit inforsignal(SpO2,heart_rate);}}

error:qDebug()<<"error";}

主线程

mainwindow.ui

这是 ui 界面的设计,可以自行改动,用到的图片资源可以参考源码

mainwindow.h

#include <QMainWindow>

#include "threadgetheart.h"

namespace Ui {classMainWindow;}QT_END_NAMESPACEclassMainWindow:public QMainWindow

{Q_OBJECTpublic:MainWindow(QWidget *parent = nullptr);~MainWindow();private:

 Ui::MainWindow *ui;

 Threadgetheart *thread1;voidbeepring();voidbeepunring();private slots:voidpushbuttonSlot();voidsetLable(int SpO2,int heart_rate);voidpushbutton_stopSlot();voidpushbutton_restartSlot();voidringSlot();

signals:voidring();};

#endif // MAINWINDOW_H

mainwindow.cpp

子线程实时并计算心率和血氧,通过信号发送给主线程,主线程将心率和血氧显示到 UI 界面上,并设置超过指定心率蜂鸣器报警。

#include "mainwindow.h"

#include "ui_mainwindow.h"

#include <stdint.h>

#include <string.h>

#include <fcntl.h>

#include <stdio.h>

#include <stdlib.h>

#include <linux/input.h>

#include <unistd.h>

MainWindow::MainWindow(QWidget *parent):QMainWindow(parent),ui(newUi::MainWindow){

 ui->setupUi(this);

 thread1 =newThreadgetheart;connect(ui->pushButton,SIGNAL(clicked()),this,SLOT(pushbuttonSlot()));connect(ui->pushButton_stop,SIGNAL(clicked()),this,SLOT(pushbutton_stopSlot()));connect(thread1,SIGNAL(inforsignal(int,int)),this,SLOT(setLable(int,int)));connect(ui->pushButton_restart,SIGNAL(clicked()),this,SLOT(pushbutton_restartSlo

t()));connect(this,SIGNAL(ring()),this,SLOT(ringSlot()));}

MainWindow::~MainWindow(){delete ui;}void MainWindow::pushbuttonSlot(){

 thread1->start();

 ui->pushButton->setEnabled(false);}void MainWindow::pushbutton_restartSlot(){system("echo 1 > /sys/bus/iio/devices/iio:device2/buffer/enable");}void MainWindow::pushbutton_stopSlot(){system("echo 0 > /sys/bus/iio/devices/iio:device2/buffer/enable");

 ui->label_heart->setText(QString::number(0));

 ui->label_SpO2->setText(QString::number(0));}void MainWindow::setLable(int SpO2,int heart_rate){

 ui->label_heart->setText(QString::number(heart_rate));

 ui->label_SpO2->setText(QString::number(SpO2));if(heart_rate >100){

 ui->label_heart_scope_Normal->setText("异常");// beepring();

 emit ring();if(heart_rate >200)

 ui->label_heart_scope_info->setText("请检查手离传感器的距离,重

新测量");}else{

 ui->label_heart_scope_Normal->setText("正常");

 ui->label_heart_scope_info->clear();beepunring();}}void MainWindow::beepring(){

int fd;

 struct input_event event;

 struct timeval time;

 fd =open("/dev/input/by-path/platform-beeper-event",O_RDWR);

 event.type =EV_SND;

 event.code =SND_TONE;

 event.value =1000;

 time.tv_sec =1;

 time.tv_usec =0;

 event.time = time;write(fd,&event,sizeof(struct input_event));}void MainWindow::beepunring(){

 int fd;

 struct input_event event;

 struct timeval time;

 fd =open("/dev/input/by-path/platform-beeper-event",O_RDWR);

 event.type =EV_SND;

 event.code =SND_TONE;

 event.value =0;

 time.tv_sec =0;

 time.tv_usec =0;

 event.time = time;write(fd,&event,sizeof(struct input_event));}void MainWindow::ringSlot(){beepring();usleep(500);beepunring();}

实验源码

源码路径【5_智慧医疗\实验源码\5_znyl】

注意事项

1.在开发板运行时,需要导入中文字库,否则会因为识别不了中文。

将【5_智慧医疗\工具软件\wqy-zenhei-0.9.47-nightlybuild.tar.gz 或 wqyzenhei-0.8.38-1.tar.gz】复制到 ubuntu 下。并使用 scp 命令将文件拷贝到开发板的 usr/share/fonts 目录下,使用 tar 命令解压后即可。

linux@ubuntu:~$ scp wqy-zenhei-0.8.38-1.tar.gz

root@192.168.10.128:/usr/share/fonts/

2.如果使用 mipi 五寸屏运行此项目,需要进行屏幕旋转以适应屏幕,具体步骤如下:

在/etc/profile.d/qt-eglfs.sh 添加环境变量如下

下面变量的 event0 设备需要填实际的触摸屏设备

这里即填 event0

exportQT_QPA_EGLFS_ROTATION=90exportQT_QPA_EGLFS_NO_LIBINPUT=1exportQT_QPA_EVDEV_TOUCHSCREEN_PARAMETERS=/dev/input/event0:rotate=90

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/346510.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

吃瓜教程 | Datawhale 打卡(Task 01)

第1章 绪论 引言 机器学习致力于研究如何通过计算的手段&#xff0c;利用经验来改善系统自身的性能。 “经验”通常以“数据”的形式存在。 机器学习研究的主要内容&#xff1a; 在计算机上从数据中产生“模型”&#xff08;model&#xff09;的算法&#xff0c;即“学习算法…

软件开发风险 需要规避的4个重点

1、前期减少投资 开发软件&#xff0c;最好减少前期的投入。软件开发成本高低是取决于所需开发的功能&#xff0c;需求越多&#xff0c;需要实现的功能越多&#xff0c;开发成本就越高。在不确定开发软件能够带来预期价值的情况下&#xff0c;建议先开发核心功能&#xff0c;辅…

分享166个HTML医疗保健模板,总有一款适合您

分享166个HTML医疗保健模板&#xff0c;总有一款适合您 166个HTML医疗保健模板下载链接&#xff1a;https://pan.baidu.com/s/1tBFEInec5Jnw_ShQd21MJg?pwdakif 提取码&#xff1a;akif Python采集代码下载链接&#xff1a;采集代码.zip - 蓝奏云 import os import shuti…

【残差稠密网络:医学图像:超分】

Residual dense network for medical magnetic resonance images super-resolution &#xff08;医学磁共振图像超分辨率的残差稠密网络&#xff09; 高分辨率磁共振成像&#xff08;MRI&#xff09;有助于专家定位病灶和诊断疾病&#xff0c;但高分辨率MRI难以获得。此外&am…

PyQt5数据库开发1 4.2 配置SQL Server 2008 数据源(ODBC编程)

文章目录 配置SQL Server 2008 数据源&#xff08;ODBC编程&#xff09; 1. 了解要配置的数据源服务器名称&#xff0c;以及数据库和对应表 2. 打开控制面板&#xff0c;点击管理工具 3. 双击数据源 4. 选择“用户DSN”选项卡&#xff0c;点击“添加” 5. 选择SQL Serv…

晚上下班之后可以做什么副业,业余时间需要利用起来

对大多数普通人来说&#xff0c;他们晚上有很多空闲时间&#xff0c;但他们总是在手机上玩游戏&#xff0c;刷视频&#xff0c;白白度过一夜。事实上&#xff0c;近年来&#xff0c;很多朋友都想利用晚上的时间做一些副业&#xff0c;因为目前的工资已经不能满足自己的需求&…

第05章_MySQL排序与分页

第05章_排序与分页 讲师&#xff1a;尚硅谷-宋红康&#xff08;江湖人称&#xff1a;康师傅&#xff09; 官网&#xff1a;http://www.atguigu.com 1. 排序数据 如果没有使用排序操作&#xff0c;默认情况下查询返回的数据时按照添加数据的顺序显示的 SELECT employee_id, la…

yocto创建自己的machine

前面讲了如何离线构建yocto工程&#xff0c;这节讲如何创建自己的machine&#xff0c;在初始化yocto启动bitbake需要输入如下命令 DISTROfsl-imx-fb MACHINEimx6ull14x14evk source imx-setup-release.sh -b build其中DISTRO用来指定发行版本 MACHINE用来指定硬件平台 build为构…

HTML复习1

VSCode 工具生成骨架标签新增代码 < !DOCTYPE html>表示的什么意思&#xff1f; 这句代码的意思是&#xff1a;当前页面采取的是HTML5版本来显示网页. 注意&#xff1a; < !DOCTYPE> 声明位于文档中的最前面的位置&#xff0c;处于 < html> 标签之前。< …

SAP S/4HANA Cloud 2302 财务模块亮点

&#xff08;亮点1:&#xff09;含项目制造的按订单设计 (ETO) 使用 SAP S/4HANA Cloud 2302&#xff0c;新的范围项目 6GD首先发布在德国和美国的国家版本下&#xff0c;提供项目制造的按订单设计 (ETO)的功能。 价值体现 借助 ETO 解决方案&#xff0c;您可以&#xff1a;…

【模型部署】TensorRT的安装与使用

文章目录1.TensorRT的安装1.1 cuda/cudnn以及虚拟环境的创建1.2 根据cuda版本安装相对应版本的tensorRT2. TensorRT的使用2.1 直接构建2.2 使用 Python API 构建2.3 使用 C API 构建2.3.1 属性配置2.3.2 验证2.4 IR 转换模型2.4.1 使用 Python API 转换2.4.2 使用 C API 转换2.…

Moonbeam生态说|走近生态项目SubWallet

「Moonbeam生态说」是Moonbeam中文爱好者社区联合Moonbeam中文高级大使组织的社区AMA活动。该活动为已部署Moonriver或Moonbeam的项目方提供了在主流Moonbeam非官方中文社区内介绍自己的项目信息&#xff0c;包括&#xff1a;项目介绍、团队介绍、技术优势等&#xff0c;帮助社…

【Unity VR开发】结合VRTK4.0:创建抽屉

语录&#xff1a; 为有牺牲多壮志&#xff0c;敢叫日月换新天。 前言&#xff1a; 前面我们知道了门的基本实现原理是通过角度驱动器实现的&#xff0c;那么今天我们来实现一下抽屉的实现原理&#xff1a;线性驱动器。 正文&#xff1a; 步骤一&#xff1a; 首先我们需要在新…

零基础学Java要具备哪些前提条件?

很多零基础的学员对于学Java比较迷茫&#xff0c;想通过学Java掌握一技之长&#xff0c;却不知道入门需要具备哪些条件?不知道怎么去学习?下面详细来和大家聊聊&#xff1a;首先&#xff0c;要对Java语言感兴趣&#xff0c;兴趣是最好的老师&#xff0c;只有拥有兴趣才能在学…

浅析云边端协同架构的应用意义与EasyCVR视频融合能力升级

随着5G时代的到来&#xff0c;万物互联产生了海量数据&#xff0c;据IDC预测&#xff0c;到2025年全球设备连接总数将达到1000亿&#xff0c;集中式处理模型下核心网络无法承载如此大的数据量传输&#xff0c;数据也无法在云中心存储计算&#xff0c;因此基于云边端的架构模式成…

ATR指标在外汇交易中的另类运用方法

当涉及到外汇交易时&#xff0c;有许多不同的指标可以使用。然而&#xff0c;ATR指标可能是一个被低估的工具&#xff0c;可以帮助您发现有利可图的交易机会。本文将介绍ATR指标是什么&#xff0c;如何使用它来识别价格波动和制定交易策略&#xff0c;以及如何在外汇市场中另辟…

DRF之实战总结

前言 DRF概念&#xff1a;Django REST framework框架是一个用于构建Web API的强大而又灵活的工具. 通常简称为DRF框架 或 REST framework框架 特点&#xff1a; 提供了定义序列化器serializer的方法,可以快速根据Django ORM或者其他库自动序列化/反序列化;提供了丰富的类视图…

解决rimraf使用时提示unexpected token “.”

解决rimraf使用时提示unexpected token “.” 前言 最近运行一个Cordova项目时&#xff0c;npm install后打包&#xff0c;命令栏提示了下面这个问题&#xff1a; 很奇怪啊&#xff0c;就我这里有问题&#xff0c;别人之前都没事&#xff0c;很头疼。 问题原因 经过一番摸索…

centos安装FastDFS,集成到SpringBoot中

前言 本教程采用centos7 实测 安装fastdfs&#xff0c;每一步都存在截图&#xff0c;安装不成功你就我 最关键的是采用springboot 集成 fastdfs&#xff0c;上传保存文件信息 小序 FastDFS是一个开源的分布式文件系统&#xff0c;她对文件进行管理&#xff0c;功能包括&…

为什么时间序列预测这么难?本文将给你答案

机器学习和深度学习已越来越多应用在时序预测中。ARIMA 或指数平滑等经典预测方法正在被 XGBoost、高斯过程或深度学习等机器学习回归算法所取代。 尽管时序模型越来越复杂&#xff0c;但人们对时序模型的性能表示怀疑。有研究表明&#xff0c;复杂的时序模型并不一定会比时序…