Android14 系统左右声音通道设置代码

Android14 系统左右声音通道设置代码

文章目录

Android14 系统左右声音通道设置代码一、前言二、系统级设置左右声音通道分析1、各方案设置左右声音通道的主要代码(1)3588 Android13 方案的实现(2)9679 Android14 方案的实现(3)311D2 Android13 方案的实现

2、串口验证左右声道平衡设置3、AudioService.java 监听并设置左右声音通道平衡4、AudioSystem.java 上层设置左右声音通道平衡的主要代码

三、其他1、Android设置左右声音通道代码小结(1)串口设置左右平衡声音通道:

2、为啥不用方案需要调用不同的声音通道接口实现?

一、前言

Android 系统左右声音通道设置在有些场景下会有用,信源或者多音箱的条件下可能需要设置, 还有些需求比如:用户偏好设置、将声音定位到特定方向,提升沉浸感。

最简单的理解:

比如耳机接入Android设备后,设置声音通道左平衡,只有左边的耳机出声;

设置声音通道右平衡,只有右边的耳机出声;

上面这样就可以简单验证系统左右声音通道是否有效。

网上基本找不到系统级的设置左右声音通道的代码,即使使用AI工具也搜不到。 网上设置左右声音通道都是针对某个播放媒体对象进行设置,并不是全局的。 网上设置左右声音通道关键部分代码如下:

// 1、应用左右声道平衡,audioTrack 设置

for (int i = 0; i < audioData.length; i += 2) {

audioData[i] = (short) (audioData[i] * leftVolume); // 左声道

audioData[i + 1] = (short) (audioData[i + 1] * rightVolume); // 右声道

}

audioTrack.write(audioData, 0, audioData.length);

// 2、应用左右声道平衡,MediaPlayer 设置

mediaPlayer = MediaPlayer.create(this, R.raw.your_audio_file);

if (mediaPlayer != null) {

// 设置左声道音量为 0.2,右声道音量为 0.8

mediaPlayer.setVolume(0.2f, 0.8f);

mediaPlayer.start();

}

上面的两种方法都是针对某个音频流对象,并不针对系统的。

那么系统有接口方法吗?

其实是有设置左右声音通道接口的,本文记录总结一下。

二、系统级设置左右声音通道分析

目前接触的系统芯片方案主要有:RK、MTK、AML;他们的设置左右声音通道代码居然都不一样。

1、各方案设置左右声音通道的主要代码

(1)3588 Android13 方案的实现

@Override

public int getBalance() throws RemoteException {

float result = 0.0f;

int value_int = 0;

String value = "";

try {

result = Settings.System.getFloatForUser(mContext.getContentResolver(),

Settings.System.MASTER_BALANCE, UserHandle.USER_CURRENT);

value_int = (int) Math.round(result);

Log.d(TAG,"getBalance result == " + result+",value_int:"+value_int);

} catch (Exception e) {

e.printStackTrace();

}

return value_int;

}

@Override

public void setBalance(int balance) throws RemoteException {

synchronized (lock) {

Settings.System.putFloatForUser(mContext.getContentResolver(),

Settings.System.MASTER_BALANCE, uiValue2parameterValue(balance), UserHandle.USER_CURRENT);

}

}

//将UI 输入值 [-50, 50] 映射到 参数范围 [-1.0, 1.0]

public float uiValue2parameterValue(int value) {

float y = ((float)(value + 50) / 50) - 1.0f;

return y;

}

上面看起来是比较原生的设置代码; 通过读取和设置 Settings.System.MASTER_BALANCE 的值设置左右声音平衡 系统中的 System.MASTER_BALANCE 的是值-1f ~ 1f,需要转换成-50 ~ 50 用户获取和设置的左右平衡值范围是-50 ~ 50。

(2)9679 Android14 方案的实现

import com.mediatek.tv.oneworld.tvapi.audio.TvAudioManager; //mtk api

private TvAudioManager mTvAudioManager;

mTvAudioManager = new TvAudioManager(mContext);

@Override

public int getBalance() throws RemoteException {

int result = mTvAudioManager.get(mContext, TvAudioManager.CFG_AUD_AUD_BALANCE, 0);

DebugLog.debug("getBalance result == " + result);

return result;

}

