init.rc中添加服务以在开机时启动脚本

在我们的日常开发过程中,做一些系统初始化操作那是司空见惯的,那怎么实现这样的功能则需要看实际的应用场景。最常用的操作是就是在系统模块起来时执行初始化,创建实例等等,可以在 WMS,AMS这些地方加入,或是 SystemReady 的时候加上,至于具体加在哪里,当然得看你的实际应用场景,保持模块的干净整洁和解耦也是一大任务和要素。
但这次我们却收到了一个特殊的需求,需要第一次开机时做一些数据的拷贝。单纯的拷贝而不带任何业务逻辑。于是我们一开始便选择将拷贝的操作放在一个预制数据的模块当中,但后来的测试中发现,该模块和实际始终的模块其实是两个进程。有可能会导致数据库未拷贝完全就被拿去使用并且刷新,导致数据不正常,从而引发开机后无法运行的问题。而实际使用的模块又紧紧只是 system 权限,没有 mount 和 mkdir 的权力。GG。无奈之下,只能尝试在原来的模块做一个 md5 的校验,不对则重新拷贝。但问题又来了,在 Cpp 代码中做 MD5 校验没有现成的方法,还要导入一堆函数,而且执行时间也相对较晚,不够符合我们的要求,最终喜欢偷懒的我们想到了在 init.rc 中执行脚本的方法来完成,重点是用命令行执行 MD5 校验实在是方便了很多很多。
init.rc 中可以启动一个脚本,来完成特定的操作。

service cp_xxx_database /system/etc/init.xxx.database.sh
    class main
    user root
    group root
    oneshot
复制代码

上面的代码说明了一下几点:

  1. 启动的 section 是一个 service,该服务名字为 cp_xxx_database
  2. 启动之后执行的操作是执行一个位于 /system/etc/ 目录下的 init.xxx.database.sh 脚本。
  3. class main 标注了启动方式,通过在 init.rc 中的 class_start main 指令来启动该服务。
  4. user rootgroup root 说明了使用的是 root 权限。
  5. oneshot 说明的是该操作只会执行一次,并不像其他带有 restart 指令的 service 一样当被 kill 调之后会重新调起。

而该脚本则只是一个普通的 shell 脚本。并不复杂,只是跟在服务器上的环境不一样,有些命令需要标明路径。例如脚本开头是 #!/system/bin/sh。而不是我们平时普通的 #!/bin/sh,这一点需要注意一下。

#!/system/bin/sh

XxxDatabase="/data/xxx/data/DATABASE"
PreXxxDatabase="/system/etc/XXXXX_XXX_DATABASE"

dbMd5=""
preDbMd5=""

echo "check /data/xxx/data/TAG" > /dev/ttyAMA0

if [ -f "/data/xxx/data/TAG" ];then
	echo "xxxxx xxx database is exit" > /dev/ttyAMA0
	exit 0;
fi   

echo "xxxxx xxx database is not exit, it will cp" > /dev/ttyAMA0

while :
do  
    rm -rf $XxxDatabase
	mkdir -p /data/xxx/data/
	chmod 777 /data/xxx/data/
	cp -f $PreXxxDatabase $XxxDatabase
	sync
	preDbMd5=$(md5 $PreXxxDatabase | busybox awk '{print $1}')
	dbMd5=$(md5 $XxxDatabase | busybox awk '{print $1}')
	echo "cping" > /dev/ttyAMA0
	if [ "$dbMd5" == "$preDbMd5" ];then
		touch /data/xxx/data/TAG
		chmod 777 /data/xxx/data/TAG
		break
	fi
	sleep 0.3
done 

echo "copy done" > /dev/ttyAMA0
复制代码

你可能感兴趣的:(init.rc中添加服务以在开机时启动脚本)