Blame SOURCES/bz1091101-nfs-updates.patch

b9953b
From 3fc7c73f7caa212ef4de4b00e38b7bde2edb0cea Mon Sep 17 00:00:00 2001
b9953b
From: David Vossel <dvossel@redhat.com>
b9953b
Date: Thu, 10 Jul 2014 09:50:48 -0500
b9953b
Subject: [PATCH] nfs updates
b9953b
b9953b
---
b9953b
 doc/man/Makefile.am   |   1 +
b9953b
 heartbeat/Makefile.am |   1 +
b9953b
 heartbeat/exportfs    | 122 ++++++-------
b9953b
 heartbeat/nfsnotify   | 315 ++++++++++++++++++++++++++++++++
b9953b
 heartbeat/nfsserver   | 492 ++++++++++++++++++++++++++++++++++++++++----------
b9953b
 5 files changed, 764 insertions(+), 167 deletions(-)
b9953b
 create mode 100644 heartbeat/nfsnotify
b9953b
b9953b
diff --git a/doc/man/Makefile.am b/doc/man/Makefile.am
b9953b
index 344d00d..e97c7e9 100644
b9953b
--- a/doc/man/Makefile.am
b9953b
+++ b/doc/man/Makefile.am
b9953b
@@ -112,6 +112,7 @@ man_MANS	       = ocf_heartbeat_AoEtarget.7 \
b9953b
                           ocf_heartbeat_mysql.7 \
b9953b
                           ocf_heartbeat_mysql-proxy.7 \
b9953b
                           ocf_heartbeat_named.7 \
b9953b
+                          ocf_heartbeat_nfsnotify.7 \
b9953b
                           ocf_heartbeat_nfsserver.7 \
b9953b
                           ocf_heartbeat_nginx.7 \
b9953b
                           ocf_heartbeat_oracle.7 \
b9953b
diff --git a/heartbeat/Makefile.am b/heartbeat/Makefile.am
b9953b
index b67c98e..aab521f 100644
b9953b
--- a/heartbeat/Makefile.am
b9953b
+++ b/heartbeat/Makefile.am
b9953b
@@ -90,6 +90,7 @@ ocf_SCRIPTS	     =  ClusterMon		\
b9953b
 			mysql			\
b9953b
 			mysql-proxy		\
b9953b
 			named			\
b9953b
+			nfsnotify		\
b9953b
 			nfsserver		\
b9953b
 			oracle			\
b9953b
 			oralsnr			\
b9953b
diff --git a/heartbeat/exportfs b/heartbeat/exportfs
b9953b
index ff5d4f1..471da24 100755
b9953b
--- a/heartbeat/exportfs
b9953b
+++ b/heartbeat/exportfs
b9953b
@@ -95,6 +95,12 @@ Unique fsid within cluster.
b9953b
 Relinquish NFS locks associated with this filesystem when the resource
b9953b
 stops. Enabling this parameter is highly recommended unless the path exported
b9953b
 by this ${__SCRIPT_NAME} resource is also exported by a different resource.
b9953b
+
b9953b
+Note: Unlocking is only possible on Linux systems where
b9953b
+/proc/fs/nfsd/unlock_filesystem exists and is writable. If your system does
b9953b
+not fulfill this requirement (on account of having an nonrecent kernel,
b9953b
+for example), you may set this parameter to 0 to silence the associated
b9953b
+warning.
b9953b
 </longdesc>
b9953b
 <shortdesc lang="en">
b9953b
 Unlock filesystem on stop?
b9953b
@@ -141,7 +147,7 @@ Location of the rmtab backup, relative to directory.
b9953b
 
b9953b
 <actions>
b9953b
 <action name="start"   timeout="40" />
b9953b
-<action name="stop"    timeout="10" />
b9953b
+<action name="stop"    timeout="120" />
b9953b
 <action name="monitor" depth="0"  timeout="20" interval="10" />
b9953b
 <action name="meta-data"  timeout="5" />
b9953b
 <action name="validate-all"  timeout="30" />
b9953b
@@ -152,28 +158,41 @@ END
b9953b
 return $OCF_SUCCESS
b9953b
 }
b9953b
 
b9953b
+exportfs_methods() {
b9953b
+  cat <<-!
b9953b
+	start
b9953b
+	stop
b9953b
+	status
b9953b
+	monitor
b9953b
+	validate-all
b9953b
+	methods
b9953b
+	meta-data
b9953b
+	usage
b9953b
+	!
b9953b
+}
b9953b
+
b9953b
 backup_rmtab() {
b9953b
-    local rmtab_backup
b9953b
-    if [ ${OCF_RESKEY_rmtab_backup} != "none" ]; then
b9953b
-	rmtab_backup="${OCF_RESKEY_directory}/${OCF_RESKEY_rmtab_backup}"
b9953b
-	grep ":${OCF_RESKEY_directory}:" /var/lib/nfs/rmtab > ${rmtab_backup}
b9953b
-    fi
b9953b
+	local rmtab_backup
b9953b
+	if [ ${OCF_RESKEY_rmtab_backup} != "none" ]; then
b9953b
+		rmtab_backup="${OCF_RESKEY_directory}/${OCF_RESKEY_rmtab_backup}"
b9953b
+		grep ":${OCF_RESKEY_directory}:" /var/lib/nfs/rmtab > ${rmtab_backup}
b9953b
+	fi
b9953b
 }
b9953b
 
b9953b
 restore_rmtab() {
b9953b
-    local rmtab_backup
b9953b
-    if [ ${OCF_RESKEY_rmtab_backup} != "none" ]; then
b9953b
+	local rmtab_backup
b9953b
+	if [ ${OCF_RESKEY_rmtab_backup} != "none" ]; then
b9953b
 	rmtab_backup="${OCF_RESKEY_directory}/${OCF_RESKEY_rmtab_backup}"
b9953b
 	if [ -r ${rmtab_backup} ]; then
b9953b
-	    local tmpf=`mktemp`
b9953b
-	    sort -u ${rmtab_backup} /var/lib/nfs/rmtab > $tmpf &&
b9953b
+		local tmpf=`mktemp`
b9953b
+		sort -u ${rmtab_backup} /var/lib/nfs/rmtab > $tmpf &&
b9953b
 		install -o root -m 644 $tmpf /var/lib/nfs/rmtab
b9953b
-	    rm -f $tmpf
b9953b
-	    ocf_log debug "Restored `wc -l ${rmtab_backup}` rmtab entries from ${rmtab_backup}."
b9953b
+		rm -f $tmpf
b9953b
+		ocf_log debug "Restored `wc -l ${rmtab_backup}` rmtab entries from ${rmtab_backup}."
b9953b
 	else
b9953b
-	    ocf_log warn "rmtab backup ${rmtab_backup} not found or not readable."
b9953b
+		ocf_log warn "rmtab backup ${rmtab_backup} not found or not readable."
b9953b
+	fi
b9953b
 	fi
b9953b
-    fi
b9953b
 }
b9953b
 
b9953b
 exportfs_usage() {
b9953b
@@ -186,8 +205,8 @@ is_exported() {
b9953b
 	local dir=$1
b9953b
 	local spec=$2
b9953b
 	exportfs |
b9953b
-	        sed -e '$! N; s/\n[[:space:]]\+/ /; t; s/[[:space:]]\+\([^[:space:]]\+\)\(\n\|$\)/ \1\2/g; P;D;' |
b9953b
-			grep -q -x -F "$dir $spec"
b9953b
+		sed -e '$! N; s/\n[[:space:]]\+/ /; t; s/[[:space:]]\+\([^[:space:]]\+\)\(\n\|$\)/ \1\2/g; P;D;' |
b9953b
+		grep -q -x -F "$dir $spec"
b9953b
 }
b9953b
 
b9953b
 exportfs_monitor ()
b9953b
@@ -209,8 +228,10 @@ exportfs_monitor ()
b9953b
 #Adapt grep status code to OCF return code
b9953b
 	case $rc in
b9953b
 	0)