@Override

public void setBalance(int balance) throws RemoteException {

// balance : (-50 ~ 50)

mTvAudioManager.set(mContext, TvAudioManager.CFG_AUD_AUD_BALANCE, String.valueOf(balance));

DebugLog.debug("setBalance: balance == " + balance);

}

(3)311D2 Android13 方案的实现

import com.droidlogic.app.AudioEffectManager; //AML方案

private AudioEffectManager mAudioEffectManager;

public SkgVoiceService(Context applicationContext) {

mContext = applicationContext;

mAudioEffectManager = AudioEffectManager.getInstance(applicationContext);

DebugLog.debug("");

}

@Override

public int getBalance() throws RemoteException {

int result = mAudioEffectManager.getBalanceStatus();

DebugLog.debug("getBalance result == " + result);

return result;

}

@Override

public void setBalance(int balance) throws RemoteException {

mAudioEffectManager.setBalance(balance);

DebugLog.debug("setBalance and no return balance == " + balance);

}

后面的MTK 和AML 方案上,实现都不一样,都是供应商封装的接口; 通过Settings属性查看并没有修改上层的 System.MASTER_BALANCE 属性值; 可能是直接控制底层接口完成的左右声音通道平衡设置,这里不进行深入研究。

从上面的代码看,RK实现左右声音通道平衡设置的代码可以移植使用。

那么其他方案也可以用Settings属性值设置的方式吗?

2、串口验证左右声道平衡设置

属性:Settings.System.MASTER_BALANCE 的设置和读取: settings 属性名称都是纯小写的。

settings get system master_balance //获取当前左右声音通道平衡值,默认 null

settings put system master_balance -1 //设置左平衡,左边音箱出声

settings put system master_balance 1 //设置右平衡,右边音箱出声

settings put system master_balance 0 //设置左右平衡,左右音箱都出声,默认情况

通过串口命令验证了下确实可用。MTK方案上试了有用。

为啥会有用呢,应该是系统某个地方监听了settings属性变化, 最后调用某个接口,实现了调用逻辑。

查看了系统代码,确实有监听的地方,就在 AudioService.java

3、AudioService.java 监听并设置左右声音通道平衡

framework\base\services\core\java\com\android\server\audio\AudioService.java

下面是主要代码:

private class SettingsObserver extends ContentObserver {

SettingsObserver() {

super(new Handler());

... //十几个Settings 属性状态的监听

mContentResolver.registerContentObserver(Settings.Global.getUriFor(

Settings.Global.DOCK_AUDIO_MEDIA_ENABLED), false, this);

mContentResolver.registerContentObserver(Settings.System.getUriFor(

Settings.System.MASTER_MONO), false, this);

//这个有注册监听 Settings.System.MASTER_BALANCE 的变化

mContentResolver.registerContentObserver(Settings.System.getUriFor(

Settings.System.MASTER_BALANCE), false, this);

}

@Override

public void onChange(boolean selfChange) {

super.onChange(selfChange);

synchronized (mSettingsLock) {

readDockAudioSettings(mContentResolver); //读取停靠音频设置

updateMasterMono(mContentResolver); //更新主声道单声道

updateMasterBalance(mContentResolver); //更新主声道平衡,左右声音通道平衡设置

updateEncodedSurroundOutput(); //更新编码环绕声输出

sendEnabledSurroundFormats(mContentResolver, mSurroundModeChanged); //发送启用的环绕声格式

updateAssistantUIdLocked(/* forceUpdate= */ false); //并更新助手UID(不强制更新)

}

}

...

}

//继续追踪左右声道平衡的代码逻辑

private void updateMasterBalance(ContentResolver cr) {

//或者左右声音通道平衡的具体值

final float masterBalance = System.getFloatForUser(

cr, System.MASTER_BALANCE, 0.f /* default */, UserHandle.USER_CURRENT);

if (DEBUG_VOL) { // DEBUG_VOL 默认true

Log.d(TAG, String.format("Master balance %f", masterBalance));

}

//调用 AudioSystem.setMasterBalance 方法设置左右声音通道

if (AudioSystem.setMasterBalance(masterBalance) != 0) {

Log.e(TAG, String.format("setMasterBalance failed for %f", masterBalance));

}

}

