前言
今日公司,安卓设备的音量显示不正常,让我来修复这个bug,现在已修复,做个博客,记录一下,以后碰到类似一下子就好解决。
Android音量调节相关
路径
frameworks\base\services\core\java\com\android\server\audio\AudioService.java
这个文件包含了关于android音量调节的代码逻辑。
问题
音量调节那里,调小会直接变成0,而且过程我发现他会有跳动的感觉。
思路
找到apk,调用这个文件的接口,添加打印了解对应逻辑。
找到问题点
- 音量数值对不上
- 音量多次调用onSetStreamVolume函数(每个音乐流都调用一次)
解决问题
-
数值对不上(修改apk与底层数值对应关系)
-
多次调用(把对应干扰流都屏蔽掉)
setStreamVolume接口
/*** 设置音频流的音量。** @param streamType 要设置音量的音频流类型,例如 AudioManager.STREAM_MUSIC。* @param index 要设置的音量索引,通常是一个在音频流范围内的整数值。* @param flags 设置标志,包括有关音量设置的额外信息,例如 AudioManager.FLAG_FIXED_VOLUME。* @param callingPackage 调用该方法的应用程序的包名。* @param caller 调用者的标识。* @param uid 调用者的用户标识。* @param hasModifyAudioSettings 是否有修改音频设置的权限。*/private void setStreamVolume(int streamType, int index, int flags, String callingPackage,String caller, int uid, boolean hasModifyAudioSettings) {if (DEBUG_VOL) {Log.d(TAG, "setStreamVolume(stream=" + streamType + ", index=" + index+ ", calling=" + callingPackage + ")");}if(streamType != 3){return ;}// 如果使用了固定音量,直接返回if (mUseFixedVolume) {return;}Log.d(TAG, "************************");// 确保音频流类型有效ensureValidStreamType(streamType);// 获取音频流类型别名和对应的音量流状态int streamTypeAlias = mStreamVolumeAlias[streamType];VolumeStreamState streamState = mStreamStates[streamTypeAlias];// 获取与音频流类型别名对应的设备final int device = getDeviceForStream(streamType);int oldIndex;// 如果不是 A2DP 设备,并且标志包含了 AudioManager.FLAG_BLUETOOTH_ABS_VOLUME,// 则跳过 A2DP 绝对音量控制请求if (!AudioSystem.DEVICE_OUT_ALL_A2DP_SET.contains(device)&& (flags & AudioManager.FLAG_BLUETOOTH_ABS_VOLUME) != 0) {return;}// 如果是系统调用(例如硬件按键),检查当前用户以正确处理用户限制if (uid == android.os.Process.SYSTEM_UID) {uid = UserHandle.getUid(getCurrentUserId(), UserHandle.getAppId(uid));}// 验证调用包和 app op 权限if (!checkNoteAppOp(STREAM_VOLUME_OPS[streamTypeAlias], uid, callingPackage)) {return;}// 在 Android N 及以上版本,如果音量调整将切换 Zen 模式,// 则检查是否已授予调用包通知策略的访问权限if (isAndroidNPlus(callingPackage)&& wouldToggleZenMode(getNewRingerMode(streamTypeAlias, index, flags))&& !mNm.isNotificationPolicyAccessGrantedForPackage(callingPackage)) {throw new SecurityException("Not allowed to change Do Not Disturb state");}// 如果音量调整在当前 Do Not Disturb 模式下不允许,则直接返回if (!volumeAdjustmentAllowedByDnd(streamTypeAlias, flags)) {return;}synchronized (mSafeMediaVolumeStateLock) {// 重置任何挂起的音量命令mPendingVolumeCommand = null;// 获取旧的音量索引oldIndex = streamState.getIndex(device);// 对索引进行重新缩放Log.d(TAG, "缩放前 index=+"+index);index = index * 10;//rescaleIndex(index * 10, streamType, streamTypeAlias);Log.d(TAG, "缩放后 index=+"+index);// 如果音频流类型别名是 STREAM_MUSIC// 且设备是 A2DP 设备并且标志不包含 AudioManager.FLAG_BLUETOOTH_ABS_VOLUME,// 则发送 Avrcp 绝对音量索引的消息if (streamTypeAlias == AudioSystem.STREAM_MUSIC&& AudioSystem.DEVICE_OUT_ALL_A2DP_SET.contains(device)&& (flags & AudioManager.FLAG_BLUETOOTH_ABS_VOLUME) == 0) {if (DEBUG_VOL) {Log.d(TAG, "setStreamVolume postSetAvrcpAbsoluteVolumeIndex index=" + index+ "stream=" + streamType);}mDeviceBroker.postSetAvrcpAbsoluteVolumeIndex(index / 10);}// 如果设备是 HEARING_AID 设备且音频流是 HEARING_AID 类型,// 则发送设置 HEARING_AID 音量索引的消息if (device == AudioSystem.DEVICE_OUT_HEARING_AID&& streamType == getHearingAidStreamType()) {Log.i(TAG, "setStreamVolume postSetHearingAidVolumeIndex index=" + index+ " stream=" + streamType);mDeviceBroker.postSetHearingAidVolumeIndex(index, streamType);}// 如果音频流类型别名是 STREAM_MUSIC,发送设置系统音频音量的消息if (streamTypeAlias == AudioSystem.STREAM_MUSIC) {setSystemAudioVolume(oldIndex, index, getStreamMaxVolume(streamType), flags);}// 清除 AudioManager.FLAG_FIXED_VOLUME 标志flags &= ~AudioManager.FLAG_FIXED_VOLUME;// 如果音频流类型别名是 STREAM_MUSIC 且是固定音量设备,则设置 AudioManager.FLAG_FIXED_VOLUME 标志if (streamTypeAlias == AudioSystem.STREAM_MUSIC && isFixedVolumeDevice(device)) {flags |= AudioManager.FLAG_FIXED_VOLUME;// 对于固定音量设备,音量要么为 0,要么为允许的最大值if (index != 0) {if (mSafeMediaVolumeState == SAFE_MEDIA_VOLUME_ACTIVE &&mSafeMediaVolumeDevices.contains(device)) {index = safeMediaVolumeIndex(device);} else {index = streamState.getMaxIndex();}}Log.d(TAG,"STREAM_MUSIC index = " + index);}// 如果音量超出安全范围,显示安全音量警告,并设置挂起的音量命令if (!checkSafeMediaVolume(streamTypeAlias, index, device)) {mVolumeController.postDisplaySafeVolumeWarning(flags);mPendingVolumeCommand = new StreamVolumeCommand(streamType, index, flags, device);} else {// 否则,调用 onSetStreamVolume 设置音量onSetStreamVolume(streamType, index, flags, device, caller, hasModifyAudioSettings);// 获取最终的音量索引index = mStreamStates[streamType].getIndex(device);Log.d(TAG, "获取最终的音量索引"+ "index=" + index + ")");}}
问题 1关键点
问题 2关键点
结束
轻描淡写的俩个问题,看上去简单,如果你什么也不知道情况下调试,你就知道了。
刚解决完,马上发一篇博客,帮助更多的人,因为有问题找不到答案真的很无助,做点力所能及的事情吧。