mirror of
https://github.com/topjohnwu/Magisk.git
synced 2024-12-26 21:17:38 +00:00
Move functions
This commit is contained in:
parent
963edfe8ab
commit
d4798b02ac
@ -7,15 +7,21 @@ import android.widget.Toast;
|
|||||||
|
|
||||||
import com.topjohnwu.magisk.MagiskManager;
|
import com.topjohnwu.magisk.MagiskManager;
|
||||||
import com.topjohnwu.magisk.R;
|
import com.topjohnwu.magisk.R;
|
||||||
|
import com.topjohnwu.magisk.container.JarMap;
|
||||||
import com.topjohnwu.magisk.container.Policy;
|
import com.topjohnwu.magisk.container.Policy;
|
||||||
import com.topjohnwu.magisk.utils.Utils;
|
import com.topjohnwu.magisk.utils.Utils;
|
||||||
import com.topjohnwu.magisk.utils.ZipUtils;
|
import com.topjohnwu.magisk.utils.ZipUtils;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.jar.JarEntry;
|
||||||
|
|
||||||
public class HideManager extends ParallelTask<Void, Void, Boolean> {
|
public class HideManager extends ParallelTask<Void, Void, Boolean> {
|
||||||
|
|
||||||
|
private static final String UNHIDE_APK = "unhide.apk";
|
||||||
|
private static final String ANDROID_MANIFEST = "AndroidManifest.xml";
|
||||||
|
private static final byte[] UNHIDE_PKG_NAME = "com.topjohnwu.unhide\0".getBytes();
|
||||||
|
|
||||||
public HideManager(Context context) {
|
public HideManager(Context context) {
|
||||||
super(context);
|
super(context);
|
||||||
}
|
}
|
||||||
@ -34,7 +40,42 @@ public class HideManager extends ParallelTask<Void, Void, Boolean> {
|
|||||||
// Generate a new unhide app with random package name
|
// Generate a new unhide app with random package name
|
||||||
File unhideAPK = new File(Environment.getExternalStorageDirectory() + "/MagiskManager", "unhide.apk");
|
File unhideAPK = new File(Environment.getExternalStorageDirectory() + "/MagiskManager", "unhide.apk");
|
||||||
unhideAPK.getParentFile().mkdirs();
|
unhideAPK.getParentFile().mkdirs();
|
||||||
String pkg = ZipUtils.generateUnhide(mm, unhideAPK);
|
String pkg;
|
||||||
|
|
||||||
|
try {
|
||||||
|
JarMap asset = new JarMap(mm.getAssets().open(UNHIDE_APK));
|
||||||
|
JarEntry je = new JarEntry(ANDROID_MANIFEST);
|
||||||
|
byte xml[] = asset.getRawData(je);
|
||||||
|
int offset = -1;
|
||||||
|
|
||||||
|
// Linear search pattern offset
|
||||||
|
for (int i = 0; i < xml.length - UNHIDE_PKG_NAME.length; ++i) {
|
||||||
|
boolean match = true;
|
||||||
|
for (int j = 0; j < UNHIDE_PKG_NAME.length; ++j) {
|
||||||
|
if (xml[i + j] != UNHIDE_PKG_NAME[j]) {
|
||||||
|
match = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (match) {
|
||||||
|
offset = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (offset < 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Patch binary XML with new package name
|
||||||
|
pkg = Utils.genPackageName("com.", UNHIDE_PKG_NAME.length - 1);
|
||||||
|
System.arraycopy(pkg.getBytes(), 0, xml, offset, pkg.length());
|
||||||
|
asset.getOutputStream(je).write(xml);
|
||||||
|
|
||||||
|
// Sign the APK
|
||||||
|
ZipUtils.signZip(mm, asset, unhideAPK, false);
|
||||||
|
} catch (Exception e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
// Install the application
|
// Install the application
|
||||||
List<String> ret = getShell().su("pm install " + unhideAPK + ">/dev/null && echo true || echo false");
|
List<String> ret = getShell().su("pm install " + unhideAPK + ">/dev/null && echo true || echo false");
|
||||||
|
@ -11,7 +11,6 @@ import android.widget.Toast;
|
|||||||
|
|
||||||
import com.topjohnwu.magisk.FlashActivity;
|
import com.topjohnwu.magisk.FlashActivity;
|
||||||
import com.topjohnwu.magisk.R;
|
import com.topjohnwu.magisk.R;
|
||||||
import com.topjohnwu.magisk.utils.Logger;
|
|
||||||
import com.topjohnwu.magisk.utils.Shell;
|
import com.topjohnwu.magisk.utils.Shell;
|
||||||
import com.topjohnwu.magisk.utils.Utils;
|
import com.topjohnwu.magisk.utils.Utils;
|
||||||
import com.topjohnwu.magisk.utils.WebService;
|
import com.topjohnwu.magisk.utils.WebService;
|
||||||
@ -26,6 +25,9 @@ import java.io.IOException;
|
|||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.io.OutputStream;
|
import java.io.OutputStream;
|
||||||
import java.net.HttpURLConnection;
|
import java.net.HttpURLConnection;
|
||||||
|
import java.util.jar.JarEntry;
|
||||||
|
import java.util.jar.JarInputStream;
|
||||||
|
import java.util.jar.JarOutputStream;
|
||||||
|
|
||||||
public class ProcessRepoZip extends ParallelTask<Void, Object, Boolean> {
|
public class ProcessRepoZip extends ParallelTask<Void, Object, Boolean> {
|
||||||
|
|
||||||
@ -46,6 +48,34 @@ public class ProcessRepoZip extends ParallelTask<Void, Object, Boolean> {
|
|||||||
mInstall = install;
|
mInstall = install;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void removeTopFolder(InputStream in, File output) throws IOException {
|
||||||
|
JarInputStream source = new JarInputStream(in);
|
||||||
|
JarOutputStream dest = new JarOutputStream(new BufferedOutputStream(new FileOutputStream(output)));
|
||||||
|
JarEntry entry;
|
||||||
|
String path;
|
||||||
|
int size;
|
||||||
|
byte buffer[] = new byte[4096];
|
||||||
|
while ((entry = source.getNextJarEntry()) != null) {
|
||||||
|
// Remove the top directory from the path
|
||||||
|
path = entry.getName().substring(entry.getName().indexOf("/") + 1);
|
||||||
|
// If it's the top folder, ignore it
|
||||||
|
if (path.isEmpty()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
// Don't include placeholder
|
||||||
|
if (path.equals("system/placeholder")) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
dest.putNextEntry(new JarEntry(path));
|
||||||
|
while((size = source.read(buffer)) != -1) {
|
||||||
|
dest.write(buffer, 0, size);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
source.close();
|
||||||
|
dest.close();
|
||||||
|
in.close();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onPreExecute() {
|
protected void onPreExecute() {
|
||||||
Activity activity = getActivity();
|
Activity activity = getActivity();
|
||||||
@ -86,7 +116,7 @@ public class ProcessRepoZip extends ParallelTask<Void, Object, Boolean> {
|
|||||||
temp1.getParentFile().mkdir();
|
temp1.getParentFile().mkdir();
|
||||||
|
|
||||||
// First remove top folder in Github source zip, Web -> temp1
|
// First remove top folder in Github source zip, Web -> temp1
|
||||||
ZipUtils.removeTopFolder(in, temp1);
|
removeTopFolder(in, temp1);
|
||||||
|
|
||||||
conn.disconnect();
|
conn.disconnect();
|
||||||
publishProgress(SHOW_PROCESSING);
|
publishProgress(SHOW_PROCESSING);
|
||||||
@ -116,7 +146,6 @@ public class ProcessRepoZip extends ParallelTask<Void, Object, Boolean> {
|
|||||||
|
|
||||||
return true;
|
return true;
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
Logger.error("ProcessRepoZip: Error!");
|
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -144,7 +173,8 @@ public class ProcessRepoZip extends ParallelTask<Void, Object, Boolean> {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ParallelTask<Void, Object, Boolean> exec(Void... voids) {
|
public ParallelTask<Void, Object, Boolean> exec(Void... voids) {
|
||||||
Utils.runWithPermission(getActivity(), Manifest.permission.WRITE_EXTERNAL_STORAGE, () -> super.exec(voids));
|
Utils.runWithPermission(getActivity(), Manifest.permission.WRITE_EXTERNAL_STORAGE,
|
||||||
|
() -> super.exec(voids));
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -70,14 +70,10 @@ public class ZipUtils {
|
|||||||
// File name in assets
|
// File name in assets
|
||||||
private static final String PUBLIC_KEY_NAME = "public.certificate.x509.pem";
|
private static final String PUBLIC_KEY_NAME = "public.certificate.x509.pem";
|
||||||
private static final String PRIVATE_KEY_NAME = "private.key.pk8";
|
private static final String PRIVATE_KEY_NAME = "private.key.pk8";
|
||||||
private static final String UNHIDE_APK = "unhide.apk";
|
|
||||||
|
|
||||||
private static final String CERT_SF_NAME = "META-INF/CERT.SF";
|
private static final String CERT_SF_NAME = "META-INF/CERT.SF";
|
||||||
private static final String CERT_SIG_NAME = "META-INF/CERT.%s";
|
private static final String CERT_SIG_NAME = "META-INF/CERT.%s";
|
||||||
|
|
||||||
private static final String ANDROID_MANIFEST = "AndroidManifest.xml";
|
|
||||||
private static final byte[] UNHIDE_PKG_NAME = "com.topjohnwu.unhide\0".getBytes();
|
|
||||||
|
|
||||||
private static Provider sBouncyCastleProvider;
|
private static Provider sBouncyCastleProvider;
|
||||||
// bitmasks for which hash algorithms we need the manifest to include.
|
// bitmasks for which hash algorithms we need the manifest to include.
|
||||||
private static final int USE_SHA1 = 1;
|
private static final int USE_SHA1 = 1;
|
||||||
@ -91,78 +87,6 @@ public class ZipUtils {
|
|||||||
|
|
||||||
public native static void zipAdjust(String filenameIn, String filenameOut);
|
public native static void zipAdjust(String filenameIn, String filenameOut);
|
||||||
|
|
||||||
public static String generateUnhide(Context context, File output) {
|
|
||||||
try {
|
|
||||||
String pkg;
|
|
||||||
JarMap apk = new JarMap(context.getAssets().open(UNHIDE_APK));
|
|
||||||
JarEntry je = new JarEntry(ANDROID_MANIFEST);
|
|
||||||
byte xml[] = apk.getRawData(je);
|
|
||||||
int offset = -1;
|
|
||||||
|
|
||||||
// Linear search pattern offset
|
|
||||||
for (int i = 0; i < xml.length - UNHIDE_PKG_NAME.length; ++i) {
|
|
||||||
boolean match = true;
|
|
||||||
for (int j = 0; j < UNHIDE_PKG_NAME.length; ++j) {
|
|
||||||
if (xml[i + j] != UNHIDE_PKG_NAME[j]) {
|
|
||||||
match = false;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (match) {
|
|
||||||
offset = i;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (offset < 0)
|
|
||||||
return "";
|
|
||||||
|
|
||||||
// Patch binary XML with new package name
|
|
||||||
pkg = Utils.genPackageName("com.", UNHIDE_PKG_NAME.length - 1);
|
|
||||||
System.arraycopy(pkg.getBytes(), 0, xml, offset, pkg.length());
|
|
||||||
apk.getOutputStream(je).write(xml);
|
|
||||||
|
|
||||||
// Sign the APK
|
|
||||||
signZip(context, apk, output, false);
|
|
||||||
|
|
||||||
return pkg;
|
|
||||||
} catch (IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void removeTopFolder(InputStream in, File output) throws IOException {
|
|
||||||
try {
|
|
||||||
JarInputStream source = new JarInputStream(in);
|
|
||||||
JarOutputStream dest = new JarOutputStream(new BufferedOutputStream(new FileOutputStream(output)));
|
|
||||||
JarEntry entry;
|
|
||||||
String path;
|
|
||||||
int size;
|
|
||||||
byte buffer[] = new byte[4096];
|
|
||||||
while ((entry = source.getNextJarEntry()) != null) {
|
|
||||||
// Remove the top directory from the path
|
|
||||||
path = entry.getName().substring(entry.getName().indexOf("/") + 1);
|
|
||||||
// If it's the top folder, ignore it
|
|
||||||
if (path.isEmpty()) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
// Don't include placeholder
|
|
||||||
if (path.equals("system/placeholder")) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
dest.putNextEntry(new JarEntry(path));
|
|
||||||
while((size = source.read(buffer)) != -1) {
|
|
||||||
dest.write(buffer, 0, size);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
source.close();
|
|
||||||
dest.close();
|
|
||||||
in.close();
|
|
||||||
} catch (IOException e) {
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void unzip(File zip, File folder, String path, boolean junkPath) throws Exception {
|
public static void unzip(File zip, File folder, String path, boolean junkPath) throws Exception {
|
||||||
InputStream in = new BufferedInputStream(new FileInputStream(zip));
|
InputStream in = new BufferedInputStream(new FileInputStream(zip));
|
||||||
unzip(in, folder, path, junkPath);
|
unzip(in, folder, path, junkPath);
|
||||||
@ -201,63 +125,46 @@ public class ZipUtils {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void signZip(Context context, InputStream is, File output, boolean minSign) {
|
public static void signZip(Context context, InputStream is, File output, boolean minSign) throws Exception {
|
||||||
try {
|
signZip(context, new JarMap(is, false), output, minSign);
|
||||||
signZip(context, new JarMap(is, false), output, minSign);
|
|
||||||
} catch (IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void signZip(Context context, File input, File output, boolean minSign) {
|
public static void signZip(Context context, File input, File output, boolean minSign) throws Exception {
|
||||||
try {
|
signZip(context, new JarMap(new FileInputStream(input), false), output, minSign);
|
||||||
signZip(context, new JarMap(new FileInputStream(input), false), output, minSign);
|
|
||||||
} catch (IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void signZip(Context context, JarMap input, File output, boolean minSign) {
|
public static void signZip(Context context, JarMap input, File output, boolean minSign) throws Exception {
|
||||||
int alignment = 4;
|
int alignment = 4;
|
||||||
BufferedOutputStream outputFile = null;
|
BufferedOutputStream outputFile = null;
|
||||||
int hashes = 0;
|
int hashes = 0;
|
||||||
try {
|
X509Certificate publicKey = readPublicKey(context.getAssets().open(PUBLIC_KEY_NAME));
|
||||||
X509Certificate publicKey = readPublicKey(context.getAssets().open(PUBLIC_KEY_NAME));
|
hashes |= getDigestAlgorithm(publicKey);
|
||||||
hashes |= getDigestAlgorithm(publicKey);
|
|
||||||
|
|
||||||
// Set the ZIP file timestamp to the starting valid time
|
// Set the ZIP file timestamp to the starting valid time
|
||||||
// of the 0th certificate plus one hour (to match what
|
// of the 0th certificate plus one hour (to match what
|
||||||
// we've historically done).
|
// we've historically done).
|
||||||
long timestamp = publicKey.getNotBefore().getTime() + 3600L * 1000;
|
long timestamp = publicKey.getNotBefore().getTime() + 3600L * 1000;
|
||||||
PrivateKey privateKey = readPrivateKey(context.getAssets().open(PRIVATE_KEY_NAME));
|
PrivateKey privateKey = readPrivateKey(context.getAssets().open(PRIVATE_KEY_NAME));
|
||||||
|
|
||||||
outputFile = new BufferedOutputStream(new FileOutputStream(output));
|
outputFile = new BufferedOutputStream(new FileOutputStream(output));
|
||||||
if (minSign) {
|
if (minSign) {
|
||||||
ZipUtils.signWholeFile(input.getInputStream(), publicKey, privateKey, outputFile);
|
ZipUtils.signWholeFile(input.getInputStream(), publicKey, privateKey, outputFile);
|
||||||
} else {
|
} else {
|
||||||
JarOutputStream outputJar = new JarOutputStream(outputFile);
|
JarOutputStream outputJar = new JarOutputStream(outputFile);
|
||||||
// For signing .apks, use the maximum compression to make
|
// For signing .apks, use the maximum compression to make
|
||||||
// them as small as possible (since they live forever on
|
// them as small as possible (since they live forever on
|
||||||
// the system partition). For OTA packages, use the
|
// the system partition). For OTA packages, use the
|
||||||
// default compression level, which is much much faster
|
// default compression level, which is much much faster
|
||||||
// and produces output that is only a tiny bit larger
|
// and produces output that is only a tiny bit larger
|
||||||
// (~0.1% on full OTA packages I tested).
|
// (~0.1% on full OTA packages I tested).
|
||||||
outputJar.setLevel(9);
|
outputJar.setLevel(9);
|
||||||
Manifest manifest = addDigestsToManifest(input, hashes);
|
Manifest manifest = addDigestsToManifest(input, hashes);
|
||||||
copyFiles(manifest, input, outputJar, timestamp, alignment);
|
copyFiles(manifest, input, outputJar, timestamp, alignment);
|
||||||
signFile(manifest, input, publicKey, privateKey, outputJar);
|
signFile(manifest, input, publicKey, privateKey, outputJar);
|
||||||
outputJar.close();
|
outputJar.close();
|
||||||
}
|
|
||||||
} catch (Exception e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
} finally {
|
|
||||||
try {
|
|
||||||
if (input != null) input.close();
|
|
||||||
if (outputFile != null) outputFile.close();
|
|
||||||
} catch (IOException e) {
|
|
||||||
e.printStackTrace();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
input.close();
|
||||||
|
outputFile.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user