b9953b
-		ocf_log info "Directory ${OCF_RESKEY_directory} is exported to ${OCF_RESKEY_clientspec} (started)."
b9953b
-	        # Backup the rmtab to ensure smooth NFS-over-TCP failover
b9953b
+		if [ "$__OCF_ACTION" = "start" ]; then
b9953b
+			ocf_log info "Directory ${OCF_RESKEY_directory} is exported to ${OCF_RESKEY_clientspec} (started)."
b9953b
+		fi
b9953b
+		# Backup the rmtab to ensure smooth NFS-over-TCP failover
b9953b
 		backup_rmtab
b9953b
 		return $OCF_SUCCESS
b9953b
 		;;
b9953b
@@ -324,60 +345,23 @@ exportfs_stop ()
b9953b
 	fi
b9953b
 }
b9953b
 
b9953b
-exportfs_validate ()
b9953b
+exportfs_validate_all ()
b9953b
 {
b9953b
-	# Checks for required parameters
b9953b
-	if [ -z "$OCF_RESKEY_directory" ]; then
b9953b
-		ocf_log err "Missing required parameter \"directory\""
b9953b
-		exit $OCF_ERR_CONFIGURED
b9953b
-	fi
b9953b
-	if [ -z "$OCF_RESKEY_fsid" ]; then
b9953b
-		ocf_log err "Missing required parameter \"fsid\""
b9953b
-		exit $OCF_ERR_CONFIGURED
b9953b
-	fi
b9953b
-	if [ -z "$OCF_RESKEY_clientspec" ]; then
b9953b
-		ocf_log err "Missing required parameter \"clientspec\""
b9953b
-		exit $OCF_ERR_CONFIGURED
b9953b
-	fi
b9953b
-	
b9953b
-	# Checks applicable only to non-probes
b9953b
-	if ! ocf_is_probe; then
b9953b
-		if [ ! -d $OCF_RESKEY_directory ]; then
b9953b
-			ocf_log err "$OCF_RESKEY_directory does not exist or is not a directory"
b9953b
-			exit $OCF_ERR_INSTALLED
b9953b
-		fi
b9953b
+	if [ ! -d $OCF_RESKEY_directory ]; then
b9953b
+		ocf_log err "$OCF_RESKEY_directory does not exist or is not a directory"
b9953b
+		return $OCF_ERR_INSTALLED
b9953b
 	fi
b9953b
 }
b9953b
 
