当前位置: 首页 > news >正文

【嵌入式人工智能产品开发实战】(二十二)—— 政安晨:改造小智AI开发智能体硬件(案例:移植PowerManager后麦克风不工作)

政安晨的个人主页:政安晨

欢迎 👍点赞✍评论⭐收藏

希望政安晨的博客能够对您有所裨益,如有不足之处,欢迎在评论区提出指正!

目录

确定你硬件的关键点

案例分析

🔍 一、关键代码分析

✅ 1. power_save_timer_->WakeUp(); 的作用

📌 相关逻辑:

✅ 2. 麦克风是否被关闭?

🧨 二、可能导致麦克风失效的原因

🛠️ 三、调试建议与修复方法

✅ 1. 添加日志确认当前状态

A. WakeUp() 函数内:

B. OnEnterSleepMode / OnExitSleepMode:

C. InitializePowerManager() 的回调:

✅ 2. 强制恢复音频输入

✅ 3. 检查 GetAudioCodec() 是否有效

✅ 4. 手动触发一次唤醒

✅ 5. 检查硬件连接

🧪 四、建议临时修复方案

五、总结


小智AI推出已经很久了,新鲜感过去之后,我们不禁要思考:我们可以利用这套框架干点什么呢?

为自己打个基础吧,先从给自己做块硬件开始。

接下来,我将列出当你拿到一块全新的嵌入式硬件主板后,如何将小智AI在这块板上运行起来。

确定你硬件的关键点

当你要集成小智AI时,为避免踩坑,你最好确认以下关键点:

1. 你的硬件芯片是ESP32的C3还是S3,或者其它型号?

2. 你的硬件是否有外部PSRAM,它的大小是多少?

3. 你的屏幕器件是什么型号?分辨率多少?

4. 你的麦克风是单/双麦克风?

5. 你硬件上的CODEC芯片是什么型号?I2C口多少,地址多少?

6. 你最好能拿到这个硬件的IO口映射表。

案例分析

在自己的硬件配置中增加powermanager功能后,发现唤醒不工作(初步判断为麦克风功能受到影响)

🔍 一、关键代码分析

✅ 1. power_save_timer_->WakeUp(); 的作用

main_button_.OnClick([this]() {ESP_LOGI(TAG, "main_button_ OnClick volume");power_save_timer_->WakeUp();...
});
  • 调用 power_save_timer_->WakeUp() 是用于唤醒设备,退出休眠模式。
  • 如果当前设备处于睡眠状态,这个调用会触发 OnExitSleepMode 回调。
