修改Android源码实现原⽣应⽤双开,应⽤多开
1. 准备
把某系统双开的两个app的信息进⾏对⽐
1.1 ⽬录的对⽐
1.1.1 data⽬录对⽐
原应⽤:
/data/ur/0/ackme/files
被复制的应⽤:
/data/ur/999/ackme/files
1.1.2 apk所在⽬录对⽐
原应⽤:
/
data/app/ackme-H1Dvbka0t42rzlCAqSpgHQ==/ba.apk
被复制的应⽤:
/data/app/ackme-H1Dvbka0t42rzlCAqSpgHQ==/ba.apk
通过对⽐apk安装⽬录和数据⽬录,我们可以知道,该系统的双开是共⽤同⼀个apk,但是却拥有独⽴的数据⽬录。
1.2 进程信息对⽐
USER PID PPID VSZ RSS WCHAN ADDR S NAME
u0_a161 30284 918 2276572 48420 SyS_epoll_wait 0 S ackme
u999_a161 30311 918 2276572 48004 SyS_epoll_wait 0 S ackme
通过查看进程信息,可以知道,这两个应⽤运⾏于不同的⽤户中。
为了实现和它相似的功能,我们做下⽂的配置。
2. 修改创建⽤户限制
从Android5.0开始,Android⽀持创建Profile.默认情况下,系统只允许创建⼀个新的多开⽤户,也就是只能双开,但是修改源码可以达到创建多个⽤户。修改frameworks/ba/rvices/core/java/com/android/rver/pm/UrManagerService.java
的MAX_MANAGED_PROFILES字段,改成⾃⼰想要创建的最⼤⽤户数,它的默认值是1.
3. 创建⽤户
创建⼀个⽤户即创建⼀个多开容器,调⽤createProfile⽅法,flag传⼊0x00000020,以创建⼀个⽤户并将它开启第一次骑自行车
private static int getUrIdFromUrInfo(Object urInfo) {
int urId = -1;
try {
Field field_id = Class().getDeclaredField("id");
field_id.tAccessible(true);
urId = (Integer)(urInfo);
} catch (Exception e) {
e.printStackTrace();
}
return urId;
小荷才露尖尖角全诗}
public boolean startUr(int urId){
Object iActivityManager = null;
try {
iActivityManager = Class.forName("android.app.ActivityManagerNative").getMethod("getDefault").invoke(null);
boolean isOk=(Class().getMethod("startUrInBackground",int.class)
.invoke(iActivityManager,urId);
return isOk;
} catch (Exception e) {
e.printStackTrace();
}
return fal;
}
public String createProfile(Context context, String urName, int flag) {
UrManager mUrManager = (UrManager) SystemService(Context.USER_SERVICE);
UrHandle urHandle = UrHandleForUid(0);
Log.d(TAG,"urHandle = "+String());
try {
int getIdentifier=(Class().getMethod("getIdentifier").invoke(urHandle);
Log.d(TAG,"Identifier = "+getIdentifier);
Class().getMethod("createProfileForUr",String.class, int.class, int.class)
.invoke(mUrManager
,urName
, flag
,getIdentifier);
if(mUrInfo==null){
Log.d(TAG, "mUrInfo is null!");
return null;
}
int urId = getUrIdFromUrInfo(mUrInfo);
boolean isOk=startUr(urId);
Log.d(TAG, "startUrInBackground() urId = " + urId + " | isOk = " + isOk);
return isOk ? ""+urId : null;幼儿园防踩踏安全教案
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
注:创建⽤户的App要在l的manifest节点下加⼊android:sharedUrId="android.uid.system"字段,加⼊<us-permission
android:name="android.permission.MANAGE_USERS"/>权限,还要使⽤系统的platform签名对apk进⾏签名。
4. 配置系统应⽤不安装到⼦⽤户
默认情况下,在创建⼀个新⽤户的时候,系统会给新⽤户复制⼀份系统应⽤,但是在⼦⽤户中我们并不需要系统应⽤,所以我们要在⼦⽤户中取消安装这些系统应⽤。
注:系统应⽤可以不安装到⼦⽤户,但是系统服务⼀定要安装到⼦⽤户,否则,运⾏在⼦⽤户的app可能⽆法正常运⾏。
修改frameworks/ba/rvices/core/java/com/android/rver/pm/Settings.java
createNewUrLI⽅法,对系统应⽤和系统服务是否安装到⼦⽤户进⾏配置。
private final String[] excludeLiStrings={
"android",
"vices",
"shared",
"com.android.bluetooth",
"com.android.htmlviewer",
"com.android.inputdevices",
"com.android.shell",
"installer",
锡林郭勒大草原"alstorage",
"com.acts",
"com.android.providers.downloads",
"com.dia",
"com.ings",
"com.android.providers.urdictionary",
顺治帝
"com.lecom",
"com.android.packageinstaller",
"ings",
"com.lephony",
"s.rvice",
"com.android.webview",
"com.android.location.fud",
"sshim",
"com.android.statementrvice",
"com.android.defcontainer",
"com.android.keychain",
"com.android.proxyhandler",
"com.android.dreams.basic",
"com.android.printspooler",
"com.android.pacprocessor",
"com.android.providers.downloads.ui"
};
private boolean isInExcludeList(String pkg){
for(String excludePkg:excludeLiStrings){
if(excludePkg.equals(pkg)){
return true;
}
}
return fal;
}
void createNewUrLI(@NonNull PackageManagerService rvice, @NonNull Installer installer,
int urHandle) {
String[] volumeUuids;
String[] names;
int[] appIds;
String[] infos;
int[] targetSdkVersions;
int packagesCount;
synchronized (mPackages) {
Collection<PackageSetting> packages = mPackages.values();
packagesCount = packages.size();
volumeUuids = new String[packagesCount];
names = new String[packagesCount];
appIds = new int[packagesCount];
infos = new String[packagesCount];
targetSdkVersions = new int[packagesCount];
Iterator<PackageSetting> packagesIterator = packages.iterator();
for (int i = 0; i < packagesCount; i++) {
PackageSetting ps = ();
if (ps.pkg == null || ps.pkg.applicationInfo == null) {
continue;
}
// Only system apps are initially installed.
//Slog.w(TAG, "Ur handle:"+urHandle+",pkg name:"+ps.name);
//修改的地⽅,在列表外的应⽤不安装到⼦⽤户
if(urHandle > 0 && !isInExcludeList(ps.name)){
ps.tInstalled(fal, urHandle);
}
el{
ps.tInstalled(ps.isSystem(), urHandle);
}
// Need to create a data directory for all apps under this ur. Accumulate all
// required args and call the installer after mPackages lock has been relead
volumeUuids[i] = ps.volumeUuid;
names[i] = ps.name;
appIds[i] = ps.appId;
infos[i] = ps.pkg.applicationInfo.info;
targetSdkVersions[i] = ps.pkg.applicationInfo.targetSdkVersion;
}
}
for (int i = 0; i < packagesCount; i++) {
if (names[i] == null) {
continue;
}
// TODO: triage flags!
final int flags = StorageManager.FLAG_STORAGE_CE | StorageManager.FLAG_STORAGE_DE;
try {
infos[i], targetSdkVersions[i]);
} catch (InstallerException e) {
Slog.w(TAG, "Failed to prepare app data", e);
}
}
synchronized (mPackages) {
applyDefaultPreferredAppsLPw(rvice, urHandle);
}
}
5. 给⾮系统⽤户安装和卸载软件
安装app到指定⽤户
pm install -t -r --ur <urId> <apkPath>
-t 允许安装测试应⽤
-r 替换存在的
-
-ur 指定安装到的⽤户
注:安装app后要重启默认启动器(Launcher),不然可能会出现奇怪的问题
从指定⽤户卸载app
pm uninstall --ur <urId> <pkgName>
6. 设置App默认只安装到主⽤户
开启⼦⽤户后,如果调⽤adb install或者pm install来安装apk,会把apk安装所有⽤户。这不是我们想要的,所以,我们修改成执⾏这些命令时,只把app安装到主⽤户。
第⼀步:针对⽤pm install命令安装apk的⽅式
frameworks/ba/cmds/pm/src/com/android/commands/pm/Pm.java
private static class InstallParams {
SessionParams ssionParams;
String installerPackageName;
//int urId = UrHandle.USER_ALL;
int urId = UrHandle.USER_SYSTEM;
}
第⼆步:针对⽤adb install命令安装apk的⽅式
在旧版本系统上,adb install会调⽤pm install来安装apk,但在新版的系统上会调⽤cmd package命令来安装apk。
package命令的具体实现在:
frameworks/ba/rvices/core/java/com/android/rver/pm/PackageManagerShellCommand.java
所以,修改以下代码:
private static class InstallParams {
SessionParams ssionParams;
String installerPackageName;
//int urId = UrHandle.USER_ALL;
int urId = UrHandle.USER_SYSTEM;
}
7. 删除⽤户
adb shell pm remove-ur <urId>
或者调⽤以下代码删除
public void deleteUr(Context context,int urId){
UrManager urManager=(UrManager) SystemService(Context.USER_SERVICE);
try {
} catch (Exception e) {
e.printStackTrace();
}
}
8. 修改⽤户App右下⾓标
开启多⽤户后,如果给多个⼦⽤户安装相同的App,它们会显⽰相同的右下⼩图标(在源码中被叫做Badge),让我们难以辨别,我们可以让不同的⽤户显⽰不同的右下⼩图标,数字图标就是不错的选择,我们来看看如何修改
frameworks/ba/core/java/android/app/ApplicationPackageManager.java的getBadgeResIdForUr⽅法,返回⼀个Drawable资源id,作为⼦⽤户App的右下⼩图标
private int getBadgeResIdForUr(int urId) {
// Return the framework-provided badge.
if (isManagedProfile(urId)) {
return com.android.internal.R.drawable.ic_corp_icon_badge;
}
return 0;
}
这个图标是要在右下⾓的,所以在做图标的时候,要做⼀张⼤的图标,它的右下⾓放着要显⽰的⼩图标,其余部分以透明填充。做好图标后要⽣成svg格式,⽤Android Studio以Vector Asts导⼊,⼤⼩64x64,导⼊成功会在项⽬的res/drawable⽣成drawable资源⽂件,把资源⽂件替换
从祖父
到frameworks/ba/core/res/res/drawable/ic_corp_l,并在frameworks/ba/core/res/res/l添加对drawable⽂件的声明9. 在最新任务列表出现多开应⽤
默认情况下,最近任务列表是不会出现多开应⽤的。
复活节几月几日在frameworks/ba/rvices/core/java/com/android/rver/am/ActivityManagerService.java的getRecentTasks⽅法中,有⼀段校验:
for (int i = 0; i < recentsCount && maxNum > 0; i++) {
TaskRecord tr = (i);
彼得潘领//....
if (!tr.mUrSetupComplete) {
// Don't include task launched while ur is not done tting-up.
if (DEBUG_RECENTS) Slog.d(TAG_RECENTS,
"Skipping, ur tup not complete: " + tr);
continue;
}
//....
res.add(rti);
/
/....
}
可以将这段校验注释掉,tr是frameworks/ba/rvices/core/java/com/android/rver/am/TaskRecord.java类实例,mUrSetupComplete赋值如下:mUrSetupComplete = IntForUr(ContentResolver(),
USER_SETUP_COMPLETE, 0, urId) != 0;
10. 修改多开应⽤最近任务的名称
⼦⽤户默认会在最近任务的应⽤名称前加上"⼯作"这两个字,语⾔是英⽂会显⽰"Work",如果想隐藏它们
中⽂:
frameworks/ba/core/res/res/l
英⽂:
frameworks/ba/core/res/res/l
修改managed_profile_label_badge字段,去掉"⼯作"或者"Work"即可。
11. 修改卸载时的提⽰⽂本
如果是多开应⽤,卸载时提⽰的⽂本和主⽤户是不⼀样的,如果需要修改,则修改下⾯的⽂件
中⽂:
packages/apps/PackageInstaller/res/l
英⽂:
packages/apps/PackageInstaller/res/l
修改uninstall_application_text_ur字段的值
12. 参考