Android雜談:音頻調(diào)試小計

經(jīng)常要搞音頻調(diào)試偎窘,很麻煩蔫巩,現(xiàn)在記錄下鸣驱。

常用數(shù)據(jù)結(jié)構(gòu)

/system/media/audio/include/system/audio.h
定義了常用的stream類型,例如3就是音樂的stream哮内。

/* Audio stream types */
typedef enum {
    /* These values must kept in sync with
     * frameworks/base/media/java/android/media/AudioSystem.java
     */
    AUDIO_STREAM_DEFAULT          = -1,
    AUDIO_STREAM_MIN              = 0,
    AUDIO_STREAM_VOICE_CALL       = 0,
    AUDIO_STREAM_SYSTEM           = 1,
    AUDIO_STREAM_RING             = 2,
    AUDIO_STREAM_MUSIC            = 3,
    AUDIO_STREAM_ALARM            = 4,
    AUDIO_STREAM_NOTIFICATION     = 5,
    AUDIO_STREAM_BLUETOOTH_SCO    = 6,
    AUDIO_STREAM_ENFORCED_AUDIBLE = 7, /* Sounds that cannot be muted by user
                                        * and must be routed to speaker
                                        */
    AUDIO_STREAM_DTMF             = 8,
    AUDIO_STREAM_TTS              = 9,  /* Transmitted Through Speaker.
                                         * Plays over speaker only, silent on other devices.
                                         */
    AUDIO_STREAM_ACCESSIBILITY    = 10, /* For accessibility talk back prompts */
    AUDIO_STREAM_REROUTING        = 11, /* For dynamic policy output mixes */
    AUDIO_STREAM_PATCH            = 12, /* For internal audio flinger tracks. Fixed volume */
    AUDIO_STREAM_PUBLIC_CNT       = AUDIO_STREAM_TTS + 1,
    AUDIO_STREAM_CNT              = AUDIO_STREAM_PATCH + 1,
} audio_stream_type_t;

定義了輸出設(shè)備id,例如0x4對應(yīng)設(shè)備就是有線耳機

    /* output devices */
    AUDIO_DEVICE_OUT_EARPIECE                  = 0x1,
    AUDIO_DEVICE_OUT_SPEAKER                   = 0x2,
    AUDIO_DEVICE_OUT_WIRED_HEADSET             = 0x4,
    AUDIO_DEVICE_OUT_WIRED_HEADPHONE           = 0x8,

2.常用調(diào)試方法

1.查看音量的log

關(guān)鍵tag是AudioMTKGainController
插入耳機圈暗,調(diào)小音量粥鞋,出現(xiàn)下面log

01-01 00:29:51.752251   499   906 D AudioMTKGainController: setNormalVolume(), stream 8, devices 0x8, index 8, mode 0x0

上面的log就是把設(shè)備耳機的stream 8(DTMF聲)的音量設(shè)置為8缘挽。

拔出耳機,調(diào)小音量呻粹,出現(xiàn)下面log

AudioMTKGainController: setNormalVolume(), stream 1, devices 0x2, index 13, mode 0x0
AudioMTKGainController: getGainDevice(), input devices = 0x2, return gainDevice = 2
AudioMTKGainController: setSpeakerGain(), gain = 10, spkAnaType = 2, spkLMixerName = Audio_Speaker_PGA_gain, spkRMixerName = Audio_Speaker_PGA_gain

上面的log就是把設(shè)備揚聲器的stream 1(系統(tǒng)聲)的音量設(shè)置為13壕曼。這里的13只是上層表示的一個音量index,要轉(zhuǎn)換成底層的gain等浊,例如index=13時腮郊,gain就是10.
下面是設(shè)置speaker音量的流程

