Blame SOURCES/alsa-delay

a7e41e
#!/bin/bash
a7e41e
a7e41e
version=0.1.0
a7e41e
tmpdir=$TMPDIR
a7e41e
if test -z "$tmpdir"; then
a7e41e
	tmpdir="/tmp"
a7e41e
fi
a7e41e
tmpdir=$tmpdir/alsa-delay-script
a7e41e
a7e41e
delay=
a7e41e
pcard=
a7e41e
pdev=
a7e41e
ccard=
a7e41e
cdev=
a7e41e
a7e41e
yes=
a7e41e
quiet=
a7e41e
clean=
a7e41e
remove=
a7e41e
a7e41e
pdevice="default:%s"
a7e41e
pdevice_set=
a7e41e
cdevice="plughw:%s,%s,%s"
a7e41e
pctl=
a7e41e
cctl=
a7e41e
one=
a7e41e
arg=
a7e41e
mix=
a7e41e
pa=
a7e41e
a7e41e
useprocfs=yes
a7e41e
a7e41e
fuser_prg=fuser
a7e41e
insmod_prg=insmod
a7e41e
rmmod_prg=rmmod
a7e41e
lsmod_prg=lsmod
a7e41e
modprobe_prg=modprobe
a7e41e
chkconfig_prg=chkconfig
a7e41e
systemctl_prg=systemctl
a7e41e
pidof_prg=pidof
a7e41e
alsaloop_prg=alsaloop
a7e41e
amixer_prg=amixer
a7e41e
test -x /sbin/fuser && fuser_prg=/sbin/fuser
a7e41e
test -x /sbin/insmod && insmod_prg=/sbin/insmod
a7e41e
test -x /sbin/rmmod && rmmod_prg=/sbin/rmmod
a7e41e
test -x /sbin/lsmod && lsmod_prg=/sbin/lsmod
a7e41e
test -x /sbin/modprobe && modprobe_prg=/sbin/modprobe
a7e41e
test -x /sbin/chkconfig && chkconfig_prg=/sbin/chkconfig
a7e41e
test -x /bin/systemctl && systemctl_prg=/bin/systemctl
a7e41e
test -x /sbin/pidof && pidof_prg=/sbin/pidof
a7e41e
test -x /usr/bin/alsaloop && alsaloop_prg=/usr/bin/alsaloop
a7e41e
test -x /usr/local/bin/alsaloop && alsaloop_prg=/usr/local/bin/alsaloop
a7e41e
test -x /usr/bin/amixer && amixer_prg=/usr/bin/amixer
a7e41e
a7e41e
modprobeconf=/etc/modprobe.d/alsa.conf
a7e41e
test -r /etc/modprobe.conf && modprobeconf=/etc/modprobe.conf && useprocfs=""
a7e41e
alsaloopconf=/etc/alsaloop.conf
a7e41e
a7e41e
test -r modprobe.work && modprobeconf=modprobe.work
a7e41e
test -r modprobe.work && alsaloopconf=alsaloop.conf
a7e41e
a7e41e
usage() {
a7e41e
	echo "Usage: $0 [OPTION]... <requested_delay_in_ms> [<output_card>[,<device>]]"
a7e41e
	cat <
a7e41e
a7e41e
This is a delay utility version $version. The snd-aloop ALSA driver is
a7e41e
used to send all PCM streams back to the user space and the alsaloop
a7e41e
utility is used to send this stream to a real hardware.
a7e41e
a7e41e
<output_card> is ALSA card index (number) or string card identifier
a7e41e
<device> is ALSA device number
a7e41e
a7e41e
Use 'aplay -l' to list available cards and devices.
a7e41e
a7e41e
Operation modes:
a7e41e
  -h, --help		print this help, then exit
a7e41e
  -q, --quiet		quiet mode
a7e41e
  -y, --yes		do not ask any questions - answer is always yes
a7e41e
  -c, --clean           clean temporary directory and exit
a7e41e
  -r, --remove          remove the alsa-delay config modifications and exit
a7e41e
  --tmpdir=<DIR>        set temporary directory
a7e41e
Alsaloop options:
a7e41e
  --pdevice=<DEV>       force playback device
a7e41e
  --cdevice=<DEV>       force capture device
a7e41e
  --pctl=<DEV>          force playback ctl device
a7e41e
  --cctl=<DEV>          force capture ctl device
a7e41e
  --one=<ARG>           pass this argument to last thread
a7e41e
  --arg=<ARG>           pass this argument to all threads
a7e41e
  --mix=<ARG>           redirect ALSA mixer controls to OSS mixer
a7e41e
			(default is Master)
a7e41e
  --pa			Redirect PA to alsaloop
a7e41e
a7e41e
Note: For devices, the string %s is replaced with the card index
a7e41e
      and second string %s is replaced with the device index
a7e41e
      and third string %s is replaced the substream index (0-7).
a7e41e
      Example: hw:%s,%s,%s    (card, device, substream)
a7e41e
EOF
a7e41e
}
a7e41e
a7e41e
while :
a7e41e
do
a7e41e
	case "$1" in
a7e41e
	-h|--help)
a7e41e
		usage
a7e41e
		exit 0
a7e41e
		;;
a7e41e
	-q|--quiet)
a7e41e
		quiet=true ;;
a7e41e
	-y|--yes)
a7e41e
		yes=true ;;
a7e41e
	-c|--clean)
a7e41e
		clean="full" ;;
a7e41e
	-r|--remove)
a7e41e
		remove="full" ;;
a7e41e
	--tmpdir*)
a7e41e
		case "$#,$1" in
a7e41e
		*,*=*)
