ffmpeg中samba網(wǎng)絡(luò)協(xié)議的兼容分析(二)

上回對avformat_open_input進(jìn)行了解析,在avformat_open_input內(nèi)部,對samba網(wǎng)絡(luò)協(xié)議的匹配其實(shí)是通過函數(shù)指針去調(diào)用samba的libsmbc_open()

我們先回顧一下:

static av_cold int libsmbc_open(URLContext *h, const char *url, int flags)
{
    LIBSMBContext *libsmbc = h->priv_data;
    int access, ret;
    struct stat st;

    libsmbc->fd = -1;
    libsmbc->filesize = -1;

    if ((ret = libsmbc_connect(h)) < 0)
        goto fail;

    if ((flags & AVIO_FLAG_WRITE) && (flags & AVIO_FLAG_READ)) {
        access = O_CREAT | O_RDWR;
        if (libsmbc->trunc)
            access |= O_TRUNC;
    } else if (flags & AVIO_FLAG_WRITE) {
        access = O_CREAT | O_WRONLY;
        if (libsmbc->trunc)
            access |= O_TRUNC;
    } else
        access = O_RDONLY;

    /* 0666 = -rw-rw-rw- = read+write for everyone, minus umask */
    if ((libsmbc->fd = smbc_open(url, access, 0666)) < 0) {
        ret = AVERROR(errno);
        av_log(h, AV_LOG_ERROR, "File open failed: %s\n", strerror(errno));
        goto fail;
    }

    if (smbc_fstat(libsmbc->fd, &st) < 0)
        av_log(h, AV_LOG_WARNING, "Cannot stat file: %s\n", strerror(errno));
    else
        libsmbc->filesize = st.st_size;

    return 0;
  fail:
    libsmbc_close(h);
    return ret;
}

此函數(shù)內(nèi)部調(diào)用了libsmbclient庫的smbc_open()函數(shù)

回到上回的問題:為什么基于ffmpeg的播放器在模擬器上可以正常播放smb://的鏈接拌阴,到真機(jī)上就是報(bào)錯(cuò)了呢:

[smb @ 0x107d041d0] File open failed: Permission denied

為了查明真相,我們先分析問題:

  • 1.報(bào)錯(cuò)為Permission denied,是否真的是權(quán)限問題因篇?
其實(shí)就是對ffmpeg內(nèi)部日志的不信任

當(dāng)我們遇到問題時(shí),如果對日志表示懷疑的笔横,那么我們就要去源碼里面才能找到答案

所以竞滓,此問題一般很棘手,我們先放著吹缔,稍后在分析
  • 2.權(quán)限報(bào)錯(cuò)商佑,一般分為密碼錯(cuò)誤或者賬戶錯(cuò)誤或者根本所用賬戶沒有訪問權(quán)限
我的賬戶默認(rèn)開啟了Guest訪問可讀寫權(quán)限,并且模擬器可以正常播放(訪問)厢塘,為何真機(jī)不行莉御?

是不是真機(jī)內(nèi)部使用的默認(rèn)賬戶并非Guest?

是不是真機(jī)內(nèi)部沒有傳遞賬戶驗(yàn)證信息俗冻?

真機(jī)和模擬器的CPU架構(gòu)不同礁叔,是否ffmpeg在調(diào)用smbc_open時(shí)沒有使用驗(yàn)證信息

解決思路

帶著上面的疑問,首先想到的就是要調(diào)試迄薄,那如何調(diào)試呢琅关?當(dāng)然是寫一個(gè)demo,一個(gè)用于測試smbc_open的demo讥蔽。

回到最開始關(guān)于libsmbc_open的ffmpeg源碼涣易,我發(fā)現(xiàn)在使用smbc_open之前還需要建立smb的連接:

static av_cold int libsmbc_connect(URLContext *h)
{
    LIBSMBContext *libsmbc = h->priv_data;

    libsmbc->ctx = smbc_new_context();
    if (!libsmbc->ctx) {
        int ret = AVERROR(errno);
        av_log(h, AV_LOG_ERROR, "Cannot create context: %s.\n", strerror(errno));
        return ret;
    }
    if (!smbc_init_context(libsmbc->ctx)) {
        int ret = AVERROR(errno);
        av_log(h, AV_LOG_ERROR, "Cannot initialize context: %s.\n", strerror(errno));
        return ret;
    }
    smbc_set_context(libsmbc->ctx);

    smbc_setOptionUserData(libsmbc->ctx, h);
    smbc_setFunctionAuthDataWithContext(libsmbc->ctx, libsmbc_get_auth_data);

    if (libsmbc->timeout != -1)
        smbc_setTimeout(libsmbc->ctx, libsmbc->timeout);
    if (libsmbc->workgroup)
        smbc_setWorkgroup(libsmbc->ctx, libsmbc->workgroup);

    if (smbc_init(NULL, 0) < 0) {
        int ret = AVERROR(errno);
        av_log(h, AV_LOG_ERROR, "Initialization failed: %s\n", strerror(errno));
        return ret;
    }
    return 0;
}