b9953b
-if [ $# -ne 1 ]; then
b9953b
-	exportfs_usage
b9953b
-	exit $OCF_ERR_ARGS
b9953b
+# If someone puts a trailing slash at the end of the export directory,
b9953b
+# this agent is going to fail in some unexpected ways due to how
b9953b
+# export strings are matched.  The simplest solution here is to strip off
b9953b
+# a trailing '/' in the directory before processing anything.
b9953b
+newdir=$(echo "$OCF_RESKEY_directory" | sed -n -e 's/^\(.*\)\/$/\1/p')
b9953b
+if [ -n "$newdir" ]; then
b9953b
+	OCF_RESKEY_directory=$newdir
b9953b
 fi
b9953b
 
b9953b
-case $__OCF_ACTION in
b9953b
-	meta-data)  exportfs_meta_data
b9953b
-		exit $OCF_SUCCESS
b9953b
-		;;
b9953b
-	usage|help) exportfs_usage
b9953b
-		exit $OCF_SUCCESS
b9953b
-		;;
b9953b
-	*)
b9953b
-		;;
b9953b
-esac
b9953b
-
b9953b
-exportfs_validate
b9953b
-
b9953b
-case $__OCF_ACTION in
b9953b
-	start)		exportfs_start
b9953b
-		;;
b9953b
-	stop)		exportfs_stop
b9953b
-		;;
b9953b
-	status|monitor)	exportfs_monitor
b9953b
-		;;
b9953b
-	validate-all)
b9953b
-		# nothing to do -- we're already validated
b9953b
-		;;
b9953b
-	*)		exportfs_usage
b9953b
-			exit $OCF_ERR_UNIMPLEMENTED
b9953b
-		;;
b9953b
-esac
b9953b
+OCF_REQUIRED_PARAMS="directory fsid clientspec"
b9953b
+OCF_REQUIRED_BINARIES="exportfs"
b9953b
+ocf_rarun $*
b9953b
diff --git a/heartbeat/nfsnotify b/heartbeat/nfsnotify
b9953b
new file mode 100644
b9953b
index 0000000..2e242de
b9953b
--- /dev/null
b9953b
+++ b/heartbeat/nfsnotify
b9953b
@@ -0,0 +1,315 @@
b9953b
+#!/bin/bash
b9953b
+#
b9953b
+# Copyright (c) 2014 David Vossel <dvossel@redhat.com>
b9953b
+#                    All Rights Reserved.
b9953b
+#
b9953b
+# This program is free software; you can redistribute it and/or modify
b9953b
+# it under the terms of version 2 of the GNU General Public License as
b9953b
+# published by the Free Software Foundation.
b9953b
+#
b9953b
+# This program is distributed in the hope that it would be useful, but
b9953b
+# WITHOUT ANY WARRANTY; without even the implied warranty of
b9953b
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
b9953b
+#
b9953b
+# Further, this software is distributed without any warranty that it is
b9953b
+# free of the rightful claim of any third person regarding infringement
b9953b
+# or the like.  Any license provided herein, whether implied or
b9953b
+# otherwise, applies only to this software file.  Patent licenses, if
b9953b
+# any, provided herein do not apply to combinations of this program with
b9953b
+# other software, or any other product whatsoever.
b9953b
+#
b9953b
+# You should have received a copy of the GNU General Public License
b9953b
+# along with this program; if not, write the Free Software Foundation,
b9953b
+# Inc., 59 Temple Place - Suite 330, Boston MA 02111-1307, USA.
b9953b
+#
b9953b
+
b9953b
+#######################################################################
b9953b
+# Initialization:
b9953b
+
b9953b
+: ${OCF_FUNCTIONS_DIR=${OCF_ROOT}/lib/heartbeat}
b9953b
+. ${OCF_FUNCTIONS_DIR}/ocf-shellfuncs
b9953b
+. ${OCF_FUNCTIONS_DIR}/ocf-directories
b9953b
+
b9953b
+#######################################################################
b9953b
+
b9953b
+sbindir=$HA_SBIN_DIR
b9953b
+if [ -z "$sbindir" ]; then
b9953b
+	sbindir=/usr/sbin
b9953b
+fi
b9953b
+
b9953b
+SELINUX_ENABLED=-1
b9953b
+
b9953b
+NFSNOTIFY_TMP_DIR="${HA_RSCTMP}/nfsnotify_${OCF_RESOURCE_INSTANCE}/"
b9953b
+HA_STATD_PIDFILE="$NFSNOTIFY_TMP_DIR/rpc.statd_${OCF_RESOURCE_INSTANCE}.pid"
b9953b
+HA_STATD_PIDFILE_PREV="$NFSNOTIFY_TMP_DIR/rpc.statd_${OCF_RESOURCE_INSTANCE}.pid.prev"
b9953b
+STATD_PATH="/var/lib/nfs/statd"
b9953b
+SM_NOTIFY_BINARY="${sbindir}/sm-notify"
b9953b
+IS_RENOTIFY=0
b9953b
+
b9953b
+meta_data() {
b9953b
+	cat <
b9953b
+
b9953b
+
b9953b
+<resource-agent name="nfsnotify" version="0.9">
b9953b
+<version>1.0</version>
b9953b
+
b9953b
+<longdesc lang="en">
b9953b
+This agent sends NFSv3 reboot notifications to clients which informs clients to reclaim locks.
b9953b
+</longdesc>
b9953b
+<shortdesc lang="en">sm-notify reboot notifications</shortdesc>
b9953b
+
b9953b
+<parameters>
b9953b
+
b9953b
+<parameter name="source_host" unique="0" required="0">
b9953b
+<longdesc lang="en">
b9953b
+Comma separated list of floating IP addresses or host names that clients use
b9953b
+to access the nfs service.  This will be used to set the source address and
b9953b
+mon_name of the SN_NOTIFY reboot notifications.
b9953b
+</longdesc>
b9953b
+<shortdesc lang="en">source IP addresses</shortdesc>
b9953b
+<content type="string" default="" />
b9953b
+</parameter>
b9953b
+
b9953b
+<parameter name="notify_args" unique="0" required="0">
b9953b
+<longdesc lang="en">
b9953b
+Additional arguments to send to the sm-notify command. By default
b9953b
+this agent will always set sm-notify's '-f' option.  When the
b9953b
+source_host option is set, the '-v' option will be used automatically
b9953b
+to set the proper source address. Any additional sm-notify arguments
b9953b
+set with this option will be used in addition to the previous default
b9953b
+arguments.
b9953b
+</longdesc>
b9953b
+<shortdesc lang="en">sm-notify arguments</shortdesc>
b9953b
+<content type="string" default="false" />
b9953b
+</parameter>
b9953b
+
b9953b
+</parameters>
b9953b
+
b9953b
+<actions>
b9953b
+<action name="start"        timeout="90" />
b9953b
+<action name="stop"         timeout="90" />
b9953b
+<action name="monitor"      timeout="90" interval="30" depth="0" />
b9953b
+<action name="reload"       timeout="90" />
b9953b
+<action name="meta-data"    timeout="10" />
b9953b
+<action name="validate-all"   timeout="20" />
b9953b
+</actions>
b9953b
+</resource-agent>
b9953b
+END
b9953b
+}
b9953b
+
b9953b
+v3notify_usage()
b9953b
+{
b9953b
+	cat <
b9953b
+usage: $0 {start|stop|monitor|validate-all|meta-data}
b9953b
+
b9953b
+Expects to have a fully populated OCF RA-compliant environment set.
b9953b
+END
b9953b
+}
b9953b
+
b9953b
+v3notify_validate()
b9953b
+{
b9953b
+	# check_binary will exit with OCF_ERR_INSTALLED when binary is missing
b9953b
+	check_binary "$SM_NOTIFY_BINARY"
b9953b
+	check_binary "pgrep"
b9953b
+	check_binary "killall"
b9953b
+
b9953b
+	return $OCF_SUCCESS
b9953b
+}
b9953b
+
b9953b
+killall_smnotify()
b9953b
+{
b9953b
+	# killall sm-notify 
b9953b
+	killall -TERM $SM_NOTIFY_BINARY > /dev/null 2>&1
b9953b
+	if [ $? -eq 0 ]; then
b9953b
+		# it is useful to know if sm-notify processes were actually left around
b9953b
+		# or not during the stop/start operation. Whether this condition is true
b9953b
+		# or false does not indicate a failure. It does indicate that 
b9953b
+		# there are probably some unresponsive nfs clients out there that are keeping
b9953b
+		# the sm-notify processes retrying.
b9953b
+		ocf_log info "previous sm-notify processes terminated before $__OCF_ACTION action."
b9953b
+	fi
b9953b
+}
b9953b
+
b9953b
+v3notify_stop()
b9953b
+{
b9953b
+	killall_smnotify
b9953b
+
b9953b
+	rm -f $HA_STATD_PIDFILE_PREV > /dev/null 2>&1
b9953b
+	mv $HA_STATD_PIDFILE $HA_STATD_PIDFILE_PREV > /dev/null 2>&1
b9953b
+
b9953b
+	return $OCF_SUCCESS
b9953b
+}
b9953b
+
b9953b
+check_statd_pidfile()
b9953b
+{
b9953b
+	local binary="rpc.statd"
b9953b
+	local pidfile="$HA_STATD_PIDFILE"
b9953b
+
b9953b
+	ocf_log debug "Checking status for ${binary}."
b9953b
+	if [ -e "$pidfile" ]; then
b9953b
+		cat /proc/$(cat $pidfile)/cmdline 2>/dev/null | grep -a "${binary}" > /dev/null 2>&1
b9953b
+		if [ $? -eq 0 ]; then
b9953b
+			return $OCF_SUCCESS
b9953b
+		fi
b9953b
+
b9953b
+		ocf_log err "$(cat $pidfile) for $binary is no longer running, sm-notify needs to re-notify clients"
b9953b
+		return $OCF_ERR_GENERIC
b9953b
+	fi
b9953b
+
b9953b
+	# if we don't have a pid file for rpc.statd, we have not yet sent the notifications
b9953b
+	return $OCF_NOT_RUNNING
b9953b
+}
b9953b
+
b9953b
+write_statd_pid()
b9953b
+{
b9953b
+	local binary="rpc.statd"
b9953b
+	local pidfile="$HA_STATD_PIDFILE"
b9953b
+	local pid
b9953b
+
b9953b
+	pid=$(pgrep ${binary})
b9953b
+	case $? in
b9953b
+		0)
b9953b
+			ocf_log info "PID file (pid:${pid} at $pidfile) created for ${binary}."
b9953b
+			mkdir -p $(dirname $pidfile)
b9953b
+			echo "$pid" > $pidfile
b9953b
+			return $OCF_SUCCESS;;
b9953b
+		1)
b9953b
+			rm -f "$pidfile" > /dev/null 2>&1 
b9953b
+			ocf_log info "$binary is not running"
b9953b
+			return $OCF_NOT_RUNNING;;
b9953b
+		*)
b9953b
+			rm -f "$pidfile" > /dev/null 2>&1 
b9953b
+			ocf_log err "Error encountered detecting pid status of $binary"
b9953b
+			return $OCF_ERR_GENERIC;;
b9953b
+	esac
b9953b
+}
b9953b
+
b9953b
+copy_statd()
b9953b
+{
b9953b
+	local src=$1
b9953b
+	local dest=$2
b9953b
+
b9953b
+	if ! [ -d "$dest" ]; then
b9953b
+		mkdir -p "$dest"
b9953b
+	fi
b9953b
+
b9953b
+	cp -rpfn $src/sm $src/sm.bak $src/state $dest > /dev/null 2>&1
b9953b
+
b9953b
+	# make sure folder ownership and selinux lables stay consistent
b9953b
+	[ -n "`id -u rpcuser`" -a "`id -g rpcuser`" ] && chown rpcuser.rpcuser "$dest"
b9953b
+	[ $SELINUX_ENABLED -eq 0 ] && chcon -R "$SELINUX_LABEL" "$dest"
b9953b
+}
b9953b
+
b9953b
+v3notify_start()
b9953b
+{
b9953b
+	local rc=$OCF_SUCCESS
b9953b
+	local cur_statd
b9953b
+	local statd_backup
b9953b
+	local is_renotify=0
b9953b
+
b9953b
+	# monitor, see if we need to notify or not
b9953b
+	v3notify_monitor
b9953b
+	if [ $? -eq 0 ]; then
b9953b
+		return $OCF_SUCCESS
b9953b
+	fi
b9953b
+
b9953b
+	# kill off any other sm-notify processes that might already be running.
b9953b
+	killall_smnotify
b9953b
+
b9953b
+	# record the pid of rpc.statd. if this pid ever changes, we have to re-notify
b9953b
+	write_statd_pid
b9953b
+	rc=$?
b9953b
+	if [ $rc -ne 0 ]; then
b9953b
+		return $rc
b9953b
+	fi
b9953b
+
b9953b
+	# if the last time we ran nfs-notify, it was with the same statd process,
b9953b
+	# consider this a re-notification. During re-notifications we do not let the
b9953b
+	# sm-notify binary have access to the real statd directory.
b9953b
+	if [ "$(cat $HA_STATD_PIDFILE)" = "$(cat $HA_STATD_PIDFILE_PREV 2>/dev/null)" ]; then
b9953b
+		ocf_log info "Renotifying clients"
b9953b
+		is_renotify=1
b9953b
+	fi
b9953b
+
b9953b
+	statd_backup="$STATD_PATH/nfsnotify.bu"
b9953b
+	copy_statd "$STATD_PATH" "$statd_backup"
b9953b
+
b9953b
+	if [ -z "$OCF_RESKEY_source_host" ]; then
b9953b
+		if [ "$is_renotify" -eq 0 ]; then
b9953b
+			cur_statd="$STATD_PATH"
b9953b
+		else 
b9953b
+			cur_statd="$statd_backup"
b9953b
+		fi
b9953b
+		ocf_log info "sending notifications on default source address."
b9953b
+		$SM_NOTIFY_BINARY -f $OCF_RESKEY_notify_args -P $cur_statd
b9953b
+		if [ $? -ne 0 ]; then
b9953b
+			ocf_log err "sm-notify failed, view syslog for more information."
b9953b
+			return $OCF_ERR_GENERIC
b9953b
+		fi
b9953b
+		
b9953b
+		return $OCF_SUCCESS
b9953b
+	fi
b9953b
+
b9953b
+	# do sm-notify for each ip
b9953b
+	for ip in `echo ${OCF_RESKEY_source_host} | sed 's/,/ /g'`; do
b9953b
+
b9953b
+		# have the first sm-notify use the actual statd directory so the
b9953b
+		# notify list can be managed properly.
b9953b
+		if [ "$is_renotify" -eq 0 ]; then
b9953b
+			cur_statd="$STATD_PATH"
b9953b
+			# everything after the first notify we are considering a renotification
b9953b
+			# which means we don't use the real statd directory. 
b9953b
+			is_renotify=1
b9953b
+		else 
b9953b
+			# use our copied statd directory for the remaining ip addresses
b9953b
+			cur_statd="$STATD_PATH/nfsnotify_${OCF_RESOURCE_INSTANCE}_${ip}"
b9953b
+			copy_statd "$statd_backup" "$cur_statd"
b9953b
+		fi
b9953b
+
b9953b
+		ocf_log info "sending notifications with source address $ip"
b9953b
+		$SM_NOTIFY_BINARY -f $OCF_RESKEY_notify_args -v $ip -P "$cur_statd"
b9953b
+		if [ $? -ne 0 ]; then
b9953b
+			ocf_log err "sm-notify with source host set to, $source_host, failed. view syslog for more information"
b9953b
+			return $OCF_ERR_GENERIC
b9953b
+		fi
b9953b
+	done
b9953b
+
b9953b
+	return $OCF_SUCCESS
b9953b
+}
b9953b
+
b9953b
+v3notify_monitor()
b9953b
+{
b9953b
+	# verify rpc.statd is up, and that the rpc.statd pid is the same one we
b9953b
+	# found during the start. otherwise rpc.statd recovered and we need to notify
b9953b
+	# again.
b9953b
+	check_statd_pidfile
b9953b
+}
b9953b
+
b9953b
+case $__OCF_ACTION in
b9953b
+	meta-data)   meta_data
b9953b
+		exit $OCF_SUCCESS;;
b9953b
+	usage|help)    v3notify_usage
b9953b
+		exit $OCF_SUCCESS;;
b9953b
+	*)
b9953b
+		;;
b9953b
+esac
b9953b
+
b9953b
+which restorecon > /dev/null 2>&1 && selinuxenabled
b9953b
+SELINUX_ENABLED=$?
b9953b
+if [ $SELINUX_ENABLED -eq 0 ]; then
b9953b
+	export SELINUX_LABEL="$(ls -ldZ $STATD_PATH | cut -f4 -d' ')"
b9953b
+fi
b9953b
+
b9953b
+case $__OCF_ACTION in
b9953b
+	start)         v3notify_start;;
b9953b
+	stop)          v3notify_stop;;
b9953b
+	monitor)       v3notify_monitor;;
b9953b
+	validate-all)  v3notify_validate;;
b9953b
+	*)             v3notify_usage
b9953b
+	               exit $OCF_ERR_UNIMPLEMENTED;;
b9953b
+esac
b9953b
+
b9953b
+rc=$?
b9953b
+ocf_log debug "${OCF_RESOURCE_INSTANCE} $__OCF_ACTION : $rc"
b9953b
+exit $rc
b9953b
+
b9953b
diff --git a/heartbeat/nfsserver b/heartbeat/nfsserver
b9953b
index bc326e5..e44da1c 100755
b9953b
--- a/heartbeat/nfsserver
b9953b
+++ b/heartbeat/nfsserver
b9953b
@@ -13,13 +13,22 @@ else
b9953b
 fi
