安装在 system/priv-app 中的应用程序在不重启的情况下无法启动(root)

nag*_*ojv 1 android android-intent android-service rooted-device

我正在研究有根设备。我创建了一个应用程序作为系统应用程序运行。在以 su 用户身份从 adb shell 挂载 /system 目录后,我已将 apk 文件复制到 SD 卡,然后复制到 /system/priv-app。清单文件如下所示

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="abc.def.settings"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="19" />

    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
    <uses-permission android:name="android.permission.WRITE_SETTINGS" />
    <uses-permission android:name="android.permission.WRITE_APN_SETTINGS" />
    <uses-permission android:name="android.permission.MODIFY_PHONE_STATE" />
    <uses-permission android:name="android.permission.READ_PHONE_STATE" />
    <uses-permission android:name="android.permission.CALL_PHONE" />
    <uses-permission android:name="android.permission.READ_CALL_LOG" />
    <uses-permission android:name="android.permission.WRITE_CALL_LOG" />
    <uses-permission android:name="android.permission.WRITE_SECURE_SETTINGS" />
    <uses-permission android:name="android.permission.CHANGE_NETWORK_STATE" />
    <uses-permission android:name="android.permission.STATUS_BAR" />
    <uses-permission android:name="android.permission.EXPAND_STATUS_BAR" />

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name" >

        <service
            android:name="abc.def.settings.service.SystemSettingsService" >
        </service>

        <receiver android:name="abc.def.settings.boot.DeviceBootReceiver" >
            <intent-filter>
                <action android:name="android.intent.action.BOOT_COMPLETED"/>
                <action android:name="abc.def.settings.start_after_reinstall"/>
            </intent-filter>
        </receiver>
    </application>

</manifest>
Run Code Online (Sandbox Code Playgroud)

将 apk 复制到 /system/priv-app 后,我将权限从 shell 更改为 777。该应用程序在设备启动时启动得非常好。但是在重新启动之前我无法启动该服务。我的限制之一是不应重新启动设备以启动此服务。

我尝试了以下但没有一个工作。

adb shell am startservice -n abc.def.settings/.SystemSettingsService
adb shell am startservice -n abc.def.settings/.service.SystemSettingsService
adb shell am startservice -a abc.def.settings.start_after_reinstall
adb shell am broadcast -a abc.def.settings.start_after_reinstall
adb shell am broadcast -a abc.def.settings.start_after_reinstall -f 32
Run Code Online (Sandbox Code Playgroud)

有什么我想念的吗?!提前致谢。

Bob*_*ick 6

我知道这是一个老问题,但我想在这里发布一个答案,以供其他可能希望自定义 ROM 或在使用内部或受限 API 的现有设备上安装他们自己的应用程序的人使用。我不得不搜索相当长的时间来收集我在这里提供的信息,因为信息可能很难找到。

为了使priv-appAPK 正常工作,甚至完全取决于权限,它需要使用系统密钥进行签名,该系统密钥已用于 ROM 中的关键系统应用程序和框架 JAR 文件(通常在/system/framework.

在某些情况下(例如基于联发科 CPU 的中文 ROM 等),所使用的 APK 签名密钥是标准的 Android AOSP(Android 开源项目)签名密钥,可以在 Android 源代码分发中找到。作为 AOSP 发行版的一部分,提供了 4 个默认密钥,每个密钥都是私有/公共对。他们的名字是:

media.pk8
media.x509.pem
platform.pk8
platform.x509.pem
shared.pk8
shared.x509.pem
testkey.pk8
testkey.x509.pem
Run Code Online (Sandbox Code Playgroud)

如果您想将 APK 添加到使用这些签名密钥的 ROM,那么您很幸运!我已经包含了一个脚本和一些指向下面所需文件的链接,以完成该过程。

但是,如果您尝试自定义来自三星等主要制造商的 ROM,您可能会发现您需要使用新密钥重新签署所有系统 APK 文件,以使您的应用程序在/system/priv-app目录中工作。

您还需要修改 ROM 的框架 APK,以便它们也由您的新密钥签名,并且您可以想象这个过程充满危险,因为如果您错过一些重要的东西,它可能会导致 ROM 无法使用。此过程超出了答案的范围,但您可以在 XDA Developers 上找到适用于不同制造商的一些好的资源。我会链接它,但目前我的声誉太低,无法分享两个以上的链接。

如果您的应用程序使用任何仅可用于已签名特权系统应用程序的内部 API,则您必须执行此步骤才能调用这些 API。这些 API 包括使用反射访问的 API,以及对这些 API 的直接调用。

我在您的清单中看到您在 APK 中使用了至少两个受限权限。那些是:

<uses-permission android:name="android.permission.WRITE_APN_SETTINGS" />
Run Code Online (Sandbox Code Playgroud)

<uses-permission android:name="android.permission.WRITE_SECURE_SETTINGS" />
Run Code Online (Sandbox Code Playgroud)

没有签名,这些权限几乎没有用,尤其是WRITE_SECURE_SETTINGS权限。

我编写了一个小的 shell 脚本,priv-app在需要时对我的APK 进行签名,这是它的源代码。zip-align执行签名过程后,您还需要使用APK。它适用于 OS X/macOS 和 Linux:

#!/bin/bash
echo Signing with Android Platform key...
java -jar signapk.jar -w platform.x509.pem platform.pk8 <app-name-for-signing>.apk <app-signed-output>.apk
echo Zip-Aligning...
./zipalign -f 4 <app-signed-output>.apk <final-app-name-signed-zipaligned>.apk
echo Done!
echo
Run Code Online (Sandbox Code Playgroud)

您需要获取适用于您平台的工具signapk.jarzipalign工具,您可以在此处找到这些工具:

[跨平台] APK 签名/Zipalign/安装

您还需要获取 AOSP 标准密钥,这些密钥可在许多位置获得,包括https://github.com/android/platform_build/tree/master/target/product/security

如果您想找出priv-appROM 中的APK 文件使用哪个签名密钥,您可以使用标准 Java 工具执行两步过程,因此请确保安装了 Java SDK(任何版本),然后按照步骤操作在这个答案中:

如何找出用于签署应用程序的密钥库?

如果您发现 ROM 正在使用 AOSP 密钥,那么您只需要使用相同的密钥签署您自己的 APK 文件并将其放入priv-app目录中即可正常工作。