status_t AudioMTKGainController::setNormalVolume(int stream, int index, int devices, audio_mode_t mode)
{
    ALOGD("setNormalVolume(), stream %d, devices 0x%x, index %d, mode 0x%x", stream, devices, index, mode);
    // get gain device
    GAIN_DEVICE gainDevice = getGainDevice(devices); //獲取要設(shè)置增益的設(shè)備,2是speaker
    if (isSpeakerCategory(gainDevice))
    {
        if (mSpec.spkAnaType >= 0 && mSpec.spkAnaType < NUM_GAIN_ANA_TYPE)
        {
            unsigned char gain = mGainTable.streamGain[stream][gainDevice][index].analog[mSpec.spkAnaType];   //把index轉(zhuǎn)換成底層的gain增益
            setSpeakerGain(gain);         //設(shè)置speaker的增益
        }
    }

2.查看設(shè)置Parameter的log

在音頻控制中筹燕,常常有一種比較粗暴的設(shè)置方法轧飞,就是直接設(shè)置參數(shù)Parameter,這個可以在java層直接設(shè)置撒踪,然后在hal層進行處理过咬。例如在fm中,在java層設(shè)置參數(shù)
/packages/apps/FMRadio/src/com/android/fmradio/FmService.java

mAudioManager.setParameters("AudioFmPreStop=1");

在AudioSystem中進行處理
/frameworks/av/media/libmedia/AudioSystem.cpp

status_t AudioSystem::setParameters(audio_io_handle_t ioHandle, const String8& keyValuePairs)
{
    const sp<IAudioFlinger>& af = AudioSystem::get_audio_flinger();
    if (af == 0) return PERMISSION_DENIED;
    return af->setParameters(ioHandle, keyValuePairs);
}

String8 AudioSystem::getParameters(audio_io_handle_t ioHandle, const String8& keys)
{
    const sp<IAudioFlinger>& af = AudioSystem::get_audio_flinger();
    String8 result = String8("");
    if (af == 0) return result;

    result = af->getParameters(ioHandle, keys);
    return result;
}

上面是原生的代碼,是在AudioFlinger中處理制妄,而在mtk中掸绞,往往是有AudioPolicy處理

 aps->SetPolicyManagerParameters (POLICY_SET_FM_PRESTOP,value,0,0);

3.了解路由規(guī)則

下面的log表示路由到耳機

01-03 06:44:36.621514   570  1265 D AudioALSAStreamOut: +setParameters(): routing=8

定義是在
http://androidxref.com/6.0.0_r1/xref/frameworks/base/media/java/android/media/AudioSystem.java#122

    /* Routing bits for the former setRouting/getRouting API */
    /** @deprecated */
    @Deprecated public static final int ROUTE_EARPIECE          = (1 << 0);
    /** @deprecated */
    @Deprecated public static final int ROUTE_SPEAKER           = (1 << 1);
    /** @deprecated use {@link #ROUTE_BLUETOOTH_SCO} */
    @Deprecated public static final int ROUTE_BLUETOOTH = (1 << 2);
    /** @deprecated */
    @Deprecated public static final int ROUTE_BLUETOOTH_SCO     = (1 << 2);
    /** @deprecated */
    @Deprecated public static final int ROUTE_HEADSET           = (1 << 3);
    /** @deprecated */
    @Deprecated public static final int ROUTE_BLUETOOTH_A2DP    = (1 << 4);
    /** @deprecated */
    @Deprecated public static final int ROUTE_ALL               = 0xFFFFFFFF;

3.了解Audio Patch

Audio Patch是一個通路,包含源source和目標(biāo)sink忍捡。例如FM就是源集漾,耳機就是目標(biāo)sink。

4.了解Audio Mode

Audio的模式砸脊,例如正常模式具篇,通話模式等
/frameworks/base/media/java/android/media/AudioSystem.java

    /* modes for setPhoneState, must match AudioSystem.h audio_mode */
    public static final int MODE_INVALID            = -2;
    public static final int MODE_CURRENT            = -1;
    public static final int MODE_NORMAL             = 0;
    public static final int MODE_RINGTONE           = 1;
    public static final int MODE_IN_CALL            = 2;
    public static final int MODE_IN_COMMUNICATION   = 3;
    public static final int NUM_MODES               = 4;

5.了解Audio Trace

看看下面的log

01-03 08:26:37.051925  5353  5353 V ToneGenerator: AudioTrack(0xf2f36700) created
01-03 08:26:37.051999  5353  5353 D AudioTrack: set(): 0xf2f36700, streamType 8, sampleRate 44100, format 0x1, channelMask 0x1, frameCount 882, flags #4, notificationFrames 0, sessionId 0, transferType 1, uid -1, pid -1

上面的log是點擊撥號盤,發(fā)出聲音打出來的凌埂。0xf2f36700是AudioTrack對象的ID驱显。streamType 是按鍵音。

6.關(guān)于SessionId

抄自網(wǎng)上:
一個Session就是一個會話。每個會話都有一個獨一無二的Id來標(biāo)識埃疫。該Id的最終管理在AudioFlinger中伏恐。
一個會話可以被多個AudioTrack對象和MediaPlayer共用。
共用一個Session的AudioTrack和MediaPlayer共享相同的AudioEffect栓霜。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末翠桦,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子胳蛮,更是在濱河造成了極大的恐慌销凑,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,270評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件仅炊,死亡現(xiàn)場離奇詭異斗幼,居然都是意外死亡,警方通過查閱死者的電腦和手機抚垄,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,489評論 3 395
  • 文/潘曉璐 我一進店門蜕窿,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人呆馁,你說我怎么就攤上這事桐经。” “怎么了智哀?”我有些...
    開封第一講書人閱讀 165,630評論 0 356
  • 文/不壞的土叔 我叫張陵次询,是天一觀的道長。 經(jīng)常有香客問我瓷叫,道長屯吊,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,906評論 1 295
  • 正文 為了忘掉前任摹菠,我火速辦了婚禮盒卸,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘次氨。我一直安慰自己蔽介,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,928評論 6 392
  • 文/花漫 我一把揭開白布煮寡。 她就那樣靜靜地躺著虹蓄,像睡著了一般。 火紅的嫁衣襯著肌膚如雪幸撕。 梳的紋絲不亂的頭發(fā)上薇组,一...
    開封第一講書人閱讀 51,718評論 1 305
  • 那天,我揣著相機與錄音坐儿,去河邊找鬼律胀。 笑死宋光,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的炭菌。 我是一名探鬼主播罪佳,決...
    沈念sama閱讀 40,442評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼黑低!你這毒婦竟也來了赘艳?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,345評論 0 276
  • 序言:老撾萬榮一對情侶失蹤投储,失蹤者是張志新(化名)和其女友劉穎第练,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體玛荞,經(jīng)...
    沈念sama閱讀 45,802評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,984評論 3 337
  • 正文 我和宋清朗相戀三年呕寝,在試婚紗的時候發(fā)現(xiàn)自己被綠了勋眯。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,117評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡下梢,死狀恐怖客蹋,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情孽江,我是刑警寧澤讶坯,帶...
    沈念sama閱讀 35,810評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站岗屏,受9級特大地震影響辆琅,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜这刷,卻給世界環(huán)境...
    茶點故事閱讀 41,462評論 3 331
  • 文/蒙蒙 一婉烟、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧暇屋,春花似錦似袁、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,011評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至定鸟,卻和暖如春而涉,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背仔粥。 一陣腳步聲響...
    開封第一講書人閱讀 33,139評論 1 272
  • 我被黑心中介騙來泰國打工婴谱, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留蟹但,地道東北人。 一個月前我還...
    沈念sama閱讀 48,377評論 3 373
  • 正文 我出身青樓谭羔,卻偏偏與公主長得像华糖,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子瘟裸,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,060評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 提醒一下客叉,純個人筆記,你完全可能看暈 一话告、音頻數(shù)字化基礎(chǔ)知識 見書兼搏,列出知識點如下: 聲音聲波,聲音頻率沙郭、響度佛呻, ...
    YY17閱讀 31,298評論 6 48
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)病线,斷路器吓著,智...
    卡卡羅2017閱讀 134,665評論 18 139
  • 第一部分 AudioTrack分析 一、目的 本文的目的是通過從Audio系統(tǒng)來分析Android的代碼送挑,包括An...
    口袋FPV閱讀 5,398評論 0 9
  • 任何吸引人的游戲都少不了聲音绑莺。iOS開發(fā)者在游戲中需要使用聲音時有多種選擇,取決于對游戲中音頻的控制需求惕耕,可以選擇...
    wingsmm閱讀 12,919評論 4 24
  • 見到見到了,怎么還打個問號惭缰? 因為不確定浪南,我不確定自己見到的成人世界是不是普世標(biāo)準(zhǔn)下的成人世界;我也不確定自己見到...
    一顆大紅棗閱讀 523評論 2 0