functions.sh 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488
  1. #!/bin/sh
  2. # Copyright (C) 2006-2011 OpenWrt.org
  3. # Copyright (C) 2006 Fokus Fraunhofer <[email protected]>
  4. debug () {
  5. ${DEBUG:-:} "$@"
  6. }
  7. # newline
  8. N="
  9. "
  10. _C=0
  11. NO_EXPORT=1
  12. LOAD_STATE=1
  13. LIST_SEP=" "
  14. hotplug_dev() {
  15. env -i ACTION=$1 INTERFACE=$2 /sbin/hotplug-call net
  16. }
  17. append() {
  18. local var="$1"
  19. local value="$2"
  20. local sep="${3:- }"
  21. eval "export ${NO_EXPORT:+-n} -- \"$var=\${$var:+\${$var}\${value:+\$sep}}\$value\""
  22. }
  23. list_contains() {
  24. local var="$1"
  25. local str="$2"
  26. local val
  27. eval "val=\" \${$var} \""
  28. [ "${val%% $str *}" != "$val" ]
  29. }
  30. list_remove() {
  31. local var="$1"
  32. local remove="$2"
  33. local val
  34. eval "val=\" \${$var} \""
  35. val1="${val%% $remove *}"
  36. [ "$val1" = "$val" ] && return
  37. val2="${val##* $remove }"
  38. [ "$val2" = "$val" ] && return
  39. val="${val1## } ${val2%% }"
  40. val="${val%% }"
  41. eval "export ${NO_EXPORT:+-n} -- \"$var=\$val\""
  42. }
  43. config_load() {
  44. [ -n "$IPKG_INSTROOT" ] && return 0
  45. uci_load "$@"
  46. }
  47. reset_cb() {
  48. config_cb() { return 0; }
  49. option_cb() { return 0; }
  50. list_cb() { return 0; }
  51. }
  52. reset_cb
  53. package() {
  54. return 0
  55. }
  56. config () {
  57. local cfgtype="$1"
  58. local name="$2"
  59. export ${NO_EXPORT:+-n} CONFIG_NUM_SECTIONS=$(($CONFIG_NUM_SECTIONS + 1))
  60. name="${name:-cfg$CONFIG_NUM_SECTIONS}"
  61. append CONFIG_SECTIONS "$name"
  62. [ -n "$NO_CALLBACK" ] || config_cb "$cfgtype" "$name"
  63. export ${NO_EXPORT:+-n} CONFIG_SECTION="$name"
  64. export ${NO_EXPORT:+-n} "CONFIG_${CONFIG_SECTION}_TYPE=$cfgtype"
  65. }
  66. option () {
  67. local varname="$1"; shift
  68. local value="$*"
  69. export ${NO_EXPORT:+-n} "CONFIG_${CONFIG_SECTION}_${varname}=$value"
  70. [ -n "$NO_CALLBACK" ] || option_cb "$varname" "$*"
  71. }
  72. list() {
  73. local varname="$1"; shift
  74. local value="$*"
  75. local len
  76. config_get len "$CONFIG_SECTION" "${varname}_LENGTH" 0
  77. [ $len = 0 ] && append CONFIG_LIST_STATE "${CONFIG_SECTION}_${varname}"
  78. len=$(($len + 1))
  79. config_set "$CONFIG_SECTION" "${varname}_ITEM$len" "$value"
  80. config_set "$CONFIG_SECTION" "${varname}_LENGTH" "$len"
  81. append "CONFIG_${CONFIG_SECTION}_${varname}" "$value" "$LIST_SEP"
  82. list_cb "$varname" "$*"
  83. }
  84. config_rename() {
  85. local OLD="$1"
  86. local NEW="$2"
  87. local oldvar
  88. local newvar
  89. [ -n "$OLD" -a -n "$NEW" ] || return
  90. for oldvar in `set | grep ^CONFIG_${OLD}_ | \
  91. sed -e 's/\(.*\)=.*$/\1/'` ; do
  92. newvar="CONFIG_${NEW}_${oldvar##CONFIG_${OLD}_}"
  93. eval "export ${NO_EXPORT:+-n} \"$newvar=\${$oldvar}\""
  94. unset "$oldvar"
  95. done
  96. export ${NO_EXPORT:+-n} CONFIG_SECTIONS="$(echo " $CONFIG_SECTIONS " | sed -e "s, $OLD , $NEW ,")"
  97. [ "$CONFIG_SECTION" = "$OLD" ] && export ${NO_EXPORT:+-n} CONFIG_SECTION="$NEW"
  98. }
  99. config_unset() {
  100. config_set "$1" "$2" ""
  101. }
  102. config_clear() {
  103. local SECTION="$1"
  104. local oldvar
  105. list_remove CONFIG_SECTIONS "$SECTION"
  106. export ${NO_EXPORT:+-n} CONFIG_SECTIONS="${SECTION:+$CONFIG_SECTIONS}"
  107. for oldvar in `set | grep ^CONFIG_${SECTION:+${SECTION}_} | \
  108. sed -e 's/\(.*\)=.*$/\1/'` ; do
  109. unset $oldvar
  110. done
  111. }
  112. # config_get <variable> <section> <option> [<default>]
  113. # config_get <section> <option>
  114. config_get() {
  115. case "$3" in
  116. "") eval echo "\${CONFIG_${1}_${2}:-\${4}}";;
  117. *) eval export ${NO_EXPORT:+-n} -- "${1}=\${CONFIG_${2}_${3}:-\${4}}";;
  118. esac
  119. }
  120. # config_get_bool <variable> <section> <option> [<default>]
  121. config_get_bool() {
  122. local _tmp
  123. config_get _tmp "$2" "$3" "$4"
  124. case "$_tmp" in
  125. 1|on|true|enabled) _tmp=1;;
  126. 0|off|false|disabled) _tmp=0;;
  127. *) _tmp="$4";;
  128. esac
  129. export ${NO_EXPORT:+-n} "$1=$_tmp"
  130. }
  131. config_set() {
  132. local section="$1"
  133. local option="$2"
  134. local value="$3"
  135. local old_section="$CONFIG_SECTION"
  136. CONFIG_SECTION="$section"
  137. option "$option" "$value"
  138. CONFIG_SECTION="$old_section"
  139. }
  140. config_foreach() {
  141. local ___function="$1"
  142. [ "$#" -ge 1 ] && shift
  143. local ___type="$1"
  144. [ "$#" -ge 1 ] && shift
  145. local section cfgtype
  146. [ -z "$CONFIG_SECTIONS" ] && return 0
  147. for section in ${CONFIG_SECTIONS}; do
  148. config_get cfgtype "$section" TYPE
  149. [ -n "$___type" -a "x$cfgtype" != "x$___type" ] && continue
  150. eval "$___function \"\$section\" \"\$@\""
  151. done
  152. }
  153. config_list_foreach() {
  154. [ "$#" -ge 3 ] || return 0
  155. local section="$1"; shift
  156. local option="$1"; shift
  157. local function="$1"; shift
  158. local val
  159. local len
  160. local c=1
  161. config_get len "${section}" "${option}_LENGTH"
  162. [ -z "$len" ] && return 0
  163. while [ $c -le "$len" ]; do
  164. config_get val "${section}" "${option}_ITEM$c"
  165. eval "$function \"\$val\" \"$@\""
  166. c="$(($c + 1))"
  167. done
  168. }
  169. load_modules() {
  170. [ -d /etc/modules.d ] && {
  171. cd /etc/modules.d
  172. sed 's/^[^#]/insmod &/' $* | ash 2>&- || :
  173. }
  174. }
  175. include() {
  176. local file
  177. for file in $(ls $1/*.sh 2>/dev/null); do
  178. . $file
  179. done
  180. }
  181. find_mtd_index() {
  182. local PART="$(grep "\"$1\"" /proc/mtd | awk -F: '{print $1}')"
  183. local INDEX="${PART##mtd}"
  184. echo ${INDEX}
  185. }
  186. find_mtd_part() {
  187. local INDEX=$(find_mtd_index "$1")
  188. local PREFIX=/dev/mtdblock
  189. [ -d /dev/mtdblock ] && PREFIX=/dev/mtdblock/
  190. echo "${INDEX:+$PREFIX$INDEX}"
  191. }
  192. find_mtd_chardev() {
  193. local INDEX=$(find_mtd_index "$1")
  194. local PREFIX=/dev/mtd
  195. [ -d /dev/mtd ] && PREFIX=/dev/mtd/
  196. echo "${INDEX:+$PREFIX$INDEX}"
  197. }
  198. strtok() { # <string> { <variable> [<separator>] ... }
  199. local tmp
  200. local val="$1"
  201. local count=0
  202. shift
  203. while [ $# -gt 1 ]; do
  204. tmp="${val%%$2*}"
  205. [ "$tmp" = "$val" ] && break
  206. val="${val#$tmp$2}"
  207. export ${NO_EXPORT:+-n} "$1=$tmp"; count=$((count+1))
  208. shift 2
  209. done
  210. if [ $# -gt 0 -a -n "$val" ]; then
  211. export ${NO_EXPORT:+-n} "$1=$val"; count=$((count+1))
  212. fi
  213. return $count
  214. }
  215. jffs2_mark_erase() {
  216. local part="$(find_mtd_part "$1")"
  217. [ -z "$part" ] && {
  218. echo Partition not found.
  219. return 1
  220. }
  221. echo -e "\xde\xad\xc0\xde" | mtd -qq write - "$1"
  222. }
  223. uci_apply_defaults() {
  224. cd /etc/uci-defaults || return 0
  225. files="$(ls)"
  226. [ -z "$files" ] && return 0
  227. mkdir -p /tmp/.uci
  228. for file in $files; do
  229. ( . "./$(basename $file)" ) && rm -f "$file"
  230. done
  231. uci commit
  232. }
  233. group_add() {
  234. local name="$1"
  235. local gid="$2"
  236. local rc
  237. [ -f "${IPKG_INSTROOT}/etc/group" ] || return 1
  238. [ -n "$IPKG_INSTROOT" ] || lock /var/lock/group
  239. echo "${name}:x:${gid}:" >> ${IPKG_INSTROOT}/etc/group
  240. rc=$?
  241. [ -n "$IPKG_INSTROOT" ] || lock -u /var/lock/group
  242. return $rc
  243. }
  244. group_exists() {
  245. grep -qs "^${1}:" ${IPKG_INSTROOT}/etc/group
  246. }
  247. user_add() {
  248. local name="${1}"
  249. local uid="${2}"
  250. local gid="${3:-$2}"
  251. local desc="${4:-$1}"
  252. local home="${5:-/var/run/$1}"
  253. local shell="${6:-/bin/false}"
  254. local rc
  255. [ -f "${IPKG_INSTROOT}/etc/passwd" ] || return 1
  256. [ -n "$IPKG_INSTROOT" ] || lock /var/lock/passwd
  257. echo "${name}:x:${uid}:${gid}:${desc}:${home}:${shell}" >> ${IPKG_INSTROOT}/etc/passwd
  258. echo "${name}:x:0:0:99999:7:::" >> ${IPKG_INSTROOT}/etc/shadow
  259. rc=$?
  260. [ -n "$IPKG_INSTROOT" ] || lock -u /var/lock/passwd
  261. return $rc
  262. }
  263. user_exists() {
  264. grep -qs "^${1}:" ${IPKG_INSTROOT}/etc/passwd
  265. }
  266. pi_include() {
  267. if [ -f "/tmp/overlay/$1" ]; then
  268. . "/tmp/overlay/$1"
  269. elif [ -f "$1" ]; then
  270. . "$1"
  271. elif [ -d "/tmp/overlay/$1" ]; then
  272. if [ -n "$(ls /tmp/overlay/$1/*.sh 2>/dev/null)" ]; then
  273. for src_script in /tmp/overlay/$1/*.sh; do
  274. . "$src_script"
  275. done
  276. fi
  277. elif [ -d "$1" ]; then
  278. if [ -n "$(ls $1/*.sh 2>/dev/null)" ]; then
  279. for src_script in $1/*.sh; do
  280. . "$src_script"
  281. done
  282. fi
  283. else
  284. echo "WARNING: $1 not found"
  285. return 1
  286. fi
  287. return 0
  288. }
  289. #!/bin/sh
  290. # Copyright (C) 2006-2010 OpenWrt.org
  291. # Copyright (C) 2010 Vertical Communications
  292. boot_hook_splice_start() {
  293. export -n PI_HOOK_SPLICE=1
  294. }
  295. boot_hook_splice_finish() {
  296. local hook
  297. for hook in $PI_STACK_LIST; do
  298. local v; eval "v=\${${hook}_splice:+\$${hook}_splice }$hook"
  299. export -n "${hook}=${v% }"
  300. export -n "${hook}_splice="
  301. done
  302. export -n PI_HOOK_SPLICE=
  303. }
  304. boot_hook_init() {
  305. local hook="${1}_hook"
  306. export -n "PI_STACK_LIST=${PI_STACK_LIST:+$PI_STACK_LIST }$hook"
  307. export -n "$hook="
  308. }
  309. boot_hook_add() {
  310. local hook="${1}_hook${PI_HOOK_SPLICE:+_splice}"
  311. local func="${2}"
  312. [ -n "$func" ] && {
  313. local v; eval "v=\$$hook"
  314. export -n "$hook=${v:+$v }$func"
  315. }
  316. }
  317. boot_hook_shift() {
  318. local hook="${1}_hook"
  319. local rvar="${2}"
  320. local v; eval "v=\$$hook"
  321. [ -n "$v" ] && {
  322. local first="${v%% *}"
  323. [ "$v" != "${v#* }" ] && \
  324. export -n "$hook=${v#* }" || \
  325. export -n "$hook="
  326. export -n "$rvar=$first"
  327. return 0
  328. }
  329. return 1
  330. }
  331. boot_run_hook() {
  332. local hook="$1"
  333. local func
  334. while boot_hook_shift "$hook" func; do
  335. local ran; eval "ran=\$PI_RAN_$func"
  336. [ -n "$ran" ] || {
  337. export -n "PI_RAN_$func=1"
  338. $func "$1" "$2"
  339. }
  340. done
  341. }
  342. jffs2_ready () {
  343. mtdpart="$(find_mtd_part rootfs_data)"
  344. [ -z "$mtdpart" ] && return 1
  345. magic=$(hexdump $mtdpart -n 4 -e '4/1 "%02x"')
  346. [ "$magic" != "deadc0de" ]
  347. }
  348. dupe() { # <new_root> <old_root>
  349. cd $1
  350. echo -n "creating directories... "
  351. {
  352. cd $2
  353. find . -xdev -type d
  354. echo "./dev ./overlay ./mnt ./proc ./tmp"
  355. # xdev skips mounted directories
  356. cd $1
  357. } | xargs mkdir -p
  358. echo "done"
  359. echo -n "setting up symlinks... "
  360. for file in $(cd $2; find . -xdev -type f;); do
  361. case "$file" in
  362. ./rom/note) ;; #nothing
  363. ./etc/config*|\
  364. ./usr/lib/opkg/info/*) cp -af $2/$file $file;;
  365. *) ln -sf /rom/${file#./*} $file;;
  366. esac
  367. done
  368. for file in $(cd $2; find . -xdev -type l;); do
  369. cp -af $2/${file#./*} $file
  370. done
  371. echo "done"
  372. }
  373. pivot() { # <new_root> <old_root>
  374. mount -o noatime,move /proc $1/proc && \
  375. pivot_root $1 $1$2 && {
  376. mount -o noatime,move $2/dev /dev
  377. mount -o noatime,move $2/tmp /tmp
  378. mount -o noatime,move $2/sys /sys 2>&-
  379. mount -o noatime,move $2/overlay /overlay 2>&-
  380. return 0
  381. }
  382. }
  383. fopivot() { # <rw_root> <ro_root> <dupe?>
  384. root=$1
  385. {
  386. if grep -q overlay /proc/filesystems; then
  387. mount -o noatime,lowerdir=/,upperdir=$1 -t overlayfs "overlayfs:$1" /mnt && root=/mnt
  388. elif grep -q mini_fo /proc/filesystems; then
  389. mount -t mini_fo -o noatime,base=/,sto=$1 "mini_fo:$1" /mnt 2>&- && root=/mnt
  390. else
  391. mount --bind -o noatime / /mnt
  392. mount --bind -o noatime,union "$1" /mnt && root=/mnt
  393. fi
  394. } || {
  395. [ "$3" = "1" ] && {
  396. mount | grep "on $1 type" 2>&- 1>&- || mount -o noatime,bind $1 $1
  397. dupe $1 $rom
  398. }
  399. }
  400. pivot $root $2
  401. }
  402. ramoverlay() {
  403. mkdir -p /tmp/root
  404. mount -t tmpfs -o noatime,mode=0755 root /tmp/root
  405. fopivot /tmp/root /rom 1
  406. }
  407. [ -z "$IPKG_INSTROOT" -a -f /lib/config/uci.sh ] && . /lib/config/uci.sh