b9953b
 
b9953b
 DEFAULT_INIT_SCRIPT="/etc/init.d/nfsserver"
b9953b
-DEFAULT_NOTIFY_CMD="/sbin/sm-notify"
b9953b
+if ! [ -f $DEFAULT_INIT_SCRIPT ]; then
b9953b
+	# On some systems, the script is just called nfs
b9953b
+	DEFAULT_INIT_SCRIPT="/etc/init.d/nfs"
b9953b
+fi
b9953b
+
b9953b
+DEFAULT_NOTIFY_CMD=`which sm-notify`
b9953b
+DEFAULT_NOTIFY_CMD=${DEFAULT_NOTIFY_CMD:-"/sbin/sm-notify"}
b9953b
 DEFAULT_NOTIFY_FOREGROUND="false"
b9953b
 DEFAULT_RPCPIPEFS_DIR="/var/lib/nfs/rpc_pipefs"
b9953b
 EXEC_MODE=0
b9953b
 SELINUX_ENABLED=-1
b9953b
 STATD_PATH="/var/lib/nfs"
b9953b
 STATD_DIR=""
b9953b
+NFS_SYSCONFIG="/etc/sysconfig/nfs"
b9953b
+NFS_SYSCONFIG_LOCAL_BACKUP="/etc/sysconfig/nfs.ha.bu"
b9953b
+NFS_SYSCONFIG_AUTOGEN_TAG="AUTOGENERATED by $0 high availability resource-agent"
b9953b
 
