2025-04-30

前段時間對Android 的SDCard unmount 流程進行瞭幾篇簡短的分析,由於當時隻是紙上談兵,沒有實際上的跟進,可能會有一些誤導人或者小錯誤。今天重新梳理瞭頭緒,針對mount的流程再重新分析一次。
 本篇大綱
 
• android 系統如何開機啟動監聽mount服務
• 默認掛載點在Android 系統的哪個目錄
• vold.fstab 配置文件的分析 
• vold 裡面啟動頁面main做瞭些什麼
android 系統如何開機啟動監聽mount服務
android sdcard 熱插拔監測和執行操作是由一個啟動文件vold  所統領的,系統開機會讀取初始化配置文件init.rc,該文件位於比如我的板子是:device/ti/omap3evm/init.rc,具體根據自己平臺查找。裡面有一個是默認啟動vold 服務的代碼,如下:
service vold /system/bin/vold
    socket vold stream 0660 root mount
    ioprio be 2 

 如果要對該文件做出修改之類,要重新編一下boot.img 鏡像文件,燒錄進android 系統,之後可以在android的文件系統根目錄找到init.rc文件。上述代碼為啟動vold 啟動文件,也可以在init.rc 增加多一些我們想要的文件目錄,比如增加一個可以存放多分區掛載的目錄等,這個是後話。
 
 默認掛載點在Android 系統的哪個目錄
 usbdisk 或者 sdcard 熱插拔的時候,kernel 會發出命令執行mount或者unmount 操作,但這都是驅動級的。而mount 目錄會在android 的文件系統目錄下:/dev/block/vold 這個目錄由vold 生成,用來存放所有的usbdisk 或者 sdcard 的掛載點。代碼位於main裡面最優先執行:
 
mkdir("/dev/block/vold", 0755);
 
 可以根據這個目錄找到如下節點:
sh-4.1# ls /dev/block/vold/
179:0  179:1  8:0    8:1    8:2    8:3    8:4 

節點的小介紹:
0代表當前的整個設備,1代碼當前設備的分區名稱代號。
所以你會發現,sdcard隻有一個分區它卻生成瞭兩個如:179:0 179:1
而usbdisk 有四個分區,它會生成五個掛載點: 8:0    8:1    8:2    8:3    8:4  就是這個原因。
 
 
 vold.fstab 配置文件的分析
vold 裡面會通過指定文件來讀取預先配置好的sdcard或者多分區配置文件,該文件位於
/system/core/rootdir/etc/vold.fstab
如以下的配置文件為:
dev_mount sdcard /mnt/sdcard auto /devices/platform/goldfish_mmc.0 /devices/platform/msm_sdcc.2/mmc_host/mmc1
 
 dev_mount 代表掛載格式
 sdcard 代表掛載的標簽
/mnt/sdcard 代表掛載點
 auto 為自定義選項可以為任何,但必須在main 裡面自己判斷比如這裡的意思為自動掛載
後面兩個目錄為設備路徑,第一個如果被占用會選擇第二個
 
配置文件可以根據自己的需要編寫,並不是固定的,但最好遵循google vold 啟動文件代碼的格式編寫,要不然會給我們修改代碼或者增加多分區功能帶來不小的麻煩,如以下我自己編寫的多分區掛載支持vold.fstab 配置文件:
 
 dev_mount sdcard external /mnt/sdcard auto /devices/platform/mmci-omap-hs.0/mmc_host/mmc0 /devices/platform/mmci-omap-hs.0/mmc_host/mmc1
dev_mount usb1 external /mnt/usbdisk/usb1-disk%d all /devices/platform/ehci-omap.0/usb1/1-2/1-2.1/
dev_mount usb2 external /mnt/usbdisk/usb2-disk%d all /devices/platform/ehci-omap.0/usb1/1-2/1-2.2/
dev_mount usb3 external /mnt/usbdisk/usb3-disk%d all /devices/platform/ehci-omap.0/usb1/1-2/1-2.3/

 該文件修改後經系統編譯會在android 系統目錄裡/system/etc/vold.fstab找到。
 /devices/platform/ehci-omap.0/usb1/1-2/1-2.1/  代表要掛載的USB口。
vold.fstab 隻是一個單純的配置文件,具體的讀取和取數據還 是要靠main裡面的process_config函數。看代碼,裡面正有一段用來讀取配置文件:
  if (!(fp = fopen("/etc/vold.fstab", "r"))) {
        return -1;
    }

在這個函數裡面會根據讀取到的數據存放起來,然後滿足條件時執行操作。比如代碼裡面的:
 
if (!strcmp(type, "dev_mount")) {
            DirectVolume *dv = NULL;
            char *part;

            if (!(part = strtok_r(NULL, delim, &save_ptr))) {
                SLOGE("Error parsing partition");
                goto out_syntax;
            }
            if (strcmp(part, "auto") && atoi(part) == 0) {
                SLOGE("Partition must either be 'auto' or 1 based index instead of '%s'", part);
                goto out_syntax;
            }

            if (!strcmp(part, "auto")) {
                dv = new DirectVolume(vm, label, mount_point, -1);
            } else {
                dv = new DirectVolume(vm, label, mount_point, atoi(part));
            }

            while ((sysfs_path = strtok_r(NULL, delim, &save_ptr))) {
                if (*sysfs_path != '/') {
                    /* If the first character is not a '/', it must be flags */
                    break;
                }
                if (dv->addPath(sysfs_path)) {
                    SLOGE("Failed to add devpath %s to volume %s", sysfs_path,
                         label);
                    goto out_fail;
                }
            }

            /* If sysfs_path is non-null at this point, then it contains
             * the optional flags for this volume
             */
            if (sysfs_path)
                flags = parse_mount_flags(sysfs_path);
            else
                flags = 0;
            dv->setFlags(flags);

            vm->addVolume(dv);
        }

 
DirectVolume後面會講到,執行mount 和unmount 都是它在做。
另外,有時後讀取配置文件會有問題,這是因為它讀取是通過指標下標遞增的方式在讀,如果有問題可以跟蹤打印一下配置文件,看哪裡需要修改。
 
 vold 裡面啟動頁面main做瞭些什麼
main 主要是初始化socket 連接監聽數據變化,在系統起來時第一時間啟動,並且通過讀取配置文件來識別usb口或者sdcard 的設備地址,來mount 或者unmount 。其它執行mount 、 unmount  或者刪除節點等操作都是由上層或者framework 發送命令給main讓其通知volumeManage 執行相應的操作。

 

摘自  Terry_龍 

發佈留言

發佈留言必須填寫的電子郵件地址不會公開。 必填欄位標示為 *