分析上述libsmbc_connect,可以學(xué)習(xí)到smbc建立連接的方法:

smbc也是純c的思想冶伞,但凡寫得好的第三方c語言庫新症,都會(huì)用到context思想,一個(gè)神秘的翻譯:上下文响禽。

后續(xù)操作都是基于context的操作徒爹。

廢話不多說荚醒,查閱smbclient的文檔,模仿ffmpeg的調(diào)用把這個(gè)demo寫成:


+ (void)test
{
    //構(gòu)造context
    SMBCCTX * ctx = smbc_new_context();
    if (!ctx) {
        NSLog(@"smbc_new_context failed");
    }
    
    //初始化context
    if (!smbc_init_context(ctx))
    {
        NSLog(@"smbc_init_context failed");
    }
    smbc_set_context(ctx);

    //初始化smbc
    if (smbc_init(NULL, 0) < 0) {
      NSLog(@"smbc_init failed");
    }
    
    //打開鏈接
    if ((smbc_open("smb://172.16.9.10/video/test.mp4", O_RDONLY | O_WRONLY, 0666)) < 0) {
        NSLog(@"File open failed");
    }


到此運(yùn)行隆嗅,在模擬器上界阁,發(fā)現(xiàn)smbc_open是成功了的

我切換到真機(jī),居然把我們的之前遇到的問題復(fù)現(xiàn)了

這就好辦了胖喳,那既然問題能復(fù)現(xiàn)泡躯,并且范圍也被縮小到這段代碼之內(nèi),還排除了ffmpeg的嫌疑(對應(yīng)問題1:Permission denied的準(zhǔn)確性)

接下來劃重點(diǎn)了

既然我們通過上述幾行代碼能建立smb的連接丽焊,那之前提到的--傳遞賬戶驗(yàn)證信息--又是在哪里調(diào)用呢较剃?

答案還是在ffmpeg的源碼中,我們不難發(fā)現(xiàn)libsmbc_connect中有個(gè)smbc_setFunctionAuthDataWithContext函數(shù)技健,此函數(shù)正式設(shè)置context驗(yàn)證信息的

smbc_setFunctionAuthDataWithContext為context設(shè)置一個(gè)回調(diào)函數(shù)写穴,用于傳遞auth三要素

SMBCCTX的賬戶驗(yàn)證信息包括三個(gè)部分:

  • 工作組:默認(rèn)WORKGROUP
  • 賬戶:默認(rèn)GUEST
  • 密碼:默認(rèn)為空

那么,我么繼續(xù)優(yōu)化demo:


static void my_smbc_get_auth_data_with_context_fn(SMBCCTX *c,
                                                  const char *srv,
                                                  const char *shr,
                                                  char *workgroup, int wglen,
                                                  char *username, int unlen,
                                                  char *password, int pwlen)
{

}

+ (void)test
{
    SMBCCTX * ctx = smbc_new_context();
    if (!ctx) {
        NSLog(@"smbc_new_context failed");
    }
    
    if (!smbc_init_context(ctx))
    {
        NSLog(@"smbc_init_context failed");
    }
    smbc_set_context(ctx);
    
    //為context設(shè)置auth的回調(diào)函數(shù)
    smbc_setFunctionAuthDataWithContext(ctx, my_smbc_get_auth_data_with_context_fn);


    if (smbc_init(NULL, 0) < 0) {
      NSLog(@"smbc_init failed");
    }
    
    if ((smbc_open("smb://172.16.9.10/video/test.mp4", O_RDONLY | O_WRONLY, 0666)) < 0) {
        NSLog(@"File open failed");
    }


這里關(guān)鍵的smbc_setFunctionAuthDataWithContext是為SMBCCTX設(shè)置賬戶驗(yàn)證信息的

通過調(diào)試發(fā)現(xiàn)凫乖,在執(zhí)行到smbc_open時(shí)确垫,會(huì)調(diào)用my_smbc_get_auth_data_with_context_fn,我們打個(gè)斷點(diǎn)看看此方法中帽芽,參數(shù)都是些什么妖魔鬼怪

image

image

image

image

image

上面的調(diào)試信息是在真機(jī)運(yùn)行下出現(xiàn)的删掀,我們一眼看出username為mobile肯定是不對的,因?yàn)槲也]有為我的samba服務(wù)器配置過名字為mobile的用戶

