【香橙派系列教程】(四)基于ARM-Linux架构的语音控制刷抖音项目
【四】基于ARM-Linux架构的语音控制刷抖音项目
文章目录
- 【四】基于ARM-Linux架构的语音控制刷抖音项目
- 1.语音模块配置
- 1.创建产品
- 2.引脚配置
- 3.词条定义
- 4.添加控制
- 5.发布版本
- 6.烧录固件
- 2.编程实现语音和开发板通信
- 3.手机接入Linux热拔插
- 1.dmesg命令
- 2.adb调试
- 踩坑问题
- 3.总结
- 4. 用shell指令来操作手机屏幕,模拟手动滑屏幕
- 5.整合
- 接线图
- 代码
1.语音模块配置
整体逻辑是:
- 语音模块负责接收用户说话的信息,通过串口发送给H616
- H616写好业务逻辑的处理,当接收不同信号时来处理不同的业务
- 通过USB接口连接手机,利用adb指令来控制安卓手机
安卓的底层是Linux,所以也可以接收一些Linux指令
智能公元:智能公元/AI产品零代码平台 (smartpi.cn)
进入官方平台,配置词条和识别后的串口输出指令。
1.创建产品
这里用到的语音模块型号为SU-03T
2.引脚配置
3.词条定义
4.添加控制
5.发布版本
其他的选项可以自行探索
6.烧录固件
首先连接好串口(记得交叉接线),插到电脑上面下载固件
下载需要冷启动:先把模块断电,然后点下载,下载进度变黄,之后再给模块上电
TX | GPIO_B7 |
---|---|
RX | GPIO_B6 |
1.下载SDK固件包(里面自带烧录工具)
2.找到烧录工具
烧录器
固件位置与烧录器一个位置
开始烧录
烧录完固件之后,可以插到电脑上面配合串口助手看一下发送的信息
2.编程实现语音和开发板通信
首先将语音与开发板通过串口进行连接。
编译我们上一篇写的串口代码:
多文件编译方法: “ * ”为通配符,统一编译所有以uartT开头的文件
运行:
因为输出是乱码,所以需要我们修改一下发送的函数。
uartTool.c
//加入这段话
char myserialGetchar (const int fd)
{char x ;if (read (fd, &x, 1) != 1)return -1 ;return x ;
}
uartTool.h
char myserialGetchar (const int fd);
uartTest.c
添加串口读取一个字符的接口
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <pthread.h>
#include <string.h>
#include <stdio.h>
#include "uartTool.h"
int fd;
void* readSerial()
{char cmd;while(1){cmd = myserialGetchar(fd);switch(cmd){case 'N':printf("next\n");break;case 'P':printf("pre\n");break;case 'Z':printf("zan\n");break;case 'Q':printf("qu\n");break;}}
}
int main(int argc, char **argv)
{char deviceName[32] = {'\0'};pthread_t readt;if(argc < 2){printf("uage:%s /dev/ttyS?\n",argv[0]);return -1;}strcpy(deviceName, argv[1]);if( (fd = myserialOpen(deviceName, 115200)) == -1){printf("open %s error\n",deviceName);return -1;}pthread_create(&readt, NULL, readSerial,NULL);while(1){sleep(10);}
}
3.手机接入Linux热拔插
1.dmesg命令
Linux dmesg(英文全称:display message):命令用于显示开机信息。
用法:dmesg
说明:查看所有开机日志信息
- kernel 会将开机信息存储在 ring buffer 中。您若是开机时来不及查看信息,可利用 dmesg 来查看。
- 开机信息亦保存在 /var/log 目录中,名称为 dmesg 的文件里。
1、查看命令版本
(base) [root@s186 ~]# dmesg -V
dmesg,来自 util-linux 2.23.2
2.过滤想查看信息
建议使用-i参数过滤时忽略大小写
(base) [root@s186 ~]# dmesg |grep -i cpu
2.adb调试
第一步:连接
用Type-c连接手机和orangepi zero2
输入指令查看usb设备:dmesg
此时能看到usb设备号和名称,这时只是单纯的识别了这个usb设备,但没有什么办法调试这个手机。
第二步:调试
如何调试?使用adb指令
//查看adb是否安装
adb version
//默认是没有安装adb的,安装指令:
sudo apt-get -y install adb//调试
adb devices
//进入shell指令
adb shell
踩坑问题
**第一个坑:**OK了,还有一个点我们需要注意的是要打开手机的开发者选项 -> USB调试功能,不然就会出现以下情况
即使我们已经连接成功了,dmesg也能监测到,但是还是看不到任何信息。
我们打开手机USB调试功能之后,再次运行:
手机会弹出来usb调试确认界面,需要手动点击确认(有时候弹不出来,拔插多试试)
到此adb 和Android连接成功,此时adb可以调试Android,通过shell指令控制手机。
**第二个坑:**当我输入adb shell时,也会报错
这是我出现的一种情况1:
手机:选择USB用于传输文件,可以进入adb调试,但是到达shell指令这块会出问题
这个的解决办法在第三个坑介绍
还有另外一种情况2:
手机:选择USB用于仅充电,直接卡在访问设备这里,进入不了adb调试
没有访问权限,也就是没有生成文件柄,应用层无法操作接入的硬件设备:
此时,需要配置文件,以支持USB设备的热拔插,支持UDEV的机制
linux下一切皆文件:
udev机制:说白了就是为插入或者拔掉的设备,添加文件柄删除文件柄。
- 在/etc/udev/rules.d文件夹下创建规则文件
cd /etc/udev/rules.d/
sudo vim oppo-android.rules
- 在文件中添加内容
//1.USB子系统 2.环境变量:USB设备 3.权限
SUBSYSTEM=="usb", ENV{DEVTYPE}=="usb_device", MODE="0666"
**第三个坑:**当我使用adb指令的时候,始终报错这个
这是因为adb工具是安装在orangepi上面的,进入手机的shell指令肯定是找不到的,我们需要退回orangepi的命令行,就可以正常使用adb shell指令了。
3.总结
a. 把手机接入开发板
b. 安装adb工具,在终端输入adb安装指令: sudo apt-get install adb
c. dmeg能查看到手机接入的信息,但是输入adb devices会出现提醒dinsufficient permissions for device: user in plugdev group; are your udevrules wrong?
d. 配置文件,以支持USB设备的热拔插,支持UDEV的机制在/etc/udev/rules.d 文件夹下创建规则文件cd /etc/udev/rules.d/sudo vim 51-android.rules在文件中添加内容 SUBSYSTEM=="usb", ENV{DEVTYPE}=="usb_device", MODE="0666"
e. 在手机开发者选项中,打开USB调试,重新拔插手机
f. 手机弹出调试提醒,点确认手机调试模式
4. 用shell指令来操作手机屏幕,模拟手动滑屏幕
目前我们语音控制刷抖音,只需要这四个指令即可。
adb shell input swipe 540 1300 540 500 100 //向下滑动 540是水平的,1300是竖直方向,下是500,100是100毫秒内完成
adb shell input swipe 540 500 540 1300 100 //向上滑动
adb shell "seq 2 | while read i;do input tap 350 1050 & input tap 350 1050 & sleep 0.2;done;" //点赞
adb shell input keyevent 26 //锁屏
这里多介绍一些常用的shell指令(吃不饱的同学可以加餐哈!哈哈哈哈)
1.adb相关命令
-
关闭adb服务:
adb kill-server
-
开启adb服务:
adb start-server
-
查看当前连接的手机设备:
adb devices
-
多设设备操作,< -s 虚拟设备名称 >
-
重启设备:
adb reboot
指定虚拟设备:adb -s 设备名称 reboot
-
查看日志:
adb logcat
清除日志:adb logcat -c
-
进入linux shell下 adb shell 其中常用的linux命令 cd cat 等等 输入su可以获取超级管理员名称了 要确定是否有哪些命令 进入 system/bin目录 就知道了
-
代理链接:
adb connect [ip:post]
-
获取MAC地址:
adb shell cat /sys/class/net/wlan0/address
-
安装应用:
adb install <name.apk>
-
卸载安装包:
adb uninstall <name.apk>
-
保存缓存文件重新安装:
adb install -r <name.apk>
-
断开连接:
adb disconnect <设备名>
-
文件复制到移动设备 ,后面跟设备目录:
adb push filename /sdcard/
-
模拟位置点击(x,y):
adb shell input tap
-
模拟滑动( x,yx1,y1 ),还可传入滑动缓冲:
adb shell input swipe
-
获取当前页面的UI层 次,保存为xml文件:
uiautomator dump dump: creates an XML dump of current UI hierarchy
18.模式实现输入:adb shell input text <输入任意内容>
2.adb模拟按键对照表
keycode | 含义 |
---|---|
3 | HOME 键 |
4 | 返回键 |
5 | 打开拨号应用 |
6 | 挂断电话 |
24 | 增加音量 |
25 | 降低音量 |
26 | 电源键 |
27 | 拍照(需要在相机应用里) |
64 | 打开浏览器 |
82 | 菜单键 |
85 | 播放/暂停 |
86 | 停止播放 |
87 | 播放下一首 |
88 | 播放上一首 |
122 | 移动光标到行首或列表顶部 |
123 | 移动光标到行末或列表底部 |
126 | 恢复播放 |
127 | 暂停播放 |
164 | 静音 |
176 | 打开系统设置 |
187 | 切换应用 |
207 | 打开联系人 |
208 | 打开日历 |
209 | 打开音乐 |
210 | 打开计算器 |
220 | 降低屏幕亮度 |
221 | 提高屏幕亮度 |
223 | 系统休眠 |
224 | 点亮屏幕 |
231 | 打开语音助手 |
276 | 如果没有 wakelock 则让系统休眠 |
调用实例说明:
命令: adb shell input keyevent <keycode>
- 示例,点击电源键:
adb shell input keyevent 26
#执行效果相当于按电源键。 - 示例,输入"你好啊中国":
adb shell input text "你好啊中国"
#执行效果相当于用户输入"你好啊中国"。 - 示例,模拟点击任何位置:
adb shell input tap x y
(x和y表示点击的坐标)
5.整合
接线图
语音模块
TX | GPIO_B7 |
---|---|
RX | GPIO_B6 |
B6->8 B7->10
手机USB接口插到开发板上面。
代码
对于代码中system函数的使用,在进程一篇中有过讲解:进程
uartTest.c
#include <stdio.h>
#include <stdlib.h>
#include <stdint.h>
#include <stdarg.h>
#include <string.h>
#include <termios.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <pthread.h>
#include "uartTool.h"
int fd;
void* readSerial()
{char cmd;while(1){cmd = myserialGetchar(fd);switch(cmd){case 'N':printf("next\n");system("adb shell input swipe 540 1300 540 500 100");break;case 'P':printf("pre\n");system("adb shell input swipe 540 500 540 1300 100");break;case 'Z':printf("zan\n");system("adb shell \"seq 2 | while read i;do input tap 350 1050 &input tap 350 1050 & sleep 0.2;done;\"");break;case 'Q':printf("qu\n");system("adb shell input keyevent 26");break;}}
}
int main(int argc, char **argv)
{char deviceName[32] = {'\0'};pthread_t readt;if(argc < 2){printf("uage:%s /dev/ttyS?\n",argv[0]);return -1;}strcpy(deviceName, argv[1]);if( (fd = myserialOpen(deviceName, 115200)) == -1){printf("open %s error\n",deviceName);return -1;}pthread_create(&readt, NULL, readSerial,NULL);while(1){sleep(10);}
}
uartTool.c
//uartTool.c
#include <stdio.h>
#include <stdlib.h>
#include <stdint.h>
#include <stdarg.h>
#include <string.h>
#include <termios.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include "wiringSerial.h"
int myserialOpen (const char *device, const int baud)
{struct termios options ;speed_t myBaud ;int status, fd ;switch (baud){case 9600: myBaud = B9600 ; break ;case 115200: myBaud = B115200 ; break ;}if ((fd = open (device, O_RDWR | O_NOCTTY | O_NDELAY | O_NONBLOCK)) == -1)return -1 ;fcntl (fd, F_SETFL, O_RDWR) ;// Get and modify current options:tcgetattr (fd, &options) ;cfmakeraw (&options) ;cfsetispeed (&options, myBaud) ;cfsetospeed (&options, myBaud) ;options.c_cflag |= (CLOCAL | CREAD) ;options.c_cflag &= ~PARENB ;options.c_cflag &= ~CSTOPB ;options.c_cflag &= ~CSIZE ;options.c_cflag |= CS8 ;options.c_lflag &= ~(ICANON | ECHO | ECHOE | ISIG) ;options.c_oflag &= ~OPOST ;options.c_cc [VMIN] = 0 ;options.c_cc [VTIME] = 100 ; // Ten seconds (100 deciseconds)tcsetattr (fd, TCSANOW, &options) ;ioctl (fd, TIOCMGET, &status);status |= TIOCM_DTR ;status |= TIOCM_RTS ;ioctl (fd, TIOCMSET, &status);usleep (10000) ; // 10mSreturn fd ;
}
void serialSendstring (const int fd, const char *s)
{int ret;ret = write (fd, s, strlen (s));if (ret < 0)printf("Serial Puts Error\n");
}
int serialGetstring (const int fd, char *buffer)
{int n_read;n_read = read(fd, buffer,32);return n_read;
}
//加入这段话
char myserialGetchar (const int fd)
{char x ;if (read (fd, &x, 1) != 1)return -1 ;return x ;
}
uartTool.h
//**uartTool.h**
int myserialOpen (const char *device, const int baud);
void serialSendstring (const int fd, const char *s);
int serialGetstring (const int fd, char *buffer);
char myserialGetchar (const int fd);
编译:gcc uartTest.c uartTool.c -pthread
运行:./a.out /dev/ttyS5
相关文章:

【香橙派系列教程】(四)基于ARM-Linux架构的语音控制刷抖音项目
【四】基于ARM-Linux架构的语音控制刷抖音项目 文章目录 【四】基于ARM-Linux架构的语音控制刷抖音项目1.语音模块配置1.创建产品2.引脚配置3.词条定义4.添加控制5.发布版本6.烧录固件 2.编程实现语音和开发板通信3.手机接入Linux热拔插1.dmesg命令2.adb调试踩坑问题 3.总结 4.…...

Java----反射
什么是反射? 反射就是允许对成员变量、成员方法和构造方法的信息进行编程访问。换句话来讲,就是通过反射,我们可以在不需要创建其对象的情况下就可以获取其定义的各种属性值以及方法。常见的应用就是IDEA中的提示功能,当我…...

相似度计算方法
一、相似度计算方法 相似度算法是计算两个或多个对象之间相似程度的方法,这些对象可以是文本、图像、音频等不同类型的数据。在计算机科学、信息检索、推荐系统、数据挖掘等领域中,相似度算法具有广泛的应用。 二、应用场景 搜索引擎:用于文…...
Vue 点击markdown页内链接,路由设置不跳转
在路由index.js里添加路由守卫: router.beforeEach((to,from,next)>{//如果是md页内链接“#xxx”,则不跳转const hash window.location.hash;if(hash.startsWith(#)) {next(false);}else{...其他控制代码next();} });当markdown用[标题链接](#标题名…...

IOday4
一、思维导图 二、练习 1、使用父子进程完成两个文件的拷贝,父进程拷贝前一半内容,子进程拷贝后一半内容,子进程结束后退出,父进程回收子进程的资源 #include<myhead.h> int main(int argc, const char *argv[]) {//判断终…...

智能座舱背后主流车机平台(SA8155/SA8295)的高通Hexagon DSP是什么?
智能座舱背后主流车机平台(SA8155/SA8295)的高通Hexagon DSP是什么? 一、高通Hexagon DSP的辉煌发展历程 高通,作为全球领先的无线通信技术创新者,其处理器技术一直走在行业前列。随着智能手机和物联网设备的普及,对处理器性能的…...

linux进程控制——进程等待——wait、waitpid
前言:本节内容仍然是进程的控制,上一节博主讲解的是进程控制里面的进程创建、进程退出、终止。本节内容将讲到进程的等待——等待是为了能够将子进程的资源回收,是父进程等待子进程。 我们前面的章节也提到过等待, 那里的等待是进…...

Shell脚本的进程管理
进程管理是系统管理的重要方面,通过对进程的监控、启动、停止和重启,可以保证系统的稳定运行。Shell脚本是一种强大的工具,可以对进程进行自动化管理,提高效率和准确性。 参考:shell脚本进程管理 - CSDN文库 shell脚本…...

JLink烧录失败
1. 现象: 这个位置是灰色的,没有SW Device信息。 MDK下面的打印: J-Flash的打印: windows上面的弹框的现象没有截屏。 2. 解决办法: 1.打开J-Link Commander,输入unlock kinetis,看现象不起作用,网…...

Monorepo简介
Monorepo 第一章:与Monorepo的邂逅第二章:Multirepo的困境第三章:Monorepo的魔力 - 不可思议的解决问题能力第四章:Monorepo的挑战与应对策略第五章:总结第六章:参考 第一章:与Monorepo的邂逅 …...
SpringBoot打包为jar包,打包前注意事项及打包教程
在打包 Spring Boot 项目为 JAR 包之前,有一些重要的注意事项和步骤,以确保打包过程顺利并生成一个可正常运行的 JAR 包: 1. 检查依赖和版本 确保所有依赖项和插件版本是最新且兼容的,特别是 Spring Boot 版本和其相关依赖的版本…...

B端系统UI个性化设计:感受定制之美
B端系统UI个性化设计:感受定制之美 引言 艾斯视觉作为ui设计和前端开发从业者,其观点始终认为:在当今竞争激烈的商业环境中,B端(Business-to-Business)系统的设计不再仅仅是功能性的堆砌,而是…...
前端常用 utils 工具封装
// 函数防抖 export function debounce(fn, interval) {let timerreturn function (this, ...args) {clearTimeout(timer)const context thislet params [...args]timer setTimeout(() > {fn.call(context, ...params)}, interval || 1000)} }// 函数节流 export functio…...

项目都做完了,领导要求国际化????--JAVA后端篇
springboot项目国际化相信各位小伙伴都会,很简单,但是怎么项目都做完了,领导却要求国际化文件就很头疼了 国际化的SpringBoot代码: 第一步:创建工具类 /*** 获取i18n资源文件** author bims*/ public class Message…...

国内备受好评PostgreSQL数据库性能如何?
为什么国内很多数据库采用PostgreSQL数据库作为基础,再次开发自己的产品呢?不仅仅是因为PostgreSQL数据库开源免费、PostgreSQL 数据库的性能也是相当出色的,具有以下几个方面的特点: 1. 处理大规模数据: - 能够有效地管理和处…...

彻底搞懂前端跨域解决方案
一、浏览器的同源策略 1、同源策略概述 同源策略是浏览器为确保资料安全,而遵循的一种策略,该策略对访问资源进行了一些限制。 2、什么是源(origin)? 3、示例 4、同源请求 5、非同源请求 二、跨域会受到哪些限制 1…...

Kafka基础概念
MQ消息中间件 1)总览: 消息中间件 这里我们主要学习的是kafka的基础概念 具体参考黑马头条:https://www.bilibili.com/video/BV1Qs4y1v7x4/?spm_id_from333.337.search-card.all.click 2)消息中间件对比 3)Kafka介…...

【论文阅读笔记】DeepCAD: A Deep Generative Network for Computer-Aided Design Models
1 引言 现有3D生成模型: 3D点云:大量离散的3D点组成的数据表示形式; 多边形网格:一系列相连的多边形组成的3D模型; 水平集场:使用数值函数来表示物体的边界,并根据函数值的正负来确定物体内部…...

《如鸢》开通官号,女性向游戏爆款预定
今天,备受瞩目的沉浸式剧情卡牌手游《如鸢》正式开通了官方社媒账号并发布了玩家信。 《如鸢》由灵犀互娱倾力打造,游戏不仅拥有跌宕起伏的权谋剧情,更采用Live2D技术,为玩家带来沉浸式的游戏体验,吸引了众多玩家关注。…...
OpenAI再下一城:发布Voice Engine,可使用文本和参考语音合成说话者的新语音!
转自 机器学习算法工程师 OpenAI又发布了一个最新的工作:Voice Engine。Voice Engine可以使用文本输入和单个 15 秒音频样本生成听起来自然且与原始说话者非常相似的语音。而且,一个小型模型仅通过一个 15 秒的样本就能创造出富有情感且逼真的语音。Voi…...

MPNet:旋转机械轻量化故障诊断模型详解python代码复现
目录 一、问题背景与挑战 二、MPNet核心架构 2.1 多分支特征融合模块(MBFM) 2.2 残差注意力金字塔模块(RAPM) 2.2.1 空间金字塔注意力(SPA) 2.2.2 金字塔残差块(PRBlock) 2.3 分类器设计 三、关键技术突破 3.1 多尺度特征融合 3.2 轻量化设计策略 3.3 抗噪声…...
生成xcframework
打包 XCFramework 的方法 XCFramework 是苹果推出的一种多平台二进制分发格式,可以包含多个架构和平台的代码。打包 XCFramework 通常用于分发库或框架。 使用 Xcode 命令行工具打包 通过 xcodebuild 命令可以打包 XCFramework。确保项目已经配置好需要支持的平台…...
Ubuntu系统下交叉编译openssl
一、参考资料 OpenSSL&&libcurl库的交叉编译 - hesetone - 博客园 二、准备工作 1. 编译环境 宿主机:Ubuntu 20.04.6 LTSHost:ARM32位交叉编译器:arm-linux-gnueabihf-gcc-11.1.0 2. 设置交叉编译工具链 在交叉编译之前&#x…...

使用分级同态加密防御梯度泄漏
抽象 联邦学习 (FL) 支持跨分布式客户端进行协作模型训练,而无需共享原始数据,这使其成为在互联和自动驾驶汽车 (CAV) 等领域保护隐私的机器学习的一种很有前途的方法。然而,最近的研究表明&…...
基于Uniapp开发HarmonyOS 5.0旅游应用技术实践
一、技术选型背景 1.跨平台优势 Uniapp采用Vue.js框架,支持"一次开发,多端部署",可同步生成HarmonyOS、iOS、Android等多平台应用。 2.鸿蒙特性融合 HarmonyOS 5.0的分布式能力与原子化服务,为旅游应用带来…...

ETLCloud可能遇到的问题有哪些?常见坑位解析
数据集成平台ETLCloud,主要用于支持数据的抽取(Extract)、转换(Transform)和加载(Load)过程。提供了一个简洁直观的界面,以便用户可以在不同的数据源之间轻松地进行数据迁移和转换。…...

【配置 YOLOX 用于按目录分类的图片数据集】
现在的图标点选越来越多,如何一步解决,采用 YOLOX 目标检测模式则可以轻松解决 要在 YOLOX 中使用按目录分类的图片数据集(每个目录代表一个类别,目录下是该类别的所有图片),你需要进行以下配置步骤&#x…...
拉力测试cuda pytorch 把 4070显卡拉满
import torch import timedef stress_test_gpu(matrix_size16384, duration300):"""对GPU进行压力测试,通过持续的矩阵乘法来最大化GPU利用率参数:matrix_size: 矩阵维度大小,增大可提高计算复杂度duration: 测试持续时间(秒&…...
Android第十三次面试总结(四大 组件基础)
Activity生命周期和四大启动模式详解 一、Activity 生命周期 Activity 的生命周期由一系列回调方法组成,用于管理其创建、可见性、焦点和销毁过程。以下是核心方法及其调用时机: onCreate() 调用时机:Activity 首次创建时调用。…...

使用Spring AI和MCP协议构建图片搜索服务
目录 使用Spring AI和MCP协议构建图片搜索服务 引言 技术栈概览 项目架构设计 架构图 服务端开发 1. 创建Spring Boot项目 2. 实现图片搜索工具 3. 配置传输模式 Stdio模式(本地调用) SSE模式(远程调用) 4. 注册工具提…...