永远选择相信同步原语

起因

qemu有一个可以让虚拟机(guest)使用宿主机(host)的音频播放的参数,-device audio。这个参数在x86上效果不错,但是在arm上效果不行,杂音很多,所以我们需要解决这个问题。方案是使用共享内存ivshmem。

具体的说,就是改写微软的官方样例驱动。虽然这些bug看起来简单,但是由于windows的奇怪特性,以及ARM上没有合适的调试工具,所以我们调试了很久很久……

调试的故事

调试的方法无非是打日志、看coredump、单步调试。

  1. 打日志

打日志可以用DbgPrint,它的输出可以用过DbgView看到。

DbgView有ARM架构版本的,也有x86版本的。虽然微软确实有x86转ARM指令集的功能,但是DbgView ARM版本异常容易崩溃,只能够用一小段时间,然后就会崩溃……据说qemu有串口输出日志的功能,但是由于我们要使用内核态,所以这个功能难以试出来。

  1. 看coredump

这个功能在x86上是可用的,在ARM上因为一些奇奇怪怪的问题,我们要用BcdEdit开启。大致只记得开启了测试模式,然后对照x86去修改其他bcd选项,然后又开了其他几个选项,不限制coredump大小,此时才有概率能够产生coredump。好在崩溃问题几乎是必现的,所以最终能够在几次重启之后正常获得coredump。

  1. 单步调试

据同事说,他曾在x86上用过这个功能对内核态驱动实体机单步调试,但是我们试过很多方法,都没有正常用的方法。

具体说,要使用单步调试,我们就需要网络,在进入系统前就必须要有网络。但是!我们只能在虚拟机中调试,而虚拟机中的驱动是在系统启动之后才加载,所以我们用不了这种方法。

当然,微软还提供了几个选择,比如可以用串口来单步调试,但是估计串口驱动也可能是在启动之后加载的,再加上在调试过程中我已经找到了崩溃的原因,所以就没有继续研究了。

所以,微软,我xxx

BUG1:缺乏同步原语导致的杂音

这是在录音这一部分遇到的bug。这里说的杂音是说,如果你给他录制人声,你能够听得出是这个人说的话,但是你听不清他说的是什么内容。如下图

也就是说,在真正写入数据之前就已经开始录音了,表现的效果就是听不清在说啥。

而播放是没有这个杂音问题的,也需要画图说明。

也就是说,播放的时候,在本次数据完整交给host播放之前,下一次数据是阻塞等待的。

所以录音有杂音,而播放无杂音,本质上就是同步原语的问题。我们无法修改windows内核的代码,所以没法用类似于播放的方法。

至于为什么非得要一个IO线程,就得看气人的安全措施protection ring

https://en.wikipedia.org/wiki/Protection_ring

这个ring中,ring2级别的无法访问ring0、1级别的内容。而我们的驱动运行在两个级别中,初始化的时候是0,初始化之后,读取、写入音频数据的时候是2,而在IO线程中是0,而IVSHMEM共享内存的ring级别就是0。

所以摆在我们面前的只有两个方案:

  1. 在IO队列与驱动之间加一个缓冲区

IO队列将所有能够读到的数据放入缓冲区,然后在windows请求读取的时候,之间从该缓冲区中读取。这样就可以解决这个问题了。

这个方案有几个问题。初始化的时候获取的ring级别是0吗?加入缓冲区导致的延迟可以接受吗?

  1. 让IVSHMEM可以在ring2级别中读取,规避掉IO线程

这是我们最终的方案,修改qemu的IVSHMEM源代码,让它能够跑在ring2级别中。具体修改的过程,只记得是IVSHMEM明明已经支持了这个功能,但是它没有开放给用户用,所以稍微加一个if-else就搞定了

BUG2:作为同步使用的WorkItem为NULL导致的驱动崩溃

这个bug似乎是和对象构造、析构相关的。它在播放与录音中都出现过,但是特点是:

只在arm上出现,不会在x86上出现

不过由于我们最后规避掉了IO线程,所以这个问题只在播放中出现过。

就是因为这个bug,我们花了大把时间在开启coredump上。coredump的问题已经讨论过了。最后修复它的方案就是分割驱动为一个录音驱动和一个播放驱动。虽然确实可以查出来该问题的根本并修复,但是我们发现用简单的方法处理了之后,在ARM上播放依然有杂音。所以我们当时推测是播放录音有相互干扰,所以分割为两个驱动,到此发现杂音没了。

如果时间够的话,我还真想排查出来WorkItem为什么会被释放。我们时间不够了,必须要赶紧做完这东西。

现在回想起来,说不定是ARM的内存屏障没有x86那么严格。我听说有一个公司将x86的java程序放arm上执行的时候就因为这个出现过毛病。

