/** * OpenAtlasForAndroid Project * <p> * The MIT License (MIT) * Copyright (c) 2015 Bunny Blue * <p> * Permission is hereby granted, free of charge, to any person obtaining a copy of this software * and associated documentation files (the "Software"), to deal in the Software * without restriction, including without limitation the rights to use, copy, modify, * merge, publish, distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to the following conditions: * <p> * The above copyright notice and this permission notice shall be included in all copies * or substantial portions of the Software. * <p> * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, * INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR * PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE * FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. * * @author BunnyBlue * @author BunnyBlue */ /** * @author BunnyBlue */ package com.openatlas.android.initializer; import android.app.Application; import android.content.Intent; import android.util.Log; import com.openatlas.boot.PlatformConfigure; import com.openatlas.framework.Atlas; import com.openatlas.framework.AtlasConfig; import com.openatlas.framework.BundleImpl; import com.openatlas.framework.bundlestorage.BundleArchiveRevision.DexLoadException; import org.osgi.framework.Bundle; public class OptDexProcess { private static OptDexProcess mOptDexProcess; private Application mApplication; private boolean isInitialized; private boolean isExecuted; private OptDexProcess() { } public static synchronized OptDexProcess getInstance() { if (mOptDexProcess != null) { return mOptDexProcess; } synchronized (OptDexProcess.class) { if (mOptDexProcess == null) { mOptDexProcess = new OptDexProcess(); } } return mOptDexProcess; } /*** 初始化OptDexProcess ***/ void init(Application application) { this.mApplication = application; this.isInitialized = true; } /** * 处理Bundles * * @param optAuto * 是否只处理安装方式为AUTO的Bundle * @param notifyResult * 通知UI安装结果 * ******/ public synchronized void processPackages(boolean optAuto, boolean notifyResult) { if (!this.isInitialized) { Log.e("OptDexProcess", "Bundle Installer not initialized yet, process abort!"); } else if (!this.isExecuted || notifyResult) { long currentTimeMillis; if (optAuto) { currentTimeMillis = System.currentTimeMillis(); optAUTODex(); if (!notifyResult) { finishInstalled(); } Log.e("debug", "dexopt auto start bundles cost time = " + (System.currentTimeMillis() - currentTimeMillis) + " ms"); } else { currentTimeMillis = System.currentTimeMillis(); optStoreDex(); Log.e("debug", "dexopt bundles not delayed cost time = " + (System.currentTimeMillis() - currentTimeMillis) + " ms"); if (!notifyResult) { finishInstalled(); } currentTimeMillis = System.currentTimeMillis(); getInstance().optStoreDex2(); Log.e("debug", "dexopt delayed bundles cost time = " + (System.currentTimeMillis() - currentTimeMillis) + " ms"); } if (!notifyResult) { this.isExecuted = true; } } } /** 通知UI安装完成 **/ private void finishInstalled() { Utils.saveAtlasInfoBySharedPreferences(this.mApplication); System.setProperty("BUNDLES_INSTALLED", "true"); this.mApplication.sendBroadcast(new Intent(PlatformConfigure.ACTION_BROADCAST_BUNDLES_INSTALLED)); } /**** 对已安装并且安装方式为STORE的Bundle进行dexopt操作 ****/ private void optStoreDex() { for (Bundle bundle : Atlas.getInstance().getBundles()) { if (!(bundle == null || contains(AtlasConfig.STORE, bundle.getLocation()))) { try { ((BundleImpl) bundle).optDexFile(); Log.e("OptDexProcess", "optStoreDex"); } catch (Throwable e) { if (e instanceof DexLoadException) { throw ((RuntimeException) e); } Log.e("OptDexProcess", "Error while dexopt >>>", e); } } } } /**** 对全部安装方式为Store的Bundle进行dexopt操作 ***/ private void optStoreDex2() { for (String bundle : AtlasConfig.STORE) { Bundle bundle2 = Atlas.getInstance().getBundle(bundle); if (bundle2 != null) { try { ((BundleImpl) bundle2).optDexFile(); Log.e("OptDexProcess", "optStoreDex2"); } catch (Throwable e) { if (e instanceof DexLoadException) { throw ((RuntimeException) e); } Log.e("OptDexProcess", "Error while dexopt >>>", e); } } } } /** 对随宿主启动的插件进行dexopt操作 ****/ private void optAUTODex() { for (String bundleName : AtlasConfig.AUTO) { Bundle bundle = Atlas.getInstance().getBundle(bundleName); if (bundle != null) { try { ((BundleImpl) bundle).optDexFile(); Log.e("OptDexProcess", "optAUTODex"); } catch (Throwable e) { if (e instanceof DexLoadException) { throw ((RuntimeException) e); } Log.e("OptDexProcess", "Error while dexopt >>>", e); } } } } private boolean contains(String[] bundleNames, String location) { if (bundleNames == null || location == null) { return false; } for (String bundleName : bundleNames) { if (bundleName != null && bundleName.equals(location)) { return true; } } return false; } }