日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

从源码角度解析Android中APK安装过程

發布時間:2025/1/21 Android 77 豆豆
生活随笔 收集整理的這篇文章主要介紹了 从源码角度解析Android中APK安装过程 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

從源碼角度解析Android中APK的安裝過程

1. Android中APK簡介

Android應用Apk的安裝有如下四種方式:

1.1 系統應用安裝

沒有安裝界面,在開機時自動完成

1.2 網絡下載應用安裝

沒有安裝界面,在應用市場完成

1.3 ADB命令安裝

沒有安裝界面,通過命令直接安裝

1.4 外部設備安裝

有安裝界面,通過SD卡等外部設備安裝,由packageInstaller處理安裝邏輯

2. APK安裝涉及到的幾個常用目錄

  • system/app : 系統自帶的應用程序,獲得root權限才能刪除

  • data/app : 用戶程序安裝目錄,安裝時會把apk文件復制到此目錄下

  • data/data : 存放應用程序的數據

  • data/dalvik-cache : 將apk中的dex文件安裝到該目錄下(dex文件是dalvik虛擬機的可執行文件,大小約為原始apk的四分之一)

3. APK安裝的四大步驟

(1)拷貝apk到指定的目錄

默認情況下,用戶安裝的apk首先會拷貝到/data/app下,用戶有訪問/data/app目錄的權限,但系統出廠的apk文件 會被放到/system分區下,包括/system/app,/system/vendor/app,以及/system/priv-app等。該分區需要 root權限的用戶才能訪問。

(2)加壓apk、拷貝文件、創建應用的數據目錄

為了加快APP的啟動速度,apk在安裝的時候,會首先將APP的可執行文件(dex)拷貝到/data/dalvik-cache目錄 下,緩存起來。再在/data/data/目錄下創建應用程序的數據目錄(以應用包名命令),用來存放應用的數據庫、xml 文件、cache、二進制的so動態庫等。

(3)解析apk的AndroidManifest.xml文件

在安裝apk的過程中,會解析apk的AndroidManifest.xml文件,將apk的權限、應用包名、apk的安裝位置、版本、 userID等重要信息保存在/data/system/packages.xml文件中。這些操作都是在PackageManagerService中完成 的。

(4)顯示icon圖標

應用程序經過PMS中的邏輯處理后,相當于已經注冊好了,如果想要在Android桌面上看到icon圖標,則需要 Launcher將系統中已經安裝的程序展現在桌面上。

4. APK安裝的預備知識點

(1)PackageManagerService是由SystemServer啟動,PMS負責應用的安裝、卸載、權限檢查等工作

(2)在/system/app和/data/app目錄下的apk文件,PMS在啟動過程中,都會掃描安裝

(3)每次開機時,PMS都會在構造函數中對指定目錄下的apk進行掃描,沒有安裝的apk就會觸發安裝。

(4) 本文的源碼是基于Android6.0

5. 系統應用安裝

在創建PackageManagerService實例時,會在PMS的構造函數中開始執行安裝應用程序的邏輯。在PMS的構造函數中做了如下幾點重要操作。

(1)創建Settings對象,添加shareUserId

mSettings = new Settings(mPackages);mSettings.addSharedUserLPw("android.uid.system", Process.SYSTEM_UID,ApplicationInfo.FLAG_SYSTEM, ApplicationInfo.PRIVATE_FLAG_PRIVILEGED);mSettings.addSharedUserLPw("android.uid.phone", RADIO_UID,ApplicationInfo.FLAG_SYSTEM, ApplicationInfo.PRIVATE_FLAG_PRIVILEGED);mSettings.addSharedUserLPw("android.uid.log", LOG_UID,ApplicationInfo.FLAG_SYSTEM, ApplicationInfo.PRIVATE_FLAG_PRIVILEGED);mSettings.addSharedUserLPw("android.uid.nfc", NFC_UID,ApplicationInfo.FLAG_SYSTEM, ApplicationInfo.PRIVATE_FLAG_PRIVILEGED);mSettings.addSharedUserLPw("android.uid.bluetooth", BLUETOOTH_UID,ApplicationInfo.FLAG_SYSTEM, ApplicationInfo.PRIVATE_FLAG_PRIVILEGED);mSettings.addSharedUserLPw("android.uid.shell", SHELL_UID,ApplicationInfo.FLAG_SYSTEM, ApplicationInfo.PRIVATE_FLAG_PRIVILEGED);

(2)通過PackageManagerService構造函數參數獲取應用安裝器Installer

mInstaller = installer;

(3)獲取SystemConfig實例,讀取“/system/etc/permissions/*.xml”資源文件

從資源文件中獲取mGlobalsGids(Group-ids)、mSystemPermissions(系統權限)、mAvailableFeatures(系統支持的features)屬性。

SystemConfig systemConfig = SystemConfig.getInstance();mGlobalGids = systemConfig.getGlobalGids();mSystemPermissions = systemConfig.getSystemPermissions();mAvailableFeatures = systemConfig.getAvailableFeatures();

(4)創建系統消息處理線程

mHandlerThread = new ServiceThread(TAG,Process.THREAD_PRIORITY_BACKGROUND, true /*allowIo*/);mHandlerThread.start();mHandler = new PackageHandler(mHandlerThread.getLooper());Watchdog.getInstance().addThread(mHandler, WATCHDOG_TIMEOUT);

(5)執行com.android.server.pm.Settings中的readLPw方法,讀取安裝包中的信息,并解析成對應的數據結構。

File dataDir = Environment.getDataDirectory();mAppDataDir = new File(dataDir, "data");mAppInstallDir = new File(dataDir, "app");mAppLib32InstallDir = new File(dataDir, "app-lib");mAsecInternalPath = new File(dataDir, "app-asec").getPath();mUserAppDataDir = new File(dataDir, "user");mDrmAppPrivateInstallDir = new File(dataDir, "app-private");sUserManager = new UserManagerService(context, this,mInstallLock, mPackages);// Propagate permission configuration in to package manager.ArrayMap<String, SystemConfig.PermissionEntry> permConfig= systemConfig.getPermissions();for (int i=0; i<permConfig.size(); i++) {SystemConfig.PermissionEntry perm = permConfig.valueAt(i);BasePermission bp = mSettings.mPermissions.get(perm.name);if (bp == null) {bp = new BasePermission(perm.name, "android", BasePermission.TYPE_BUILTIN);mSettings.mPermissions.put(perm.name, bp);}if (perm.gids != null) {bp.setGids(perm.gids, perm.perUser);}}ArrayMap<String, String> libConfig = systemConfig.getSharedLibraries();for (int i=0; i<libConfig.size(); i++) {mSharedLibraries.put(libConfig.keyAt(i),new SharedLibraryEntry(libConfig.valueAt(i), null));}mFoundPolicyFile = SELinuxMMAC.readInstallPolicy();mRestoredSettings = mSettings.readLPw(this, sUserManager.getUsers(false),mSdkVersion, mOnlyCore);

其中,讀取的重要文件有如下幾個:

  • packages.xml:記錄了系統中所有安裝應用的信息,包括基本信息、簽名和權限。

  • packages-backup.xml:packages.xml的備份文件

  • packages.list:保存普通應用的數據目錄和uid等信息

  • packages-stopped.xml:記錄系統中被強制停止運行應用的信息。系統在強制停止某個應用時,會將應用的信息記錄到該文件中

  • packages-stopped-backup.xml:packages-stopped.xml的備份文件

這幾個目錄在創建Settings對象時就已經被封裝成了對應的File文件。

(6)執行PMS中的scanDirLI方法掃描系統安裝目錄和非系統apk信息

File vendorOverlayDir = new File(VENDOR_OVERLAY_DIR);scanDirLI(vendorOverlayDir, PackageParser.PARSE_IS_SYSTEM| PackageParser.PARSE_IS_SYSTEM_DIR, scanFlags | SCAN_TRUSTED_OVERLAY, 0);// Find base frameworks (resource packages without code).scanDirLI(frameworkDir, PackageParser.PARSE_IS_SYSTEM| PackageParser.PARSE_IS_SYSTEM_DIR| PackageParser.PARSE_IS_PRIVILEGED,scanFlags | SCAN_NO_DEX, 0);// Collected privileged system packages.final File privilegedAppDir = new File(Environment.getRootDirectory(), "priv-app");scanDirLI(privilegedAppDir, PackageParser.PARSE_IS_SYSTEM| PackageParser.PARSE_IS_SYSTEM_DIR| PackageParser.PARSE_IS_PRIVILEGED, scanFlags, 0);// Collect ordinary system packages.final File systemAppDir = new File(Environment.getRootDirectory(), "app");scanDirLI(systemAppDir, PackageParser.PARSE_IS_SYSTEM| PackageParser.PARSE_IS_SYSTEM_DIR, scanFlags, 0);// Collect all vendor packages.File vendorAppDir = new File("/vendor/app");try {vendorAppDir = vendorAppDir.getCanonicalFile();} catch (IOException e) {// failed to look up canonical path, continue with original one}scanDirLI(vendorAppDir, PackageParser.PARSE_IS_SYSTEM| PackageParser.PARSE_IS_SYSTEM_DIR, scanFlags, 0);// Collect all OEM packages.final File oemAppDir = new File(Environment.getOemDirectory(), "app");scanDirLI(oemAppDir, PackageParser.PARSE_IS_SYSTEM| PackageParser.PARSE_IS_SYSTEM_DIR, scanFlags, 0);

