[轉]如何使Android應用程序獲取系統權限來修改系統時間


在 android 的API中有提供 SystemClock.setCurrentTimeMillis()函數來修改系統時間,可惜無論你怎么調用這個函數都是沒用的,無論模擬器還是真機,在logcat中總會得到"Unable to open alarm driver: Permission denied ".這個函數需要root權限或者運行與系統進程中才可以用。

本來以為就沒有辦法在應用程序這一層改系統時間了,后來在網上搜了好久,知道這個目的還是可以達到的。

 

第一個方法簡單點,不過需要在Android系統源碼的環境下用make來編譯:

1. 在應用程序的AndroidManifest.xml中的manifest節點中加入android:sharedUserId="android.uid.system"這個屬性。

2. 修改Android.mk文件,加入LOCAL_CERTIFICATE := platform這一行

3. 使用mm命令來編譯,生成的apk就有修改系統時間的權限了。

 

 

第二個方法麻煩點,不過不用開虛擬機跑到源碼環境下用make來編譯:

1. 同上,加入android:sharedUserId="android.uid.system"這個屬性。

2. 使用eclipse編譯出apk文件,但是這個apk文件是不能用的。

3. 用壓縮軟件打開apk文件,刪掉META-INF目錄下的CERT.SF和CERT.RSA兩個文件。

4. 使用目標系統的platform密鑰來重新給apk文件簽名。這步比較麻煩,首先找到密鑰文件,在我的Android源碼目錄中的位置是"build/target/product/security",下面的platform.pk8和platform.x509.pem兩個文件。然后用Android提供的Signapk工具來簽名,signapk的源代碼是在"build/tools/signapk"下,用法為"signapk platform.x509.pem platform.pk8 input.apk output.apk",文件名最好使用絕對路徑防止找不到,也可以修改源代碼直接使用。

這樣最后得到的apk和第一個方法是一樣的。

 

最后解釋一下原理,首先加入android:sharedUserId="android.uid.system"這個屬性。通過Shared User id,擁有同一個User id的多個APK可以配置成運行在同一個進程中。那么把程序的UID配成android.uid.system,也就是要讓程序運行在系統進程中,這樣就有權限來修改系統時間了。

只是加入UID還不夠,如果這時候安裝APK的話發現無法安裝,提示簽名不符,原因是程序想要運行在系統進程中還要有目標系統的platform key,就是上面第二個方法提到的platform.pk8和platform.x509.pem兩個文件。用這兩個key簽名后apk才真正可以放入系統進程中。第一個方法中加入LOCAL_CERTIFICATE := platform其實就是用這兩個key來簽名。

這也有一個問題,就是這樣生成的程序只有在原始的Android系統或者是自己編譯的系統中才可以用,因為這樣的系統才可以拿到platform.pk8和platform.x509.pem兩個文件。要是別家公司做的Android上連安裝都安裝不了。試試原始的Android中的key來簽名,程序在模擬器上運行OK,不過放到G3上安裝直接提示"Package ... has no signatures that match those in shared user android.uid.system",這樣也是保護了系統的安全。

 

我采用的是第二種方法,第一種方案說的不清不楚,真的難以讓人理解,第二種方案也沒有說清楚,自己試了好多次才成功,現在把具體的實施步驟分享給大家:

注意:第四步以后的工作要在源碼的環境下操作,我的環境是Ubuntu + 源碼,這個源碼所編譯的鏡像應該是你目標所用的鏡像。

1、首先在manifest文件加入android:sharedUserId="android.uid.system"這個屬性。

2. 使用eclipse編譯出XXX.apk文件,但是這個apk文件是不能用的。

3. 用壓縮軟件打開XXX.apk文件,刪掉META-INF目錄下的CERT.SF和CERT.RSA兩個文件,win32系統用WinRAR就可以了。

4、使用目標系統的platform密鑰來重新給apk文件簽名。這步比較麻煩,首先找到密鑰文件,在我的Android源碼目錄中的位置是"build/target/product/security",下面的platform.pk8和platform.x509.pem兩個文件,可以把這二個文件拷貝到tmp目錄下,把第三步刪除文件以后的apk包也拷貝到這個目錄下。

5、編譯出signapk.jar文件,signapk的源碼在android源碼目錄的/build/tools/signapk下,首先CD到改目錄下,然后通過Javac signapk.java命令編譯該原文件,得到SignApk$SignatureOutputStream.class和Signapk.class文件,然后手動創建k/com/android/signapk,注意,這個目錄必須是在源碼build/tools/signapk下的目錄,pwd顯示的目錄應該是/build/tools/signapkk/com/android/signapk,然后將二個class文件放到該目錄下,然后在/build/tools/signapk目錄下執行jar cvfm signapk.jar SignaApk.mf -C k/ .這樣在該目錄下就可以生成signapk.jar文件了,得到Android提供的簽名程序,擁有系統的權限。(如果有異常就說明你操作有問題,是不對的)

6、把生成的signapk.jar文件拷貝到源碼的tmp目錄下,然后用命令:java -jar signapk.jar platform.x509.pem  platform.pk8  XXX.apk  XXX_New.apk,如果沒有異常就說明新生成的XXX_New.apk已經被簽名了。

7、把生成的apk裝載到目標機就可以運行了,如adb install  XXX_New.apk.

再來看看修改系統時間的方法吧!源碼中是這樣的:

 1 static void setDate(int year, int month, int day) {
 2         Calendar c = Calendar.getInstance();
 3 
 4 c.set(Calendar.YEAR, year);
 5         c.set(Calendar.MONTH, month);
 6         c.set(Calendar.DAY_OF_MONTH, day);
 7         long when = c.getTimeInMillis();
 8 
 9 if (when / 1000 < Integer.MAX_VALUE) {
10             SystemClock.setCurrentTimeMillis(when);
11         }
12     }
13 
14 static void setTime(int hourOfDay, int minute) {
15         Calendar c = Calendar.getInstance();
16 
17 c.set(Calendar.HOUR_OF_DAY, hourOfDay);
18         c.set(Calendar.MINUTE, minute);
19         c.set(Calendar.SECOND, 0);
20         c.set(Calendar.MILLISECOND, 0);
21         long when = c.getTimeInMillis();
22 
23 if (when / 1000 < Integer.MAX_VALUE) {
24             SystemClock.setCurrentTimeMillis(when);
25         }
26     }

 

附上源碼下載地址:http://download.csdn.net/detail/weijing331/4530873


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM