preface

A little over a month ago, I wrote a hotfix primer that introduced the implementation principles of various widely discussed and used hotfix technologies. In that article, I also said that I would continue to study the source code of the hotfix technology based on dex subcontracting.

The hot repair technology based on the dex subcontract should be the first proposed by qzone team, but they only shared the implementation principle through the technical article, and its own source code has not been made public, so THE implementation details and coding style of QQ hot repair have no chance to observe. But there are a number of teams that have implemented hot fixes based on the principles introduced in Qzone and released the source code, such as @Dodola’s RocooFix and AnoleFix (yes, he made two), and Amigo developed by Android predecessors working on Ele. me.

Since this senior specifically advertised his framework to me in the comments below my preliminary article on hot repair, I would like to analyze the details of his hot repair implementation first. However, he has also written the source code interpretation, although due to the current code update caused by his source code interpretation and the source code of some differences, but overall the logic is consistent. So actually I don’t need to analyze his framework in detail here, just pick the main ones.

Amigo hotfix framework anatomy

Amigo github: github.com/eleme/Amigo

In general, from what I have seen in the code, this is a fairly complete hotfix framework that can be applied. From detecting APK, fetching resource files, dex files, and inserting dex packages into dexElements, the process of restarting APK is quite complete and thoughtful. So, HERE I just want to talk about how Amigo actually plugged dex into dexElements, because that’s the key to the dex subcontract thermal repair technology, but it’s still a little different from the DE proposed by the Qzone team.

Amigo.java

 @Override
    public void onCreate() {
        super.onCreate(); . . . Log.e(TAG,"demoAPk.exists-->" + demoAPk.exists() + ", this--->" + this);

        ClassLoader originalClassLoader = getClassLoader();

        try {
            SharedPreferences sp = getSharedPreferences(SP_NAME, MODE_MULTI_PROCESS);

            if (checkUpgrade(sp)) {
                Log.e(TAG, "upgraded host app");
                clear(this);
                runOriginalApplication(originalClassLoader);
                return;
            }

            if(! demoAPk.exists()) { Log.e(TAG,"demoApk not exist");
                clear(this);
                runOriginalApplication(originalClassLoader);
                return;
            }

            if(! isSignatureRight(this, demoAPk)) {
                Log.e(TAG, "signature is illegal");
                clear(this);
                runOriginalApplication(originalClassLoader);
                return;
            }

            if(! checkPatchApkVersion(this, demoAPk)) {
                Log.e(TAG, "patch apk version cannot be less than host apk");
                clear(this);
                runOriginalApplication(originalClassLoader);
                return;
            }

            if(! ProcessUtils.isMainProcess(this) && isPatchApkFirstRun(sp)) {
                Log.e(TAG, "none main process and patch apk is not released yet");
                runOriginalApplication(originalClassLoader);
                return;
            }

            // only release loaded apk in the main process
            runPatchApk(sp); // This is the most important sentence. . . }Copy the code

RunPatchApk () is called in the onCreate method of the Amigo class to begin the process of replacing APK. Look again at the runPatchApk() method

  private void runPatchApk(SharedPreferences sp) throws LoadPatchApkException {
        try {
            String demoApkChecksum = getCrc(demoAPk);
            boolean isFirstRun = isPatchApkFirstRun(sp);
            Log.e(TAG, "demoApkChecksum-->" + demoApkChecksum + ", sig--->" + sp.getString(NEW_APK_SIG, ""));
            if (isFirstRun) {
                //clear previous working dir
                Amigo.clearWithoutApk(this);

                //start a new process to handle time-tense operation
                ApplicationInfo appInfo = getPackageManager().getApplicationInfo(getPackageName(), GET_META_DATA);
                String layoutName = appInfo.metaData.getString("amigo_layout");
                String themeName = appInfo.metaData.getString("amigo_theme");
                int layoutId = 0;
                int themeId = 0;
                if(! TextUtils.isEmpty(layoutName)) { layoutId = (int) readStaticField(Class.forName(getPackageName() +".R$layout"), layoutName);
                }
                if(! TextUtils.isEmpty(themeName)) { themeId = (int) readStaticField(Class.forName(getPackageName() +".R$style"), themeName);
                }
                Log.e(TAG, String.format("layoutName-->%s, themeName-->%s", layoutName, themeName));
                Log.e(TAG, String.format("layoutId-->%d, themeId-->%d", layoutId, themeId));

                ApkReleaser.work(this, layoutId, themeId);
                Log.e(TAG, "release apk once");
            } else {
                checkDexAndSoChecksum();
            }
            // Create an object inherited from the PathClassLoader class and pass in the patch APK path to construct a loader
            AmigoClassLoader amigoClassLoader = new AmigoClassLoader(demoAPk.getAbsolutePath(), getRootClassLoader());
            // This method reflects the LoadApk loader in the ActivityThread object corresponding to the app.
            setAPKClassLoader(amigoClassLoader);
            // This is the way to prepare the replacement for dex
            setDexElements(amigoClassLoader);
            // As the name implies, set the local library to load
            setNativeLibraryDirectories(amigoClassLoader);
            // Load some resource files
            AssetManager assetManager = AssetManager.class.newInstance();
            Method addAssetPath = getDeclaredMethod(AssetManager.class, "addAssetPath".String.class);
            addAssetPath.setAccessible(true);
            addAssetPath.invoke(assetManager, demoAPk.getAbsolutePath());
            setAPKResources(assetManager);

            runOriginalApplication(amigoClassLoader);
        } catch (Exception e) {
            throw newLoadPatchApkException(e); }}Copy the code

Instead of going into the setDexElements(amigoClassLoader) method, let’s look at the setAPKClassLoader(amigoClassLoader) method that sets the classloader, because this is also a key point that’s hard to ignore, so, Let’s see how he sets up the loader, okay

 private void setAPKClassLoader(ClassLoader classLoader)
            throws IllegalAccessException, NoSuchMethodException, ClassNotFoundException, InvocationTargetException {
        // Replace the "mClassLoader" property in the object returned by getLoadedApk() with the classloader of our new
        writeField(getLoadedApk(), "mClassLoader", classLoader);
    }Copy the code

The writeFiled method is used to set the classloader to mClassLoader by reflection. The key is getLoadedApk().

 private static Object getLoadedApk()
            throws IllegalAccessException, NoSuchMethodException, InvocationTargetException, ClassNotFoundException {
        / / the instance () returns a "android. App. ActivityThread" class, readField is read ActivityThread mPackages attribute in the class
        Map<String, WeakReference<Object>> mPackages = (Map<String, WeakReference<Object>>) readField(instance(), "mPackages".true);
        // The mPackage property contains a LoadedApk
        for (String s : mPackages.keySet()) {
            WeakReference wr = mPackages.get(s);
            if(wr ! =null&& wr.get() ! =null) {
                // A LoadedApk should be returned
                returnwr.get(); }}return null;
    }Copy the code

Ok, so we end up with the LoadedApk object, which is actually very important. After an APK is loaded, all the information is stored in this object (for example: DexClassLoader, Resources, Application), a package corresponds to an object, depending on the name of the package, and we just replace the Classloader in the LoadedApk object with our own classloader object, so we can load our own APK. Since our own amigoClassLoader actually inherits from the PathClassLoader, we intelligently load the APK in the specific directory, that is, our patch APK needs to be in the specific directory.

Well, with that said, let’s get back to business and replace dex for a hot fix. Continue down from setDexElements(amigoClassLoader)

private void setDexElements(ClassLoader classLoader) throws NoSuchFieldException, IllegalAccessException {
        //getPathList This is a reflection to read the pathList object in BaseDexClassLoader, which has a dexElements array wrapped around all the dex in the running APK.
        Object dexPathList = getPathList(classLoader);
        // File directory, patch apk dex file object array
        File[] listFiles = dexDir.listFiles();

        List<File> validDexes = new ArrayList<>();
        for (File listFile : listFiles) {
            if (listFile.getName().endsWith(".dex")) {
                // Add to the listvalidDexes.add(listFile); }}// Create an array of the same size files
        File[] dexes = validDexes.toArray(new File[validDexes.size()]);
        // Read the original dexElements array object in the dexPathList object by reflection
        Object originDexElements = readField(dexPathList, "dexElements");
        // Returns the type of the element in the dexElements arrayClass<? > localClass = originDexElements.getClass().getComponentType(); int length = dexes.length;// Then create a new array of the same size based on this type
        Object dexElements = Array.newInstance(localClass, length);
        for (int k = 0; k < length; K++) {assign to the arrayArray.set(dexElements, k, getElementWithDex(dexes[k], optimizedDir));
        }
        // Finally, the new array is reflected into the dexPathList object.
        writeField(dexPathList, "dexElements", dexElements);
    }Copy the code

Ok, now that the dex replacement is almost complete, there is some work to do to restart or re-run the Application. If you are not clear about BaseDexClassLoader, dexPathList, and dexElements, you can take a look at my previous article on hot repair, which has related introduction.

summary

If you really serious see my previous article hot fix a preliminary study, you will find that this framework is introduced to me that there are some discrepancy on the basis of the principle of dex subcontracting of hot fix, because this is the overall wrapped all the dex replace, also means that when need hot fix, download a file to a few bigger, may be the whole apk; Second, the framework uses a PathClassLoader instead of a DexClassLoader. PathClassLoader is limited because it can only load specified private paths, and the author directly replaces the original classloader by making extensive use of reflection. Then complete replacement of the entire dex through its own class loader. Overall, this frame has many advantages besides its larger size. (Though with reflection, it’s hard to launch an APP on Google Play, right?)

I didn’t know much about reflection in my work, but now it works really well because you can get a lot of private Android apis and properties that aren’t public……

Oh, my God. I decided to study my reflexes.

errata

no

Afterword.

Originally also continue to analyze other hot repair framework source code, but the length of this article is not small, halftime break, find the opportunity FOR me to write other framework source code implementation details in the new article to share

Finally, the performance table of each hot repair framework (accuracy not guaranteed)