至此导街,原因算是找到了披泪,真機(jī)下,smbc的默認(rèn)賬戶為mobile搬瑰,而模擬器是x86_64架構(gòu)款票,在smbc執(zhí)行時(shí)會(huì)為此類設(shè)備設(shè)置默認(rèn)賬戶名為Guest,我們再驗(yàn)證一下:

image

好了泽论,那問題來了

我們?nèi)绾螢閒fmpeg解決這個(gè)問題呢?

一個(gè)最簡單的方案就是在我們的samba服務(wù)器上添加一個(gè)無密碼的名字為“mobile”的賬戶

我偏不這樣該咋辦艾少?

先回顧ffmpeg的源碼,在libsmbclient.c中翼悴,驗(yàn)證信息回調(diào)函數(shù)被設(shè)置為libsmbc_get_auth_data缚够,有一句注釋很出戲

static void libsmbc_get_auth_data(SMBCCTX *c, const char *server, const char *share,
                                  char *workgroup, int workgroup_len,
                                  char *username, int username_len,
                                  char *password, int password_len)
{
    /* Do nothing yet. Credentials are passed via url.
     * Callback must exists, there might be a segmentation fault otherwise. */
}

正是這句:/* Do nothing yet. Credentials are passed via url.

好一個(gè)”via url“, 我又該如何鹦赎?

我沒查到資料谍椅,瞎蒙了一個(gè)(完全憑經(jīng)驗(yàn)),原鏈接為:

smb://172.16.9.10/video/test.mp4

通過”via url"啟發(fā)古话,我改為這個(gè):

smb://guest@172.16.9.10/video/test.mp4

運(yùn)氣不錯(cuò)雏吭,通過真機(jī)調(diào)試發(fā)現(xiàn)my_smbc_get_auth_data_with_context_fn回調(diào)中的username打印日志確實(shí)為“guest”,并且smbc_open成功!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末陪踩,一起剝皮案震驚了整個(gè)濱河市杖们,隨后出現(xiàn)的幾起案子悉抵,更是在濱河造成了極大的恐慌,老刑警劉巖胀莹,帶你破解...
    沈念sama閱讀 212,454評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件基跑,死亡現(xiàn)場離奇詭異婚温,居然都是意外死亡描焰,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,553評論 3 385
  • 文/潘曉璐 我一進(jìn)店門栅螟,熙熙樓的掌柜王于貴愁眉苦臉地迎上來荆秦,“玉大人,你說我怎么就攤上這事力图〔匠瘢” “怎么了?”我有些...
    開封第一講書人閱讀 157,921評論 0 348
  • 文/不壞的土叔 我叫張陵吃媒,是天一觀的道長瓤介。 經(jīng)常有香客問我,道長赘那,這世上最難降的妖魔是什么刑桑? 我笑而不...
    開封第一講書人閱讀 56,648評論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮募舟,結(jié)果婚禮上祠斧,老公的妹妹穿的比我還像新娘。我一直安慰自己拱礁,他們只是感情好琢锋,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,770評論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著呢灶,像睡著了一般吴超。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上鸯乃,一...
    開封第一講書人閱讀 49,950評論 1 291
  • 那天鲸阻,我揣著相機(jī)與錄音,去河邊找鬼飒责。 笑死赘娄,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的宏蛉。 我是一名探鬼主播遣臼,決...
    沈念sama閱讀 39,090評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼拾并!你這毒婦竟也來了揍堰?” 一聲冷哼從身側(cè)響起鹏浅,我...
    開封第一講書人閱讀 37,817評論 0 268
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎屏歹,沒想到半個(gè)月后隐砸,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,275評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡蝙眶,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,592評論 2 327
  • 正文 我和宋清朗相戀三年季希,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片幽纷。...
    茶點(diǎn)故事閱讀 38,724評論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡式塌,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出友浸,到底是詐尸還是另有隱情峰尝,我是刑警寧澤,帶...
    沈念sama閱讀 34,409評論 4 333
  • 正文 年R本政府宣布收恢,位于F島的核電站武学,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏伦意。R本人自食惡果不足惜火窒,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,052評論 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望默赂。 院中可真熱鬧沛鸵,春花似錦、人聲如沸缆八。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,815評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽奈辰。三九已至栏妖,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間奖恰,已是汗流浹背吊趾。 一陣腳步聲響...
    開封第一講書人閱讀 32,043評論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留瑟啃,地道東北人论泛。 一個(gè)月前我還...
    沈念sama閱讀 46,503評論 2 361
  • 正文 我出身青樓,卻偏偏與公主長得像蛹屿,于是被迫代替她去往敵國和親屁奏。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,627評論 2 350

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