|
1 | 1 | package inc.whew.android.fakegapps; |
2 | 2 |
|
| 3 | +import android.annotation.TargetApi; |
3 | 4 | import android.content.pm.PackageInfo; |
4 | 5 | import android.content.pm.Signature; |
| 6 | +import android.content.pm.SigningInfo; |
5 | 7 | import android.os.Build; |
| 8 | +import android.util.ArraySet; |
| 9 | +import android.util.Base64; |
| 10 | + |
| 11 | +import java.io.ByteArrayInputStream; |
| 12 | +import java.lang.reflect.Constructor; |
| 13 | +import java.security.cert.Certificate; |
| 14 | +import java.security.cert.CertificateException; |
| 15 | +import java.security.cert.CertificateFactory; |
| 16 | +import java.security.PublicKey; |
6 | 17 |
|
7 | 18 | import de.robv.android.xposed.IXposedHookLoadPackage; |
8 | 19 | import de.robv.android.xposed.XC_MethodHook; |
|
11 | 22 | import de.robv.android.xposed.callbacks.XC_LoadPackage.LoadPackageParam; |
12 | 23 |
|
13 | 24 | public class FakeSignatures implements IXposedHookLoadPackage { |
| 25 | + private static final String TAG = "FakeGApps"; |
| 26 | + private static final String _x509cert = "MIIEQzCCAyugAwIBAgIJAMLgh0ZkSjCNMA0GCSqGSIb3DQEBBAUAMHQxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpDYWxpZm9ybmlhMRYwFAYDVQQHEw1Nb3VudGFpbiBWaWV3MRQwEgYDVQQKEwtHb29nbGUgSW5jLjEQMA4GA1UECxMHQW5kcm9pZDEQMA4GA1UEAxMHQW5kcm9pZDAeFw0wODA4MjEyMzEzMzRaFw0zNjAxMDcyMzEzMzRaMHQxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpDYWxpZm9ybmlhMRYwFAYDVQQHEw1Nb3VudGFpbiBWaWV3MRQwEgYDVQQKEwtHb29nbGUgSW5jLjEQMA4GA1UECxMHQW5kcm9pZDEQMA4GA1UEAxMHQW5kcm9pZDCCASAwDQYJKoZIhvcNAQEBBQADggENADCCAQgCggEBAKtWLgDYO6IIrgqWbxJOKdoR8qtW0I9Y4sypEwPpt1TTcvZApxsdyxMJZ2JORland2qSGT2y5b+3JKkedxiLDmpHpDsz2WCbdxgxRczfey5YZnTJ4VZbH0xqWVW/8lGmPav5xVwnIiJS6HXk+BVKZF+JcWjAsb/GEuq/eFdpuzSqeYTcfi6idkyugwfYwXFU1+5fZKUaRKYCwkkFQVfcAs1fXA5V+++FGfvjJ/CxURaSxaBvGdGDhfXE28LWuT9ozCl5xw4Yq5OGazvV24mZVSoOO0yZ31j7kYvtwYK6NeADwbSxDdJEqO4k//0zOHKrUiGYXtqw/A0LFFtqoZKFjnkCAQOjgdkwgdYwHQYDVR0OBBYEFMd9jMIhF1Ylmn/Tgt9r45jk14alMIGmBgNVHSMEgZ4wgZuAFMd9jMIhF1Ylmn/Tgt9r45jk14aloXikdjB0MQswCQYDVQQGEwJVUzETMBEGA1UECBMKQ2FsaWZvcm5pYTEWMBQGA1UEBxMNTW91bnRhaW4gVmlldzEUMBIGA1UEChMLR29vZ2xlIEluYy4xEDAOBgNVBAsTB0FuZHJvaWQxEDAOBgNVBAMTB0FuZHJvaWSCCQDC4IdGZEowjTAMBgNVHRMEBTADAQH/MA0GCSqGSIb3DQEBBAUAA4IBAQBt0lLO74UwLDYKqs6Tm8/yzKkEu116FmH4rkaymUIE0P9KaMftGlMexFlaYjzmB2OxZyl6euNXEsQH8gjwyxCUKRJNexBiGcCEyj6z+a1fuHHvkiaai+KL8W1EyNmgjmyy8AW7P+LLlkR+ho5zEHatRbM/YAnqGcFh5iZBqpknHf1SKMXFh4dd239FJ1jWYfbMDMy3NS5CTMQ2XFI1MvcyUTdZPErjQfTbQe3aDQsQcafEQPD+nqActifKZ0Np0IS9L9kR/wbNvyz6ENwPiTrjV2KRkEjH78ZMcUQXg0L3BYHJ3lc69Vs5Ddf9uUGGMYldX3WfMBEmh/9iFBDAaTCK"; |
| 27 | + |
14 | 28 | @Override |
15 | | - public void handleLoadPackage(LoadPackageParam loadedPackage) { |
| 29 | + public void handleLoadPackage(LoadPackageParam loadedPackage) throws CertificateException { |
16 | 30 | if (!loadedPackage.packageName.equals("android")) |
17 | 31 | return; |
18 | 32 |
|
| 33 | + final byte[] certBytes = Base64.decode(_x509cert, Base64.DEFAULT); |
| 34 | + final CertificateFactory certFactory = CertificateFactory.getInstance("X.509"); |
| 35 | + final Certificate cert = certFactory.generateCertificate(new ByteArrayInputStream(certBytes)); |
| 36 | + |
19 | 37 | XC_MethodHook hook = new XC_MethodHook() { |
20 | 38 | @Override |
21 | 39 | protected void afterHookedMethod(MethodHookParam param) { |
22 | 40 | PackageInfo pi = (PackageInfo) param.getResult(); |
23 | 41 | if (pi != null) { |
24 | 42 | String packageName = pi.packageName; |
25 | 43 | if (packageName.equals("com.google.android.gms") || packageName.equals("com.android.vending")) { |
26 | | - String sig = "308204433082032ba003020102020900c2e08746644a308d300d06092a864886f70d01010405003074310b3009060355040613025553311330110603550408130a43616c69666f726e6961311630140603550407130d4d6f756e7461696e205669657731143012060355040a130b476f6f676c6520496e632e3110300e060355040b1307416e64726f69643110300e06035504031307416e64726f6964301e170d3038303832313233313333345a170d3336303130373233313333345a3074310b3009060355040613025553311330110603550408130a43616c69666f726e6961311630140603550407130d4d6f756e7461696e205669657731143012060355040a130b476f6f676c6520496e632e3110300e060355040b1307416e64726f69643110300e06035504031307416e64726f696430820120300d06092a864886f70d01010105000382010d00308201080282010100ab562e00d83ba208ae0a966f124e29da11f2ab56d08f58e2cca91303e9b754d372f640a71b1dcb130967624e4656a7776a92193db2e5bfb724a91e77188b0e6a47a43b33d9609b77183145ccdf7b2e586674c9e1565b1f4c6a5955bff251a63dabf9c55c27222252e875e4f8154a645f897168c0b1bfc612eabf785769bb34aa7984dc7e2ea2764cae8307d8c17154d7ee5f64a51a44a602c249054157dc02cd5f5c0e55fbef8519fbe327f0b1511692c5a06f19d18385f5c4dbc2d6b93f68cc2979c70e18ab93866b3bd5db8999552a0e3b4c99df58fb918bedc182ba35e003c1b4b10dd244a8ee24fffd333872ab5221985edab0fc0d0b145b6aa192858e79020103a381d93081d6301d0603551d0e04160414c77d8cc2211756259a7fd382df6be398e4d786a53081a60603551d2304819e30819b8014c77d8cc2211756259a7fd382df6be398e4d786a5a178a4763074310b3009060355040613025553311330110603550408130a43616c69666f726e6961311630140603550407130d4d6f756e7461696e205669657731143012060355040a130b476f6f676c6520496e632e3110300e060355040b1307416e64726f69643110300e06035504031307416e64726f6964820900c2e08746644a308d300c0603551d13040530030101ff300d06092a864886f70d010104050003820101006dd252ceef85302c360aaace939bcff2cca904bb5d7a1661f8ae46b2994204d0ff4a68c7ed1a531ec4595a623ce60763b167297a7ae35712c407f208f0cb109429124d7b106219c084ca3eb3f9ad5fb871ef92269a8be28bf16d44c8d9a08e6cb2f005bb3fe2cb96447e868e731076ad45b33f6009ea19c161e62641aa99271dfd5228c5c587875ddb7f452758d661f6cc0cccb7352e424cc4365c523532f7325137593c4ae341f4db41edda0d0b1071a7c440f0fe9ea01cb627ca674369d084bd2fd911ff06cdbf2cfa10dc0f893ae35762919048c7efc64c7144178342f70581c9de573af55b390dd7fdb9418631895d5f759f30112687ff621410c069308a"; |
| 44 | + pi.signatures = new Signature[]{new Signature(certBytes)}; |
| 45 | + |
| 46 | + if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.P) { |
| 47 | + SigningInfo signingInfo = createSigningInfo(new Signature(certBytes), cert.getPublicKey()); |
| 48 | + if (signingInfo != null) { |
| 49 | + pi.signingInfo = signingInfo; |
| 50 | + } |
| 51 | + } |
27 | 52 |
|
28 | | - pi.signatures = new Signature[]{new Signature(sig)}; |
29 | 53 | param.setResult(pi); |
30 | 54 | } |
31 | 55 | } |
@@ -80,4 +104,37 @@ protected void afterHookedMethod(MethodHookParam param) { |
80 | 104 | final Class<?> hookedClass = XposedHelpers.findClass(classToHook, loadedPackage.classLoader); |
81 | 105 | XposedBridge.hookAllMethods(hookedClass, "generatePackageInfo", hook); |
82 | 106 | } |
| 107 | + |
| 108 | + @TargetApi(android.os.Build.VERSION_CODES.P) |
| 109 | + private SigningInfo createSigningInfo(Signature sig, PublicKey publicKey) { |
| 110 | + final int SIGNING_BLOCK_V3 = 3; |
| 111 | + final Signature[] sigs = new Signature[]{sig}; |
| 112 | + final ArraySet<PublicKey> pks = new ArraySet<>(); |
| 113 | + pks.add(publicKey); |
| 114 | + |
| 115 | + // Unfortunately, SigningDetails is not exported in SDK, so we have to rely on reflection. |
| 116 | + // Also, public SigningInfo constructor is only available from API 35, so we can't use it. |
| 117 | + try { |
| 118 | + Class<?> signingDetailsClass = Class.forName("android.content.pm.SigningDetails"); |
| 119 | + // https://cs.android.com/android/platform/superproject/+/1c19b376095446666df2b2d9290dac3ef71da846:frameworks/base/core/java/android/content/pm/SigningDetails.java;l=146 |
| 120 | + Constructor<?> signingDetailsConstructor = signingDetailsClass.getDeclaredConstructor( |
| 121 | + Signature[].class, // signatures |
| 122 | + int.class, // signatureSchemeVersion |
| 123 | + ArraySet.class, // keys |
| 124 | + Signature[].class // pastSigningCertificates |
| 125 | + ); |
| 126 | + Constructor<SigningInfo> signingInfoConstructor = SigningInfo.class.getDeclaredConstructor(signingDetailsClass); |
| 127 | + |
| 128 | + signingDetailsConstructor.setAccessible(true); |
| 129 | + signingInfoConstructor.setAccessible(true); |
| 130 | + |
| 131 | + Object signingDetails = signingDetailsConstructor.newInstance(sigs, SIGNING_BLOCK_V3, pks, null); |
| 132 | + return signingInfoConstructor.newInstance(signingDetails); |
| 133 | + } catch (Exception e) { |
| 134 | + XposedBridge.log(String.format("%s failed to create signingInfo", TAG)); |
| 135 | + XposedBridge.log(e); |
| 136 | + } |
| 137 | + |
| 138 | + return null; |
| 139 | + } |
83 | 140 | } |
0 commit comments