All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.appmetrica.gradle.aarcheck.utils.ProguardMap.kt Maven / Gradle / Ivy

/*
 * Copyright (C) 2017 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package io.appmetrica.gradle.aarcheck.utils

import java.io.BufferedReader
import java.io.File
import java.io.FileNotFoundException
import java.io.FileReader
import java.io.IOException
import java.io.Reader
import java.text.ParseException

// Class used to deobfuscate classes, fields, and stack frames.
class ProguardMap {
    private class FrameData(
        var clearMethodName: String,
        var lineDelta: Int // lineDelta = obfuscatedLine - clearLine
    )

    // Constructs a ClassData object for a class with the given clear name.
    private class ClassData(val clearName: String) {

        // Mapping from obfuscated field name to clear field name.
        private val mFields: MutableMap = HashMap()

        // obfuscatedMethodName + clearSignature -> FrameData
        private val mFrames: MutableMap = HashMap()

        fun addField(obfuscatedName: String, clearName: String) {
            mFields[obfuscatedName] = clearName
        }

        // Get the clear name for the field in this class with the given
        // obfuscated name. Returns the original obfuscated name if a clear
        // name for the field could not be determined.
        // TODO: Do we need to take into account the type of the field to
        // propery determine the clear name?
        fun getField(obfuscatedName: String): String {
            val clearField = mFields[obfuscatedName]
            return clearField ?: obfuscatedName
        }

        // TODO: Does this properly interpret the meaning of line numbers? Is
        // it possible to have multiple frame entries for the same method
        // name and signature that differ only by line ranges?
        fun addFrame(
            obfuscatedMethodName: String,
            clearMethodName: String,
            clearSignature: String,
            obfuscatedLine: Int,
            clearLine: Int
        ) {
            val key = obfuscatedMethodName + clearSignature
            mFrames[key] = FrameData(clearMethodName, obfuscatedLine - clearLine)
        }

        fun getFrame(
            clearClassName: String,
            obfuscatedMethodName: String,
            clearSignature: String,
            obfuscatedFilename: String?,
            obfuscatedLine: Int
        ): Frame {
            val key = obfuscatedMethodName + clearSignature
            val frame = mFrames[key] ?: FrameData(obfuscatedMethodName, 0)
            return Frame(
                frame.clearMethodName,
                clearSignature,
                getFileName(clearClassName),
                obfuscatedLine - frame.lineDelta
            )
        }
    }

    private val mClassesFromClearName: MutableMap = HashMap()
    private val mClassesFromObfuscatedName: MutableMap = HashMap()

    class Frame(val methodName: String, val signature: String, val filename: String, val line: Int)

    // Read in proguard mapping information from the given file.
    @Throws(FileNotFoundException::class, IOException::class, ParseException::class)
    fun readFromFile(mapFile: File?) {
        readFromReader(FileReader(mapFile))
    }

    // Read in proguard mapping information from the given Reader.
    @Throws(IOException::class, ParseException::class)
    fun readFromReader(mapReader: Reader?) {
        val reader = BufferedReader(mapReader)
        var line = reader.readLine()
        while (line != null) {
            // Line may start with '#' as part of R8 markers, e.g.,
            //   '# compiler: R8'
            // Allow comments or empty lines in class mapping lines.
            var trimmed = line.trim()
            if (trimmed.isEmpty() || trimmed.startsWith("#")) {
                line = reader.readLine()
                continue
            }
            // Class lines are of the form:
            //   'clear.class.name -> obfuscated_class_name:'
            var sep = line.indexOf(" -> ")
            if (sep == -1 || sep + 5 >= line.length) {
                parseException("Error parsing class line: '$line'")
            }
            val clearClassName = line.substring(0, sep)
            val obfuscatedClassName = line.substring(sep + 4, line.length - 1)
            val classData = ClassData(clearClassName)
            mClassesFromClearName[clearClassName] = classData
            mClassesFromObfuscatedName[obfuscatedClassName] = classData

            // After the class line comes zero or more field/method lines of the form:
            //   '    type clearName -> obfuscatedName'
            line = reader.readLine()
            while (line != null) {
                trimmed = line.trim()
                // Allow comments or empty lines in field/method mapping lines.
                if (trimmed.isEmpty() || trimmed.startsWith("#")) {
                    line = reader.readLine()
                    continue
                }
                // After skipping comments or empty line,
                // make sure this is a field/method mapping line.
                if (!line.startsWith("    ")) {
                    break
                }
                val ws = trimmed.indexOf(' ')
                sep = trimmed.indexOf(" -> ")
                if (ws == -1 || sep == -1) {
                    parseException("Error parse field/method line: '$line'")
                }
                var type = trimmed.substring(0, ws)
                var clearName = trimmed.substring(ws + 1, sep)
                val obfuscatedName = trimmed.substring(sep + 4, trimmed.length)

                // If the clearName contains '(', then this is for a method instead of a
                // field.
                if (clearName.indexOf('(') == -1) {
                    classData.addField(obfuscatedName, clearName)
                } else {
                    // For methods, the type is of the form: [#:[#:]]
                    var obfuscatedLine = 0
                    var colon = type.indexOf(':')
                    if (colon != -1) {
                        obfuscatedLine = type.substring(0, colon).toInt()
                        type = type.substring(colon + 1)
                    }
                    colon = type.indexOf(':')
                    if (colon != -1) {
                        type = type.substring(colon + 1)
                    }

                    // For methods, the clearName is of the form: [:#[:#]]
                    val op = clearName.indexOf('(')
                    val cp = clearName.indexOf(')')
                    if (op == -1 || cp == -1) {
                        parseException("Error parse method line: '$line'")
                    }
                    val sig = clearName.substring(op, cp + 1)
                    var clearLine = obfuscatedLine
                    colon = clearName.lastIndexOf(':')
                    if (colon != -1) {
                        clearLine = clearName.substring(colon + 1).toInt()
                        clearName = clearName.substring(0, colon)
                    }
                    colon = clearName.lastIndexOf(':')
                    if (colon != -1) {
                        clearLine = clearName.substring(colon + 1).toInt()
                        clearName = clearName.substring(0, colon)
                    }
                    clearName = clearName.substring(0, op)
                    val clearSig = fromProguardSignature(sig + type)
                    classData.addFrame(
                        obfuscatedName,
                        clearName,
                        clearSig,
                        obfuscatedLine,
                        clearLine
                    )
                }
                line = reader.readLine()
            }
        }
        reader.close()
    }

    // Returns the deobfuscated version of the given class name. If no
    // deobfuscated version is known, the original string is returned.
    fun getClassName(obfuscatedClassName: String): String {
        // Class names for arrays may have trailing [] that need to be
        // stripped before doing the lookup.
        var baseName = obfuscatedClassName
        var arraySuffix = ""
        while (baseName.endsWith(ARRAY_SYMBOL)) {
            arraySuffix += ARRAY_SYMBOL
            baseName = baseName.substring(0, baseName.length - ARRAY_SYMBOL.length)
        }
        val classData = mClassesFromObfuscatedName[baseName]
        val clearBaseName = classData?.clearName ?: baseName
        return clearBaseName + arraySuffix
    }

    // Returns the deobfuscated version of the given field name for the given
    // (clear) class name. If no deobfuscated version is known, the original
    // string is returned.
    fun getFieldName(clearClass: String, obfuscatedField: String): String {
        val classData = mClassesFromClearName[clearClass] ?: return obfuscatedField
        return classData.getField(obfuscatedField)
    }

    // Returns the deobfuscated frame for the given obfuscated frame and (clear)
    // class name. As much of the frame is deobfuscated as can be.
    fun getFrame(
        clearClassName: String,
        obfuscatedMethodName: String,
        obfuscatedSignature: String,
        obfuscatedFilename: String,
        obfuscatedLine: Int
    ): Frame {
        val clearSignature = getSignature(obfuscatedSignature)
        val classData = mClassesFromClearName[clearClassName]
            ?: return Frame(
                obfuscatedMethodName,
                clearSignature,
                obfuscatedFilename,
                obfuscatedLine
            )
        return classData.getFrame(
            clearClassName,
            obfuscatedMethodName,
            clearSignature,
            obfuscatedFilename,
            obfuscatedLine
        )
    }

    // Return a clear signature for the given obfuscated signature.
    private fun getSignature(obfuscatedSig: String): String {
        val builder = StringBuilder()
        var i = 0
        while (i < obfuscatedSig.length) {
            if (obfuscatedSig[i] == 'L') {
                val e = obfuscatedSig.indexOf(';', i)
                builder.append('L')
                val cls = obfuscatedSig.substring(i + 1, e).replace('/', '.')
                builder.append(getClassName(cls).replace('.', '/'))
                builder.append(';')
                i = e
            } else {
                builder.append(obfuscatedSig[i])
            }
            i++
        }
        return builder.toString()
    }

    companion object {
        private const val ARRAY_SYMBOL = "[]"

        @Throws(ParseException::class)
        private fun parseException(msg: String) {
            throw ParseException(msg, 0)
        }

        // Converts a proguard-formatted method signature into a Java formatted
        // method signature.
        @Throws(ParseException::class)
        private fun fromProguardSignature(sig: String): String {
            return if (sig.startsWith("(")) {
                val end = sig.indexOf(')')
                if (end == -1) {
                    parseException("Error parsing signature: $sig")
                }
                val converted = StringBuilder()
                converted.append('(')
                if (end > 1) {
                    for (arg in sig.substring(1, end).split(",".toRegex()).dropLastWhile { it.isEmpty() }
                        .toTypedArray()) {
                        converted.append(fromProguardSignature(arg))
                    }
                }
                converted.append(')')
                converted.append(fromProguardSignature(sig.substring(end + 1)))
                converted.toString()
            } else if (sig.endsWith(ARRAY_SYMBOL)) {
                "[" + fromProguardSignature(sig.substring(0, sig.length - 2))
            } else if (sig == "boolean") {
                "Z"
            } else if (sig == "byte") {
                "B"
            } else if (sig == "char") {
                "C"
            } else if (sig == "short") {
                "S"
            } else if (sig == "int") {
                "I"
            } else if (sig == "long") {
                "J"
            } else if (sig == "float") {
                "F"
            } else if (sig == "double") {
                "D"
            } else if (sig == "void") {
                "V"
            } else {
                "L" + sig.replace('.', '/') + ";"
            }
        }

        // Return a file name for the given clear class name.
        private fun getFileName(clearClass: String): String {
            var filename = clearClass
            val dot = filename.lastIndexOf('.')
            if (dot != -1) {
                filename = filename.substring(dot + 1)
            }
            val dollar = filename.indexOf('$')
            if (dollar != -1) {
                filename = filename.substring(0, dollar)
            }
            return "$filename.java"
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy