summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRobert Nagy <robert@cvs.openbsd.org>2014-07-12 10:14:04 +0000
committerRobert Nagy <robert@cvs.openbsd.org>2014-07-12 10:14:04 +0000
commitd3fa9b25ec8da206f184b0dc56125e71d89db262 (patch)
tree33b1901e8e13db010107d890d1e5c24135b99315
parent4697827b62393caebc5c11a7b9169add6ccbf52a (diff)
Make rc.conf a parsed configuration file and stop sourcing it as a shell
script. From now on rc.conf has a fixed syntax (key=val) and it is not allowed to add anything to it besides the supported syntax, it all going to be ignored. discussed with and help from deraadt@ and halex@
-rw-r--r--etc/netstart7
-rw-r--r--etc/rc9
-rw-r--r--etc/rc.conf35
-rw-r--r--etc/rc.d/rc.subr180
4 files changed, 134 insertions, 97 deletions
diff --git a/etc/netstart b/etc/netstart
index 464aafae782..451f77fa5b1 100644
--- a/etc/netstart
+++ b/etc/netstart
@@ -1,6 +1,6 @@
#!/bin/sh -
#
-# $OpenBSD: netstart,v 1.139 2013/08/22 07:53:11 mpi Exp $
+# $OpenBSD: netstart,v 1.140 2014/07/12 10:14:03 robert Exp $
# Strip comments (and leading/trailing whitespace if IFS is set)
# from a file and spew to stdout
@@ -148,8 +148,9 @@ ifmstart() {
done
}
-# Re-read /etc/rc.conf
-. /etc/rc.conf
+# re-read rc.subr if we are not inside /etc/rc
+[ -n ${INRC} ] && FUNCS_ONLY=1 . /etc/rc.d/rc.subr
+_rc_parse_conf
# If we were invoked with a list of interface names, just reconfigure these
# interfaces (or bridges) and return.
diff --git a/etc/rc b/etc/rc
index 48e42b7ef9b..fda7631bb76 100644
--- a/etc/rc
+++ b/etc/rc
@@ -1,4 +1,4 @@
-# $OpenBSD: rc,v 1.427 2014/04/25 17:59:53 bluhm Exp $
+# $OpenBSD: rc,v 1.428 2014/07/12 10:14:03 robert Exp $
# System startup script run by init on autoboot
# or after single-user.
@@ -224,8 +224,11 @@ if [ -f /etc/defaultdomain ]; then
domainname `stripcom /etc/defaultdomain`
fi
-# pick up option configuration
-. /etc/rc.conf
+# need to get local functions from rc.subr
+FUNCS_ONLY=1 . /etc/rc.d/rc.subr
+
+# load rc.conf into scope
+_rc_parse_conf
if [ X"$1" = X"shutdown" ]; then
random_seed
diff --git a/etc/rc.conf b/etc/rc.conf
index 7bd994d5ac3..5b768067102 100644
--- a/etc/rc.conf
+++ b/etc/rc.conf
@@ -1,6 +1,4 @@
-#!/bin/sh -
-#
-# $OpenBSD: rc.conf,v 1.191 2014/07/11 21:58:32 tedu Exp $
+# $OpenBSD: rc.conf,v 1.192 2014/07/12 10:14:03 robert Exp $
# DO NOT EDIT THIS FILE!!
#
@@ -24,7 +22,7 @@ bgpd_flags=NO # for normal use: ""
rarpd_flags=NO # for normal use: "-a"
bootparamd_flags=NO # for normal use: ""
rbootd_flags=NO # for normal use: ""
-sshd_flags="" # for normal use: ""
+sshd_flags= # for normal use: ""
named_flags=NO # for normal use: ""
nsd_flags=NO # for normal use: "-c /var/nsd/etc/nsd.conf"
unbound_flags=NO # for normal use: "-c /var/unbound/etc/unbound.conf"
@@ -54,8 +52,8 @@ hostapd_flags=NO # for normal use: ""
ifstated_flags=NO # for normal use: ""
relayd_flags=NO # for normal use: ""
snmpd_flags=NO # for normal use: ""
-smtpd_flags="" # for normal use: ""
-sndiod_flags="" # for normal use: ""
+smtpd_flags= # for normal use: ""
+sndiod_flags= # for normal use: ""
ldapd_flags=NO # for normal use: ""
npppd_flags=NO # for normal use: ""
inetd_flags=NO # for normal use: ""
@@ -66,7 +64,7 @@ tftpd_flags=NO # for normal use: "[chroot dir]"
tftpproxy_flags=NO # for normal use: ""
ldomd_flags=NO # for normal use: ""
identd_flags=NO # for normal use: "-e"
-cron_flags="" # for normal use: ""
+cron_flags= # for normal use: ""
# use -u to disable chroot, see nginx(8)
nginx_flags=NO # for normal use: ""
@@ -76,7 +74,7 @@ slowcgi_flags=NO # for normal use: ""
sendmail_flags=NO
spamd_flags=NO # for normal use: "" and see spamd(8)
spamd_black=NO # set to YES to run spamd without greylisting
-spamlogd_flags="" # use eg. "-i interface" and see spamlogd(8)
+spamlogd_flags= # use eg. "-i interface" and see spamlogd(8)
# Set to NO if ftpd is running out of inetd
ftpd_flags=NO # for non-inetd use: ""
@@ -119,24 +117,3 @@ shlib_dirs= # extra directories for ldconfig, separated
# rc.d(8) packages scripts
# started in the specified order and stopped in reverse order
pkg_scripts=
-
-unset mountd_flags nfsd_flags ypbind_flags
-
-[ -f /etc/rc.conf.local ] && . /etc/rc.conf.local
-
-# special care needed for spamlogd to avoid starting it up and failing
-# all the time
-if [ X"${spamd_flags}" = X"NO" -o X"${spamd_black}" != X"NO" ]; then
- spamlogd_flags=NO
-fi
-
-# special care needed for pflogd to avoid starting it up and failing
-# if pf is not enabled
-if [ X"${pf}" = X"NO" ]; then
- pflogd_flags=NO
-fi
-
-# backward compatibility
-: ${mountd_flags=$([ X"${nfs_server-NO}" = XYES ] || echo NO)}
-: ${nfsd_flags=$([ X"${nfs_server-NO}" = XYES ] && echo "-tun 4" || echo NO)}
-: ${ypbind_flags=$([ X"`domainname`" != X"" -a -d /var/yp/binding ] || echo NO)}
diff --git a/etc/rc.d/rc.subr b/etc/rc.d/rc.subr
index b3b251557b3..f6190913c85 100644
--- a/etc/rc.d/rc.subr
+++ b/etc/rc.d/rc.subr
@@ -1,8 +1,8 @@
-# $OpenBSD: rc.subr,v 1.72 2014/07/09 14:19:22 ajacoutot Exp $
+# $OpenBSD: rc.subr,v 1.73 2014/07/12 10:14:03 robert Exp $
#
# Copyright (c) 2010, 2011, 2014 Antoine Jacoutot <ajacoutot@openbsd.org>
# Copyright (c) 2010, 2011 Ingo Schwarze <schwarze@openbsd.org>
-# Copyright (c) 2010, 2011 Robert Nagy <robert@openbsd.org>
+# Copyright (c) 2010, 2011, 2014 Robert Nagy <robert@openbsd.org>
#
# Permission to use, copy, modify, and distribute this software for any
# purpose with or without fee is hereby granted, provided that the above
@@ -19,57 +19,41 @@
# Default functions and variables used by rc.d(8) scripts.
-rc_err() {
+_rc_err() {
echo $1 1>&2
exit 1
}
-rc_is_supported() {
+_rc_is_supported() {
local _enotsup
eval _enotsup=\${rc_$1}
[ X"${_enotsup}" != X"NO" ]
}
-rc_usage() {
+_rc_usage() {
local _a _allsup
for _a in start stop restart reload check; do
- rc_is_supported ${_a} && _allsup="${_allsup:+$_allsup|}${_a}"
+ _rc_is_supported ${_a} && _allsup="${_allsup:+$_allsup|}${_a}"
done
- rc_err "usage: $0 [-df] (${_allsup})"
+ _rc_err "usage: $0 [-df] (${_allsup})"
}
-rc_write_runfile() {
+_rc_write_runfile() {
[ -d ${_RC_RUNDIR} ] || mkdir -p ${_RC_RUNDIR} && \
print -rn -- "${pexp}" > ${_RC_RUNFILE}
}
-rc_read_runfile() {
+_rc_read_runfile() {
local _new_pexp
[ -f ${_RC_RUNFILE} ] && _new_pexp=$(< ${_RC_RUNFILE})
[ -n "${_new_pexp}" ] && pexp="${_new_pexp}"
}
-rc_rm_runfile() {
+_rc_rm_runfile() {
rm -f ${_RC_RUNFILE}
}
-rc_start() {
- ${rcexec} "${daemon} ${daemon_flags} ${_bg}"
-}
-
-rc_check() {
- pgrep -q -f "^${pexp}"
-}
-
-rc_reload() {
- pkill -HUP -f "^${pexp}"
-}
-
-rc_stop() {
- pkill -f "^${pexp}"
-}
-
-rc_do() {
+_rc_do() {
if [ -n "${_RC_DEBUG}" ]; then
echo "doing $@" && "$@"
else
@@ -77,22 +61,22 @@ rc_do() {
fi
}
-rc_exit() {
+_rc_exit() {
local _pfix
[ -z "${INRC}" -o X"$1" != X"ok" ] && _pfix="($1)"
echo ${INRC:+'-n'} "${_pfix}"
[ X"$1" = X"ok" ] && exit 0 || exit 1
}
-rc_wait() {
+_rc_wait() {
local _i=0
while [ $_i -lt ${daemon_timeout} ]; do
case "$1" in
reload|start)
- rc_do rc_check && return 0
+ _rc_do rc_check && return 0
;;
stop)
- rc_do rc_check || return 0
+ _rc_do rc_check || return 0
;;
*)
break
@@ -104,95 +88,164 @@ rc_wait() {
return 1
}
+_rc_quirks() {
+ unset mountd_flags nfsd_flags ypbind_flags
+
+ # special care needed for spamlogd to avoid starting it up and failing
+ # all the time
+ if [ X"${spamd_flags}" = X"NO" -o X"${spamd_black}" != X"NO" ]; then
+ spamlogd_flags=NO
+ fi
+
+ # special care needed for pflogd to avoid starting it up and failing
+ # if pf is not enabled
+ if [ X"${pf}" = X"NO" ]; then
+ pflogd_flags=NO
+ fi
+
+ : ${mountd_flags=$([ X"${nfs_server-NO}" = XYES ] || echo NO)}
+ : ${nfsd_flags=$([ X"${nfs_server-NO}" = XYES ] && echo "-tun 4" || echo NO)}
+ : ${ypbind_flags=$([ X"`domainname`" != X"" -a -d /var/yp/binding ] || echo NO)}
+}
+
+_rc_parse_conf() {
+ typeset -l _key
+ local _l _val
+ local _rcconf="/etc/rc.conf"
+ local _rcconf_local="/etc/rc.conf.local"
+ set -A _allowed_keys -- \
+ spamd_black pf ipsec check_quotas accounting \
+ multicast_hosts multicast_router amd_master \
+ pf_rules ipsec_rules shlib_dirs pkg_scripts \
+ nfs_server
+
+ for _rcfile in $_rcconf $_rcconf_local; do
+ [[ -f $_rcfile ]] || return
+ while IFS=' ' read -r _l; do
+ [[ $_l == [!#=]*=* ]] || continue
+ _key=${_l%%*([[:blank:]])=*}
+ [[ $_key == *_@(flags|user|timeout) ]] || \
+ [[ " ${_allowed_keys[*]} " == *" $_key "* ]] || \
+ continue
+ [[ $_key == "" ]] && continue
+ _val=${_l##*([!=])=*([[:blank:]])}
+ _val=${_val%%#*}
+ _val=${_val%%*([[:blank:]])}
+ # remove leading and trailing quotes (backwards compat)
+ [[ $_val == @(\"*\"|\'*\') ]] && _val=${_val#?} _val=${_val%?}
+ [ -n "${_RC_DEBUG}" ] && printf "%18s\t>$_val<\n" $_key
+ eval "${_key}=\${_val}"
+ done < $_rcfile
+ done
+
+ _rc_do _rc_quirks
+}
+
+[ -n "${FUNCS_ONLY}" ] && return
+
+rc_start() {
+ ${rcexec} "${daemon} ${daemon_flags} ${_bg}"
+}
+
+rc_check() {
+ pgrep -q -f "^${pexp}"
+}
+
+rc_reload() {
+ pkill -HUP -f "^${pexp}"
+}
+
+rc_stop() {
+ pkill -f "^${pexp}"
+}
+
rc_cmd() {
local _bg _n
[ "$(id -u)" -eq 0 ] || \
[ X"${rc_usercheck}" != X"NO" -a X"$1" = "Xcheck" ] || \
- rc_err "$0: need root privileges"
+ _rc_err "$0: need root privileges"
- if ! (rc_is_supported start && rc_is_supported stop); then
+ if ! (_rc_is_supported start && _rc_is_supported stop); then
rc_restart=NO
fi
- if ! rc_is_supported $1; then
+ if ! _rc_is_supported $1; then
[ -n "${INRC}" ] && exit 1
- rc_err "$0: $1 is not supported"
+ _rc_err "$0: $1 is not supported"
fi
[ X"${rc_bg}" = X"YES" ] && _bg="&"
[ -n "${_RC_DEBUG}" ] || _n="-n"
- rc_do rc_read_runfile
+ _rc_do _rc_read_runfile
case "$1" in
check)
- rc_do rc_check
+ _rc_do rc_check
;;
start)
if [ X"${daemon_flags}" = X"NO" ]; then
- rc_err "$0: need -f to force $1 since ${_name}_flags=NO"
+ _rc_err "$0: need -f to force $1 since ${_name}_flags=NO"
exit 1
fi
- [ -z "${INRC}" ] && rc_do rc_check && exit 0
+ [ -z "${INRC}" ] && _rc_do rc_check && exit 0
echo $_n "${INRC:+ }${_name}"
while true; do # no real loop, only needed to break
if type rc_pre >/dev/null; then
- rc_do rc_pre || break
+ _rc_do rc_pre || break
fi
# XXX only checks the status of the return code,
# and _not_ that the daemon is actually running
- rc_do rc_start || break
+ _rc_do rc_start || break
if [ -n "${_bg}" ]; then
sleep 1
- rc_do rc_wait start || break
+ _rc_do _rc_wait start || break
fi
- rc_do rc_write_runfile
- rc_exit ok
+ _rc_do _rc_write_runfile
+ _rc_exit ok
done
# handle failure
- type rc_post >/dev/null && rc_do rc_post
- rc_do rc_rm_runfile
- rc_exit failed
+ type rc_post >/dev/null && _rc_do rc_post
+ _rc_do _rc_rm_runfile
+ _rc_exit failed
;;
stop)
- rc_do rc_check || exit 0
+ _rc_do rc_check || exit 0
echo $_n "${INRC:+ }${_name}"
- rc_do rc_stop || rc_exit failed
- rc_do rc_wait stop || rc_exit failed
+ _rc_do rc_stop || _rc_exit failed
+ _rc_do _rc_wait stop || _rc_exit failed
if type rc_post >/dev/null; then \
- rc_do rc_post || rc_exit failed
+ _rc_do rc_post || _rc_exit failed
fi
- rc_do rc_rm_runfile
- rc_exit ok
+ _rc_do _rc_rm_runfile
+ _rc_exit ok
;;
reload)
- rc_do rc_check || exit 0
+ _rc_do rc_check || exit 0
echo $_n "${INRC:+ }${_name}"
- rc_do rc_reload || rc_exit failed
- rc_do rc_wait reload || rc_exit failed
- rc_exit ok
+ _rc_do rc_reload || _rc_exit failed
+ _rc_do _rc_wait reload || _rc_exit failed
+ _rc_exit ok
;;
restart)
$0 ${_RC_DEBUG} ${_RC_FORCE} stop &&
$0 ${_RC_DEBUG} ${_RC_FORCE} start
;;
*)
- rc_usage
+ _rc_usage
;;
esac
}
-. /etc/rc.conf
-
-[ -n "${daemon}" ] || rc_err "$0: daemon is not set"
+[ -n "${daemon}" ] || _rc_err "$0: daemon is not set"
unset _RC_DEBUG _RC_FORCE
while getopts "df" c; do
case "$c" in
d) _RC_DEBUG=-d;;
f) _RC_FORCE=-f;;
- *) rc_usage;;
+ *) _rc_usage;;
esac
done
shift $((OPTIND-1))
@@ -201,6 +254,9 @@ _name=$(basename $0)
_RC_RUNDIR=/var/run/rc.d
_RC_RUNFILE=${_RC_RUNDIR}/${_name}
+# parse /etc/rc.conf{.local} for the daemon_flags
+_rc_do _rc_parse_conf
+
eval _rcflags=\${${_name}_flags}
eval _rcuser=\${${_name}_user}
eval _rctimeout=\${${_name}_timeout}