其中,系統安裝目錄有:

  • /system/framework 系統庫

  • /system/app 默認的系統應用

  • /vendor/app 廠商定制的應用

非系統apk信息目錄:

  • /data/app/

  • /system/priv-app/

  • /data/app-private/

到此為止,PMS構造函數中主要的邏輯操作就介紹完了。

繼續探究掃描安裝過程:

(1)深入到PackageManagerService—>scanDirLI方法中

private void scanDirLI(File dir, int parseFlags, int scanFlags, long currentTime) {final File[] files = dir.listFiles();if (ArrayUtils.isEmpty(files)) {Log.d(TAG, "No files in app dir " + dir);return;}if (DEBUG_PACKAGE_SCANNING) {Log.d(TAG, "Scanning app dir " + dir + " scanFlags=" + scanFlags+ " flags=0x" + Integer.toHexString(parseFlags));}for (File file : files) {final boolean isPackage = (isApkFile(file) || file.isDirectory())&& !PackageInstallerService.isStageName(file.getName());if (!isPackage) {// Ignore entries which are not packagescontinue;}try {scanPackageLI(file, parseFlags | PackageParser.PARSE_MUST_BE_APK,scanFlags, currentTime, null);//注釋1} catch (PackageManagerException e) {Slog.w(TAG, "Failed to parse " + file + ": " + e.getMessage());// Delete invalid userdata appsif ((parseFlags & PackageParser.PARSE_IS_SYSTEM) == 0 &&e.error == PackageManager.INSTALL_FAILED_INVALID_APK) {logCriticalInfo(Log.WARN, "Deleting invalid package at " + file);if (file.isDirectory()) {mInstaller.rmPackageDir(file.getAbsolutePath());} else {file.delete();}}}} }

注釋1處,對于目錄中的每一個文件,如果是已apk作為后綴名的,就會調用PackageManagerService—>scanPackageLI方法進行掃描解析

(2)深入到PackageManagerService—>scanPackageLI方法中

private PackageParser.Package scanPackageLI(File scanFile, int parseFlags, int scanFlags,long currentTime, UserHandle user) throws PackageManagerException {if (DEBUG_INSTALL) Slog.d(TAG, "Parsing: " + scanFile);parseFlags |= mDefParseFlags;PackageParser pp = new PackageParser();//注釋1pp.setSeparateProcesses(mSeparateProcesses);pp.setOnlyCoreApps(mOnlyCore);pp.setDisplayMetrics(mMetrics);if ((scanFlags & SCAN_TRUSTED_OVERLAY) != 0) {parseFlags |= PackageParser.PARSE_TRUSTED_OVERLAY;}final PackageParser.Package pkg;try {pkg = pp.parsePackage(scanFile, parseFlags);//注釋2} catch (PackageParserException e) {throw PackageManagerException.from(e);}......省略...... // readersynchronized (mPackages) {//注釋3// Look to see if we already know about this package.String oldName = mSettings.mRenamedPackages.get(pkg.packageName);if (pkg.mOriginalPackages != null && pkg.mOriginalPackages.contains(oldName)) {// This package has been renamed to its original name. Let's// use that.ps = mSettings.peekPackageLPr(oldName);}// If there was no original package, see one for the real package name.if (ps == null) {ps = mSettings.peekPackageLPr(pkg.packageName);}// Check to see if this package could be hiding/updating a system// package. Must look for it either under the original or real// package name depending on our state.updatedPkg = mSettings.getDisabledSystemPkgLPr(ps != null ? ps.name : pkg.packageName);if (DEBUG_INSTALL && updatedPkg != null) Slog.d(TAG, "updatedPkg = " + updatedPkg);}......省略...... PackageParser.Package scannedPkg = scanPackageLI(pkg, parseFlags, scanFlags| SCAN_UPDATE_SIGNATURE, currentTime, user);//注釋4/** If the system app should be overridden by a previously installed* data, hide the system app now and let the /data/app scan pick it up* again.*/if (shouldHideSystemApp) {synchronized (mPackages) {mSettings.disableSystemPackageLPw(pkg.packageName);}}return scannedPkg; }

注釋1處:創建一個PackageParser實例。

注釋2處調用PackageParser的parsePackage函數來對apk安裝包的AndroidManifest.xml文件掃描和提取證書信息,然后構建一個PackageParser.Package對象,并將其返回。

注釋3將解析返回的PackageParser對象中的信息保存到PMS中。

注釋4調用另一個重載的scanPackageLI方法來構建一個PackageSetting對象,這個對象保存的信息最后會通過writeLPr寫入到/data/system/packages.xml文件中去。

(3)根據(2)中注釋2深入到PackageParser—>parsePackage方法中

public Package parsePackage(File packageFile, int flags) throws PackageParserException {if (packageFile.isDirectory()) {return parseClusterPackage(packageFile, flags);} else {return parseMonolithicPackage(packageFile, flags);} }

這里根據packageFile是否是目錄分別調用parseClusterPackage和parseMonolithicPackage去解析。

其中parseClusterPackage方法如下:

private Package parseClusterPackage(File packageDir, int flags) throws PackageParserException {final PackageLite lite = parseClusterPackageLite(packageDir, 0);if (mOnlyCoreApps && !lite.coreApp) {throw new PackageParserException(INSTALL_PARSE_FAILED_MANIFEST_MALFORMED,"Not a coreApp: " + packageDir);}final AssetManager assets = new AssetManager();try {...省略...final File baseApk = new File(lite.baseCodePath);final Package pkg = parseBaseApk(baseApk, assets, flags);//注釋1if (pkg == null) {throw new PackageParserException(INSTALL_PARSE_FAILED_NOT_APK,"Failed to parse base APK: " + baseApk);}...省略...pkg.codePath = packageDir.getAbsolutePath();return pkg;} finally {IoUtils.closeQuietly(assets);} }

parseMonolithicPackage方法如下:

public Package parseMonolithicPackage(File apkFile, int flags) throws PackageParserException {...省略...final AssetManager assets = new AssetManager();try {final Package pkg = parseBaseApk(apkFile, assets, flags);//注釋2pkg.codePath = apkFile.getAbsolutePath();return pkg;} finally {IoUtils.closeQuietly(assets);} }

從parseClusterPackage的注釋1和parseMonolithicPackage的注釋2可以看出都是調用parseBaseApk去解析。

繼續探究parseBaseApk方法:

private Package parseBaseApk(File apkFile, AssetManager assets, int flags)throws PackageParserException {...省略...try {res = new Resources(assets, mMetrics, null);assets.setConfiguration(0, 0, null, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,Build.VERSION.RESOURCES_SDK_INT);parser = assets.openXmlResourceParser(cookie, ANDROID_MANIFEST_FILENAME);final String[] outError = new String[1];final Package pkg = parseBaseApk(res, parser, flags, outError);//注釋1if (pkg == null) {throw new PackageParserException(mParseError,apkPath + " (at " + parser.getPositionDescription() + "): " + outError[0]);}...省略...} catch (PackageParserException e) {throw e;} catch (Exception e) {throw new PackageParserException(INSTALL_PARSE_FAILED_UNEXPECTED_EXCEPTION,"Failed to read manifest from " + apkPath, e);} finally {IoUtils.closeQuietly(parser);} }

注釋1處調用另一個重載的parseBaseApk方法對apk進行解析。

(4)根據(2)中注釋4深入到PackageManagerService—>另一個scanPackageLI方法中

private PackageParser.Package scanPackageLI(PackageParser.Package pkg, int parseFlags,int scanFlags, long currentTime, UserHandle user) throws PackageManagerException {boolean success = false;try {final PackageParser.Package res = scanPackageDirtyLI(pkg, parseFlags, scanFlags,currentTime, user);//注釋1success = true;return res;} finally {if (!success && (scanFlags & SCAN_DELETE_DATA_ON_FAILURES) != 0) {removeDataDirsLI(pkg.volumeUuid, pkg.packageName);}} }

由注釋1繼續深入PackageManagerService—>scanPackageDirtyLI方法中:

private PackageParser.Package scanPackageDirtyLI(PackageParser.Package pkg, int parseFlags,int scanFlags, long currentTime, UserHandle user) throws PackageManagerException {final File scanFile = new File(pkg.codePath);......省略......// And now re-install the app.ret = createDataDirsLI(pkg.volumeUuid, pkgName, pkg.applicationInfo.uid,HasSystemUidErrors = true;//注釋1......省略......//invoke installer to do the actual installationint ret = createDataDirsLI(pkg.volumeUuid, pkgName, pkg.applicationInfo.uid,pkg.applicationInfo.seinfo);//注釋2if (ret < 0) {// Error from installerthrow new PackageManagerException(INSTALL_FAILED_INSUFFICIENT_STORAGE,"Unable to create data dirs [errorCode=" + ret + "]");}......省略...... }

從注釋1和注釋2處看出調用了PMS中的createDataDirsLI方法,給installed發送消息,為應用程序創建對應的數據目錄,如果目錄已經存在,也會重新創建一遍。

深入到PMS的createDataDirsLI方法中:

private int createDataDirsLI(String volumeUuid, String packageName, int uid, String seinfo) {int[] users = sUserManager.getUserIds();int res = mInstaller.install(volumeUuid, packageName, uid, uid, seinfo);//注釋1if (res < 0) {return res;}for (int user : users) {if (user != 0) {res = mInstaller.createUserData(volumeUuid, packageName,//注釋2UserHandle.getUid(user, uid), user, seinfo);if (res < 0) {return res;}}}return res; }

從上面可知調用mInstaller.install()函數完成APK安裝。

到此為止,系統應用的安裝流程差不多就完成了。

6. 網絡下載應用安裝

網絡應用下載完成后,會自動調用PackageManagerService中的installPackage方法:

@Override public void installPackage(String originPath, IPackageInstallObserver2 observer,int installFlags, String installerPackageName, VerificationParams verificationParams,String packageAbiOverride) {installPackageAsUser(originPath, observer, installFlags, installerPackageName,verificationParams, packageAbiOverride, UserHandle.getCallingUserId()); }

可以看到主要是調用了PMS—>installPackageAsUser方法:

public void installPackageAsUser(String originPath, IPackageInstallObserver2 observer,int installFlags, String installerPackageName, VerificationParams verificationParams,String packageAbiOverride, int userId) {mContext.enforceCallingOrSelfPermission(android.Manifest.permission.INSTALL_PACKAGES, null);......省略......final Message msg = mHandler.obtainMessage(INIT_COPY);msg.obj = new InstallParams(origin, null, observer, params.installFlags,installerPackageName, params.volumeUuid, verifParams, user, params.abiOverride,params.grantedRuntimePermissions);mHandler.sendMessage(msg); }

主要是獲取用戶安裝位置,將InstallParams對象封裝在Message里,然后發一個Handler消息。

深入到處理INIT_COPY的地方:PMS—>doHandleMessage方法中:

void doHandleMessage(Message msg) {switch (msg.what) {case INIT_COPY: {......省略...... if (!mBound) {// If this is the only one pending we might// have to bind to the service again.if (!connectToService()) {Slog.e(TAG, "Failed to bind to media container service");params.serviceError();return;} else {// Once we bind to the service, the first// pending request will be processed.mPendingInstalls.add(idx, params);}} else {mPendingInstalls.add(idx, params);// Already bound to the service. Just make// sure we trigger off processing the first request.if (idx == 0) {mHandler.sendEmptyMessage(MCS_BOUND);}}break;}case MCS_BOUND: {if (DEBUG_INSTALL) Slog.i(TAG, "mcs_bound");if (msg.obj != null) {mContainerService = (IMediaContainerService) msg.obj;}if (mContainerService == null) {......省略......} else if (mPendingInstalls.size() > 0) {HandlerParams params = mPendingInstalls.get(0);if (params != null) {if (params.startCopy()) {//注釋1// We are done... look for more work or to// go idle.if (DEBUG_SD_INSTALL) Log.i(TAG,"Checking for more work or unbind...");// Delete pending installif (mPendingInstalls.size() > 0) {mPendingInstalls.remove(0);}if (mPendingInstalls.size() == 0) {if (mBound) {if (DEBUG_SD_INSTALL) Log.i(TAG,"Posting delayed MCS_UNBIND");removeMessages(MCS_UNBIND);Message ubmsg = obtainMessage(MCS_UNBIND);// Unbind after a little delay, to avoid// continual thrashing.sendMessageDelayed(ubmsg, 10000);}} else {// There are more pending requests in queue.// Just post MCS_BOUND message to trigger processing// of next pending install.if (DEBUG_SD_INSTALL) Log.i(TAG,"Posting MCS_BOUND for next work");mHandler.sendEmptyMessage(MCS_BOUND);}}}} else {// Should never happen ideally.Slog.w(TAG, "Empty queue");}break;}

如果msg.what是INIT_COPY:

則連接DefaultContainerService服務,將我們要安裝的信息方法HandlerParams的mPendingInstalls中,然后再發送MCS_BOUND消息。

如果msg.what是MCS_BOUND:

則通過HandlerParams params = mPendingInstalls.get(0)獲取要安裝包的信息,然后清除包信息,如果還有其他包,則繼續發MCS_BOUND消息,一直循環,直到安裝包都安裝完。

然后再調用PackageManagerService.HandlerParams—>startCopy方法:

final boolean startCopy() {boolean res;try {if (DEBUG_INSTALL) Slog.i(TAG, "startCopy " + mUser + ": " + this);if (++mRetries > MAX_RETRIES) {Slog.w(TAG, "Failed to invoke remote methods on default container service. Giving up");mHandler.sendEmptyMessage(MCS_GIVE_UP);handleServiceError();return false;} else {handleStartCopy();//注釋1res = true;}} catch (RemoteException e) {if (DEBUG_INSTALL) Slog.i(TAG, "Posting install MCS_RECONNECT");mHandler.sendEmptyMessage(MCS_RECONNECT);res = false;}handleReturnCode();//注釋2return res;}

其中startCopy中有兩個重要的方法handleStartCopy和handleReturnCode。

handleStartCopy中會檢查應用是否能安裝成功,如果不能安裝成功,則會返回failed的CODE;返回res標識,判斷是否安裝成功。

handleReturnCode方法有兩處重載了,一個是刪除時調用的,一個是安裝時調用的,下面列出安裝的地方:

@Overridevoid handleReturnCode() {// If mArgs is null, then MCS couldn't be reached. When it// reconnects, it will try again to install. At that point, this// will succeed.if (mArgs != null) {processPendingInstall(mArgs, mRet);}}

可以看到調用了processPendingInstall方法:

private void processPendingInstall(final InstallArgs args, final int currentStatus) {// Queue up an async operation since the package installation may take a little while.mHandler.post(new Runnable() {public void run() {mHandler.removeCallbacks(this);// Result object to be returnedPackageInstalledInfo res = new PackageInstalledInfo();res.returnCode = currentStatus;res.uid = -1;res.pkg = null;res.removedInfo = new PackageRemovedInfo();if (res.returnCode == PackageManager.INSTALL_SUCCEEDED) {args.doPreInstall(res.returnCode);synchronized (mInstallLock) {installPackageLI(args, res);//注釋1}args.doPostInstall(res.returnCode, res.uid);}......省略...... }

從注釋1處可以看到,調用的是installPackageLI方法:

private void installPackageLI(InstallArgs args, PackageInstalledInfo res) {......省略......if (replace) {replacePackageLI(pkg, parseFlags, scanFlags | SCAN_REPLACING, args.user,installerPackageName, volumeUuid, res);} else {installNewPackageLI(pkg, parseFlags, scanFlags | SCAN_DELETE_DATA_ON_FAILURES,args.user, installerPackageName, volumeUuid, res);}......省略...... }

如果是替換安裝,則走replacePackageLI的邏輯。追蹤replacePackageLI的邏輯流程:

  • replacePackageLI—>如果是系統應用則調用replaceSystemPackageLI—>再調用PackageParser.Package的scanPackageLI方法。

  • replacePackageLI—>如果是非系統應用則調用replaceNonSystemPackageLI—>再調用PackageParser.Package的scanPackageLI方法。

如果是第一次安裝,則走installNewPackageLI。追蹤installNewPackageLI的邏輯流程:

  • installNewPackageLI—>再調用PackageParser.Package的scanPackageLI方法。

到了調用PackageParser.Package的scanPackageLI方法就到了系統應用安裝過程中的“繼續探究掃描安裝過程”邏輯,后面的安裝邏輯就與系統應用安裝的邏輯一樣了。

7. 通過命令安裝應用

命令安裝應用的入口是:

framework/base/cmds/pm/src/com/android/commands/pm/Pm.java

應用安裝的時候就會調用runInstall()方法:

private int runInstall() {......省略......try {VerificationParams verificationParams = new VerificationParams(verificationURI,originatingURI, referrerURI, VerificationParams.NO_UID, null);mPm.installPackageAsUser(apkFilePath, obs.getBinder(), installFlags,installerPackageName, verificationParams, abi, userId);//注釋1......省略...... } catch (RemoteException e) {System.err.println(e.toString());System.err.println(PM_NOT_RUNNING_ERR);return 1;} }

從注釋1處可以看到安裝邏輯調用的是PackageManagerService—>installPackageAsUser方法。后面的調用邏輯就與“網絡下載應用安裝”的邏輯一樣了。

8. 通過外部設備安裝應用

通過外部設備安裝應用,調用的是Android內部的PackageInstaller去完成的,其本身也是一個apk。代碼位置在

/packages/apps/PackageInstaller/

用于顯示安裝apk,但其最終的本質還是調用PackageManagerService去完成安裝的。

當點擊文件管理器中的apk時,文件管理器就會啟動PackageInstaller中的PackageInstallerActivity,并將apk的信息通過intent傳遞給PackageInstallerActivity。

(1)深入到PackageInstallerActivity—>onCreate方法:

@Override protected void onCreate(Bundle icicle) {super.onCreate(icicle);mPm = getPackageManager();mInstaller = mPm.getPackageInstaller();//注釋1mUserManager = (UserManager) getSystemService(Context.USER_SERVICE);final Intent intent = getIntent();//注釋2if (PackageInstaller.ACTION_CONFIRM_PERMISSIONS.equals(intent.getAction())) {final int sessionId = intent.getIntExtra(PackageInstaller.EXTRA_SESSION_ID, -1);final PackageInstaller.SessionInfo info = mInstaller.getSessionInfo(sessionId);if (info == null || !info.sealed || info.resolvedBaseCodePath == null) {Log.w(TAG, "Session " + mSessionId + " in funky state; ignoring");finish();return;}mSessionId = sessionId;mPackageURI = Uri.fromFile(new File(info.resolvedBaseCodePath));mOriginatingURI = null;mReferrerURI = null;} else {mSessionId = -1;mPackageURI = intent.getData();mOriginatingURI = intent.getParcelableExtra(Intent.EXTRA_ORIGINATING_URI);mReferrerURI = intent.getParcelableExtra(Intent.EXTRA_REFERRER);}final boolean unknownSourcesAllowedByAdmin = isUnknownSourcesAllowedByAdmin();final boolean unknownSourcesAllowedByUser = isUnknownSourcesEnabled();//注釋3boolean requestFromUnknownSource = isInstallRequestFromUnknownSource(intent);......省略.....// Block the install attempt on the Unknown Sources setting if necessary.if (!requestFromUnknownSource) {initiateInstall();//注釋4return;}......省略.....if (!unknownSourcesAllowedByAdmin|| (!unknownSourcesAllowedByUser && isManagedProfile)) {showDialogInner(DLG_ADMIN_RESTRICTS_UNKNOWN_SOURCES);mInstallFlowAnalytics.setFlowFinished(InstallFlowAnalytics.RESULT_BLOCKED_BY_UNKNOWN_SOURCES_SETTING);} else if (!unknownSourcesAllowedByUser) {// Ask user to enable setting firstshowDialogInner(DLG_UNKNOWN_SOURCES);mInstallFlowAnalytics.setFlowFinished(InstallFlowAnalytics.RESULT_BLOCKED_BY_UNKNOWN_SOURCES_SETTING);} else {initiateInstall();//注釋5} }

注釋1:獲取PackageInstaller對象

注釋2:PackageInstaller檢查權限

注釋3:PackageInstaller檢查是否開啟未知來源

注釋4、注釋5:調用initiateInstall方法

(2)深入到PackageInstallerActivity—>initiateInstall方法

private void initiateInstall() {String pkgName = mPkgInfo.packageName;// Check if there is already a package on the device with this name// but it has been renamed to something else.String[] oldName = mPm.canonicalToCurrentPackageNames(new String[] { pkgName });if (oldName != null && oldName.length > 0 && oldName[0] != null) {pkgName = oldName[0];mPkgInfo.packageName = pkgName;mPkgInfo.applicationInfo.packageName = pkgName;}// Check if package is already installed. display confirmation dialog if replacing pkgtry {// This is a little convoluted because we want to get all uninstalled// apps, but this may include apps with just data, and if it is just// data we still want to count it as "installed".mAppInfo = mPm.getApplicationInfo(pkgName,PackageManager.GET_UNINSTALLED_PACKAGES);if ((mAppInfo.flags&ApplicationInfo.FLAG_INSTALLED) == 0) {mAppInfo = null;}} catch (NameNotFoundException e) {mAppInfo = null;}mInstallFlowAnalytics.setReplace(mAppInfo != null);mInstallFlowAnalytics.setSystemApp((mAppInfo != null) && ((mAppInfo.flags & ApplicationInfo.FLAG_SYSTEM) != 0));startInstallConfirm();//調用 }

initiateInstall方法主要負責檢查是否已經安裝過,是否是系統應用等。然后繼續調用了startInstallConfirm方法。

(3) 深入到PackageInstallerActivity—>startInstallConfirm方法

private void startInstallConfirm() {TabHost tabHost = (TabHost)findViewById(android.R.id.tabhost);tabHost.setup();ViewPager viewPager = (ViewPager)findViewById(R.id.pager);TabsAdapter adapter = new TabsAdapter(this, tabHost, viewPager);adapter.setOnTabChangedListener(new TabHost.OnTabChangeListener() {@Overridepublic void onTabChanged(String tabId) {if (TAB_ID_ALL.equals(tabId)) {mInstallFlowAnalytics.setAllPermissionsDisplayed(true);} else if (TAB_ID_NEW.equals(tabId)) {mInstallFlowAnalytics.setNewPermissionsDisplayed(true);}}});// If the app supports runtime permissions the new permissions will// be requested at runtime, hence we do not show them at install.boolean supportsRuntimePermissions = mPkgInfo.applicationInfo.targetSdkVersion>= Build.VERSION_CODES.M;boolean permVisible = false;mScrollView = null;mOkCanInstall = false;int msg = 0;AppSecurityPermissions perms = new AppSecurityPermissions(this, mPkgInfo);final int N = perms.getPermissionCount(AppSecurityPermissions.WHICH_ALL);if (mAppInfo != null) {msg = (mAppInfo.flags & ApplicationInfo.FLAG_SYSTEM) != 0? R.string.install_confirm_question_update_system: R.string.install_confirm_question_update;mScrollView = new CaffeinatedScrollView(this);mScrollView.setFillViewport(true);boolean newPermissionsFound = false;if (!supportsRuntimePermissions) {newPermissionsFound =(perms.getPermissionCount(AppSecurityPermissions.WHICH_NEW) > 0);mInstallFlowAnalytics.setNewPermissionsFound(newPermissionsFound);if (newPermissionsFound) {permVisible = true;mScrollView.addView(perms.getPermissionsView(AppSecurityPermissions.WHICH_NEW));}}if (!supportsRuntimePermissions && !newPermissionsFound) {LayoutInflater inflater = (LayoutInflater)getSystemService(Context.LAYOUT_INFLATER_SERVICE);TextView label = (TextView)inflater.inflate(R.layout.label, null);label.setText(R.string.no_new_perms);mScrollView.addView(label);}adapter.addTab(tabHost.newTabSpec(TAB_ID_NEW).setIndicator(getText(R.string.newPerms)), mScrollView);} else {findViewById(R.id.tabscontainer).setVisibility(View.GONE);findViewById(R.id.divider).setVisibility(View.VISIBLE);}if (!supportsRuntimePermissions && N > 0) {permVisible = true;LayoutInflater inflater = (LayoutInflater)getSystemService(Context.LAYOUT_INFLATER_SERVICE);View root = inflater.inflate(R.layout.permissions_list, null);if (mScrollView == null) {mScrollView = (CaffeinatedScrollView)root.findViewById(R.id.scrollview);}((ViewGroup)root.findViewById(R.id.permission_list)).addView(perms.getPermissionsView(AppSecurityPermissions.WHICH_ALL));adapter.addTab(tabHost.newTabSpec(TAB_ID_ALL).setIndicator(getText(R.string.allPerms)), root);}mInstallFlowAnalytics.setPermissionsDisplayed(permVisible);if (!permVisible) {if (mAppInfo != null) {// This is an update to an application, but there are no// permissions at all.msg = (mAppInfo.flags & ApplicationInfo.FLAG_SYSTEM) != 0? R.string.install_confirm_question_update_system_no_perms: R.string.install_confirm_question_update_no_perms;} else {// This is a new application with no permissions.msg = R.string.install_confirm_question_no_perms;}tabHost.setVisibility(View.GONE);mInstallFlowAnalytics.setAllPermissionsDisplayed(false);mInstallFlowAnalytics.setNewPermissionsDisplayed(false);findViewById(R.id.filler).setVisibility(View.VISIBLE);findViewById(R.id.divider).setVisibility(View.GONE);mScrollView = null;}if (msg != 0) {((TextView)findViewById(R.id.install_confirm_question)).setText(msg);}mInstallConfirm.setVisibility(View.VISIBLE);mOk = (Button)findViewById(R.id.ok_button);mOk.requestFocus();mCancel = (Button)findViewById(R.id.cancel_button);mOk.setOnClickListener(this);mCancel.setOnClickListener(this); //add by wangqi begin{@TabWidget tabWidget = tabHost.getTabWidget();int childCount = tabWidget.getChildCount();if (childCount == 2) {final View left = tabWidget.getChildAt(0);final View right = tabWidget.getChildAt(1);left.setId(tabWidget.getId() + 1);right.setId(tabWidget.getId() + 2);right.setNextFocusLeftId(left.getId());right.setOnKeyListener(new View.OnKeyListener() {@Overridepublic boolean onKey(View v, int keyCode, KeyEvent event) {if (keyCode == KeyEvent.KEYCODE_DPAD_LEFT && event.getAction() == KeyEvent.ACTION_DOWN) {left.requestFocus();}return true;}});} // end @}if (mScrollView == null) {// There is nothing to scroll view, so the ok button is immediately// set to install.//mOk.setText(R.string.install);mOkCanInstall = true;} else {mScrollView.setFullScrollAction(new Runnable() {@Overridepublic void run() {//mOk.setText(R.string.install);mOkCanInstall = true;}});} }

startInstallConfirm主要負責界面初始化,顯示權限信息等。

界面初始化完成后,安裝界面就會呈現在用戶面前,如果用戶想要安裝這個應用程序,可以直接點擊確認按鈕,此時就會調用PackageInstallerActivity中的onClick方法:

public void onClick(View v) {if (v == mOk) {if (mOkCanInstall || mScrollView == null) {mInstallFlowAnalytics.setInstallButtonClicked();if (mSessionId != -1) {mInstaller.setPermissionsResult(mSessionId, true);// We're only confirming permissions, so we don't really know how the// story ends; assume success.mInstallFlowAnalytics.setFlowFinishedWithPackageManagerResult(PackageManager.INSTALL_SUCCEEDED);finish();} else {startInstall();//注釋1}} else {mScrollView.pageScroll(View.FOCUS_DOWN);}} else if(v == mCancel) {// Cancel and finishsetResult(RESULT_CANCELED);if (mSessionId != -1) {mInstaller.setPermissionsResult(mSessionId, false);}mInstallFlowAnalytics.setFlowFinished(InstallFlowAnalytics.RESULT_CANCELLED_BY_USER);finish();} }

onClick方法中分別會對取消和確定按鈕做處理,如果是確定按鈕,就會調用注釋1處的startInstall方法。

(4) 深入到PackageInstallerActivity—>startInstall方法

private void startInstall() {// Start subactivity to actually install the applicationIntent newIntent = new Intent();newIntent.putExtra(PackageUtil.INTENT_ATTR_APPLICATION_INFO,mPkgInfo.applicationInfo);newIntent.setData(mPackageURI);newIntent.setClass(this, InstallAppProgress.class);//跳轉的ActivitynewIntent.putExtra(InstallAppProgress.EXTRA_MANIFEST_DIGEST, mPkgDigest);newIntent.putExtra(InstallAppProgress.EXTRA_INSTALL_FLOW_ANALYTICS, mInstallFlowAnalytics);String installerPackageName = getIntent().getStringExtra(Intent.EXTRA_INSTALLER_PACKAGE_NAME);......省略.....if(localLOGV) Log.i(TAG, "downloaded app uri="+mPackageURI);startActivity(newIntent);//注釋1finish(); }

startInstall方法啟動后,將會跳轉到注釋1處的InstallAppProgress界面。并關掉當前的PackageInstallerActivity

(5) 深入到InstallAppProgress.java文件中

當InstallAppProgress.java啟動時,會先調用InstallAppProgress.java—>onCreate方法:

@Override public void onCreate(Bundle icicle) {......省略.....initView(); }

onCreate方法中會調用initView方法初始化界面:

public void initView() {......省略.....if ("package".equals(mPackageURI.getScheme())) {try {pm.installExistingPackage(mAppInfo.packageName);observer.packageInstalled(mAppInfo.packageName,PackageManager.INSTALL_SUCCEEDED);} catch (PackageManager.NameNotFoundException e) {observer.packageInstalled(mAppInfo.packageName,PackageManager.INSTALL_FAILED_INVALID_APK);}} else {pm.installPackageWithVerificationAndEncryption(mPackageURI, observer, installFlags,installerPackageName, verificationParams, null);} }

可以看到有兩條安裝邏輯,我這里只探索else中的邏輯。else中實際上是調用了ApplicationPackageManager的installPackageWithVerificationAndEncryption方法來安裝。

(6) 深入到ApplicationPackageManager—>installPackageWithVerificationAndEncryption方法:

@Override public void installPackageWithVerificationAndEncryption(Uri packageURI,PackageInstallObserver observer, int flags, String installerPackageName,VerificationParams verificationParams, ContainerEncryptionParams encryptionParams) {installCommon(packageURI, observer, flags, installerPackageName, verificationParams,encryptionParams);//注釋1 }

注意installPackageWithVerificationAndEncryption方法上有個Override,說明繼承于父類的PackageManager中。

從注釋1中可以看到調用的是installCommon方法:

private void installCommon(Uri packageURI,PackageInstallObserver observer, int flags, String installerPackageName,VerificationParams verificationParams, ContainerEncryptionParams encryptionParams) {......省略.....try {mPM.installPackage(originPath, observer.getBinder(), flags, installerPackageName,verificationParams, null);//注釋1} catch (RemoteException ignored) {} }

從注釋1中可以看到調用的是PMS中的installPackage方法。到這里后續的邏輯就與前面的“網絡下載應用安裝”中的邏輯一樣了。

總結

以上是生活随笔為你收集整理的从源码角度解析Android中APK安装过程的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

五月婷婷毛片 | 亚洲最大色 | 六月丁香在线观看 | 91看片网址 | 狠狠操狠狠干2017 | 亚洲jizzjizz日本少妇 | 久久精品免视看 | 男女拍拍免费视频 | 国产黄色免费电影 | 国产精品123| 欧美乱码精品一区 | 国产亚洲久一区二区 | 国产精品人人做人人爽人人添 | 日韩中文在线播放 | 国产 视频 久久 | 91网免费观看 | 91看片一区二区三区 | 国产色a在线观看 | 一本大道久久精品懂色aⅴ 五月婷社区 | 亚洲一级二级三级 | 国产 视频 高清 免费 | 在线三级中文 | av成人资源 | www.日日日.com | 久久一区国产 | 成人羞羞视频在线观看免费 | 免费看的黄色的网站 | 97在线影院| 国产片网站 | 欧美精品一区二区三区一线天视频 | 国产麻豆精品久久一二三 | 波多野结衣视频一区二区三区 | 亚洲综合五月 | 草久久av| 91成人精品一区在线播放 | 日韩中文字幕视频在线观看 | 成人久久亚洲 | 久久99免费观看 | 毛片激情永久免费 | av福利免费| 色婷婷激情 | 日本黄色免费在线 | 婷婷色网站 | 久久这里只有精品1 | 日韩欧美精品在线视频 | 在线播放 一区 | 久久精品99国产精品酒店日本 | 日韩在线观看小视频 | 久久图| 日韩在线小视频 | 天天干天天操天天搞 | 午夜久草| 成人动漫一区二区三区 | 亚洲精品男人天堂 | 国产美女被啪进深处喷白浆视频 | 午夜在线日韩 | 免费一级毛毛片 | 精品视频在线免费观看 | 国产精品字幕 | 国产精品毛片一区二区在线 | 午夜狠狠操| 亚洲第一区在线播放 | 婷婷久久综合九色综合 | 99久久精品免费看国产一区二区三区 | 91在线播放国产 | 亚洲欧美国产精品18p | 99精品偷拍视频一区二区三区 | 亚洲精品国产精品国产 | 超碰在线免费福利 | 久草在线看片 | 日韩欧美在线影院 | 国产九九精品 | 中文字幕在线日本 | 91亚州| 99热在线国产| 亚洲天堂社区 | 麻豆国产精品va在线观看不卡 | 中日韩男男gay无套 日韩精品一区二区三区高清免费 | 免费视频一二三区 | 成人av中文字幕在线观看 | 草久久影院 | 国产精品观看视频 | 免费在线播放 | 在线观看成人网 | 粉嫩一区二区三区粉嫩91 | 草久草久 | av丝袜在线 | 国产成人精品综合久久久久99 | 午夜视频99 | 成人一级在线观看 | 最新中文字幕视频 | 热九九精品| 国产女人18毛片水真多18精品 | 国产小视频免费在线观看 | 亚洲成年人av | 午夜私人影院久久久久 | 国产麻豆成人传媒免费观看 | 97在线视频免费 | 福利视频导航网址 | 国产青春久久久国产毛片 | 成人h视频| 免费的国产精品 | 黄色日本免费 | 91桃色视频 | 97超碰人人模人人人爽人人爱 | 精品国产伦一区二区三区观看方式 | 久久久黄色av | 国产视频一区二区在线 | 99视频精品免费视频 | 在线亚洲欧美日韩 | 免费看一级特黄a大片 | 日韩欧美电影 | 人人狠| 91成人精品一区在线播放 | 在线看小早川怜子av | 91av在线免费 | 久久综合九色综合欧美狠狠 | 手机av电影在线观看 | 亚洲精品99久久久久中文字幕 | 亚洲一区二区三区毛片 | 91精品入口| 一级片视频免费观看 | 婷婷伊人五月天 | 一个色综合网站 | 97在线精品国自产拍中文 | 五月开心激情 | www·22com天天操| 一区在线播放 | 日韩天天综合 | 欧美日本国产在线观看 | 亚洲精品高清一区二区三区四区 | 天天躁天天操 | 国产精品免费成人 | 国产一区二三区好的 | 1000部国产精品成人观看 | 久射网| 久久艹精品| 亚洲国产高清在线观看视频 | 欧美伦理一区二区三区 | 欧美日韩国产精品久久 | 中文在线免费观看 | 日韩欧美高清不卡 | 久久久久久国产精品美女 | 91视频免费观看 | 综合色播 | 91成人免费看片 | 91视频a| 视频在线观看入口黄最新永久免费国产 | 欧美精品久久久久久久久久白贞 | www最近高清中文国语在线观看 | 亚洲成人动漫在线观看 | 久久色视频| www.五月激情.com | 1024在线看片 | 中文字幕免费高清av | 国产精品久久久久久久久毛片 | 99热这里只有精品国产首页 | 中文字幕在线人 | 成人福利av| 99精品免费久久久久久日本 | 狠狠色狠狠色综合系列 | 国产精品久久麻豆 | 欧美成人按摩 | 日韩高清免费无专码区 | 日韩理论片在线观看 | 在线观看成人福利 | 中文字幕色在线视频 | 特黄免费av| 亚洲精品国偷拍自产在线观看蜜桃 | 国内精品久久久久久久久 | 亚洲 欧美 综合 在线 精品 | 亚洲一二三久久 | 中文在线字幕免费观看 | 国产精品一区一区三区 | 久操视频在线播放 | 四虎永久网站 | 热久久99这里有精品 | 国产黄av | 亚洲 欧洲av | 色99在线 | 国产精品五月天 | 精品无人国产偷自产在线 | 亚洲黄色免费在线 | 国产精品麻豆99久久久久久 | 国产精品黄色在线观看 | 欧美性黑人 | 精品亚洲一区二区三区 | 亚洲激情校园春色 | 欧美一区二区三区特黄 | 亚洲高清av在线 | 国产18精品乱码免费看 | 波多野结衣精品视频 | 国产高清无线码2021 | 青青草国产精品视频 | 91精品一区二区三区久久久久久 | 国产在线第三页 | 啪嗒啪嗒免费观看完整版 | 一区二区三区视频网站 | 在线香蕉视频 | 91视频 - x99av | 一区三区在线欧 | 亚洲国产成人av网 | 精品国产亚洲一区二区麻豆 | www.久久久久 | 成人av资源| 天天爽天天爽天天爽 | 欧美一级特黄aaaaaa大片在线观看 | 在线 日韩 av| 97精品视频在线播放 | 国产精品99精品久久免费 | 视频在线国产 | 天天搞天天干 | 国产精品18久久久久久久久久久久 | 99久久久国产精品免费观看 | 中文字幕 国产 一区 | 精品视频在线视频 | 日韩有码中文字幕在线 | 久久无码精品一区二区三区 | 亚洲视频大全 | 婷婷色网视频在线播放 | 黄色毛片视频 | 亚洲国产精品电影在线观看 | 欧洲精品亚洲精品 | 免费网站黄色 | 国产经典三级 | 日日干干 | 91自拍视频在线观看 | 国产一区视频免费在线观看 | 最近更新的中文字幕 | 黄色软件视频大全免费下载 | 黄色一级大片免费看 | 中文字幕av在线不卡 | 亚洲欧美日韩国产一区二区三区 | 日韩av电影中文字幕 | 久久精品999 | 免费观看性生交大片3 | 久久久久久国产精品亚洲78 | 国产精品每日更新 | 中文字幕在线观看完整版电影 | 伊香蕉大综综综合久久啪 | 国产成人亚洲在线电影 | 久久久久综合网 | 天天操夜夜操国产精品 | 六月婷操 | 亚洲激情校园春色 | 最近中文国产在线视频 | 五月在线| 亚洲天堂网在线视频观看 | 亚洲精品久久久久999中文字幕 | av看片在线 | 亚洲第一av在线 | 欧美日韩视频免费看 | 久久成人国产精品 | 99精品成人 | 美女视频黄免费 | 成人精品久久久 | 国产麻豆剧传媒免费观看 | 久久久麻豆视频 | 色视频在线免费观看 | 国内精品视频在线播放 | 在线成人免费电影 | 国产一区二区不卡视频 | 操操操日日日 | 成人黄大片| 91在线国产观看 | 一级黄色片在线免费观看 | 中文字幕视频三区 | 欧美日韩在线免费观看视频 | 99精品久久久久 | 天天射射天天 | 在线视频国产区 | 美女久久久 | 久久情爱 | 成人av片免费看 | 久久精品官网 | 天堂在线一区二区三区 | 亚洲一级二级三级 | 亚洲免费色| 日韩av一区二区三区四区 | 天天综合天天做 | av在线等| 91黄色在线看 | 五月天婷婷在线播放 | 亚洲最新视频在线播放 | 五月天激情婷婷 | 一区二区三区四区五区在线视频 | 亚洲国产精品一区二区久久hs | 成人黄色在线 | 在线国产欧美 | av中文字幕在线播放 | 日日摸日日碰 | 免费看污片| 91在线播 | 青青草国产精品 | 亚洲国产美女精品久久久久∴ | 久久国产欧美日韩精品 | 欧美精品在线观看免费 | 欧美,日韩 | 久草视频99| 国产九九热 | 色播五月激情五月 | 国产精品自产拍在线观看桃花 | 久久久一本精品99久久精品66 | 久久精国产 | 免费网站看v片在线a | 国产精品久久久久久久久久直播 | 精品国产区 | 91久久精| 黄色特一级 | 九九九九九国产 | 国产成人久久av977小说 | 亚洲永久av| 97韩国电影| 久久久久久黄 | 五月婷婷六月丁香激情 | 天天做天天干 | 久久人人97超碰国产公开结果 | 婷婷六月丁香激情 | 国产精品一区二区无线 | 国产精品网红福利 | 午夜三级毛片 | 91av视频| 久视频在线播放 | 国产精品国产三级在线专区 | 在线播放亚洲 | 国产精品中文字幕在线观看 | 国产精品一区二区三区视频免费 | 国产999精品| 在线免费观看羞羞视频 | 日韩欧美高清一区二区三区 | 国产69久久 | 最新av免费在线观看 | 在线日韩精品视频 | 午夜精品电影一区二区在线 | 久久久精品在线观看 | 国产精品久久久久免费观看 | 亚洲四虎在线 | av在线进入 | 国产成人一区二区三区电影 | 永久免费视频国产 | 日韩中文字幕第一页 | 国产精品视频免费看 | 丝袜美腿亚洲综合 | 五月天婷婷免费视频 | 欧美尹人| 四虎影视8848aamm| 久久婷婷网 | 在线中文字幕电影 | 国产成人精品不卡 | www黄色com | 97超碰精品 | 日韩中文字幕电影 | 久久精品麻豆 | 国产 欧美 日韩 | 久久麻豆精品 | 色婷久久 | a极黄色片| 中文字幕免费中文 | 亚洲成人在线免费 | 国产最顶级的黄色片在线免费观看 | 91精品色| 日韩精品久久中文字幕 | 午夜影视剧场 | 91网页版免费观看 | 国产一级黄色免费看 | 日韩精品在线播放 | www.亚洲视频.com | 久久精品视频中文字幕 | 精品亚洲欧美一区 | 欧美乱大交 | 国产视频观看 | 国产乱码精品一区二区三区介绍 | 国产精品18p | 色资源二区在线视频 | 国产精品成久久久久三级 | 成人免费视频网站在线观看 | 色噜噜噜噜 | 色中射| 日韩高清在线不卡 | 狠狠干我| 日本激情动作片免费看 | 色操插 | 国产精品美女999 | 欧美日韩在线免费观看视频 | 911久久香蕉国产线看观看 | 日日夜夜狠狠干 | 在线看成人av | 日韩在线观看中文字幕 | 日韩在线不卡 | 韩国三级一区 | 国产美腿白丝袜足在线av | 国产精品久久一区二区三区, | 亚洲国产手机在线 | 午夜精品久久久久久久久久久久 | 久久久久久久久影院 | 国产精品一区二区三区四区在线观看 | 天天射天天舔天天干 | 99久久婷婷国产一区二区三区 | 色婷婷六月 | 天天操天天透 | 黄色a在线观看 | 日韩欧美91 | 精品一区二区影视 | 91免费观看视频网站 | 免费成人黄色片 | 国产成人一级电影 | av电影在线免费 | 亚洲精品中文在线 | 一区二区三区四区影院 | 制服丝袜欧美 | 国产精品成人aaaaa网站 | 午夜在线资源 | 97视频在线观看免费 | 美女视频a美女大全免费下载蜜臀 | 国产小视频网站 | 欧美成人影音 | 999电影免费在线观看2020 | 国产日韩欧美在线一区 | 久草视频免费观 | 免费久久99精品国产 | 久久久天堂 | 日韩中出在线 | 在线播放日韩av | 亚洲精品在线观看不卡 | 精品国产电影一区二区 | 国产综合小视频 | 午夜影院一级 | 久草成人在线 | 日韩免费网站 | 青草视频在线 | 亚洲 中文 欧美 日韩vr 在线 | 日韩精品欧美视频 | 波多野结衣视频一区 | 97在线观看视频免费 | 麻花豆传媒mv在线观看网站 | www天天干com| 久久精品永久免费 | 97人人精品 | 顶级欧美色妇4khd | 国产精品久久一区二区三区, | 成人在线小视频 | 久久在线观看视频 | 日本福利视频在线 | 91九色视频导航 | 国产午夜剧场 | 午夜影院一级 | 久久美女高清视频 | 免费日韩视 | 99精品在线观看视频 | 色婷婷狠狠 | 国产精品久久久一区二区三区网站 | 久久国产精品免费观看 | 色综合久久五月 | 久久99精品国产麻豆婷婷 | 四虎在线永久免费观看 | 欧美日韩精品免费观看视频 | 91精品亚洲影视在线观看 | 97天堂 | 1000部18岁以下禁看视频 | 97在线精品国自产拍中文 | 91精品一区二区三区久久久久久 | www黄色av | 久久久久蜜桃 | 久久久久国产精品免费网站 | 人人爽人人 | 亚洲精品国产片 | 国产在线免费观看 | 日本中文字幕网站 | 国产91精品一区二区 | 亚洲伦理一区二区 | 99国内精品久久久久久久 | 日本久久精品视频 | 久久国产热视频 | 欧美在线free | 亚洲国产午夜精品 | 亚洲爱爱视频 | 免费看色视频 | 色偷偷av男人天堂 | 青青射 | 九九九九热精品免费视频点播观看 | 亚洲国产999 | 国产久视频 | 特级黄色视频毛片 | www.久久久 | 久久大片网站 | 欧美视频不卡 | 最新国产一区二区三区 | 日韩av影视在线观看 | 日韩成人精品一区二区三区 | 国产一区二区在线免费播放 | 欧美日韩免费一区 | a成人v在线 | 99精品视频在线观看免费 | 欧美精品乱码久久久久 | 久久91久久久久麻豆精品 | 久久久久久高潮国产精品视 | 成人久久国产 | 色一级片 | 国产v在线观看 | 丁香色天天 | 黄色一级大片在线观看 | 色九九在线 | 欧洲精品视频一区二区 | 久久久久免费精品国产小说色大师 | 日韩精品视频网站 | 国产精品一区在线观看 | 久99视频 | 久久久影片 | 中文字幕在线播放一区 | 最近字幕在线观看第一季 | 久久九九精品 | 欧美日韩在线视频免费 | 成人在线视频你懂的 | 久久综合中文色婷婷 | 麻豆免费精品视频 | 成人免费中文字幕 | 亚洲经典视频 | 日韩激情片在线观看 | 手机成人av在线 | 国产美女免费观看 | 黄色大片免费播放 | 视频一区二区三区视频 | 三日本三级少妇三级99 | 国产精品久久久久久久久久久杏吧 | 97在线观看视频免费 | 国产不卡在线观看 | 国产麻豆精品免费视频 | 中日韩欧美精彩视频 | 日韩特黄一级欧美毛片特黄 | 性色在线视频 | 片黄色毛片黄色毛片 | 国产日韩中文字幕在线 | zzijzzij日本成熟少妇 | 天天摸天天操天天舔 | 久久久久久久久久影院 | 青青河边草免费观看 | 国内精品久久久久久 | 婷婷色站 | 毛片一二区 | 精品亚洲一区二区 | 日韩三级成人 | 欧美午夜理伦三级在线观看 | 99精品偷拍视频一区二区三区 | 又黄又爽又湿又无遮挡的在线视频 | 精品国产一区二区三区久久久 | 天天操天天干天天综合网 | 精品一区二区在线播放 | 伊人电影天堂 | 亚洲国内在线 | 精品国产乱码久久久久久天美 | 日韩av男人的天堂 | 97超碰人人模人人人爽人人爱 | 欧美日韩视频在线观看免费 | 亚洲高清色综合 | 色婷婷影视 | 狠狠干,狠狠操 | 久久久www成人免费精品 | 亚洲日本va中文字幕 | 九九免费观看全部免费视频 | 亚洲永久精品视频 | 久久久久99精品成人片三人毛片 | 成人免费视频网 | 又黄又爽的视频在线观看网站 | 探花视频在线版播放免费观看 | 欧美日韩在线视频免费 | 欧美成人999 | 青青草久草在线 | 久久免费视频观看 | 婷婷综合久久 | 天天爱天天操天天射 | 五月天婷婷在线视频 | 狠狠躁日日躁狂躁夜夜躁 | 精品国产一区二区三区在线 | 中文国产成人精品久久一 | 欧美日本高清视频 | 成人av高清在线观看 | 99国产精品一区 | 日本大尺码专区mv | 伊人开心激情 | 国产精品一区在线观看 | 天天色天天射综合网 | 91视频在线免费下载 | 91成人在线观看喷潮 | 天天操天天操天天操天天操天天操天天操 | 久久综合久久久久88 | 制服丝袜欧美 | 久久激情视频 久久 | 久久亚洲专区 | 天堂在线视频免费观看 | 欧洲性视频| 91视频在线看 | 国产1级毛片 | 香蕉视频国产在线观看 | 国产露脸91国语对白 | 在线看的av网站 | 成人免费一级片 | 麻豆av一区二区三区在线观看 | 久久综合福利 | 亚洲精品 在线视频 | 免费久久精品视频 | 欧美日韩一区二区久久 | 亚洲精品www. | 成人免费看电影 | 日韩精品国产一区 | 亚洲在线日韩 | 亚洲 综合 国产 精品 | 国产精品自产拍在线观看中文 | 天天躁天天操 | 99久久久国产精品美女 | 91免费看片黄| 精品久久久久久国产偷窥 | 狠狠操在线 | 色在线高清 | 五月天久久精品 | 免费日韩一区二区 | 天天综合网在线观看 | 一级大片在线观看 | 天天插天天色 | a视频免费看 | 美女视频一区 | 久久久久久毛片精品免费不卡 | 丁香六月婷婷综合 | 国产亚洲精品久久久久5区 成人h电影在线观看 | 亚洲精品成人av在线 | 精品在线观看一区二区三区 | 国产三级在线播放 | 久久久久久久久久亚洲精品 | 婷婷色亚洲 | 国产精品私拍 | 狠狠色丁香久久婷婷综合丁香 | 久草青青在线观看 | 在线视频专区 | 久久免费视频这里只有精品 | 高潮毛片无遮挡高清免费 | 久久成人高清 | 日韩综合第一页 | 国产精品久久久毛片 | 香蕉97视频观看在线观看 | 国产精品久久久久久久久岛 | 久久久久久久久毛片精品 | 一区二区三高清 | 亚洲精品视频免费看 | 久久99热久久99精品 | 在线看的毛片 | 精品视频999| av一级网站 | 麻豆视屏| 国产人成在线视频 | 国产一区欧美在线 | 天天躁日日躁狠狠躁av麻豆 | 免费高清看电视网站 | 91热这里只有精品 | 人人爽人人爽 | 成人黄色小说在线观看 | 久久亚洲在线 | 欧美日韩免费观看一区二区三区 | 97色综合| 免费福利片 | 精品一区二区日韩 | 久久人人精 | 一级c片 | 狠狠操影视 | 天天狠狠干 | 在线观看91视频 | 久久色视频 | 天天射天 | 久久99精品国产91久久来源 | www.色在线| 国产高清免费在线观看 | 久久久久免费看 | 99精品乱码国产在线观看 | 在线观看一级片 | 狠色狠色综合久久 | www.黄色片.com| 久久香蕉电影网 | 亚洲欧美国产视频 | 国产在线观看高清视频 | 亚洲一区二区视频在线 | 一级黄色片在线免费看 | 国产色就色 | 在线观看蜜桃视频 | 久久久久福利视频 | 久久国产精品二国产精品中国洋人 | 欧美福利视频一区 | 久久亚洲在线 | 在线免费看黄色 | 日韩欧美一区二区三区在线 | 五月婷婷综合在线视频 | 国产一区二区在线影院 | 天天曰天天爽 | 91久久久久久久一区二区 | av免费网 | 国产精品久免费的黄网站 | 国产xxxx做受性欧美88 | 在线观看视频日韩 | 中文字幕av最新更新 | 天天操狠狠操夜夜操 | 欧美综合在线观看 | 亚洲人视频在线 | 97成人免费 | av视屏在线播放 | av超碰在线 | 免费成人在线网站 | 亚洲综合视频在线观看 | 91成人午夜| 有码一区二区三区 | 午夜久久 | 精品在线不卡 | 在线导航福利 | 成人a免费视频 | 婷婷色六月天 | 成人在线视频在线观看 | 精品电影一区二区 | 中文在线字幕免费观 | 欧美久久久久久久久 | 日韩av手机在线看 | 国产美女黄网站免费 | 五月婷婷在线视频观看 | 成人中文字幕在线观看 | aaa日本高清在线播放免费观看 | 91精品久久久久久综合乱菊 | 国产日韩精品久久 | 国产一区高清在线 | 丁香婷婷久久 | 久久国产精品99久久久久 | 国产在线观看免费 | 免费在线中文字幕 | 国产视频2 | 亚洲视频在线播放 | 97爱| 中文有码在线 | 色婷婷狠狠18 | 久久你懂的 | 又长又大又黑又粗欧美 | 99色在线观看 | 最新日韩在线观看 | 中文字幕一区二区三 | 五月天久久综合 | 免费看wwwwwwwwwww的视频 久久久久久99精品 91中文字幕视频 | 国内久久看 | 九九视频免费在线观看 | 成人午夜片av在线看 | 亚洲成人av免费 | 91视频亚洲| 日韩精品久久久久久中文字幕8 | 久久久国产精品成人免费 | 中文字幕视频一区 | 人人爽影院 | 丁香花中文在线免费观看 | 日韩精品在线视频免费观看 | 国产成人精品免费在线观看 | 国产一级免费播放 | 色婷婷狠狠 | 99色免费 | 98福利在线| 日韩毛片一区 | 欧美一区在线看 | 久色网 | 91视频免费播放 | 精品网站999www | 99久久精品免费看国产 | 亚洲va欧美va人人爽 | 中文字幕在线日亚洲9 | 91网站在线视频 | av7777777| 热久久视久久精品18亚洲精品 | 在线欧美小视频 | 亚洲视频六区 | 99久久9| 日韩午夜在线 | 日韩v欧美v日本v亚洲v国产v | 欧美日韩国产一区二区三区 | 波多野结衣电影一区二区三区 | 国产中的精品av小宝探花 | 国产精品18久久久久久久久久久久 | 久久歪歪| 天天色天天综合网 | 亚洲免费在线看 | 另类五月激情 | 国产精品电影在线 | www.黄色片网站 | 亚洲一级影院 | 免费av影视 | 日韩av电影手机在线观看 | 91视频 - v11av| 九色porny真实丨国产18 | 91视频在线| 91色国产| 美女网站视频免费都是黄 | 精品一区电影国产 | 色综合久久久久久中文网 | 日韩日韩日韩日韩 | 免费看的黄网站软件 | 亚洲视频 中文字幕 | 免费在线91| 国产裸体视频bbbbb | 久久久久激情电影 | 国产精品99久久久久人中文网介绍 | 成人av在线直播 | 五月婷婷欧美 | 亚洲小视频在线观看 | 国产精品久久久久久久久久久久 | 久久精品2 | 2023av| 成人av电影免费 | 美州a亚洲一视本频v色道 | 国产精品手机看片 | 久草五月| 77国产精品 | 亚洲精品国偷自产在线99热 | 亚洲视频在线播放 | 青春草视频 | 成人久久18免费网站 | 国产三级午夜理伦三级 | 成人在线黄色 | 美女视频免费精品 | 欧美另类sm图片 | 国产精品久久伊人 | 蜜桃av人人夜夜澡人人爽 | 欧美成人区 | 亚洲电影第一页av | 不卡视频一区二区三区 | 91在线超碰 | 国产一级一片免费播放放a 一区二区三区国产欧美 | 中文av字幕在线观看 | 色吊丝在线永久观看最新版本 | 久草视频免费播放 | 国产免费大片 | 九九九九精品 | 五月开心婷婷网 | 亚洲天堂自拍视频 | 免费毛片一区二区三区久久久 | 国产v视频 | 99人成在线观看视频 | 国产精品av一区二区 | 亚洲视频在线观看 | 黄色免费网站 | 久草视频手机在线 | 精品视频在线视频 | 狠狠五月天 | 欧美在线视频精品 | 久草在线视频网 | 免费观看性生交 | 天天拍天天干 | 五月天久久久 | 在线а√天堂中文官网 | 久久天堂亚洲 | 91精品国产电影 | 国产精品久久久久aaaa九色 | 在线观看av不卡 | 国产精品99久久久久人中文网介绍 | 午夜色场| 国产精品初高中精品久久 | 免费99精品国产自在在线 | 中文字幕丝袜美腿 | 毛片网在线 | 波多野结衣视频一区二区三区 | 美女黄频 | 国产中文字幕免费 | 久久久久久久久久久久影院 | 国产精品美女免费 | 深夜男人影院 | 亚洲黄色小说网 | 视频一区二区在线 | 亚洲蜜桃av| 91片黄在线观 | 992tv人人草 黄色国产区 | www.天天色.com | 国产色女人 | 欧美激情视频在线免费观看 | 349k.cc看片app| 精品国产一区二区三区在线 | 亚洲丁香久久久 | 福利视频区 | av免费网站在线观看 | 久久69精品 | 午夜精品视频福利 | 天天爽夜夜操 | 99九九视频| 国产成人精品不卡 | 久久国产精品网站 | av黄免费看 | 国产1级视频 | 婷婷久久精品 | 色综合天天色综合 | 久久精品一区二区三区中文字幕 | 国产精品久久伊人 | 亚洲欧美国产精品久久久久 | 国产区久久 | 国产成人综合在线观看 | 国产高清中文字幕 | 国产乱对白刺激视频不卡 | 黄色小说在线免费观看 | 伊人色综合久久天天 | 国产亚洲精品久久久久久大师 | 成人久久18免费网站麻豆 | www久久| 亚洲精品ww| 免费a网站 | 亚洲精品美女久久17c | 欧美精品一二三 | 香蕉视频在线免费 | 欧美色图亚洲图片 | 日韩精品中文字幕在线播放 | 久久国产成人午夜av影院潦草 | 国产成人黄色片 | 欧美福利视频一区 | 欧美日韩后 | 久久久国产毛片 | 久久精品视频在线观看免费 | 91精品国自产在线观看 | 久久久久国产精品一区 | 国产高清免费观看 | 国产亚洲视频在线免费观看 | 色视频在线 | 在线观看韩国av | 日日爽视频| 中文字幕人成人 | 国产96精品 | 欧美大片在线观看一区 | 国产成人精品一区在线 | 99久久免费看| 久久高清国产视频 | 国产.精品.日韩.另类.中文.在线.播放 | 人人射人人爱 | 久久99热精品这里久久精品 | 国产精品videoxxxx | 天天干,天天干 | 欧美日韩调教 | www.久久色 | 蜜臀91丨九色丨蝌蚪老版 | 久久成年人视频 | 亚洲一片黄 | 久久久久久久久久久久影院 | 久久9999久久免费精品国产 | 久久久久99999| 国产精品免费观看国产网曝瓜 | 天天操天天操天天操天天操天天操 | 免费在线观看日韩欧美 | 久久99国产精品视频 | 精品一区二区免费在线观看 | 久久精品成人欧美大片古装 | 日韩欧美一区二区三区在线 | 欧美日韩另类在线观看 | 久久久999精品视频 国产美女免费观看 | 久久久久亚洲精品 | 少妇bbb | 国产精品正在播放 | 网站免费黄色 | 久久91网 | 狠狠伊人 | 免费网站观看www在线观看 | 久久黄网站 | 国产精品中文在线 | 国产一级片播放 | 国产不卡在线看 | 国产在线黄 | 精品毛片在线 | 亚洲成av片人久久久 | 久久福利综合 | 成人一区二区在线观看 | 九九久久国产 | 午夜视频黄 | 亚洲乱码国产乱码精品天美传媒 | 成人理论电影 | 精品在线亚洲视频 | 操操碰 | 久久99精品久久久久久 | 国内少妇自拍视频一区 | 国产91在线观看 | 久久国产精品一二三区 | 99精品国产99久久久久久福利 | 日本资源中文字幕在线 | 久久久久久久久久久电影 | 日韩啪啪小视频 | 激情五月在线视频 | 人人讲| 视频在线观看入口黄最新永久免费国产 | 97久久精品午夜一区二区 | 日韩免费高清在线观看 | 五月天婷婷在线播放 | 又爽又黄又无遮挡网站动态图 | 九九热国产视频 | 精品国产欧美一区二区三区不卡 | 久草电影在线观看 | 日韩精品电影在线播放 | 免费看片成人 | 国产高清久久 | 成人网页在线免费观看 | 免费看的黄色 | 超碰人人超碰 | av在线com| 香蕉久草在线 | 日韩免费一级电影 | 成人avav| 久久精品二区 | 一区二区三区四区精品视频 | 九九热在线视频 |