b9953b
 nfsserver_meta_data() {
b9953b
 	cat <
b9953b
@@ -53,21 +62,19 @@ Init script for nfsserver
b9953b
 <content type="string" default="auto detected" />
b9953b
 </parameter>
b9953b
 
b9953b
-<parameter name="nfs_notify_cmd" unique="0" required="0">
b9953b
+<parameter name="nfs_no_notify" unique="0" required="0">
b9953b
 <longdesc lang="en">
b9953b
-The tool to send out NSM reboot notification; it should be either sm-notify or rpc.statd.
b9953b
-Failover of nfsserver can be considered as rebooting to different machines.
b9953b
-The nfsserver resource agent use this command to notify all clients about the occurrence of failover.
b9953b
+Do not send reboot notifications to NFSv3 clients during server startup.
b9953b
 </longdesc>
b9953b
 <shortdesc lang="en">
b9953b
-The tool to send out notification.
b9953b
+Disable NFSv3 server reboot notifications
b9953b
 </shortdesc>
b9953b
-<content type="string" default="$DEFAULT_NOTIFY_CMD" />
b9953b
+<content type="boolean" default="false" />
b9953b
 </parameter>
b9953b
 
b9953b
 <parameter name="nfs_notify_foreground" unique="0" required="0">
b9953b
 <longdesc lang="en">
b9953b
-Keeps the notify tool attached to its controlling terminal and running in the foreground.
b9953b
+Keeps the sm-notify attached to its controlling terminal and running in the foreground.
b9953b
 </longdesc>
b9953b
 <shortdesc lang="en">
b9953b
 Keeps the notify tool running in the foreground.
b9953b
@@ -87,25 +94,102 @@ Specifies the length of sm-notify retry time (minutes).
b9953b
 <content type="integer" default="" />
b9953b
 </parameter>
b9953b
 
b9953b
-<parameter name="nfs_shared_infodir" unique="0" required="1">
b9953b
+<parameter name="nfs_ip" unique="0" required="0">
b9953b
 <longdesc lang="en">
b9953b
-The nfsserver resource agent will save nfs related information in this specific directory.
b9953b
-And this directory must be able to fail-over before nfsserver itself.
b9953b
+Comma separated list of floating IP addresses used to access the nfs service
b9953b
 </longdesc>
b9953b
 <shortdesc lang="en">
b9953b
-Directory to store nfs server related information.
b9953b
+IP addresses.
b9953b
 </shortdesc>
b9953b
-<content type="string" default="" />
b9953b
+<content type="string"/>
b9953b
 </parameter>
b9953b
 
b9953b
-<parameter name="nfs_ip" unique="0" required="1">
b9953b
+<parameter name="nfsd_args" unique="0" required="0">
b9953b
 <longdesc lang="en">
b9953b
-Comma separated list of floating IP addresses used to access the nfs service
b9953b
+Specifies what arguments to pass to the nfs daemon on startup. View the rpc.nfsd man page for information on what arguments are available.
b9953b
+Note that setting this value will override all settings placed in the local /etc/sysconfig/nfs file.
b9953b
 </longdesc>
b9953b
 <shortdesc lang="en">
b9953b
-IP addresses.
b9953b
+rpc.nfsd options
b9953b
 </shortdesc>
b9953b
-<content type="string"/>
b9953b
+<content type="string" />
b9953b
+</parameter>
b9953b
+
b9953b
+<parameter name="lockd_udp_port" unique="0" required="0">
b9953b
+<longdesc lang="en">
b9953b
+The udp port lockd should listen on.
b9953b
+Note that setting this value will override all settings placed in the local /etc/sysconfig/nfs file.
b9953b
+</longdesc>
b9953b
+<shortdesc lang="en">
b9953b
+lockd udp port
b9953b
+</shortdesc>
b9953b
+<content type="integer" />
b9953b
+</parameter>
b9953b
+
b9953b
+<parameter name="lockd_tcp_port" unique="0" required="0">
b9953b
+<longdesc lang="en">
b9953b
+The tcp port lockd should listen on.
b9953b
+Note that setting this value will override all settings placed in the local /etc/sysconfig/nfs file.
b9953b
+</longdesc>
b9953b
+<shortdesc lang="en">
b9953b
+lockd tcp port
b9953b
+</shortdesc>
b9953b
+<content type="integer" />
b9953b
+</parameter>
b9953b
+
b9953b
+<parameter name="statd_outgoing_port" unique="0" required="0">
b9953b
+<longdesc lang="en">
b9953b
+The source port number sm-notify uses when sending reboot notifications.
b9953b
+Note that setting this value will override all settings placed in the local /etc/sysconfig/nfs file.
b9953b
+</longdesc>
b9953b
+<shortdesc lang="en">
b9953b
+sm-notify source port
b9953b
+</shortdesc>
b9953b
+<content type="integer" />
b9953b
+</parameter>
b9953b
+
b9953b
+<parameter name="statd_port" unique="0" required="0">
b9953b
+<longdesc lang="en">
b9953b
+The port number used for RPC listener sockets.
b9953b
+Note that setting this value will override all settings placed in the local /etc/sysconfig/nfs file.
b9953b
+</longdesc>
b9953b
+<shortdesc lang="en">
b9953b
+rpc.statd listener port
b9953b
+</shortdesc>
b9953b
+<content type="integer" />
b9953b
+</parameter>
b9953b
+
b9953b
+<parameter name="mountd_port" unique="0" required="0">
b9953b
+<longdesc lang="en">
b9953b
+The port number used for rpc.mountd listener sockets.
b9953b
+Note that setting this value will override all settings placed in the local /etc/sysconfig/nfs file.
b9953b
+</longdesc>
b9953b
+<shortdesc lang="en">
b9953b
+rpc.mountd listener port
b9953b
+</shortdesc>
b9953b
+<content type="integer" />
b9953b
+</parameter>
b9953b
+
b9953b
+<parameter name="rquotad_port" unique="0" required="0">
b9953b
+<longdesc lang="en">
b9953b
+The port number used for rpc.rquotad.
b9953b
+Note that setting this value will override all settings placed in the local /etc/sysconfig/nfs file.
b9953b
+</longdesc>
b9953b
+<shortdesc lang="en">
b9953b
+rpc.rquotad port
b9953b
+</shortdesc>
b9953b
+<content type="integer" />
b9953b
+</parameter>
b9953b
+
b9953b
+<parameter name="nfs_shared_infodir" unique="0" required="0">
b9953b
+<longdesc lang="en">
b9953b
+The nfsserver resource agent will save nfs related information in this specific directory.
b9953b
+And this directory must be able to fail-over before nfsserver itself.
b9953b
+</longdesc>
b9953b
+<shortdesc lang="en">
b9953b
+Directory to store nfs server related information.
b9953b
+</shortdesc>
b9953b
+<content type="string" default="" />
b9953b
 </parameter>
b9953b
 
b9953b
 <parameter name="rpcpipefs_dir" unique="0" required="0">
b9953b
@@ -228,6 +312,7 @@ set_exec_mode()
b9953b
 	if which systemctl > /dev/null 2>&1; then
b9953b
 		if systemctl list-unit-files | grep nfs-server > /dev/null && systemctl list-unit-files | grep nfs-lock > /dev/null; then
b9953b
 			EXEC_MODE=2
b9953b
+			# when using systemd, the nfs-lock service file handles nfsv3 locking daemons for us.
b9953b
 			return 0
b9953b
 		fi
b9953b
 	fi
b9953b
@@ -236,33 +321,6 @@ set_exec_mode()
b9953b
 	exit $OCF_ERR_INSTALLED
b9953b
 }
b9953b
 
