把APK安装到SD卡和TF卡实现方案

本文介绍了如何将Android应用程序安装到SD卡或TF卡上,涉及Vold系统框架、安装流程、关键代码分析,以及需要修改的系统文件,包括Volume.cpp、DefaultContainerService.java和VolumeManager.cpp等。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

首先给大家分享一个巨牛巨牛的人工智能教程,是我无意中发现的。教程不仅零基础,通俗易懂,而且非常风趣幽默,还时不时有内涵段子,像看小说一样,哈哈~我正在学习中,觉得太牛了,所以分享给大家!点这里可以跳转到教程

               

1. 简介

    为了能把应用程序安装到SD卡和TF卡上,Android系统默认是不支持的,它只有一个asec mount点: /mnt/secure/asec,在我的系统中,此mount点由/mnt/sdcard给占用了,所以TF卡就支持不了。为了解决此问题,除了把代码读明白之外,无其它的办法。为了方便理解下面的描述,先看下Vold(管理外设热插拔)的系统框架图:


关于相关类图,引用其他仁兄的图:


2. 从U盘安装应用程序流程

PackageInstallerActivity.java::initiateInstall->
                             ::startInstallConfirm->
                   
InstallAppProgress::onCreate->
InstallAppProgress::initView->

PackageManager::installPackage->

PackageManagerService::installPackage->
 installPackageWithVerification->
       mHandler.sendMessage(msg)->

PackageHandler::handleMessage->
  PackageHandler::doHandleMessage->
  HandlerParams::startCopy->
     InstallParams::handleStartCopy-> (根据installLocation确认安装位置,只有针对/data分区有lowThreshold)
       SdInstallArgs::copyApk-> (关键的地方,下面详解)
     InstallParams::handleReturnCode->
       PackageManagerService::processPendingInstall->
         PackageManagerService::installPackageLI-> //args, res分析
           PackageManagerService::installNewPackageLI->  // or replacePackageLI
             PackageManagerService::scanPackageLI-> (KEY)->parsePackage(解析AndroidManifest.xml)
               Installer::install (mInstaller.install)-> (Socket)--->
               installd(真正做事的地方)

2.1 SdInstallArgs::copyApk

SdInstallArgs::copyApk->
DefaultContainerService::copyResourceInner->
PackageHelper::createSdDir->
MountService::createSecureContainer-> 

      (cmd = String.format(new Locale("en", "Us"),

       "asec create %s %d %s %s %d", id, sizeMb, fstype, key, ownerUid);)
NativeDaemonConnector::doCommandLocked->(向"vold" socket发送命令给vold,并等待结果)
NativeDaemonConnector::sendCommandLocked-> (Socket)

FrameworkListener::onDataAvailable-> (receive message)  "buffer=asec list"
                                  或"buffer=asec create smdl2tmp1 19 fat caf791d0426d682e5b2aafc5439d55a9 10023"
FrameworkListener::dispatchCommand->
CommandListener::AsecCmd::runCommand-> (所有与Volume::SEC_ASECDIR相关的代码都需要修改)
VolumeManager::createAsec (决定安装文件的路径Volume::SEC_ASECDIR)

2.2 复杂的mount关系

    详细代码参见Volume.cpp::

     代码默认不把sdcard && udisk mount为asec,为此需要把SD卡或TF卡mount到asec挂载点,当然系统默认只有一个/mnt/secure/asec,你可以在init.rc中创建一个/mnt/secure/asecsd或/mnt/secure/asectf,并执行mount即可。

• /mnt/secure: 下的mount点为物理外设的mount点

•/mnt/asec: 为公共挂载点,它是一个tmpfs文件系统的挂载点,安装到SD或TF上的应用程序都可以被mount到此目录下的mount点

   复杂的mount关系如下所示:

   1) /dev/block/vold/31:9    mount to /mnt/secure/staging
   2) /mnt/secure/staging/.android_secure mount to /mnt/secure/asec (MS_BIND)
   3) tmpfs  mount to /mnt/secure/staging/.android_secure (只读0字节tmpfs,把它隐藏起来, obscuring the asec image directory from non root users)
   4) 把/mnt/secure/staging下的子目录(.android_secure)移动到/mnt/sdcard下

   注:MS_BIND: 执行bind挂载,使文件或者子目录树在文件系统内的另一个点上可视。

   搞这么复杂的目的只有一个:只有root用

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值