android 底層驅(qū)動實際linux, linux中大量使用了shell。那 shell到底是什么東西呢?個人理解相當于windows中的
成都創(chuàng)新互聯(lián)公司長期為上1000+客戶提供的網(wǎng)站建設(shè)服務(wù),團隊從業(yè)經(jīng)驗10年,關(guān)注不同地域、不同群體,并針對不同對象提供差異化的產(chǎn)品和服務(wù);打造開放共贏平臺,與合作伙伴共同營造健康的互聯(lián)網(wǎng)生態(tài)環(huán)境。為饒平企業(yè)提供專業(yè)的做網(wǎng)站、網(wǎng)站制作,饒平網(wǎng)站改版等技術(shù)服務(wù)。擁有十年豐富建站經(jīng)驗和眾多成功案例,為您定制開發(fā)。
cmd,但是shell很強大,如果你能熟練使用shell,在android開發(fā)中如虎添翼。
下面就android常用的shell進行一下說明:
1. cd (change directory)
如: cd / 跳轉(zhuǎn)到根目錄 cd ~ 跳轉(zhuǎn)到用戶所在的目錄
2. ls (list)
顯示目錄結(jié)構(gòu)
3. chmod 777 path
在開發(fā)過程中,如果發(fā)現(xiàn)文件不能讀寫,首先應(yīng)該想到是否給user權(quán)限,可以通過該命令試試看
改變目錄屬性,如果目錄下面還有子目錄,加上-R
4. chown 該變目錄所有者
如果目錄還有子目錄,加上-R
5. rm 目錄
如果是文件加上-f
如果是目錄加上-r
6. find
找文件,如果找到R.java文件,然后刪除它
find . -name R.java|args rm -rf
find . -name *.svn|xargs rm -rf
find . -name *.class|xargs rm -rf
7. 替換,如某個文件中根據(jù)某個模式替換某行
如下命令就是找到FPTitlebar.java這個java文件,找不到// pm.shutDown();,用pm.shutDown();替換之。
find . -name FPTitlebar.java -exec sed -i 's\// pm.shutDown();\ pm.shutDown();\' {} \;
比較難的如下所示:
find . -name *.java -exec sed -i 's\KeyEvent.KEYCODE_2\KeyEvent.KEYCODE_CAMERA \g ' {} \;
find . -name *.java -exec sed -i 's\KeyEvent.KEYCODE_1\KeyEvent.KEYCODE_CALL \g ' {} \;
找到所有的java文件,用KeyEvent.KEYCODE_CAMERA替換KeyEvent.KEYCODE_2
8. 編譯android源碼時實際也是執(zhí)行shell命令:
//執(zhí)行build 目錄下envsetup.sh命令
. build/envsetup.sh
//彈出選擇框,分別選擇第一個,第一個,第五個,第三個
choosecombo 1 1 5 3
//設(shè)置環(huán)境變量
export ANDROID_JAVA_HOME=$JAVA_HOME
//執(zhí)行update命令
make update-api
//起4 個線程同時編譯
make -j 4
9. 如果你使用了第三so包,需要在android編譯,直接在mk文件中添加如下設(shè)置即可:
如下所示:
1. 聲明library名稱
###############################
LOCAL_STATIC_JAVA_LIBRARIES := xstream
###############################
2. 加入引入的庫文件
###############################
include $(CLEAR_VARS)
LOCAL_PREBUILT_STATIC_JAVA_LIBRARIES := xstream:xstream-1.3.1.jar
include $(BUILD_MULTI_PREBUILT)
###############################
一、Android應(yīng)用啟動服務(wù)執(zhí)行腳本
1 如何寫服務(wù)和腳本
在android源碼根目錄下有/device/tegatech/tegav2/init.rc文件相信大家對這個文件都不陌生(如果不明白就仔細研讀下android啟動流程)。如果在該腳本文件中添加諸如以下服務(wù):
service usblp_test /data/setip/init.usblpmod.sh
oneshot
disabled
注解:每個設(shè)備下都會有自己對應(yīng)的init.rc,init.設(shè)備名.rc腳本文件。oneshot disabled向我們說明了在系統(tǒng)啟動的時候這個服務(wù)是不會自動啟動的。并且該服務(wù)的目的是執(zhí)行/data/setip/init.usblpmod.sh腳本。腳本的內(nèi)容你可以隨便寫,只要符合shell語法就可以了,比如腳本可以是簡單的設(shè)置eth0:
# ! /system/bin/sh //腳本的開頭必須這樣寫。
Ifconfig eth0 172.16.100.206 netmask 255.255.0.0 up//設(shè)置ip的命令
2、如何在應(yīng)用中啟動服務(wù)
1)首先了解下在服務(wù)啟動的流程
1. 在你的應(yīng)用中讓init.rc中添加的服務(wù)啟動起來。
首先了解下在服務(wù)啟動的流程:
在設(shè)備目錄下的init.c(切記并不是system/core/init/init.rc)
Main函數(shù)的for(;;)循環(huán)中有一個handle_property_set_fd(),函數(shù):
for (i = 0; i fd_count; i++) {
if (ufds[i].revents == POLLIN) {
if (ufds[i].fd == get_property_set_fd())
handle_property_set_fd();
else if (ufds[i].fd == get_keychord_fd())
handle_keychord();
else if (ufds[i].fd == get_signal_fd())
handle_signal();
}
}
這個函數(shù)的實現(xiàn)也在system/core/init目錄下,該函數(shù)中的check_control_perms(msg.value, cr.uid, cr.gid)函數(shù)就是檢查該uid是否有權(quán)限啟動服務(wù)(msg.value就是你服務(wù)的名字),如果應(yīng)用為root或system用戶則直接返回1.之后就是調(diào)用handle_control_message((char*) msg.name + 4, (char*) msg.value),該函數(shù)的參數(shù)就是去掉1.ctl.后的start和2.你服務(wù)的名字。這個函數(shù)的詳細內(nèi)容:
void handle_control_message(const char *msg, const char *arg)
{
if (!strcmp(msg,"start")) {
msg_start(arg);
} else if (!strcmp(msg,"stop")) {
msg_stop(arg);
} else if (!strcmp(msg,"restart")) {
msg_stop(arg);
msg_start(arg);
} else {
ERROR("unknown control msg '%s'\n", msg);
}
}
匹配start后調(diào)用msg_start.服務(wù)就這樣起來了,我們的解決方案就是在檢查權(quán)限的地方“下點功夫”,因為我們不確定uid,所以就讓check_control_perms這個函數(shù)不要檢查我們的uid,直接檢查我們服務(wù)的名字,看看這個函數(shù):
static int check_control_perms(const char *name, unsigned int uid, unsigned int gid) {
int i;
if (uid == AID_SYSTEM || uid == AID_ROOT)
return 1;
/* Search the ACL */
for (i = 0; control_perms[i].service; i++) {
if (strcmp(control_perms[i].service, name) == 0) {
if ((uid control_perms[i].uid == uid) ||
(gid control_perms[i].gid == gid)) {
return 1;
}
}
}
return 0;
}
這個函數(shù)里面是必須要檢查uid的,我們只要在for循環(huán)上寫上。
if(strcmp(“usblp_test”,name)==0) //usblp_test就是我們服務(wù)的名字。
return 1;
這樣做不會破壞android原本的結(jié)構(gòu),不會有什么副作用。
init.c和init.rc都改好了,現(xiàn)在就可以編譯源碼了,編譯好了裝到機子開發(fā)板上就可以了。
rc是android初始化腳本,用android init language編寫,通過init程序來執(zhí)行。sh是linux里面的shell腳本,可以在sh或者 bash里面執(zhí)行。
在Android中使用啟動腳本init.rc,可以在系統(tǒng)的初始化中進行簡單的操作。
init.rc啟動腳本路徑:system/core/rootdir/init.rc
內(nèi)容:
Commands:命令
Actions:動作
Triggers:觸發(fā)條件
Services:服務(wù)
Options:選項
Properties:屬性
Commands是一些基本操作。如:
mkdir /system
mkdir /data 0771 system system
mkdir /persist 0771 system system
devwait /dev/block/mmcblk0p12
mount ext3 /dev/block/mmcblk0p
Action表示一系列命令,通常在Triggers中調(diào)用,如:
on init //表示一個觸發(fā)條件
sysclktz 0
loglevel 3
# setup the global environment
export PATH /sbin:/system/sbin:/system/bin:/system/xbin
export LD_LIBRARY_PATH /system/lib
export ANDROID_BOOTLOGO 1
Services通常表示啟動一個可執(zhí)行程序,Options是服務(wù)的附加內(nèi)容,用于配合服務(wù)使用。
service vold /system/bin/vold //vold是服務(wù)名稱,/system/bin/vold是所對應(yīng)的可執(zhí)行程序。
socket vold stream 0660 root mount //socket是配合服務(wù)使用的選項
ioprio be 2
service netd /system/bin/netd
socket netd stream 0660 root system
配合服務(wù)使用的選項有socket,user,group,oneshot。
oneshot表示該服務(wù)只啟動一次,而如果沒有oneshot選項,這個可執(zhí)行程序?qū)⒁恢贝嬖凇绻蓤?zhí)行程序被殺死,則會重新啟動。
Properties是系統(tǒng)中使用的一些值,可以進行設(shè)置和讀寫。
setprop ro.HIDDEN_APP_MEM 5120 //setprop用于設(shè)置屬性
setprop ro.CONTENT_PROVIDER_MEM 5632
setprop ro.EMPTY_APP_MEM 6144
...
on property:ro.kernel.qemu=1 //on property用于判斷屬性
start adbd
這里的屬性在整個android系統(tǒng)運行中都是一致的。
init腳本的關(guān)鍵字可以參考init進程中的system/core/init/keyword.h文件。如:
KEYWORD(chroot, COMMAND, 1, do_chroot) //chroot是命令,do_chroot()是調(diào)用的函數(shù),這個函數(shù)在init進程中的system/core/init/builtins.c文件中定義。
sh是腳本文件,類似與windows里的批處理文件,也有的sh文件是安裝包。然后這個就是你可以自己找下一些軟件啥的,看行不行,我也沒怎么遇到過這個,你可以就是用應(yīng)用寶啥的打開看行不行,要是能用就是可以的,不行就是再換這個windows的軟件
android中的sh不支持“((",expr,這些,要用
i=$(($1-1))
#!/system/bin/sh
i=100
while [ i -gt 0 ]
do
echo $i
i=$((i-1))
done
下面是我自己實驗的一個只循環(huán)十次的結(jié)果
root@android:/ # i=10;while [ i -gt 0 ];do echo $i;i=$(($i-1));done
10
9
8
7
6
5
4
3
2
1