b9953b
-nfs_systemd_exec()
b9953b
-{
b9953b
-	local cmd=$1
b9953b
-	local server_res
b9953b
-	local lock_res
b9953b
-
b9953b
-	if [ "$cmd" = "stop" ]; then
b9953b
-		systemctl $cmd nfs-server.service
b9953b
-		server_res=$?
b9953b
-		systemctl $cmd nfs-lock.service
b9953b
-		lock_res=$?
b9953b
-	else
b9953b
-		systemctl $cmd nfs-lock.service
b9953b
-		lock_res=$?
b9953b
-		systemctl $cmd nfs-server.service
b9953b
-		server_res=$?
b9953b
-	fi
b9953b
-
b9953b
-	if [ $lock_res -ne $server_res ]; then
b9953b
-		# If one is running and the other isn't, or for whatever other reason
b9953b
-		# the return code's aren't the same, this is bad.
b9953b
-		ocf_log err "Systemd services nfs-lock and nfs-server are not in the same state after attempting $cmd command"
b9953b
-		return $OCF_ERR_GENERIC
b9953b
-	fi
b9953b
-	return $server_res
b9953b
-}
b9953b
-
b9953b
 ##
b9953b
 # wrapper for init script and systemd calls.
b9953b
 ##
b9953b
@@ -273,21 +331,45 @@ nfs_exec()
b9953b
 
b9953b
 	case $EXEC_MODE in 
b9953b
 		1) ${OCF_RESKEY_nfs_init_script} $cmd;;
b9953b
-		2) nfs_systemd_exec $cmd;;
b9953b
+		2) systemctl $cmd nfs-server.service ;;
b9953b
 	esac
b9953b
 }
b9953b
 
b9953b
+v3locking_exec()
b9953b
+{
b9953b
+	local cmd=$1
b9953b
+	set_exec_mode
b9953b
+
b9953b
+	if [ $EXEC_MODE -eq 2 ]; then
b9953b
+		systemctl $cmd nfs-lock.service
b9953b
+	else 
b9953b
+		case $cmd in
b9953b
+			start) locking_start;;
b9953b
+			stop) locking_stop;;
b9953b
+			status) locking_status;;
b9953b
+		esac
b9953b
+	fi
b9953b
+}
b9953b
+
b9953b
 nfsserver_monitor ()
b9953b
 {
b9953b
 	fn=`mktemp`
b9953b
 	nfs_exec status > $fn 2>&1 
b9953b
 	rc=$?
b9953b
-	ocf_log debug `cat $fn`
b9953b
+	ocf_log debug "$(cat $fn)"
b9953b
 	rm -f $fn
b9953b
 
b9953b
-#Adapte LSB status code to OCF return code
b9953b
+	#Adapte LSB status code to OCF return code
b9953b
 	if [ $rc -eq 0 ]; then
b9953b
-		return $OCF_SUCCESS
b9953b
+		# don't report success if nfs servers are up
b9953b
+		# without locking daemons.
b9953b
+		v3locking_exec "status"
b9953b
+		rc=$?
b9953b
+		if [ $rc -ne 0 ]; then
b9953b
+			ocf_log error "NFS server is up, but the locking daemons are down"
b9953b
+			rc=$OCF_ERR_GENERIC
b9953b
+		fi
b9953b
+		return $rc
b9953b
 	elif [ $rc -eq 3 ]; then
b9953b
 		return $OCF_NOT_RUNNING
b9953b
 	else
b9953b
@@ -295,8 +377,79 @@ nfsserver_monitor ()
b9953b
 	fi
b9953b
 }
b9953b
 
b9953b
+set_arg()
b9953b
+{
b9953b
+	local key="$1"
b9953b
+	local value="$2"
b9953b
+	local file="$3"
b9953b
+	local requires_sysconfig="$4"
b9953b
+
b9953b
+	if [ -z "$value" ]; then
b9953b
+		return
b9953b
+	fi
b9953b
+
b9953b
+	# only write to the tmp /etc/sysconfig/nfs if sysconfig exists.
b9953b
+	# otherwise this distro does not support setting these options.
b9953b
+	if [ -d "/etc/sysconfig" ]; then
b9953b
+		echo "${key}=\"${value}\"" >> $file
b9953b
+	elif [ "$requires_sysconfig" = "true" ]; then
b9953b
+		ocf_log warn "/etc/sysconfig/nfs not found, unable to set port and nfsd args."
b9953b
+	fi
b9953b
+
b9953b
+	export ${key}="${value}"
b9953b
+}
b9953b
+
b9953b
+set_env_args()
b9953b
+{
b9953b
+	local tmpconfig=$(mktemp ${HA_RSCTMP}/nfsserver-tmp-XXXXX)
b9953b
+	local statd_args
b9953b
+
b9953b
+	# nfsd args
b9953b
+	set_arg "RPCNFSDARGS" "$OCF_RESKEY_nfsd_args" "$tmpconfig" "true"
b9953b
+
b9953b
+	# mountd args
b9953b
+	if [ -n "$OCF_RESKEY_mountd_port" ]; then
b9953b
+		set_arg "RPCMOUNTDOPTS" "-p $OCF_RESKEY_mountd_port" "$tmpconfig" "true"
b9953b
+	fi
b9953b
+
b9953b
+	# statd args. we always want to perform the notify using sm-notify after
b9953b
+	# both rpc.statd and the nfsd daemons are initialized
b9953b
+	statd_args="--no-notify"
b9953b
+	if [ -n "$OCF_RESKEY_statd_outgoing_port" ]; then
b9953b
+		statd_args="$statd_args -o $OCF_RESKEY_statd_outgoing_port"
b9953b
+	fi
b9953b
+	if [ -n "$OCF_RESKEY_statd_port" ]; then
b9953b
+		statd_args="$statd_args -p $OCF_RESKEY_statd_port"
b9953b
+	fi
b9953b
+	set_arg "STATDARG" "$statd_args" "$tmpconfig" "false"
b9953b
+
b9953b
+	# lockd ports
b9953b
+	set_arg "LOCKD_UDPPORT" "$OCF_RESKEY_lockd_udp_port" "$tmpconfig" "true"
b9953b
+	set_arg "LOCKD_TCPPORT" "$OCF_RESKEY_lockd_tcp_port" "$tmpconfig" "true"
b9953b
+
b9953b
+	# rquotad_port
b9953b
+	set_arg "RPCRQUOTADOPTS" "-p $OCF_RESKEY_rquotad_port" "$tmpconfig" "true"
b9953b
+
b9953b
+	# override local nfs config. preserve previous local config though.
b9953b
+	if [ -s $tmpconfig ]; then
b9953b
+		cat $NFS_SYSCONFIG | grep -e "$NFS_SYSCONFIG_AUTOGEN_TAG"
b9953b
+		if [ $? -ne 0 ]; then
b9953b
+			# backup local nfs config if it doesn't have our HA autogen tag in it.
b9953b
+			mv -f $NFS_SYSCONFIG $NFS_SYSCONFIG_LOCAL_BACKUP
b9953b
+		fi
b9953b
+		echo "# $NFS_SYSCONFIG_AUTOGEN_TAG" > $NFS_SYSCONFIG
b9953b
+		echo "# local config backup stored here, '$NFS_SYSCONFIG_LOCAL_BACKUP'" >> $NFS_SYSCONFIG
b9953b
+		cat $tmpconfig >> $NFS_SYSCONFIG
b9953b
+	fi
b9953b
+	rm -f $tmpconfig
b9953b
+}
b9953b
+
b9953b
 prepare_directory ()