录音虚拟驱动杂音bug修复的更多相关文章

  1. ThinkPHP 3.2.3+ORACLE插入数据BUG修复及支持获取自增Id的上次记录

    TP+ORACLE插入数据BUG修复以及获取自增Id支持getLastInsID方法 这些天在做Api接口时候,发现用TP操作Oracle数据库,发现查询修改删除都能执行, 但一旦执行插入操作老是报错 ...

  2. 仿酷狗音乐播放器开发日志十九——CTreeNodeUI的bug修复二(附源码)

    转载请说明原出处,谢谢 今天本来打算把仿酷狗播放列表的子控件拖动插入功能做一下,但是仔细使用播放列表控件时发现了几个逻辑错误,由于我的播放 列表控件是基于CTreeViewUI和CTreeNodeUI ...

  3. OJ2.0userInfo页面Modify逻辑bug修复,search功能逻辑实现

    这周的主要任务:userInfo页面Modify逻辑bug修复,search功能逻辑实现. (一)Modify逻辑bug修复: 这里存在的bug就是在我们不重置password的时候依照前面的逻辑是不 ...

  4. cocos2d-x多分辨率和随后的自适应CCListView的bug修复

    cocos2d-x多分辨率自适配及因此导致的CCListView的bug修复 cocos2d-x是一款众所周知的跨平台的游戏开发引擎.因为其跨平台的特性.多分辨率支持也自然就有其需求. 因此.在某一次 ...

  5. android-misc-widgets四向(上下左右)抽屉bug修复版--转载

     android-misc-widgets四向(上下左右)抽屉bug修复版 2013-08-04 08:58:13 标签:bug down top panel slidingdrawer 原创作品,允 ...

  6. Spring+SpringMVC+MyBatis+easyUI整合基础篇(八)mysql中文查询bug修复

    写在前面的话 在测试搜索时出现的问题,mysql通过中文查询条件搜索不出数据,但是英文和数字可以搜索到记录,中文无返回记录.本文就是写一下发现问题的过程及解决方法.此bug在第一个项目中点这里还存在, ...

  7. 微信小程序(有始有终,全部代码)开发---跑步App+音乐播放器 Bug修复

    开篇语 昨晚发了一篇: <简年15: 微信小程序(有始有终,全部代码)开发---跑步App+音乐播放器 > 然后上午起来吃完午饭之后,我就准备继续开工的,但是突然的,想要看B站.然后在一股 ...

  8. Saiku Table展示数据合并bug修复(二十五)

    Saiku Table展示数据合并bug修复 Saiku以table的形式展示数据,如果点击了 非空的字段 按钮,则会自动进行数据合并,为空的数据行以及数据列都会自动隐藏掉. 首先我们应该定位问题: ...

  9. git bug修复

    在Git中,由于分支是如此的强大,所以,每个bug都可以通过一个新的临时分支来修复,修复后,合并分支,然后将临时分支删除. 当你接到一个修复一个代号101的bug的任务时,很自然地,你想创建一个分支i ...

  10. 高大上网站-CSS3总结1-图片2D处理以及BUG修复

    高大上网站-CSS3总结1-图片2D处理以及BUG修复 一,前言: 现在的前端UI相对JS来说,重视并不够. 但是CSS3提供的新特性,将现在的网站赤裸裸的划分为两类:一类还在写着老旧样式,或者通过b ...

随机推荐

  1. Element——前端样式美化

    Element 简介    Element 快速入门      https://element.eleme.cn/#/zh-CN/component/button Element 布局      ht ...

  2. Servlet——Response对象

    Response对象         Response 设置响应数据   1.响应行          void setStatus(int sc):设置响应状态码   2.响应头           ...

  3. 从0到1搭建权限管理系统系列三 .net8 JWT创建Token并使用

    说明 该文章是属于OverallAuth2.0系列文章,每周更新一篇该系列文章(从0到1完成系统开发). 该系统文章,我会尽量说的非常详细,做到不管新手.老手都能看懂. 说明:OverallAuth2 ...

  4. [C103] 斐波那契数列

    设 \((i,j)=gcd(i,j)\) \[f_{i}=f_{i-1}+f_{i-2} \] \[f_{i}=f_{i-2}\times f_{1}+f_{i-1}\times f_{2} \] \ ...

  5. IntelliJ IDEA插入时间文本

    IntelliJ IDEA插入时间文本 需求: 在使用IDEA编辑一些文本时,需要插入指定格式的当前时间文本,首先想到的是找找有没有相关的IDEA插件,看到确实有别的猿做过相关的插件,但当时找到的文章 ...

  6. 高强度学习训练第七天总结:JVM分配内存机制

    理解JVM内存分配策略 三大原则+担保机制 JVM分配内存机制有三大原则和担保机制 具体如下所示: 优先分配到eden区 大对象,直接进入到老年代 长期存活的对象分配到老年代 空间分配担保 对象优先在 ...

  7. 2022年9月中国数据库排行榜:榜眼、探花纷纷易主,AnalyticDB蝉联榜单十强

    夏去秋来清风至,榜首前三起涟漪. 2022年9月的 墨天轮中国数据库流行度排行榜火热出炉,相比上月新增两个数据库, 本月榜单前十名可以用一句话概括为:OceanBase新版发布先声夺人,达梦低调做事暂 ...

  8. 深入解析Spring AI框架:在Java应用中实现智能化交互的关键

    今天我们的Spring AI源码分析主题即将结束.我已经对自己感兴趣的基本内容进行了全面的审视,并将这些分析分享给大家.如果你对这个主题感兴趣,可以阅读以下几篇文章.每篇文章都层层递进,深入探讨相关内 ...

  9. kotlin类与对象——>数据类、密封类、泛型

    数据类,用来保存数据的类,使用data对class进行标记 data class User(val name: String, val age: Int) //编译器自动从主构造函数中声明的所有属性导 ...

  10. kotlin类和对象—>属性与字段

    1.声明属性,Kotlin 类中的属性既可以用关键字 var 声明为可变的,也可以用关键字 val 声明为只读的 class Address { var name: String = "Ho ...