2016-02-19 14:11:08 -07:00
|
|
|
#!/usr/bin/env bash
|
|
|
|
|
|
|
|
set -e
|
|
|
|
|
|
|
|
rootdir=$(readlink -f $(dirname $0))/..
|
|
|
|
|
|
|
|
function linux_iter_pci {
|
|
|
|
# Argument is the class code
|
|
|
|
# TODO: More specifically match against only class codes in the grep
|
|
|
|
# step.
|
|
|
|
lspci -mm -n | grep $1 | tr -d '"' | awk -F " " '{print "0000:"$1}'
|
|
|
|
}
|
|
|
|
|
2016-04-14 12:21:32 -07:00
|
|
|
function linux_bind_driver() {
|
|
|
|
bdf="$1"
|
|
|
|
driver_name="$2"
|
|
|
|
old_driver_name="no driver"
|
|
|
|
ven_dev_id=$(lspci -n -s $bdf | cut -d' ' -f3 | sed 's/:/ /')
|
|
|
|
|
|
|
|
if [ -e "/sys/bus/pci/devices/$bdf/driver" ]; then
|
|
|
|
old_driver_name=$(basename $(readlink /sys/bus/pci/devices/$bdf/driver))
|
|
|
|
|
|
|
|
if [ "$driver_name" = "$old_driver_name" ]; then
|
|
|
|
return 0
|
|
|
|
fi
|
|
|
|
|
|
|
|
echo "$ven_dev_id" > "/sys/bus/pci/devices/$bdf/driver/remove_id" 2> /dev/null || true
|
|
|
|
echo "$bdf" > "/sys/bus/pci/devices/$bdf/driver/unbind"
|
|
|
|
fi
|
|
|
|
|
|
|
|
echo "$bdf ($ven_dev_id): $old_driver_name -> $driver_name"
|
|
|
|
|
|
|
|
echo "$ven_dev_id" > "/sys/bus/pci/drivers/$driver_name/new_id" 2> /dev/null || true
|
|
|
|
echo "$bdf" > "/sys/bus/pci/drivers/$driver_name/bind" 2> /dev/null || true
|
2016-04-06 11:03:28 +08:00
|
|
|
|
|
|
|
iommu_group=$(basename $(readlink -f /sys/bus/pci/devices/$bdf/iommu_group))
|
|
|
|
if [ -e "/dev/vfio/$iommu_group" ]; then
|
|
|
|
chown "$username" "/dev/vfio/$iommu_group"
|
|
|
|
fi
|
2016-04-14 12:21:32 -07:00
|
|
|
}
|
|
|
|
|
2016-02-19 14:11:08 -07:00
|
|
|
function configure_linux {
|
|
|
|
driver_name=vfio-pci
|
|
|
|
if [ -z "$(ls /sys/kernel/iommu_groups)" ]; then
|
|
|
|
# No IOMMU. Use uio.
|
|
|
|
driver_name=uio_pci_generic
|
|
|
|
fi
|
|
|
|
|
|
|
|
# NVMe
|
|
|
|
modprobe $driver_name || true
|
|
|
|
for bdf in $(linux_iter_pci 0108); do
|
2016-04-14 12:21:32 -07:00
|
|
|
linux_bind_driver "$bdf" "$driver_name"
|
2016-02-19 14:11:08 -07:00
|
|
|
done
|
|
|
|
|
|
|
|
|
|
|
|
# IOAT
|
|
|
|
TMP=`mktemp`
|
|
|
|
#collect all the device_id info of ioat devices.
|
|
|
|
grep "PCI_DEVICE_ID_INTEL_IOAT" $rootdir/lib/ioat/ioat_pci.h \
|
|
|
|
| awk -F"x" '{print $2}' > $TMP
|
|
|
|
|
|
|
|
for dev_id in `cat $TMP`; do
|
|
|
|
# Abuse linux_iter_pci by giving it a device ID instead of a class code
|
|
|
|
for bdf in $(linux_iter_pci $dev_id); do
|
2016-04-14 12:21:32 -07:00
|
|
|
linux_bind_driver "$bdf" "$driver_name"
|
2016-02-19 14:11:08 -07:00
|
|
|
done
|
|
|
|
done
|
|
|
|
rm $TMP
|
|
|
|
|
|
|
|
echo "1" > "/sys/bus/pci/rescan"
|
2016-04-14 13:22:11 -07:00
|
|
|
|
|
|
|
if ! mount | grep -q hugetlbs; then
|
|
|
|
mkdir -p /mnt/huge
|
|
|
|
mount -t hugetlbfs nodev /mnt/huge
|
|
|
|
fi
|
|
|
|
echo "$NRHUGE" > /proc/sys/vm/nr_hugepages
|
2016-04-06 11:03:28 +08:00
|
|
|
|
|
|
|
if [ "$driver_name" = "vfio-pci" ]; then
|
|
|
|
chown "$username" /dev/hugepages
|
|
|
|
|
|
|
|
MEMLOCK_AMNT=`ulimit -l`
|
|
|
|
if [ "$MEMLOCK_AMNT" != "unlimited" ] ; then
|
|
|
|
MEMLOCK_MB=`expr $MEMLOCK_AMNT / 1024`
|
|
|
|
echo ""
|
|
|
|
echo "Current user memlock limit: ${MEMLOCK_MB} MB"
|
|
|
|
echo ""
|
|
|
|
echo "This is the maximum amount of memory you will be"
|
|
|
|
echo "able to use with DPDK and VFIO if run as current user."
|
|
|
|
echo -n "To change this, please adjust limits.conf memlock "
|
|
|
|
echo "limit for current user."
|
|
|
|
|
|
|
|
if [ $MEMLOCK_AMNT -lt 65536 ] ; then
|
|
|
|
echo ""
|
|
|
|
echo "## WARNING: memlock limit is less than 64MB"
|
|
|
|
echo -n "## DPDK with VFIO may not be able to initialize "
|
|
|
|
echo "if run as current user."
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
fi
|
2016-02-19 14:11:08 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
function reset_linux {
|
|
|
|
# NVMe
|
|
|
|
modprobe nvme || true
|
|
|
|
for bdf in $(linux_iter_pci 0108); do
|
2016-04-14 12:21:32 -07:00
|
|
|
linux_bind_driver "$bdf" nvme
|
2016-02-19 14:11:08 -07:00
|
|
|
done
|
|
|
|
|
|
|
|
|
|
|
|
# IOAT
|
|
|
|
TMP=`mktemp`
|
|
|
|
#collect all the device_id info of ioat devices.
|
|
|
|
grep "PCI_DEVICE_ID_INTEL_IOAT" $rootdir/lib/ioat/ioat_pci.h \
|
|
|
|
| awk -F"x" '{print $2}' > $TMP
|
|
|
|
|
|
|
|
modprobe ioatdma || true
|
|
|
|
for dev_id in `cat $TMP`; do
|
|
|
|
# Abuse linux_iter_pci by giving it a device ID instead of a class code
|
|
|
|
for bdf in $(linux_iter_pci $dev_id); do
|
2016-04-14 12:21:32 -07:00
|
|
|
linux_bind_driver "$bdf" ioatdma
|
2016-02-19 14:11:08 -07:00
|
|
|
done
|
|
|
|
done
|
|
|
|
rm $TMP
|
|
|
|
|
|
|
|
echo "1" > "/sys/bus/pci/rescan"
|
|
|
|
}
|
|
|
|
|
|
|
|
function configure_freebsd {
|
|
|
|
TMP=`mktemp`
|
|
|
|
AWK_PROG="{if (count > 0) printf \",\"; printf \"%s:%s:%s\",\$2,\$3,\$4; count++}"
|
|
|
|
echo $AWK_PROG > $TMP
|
|
|
|
PCICONF=`pciconf -l | grep 'class=0x010802\|^ioat'`
|
|
|
|
BDFS=`echo $PCICONF | awk -F: -f $TMP`
|
|
|
|
kldunload nic_uio.ko || true
|
|
|
|
kenv hw.nic_uio.bdfs=$BDFS
|
|
|
|
kldload nic_uio.ko
|
|
|
|
rm $TMP
|
2016-04-14 13:22:11 -07:00
|
|
|
|
|
|
|
kldunload contigmem.ko || true
|
2016-05-25 15:29:11 -07:00
|
|
|
kenv hw.contigmem.num_buffers=$((NRHUGE * 2 / 64))
|
|
|
|
kenv hw.contigmem.buffer_size=$((64 * 1024 * 1024))
|
2016-04-14 13:22:11 -07:00
|
|
|
kldload contigmem.ko
|
2016-02-19 14:11:08 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
function reset_freebsd {
|
|
|
|
kldunload contigmem.ko || true
|
|
|
|
kldunload nic_uio.ko || true
|
|
|
|
}
|
|
|
|
|
2016-04-14 13:22:11 -07:00
|
|
|
NRHUGE=1024
|
|
|
|
|
2016-04-06 11:03:28 +08:00
|
|
|
username=$1
|
|
|
|
mode=$2
|
|
|
|
|
|
|
|
if [ "$username" = "reset" -o "$username" = "config" ]; then
|
|
|
|
mode="$username"
|
|
|
|
username=""
|
|
|
|
fi
|
|
|
|
|
2016-02-19 14:11:08 -07:00
|
|
|
if [ "$mode" == "" ]; then
|
|
|
|
mode="config"
|
|
|
|
fi
|
|
|
|
|
2016-04-06 11:03:28 +08:00
|
|
|
if [ "$username" = "" ]; then
|
|
|
|
username=`logname`
|
|
|
|
fi
|
|
|
|
|
2016-02-19 14:11:08 -07:00
|
|
|
if [ `uname` = Linux ]; then
|
|
|
|
if [ "$mode" == "config" ]; then
|
|
|
|
configure_linux
|
|
|
|
elif [ "$mode" == "reset" ]; then
|
|
|
|
reset_linux
|
|
|
|
fi
|
|
|
|
else
|
|
|
|
if [ "$mode" == "config" ]; then
|
|
|
|
configure_freebsd
|
|
|
|
elif [ "$mode" == "reset" ]; then
|
|
|
|
reset_freebsd
|
|
|
|
fi
|
|
|
|
fi
|