b9953b
 {
b9953b
+	if [ -z "$fp" ]; then
b9953b
+		return
b9953b
+	fi
b9953b
+
b9953b
 	[ -d "$fp" ] || mkdir -p $fp
b9953b
 	[ -d "$rpcpipefs_make_dir" ] || mkdir -p $rpcpipefs_make_dir
b9953b
 	[ -d "$fp/v4recovery" ] || mkdir -p $fp/v4recovery
b9953b
@@ -311,6 +464,8 @@ prepare_directory ()
b9953b
 	[ -f "$fp/xtab" ] || touch "$fp/xtab"
b9953b
 	[ -f "$fp/rmtab" ] || touch "$fp/rmtab"
b9953b
 
b9953b
+	dd if=/dev/urandom of=$fp/$STATD_DIR/state bs=1 count=4 &> /dev/null
b9953b
+	[ -n "`id -u rpcuser`" -a "`id -g rpcuser`" ] && chown rpcuser.rpcuser "$fp/$STATD_DIR/state"
b9953b
 	[ $SELINUX_ENABLED -eq 0 ] && chcon -R "$SELINUX_LABEL" "$fp"
b9953b
 }
b9953b
 
b9953b
@@ -325,6 +480,10 @@ is_bound ()
b9953b
 
b9953b
 bind_tree ()
b9953b
 {
b9953b
+	if [ -z "$fp" ]; then
b9953b
+		return
b9953b
+	fi
b9953b
+
b9953b
 	if is_bound /var/lib/nfs; then
b9953b
 		ocf_log debug "$fp is already bound to /var/lib/nfs"
b9953b
 		return 0
b9953b
@@ -343,25 +502,195 @@ unbind_tree ()
b9953b
 	fi
b9953b
 }
b9953b
 
b9953b
+binary_status()
b9953b
+{
b9953b
+	local binary=$1
b9953b
+	local pid
b9953b
+
b9953b
+	pid=$(pgrep ${binary})
b9953b
+	case $? in
b9953b
+		0)
b9953b
+			echo "$pid"
b9953b
+			return $OCF_SUCCESS;;
b9953b
+		1)
b9953b
+			return $OCF_NOT_RUNNING;;
b9953b
+		*)
b9953b
+			return $OCF_ERR_GENERIC;;
b9953b
+	esac
b9953b
+}
b9953b
+
b9953b
+locking_status()
b9953b
+{
b9953b
+	binary_status "rpc.statd" > /dev/null 2>&1
b9953b
+}
b9953b
+
b9953b
+locking_start()
b9953b
+{
b9953b
+	local ret=$OCF_SUCCESS
b9953b
+
b9953b
+	ocf_log info "Starting rpc.statd."
b9953b
+
b9953b
+	rpc.statd $STATDARG
b9953b
+
b9953b
+	ret=$?
b9953b
+	if [ $ret -ne 0 ]; then
b9953b
+		ocf_log err "Failed to start rpc.statd"
b9953b
+		return $ret
b9953b
+	fi
b9953b
+	touch /var/lock/subsys/nfslock
b9953b
+
b9953b
+	return $ret
b9953b
+}
b9953b
+
b9953b
+terminate()
b9953b
+{
b9953b
+	declare pids
b9953b
+	declare i=0
b9953b
+
b9953b
+	while : ; do
b9953b
+		pids=$(binary_status $1)
b9953b
+		[ -z "$pids" ] && return 0
b9953b
+	 	kill $pids
b9953b
+		sleep 1
b9953b
+		((i++))
b9953b
+		[ $i -gt 3 ] && return 1
b9953b
+	done
b9953b
+}
b9953b
+
b9953b
+
b9953b
+killkill()
b9953b
+{
b9953b
+	declare pids
b9953b
+	declare i=0
b9953b
+
b9953b
+	while : ; do
b9953b
+		pids=$(binary_status $1)
b9953b
+		[ -z "$pids" ] && return 0
b9953b
+	 	kill -9 $pids
b9953b
+		sleep 1
b9953b
+		((i++))
b9953b
+		[ $i -gt 3 ] && return 1
b9953b
+	done
b9953b
+}
b9953b
+
b9953b
+stop_process()
b9953b
+{
b9953b
+	declare process=$1
b9953b
+
b9953b
+	ocf_log info "Stopping $process"
b9953b
+	if terminate $process; then
b9953b
+		ocf_log debug "$process is stopped"
b9953b
+	else
b9953b
+		if killkill $process; then
b9953b
+			ocf_log debug "$process is stopped"
b9953b
+		else
b9953b
+			ocf_log debug "Failed to stop $process"
b9953b
+			return 1
b9953b
+		fi
b9953b
+	fi
b9953b
+	return 0
b9953b
+}
b9953b
+
b9953b
+locking_stop()
b9953b
+{
b9953b
+	ret=0
b9953b
+
b9953b
+	# sm-notify can prevent umount of /var/lib/nfs/statd if
b9953b
+	# it is still trying to notify unresponsive clients.
b9953b
+	stop_process sm-notify
b9953b
+	if [ $? -ne 0 ]; then
b9953b
+		ret=$OCF_ERR_GENERIC
b9953b
+	fi
b9953b
+
b9953b
+	stop_process rpc.statd
b9953b
+	if [ $? -ne 0 ]; then
b9953b
+		ret=$OCF_ERR_GENERIC
b9953b
+	fi
b9953b
+
b9953b
+	return $ret
b9953b
+}
b9953b
+
b9953b
+notify_locks()
b9953b
+{
b9953b
+	if ocf_is_true "$OCF_RESKEY_nfs_no_notify"; then
b9953b
+		# we've been asked not to notify clients
b9953b
+		return;
b9953b
+	fi
b9953b
+
b9953b
+	# run in foreground, if requested
b9953b
+	if ocf_is_true "$OCF_RESKEY_nfs_notify_foreground"; then
b9953b
+		opts="-d"
b9953b
+	fi
b9953b
+
b9953b
+	if [ -n "$OCF_RESKEY_nfs_smnotify_retry_time" ]; then
b9953b
+		opts="$opts -m $OCF_RESKEY_nfs_smnotify_retry_time"
b9953b
+	fi
b9953b
+
b9953b
+	if [ -n "$OCF_RESKEY_statd_outgoing_port" ]; then
b9953b
+		opts="$opts -p $OCF_RESKEY_statd_outgoing_port"
b9953b
+	fi
b9953b
+
b9953b
+	# forces re-notificaiton regardless if notifies have already gone out
b9953b
+	opts="$opts -f"
b9953b
+
b9953b
+	ocf_log info "executing sm-notify"
b9953b
+	if [ -n "$OCF_RESKEY_nfs_ip" ]; then
b9953b
+		for ip in `echo ${OCF_RESKEY_nfs_ip} | sed 's/,/ /g'`; do
b9953b
+			cp -rpfn $STATD_PATH/sm.ha/* $STATD_PATH/  > /dev/null 2>&1
b9953b
+			sm-notify $opts -v $ip
b9953b
+		done
b9953b
+	else
b9953b
+		sm-notify $opts
b9953b
+	fi
b9953b
+}
b9953b
+
b9953b
 nfsserver_start ()
b9953b
 {
b9953b
+	local rc;
b9953b
+
b9953b
 	if nfsserver_monitor; then
b9953b
 		ocf_log debug "NFS server is already started"
b9953b
 		return $OCF_SUCCESS
b9953b
 	fi
b9953b
 
b9953b
+	set_env_args
b9953b
 	prepare_directory
b9953b
 	bind_tree
b9953b
 
b9953b
+	# remove the sm-notify pid so sm-notify will be allowed to run again without requiring a reboot.
b9953b
+	rm -f /var/run/sm-notify.pid
b9953b
+	#
b9953b
+	# Synchronize these before starting statd
b9953b
+	#
b9953b
+	cp -rpfn $STATD_PATH/sm.ha/* $STATD_PATH/ > /dev/null 2>&1
b9953b
 	rm -rf $STATD_PATH/sm.ha/* > /dev/null 2>&1
b9953b
-	cp -rf $STATD_PATH/sm $STATD_PATH/sm.bak /var/lib/nfs/state $STATD_PATH/sm.ha > /dev/null 2>&1
b9953b
+	cp -rpf $STATD_PATH/sm $STATD_PATH/sm.bak /var/lib/nfs/state $STATD_PATH/sm.ha > /dev/null 2>&1
b9953b
 
b9953b
 	ocf_log info "Starting NFS server ..."
b9953b
 
b9953b
+	# mounts /proc/fs/nfsd for us
b9953b
+	lsmod | grep -q nfsd
b9953b
+	if [ $? -ne 0 ]; then
b9953b
+		modprobe nfsd
b9953b
+	fi
b9953b
+
b9953b
+	# check to see if we need to start rpc.statd
b9953b
+	v3locking_exec "status"
b9953b
+	if [ $? -ne $OCF_SUCCESS ]; then
b9953b
+		v3locking_exec "start"
b9953b
+		rc=$?
b9953b
+		if [ $rc -ne 0 ]; then
b9953b
+			ocf_log error "Failed to start NFS server locking daemons"
b9953b
+			return $rc
b9953b
+		fi
b9953b
+	else
b9953b
+		ocf_log info "rpc.statd already up"
b9953b
+	fi
b9953b
+
b9953b
 	fn=`mktemp`
b9953b
 	nfs_exec start > $fn 2>&1
b9953b
 	rc=$?
b9953b
-	ocf_log debug `cat $fn`
b9953b
+	ocf_log debug "$(cat $fn)"
b9953b
 	rm -f $fn
b9953b
 
b9953b
 	if [ $rc -ne 0 ]; then
b9953b
@@ -369,42 +698,7 @@ nfsserver_start ()
b9953b
 		return $rc
b9953b
 	fi	
b9953b
 
b9953b
-	#Notify the nfs server has been moved or rebooted
b9953b
-	#The init script do that already, but with the hostname, which may be ignored by client
b9953b
-	#we have to do it again with the nfs_ip 
b9953b
-	local opts
b9953b
-
b9953b
-	case ${OCF_RESKEY_nfs_notify_cmd##*/} in 
b9953b
-	sm-notify)
b9953b
-		# run in foreground, if requested
b9953b
-		if ocf_is_true "$OCF_RESKEY_nfs_notify_foreground"; then
b9953b
-			opts="-d"
b9953b
-		fi
b9953b
-
b9953b
-		if [ -n "$OCF_RESKEY_nfs_smnotify_retry_time" ]; then
b9953b
-			opts="$opts -m $OCF_RESKEY_nfs_smnotify_retry_time"
b9953b
-		fi
b9953b
-
b9953b
-		opts="$opts -f -v"
b9953b
-		;;
b9953b
-
b9953b
-	rpc.statd)
b9953b
-		if ocf_is_true "$OCF_RESKEY_nfs_notify_foreground"; then
b9953b
-			opts="-F"
b9953b
-		fi
b9953b
-		opts="$opts -n"
b9953b
-		;;
b9953b
-
b9953b
-	esac
b9953b
-
b9953b
-	rm -rf $STATD_PATH/sm.ha.save > /dev/null 2>&1
b9953b
-	cp -rf $STATD_PATH/sm.ha $STATD_PATH/sm.ha.save > /dev/null 2>&1
b9953b
-	for ip in `echo ${OCF_RESKEY_nfs_ip} | sed 's/,/ /g'`; do
b9953b
-	  ${OCF_RESKEY_nfs_notify_cmd} $opts $ip -P $STATD_PATH/sm.ha
b9953b
-	  rm -rf $STATD_PATH/sm.ha > /dev/null 2>&1
b9953b
-	  cp -rf $STATD_PATH/sm.ha.save $STATD_PATH/sm.ha > /dev/null 2>&1
b9953b
-	done
b9953b
-
b9953b
+	notify_locks
b9953b
 