📌 相关逻辑:
void PowerSaveTimer::WakeUp() {ticks_ = 0;if (in_sleep_mode_) {in_sleep_mode_ = false;// 恢复性能配置(如CPU频率)...// 触发退出休眠回调if (on_exit_sleep_mode_) {on_exit_sleep_mode_();}}
}

✅ 2. 麦克风是否被关闭?

在 InitializePowerSaveTimer() 中有如下逻辑:

power_save_timer_->OnEnterSleepMode([this]() {auto display = GetDisplay();display->SetChatMessage("system", "");display->SetEmotion("sleepy");auto codec = GetAudioCodec();codec->EnableInput(false); // 关闭音频输入
});power_save_timer_->OnExitSleepMode([this]() {auto codec = GetAudioCodec();codec->EnableInput(true); // 恢复音频输入auto display = GetDisplay();display->SetChatMessage("system", "");display->SetEmotion("neutral");
});
  • 如果设备进入休眠模式,会执行 codec->EnableInput(false),这会导致麦克风被关闭。
  • 如果没有正确触发 OnExitSleepMode,或者 WakeUp() 没有恢复音频输入,麦克风就无法再次启用。

🧨 二、可能导致麦克风失效的原因

原因分析
1. 设备长时间处于休眠状态如果没有外部唤醒事件(比如按钮按下),设备一直处在休眠状态,EnableInput(false) 就不会被恢复。
2. WakeUp() 未成功触发 OnExitSleepMode可能由于某些条件判断或错误处理跳过了 OnExitSleepMode 的执行,导致麦克风未重新启用。
3. GetAudioCodec() 返回空指针或无效对象如果 GetAudioCodec() 初始化失败或返回了无效对象,后续操作将无效,也无法恢复音频输入。
4. InitializePowerManager() 影响了电源管理如果设备检测到充电状态变化并频繁触发 SetEnabled(true/false),可能会导致休眠/唤醒机制紊乱。
5. 音频驱动问题(硬件级)如果底层音频驱动存在 bug 或初始化失败,也可能导致麦克风始终无法工作。

🛠️ 三、调试建议与修复方法

✅ 1. 添加日志确认当前状态

在以下关键位置添加日志输出,便于定位问题:

A. WakeUp() 函数内:
void PowerSaveTimer::WakeUp() {ESP_LOGI(TAG, "WakeUp() called, in_sleep_mode: %d", in_sleep_mode_);...
}
B. OnEnterSleepMode / OnExitSleepMode
power_save_timer_->OnEnterSleepMode([this]() {ESP_LOGI(TAG, "Entering sleep mode, disabling mic");auto codec = GetAudioCodec();codec->EnableInput(false);
});power_save_timer_->OnExitSleepMode([this]() {ESP_LOGI(TAG, "Exiting sleep mode, enabling mic");auto codec = GetAudioCodec();codec->EnableInput(true);
});
C. InitializePowerManager() 的回调:
power_manager_->OnChargingStatusChanged([this](bool is_charging) {ESP_LOGI(TAG, "OnChargingStatusChanged: %d", is_charging);if (is_charging) {power_save_timer_->SetEnabled(false);} else {power_save_timer_->SetEnabled(true);}
});
✅ 2. 强制恢复音频输入

你可以手动调用一次恢复麦克风的接口:

auto codec = GetAudioCodec();
if (codec != nullptr) {codec->EnableInput(true);ESP_LOGI(TAG, "Forced enable audio input.");
} else {ESP_LOGE(TAG, "Failed to get audio codec.");
}

放在以下任意一个地方进行测试:

  • 在 Application::GetInstance().StartListening() 前;
  • 在 boot_button_.OnClick() 或 main_button_.OnClick() 中;
  • 在 Initialize() 构造函数最后;
✅ 3. 检查 GetAudioCodec() 是否有效

确保 GetAudioCodec() 正确初始化且返回非空指针:

virtual AudioCodec* GetAudioCodec() override {static BoxAudioCodec audio_codec(...);return &audio_codec;
}

可以加日志验证:

auto codec = GetAudioCodec();
ESP_LOGI(TAG, "GetAudioCodec address: %p", codec);
✅ 4. 手动触发一次唤醒

你可以通过按键模拟触发一次 power_save_timer_->WakeUp(),看是否能恢复麦克风功能。

✅ 5. 检查硬件连接
  • 确认麦克风是否插拔良好;
  • 检查 I2C 总线是否正常通信;
  • 查看是否有硬件中断或 GPIO 设置冲突。

🧪 四、建议临时修复方案

你可以先尝试在启动时强制开启音频输入:

ZasmartHeadBoard() : boot_button_(BOOT_BUTTON_GPIO), main_button_(POWER_BUTTON_GPIO) {...InitializePowerSaveTimer();InitializePowerManager();// 强制恢复音频输入auto codec = GetAudioCodec();if (codec) {codec->EnableInput(true);}
}

五、总结

问题点排查建议
麦克风未恢复检查 OnExitSleepMode 是否执行
WakeUp() 失效添加日志,确认是否真正唤醒
GetAudioCodec() 失败检查是否为空
InitializePowerManager() 干扰检查是否频繁切换状态
音频驱动问题检查硬件连接、I2C 通信等

如果你能提供以下信息,我可以进一步协助你精准定位问题:

  1. 串口打印的日志内容(尤其是 WakeUp、Sleep Mode、Audio Enable)
  2. 你目前麦克风失灵的具体场景(例如:刚开机、使用一段时间后、断电再开、按某个键之后等)
  3. 是否曾修改过 InitializePowerManager() 或 InitializePowerSaveTimer() 的逻辑?


以笔者二十年智能硬件的开发经验,调试问题就是一个积累经验,所谓见多识广、假设验证,就是精华,小伙伴们共勉。

相关文章:

  • Parasoft为可口可乐赋能: 强化软件开发与质量保证
  • 二进制编码、定点数与浮点数
  • 左手腾讯CodeBuddy 、华为通义灵码,右手微软Copilot,旁边还有个Cursor,程序员幸福指数越来越高了
  • 力扣-两数之和
  • uniapp-商城-64-后台 商品列表(商品修改---页面跳转,深浅copy应用,递归调用等)
  • 【Java学习笔记】main方法
  • 电脑中所有word文件图标变白怎么恢复
  • (vue)前端实现下载后端提供的URL文件
  • 鸿蒙devEco studio如何创建模拟器
  • C++线程池实现
  • JAVA项目中常见的注解总结
  • 学习笔记:黑马程序员JavaWeb开发教程(2025.4.8)
  • 使用Jenkins部署nodejs前端项目
  • 【物联网】基于树莓派的物联网开发【6】——汉化+字体库输入法安装
  • 【Linux系统】第五章 - 第七章 -2 逻辑卷 + RAID阵列
  • 基于R语言的贝叶斯网络模型实践技术应用:开启科研新视角
  • js不同浏览器标签页、窗口或 iframe 之间可以相互通信
  • Redis语法大全
  • 解决dedecms织梦系统{dede:arclist keyword=‘动态获取关键词‘}只生效一次
  • 多模态大语言模型arxiv论文略读(八十七)
  • wordpress批量更换文章的关键字/武汉seo百度
  • 是想建个网站 用本地做服务器/百度一下百度搜索网站