上面可以看到左右声音通道平衡的设置最终是调用了 AudioSystem 的接口方法。

import android.media.AudioSystem;

AudioSystem.setMasterBalance(masterBalance); //设置值范围:-1f ~ 1f

AudioSystem 接口和方法默认是隐藏的,系统framework中可以调用到。 系统签名应用也是无法调用到,需要导入系统的framework jar包就可能调用到AudioSystem 的类。

4、AudioSystem.java 上层设置左右声音通道平衡的主要代码

framework\base\media\java\android\media\AudioSystem.java

/**

* @hide

*/

@TestApi

public class AudioSystem

{

private static final boolean DEBUG_VOLUME = false;

private static final String TAG = "AudioSystem";

// private constructor to prevent instantiating AudioSystem

private AudioSystem() { //不能 new

throw new UnsupportedOperationException("Trying to instantiate AudioSystem");

}

//暴露的都是静态方法,所以不用new。

/** @hide returns master balance value in range -1.f -> 1.f, where 0.f is dead center. */

@TestApi

@RequiresPermission(android.Manifest.permission.MODIFY_AUDIO_SETTINGS)

public static native float getMasterBalance();

/** @hide Changes the audio balance of the device. */

@TestApi

@RequiresPermission(android.Manifest.permission.MODIFY_AUDIO_SETTINGS)

public static native int setMasterBalance(float balance);

}

上面已有提示 左右声音通道的值范围是:-1.f -> 1.f

声音通道平衡的具体实现是nativie的调用,后续不做进一步分析。 有兴趣的自行查阅:framework\base\core\jni\android_media_AudioSystem.cpp

三、其他

1、Android设置左右声音通道代码小结

(1)串口设置左右平衡声音通道:

settings get system master_balance //获取当前左右声音通道平衡值,默认 null

settings put system master_balance -1 //设置左平衡,左边音箱出声

settings put system master_balance 1 //设置右平衡,右边音箱出声

settings put system master_balance 0 //设置左右平衡,左右音箱都出声,默认情况

Android 原生设置左右声音通道代码:

//1、Settings属性设置,AudioService.java 有监听变化和调用接口

Settings.System.putFloat(mContext.getContentResolver(),Settings.System.MASTER_BALANCE, balance);

//2、AudioSystem 的静态方法

AudioSystem.setMasterBalance(masterBalance)

如果是系统签名应用,但是又没有导入framework的jar包, 是无法识别AudioSystem的,并且System.MASTER_BALANCE 也是无法识别的;

可以使用代码直接设置具体属性就行:

Settings.System.putFloat(mContext.getContentResolver(), "master_balance", balance);

2、为啥不用方案需要调用不同的声音通道接口实现?

这里不去看他们的具体实现,直接简单分析一下。

对比了不同的Android版本的系统代码:

Android 9或者更旧的代码 AudioSystem.java 里面是没有setMasterBalance方法的;

Android11 的系统代码中是有AudioSystem.setMasterBalance 方法的。

所以之前Android9之前的版本,不得不使用自定义的接口调用底层声音设置。 所以不同供应商提供的调用接口是有差异的。 这个只是我的一个猜测哈。

测试了一下MTK方案:

使用供应商提供的设置声音接口或者使用Settings属性都是可以设置左右声音通道平衡的;

但是有个小问题:Settings属性设置左声道后,再次调用供应商的接口设置左声道或者右声道,会导致设备没有声音;

重新用供应商的接口设置左右声音通道值为0 ,设备就有声音了。

估计是底层兼容有问题!同时设置可能标志位会错乱。 如果能只用Google接口实现是最好的。毕竟Google考虑的应该比较全一点。

最近也是发现一个供应商的蓝牙音箱左右声音通道问题: 连接耳机设置左右声音通道都是没有问题的。

AML 和MTK 的方案都是存在的:

连接多音箱的蓝牙设备,设置左右声音通道没有用!

RK 的方案是没有问题的,因为它是使用的Google原生接口设置的左右声音通道实现。

这种情况有可能就是 AML 和MTK 的方案 没有考虑蓝牙连接下的左右声音通道平衡导致 。