b9953b
 	ocf_log info "NFS server started"
b9953b
 	return $OCF_SUCCESS
b9953b
@@ -414,12 +708,23 @@ nfsserver_stop ()
b9953b
 {
b9953b
 	ocf_log info "Stopping NFS server ..."
b9953b
 
b9953b
+	# backup the current sm state information to the ha folder before stopping.
b9953b
+	# the ha folder will be synced after startup, restoring the statd client state
b9953b
+	rm -rf $STATD_PATH/sm.ha/* > /dev/null 2>&1
b9953b
+	cp -rpf $STATD_PATH/sm $STATD_PATH/sm.bak /var/lib/nfs/state $STATD_PATH/sm.ha > /dev/null 2>&1
b9953b
+
b9953b
 	fn=`mktemp`
b9953b
 	nfs_exec stop > $fn 2>&1
b9953b
 	rc=$?
b9953b
-	ocf_log debug `cat $fn`
b9953b
+	ocf_log debug "$(cat $fn)"
b9953b
 	rm -f $fn
b9953b
 
b9953b
+	v3locking_exec "stop"
b9953b
+	if [ $? -ne 0 ]; then
b9953b
+		ocf_log err "Failed to stop NFS locking daemons"
b9953b
+		rc=$OCF_ERR_GENERIC
b9953b
+	fi
b9953b
+
b9953b
 	if [ $rc -eq 0 ]; then
b9953b
 		unbind_tree 
b9953b
 		ocf_log info "NFS server stopped"
b9953b
@@ -437,13 +742,9 @@ nfsserver_validate ()
b9953b
 	set_exec_mode
b9953b
 	check_binary ${OCF_RESKEY_nfs_notify_cmd}
b9953b
 
b9953b
-	if [ x = x"${OCF_RESKEY_nfs_ip}" ]; then
b9953b
-		ocf_log err "nfs_ip not set"
b9953b
-		exit $OCF_ERR_CONFIGURED
b9953b
-	fi
b9953b
 
b9953b
-	if [ x = "x$OCF_RESKEY_nfs_shared_infodir" ]; then
b9953b
-		ocf_log err "nfs_shared_infodir not set"
b9953b
+	if [ -n "$OCF_RESKEY_CRM_meta_clone" ] && [ -n "$OCF_RESKEY_nfs_shared_infodir" ]; then
b9953b
+		ocf_log err "This RA does not support clone mode when a shared info directory is in use."
b9953b
 		exit $OCF_ERR_CONFIGURED
b9953b
 	fi
b9953b
 
b9953b
@@ -465,11 +766,6 @@ nfsserver_validate ()
b9953b
 	return $OCF_SUCCESS
b9953b
 }
b9953b
 
b9953b
-if [ -n "$OCF_RESKEY_CRM_meta_clone" ]; then
b9953b
-	ocf_log err "THIS RA DO NOT SUPPORT CLONE MODE!"
b9953b
-	exit $OCF_ERR_CONFIGURED
b9953b
-fi
b9953b
-
b9953b
 nfsserver_validate
b9953b
 
b9953b
 case $__OCF_ACTION in
b9953b
-- 
b9953b
1.8.4.2
b9953b