#! /bin/sh
# Copyright (c) 1996 - 2001 SuSE GmbH Nuernberg, Germany.  All rights reserved.
#
# Author: Florian La Roche <florian@suse.de>, 1996
#         Werner Fink <werner@suse.de>, 1996,98
#         Burchard Steinbild <bs@suse.de>, 1997
#         Thorsten Kukuk <kukuk@suse.de>, 2000,01
#
# /etc/init.d/nfsserver
#
### BEGIN INIT INFO
# Provides: nfsserver
# Required-Start: $network $remote_fs $named portmap
# Required-Stop: $network portmap
# Should-Start: ypbind svcgssd idmapd
# Should-Stop:
# Default-Start: 3 5
# Default-Stop: 0 1 2 6
# Description: Start the kernel based NFS daemon
### END INIT INFO

. /etc/sysconfig/nfs

# Shell functions sourced from /etc/rc.status:
#      rc_check         check and set local and overall rc status
#      rc_status        check and set local and overall rc status
#      rc_status -v     ditto but be verbose in local rc status
#      rc_status -v -r  ditto and clear the local rc status
#      rc_failed        set local and overall rc status to failed
#      rc_failed <num>  set local and overall rc status to <num><num>
#      rc_reset         clear local rc status (overall remains)
#      rc_exit          exit appropriate to overall rc status
. /etc/rc.status

# First reset status of this service
rc_reset

# Return values acc. to LSB for all commands but status:
# 0 - success
# 1 - generic or unspecified error
# 2 - invalid or excess argument(s)
# 3 - unimplemented feature (e.g. "reload")
# 4 - insufficient privilege
# 5 - program is not installed
# 6 - program is not configured
# 7 - program is not running
# 
# Note that starting an already running service, stopping
# or restarting a not-running service as well as the restart
# with force-reload (in case signalling is not supported) are
# considered a success.

case `uname -r` in
    0.*|1.*|2.0.*) exit 3
esac

check_for_nfsdfs() {
    HAVE_NFSDFS="no"
    while read dummy type ; do
	if [ "$type" = "nfsd" ] ; then
	    HAVE_NFSDFS="yes"
	fi
    done < /proc/filesystems
}

nfs4_bind_mounts() {

    # In case of doubt, try "man sed" :-)
    cat /etc/exports |
	    sed -n 'H;g;s/\\$//;h;t;s/^\n//;s/\n[[:space:]]*//g;s/#.*//;p;s/.*//;h' |
	    sed 's/^\([^[:space:]]*\).*bind=\([^,)]*\).*/\1 \2/;t;d' |
	    sort |
	    while read export dir; do
		test -d $export || mkdir -p $export
		# Fortunately, mount ignores unknown
		# options, so we have an easy way to
		# tag our "magic" bind mounts
	    	mount -o bind,nfsexp $dir $export
	    done
}

nfs4_unbind_mounts() {

    cat /etc/mtab |
	    grep '\<nfsexp\>' |
	    sort -r -k2 |
	    while read src mountpoint crap; do
		umount $mountpoint
	    done
}

case "$1" in
    start)
      PARAMS=3
      test "$USE_KERNEL_NFSD_NUMBER" -gt 0 && PARAMS="$USE_KERNEL_NFSD_NUMBER"

      echo -n "Starting kernel based NFS server"
      modprobe nfsd

      # this can be removed when modprobe allows the sysctl to be
      # moved to modprobe.d without breaking --show-depends
      grep '^fs.nfs.n[sl]m' /etc/sysctl.conf | sysctl -q -e -n -p /dev/stdin

      check_for_nfsdfs
      if [ "$HAVE_NFSDFS" = "yes" -a ! -f /proc/fs/nfsd/exports ] ; then
	mount -t nfsd nfsd /proc/fs/nfsd
	rc_status
      fi
      if [ "$NFS4_SUPPORT" = "yes" ]; then
       echo "+2 +3 +4" > /proc/fs/nfsd/versions
      else
       echo "+2 +3 -4" > /proc/fs/nfsd/versions
      fi
      nfs4_bind_mounts
      /usr/sbin/exportfs -r 
      rc_status
      /usr/sbin/rpc.nfsd $PARAMS
      rc_status
      if [ -n "$MOUNTD_PORT" ] ; then
	startproc /usr/sbin/rpc.mountd -p $MOUNTD_PORT
      else
	startproc /usr/sbin/rpc.mountd
      fi
      if [ "$NFS4_SUPPORT" = "yes" ]; then
	checkproc /usr/sbin/rpc.idmapd
	if [ $? -eq 0 ] ; then
	  ## Let idmapd know that nfs server is starting up
	  /etc/init.d/idmapd reload
	else
	  echo
	  echo -n 'WARNING: idmapd is not running - run "chkconfig nfsserver on"'
	fi
      fi 
      rc_status -v
      ;;
    stop)
      echo -n "Shutting down kernel based NFS server"
      killproc -n -KILL nfsd 
      rc_status
      killproc    -TERM /usr/sbin/rpc.mountd
      rc_status
      check_for_nfsdfs
      if [ "$HAVE_NFSDFS" = "yes" -a -f /proc/fs/nfsd/exports ] ; then
	umount /proc/fs/nfsd
	rc_status
      fi
      nfs4_unbind_mounts
      rc_status -v
      ;;
    try-restart)
        ## Stop the service and if this succeeds (i.e. the 
        ## service was running before), start it again.
        $0 status >/dev/null &&  $0 restart

        # Remember status and be quiet
        rc_status
        ;;
    restart)
        ## Stop the service and regardless of whether it was
        ## running or not, start it again.
        $0 stop
        $0 start

        # Remember status and be quiet
        rc_status
        ;;
    reload|force-reload)
	echo -n "Reload kernel based NFS server"
	# Unfortunately, there's no sane way of doing this:
        nfs4_unbind_mounts; nfs4_bind_mounts
	/usr/sbin/exportfs -r
	rc_status -v
	;;
    status)
        echo -n "Checking for kernel based NFS server: "
        ## Check status with checkproc(8), if process is running
        ## checkproc will return with exit status 0.

        # Status has a slightly different for the status command:
        # 0 - service running
        # 1 - service dead, but /var/run/  pid  file exists
        # 2 - service dead, but /var/lock/ lock file exists
        # 3 - service not running

        # NOTE: checkproc returns LSB compliant status values.
        checkproc /usr/sbin/rpc.mountd
	rc_status
        checkproc -n nfsd
        rc_status -v
        ;;
    *)
	echo "Usage: $0 {start|stop|status|try-restart|restart|force-reload|reload}"
	exit 1
	;;
esac

rc_exit