a7e41e
			tmpdir=`expr "z$1" : 'z-[^=]*=\(.*\)'` ;;
a7e41e
		1,*)
a7e41e
			usage ;;
a7e41e
		*)
a7e41e
			tmpdir="$2"
a7e41e
			shift ;;
a7e41e
		esac
a7e41e
		tmpdir="$tmpdir/alsa-compile-script"
a7e41e
		;;
a7e41e
	--pdevice*)
a7e41e
		case "$#,$1" in
a7e41e
		*,*=*)
a7e41e
			pdevice=`expr "z$1" : 'z-[^=]*=\(.*\)'`
a7e41e
			pdevice_set=yes ;;
a7e41e
		1,*)
a7e41e
			usage ;;
a7e41e
		*)
a7e41e
			pdevice="$2"
a7e41e
			pdevice_set=yes
a7e41e
			shift ;;
a7e41e
		esac
a7e41e
		;;
a7e41e
	--cdevice*)
a7e41e
		case "$#,$1" in
a7e41e
		*,*=*)
a7e41e
			cdevice=`expr "z$1" : 'z-[^=]*=\(.*\)'` ;;
a7e41e
		1,*)
a7e41e
			usage ;;
a7e41e
		*)
a7e41e
			cdevice="$2"
a7e41e
			shift ;;
a7e41e
		esac
a7e41e
		;;
a7e41e
	--pctl*)
a7e41e
		case "$#,$1" in
a7e41e
		*,*=*)
a7e41e
			pctl=`expr "z$1" : 'z-[^=]*=\(.*\)'` ;;
a7e41e
		1,*)
a7e41e
			usage ;;
a7e41e
		*)
a7e41e
			pctl="$2"
a7e41e
			shift ;;
a7e41e
		esac
a7e41e
		;;
a7e41e
	--cctl*)
a7e41e
		case "$#,$1" in
a7e41e
		*,*=*)
a7e41e
			cctl=`expr "z$1" : 'z-[^=]*=\(.*\)'` ;;
a7e41e
		1,*)
a7e41e
			usage ;;
a7e41e
		*)
a7e41e
			cctl="$2"
a7e41e
			shift ;;
a7e41e
		esac
a7e41e
		;;
a7e41e
	--one*)
a7e41e
		case "$#,$1" in
a7e41e
		*,*=*)
a7e41e
			one1=`expr "z$1" : 'z-[^=]*=\(.*\)'` ;;
a7e41e
		1,*)
a7e41e
			usage ;;
a7e41e
		*)
a7e41e
			one1="$2"
a7e41e
			shift ;;
a7e41e
		esac
a7e41e
		one="$one $one1"
a7e41e
		;;
a7e41e
	--arg*)
a7e41e
		case "$#,$1" in
a7e41e
		*,*=*)
a7e41e
			arg1=`expr "z$1" : 'z-[^=]*=\(.*\)'` ;;
a7e41e
		1,*)
a7e41e
			usage ;;
a7e41e
		*)
a7e41e
			arg1="$2"
a7e41e
			shift ;;
a7e41e
		esac
a7e41e
		arg="$arg $arg1"
a7e41e
		;;
a7e41e
	--mix*)
a7e41e
		case "$#,$1" in
a7e41e
		*,*=*)
a7e41e
			mix=`expr "z$1" : 'z-[^=]*=\(.*\)'` ;;
a7e41e
		1,*)
a7e41e
			usage ;;
a7e41e
		*)
a7e41e
			mix="$2"
a7e41e
			shift ;;
a7e41e
		esac
a7e41e
		arg="$arg $arg1"
a7e41e
		;;
a7e41e
	--pa)
a7e41e
		pa=true
a7e41e
		;;
a7e41e
	*)
a7e41e
		if test -n "$1"; then
a7e41e
			ok=
a7e41e
			if test -z "$delay"; then
a7e41e
				delay="$1"
a7e41e
				if test "$delay" -lt 1000; then
a7e41e
					delay=$[$delay * 1000]
a7e41e
				fi
a7e41e
				ok=true
a7e41e
			fi
a7e41e
			if test -z "$ok" -a -z "$pcard"; then
a7e41e
				pcard="$1"
a7e41e
				ok=true
a7e41e
			fi
a7e41e
			if test -z "$ok"; then
a7e41e
				echo "Unknown parameter '$1'"
a7e41e
				break
a7e41e
			fi
a7e41e
		else
a7e41e
			break
a7e41e
		fi
a7e41e
		;;
a7e41e
	esac
a7e41e
	shift
a7e41e
done
a7e41e
a7e41e
test -z "$pdevice_set" -a -n "$pa" && pdevice="plug:dmix:%s"
a7e41e
test -z "$delay" && delay="50000"
a7e41e
test -z "$pcard" && pcard="1"
a7e41e
pdev=$(echo $pcard | cut -s -d , -f 2)
a7e41e
pcard=$(echo $pcard | cut -d , -f 1)
a7e41e
test -z "$pdev" && pdev="0"
a7e41e
test -z "$ccard" && ccard="Loopback,1"
a7e41e
cdev=$(echo $ccard | cut -s -d , -f 2)
a7e41e
ccard=$(echo $ccard | cut -d , -f 1)
a7e41e
test -z "$cdev" && cdev="0"
a7e41e
a7e41e
# Echo "true" or "false", depending on $yes and user response to prompt
a7e41e
# $1 is prompt message
a7e41e
question_bool() {
a7e41e
	if test "$yes" = "yes"; then
a7e41e
		echo "true"
a7e41e
	else
a7e41e
		echo >&2 -n "$1 (Y/ ) "
a7e41e
		read i
a7e41e
		local i=${i:0:1}
a7e41e
		if test "$i" = "Y" -o "$i" = "y"; then
a7e41e
			echo "true"
a7e41e
		else
a7e41e
			echo "false"
a7e41e
		fi
a7e41e
	fi
a7e41e
}
a7e41e
a7e41e
# Safe exit
a7e41e
safe_exit() {
a7e41e
	if test -r /etc/pulse/client.conf ; then
a7e41e
		if grep "# alsa-delay-line-to-be-removed-1234" /etc/pulse/client.conf 2> /dev/null > /dev/null ; then
a7e41e
			grep -v alsa-delay-line-to-be-removed-1234 /etc/pulse/client.conf > /etc/pulse/client.conf.new
a7e41e
			if test -s /etc/pulse/client.conf.new; then
a7e41e
				mv /etc/pulse/client.conf.new /etc/pulse/client.conf
a7e41e
			fi
a7e41e
		fi
a7e41e
	fi
a7e41e
	exit $1
a7e41e
}
a7e41e
a7e41e
# Log and execute $@ and check success
a7e41e
do_cmd() {
a7e41e
	if test -z "$quiet"; then
a7e41e
		echo "> $@"
a7e41e
	fi
a7e41e
	$@ || safe_exit 1
a7e41e
}
a7e41e
a7e41e
# Cache or restore $protocol and $url and $package in $tmpdir
a7e41e
check_environment() {
a7e41e
	if ! test -d $tmpdir ; then
a7e41e
		mkdir -p $tmpdir
a7e41e
		if ! test -d $tmpdir; then
a7e41e
			echo >&2 "Unable to create directory $tmpdir."
a7e41e
			exit 1
a7e41e
		fi
a7e41e
	fi
a7e41e
	echo "Using temporary tree: $tmpdir"
a7e41e
	test -x /bin/depmod && depmodbin=/bin/depmod
a7e41e
	test -x /sbin/depmod && depmodbin=/sbin/depmod
a7e41e
	if test -z "$depmodbin"; then
a7e41e
		echo >&2 "Unable to find depmod utility."
a7e41e
		exit 1
a7e41e
	fi
a7e41e
	test -x /bin/modinfo && modinfobin=/bin/modinfo
a7e41e
	test -x /sbin/modinfo && modinfobin=/sbin/modinfo
a7e41e
	if test -z "$modinfobin"; then
a7e41e
		echo >&2 "Unable to find modinfo utility."
a7e41e
		exit 1
a7e41e
	fi
a7e41e
}
a7e41e
a7e41e
# Kill processes currently accessing the audio devices
a7e41e
kill_audio_apps() {
a7e41e
	local pids0=$($fuser_prg /dev/snd/* 2> /dev/null)
a7e41e
	local pids1=$($fuser_prg /dev/mixer* 2> /dev/null)
a7e41e
	local pids2=$($fuser_prg /dev/sequencer* 2> /dev/null)
a7e41e
	local pids=
a7e41e
	for pid in $pids0 $pids1 $pids2; do
a7e41e
		local pids="$pids $pid"
a7e41e
	done
a7e41e
	if ! test -z "$pids"; then
a7e41e
		echo
a7e41e
		echo "WARNING! An audio application uses ALSA driver:"
a7e41e
		echo
a7e41e
		for pid in $pids; do
a7e41e
			ps --no-headers -p $pids || safe_exit 1
a7e41e
		done
a7e41e
		echo
a7e41e
		if test $(question_bool "Would you like to kill these apps?") = "true"; then
a7e41e
			if test -w /etc/pulse/client.conf ; then
a7e41e
				echo "autospawn = no # alsa-delay-line-to-be-removed-1234" >> /etc/pulse/client.conf
a7e41e
			fi
a7e41e
			for pid in $pids; do
a7e41e
				do_cmd kill $pid
a7e41e
			done
a7e41e
			sleep 2
a7e41e
			local killed=
a7e41e
			for pid in $pids; do
a7e41e
				local a=$(ps --no-headers -p $pids)
a7e41e
				if test -n "$a"; then
a7e41e
					do_cmd kill -9 $pid
a7e41e
					local killed="true"
a7e41e
				fi
a7e41e
			done
a7e41e
			if test "$killed" = "true"; then
a7e41e
				sleep 2
a7e41e
				for pid in $pids; do
a7e41e
					local a=$(ps --no-headers -p $pids)
a7e41e
					if test -n "$a"; then
a7e41e
						echo >&2 "Unable to kill application:"
a7e41e
						echo >&2 "  $a"
a7e41e
						safe_exit 1
a7e41e
					fi
a7e41e
				done
a7e41e
			fi
a7e41e
		else
a7e41e
			echo >&2 "Cannot continue with running audio applications."
a7e41e
			safe_exit 1
a7e41e
		fi
a7e41e
	fi
a7e41e
}
a7e41e
a7e41e
# Echo the list of configured sound modules
a7e41e
configured_modules() {
a7e41e
	if test -z "$useprocfs"; then
a7e41e
		cat $modprobeconf | grep -E "^alias snd-card-" | cut -d ' ' -f 3
a7e41e
	else
a7e41e
		cat /proc/asound/modules | colrm 1 3
a7e41e
	fi
a7e41e
}
a7e41e
a7e41e
# Echo the list of loaded sound modules
a7e41e
current_modules() {
a7e41e
	$lsmod_prg | cut -d ' ' -f 1 | grep -E "^(snd[_-])"
a7e41e
}
a7e41e
a7e41e
# The loopback kernel driver detection
a7e41e
aloop_present() {
a7e41e
	if test -r /proc/asound/card0/id; then
a7e41e
		local id=$(cat /proc/asound/card0/id)
a7e41e
	else
a7e41e
		local id=""
a7e41e
	fi
a7e41e
	if test "$id" = "Loopback"; then
a7e41e
		echo "yes"
a7e41e
	fi
a7e41e
}
a7e41e
a7e41e
# Remove kernel modules, using two phases
a7e41e
# $@ is module names
a7e41e
my_rmmod() {
a7e41e
	local phase2=
a7e41e
	while test -n "$1"; do
a7e41e
		if ! $rmmod_prg $1 2> /dev/null > /dev/null; then
a7e41e
			local phase2="$phase2 $1"
a7e41e
		else
a7e41e
			echo "> rmmod $1"
a7e41e
		fi
a7e41e
		shift
a7e41e
	done
a7e41e
	for mod in $phase2; do
a7e41e
		echo "> rmmod $mod"
a7e41e
		if ! $rmmod_prg $mod ; then
a7e41e
			echo >&2 "Unable to remove kernel module $mod."
a7e41e
			safe_exit 1
a7e41e
		fi
a7e41e
	done
a7e41e
}
a7e41e
a7e41e
# Reload kernel modules
a7e41e
kernel_modules() {
a7e41e
	kill_audio_apps
a7e41e
	if test "$1" = "unload"; then
a7e41e
		local present=$(aloop_present)
a7e41e
		if test "$present" = "yes"; then
a7e41e
			if ! $rmmod_prg snd-aloop; then
a7e41e
				echo >&2 "Unable to remove kernel module snd-aloop."
a7e41e
				safe_exit 1
a7e41e
			fi
a7e41e
		fi
a7e41e
	fi
a7e41e
	local curmods=$(current_modules)
a7e41e
	local usermods=$(configured_modules)
a7e41e
	my_rmmod $curmods
a7e41e
	$modprobe_prg soundcore || safe_exit 1
a7e41e
	if test "$1" = "load"; then
a7e41e
		if ! $modprobe_prg snd-aloop; then
a7e41e
			echo >&2 "Unable to install kernel module snd-aloop."
a7e41e
			safe_exit 1
a7e41e
		fi
a7e41e
	fi
a7e41e
	for mod in $usermods; do
a7e41e
		if ! $modprobe_prg $mod; then
a7e41e
			echo >&2 "Unable to install kernel module $mod."
a7e41e
			safe_exit 1
a7e41e
		fi
a7e41e
	done
a7e41e
	echo "Kernel modules ready:"
a7e41e
	cat /proc/asound/cards
a7e41e
	sleep 0.5
a7e41e
}
a7e41e
a7e41e
# If $package is alsa-driver then remove current modules
a7e41e
kernel_modules_remove() {
a7e41e
	local curmods=$(current_modules)
a7e41e
	if test -z "$curmods"; then
a7e41e
		echo "No ALSA kernel modules to remove."
a7e41e
		safe_exit 0
a7e41e
	fi
a7e41e
	kill_audio_apps
a7e41e
	my_rmmod $curmods
a7e41e
	echo "ALSA kernel modules removed."
a7e41e
}
a7e41e
a7e41e
function clean() {
a7e41e
	echo -n "Removing tree $tmpdir:"
a7e41e
	if test -d "$tmpdir"; then
a7e41e
		if ! rm -rf "$tmpdir"; then
a7e41e
			echo " failed"
a7e41e
			safe_exit 1
a7e41e
		fi
a7e41e
	fi
a7e41e
	echo " success"
a7e41e
}
a7e41e
a7e41e
function reindex_modprobe_conf() {
a7e41e
	if test -z "$useprocfs"; then
a7e41e
		cat > $tmpdir/run.awk <
a7e41e
function rewrite_line(line,   res, c, i, j) {
a7e41e
	split(line, l, " ")
a7e41e
	if (l[1] == "alias") {
a7e41e
		printf(l[1])
a7e41e
		c = int(substr(l[2], 9)) + 1
a7e41e
		printf(" snd-card-%i %s\n", c, l[3])
a7e41e
	} else if (l[1] == "options") {
a7e41e
		printf(l[1])
a7e41e
		for (i = 2; i <= length(l); i++) {
a7e41e
			c = substr(l[i], 0, 9)
a7e41e
			if (c == "snd-card-") {
a7e41e
				c = int(substr(l[i], 9)) + 1
a7e41e
				printf(" snd-card-%i", c)
a7e41e
				continue
a7e41e
			}
a7e41e
			c = substr(l[i], 0, 6)
a7e41e
			if (c == "index=") {
a7e41e
				printf(" index=")
a7e41e
				c = substr(l[i], 6)
a7e41e
				split(c, y, ",")
a7e41e
				for (j = 1; j <= length(y); j++) {
a7e41e
					c = int(y[j]) + 1
a7e41e
					if (j > 1)
a7e41e
						printf(",")
a7e41e
					printf("%i", c)
a7e41e
				}
a7e41e
				continue
a7e41e
			}
a7e41e
			printf(" %s", l[i])
a7e41e
		}
a7e41e
		printf("\n")
a7e41e
	} else if (l[1] == "remove") {
a7e41e
		printf(l[1])
a7e41e
		flag = 0
a7e41e
		for (i = 2; i <= length(l); i++) {
a7e41e
			if (l[i] == "/usr/sbin/alsactl" || l[i] == "store") {
a7e41e
				flag++;
a7e41e
				printf(" %s", l[i]);
a7e41e
				continue;
a7e41e
			}
a7e41e
			if (flag == 2) {
a7e41e
				c = int(l[i]) + 1;
a7e41e
				printf(" %i", c);
a7e41e
				flag = 0;
a7e41e
				continue;
a7e41e
			}
a7e41e
			flag = 0;
a7e41e
			printf(" %s", l[i]);
a7e41e
		}
a7e41e
		printf("\n")
a7e41e
	} else {
a7e41e
		print line
a7e41e
	}
a7e41e
}
a7e41e
a7e41e
BEGIN			{ aloop=0; }
a7e41e
/alias snd-card-0 snd-aloop/ { aloop=1; next; }
a7e41e
/options snd-card-0 index=0/ { if (aloop) next; }
a7e41e
/options snd-aloop/	{ next; }
a7e41e
/alias snd-card-/	{ rewrite_line(\$0); next; }
a7e41e
/options snd-/		{ rewrite_line(\$0); next; }
a7e41e
/remove snd-/		{ rewrite_line(\$0); next; }
a7e41e
			{ print \$0; }
a7e41e
EOF
a7e41e
		cat $modprobeconf | awk -f $tmpdir/run.awk > $modprobeconf.new
a7e41e
		rm $tmpdir/run.awk || safe_exit 1
a7e41e
	else
a7e41e
		local index=1
a7e41e
		echo -n "" > $modprobeconf.new
a7e41e
		declare -a modules printed
a7e41e
		for mod in $(cat /proc/asound/modules | grep -Ev "^2[6789]" | colrm 1 3); do
a7e41e
			if test "$mod" != "snd_aloop" -a "$mod" != "snd-aloop"; then
a7e41e
				if test -z ${modules[$mod]}; then
a7e41e
					modules[$mod]=$index
a7e41e
				else
a7e41e
					modules[$mod]="${modules[mod]},$index"
a7e41e
				fi
a7e41e
				echo "alias snd-card-$index $mod" >> $modprobeconf.new
a7e41e
				echo "options snd-card-$index index=$index" >> $modprobeconf.new
a7e41e
			fi
a7e41e
			index=$(expr $index + 1)
a7e41e
		done
a7e41e
		for mod in $(cat /proc/asound/modules | grep -Ev "^2[6789]" | colrm 1 3); do
a7e41e
			if test "$mod" != "snd_aloop" -a "$mod" != "snd-aloop"; then
a7e41e
				if test -z ${printed[$mod]}; then
a7e41e
					echo "options $mod index=${modules[$mod]}" >> $modprobeconf.new
a7e41e
					printed[$mod]=yes
a7e41e
				fi
a7e41e
			fi
a7e41e
		done
a7e41e
	fi
a7e41e
cat >> $modprobeconf.new <
a7e41e
alias snd-card-0 snd-aloop
a7e41e
options snd-card-0 index=0
a7e41e
options snd-aloop index=0
a7e41e
EOF
a7e41e
	if ! test -f $modprobeconf.alsa-delay.save ; then
a7e41e
		if ! test -f $modprobeconf ; then
a7e41e
			touch $modprobeconf.alsa-delay.save
a7e41e
		else
a7e41e
			cp $modprobeconf $modprobeconf.alsa-delay.save
a7e41e
		fi
a7e41e
	fi
a7e41e
	mv $modprobeconf.new $modprobeconf || safe_exit 1
a7e41e
}
a7e41e
a7e41e
function modify_modprobe_conf() {
a7e41e
	local check=$(cat $modprobeconf 2> /dev/null | grep -E "^alias snd-card-0 snd-aloop")
a7e41e
	local present=$(aloop_present)
a7e41e
	if test -n "$check"; then
a7e41e
		echo "Module snd-aloop is already installed."
a7e41e
		if ! test "$present" = "yes"; then
a7e41e
			kernel_modules load
a7e41e
		fi
a7e41e
	else
a7e41e
		reindex_modprobe_conf
a7e41e
		if ! test "$present" = "yes"; then
a7e41e
			kernel_modules load
a7e41e
		else
a7e41e
			kernel_modules
a7e41e
		fi
a7e41e
	fi
a7e41e
}
a7e41e
a7e41e
function myprintf() {
a7e41e
	local cnt=$(echo "$1" | grep -o "%" | wc -l)
a7e41e
	if test $cnt -eq 1; then
a7e41e
		printf "$1" "$2"
a7e41e
	else if test $cnt -eq 2; then
a7e41e
		printf "$1" "$2" "$3"
a7e41e
	else
a7e41e
		printf "$1" "$2" "$3" "$4"
a7e41e
	fi
a7e41e
	fi
a7e41e
}
a7e41e
a7e41e
function check_ctl_name() {
a7e41e
	$amixer_prg -D "$1" contents | grep "name='$2'"
a7e41e
}
a7e41e
a7e41e
function check_oss_mixer() {
a7e41e
	grep ": mixer" /proc/asound/oss/devices
a7e41e
}
a7e41e
a7e41e
function get_card_id() {
a7e41e
	$amixer_prg -D "$1" info | head -1 | cut -d '/' -f 1 | cut -d ' ' -f 3- | awk '{ print substr($0, 2, length($0)-2) }'
a7e41e
}
a7e41e
a7e41e
function generate_alsaloop_line() {
a7e41e
	local file="$1"
a7e41e
	local idx="$2"
a7e41e
	local carg=$(myprintf "$cdevice" $ccard $cdev $idx)
a7e41e
	local parg=$(myprintf "$pdevice" $pcard $pdev $idx)
a7e41e
	local res="-C $carg -P $parg -T $idx -t $delay"
a7e41e
	if test -n "$cctl"; then
a7e41e
		local carg=$(myprintf "$cctl" $pcard $cdev $idx)
a7e41e
		local res="$res -Y $carg"
a7e41e
	fi
a7e41e
	if test -n "$pctl"; then
a7e41e
		local parg=$(myprintf "$pctl" $pcard $pdev $idx)
a7e41e
		local res="$res -X $parg"
a7e41e
	fi
a7e41e
	if test -n "$arg"; then
a7e41e
		local res="$res $arg"
a7e41e
	fi
a7e41e
	if test $idx -eq 7; then
a7e41e
		if test -n "$one"; then
a7e41e
			local res="$res $one"
a7e41e
		fi
a7e41e
		if test -z "$pctl"; then
a7e41e
			local res="$res -X hw:$pcard"
a7e41e
		fi
a7e41e
		local mymix="$mix"
a7e41e
		if test -z "$mymix"; then
a7e41e
			local mymix="Master"
a7e41e
			for mymix in Master "Master Mono" Headphone Headphone2 PCM Speaker \
a7e41e
				     "Desktop Speaker" Beep Front Rear Center LFE Side \
a7e41e
				     Surround ; do
a7e41e
				local check=$(check_ctl_name hw:$pcard "$mymix Playback Switch")
a7e41e
				if test -n "$check"; then
a7e41e
					local res="$res -m \"name='$mymix Playback Switch'\""
a7e41e
				fi
a7e41e
				local check=$(check_ctl_name hw:$pcard "$mymix Playback Volume")
a7e41e
				if test -n "$check"; then
a7e41e
					local res="$res -m \"name='$mymix Playback Volume'\""
a7e41e
				fi
a7e41e
			done
a7e41e
		else
a7e41e
			local check=$(check_ctl_name hw:$pcard "$mymix Playback Switch")
a7e41e
			if test -n "$check"; then
a7e41e
				local res="$res -m \"name='$mymix Playback Switch'\""
a7e41e
			fi
a7e41e
			local check=$(check_ctl_name hw:$pcard "$mymix Playback Volume")
a7e41e
			if test -n "$check"; then
a7e41e
				local res="$res -m \"name='$mymix Playback Volume'\""
a7e41e
			fi
a7e41e
		fi
a7e41e
		local check=$(check_oss_mixer)
a7e41e
		if test -n "$check"; then
a7e41e
			local res="$res -O \"$mymix@VOLUME\""
a7e41e
		fi
a7e41e
	fi
a7e41e
	echo $res >> $file
a7e41e
}
a7e41e
a7e41e
function generate_alsaloop_conf() {
a7e41e
	local idx=0
a7e41e
	rm -f $alsaloopconf.new
a7e41e
	while test $idx -lt 8; do
a7e41e
		generate_alsaloop_line $alsaloopconf.new $idx
a7e41e
		idx=$(expr $idx + 1)
a7e41e
	done
a7e41e
	mv $alsaloopconf.new $alsaloopconf || safe_exit 1
a7e41e
}
a7e41e
a7e41e
function kill_alsaloop() {
a7e41e
	local pid=$($pidof_prg alsaloop)
a7e41e
	if test -n "$pid"; then
a7e41e
		echo "Killing alsaloop..."
a7e41e
	fi
a7e41e
	while test -n "$pid"; do
a7e41e
		kill $pid
a7e41e
		sleep 0.01
a7e41e
		pid=$($pidof_prg alsaloop)
a7e41e
	done
a7e41e
}
a7e41e
a7e41e
function restart_alsaloop() {
a7e41e
	echo "Restarting alsaloop: delay $delay us."
a7e41e
	if test -d /etc/systemd/system; then
a7e41e
		systemctl start alsaloop.service
a7e41e
	else
a7e41e
		/etc/init.d/alsaloop start
a7e41e
	fi
a7e41e
	sleep 0.4
a7e41e
	local pid=$($pidof_prg alsaloop)
a7e41e
	if test -z "$pid"; then
a7e41e
		echo "ERROR: Not started, check /var/log/messages for details"
a7e41e
	else
a7e41e
		if test -d /etc/systemd/system; then
a7e41e
			systemctl status alsaloop.service
a7e41e
		else
a7e41e
			ps -p $pid
a7e41e
		fi
a7e41e
	fi
a7e41e
}
a7e41e
a7e41e
function create_systemd_service() {
a7e41e
	if test -r /etc/systemd/system/alsaloop.service; then
a7e41e
		echo "/etc/systemd/system/alsaloop.service exists"
a7e41e
	else
a7e41e
		cat >> /etc/systemd/system/alsaloop.service <
a7e41e
# This file was created by the alsa-delay utility
a7e41e
a7e41e
[Unit]
a7e41e
Description=The alsaloop daemon service
a7e41e
After=alsa-restore.service
a7e41e
Before=shutdown.target
a7e41e
Conflicts=shutdown.target
a7e41e
a7e41e
[Service]
a7e41e
Type=simple
a7e41e
ExecPre=-/sbin/modprobe snd-aloop
a7e41e
ExecStart=-/usr/bin/alsaloop --syslog --config $alsaloopconf
a7e41e
EOF
a7e41e
		ln -sf ../alsaloop.service /etc/systemd/system/basic.target.wants/alsaloop.service
a7e41e
	fi
a7e41e
}
a7e41e
a7e41e
function create_initd() {
a7e41e
	if test -x /etc/init.d/alsaloop; then
a7e41e
		echo "/etc/init.d/alsaloop exists"
a7e41e
	else
a7e41e
		cat >> /etc/init.d/alsaloop <
a7e41e
#! /bin/sh
a7e41e
a7e41e
# alsaloop init file
a7e41e
# Copyright (C) 2010 Jaroslav Kysela <jkysela@redhat.com>
a7e41e
a7e41e
# For RedHat and cousins:
a7e41e
# chkconfig: 2345 99 01
a7e41e
# description: Start alsaloop daemon
a7e41e
# processname: alsaloop
a7e41e
a7e41e
### BEGIN INIT INFO
a7e41e
# Provides: alsaloop
a7e41e
# Required-Start: \$local_fs
a7e41e
# Required-Stop: \$local_fs
a7e41e
# Should-Start:
a7e41e
# Short-Description: ALSALOOP
a7e41e
# Description: ALSALOOP
a7e41e
### END INIT INFO
a7e41e
a7e41e
# This program is free software; you can redistribute it and/or modify it
a7e41e
# under the terms of the GNU General Public License as published by the Free
a7e41e
# Software Foundation; either version 2, or (at your option) any later
a7e41e
# version.
a7e41e
# You should have received a copy of the GNU General Public License (for
a7e41e
# example COPYING); if not, write to the Free Software Foundation, Inc., 675
a7e41e
# Mass Ave, Cambridge, MA 02139, USA.
a7e41e
# This code was originally developed as a Senior Thesis by Michael Cornwell
a7e41e
# at the Concurrent Systems Laboratory (now part of the Storage Systems
a7e41e
# Research Center), Jack Baskin School of Engineering, University of
a7e41e
# California, Santa Cruz. http://ssrc.soe.ucsc.edu/.
a7e41e
a7e41e
alsaloop_opts="--daemonize --workaround serialopen --config $alsaloopconf"
a7e41e
a7e41e
ALSALOOP_BIN=/usr/bin/alsaloop
a7e41e
test -x /usr/local/bin/alsaloop && ALSALOOP_BIN=/usr/local/bin/alsaloop
a7e41e
a7e41e
# Source function library
a7e41e
. /etc/rc.d/init.d/functions
a7e41e
a7e41e
RETVAL=0
a7e41e
prog=alsaloop
a7e41e
pidfile=/var/lock/subsys/alsaloop
a7e41e
a7e41e
start()
a7e41e
{
a7e41e
	modprobe snd-aloop
a7e41e
	echo -n \$"Starting \$prog: "
a7e41e
	daemon \$ALSALOOP_BIN \$alsaloop_opts
a7e41e
	RETVAL=\$?
a7e41e
	echo
a7e41e
	[ \$RETVAL = 0 ] && touch \$pidfile
a7e41e
	return \$RETVAL
a7e41e
}
a7e41e
a7e41e
stop()
a7e41e
{
a7e41e
	echo -n \$"Shutting down \$prog: "
a7e41e
	killproc \$ALSALOOP_BIN
a7e41e
	RETVAL=\$?
a7e41e
	echo
a7e41e
	rm -f \$pidfile
a7e41e
	return \$RETVAL
a7e41e
}
a7e41e
a7e41e
case "\$1" in
a7e41e
	start)
a7e41e
		start
a7e41e
		;;
a7e41e
	stop)
a7e41e
		stop
a7e41e
		;;
a7e41e
	restart)
a7e41e
		stop
a7e41e
		start
a7e41e
		;;
a7e41e
	try-restart)
a7e41e
		if [ -f \$pidfile ]; then
a7e41e
			stop
a7e41e
			start
a7e41e
		fi
a7e41e
		;;
a7e41e
	status)
a7e41e
		status $prog
a7e41e
		RETVAL=$?
a7e41e
		;;
a7e41e
	*)
a7e41e
		echo \$"Usage: \$0 {start|stop|restart|try-restart}"
a7e41e
		RETVAL=3
a7e41e
esac
a7e41e
a7e41e
exit \$RETVAL
a7e41e
EOF
a7e41e
		chmod 755 /etc/init.d/alsaloop || safe_exit 1
a7e41e
		$chkconfig_prg --add alsaloop
a7e41e
		$chkconfig_prg alsaloop on
a7e41e
	fi
a7e41e
}
a7e41e
a7e41e
function modify_pa_conf() {
a7e41e
a7e41e
	if grep Loopback /etc/pulse/default.pa > /dev/null; then
a7e41e
		echo "/etc/pulse/default.pa already modified"
a7e41e
	else
a7e41e
		if test -f /etc/pulse/default.pa; then
a7e41e
			local pcardid=$(get_card_id "hw:$pcard")
a7e41e
			if ! test -f /etc/pulse/default.pa.alsa-delay.save ; then
a7e41e
				cp /etc/pulse/default.pa /etc/pulse/default.pa.alsa-delay.save
a7e41e
			fi
a7e41e
			cat /etc/pulse/default.pa | \
a7e41e
				sed -e 's/^load-module module-udev-detect/#load-module module-udev-detect # commented by alsa-delay/' \
a7e41e
				    -e 's/^load-module module-detect/#load-module module-detect # commented by alsa-delay/' \
a7e41e
				    -e "s/^### Automatically load driver modules for Bluetooth hardware/load-module module-alsa-card device_id=Loopback\nload-module module-alsa-card device_id=$pcardid\n\n### Automatically load driver modules for Bluetooth hardware/" > /etc/pulse/default.pa.new
a7e41e
			if test -s /etc/pulse/default.pa.new; then
a7e41e
				mv /etc/pulse/default.pa.new /etc/pulse/default.pa
a7e41e
				echo "/etc/pulse/default.pa changed"
a7e41e
			fi
a7e41e
		fi
a7e41e
	fi
a7e41e
a7e41e
	local file=
a7e41e
	local asoundconf=
a7e41e
	if test -d /etc/alsa -a -f /etc/alsa/pulse-default.conf; then
a7e41e
		file=/etc/alsa/alsaloop-default.conf
a7e41e
		asoundconf=yes
a7e41e
	fi
a7e41e
	if test -f /usr/share/alsa/alsa.conf.d/99-pulseaudio-default.conf ; then
a7e41e
		file=/usr/share/alsa/alsa.conf.d/A0-alsaloop-default.conf
a7e41e
	fi
a7e41e
	if test -z "$file"; then
a7e41e
		echo "unknown PulseAudio setup (asound.conf)"
a7e41e
	elif test -f $file; then
a7e41e
		echo "$file exists"
a7e41e
	else
a7e41e
		cat >> $file <
a7e41e
pcm.!default {
a7e41e
	@args [ CARD ]
a7e41e
	@args.CARD {
a7e41e
		type string
a7e41e
	}
a7e41e
	type asym
a7e41e
	playback.pcm "plughw:Loopback"
a7e41e
	capture.pcm {
a7e41e
		type pulse
a7e41e
	}
a7e41e
}
a7e41e
EOF
a7e41e
		if test -n "$asoundconf"; then
a7e41e
			cat /etc/asound.conf | sed 's/pulse-default.conf/alsaloop-default.conf/g' > /etc/asound.conf.new
a7e41e
			if test -s /etc/asound.conf.new; then
a7e41e
				mv /etc/asound.conf.new /etc/asound.conf
a7e41e
			fi
a7e41e
		fi
a7e41e
	fi
a7e41e
}
a7e41e
a7e41e
function remove() {
a7e41e
	if test -f /etc/pulse/default.pa.alsa-delay.save ; then
a7e41e
		echo "Restoring /etc/pulse/default.pa"
a7e41e
		mv /etc/pulse/default.pa.alsa-delay.save /etc/pulse/default.pa
a7e41e
	fi
a7e41e
	if test -f /etc/init.d/alsaloop ; then
a7e41e
		echo "Removing /etc/init.d/alsaloop service"
a7e41e
		$chkconfig_prg alsaloop off
a7e41e
		$chkconfig_prg --del alsaloop
a7e41e
		rm /etc/init.d/alsaloop
a7e41e
	fi
a7e41e
	if test -f /etc/systemd/system/alsaloop.service ; then
a7e41e
		echo "Removing /etc/systemd/system/alsaloop.service"
a7e41e
		$systemctl_prg disable alsaloop.service
a7e41e
		rm /etc/systemd/system/alsaloop.service
a7e41e
	fi
a7e41e
	if test -f $alsaloopconf ; then
a7e41e
		echo "Removing $alsaloopconf"
a7e41e
		rm $alsaloopconf
a7e41e
	fi
a7e41e
	if test -f /etc/alsa/alsaloop-default.conf ; then
a7e41e
		echo "Removing /etc/alsa/alsaloop-default.conf"
a7e41e
		rm /etc/alsa/alsaloop-default.conf
a7e41e
	fi
a7e41e
	if test -f /usr/share/alsa/alsa.conf.d/A0-alsaloop-default.conf ; then
a7e41e
		echo "Removing /usr/share/alsa/alsa.conf.d/A0-alsaloop-default.conf"
a7e41e
		rm /usr/share/alsa/alsa.conf.d/A0-alsaloop-default.conf
a7e41e
	fi
a7e41e
	if grep "alsaloop-default.conf" /etc/asound.conf > /dev/null 2> /dev/null ; then
a7e41e
		echo "Modifying /etc/asound.conf"
a7e41e
		cat /etc/asound.conf | sed 's/alsaloop-default.conf/pulse-default.conf/g' > /etc/asound.conf.new
a7e41e
		if test -s /etc/asound.conf.new; then
a7e41e
			mv /etc/asound.conf.new /etc/asound.conf
a7e41e
		fi
a7e41e
	fi
a7e41e
	if test -f $modprobeconf.alsa-delay.save ; then
a7e41e
		echo "Restoring $modprobeconf"
a7e41e
		if test -s $modprobeconf.alsa-delay.save ; then
a7e41e
			mv $modprobeconf.alsa-delay.save $modprobeconf
a7e41e
		else
a7e41e
			rm $modprobeconf.alsa-delay.save
a7e41e
			rm $modprobeconf
a7e41e
		fi
a7e41e
	fi
a7e41e
	if test $(aloop_present) = "yes"; then
a7e41e
		kernel_modules unload
a7e41e
	fi
a7e41e
}
a7e41e
a7e41e
rundir=$(pwd)
a7e41e
export LC_ALL=C
a7e41e
export LANGUAGE=C
a7e41e
a7e41e
check_environment
a7e41e
a7e41e
if test -n "$clean"; then
a7e41e
	clean
a7e41e
	safe_exit 0
a7e41e
fi
a7e41e
a7e41e
if test -n "$remove"; then
a7e41e
	kill_alsaloop
a7e41e
	remove
a7e41e
	clean
a7e41e
	safe_exit 0
a7e41e
fi
a7e41e
a7e41e
kill_alsaloop
a7e41e
if test -d /etc/systemd/system ; then
a7e41e
  create_systemd_service
a7e41e
else
a7e41e
  create_initd
a7e41e
fi
a7e41e
modify_modprobe_conf
a7e41e
generate_alsaloop_conf
a7e41e
if test -n "$pa"; then
a7e41e
  modify_pa_conf
a7e41e
fi
a7e41e
restart_alsaloop
a7e41e
a7e41e
clean
a7e41e
a7e41e
safe_exit 0