_v
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
DLAali
literal 0
HcmV?d00001
diff --git a/service-management/src/main/java/com/weishu/upf/service_management/app/ProxyService.java b/service-management/src/main/java/com/weishu/upf/service_management/app/ProxyService.java
index df53b09..587c8cb 100644
--- a/service-management/src/main/java/com/weishu/upf/service_management/app/ProxyService.java
+++ b/service-management/src/main/java/com/weishu/upf/service_management/app/ProxyService.java
@@ -22,12 +22,15 @@ public void onCreate() {
@Override
public void onStart(Intent intent, int startId) {
Log.d(TAG, "onStart() called with " + "intent = [" + intent + "], startId = [" + startId + "]");
+
+ // 分发Service
ServiceManager.getInstance().onStart(intent, startId);
super.onStart(intent, startId);
}
@Override
public IBinder onBind(Intent intent) {
+ // TODO: 16/5/11 bindService实现
return null;
}
diff --git a/service-management/src/main/java/com/weishu/upf/service_management/app/ServiceManager.java b/service-management/src/main/java/com/weishu/upf/service_management/app/ServiceManager.java
index a964fe2..b78451b 100644
--- a/service-management/src/main/java/com/weishu/upf/service_management/app/ServiceManager.java
+++ b/service-management/src/main/java/com/weishu/upf/service_management/app/ServiceManager.java
@@ -50,7 +50,6 @@ public synchronized static ServiceManager getInstance() {
public void onStart(Intent proxyIntent, int startId) {
Intent targetIntent = proxyIntent.getParcelableExtra(AMSHookHelper.EXTRA_TARGET_INTENT);
-
ServiceInfo serviceInfo = selectPluginService(targetIntent);
if (serviceInfo == null) {
diff --git a/service-management/src/main/java/com/weishu/upf/service_management/app/UPFApplication.java b/service-management/src/main/java/com/weishu/upf/service_management/app/UPFApplication.java
index 121cee7..9b86578 100644
--- a/service-management/src/main/java/com/weishu/upf/service_management/app/UPFApplication.java
+++ b/service-management/src/main/java/com/weishu/upf/service_management/app/UPFApplication.java
@@ -26,8 +26,8 @@ protected void attachBaseContext(Context base) {
try {
// 拦截startService, stopService等操作
AMSHookHelper.hookActivityManagerNative();
- Utils.extractAssets(base, "test.apk");
- File apkFile = getFileStreamPath("test.apk");
+ Utils.extractAssets(base, "test.jar");
+ File apkFile = getFileStreamPath("test.jar");
File odexFile = getFileStreamPath("test.odex");
// Hook ClassLoader, 让插件中的类能够被成功加载
From 193f38a4297ac3218edb281b34d44c43920cfaaa Mon Sep 17 00:00:00 2001
From: tiann <923551233@qq.com>
Date: Fri, 27 May 2016 10:28:20 +0800
Subject: [PATCH 04/11] add detail description for binder hook; tks
jeremyhe_cn@qq.com.
---
.../binder_hook/app/BinderProxyHookHandler.java | 16 +++++++++++-----
1 file changed, 11 insertions(+), 5 deletions(-)
diff --git a/binder-hook/src/main/java/com/weishu/binder_hook/app/BinderProxyHookHandler.java b/binder-hook/src/main/java/com/weishu/binder_hook/app/BinderProxyHookHandler.java
index 28a1eed..cfbe99a 100644
--- a/binder-hook/src/main/java/com/weishu/binder_hook/app/BinderProxyHookHandler.java
+++ b/binder-hook/src/main/java/com/weishu/binder_hook/app/BinderProxyHookHandler.java
@@ -1,13 +1,12 @@
package com.weishu.binder_hook.app;
-import android.os.IBinder;
-import android.os.IInterface;
-import android.util.Log;
-
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
+import android.os.IBinder;
+import android.util.Log;
+
/**
* 由于ServiceManager里面的sCache里面存储的 IBinder类型基本上都是BinderProxy
* 因此, ServiceManager的使用者调用getService之后不会直接使用这个map
@@ -66,7 +65,14 @@ public Object invoke(Object proxy, Method method, Object[] args) throws Throwabl
// asInterface 的时候会检测是否是特定类型的接口然后进行强制转换
// 因此这里的动态代理生成的类型信息的类型必须是正确的
- new Class[] { IBinder.class, IInterface.class, this.iinterface },
+
+ // 这里面Hook的是一个BinderProxy对象(Binder代理) (代理Binder的queryLocalInterface正常情况下是返回null)
+ // 因此, 正常情况下 在asInterface里面会由于BinderProxy的queryLocalInterface返回null导致系统创建一个匿名的代理对象, 这样我们就无法控制了
+ // 所以我们要伪造一个对象, 瞒过这个if检测, 使得系统把这个queryLocalInterface返回的对象透传给asInterface的返回值;
+ // 检测有两个要求, 其一: 非空, 其二, IXXInterface类型。
+ // 所以, 其实返回的对象不需要是Binder对象, 我们把它当作普通的对象Hook掉就ok(拦截这个对象里面对于IXXInterface相关方法的调用)
+ // tks jeremyhe_cn@qq.com
+ new Class[] { this.iinterface },
new BinderHookHandler(base, stub));
}
From 89a9334cd5870b7768290d1aa48a39f26c0c6326 Mon Sep 17 00:00:00 2001
From: tiann <923551233@qq.com>
Date: Sun, 12 Jun 2016 17:08:51 +0800
Subject: [PATCH 05/11] =?UTF-8?q?fix=20the=20description=20of=20service=20?=
=?UTF-8?q?hook=20and=20the=20parse=20method;=20thx=20=E5=B0=8F=E9=BB=84?=
=?UTF-8?q?=E7=89=9B=E5=93=88=E5=93=88=E5=93=88;?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.../com/weishu/upf/service_management/app/ServiceManager.java | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/service-management/src/main/java/com/weishu/upf/service_management/app/ServiceManager.java b/service-management/src/main/java/com/weishu/upf/service_management/app/ServiceManager.java
index b78451b..6e18a2b 100644
--- a/service-management/src/main/java/com/weishu/upf/service_management/app/ServiceManager.java
+++ b/service-management/src/main/java/com/weishu/upf/service_management/app/ServiceManager.java
@@ -170,7 +170,7 @@ private void proxyCreateService(ServiceInfo serviceInfo) throws Exception {
}
/**
- * 解析Apk文件中的 , 并存储起来
+ * 解析Apk文件中的 , 并存储起来
* 主要是调用PackageParser类的generateServiceInfo方法
* @param apkFile 插件对应的apk文件
* @throws Exception 解析出错或者反射调用出错, 均会抛出异常
@@ -182,7 +182,7 @@ public void preLoadServices(File apkFile) throws Exception {
Object packageParser = packageParserClass.newInstance();
// 首先调用parsePackage获取到apk对象对应的Package对象
- Object packageObj = parsePackageMethod.invoke(packageParser, apkFile, PackageManager.GET_RECEIVERS);
+ Object packageObj = parsePackageMethod.invoke(packageParser, apkFile, PackageManager.GET_SERVICES);
// 读取Package对象里面的services字段
// 接下来要做的就是根据这个List 获取到Service对应的ServiceInfo
From 7527916802f26ac914200145b7d1a00feb742952 Mon Sep 17 00:00:00 2001
From: tiann <923551233@qq.com>
Date: Tue, 12 Jul 2016 11:52:17 +0800
Subject: [PATCH 06/11] content provider plugin demo
---
.idea/gradle.xml | 6 +-
.idea/misc.xml | 47 ++++++-
.idea/modules.xml | 2 +
.idea/uiDesigner.xml | 124 ------------------
ams-pms-hook/ams-pms-hook.iml | 31 +++--
binder-hook/binder-hook.iml | 31 +++--
classloader-hook/classloader-hook.iml | 31 +++--
contentprovider-management/.gitignore | 1 +
contentprovider-management/build.gradle | 24 ++++
.../contentprovider-management.iml | 96 ++++++++++++++
contentprovider-management/proguard-rules.pro | 17 +++
.../src/main/AndroidManifest.xml | 25 ++++
.../src/main/assets/TestContentProvider.java | 100 ++++++++++++++
.../MainActivity.java | 67 ++++++++++
.../ProviderHelper.java | 85 ++++++++++++
.../StubContentProvider.java | 88 +++++++++++++
.../UPFApplication.java | 40 ++++++
.../contentprovider_management/Utils.java | 63 +++++++++
.../hook/BaseDexClassLoaderHookHelper.java | 59 +++++++++
.../src/main/res/mipmap-hdpi/ic_launcher.png | Bin 0 -> 3418 bytes
.../src/main/res/mipmap-mdpi/ic_launcher.png | Bin 0 -> 2206 bytes
.../src/main/res/mipmap-xhdpi/ic_launcher.png | Bin 0 -> 4842 bytes
.../main/res/mipmap-xxhdpi/ic_launcher.png | Bin 0 -> 7718 bytes
.../main/res/mipmap-xxxhdpi/ic_launcher.png | Bin 0 -> 10486 bytes
.../src/main/res/values/colors.xml | 6 +
.../src/main/res/values/strings.xml | 3 +
.../src/main/res/values/styles.xml | 5 +
dynamic-proxy-hook/dynamic-proxy-hook.iml | 31 ++---
intercept-activity/intercept-activity.iml | 31 ++---
receiver-management/receiver-management.iml | 31 +++--
.../service-management.iml | 117 ++++++++++++++---
settings.gradle | 2 +-
understand-plugin-framework.iml | 8 ++
33 files changed, 935 insertions(+), 236 deletions(-)
delete mode 100644 .idea/uiDesigner.xml
create mode 100644 contentprovider-management/.gitignore
create mode 100644 contentprovider-management/build.gradle
create mode 100644 contentprovider-management/contentprovider-management.iml
create mode 100644 contentprovider-management/proguard-rules.pro
create mode 100644 contentprovider-management/src/main/AndroidManifest.xml
create mode 100644 contentprovider-management/src/main/assets/TestContentProvider.java
create mode 100644 contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/MainActivity.java
create mode 100644 contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/ProviderHelper.java
create mode 100644 contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/StubContentProvider.java
create mode 100644 contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/UPFApplication.java
create mode 100644 contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/Utils.java
create mode 100644 contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/hook/BaseDexClassLoaderHookHelper.java
create mode 100644 contentprovider-management/src/main/res/mipmap-hdpi/ic_launcher.png
create mode 100644 contentprovider-management/src/main/res/mipmap-mdpi/ic_launcher.png
create mode 100644 contentprovider-management/src/main/res/mipmap-xhdpi/ic_launcher.png
create mode 100644 contentprovider-management/src/main/res/mipmap-xxhdpi/ic_launcher.png
create mode 100644 contentprovider-management/src/main/res/mipmap-xxxhdpi/ic_launcher.png
create mode 100644 contentprovider-management/src/main/res/values/colors.xml
create mode 100644 contentprovider-management/src/main/res/values/strings.xml
create mode 100644 contentprovider-management/src/main/res/values/styles.xml
rename .idea/findbugs-idea.xml => service-management/service-management.iml (60%)
diff --git a/.idea/gradle.xml b/.idea/gradle.xml
index 4937155..d641bd5 100644
--- a/.idea/gradle.xml
+++ b/.idea/gradle.xml
@@ -5,17 +5,17 @@
-
-
@@ -24,9 +24,11 @@
+
+
diff --git a/.idea/misc.xml b/.idea/misc.xml
index b556505..cca2cda 100644
--- a/.idea/misc.xml
+++ b/.idea/misc.xml
@@ -1,13 +1,32 @@
-
-
-
-
-
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
@@ -18,10 +37,26 @@
-
+
+
+
+
+
+ 1.8
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.idea/modules.xml b/.idea/modules.xml
index 93081b8..1b7bd0d 100644
--- a/.idea/modules.xml
+++ b/.idea/modules.xml
@@ -5,9 +5,11 @@
+
+
diff --git a/.idea/uiDesigner.xml b/.idea/uiDesigner.xml
deleted file mode 100644
index e96534f..0000000
--- a/.idea/uiDesigner.xml
+++ /dev/null
@@ -1,124 +0,0 @@
-
-
-
-
- -
-
-
- -
-
-
- -
-
-
- -
-
-
- -
-
-
-
-
-
- -
-
-
-
-
-
- -
-
-
-
-
-
- -
-
-
-
-
-
- -
-
-
-
-
- -
-
-
-
-
- -
-
-
-
-
- -
-
-
-
-
- -
-
-
-
-
- -
-
-
-
-
- -
-
-
- -
-
-
-
-
- -
-
-
-
-
- -
-
-
-
-
- -
-
-
-
-
- -
-
-
-
-
- -
-
-
- -
-
-
- -
-
-
- -
-
-
- -
-
-
-
-
- -
-
-
- -
-
-
-
-
-
\ No newline at end of file
diff --git a/ams-pms-hook/ams-pms-hook.iml b/ams-pms-hook/ams-pms-hook.iml
index d66dfce..5c15004 100644
--- a/ams-pms-hook/ams-pms-hook.iml
+++ b/ams-pms-hook/ams-pms-hook.iml
@@ -12,10 +12,7 @@
-
-
- generateDebugAndroidTestSources
generateDebugSources
@@ -28,7 +25,7 @@
-
+
@@ -50,6 +47,13 @@
+
+
+
+
+
+
+
@@ -64,25 +68,26 @@
+
+
+
+
+
+
+
-
-
+
-
-
-
-
-
-
+
-
+
diff --git a/binder-hook/binder-hook.iml b/binder-hook/binder-hook.iml
index 59b6e55..f31b049 100644
--- a/binder-hook/binder-hook.iml
+++ b/binder-hook/binder-hook.iml
@@ -12,10 +12,7 @@
-
-
- generateDebugAndroidTestSources
generateDebugSources
@@ -28,7 +25,7 @@
-
+
@@ -50,6 +47,13 @@
+
+
+
+
+
+
+
@@ -57,6 +61,13 @@
+
+
+
+
+
+
+
@@ -65,24 +76,18 @@
-
-
+
-
-
-
-
-
-
+
-
+
diff --git a/classloader-hook/classloader-hook.iml b/classloader-hook/classloader-hook.iml
index c5e7cea..a31f499 100644
--- a/classloader-hook/classloader-hook.iml
+++ b/classloader-hook/classloader-hook.iml
@@ -12,10 +12,7 @@
-
-
- generateDebugAndroidTestSources
generateDebugSources
@@ -28,7 +25,7 @@
-
+
@@ -50,6 +47,13 @@
+
+
+
+
+
+
+
@@ -64,25 +68,26 @@
+
+
+
+
+
+
+
-
-
+
-
-
-
-
-
-
+
-
+
diff --git a/contentprovider-management/.gitignore b/contentprovider-management/.gitignore
new file mode 100644
index 0000000..796b96d
--- /dev/null
+++ b/contentprovider-management/.gitignore
@@ -0,0 +1 @@
+/build
diff --git a/contentprovider-management/build.gradle b/contentprovider-management/build.gradle
new file mode 100644
index 0000000..c7d29d8
--- /dev/null
+++ b/contentprovider-management/build.gradle
@@ -0,0 +1,24 @@
+apply plugin: 'com.android.application'
+
+android {
+ compileSdkVersion 23
+ buildToolsVersion "23.0.3"
+
+ defaultConfig {
+ applicationId "com.example.weishu.contentprovider_management"
+ minSdkVersion 15
+ targetSdkVersion 23
+ versionCode 1
+ versionName "1.0"
+ }
+ buildTypes {
+ release {
+ minifyEnabled false
+ proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
+ }
+ }
+}
+
+dependencies {
+ compile fileTree(dir: 'libs', include: ['*.jar'])
+}
diff --git a/contentprovider-management/contentprovider-management.iml b/contentprovider-management/contentprovider-management.iml
new file mode 100644
index 0000000..af968f9
--- /dev/null
+++ b/contentprovider-management/contentprovider-management.iml
@@ -0,0 +1,96 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ generateDebugSources
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/contentprovider-management/proguard-rules.pro b/contentprovider-management/proguard-rules.pro
new file mode 100644
index 0000000..10fada3
--- /dev/null
+++ b/contentprovider-management/proguard-rules.pro
@@ -0,0 +1,17 @@
+# Add project specific ProGuard rules here.
+# By default, the flags in this file are appended to flags specified
+# in /Users/weishu/dev/env/android-sdk-macosx/tools/proguard/proguard-android.txt
+# You can edit the include path and order by changing the proguardFiles
+# directive in build.gradle.
+#
+# For more details, see
+# http://developer.android.com/guide/developing/tools/proguard.html
+
+# Add any project specific keep options here:
+
+# If your project uses WebView with JS, uncomment the following
+# and specify the fully qualified class name to the JavaScript interface
+# class:
+#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
+# public *;
+#}
diff --git a/contentprovider-management/src/main/AndroidManifest.xml b/contentprovider-management/src/main/AndroidManifest.xml
new file mode 100644
index 0000000..a986594
--- /dev/null
+++ b/contentprovider-management/src/main/AndroidManifest.xml
@@ -0,0 +1,25 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/contentprovider-management/src/main/assets/TestContentProvider.java b/contentprovider-management/src/main/assets/TestContentProvider.java
new file mode 100644
index 0000000..8b45edb
--- /dev/null
+++ b/contentprovider-management/src/main/assets/TestContentProvider.java
@@ -0,0 +1,100 @@
+package com.example.weishu.contentprovider_management;
+
+import android.content.ContentProvider;
+import android.content.ContentUris;
+import android.content.ContentValues;
+import android.content.Context;
+import android.database.Cursor;
+import android.database.SQLException;
+import android.database.sqlite.SQLiteDatabase;
+import android.database.sqlite.SQLiteOpenHelper;
+import android.database.sqlite.SQLiteQueryBuilder;
+import android.net.Uri;
+import android.support.annotation.NonNull;
+import android.support.annotation.Nullable;
+
+/**
+ * @author weishu
+ * @date 16/7/8.
+ */
+@SuppressWarnings("ALL")
+public class TestContentProvider extends ContentProvider {
+
+ public static final String AUTHORITY = "com.example.weishu.testcontentprovider.TestContentProvider";
+
+ public static final Uri URI = Uri.parse("content://" + AUTHORITY);
+
+ public static final String NAME = "name";
+
+ private static final String TABLE_NAME = "person";
+
+ private TestDataBase mDb;
+
+ @Override
+ public boolean onCreate() {
+ mDb = new TestDataBase(getContext());
+ return true;
+ }
+
+ @Nullable
+ @Override
+ public Cursor query(@NonNull Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
+ SQLiteQueryBuilder qb = new SQLiteQueryBuilder();
+ SQLiteDatabase db = mDb.getReadableDatabase();
+ qb.setTables(TABLE_NAME);
+ Cursor c = qb.query(db, projection, selection, null, null, null, sortOrder);
+ c.setNotificationUri(getContext().getContentResolver(), uri);
+ return c;
+ }
+
+ @Nullable
+ @Override
+ public String getType(Uri uri) {
+ return null;
+ }
+
+ @Nullable
+ @Override
+ public Uri insert(Uri uri, ContentValues values) {
+
+ SQLiteDatabase sqlDB = mDb.getWritableDatabase();
+ long rowId = sqlDB.insert(TABLE_NAME, "", values);
+ if (rowId > 0) {
+ Uri rowUri = ContentUris.appendId(URI.buildUpon(), rowId).build();
+ getContext().getContentResolver().notifyChange(rowUri, null);
+ return rowUri;
+ }
+ throw new SQLException("Failed to insert row into " + uri);
+ }
+
+ @Override
+ public int delete(Uri uri, String selection, String[] selectionArgs) {
+ return 0;
+ }
+
+ @Override
+ public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
+ return 0;
+ }
+
+ private static class TestDataBase extends SQLiteOpenHelper {
+
+ private static int VERSION = 1;
+ private static final String DB_NAME = "persons.db";
+
+ TestDataBase(Context context) {
+ super(context, DB_NAME, null, VERSION);
+ }
+
+ @Override
+ public void onCreate(SQLiteDatabase db) {
+ db.execSQL("Create table " + TABLE_NAME + "( _id INTEGER PRIMARY KEY AUTOINCREMENT, " + NAME + ");");
+ }
+
+ @Override
+ public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
+ db.execSQL("DROP TABLE IF EXISTS " + TABLE_NAME);
+ onCreate(db);
+ }
+ }
+}
diff --git a/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/MainActivity.java b/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/MainActivity.java
new file mode 100644
index 0000000..78b719b
--- /dev/null
+++ b/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/MainActivity.java
@@ -0,0 +1,67 @@
+package com.example.weishu.contentprovider_management;
+
+import android.app.Activity;
+import android.content.ContentValues;
+import android.database.Cursor;
+import android.net.Uri;
+import android.os.Bundle;
+import android.util.Log;
+import android.view.View;
+import android.widget.Button;
+import android.widget.LinearLayout;
+
+/**
+ * @author weishu
+ * @date 16/7/8.
+ */
+public class MainActivity extends Activity {
+
+ private static final String TAG = "MainActivity";
+
+ // demo ContentProvider 的URI
+ private static Uri URI = Uri.parse("content://com.example.weishu.testcontentprovider.TestContentProvider");
+
+ static int count = 0;
+
+ @Override
+ protected void onCreate(Bundle savedInstanceState) {
+ super.onCreate(savedInstanceState);
+
+ Button query = new Button(this);
+ query.setText("query");
+ query.setOnClickListener(new View.OnClickListener() {
+ @Override
+ public void onClick(View v) {
+ Cursor cursor = getContentResolver().query(URI,
+ null, null, null, null);
+ assert cursor != null;
+ while (cursor.moveToNext()) {
+ int count = cursor.getColumnCount();
+ StringBuilder sb = new StringBuilder("column: ");
+ for (int i = 0; i < count; i++) {
+ sb.append(cursor.getString(i) + ", ");
+ }
+ Log.d(TAG, sb.toString());
+ }
+ }
+ });
+
+ Button insert = new Button(this);
+ insert.setText("insert");
+ insert.setOnClickListener(new View.OnClickListener() {
+ @Override
+ public void onClick(View v) {
+ ContentValues values = new ContentValues();
+ values.put("name", "name" + count++);
+ getContentResolver().insert(URI, values);
+ }
+ });
+
+ LinearLayout layout = new LinearLayout(this);
+ layout.setOrientation(LinearLayout.VERTICAL);
+ layout.addView(query);
+ layout.addView(insert);
+
+ setContentView(layout);
+ }
+}
diff --git a/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/ProviderHelper.java b/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/ProviderHelper.java
new file mode 100644
index 0000000..06e2190
--- /dev/null
+++ b/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/ProviderHelper.java
@@ -0,0 +1,85 @@
+package com.example.weishu.contentprovider_management;
+
+import android.content.Context;
+import android.content.pm.PackageManager;
+import android.content.pm.ProviderInfo;
+import android.util.Log;
+
+import java.io.File;
+import java.lang.reflect.Field;
+import java.lang.reflect.Method;
+import java.util.ArrayList;
+import java.util.List;
+
+/**
+ * @author weishu
+ * @date 16/7/8.
+ */
+public class ProviderHelper {
+
+ /**
+ * 解析Apk文件中的 , 并存储起来
+ * 主要是调用PackageParser类的generateProviderInfo方法
+ *
+ * @param apkFile 插件对应的apk文件
+ * @throws Exception 解析出错或者反射调用出错, 均会抛出异常
+ */
+ public static List parseProviders(File apkFile) throws Exception {
+ Class> packageParserClass = Class.forName("android.content.pm.PackageParser");
+ Method parsePackageMethod = packageParserClass.getDeclaredMethod("parsePackage", File.class, int.class);
+
+ Object packageParser = packageParserClass.newInstance();
+
+ // 首先调用parsePackage获取到apk对象对应的Package对象
+ Object packageObj = parsePackageMethod.invoke(packageParser, apkFile, PackageManager.GET_PROVIDERS);
+
+ // 读取Package对象里面的services字段
+ // 接下来要做的就是根据这个List 获取到Provider对应的ProviderInfo
+ Field providersField = packageObj.getClass().getDeclaredField("providers");
+ List providers = (List) providersField.get(packageObj);
+
+ // 调用generateProviderInfo 方法, 把PackageParser.Provider转换成ProviderInfo
+ Class> packageParser$ProviderClass = Class.forName("android.content.pm.PackageParser$Provider");
+ Class> packageUserStateClass = Class.forName("android.content.pm.PackageUserState");
+ Class> userHandler = Class.forName("android.os.UserHandle");
+ Method getCallingUserIdMethod = userHandler.getDeclaredMethod("getCallingUserId");
+ int userId = (Integer) getCallingUserIdMethod.invoke(null);
+ Object defaultUserState = packageUserStateClass.newInstance();
+
+ // 需要调用 android.content.pm.PackageParser#generateProviderInfo
+ Method generateProviderInfo = packageParserClass.getDeclaredMethod("generateProviderInfo",
+ packageParser$ProviderClass, int.class, packageUserStateClass, int.class);
+
+ List ret = new ArrayList<>();
+ // 解析出intent对应的Provider组件
+ for (Object service : providers) {
+ ProviderInfo info = (ProviderInfo) generateProviderInfo.invoke(packageParser, service, 0, defaultUserState, userId);
+ ret.add(info);
+ }
+
+ return ret;
+ }
+
+ /**
+ * 在进程内部安装provider, 也就是调用 ActivityThread.installContentProviders方法
+ *
+ * @param context you know
+ * @param apkFile
+ * @throws Exception
+ */
+ public static void installProviders(Context context, File apkFile) throws Exception {
+ List providerInfos = parseProviders(apkFile);
+
+ for (ProviderInfo providerInfo : providerInfos) {
+ providerInfo.applicationInfo.packageName = context.getPackageName();
+ }
+
+ Log.d("test", providerInfos.toString());
+ Class> activityThreadClass = Class.forName("android.app.ActivityThread");
+ Method currentActivityThreadMethod = activityThreadClass.getDeclaredMethod("currentActivityThread");
+ Object currentActivityThread = currentActivityThreadMethod.invoke(null);
+ Method installProvidersMethod = activityThreadClass.getDeclaredMethod("installContentProviders", Context.class, List.class);
+ installProvidersMethod.setAccessible(true);
+ installProvidersMethod.invoke(currentActivityThread, context, providerInfos);
+ }
+}
diff --git a/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/StubContentProvider.java b/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/StubContentProvider.java
new file mode 100644
index 0000000..a1c7227
--- /dev/null
+++ b/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/StubContentProvider.java
@@ -0,0 +1,88 @@
+package com.example.weishu.contentprovider_management;
+
+import android.content.ContentProvider;
+import android.content.ContentValues;
+import android.database.Cursor;
+import android.net.Uri;
+import android.util.Log;
+
+/**
+ * content://
+ *
+ * @author weishu
+ * @date 16/7/11.
+ */
+public class StubContentProvider extends ContentProvider {
+
+ private static final String TAG = "StubContentProvider";
+
+ public static final String AUTHORITY = "com.example.weishu.contentprovider_management.StubContentProvider";
+
+ @Override
+ public boolean onCreate() {
+ return true;
+ }
+
+ @Override
+ public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
+ //noinspection ConstantConditions
+ return getContext().getContentResolver().query(getRealUri(uri), projection, selection, selectionArgs, sortOrder);
+ }
+
+ @Override
+ public String getType(Uri uri) {
+ return null;
+ }
+
+ @Override
+ public Uri insert(Uri uri, ContentValues values) {
+ //noinspection ConstantConditions
+ return getContext().getContentResolver().insert(getRealUri(uri), values);
+ }
+
+ @Override
+ public int delete(Uri uri, String selection, String[] selectionArgs) {
+ return 0;
+ }
+
+ @Override
+ public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
+ return 0;
+ }
+
+ /**
+ * 为了使得插件的ContentProvder提供给外部使用,我们需要一个StubProvider做中转;
+ * 如果外部程序需要使用插件系统中插件的ContentProvider,不能直接查询原来的那个uri
+ * 我们对uri做一些手脚,使得插件系统能识别这个uri;
+ *
+ * 这里的处理方式如下:
+ *
+ * 原始查询插件的URI应该为:
+ * content://plugin_auth/path/query
+ *
+ * 如果需要查询插件,需要修改为:
+ *
+ * content://stub_auth/plugin_auth/path/query
+ *
+ * 也就是,我们把插件ContentProvider的信息放在URI的path中保存起来;
+ * 然后在StubProvider中做分发。
+ *
+ * 当然,也可以使用QueryParamerter,比如:
+ * content://plugin_auth/path/query/ -> content://stub_auth/path/query?plugin=plugin_auth
+ * @param raw 外部查询我们使用的URI
+ * @return 插件真正的URI
+ */
+ private Uri getRealUri(Uri raw) {
+ String rawAuth = raw.getAuthority();
+ if (!AUTHORITY.equals(rawAuth)) {
+ Log.w(TAG, "rawAuth:" + rawAuth);
+ }
+
+ String uriString = raw.toString();
+ uriString = uriString.replaceAll(rawAuth + '/', "");
+ Uri newUri = Uri.parse(uriString);
+ Log.i(TAG, "realUri:" + newUri);
+ return newUri;
+ }
+
+}
diff --git a/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/UPFApplication.java b/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/UPFApplication.java
new file mode 100644
index 0000000..6b68d59
--- /dev/null
+++ b/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/UPFApplication.java
@@ -0,0 +1,40 @@
+package com.example.weishu.contentprovider_management;
+
+import android.app.Application;
+import android.content.Context;
+
+import com.example.weishu.contentprovider_management.hook.BaseDexClassLoaderHookHelper;
+
+import java.io.File;
+
+/**
+ * 一定需要Application,并且在attachBaseContext里面Hook
+ * 因为provider的初始化非常早,比Application的onCreate还要早
+ * 在别的地方hook都晚了。
+ *
+ * @author weishu
+ * @date 16/3/29
+ */
+public class UPFApplication extends Application {
+
+ @Override
+ protected void attachBaseContext(Context base) {
+ super.attachBaseContext(base);
+
+ try {
+ File apkFile = getFileStreamPath("testcontentprovider-debug.apk");
+ if (!apkFile.exists()) {
+ Utils.extractAssets(base, "testcontentprovider-debug.apk");
+ }
+
+ File odexFile = getFileStreamPath("test.odex");
+
+ // Hook ClassLoader, 让插件中的类能够被成功加载
+ BaseDexClassLoaderHookHelper.patchClassLoader(getClassLoader(), apkFile, odexFile);
+ ProviderHelper.installProviders(base, getFileStreamPath("testcontentprovider-debug.apk"));
+ } catch (Exception e) {
+ throw new RuntimeException("hook failed", e);
+ }
+ }
+
+}
diff --git a/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/Utils.java b/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/Utils.java
new file mode 100644
index 0000000..eb59c72
--- /dev/null
+++ b/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/Utils.java
@@ -0,0 +1,63 @@
+package com.example.weishu.contentprovider_management;
+
+import android.content.Context;
+import android.content.res.AssetManager;
+
+import java.io.Closeable;
+import java.io.File;
+import java.io.FileOutputStream;
+import java.io.IOException;
+import java.io.InputStream;
+
+/**
+ * @author weishu
+ * @date 16/3/29
+ */
+public class Utils {
+
+ /**
+ * 把Assets里面得文件复制到 /data/data/files 目录下
+ *
+ * @param context
+ * @param sourceName
+ */
+ public static void extractAssets(Context context, String sourceName) {
+ AssetManager am = context.getAssets();
+ InputStream is = null;
+ FileOutputStream fos = null;
+ try {
+ is = am.open(sourceName);
+ File extractFile = context.getFileStreamPath(sourceName);
+ fos = new FileOutputStream(extractFile);
+ byte[] buffer = new byte[1024];
+ int count = 0;
+ while ((count = is.read(buffer)) > 0) {
+ fos.write(buffer, 0, count);
+ }
+ fos.flush();
+ } catch (IOException e) {
+ e.printStackTrace();
+ } finally {
+ closeSilently(is);
+ closeSilently(fos);
+ }
+
+ }
+
+
+
+ // --------------------------------------------------------------------------
+ private static void closeSilently(Closeable closeable) {
+ if (closeable == null) {
+ return;
+ }
+ try {
+ closeable.close();
+ } catch (Throwable e) {
+ // ignore
+ }
+ }
+
+ private static File sBaseDir;
+
+}
diff --git a/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/hook/BaseDexClassLoaderHookHelper.java b/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/hook/BaseDexClassLoaderHookHelper.java
new file mode 100644
index 0000000..78c7cb6
--- /dev/null
+++ b/contentprovider-management/src/main/java/com/example/weishu/contentprovider_management/hook/BaseDexClassLoaderHookHelper.java
@@ -0,0 +1,59 @@
+package com.example.weishu.contentprovider_management.hook;
+
+import java.io.File;
+import java.io.IOException;
+import java.lang.reflect.Array;
+import java.lang.reflect.Constructor;
+import java.lang.reflect.Field;
+import java.lang.reflect.InvocationTargetException;
+
+import dalvik.system.DexClassLoader;
+import dalvik.system.DexFile;
+
+/**
+ * 由于应用程序使用的ClassLoader为PathClassLoader
+ * 最终继承自 BaseDexClassLoader
+ * 查看源码得知,这个BaseDexClassLoader加载代码根据一个叫做
+ * dexElements的数组进行, 因此我们把包含代码的dex文件插入这个数组
+ * 系统的classLoader就能帮助我们找到这个类
+ *
+ * 这个类用来进行对于BaseDexClassLoader的Hook
+ * 类名太长, 不要吐槽.
+ * @author weishu
+ * @date 16/3/28
+ */
+public final class BaseDexClassLoaderHookHelper {
+
+ public static void patchClassLoader(ClassLoader cl, File apkFile, File optDexFile)
+ throws IllegalAccessException, NoSuchMethodException, IOException, InvocationTargetException, InstantiationException, NoSuchFieldException {
+ // 获取 BaseDexClassLoader : pathList
+ Field pathListField = DexClassLoader.class.getSuperclass().getDeclaredField("pathList");
+ pathListField.setAccessible(true);
+ Object pathListObj = pathListField.get(cl);
+
+ // 获取 PathList: Element[] dexElements
+ Field dexElementArray = pathListObj.getClass().getDeclaredField("dexElements");
+ dexElementArray.setAccessible(true);
+ Object[] dexElements = (Object[]) dexElementArray.get(pathListObj);
+
+ // Element 类型
+ Class> elementClass = dexElements.getClass().getComponentType();
+
+ // 创建一个数组, 用来替换原始的数组
+ Object[] newElements = (Object[]) Array.newInstance(elementClass, dexElements.length + 1);
+
+ // 构造插件Element(File file, boolean isDirectory, File zip, DexFile dexFile) 这个构造函数
+ Constructor> constructor = elementClass.getConstructor(File.class, boolean.class, File.class, DexFile.class);
+ Object o = constructor.newInstance(apkFile, false, apkFile, DexFile.loadDex(apkFile.getCanonicalPath(), optDexFile.getAbsolutePath(), 0));
+
+ Object[] toAddElementArray = new Object[] { o };
+ // 把原始的elements复制进去
+ System.arraycopy(dexElements, 0, newElements, 0, dexElements.length);
+ // 插件的那个element复制进去
+ System.arraycopy(toAddElementArray, 0, newElements, dexElements.length, toAddElementArray.length);
+
+ // 替换
+ dexElementArray.set(pathListObj, newElements);
+
+ }
+}
diff --git a/contentprovider-management/src/main/res/mipmap-hdpi/ic_launcher.png b/contentprovider-management/src/main/res/mipmap-hdpi/ic_launcher.png
new file mode 100644
index 0000000000000000000000000000000000000000..cde69bcccec65160d92116f20ffce4fce0b5245c
GIT binary patch
literal 3418
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
literal 0
HcmV?d00001
diff --git a/contentprovider-management/src/main/res/mipmap-xhdpi/ic_launcher.png b/contentprovider-management/src/main/res/mipmap-xhdpi/ic_launcher.png
new file mode 100644
index 0000000000000000000000000000000000000000..bfa42f0e7b91d006d22352c9ff2f134e504e3c1d
GIT binary patch
literal 4842
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
literal 0
HcmV?d00001
diff --git a/contentprovider-management/src/main/res/mipmap-xxhdpi/ic_launcher.png b/contentprovider-management/src/main/res/mipmap-xxhdpi/ic_launcher.png
new file mode 100644
index 0000000000000000000000000000000000000000..324e72cdd7480cb983fa1bcc7ce686e51ef87fe7
GIT binary patch
literal 7718
zcmZ{JWl)?=u?hpbj?h-6mfK3P*Eck~k0Tzeg5-hkABxtZea0_k$f-mlF
z0S@Qqtva`>x}TYzc}9LrO?P#qj+P1@HZ?W?0C;Muih9o&|G$cb@ocx1*PEUJ%~tM}
z901hB;rx4#{@jOHs_MN00ADr$2n+#$yJuJ64gh!x0KlF(07#?(0ENrf7G3D`0EUHz
zisCaq%dJ9dz%zhdRNuG*01nCjDhiPCl@b8xIMfv7^t~4jVRrSTGYyZUWqY@yW=)V_
z&3sUP1SK9v1f{4lDSN(agrKYULc;#EGDVeU*5b@#MOSY5JBn#QG8wqxQh+mdR638{mo5f>O
zLUdZIPSjFk0~F26zDrM3y_#P^P91oWtLlPaZrhnM$NR%qsbHHK#?fN?cX?EvAhY1Sr9A(1;Kw4@87~|;2QP~
z(kKOGvCdB}qr4m#)1DwQFlh^NdBZvNLkld&yg%&GU`+boBMsoj5o?8tVuY^b0?4;E
zsxoLxz8?S$y~a~x0{?dqk+6~Dd(EG7px_yH(X&NX&qEtHPUhu*JHD258=5$JS12rQ
zcN+7p>R>tbFJ3NzEcRIpS98?}YEYxBIA8}1Y8zH9wq0c{hx+EXY&ZQ!-Hvy03X
zLTMo4EZwtKfwb294-cY5XhQRxYJSybphcrNJWW2FY+b?|QB^?$5ZN=JlSs9Og(;8+
z*~-#CeeEOxt~F#aWn8wy-N_ilDDe_o+SwJD>4y?j5Lpj
z2&!EX)RNxnadPBAa?fOj5D1C{l1E0X?&G3+ckcVfk`?%2FTsoUf4@~eaS#th=zq7v
zMEJR@1T?Pi4;$xiPv`3)9rsrbVUH&b0e2{YTEG%;$GGzKUKEim;R6r>F@Q-}9JR-<
zOPpQI>W0Vt6&7d?~$d&}chKTr_rELu}
zWY;KTvtpJFr?P~ReHL4~2=ABn1`GN4Li%OI_1{mMRQi1Bf?+^Va?xdn4>h)Bq#ZRK
zYo%R_h5etrv|!$1QF8fu80fN?1oXe(Jx#e6H^$+>C}N{*i$bNbELsXDA>cxlh|iFq
zh~$yJ?1lTdcFd1Yv+Hr^PP!yupP!0H@Y6(wFcaVE+0?qjDJ1;*-Q8qL{NNPc{GAoi
z_kBH`kw^(^7ShmzArk^A-!3_$W%!M-pGaZC=K`p-ch&iT%CV0>ofS74aPd7oT&cRr
zXI30fVV6#PR*Z?c*orR0!$K6SUl9!H>hG+%`LdifNk`!Sw7Hon{Wn=|qV{a%v9nEq
zAdBW*5kq6il=yA}x8cZQt^c+RBS|TRn;!?$ue?@jIV~0w1dt1FJRYI-K5>z-^01)R
z)r}A&QXp^?-?}Uj`}ZPqB#}xO-?{0wrmi|eJOEjzdXbey4$rtKNHz)M*o?Ov+;S=K
z-l~`)xV`%7Gvzy5wfvwqc0|80K29k0G~1nuBO+y-6)w11Kz2{>yD{HTt-uybe2pe?
zUZK*Eij7TT4NwF1Jr@6R7gMuu^@qn#zPIgRtF?-SJL83LBDrh7k#{F^222EXPg}S0d4Lf0!|1
z|2k$^b~)^8$Z-yH{B-vo%7sVU@ZCvXN+Am)-fy$afZ_4HAUpK}j4p`UyXRel-+(VS
z#K>-=-oA1pH+Lo$&|!lYB|M7Y&&bF##Oi@y_G3p1X$0I{jS1!NEdTz#x0`H`d*l%X
z*8Y3>L*>j@ZQGOdPqwY(GzbA4nxqT(UAP<-tBf{_cb&Hn8hO5gEAotoV;tF6K4~wr2-M0v|2acQ!E@G*g$J
z)~&_lvwN%WW>@U_taX5YX@a~pnG7A~jGwQwd4)QKk|^d_x9j+3JYmI5H`a)XMKwDt
zk(nmso_I$Kc5m+8iVbIhY<4$34Oz!sg3oZF%UtS(sc6iq3?e8Z;P<{OFU9MACE6y(
zeVprnhr!P;oc8pbE%A~S<+NGI2ZT@4A|o9bByQ0er$rYB3(c)7;=)^?$%a${0@70N
zuiBVnAMd|qX7BE)8})+FAI&HM|BIb3e=e`b{Do8`J0jc$H>gl$zF26=haG31FDaep
zd~i}CHSn$#8|WtE06vcA%1yxiy_TH|RmZ5>pI5*8pJZk0X54JDQQZgIf1Pp3*6hepV_cXe)L2iW$Ov=RZ4T)SP^a_8V}
z+Nl?NJL7fAi<)Gt98U+LhE>x4W=bfo4F>5)qBx@^8&5-b>y*Wq19MyS(72ka8XFr2
zf*j(ExtQkjwN|4B?D
z7+WzS*h6e_Po+Iqc-2n)gTz|de%FcTd_i9n+Y5*Vb=E{8xj&|h`CcUC*(yeCf~#Mf
zzb-_ji&PNcctK6Xhe#gB0skjFFK5C4=k%tQQ}F|ZvEnPcH=#yH4n%z78?McMh!vek
zVzwC0*OpmW2*-A6xz0=pE#WdXHMNxSJ*qGY(RoV9)|eu)HSSi_+|)IgT|!7HRx~
zjM$zp%LEBY)1AKKNI?~*>9DE3Y2t5p#jeqeq`1
zsjA-8eQKC*!$%k#=&jm+JG?UD(}M!tI{wD*3FQFt8jgv2xrRUJ}t}rWx2>XWz9ndH*cxl()ZC
zoq?di!h6HY$fsglgay7|b6$cUG-f!U4blbj(rpP^1ZhHv@Oi~;BBvrv<+uC;%6QK!nyQ!bb3i3D~cvnpDAo3*3
zXRfZ@$J{FP?jf(NY7~-%Kem>jzZ2+LtbG!9I_fdJdD*;^T9gaiY>d+S$EdQrW9W62
z6w8M&v*8VWD_j)fmt?+bdavPn>oW8djd
zRnQ}{XsIlwYWPp;GWLXvbSZ8#w25z1T}!<{_~(dcR_i1U?hyAe+lL*(Y6c;j2q7l!
zMeN(nuA8Z9$#w2%ETSLjF{A#kE#WKus+%pal;-wx&tTsmFPOcbJtT?j&i(#-rB}l@
zXz|&%MXjD2YcYCZ3h4)?KnC*X$G%5N)1s!0!Ok!F9KLgV@wxMiFJIVH?E5JcwAnZF
zU8ZPDJ_U_l81@&npI5WS7Y@_gf3vTXa;511h_(@{y1q-O{&bzJ
z*8g>?c5=lUH6UfPj3=iuuHf4j?KJPq`x@en2Bp>#zIQjX5(C<9-X4X{a^S
znWF1zJ=7rEUwQ&cZgyV4L12f&2^eIc^dGIJP@ToOgrU_Qe=T)utR;W$_2Vb7NiZ+d
z$I0I>GFIutqOWiLmT~-Q<(?n5QaatHWj**>L8sxh1*pAkwG>siFMGEZYuZ)E!^Hfs
zYBj`sbMQ5MR;6=1^0W*qO*Zthx-svsYqrUbJW)!vTGhWKGEu8c+=Yc%xi}Rncu3ph
zTT1j_>={i3l#~$!rW!%ZtD9e6l6k-k8l{2w53!mmROAD^2yB^e)3f9_Qyf&C#zk`(
z|5RL%r&}#t(;vF4nO&n}`iZpIL=p9tYtYv3%r@GzLWJ6%y_D(icSF^swYM`e8-n43iwo$C~>G<)dd0ze@5}n(!^YD
zHf#OVbQ$Li@J}-qcOYn_iWF=_%)EXhrVuaYiai|B<1tXwNsow(m;XfL6^x~|Tr%L3~cs0@c)
zDvOFU-AYn1!A;RBM0S}*EhYK49H$mBAxus)CB*KW(87#!#_C0wDr<0*dZ+GN&(3wR
z6)cFLiDvOfs*-7Q75ekTAx)k!dtENUKHbP|2y4=tf*d_BeZ(9kR*m;dVzm&0fkKuD
zVw5y9N>pz9C_wR+&Ql&&y{4@2M2?fWx~+>f|F%8E@fIfvSM$Dsk26(UL32oNvTR;M
zE?F<7<;;jR4)ChzQaN((foV
z)XqautTdMYtv<=oo-3W-t|gN7Q43N~%fnClny|NNcW9bIPPP5KK7_N8g!LB8{mK#!
zH$74|$b4TAy@hAZ!;irT2?^B0kZ)7Dc?(7xawRUpO~AmA#}eX9A>+BA7{oDi)LA?F
ze&CT`Cu_2=;8CWI)e~I_65cUmMPw5fqY1^6v))pc_TBArvAw_5Y8v0+fFFT`T
zHP3&PYi2>CDO=a|@`asXnwe>W80%%<>JPo(DS}IQiBEBaNN0EF6HQ1L2i6GOPMOdN
zjf3EMN!E(ceXhpd8~<6;6k<57OFRs;mpFM6VviPN>p3?NxrpNs0>K&nH_s
ze)2#HhR9JHPAXf#viTkbc{-5C7U`N!`>J-$T!T6%=xo-)1_WO=+BG{J`iIk%tvxF39rJtK49Kj#ne;WG1JF1h7;~wauZ)nMvmBa2PPfrqREMKWX
z@v}$0&+|nJrAAfRY-%?hS4+$B%DNMzBb_=Hl*i%euVLI5Ts~UsBVi(QHyKQ2LMXf`
z0W+~Kz7$t#MuN|X2BJ(M=xZDRAyTLhPvC8i&9b=rS-T{k34X}|t+FMqf5gwQirD~N1!kK&^#+#8WvcfENOLA`Mcy@u~
zH10E=t+W=Q;gn}&;`R1D$n(8@Nd6f)9=F%l?A>?2w)H}O4avWOP@7IMVRjQ&aQDb)
zzj{)MTY~Nk78>B!^EbpT{&h
zy{wTABQlVVQG<4;UHY?;#Je#-E;cF3gVTx520^#XjvTlEX>+s{?KP#Rh@hM6R;~DE
zaQY16$Axm5ycukte}4FtY-VZHc>=Ps8mJDLx3mwVvcF<^`Y6)v5tF`RMXhW1kE-;!
z7~tpIQvz5a6~q-8@hTfF9`J;$QGQN%+VF#`>F4K3>h!tFU^L2jEagQ5Pk1U_I5&B>
z+i<8EMFGFO$f7Z?pzI(jT0QkKnV)gw=j74h4*jfkk3UsUT5PemxD`pO^Y#~;P2Cte
zzZ^pr>SQHC-576SI{p&FRy36<`&{Iej&&A&%>3-L{h(fUbGnb)*b&eaXj>i>gzllk
zLXjw`pp#|yQIQ@;?mS=O-1Tj+ZLzy+aqr7%QwWl?j=*6dw5&4}>!wXqh&j%NuF{1q
zzx$OXeWiAue+g#nkqQ#Uej@Zu;D+@z^VU*&HuNqqEm?V~(Z%7D`W5KSy^e|yF6kM7
z8Z9fEpcs^ElF9Vnolfs7^4b0fsNt+i?LwUX8Cv|iJeR|GOiFV!JyHdq+XQ&dER(KSqMxW{=M)lA?Exe&ZEB~6SmHg`zkcD7x#myq0h61+zhLr_NzEIjX
zr~NGX_Uh~gdcrvjGI(&5K_zaEf}1t*)v3uT>~Gi$r^}R;H+0FEE5El{y;&DniH2@A
z@!71_8mFHt1#V8MVsIYn={v&*0;3SWf4M$yLB^BdewOxz;Q=+gakk`S{_R_t!z2b|
z+0d^C?G&7U6$_-W9@eR6SH%+qLx_Tf&Gu5%pn*mOGU0~kv~^K
zhPeqYZMWWoA(Y+4GgQo9nNe6S#MZnyce_na@78ZnpwFenVafZC3N2lc5Jk-@V`{|l
zhaF`zAL)+($xq8mFm{7fXtHru+DANoGz-A^1*@lTnE;1?03lz8kAnD{zQU=Pb^3f`
zT5-g`z5|%qOa!WTBed-8`#AQ~wb9TrUZKU)H*O7!LtNnEd!r8!Oda)u!Gb5P`9(`b
z`lMP6CLh4OzvXC#CR|@uo$EcHAyGr=)LB7)>=s3
zvU;aR#cN3<5&CLMFU@keW^R-Tqyf4fdkOnwI(H$x#@I1D6#dkUo@YW#7MU0@=NV-4
zEh2K?O@+2e{qW^7r?B~QTO)j}>hR$q9*n$8M(4+DOZ00WXFonLlk^;os8*zI>YG#?
z9oq$CD~byz>;`--_NMy|iJRALZ#+qV8OXn=AmL^GL&|q1Qw-^*#~;WNNNbk(96Tnw
zGjjscNyIyM2CYwiJ2l-}u_7mUGcvM+puPF^F89eIBx27&$|p_NG)fOaafGv|_b9G$;1LzZ-1aIE?*R6kHg}dy%~K(Q5S2O6086
z{lN&8;0>!pq^f*Jlh=J%Rmaoed<=uf@$iKl+bieC83IT!09J&IF)9H)C?d!eW1UQ}BQwxaqQY47DpOk@`zZ
zo>#SM@oI^|nrWm~Ol7=r`!Bp9lQNbBCeHcfN&X$kjj0R(@?f$OHHt|fWe6jDrYg3(mdEd$8P2Yzjt9*EM
zLE|cp-Tzsdyt(dvLhU8}_IX&I?B=|yoZ!&<`9&H5PtApt=VUIB4l0a1NH
v0SQqt3DM`an1p};^>=lX|A*k@Y-MNT^ZzF}9G-1G696?OEyXH%^Pv9$0dR%J
literal 0
HcmV?d00001
diff --git a/contentprovider-management/src/main/res/mipmap-xxxhdpi/ic_launcher.png b/contentprovider-management/src/main/res/mipmap-xxxhdpi/ic_launcher.png
new file mode 100644
index 0000000000000000000000000000000000000000..aee44e138434630332d88b1680f33c4b24c70ab3
GIT binary patch
literal 10486
zcmai4byOU|lb&5k+^GN3bv-?^>(QkVinb
zlU9`mfQEQnq$S4VGrg6fmMQ=QFarQQ0ss(?uiys&;LQU7M-~7engIZmZaH5x#UC3m
z-zvYBd&I}<`b3rPHj1tDgVv1x|
zQss$ELI?W?E(!7PKk$lm@;7PwPX3o43{Ccd9@_BUsL4kQzSMa&=g{>4wj9#)9wgYw;=H@gH9KK{s?Be8N1_8W<
z1Rh%Lm&PAfyYb*rGB%E#3q+}riOBB~+@@X<`9mgIiAex!QP8vg-XT>=+N&y*jC-f<
zGihyr7XAly+G)|_e)qA?rnKZGG(x?=lLM7nrPk&93@5eX#7I_$g8kMX`0h=}l`HH)
z=bpOkBCx=z*-fyr{yp7A9F=%o*qm93t_#tB2lAM@O{fX9ju%X#0~)nRUMvrXClh9w
ze8|a0|0}JJg(_@$2wItI?LUY{zF78o(P2BR7;aC^@(jOp{8RE%U3m>MV5%Lu*46b@
zw*c?Nweu!TULS~}*9mi!ejNfNa=`po1*!jiYK)osxi%b59(thEyUZ>#lX@uEXSb_x?3)0kvB?8*TAh)7}IbzSm}5Ia;_?10{}M;
z7vq-OS;Ayk8%_c-gg1Ee0FsrRU5phNs#H9Lp!1t+hwyK~9W0bWCxuG$LM~wQuumEw
z=fbBD@sQE%1^j
z`T@`PZLRVyWjX@*tjc7r;w$H~aW&7vu?|war?84^sg!{J*RH|mhq?KTsCVQBC1~fR
z>99jeR=g-Q2b=d;pKwzXwYjrG>?pd3tFSsHN4in{usYLdK;01X2BdRLFI`cuB9yI)
zI_ZX?7_(bz`MX2@^mCknx7
z*f}KV@}TBBc}CXMR8T_5yInD3p`KrNROSA;HoJJtlNG3weri%utO$eeY0
z+w-NEn;(;UCBk=OM$f%=%ma24wV7$idelqyNWI>sz1>BlGwr_3UugqVjY+UYyi9P)
zxCB?&rPUetoZN?|*D%=hOOJ_${JU3GRjppY%&8Ws^G6>iokr^Bmv1&*@#2#5mXu05
zhPVXaQ`qe5i0lP-1^XL45x`ertKU5d-8b_?*1+tSU!qCeqD9gZP_>ZLq9p)RKtV(B
zOh&^x>gV^eqb&c~Oi0|HgGG|gjpbR`9aRdZhOimvS2Y3e?eCFiw+L#_mi9j
z;nU}gih+zTn{nv_|L}IllD1Dr3~@yitI}+4C&+;SR+cEfelqJ?eUjZ%&Qz)W8S750
z+vG8Lvo}xXz2C}S-m|9*uE?NWQWT#W+p@$DkH8wVn#=gLKa13M!Yva9qsfE(5Z#0V`A0pN)Ok
zP*Eq0(~e$~m@iej0#Av_z703y-7|W6`UuGDS8fpy2rUgINZs#`33@@0(S%~%XUO5G
zscEp&x^dU`8syC67USOswNLq>Z_}q#gLh2x`zR)0wvor72-IW@oDpnT0x
zWn%LZ_yvR*7geY6<}MC~SViD+4`S9XC|L}N0ANpsUU;50sAj