199 lines
6.4 KiB
Nix
199 lines
6.4 KiB
Nix
{ config, pkgs, materusArg, ... }:
|
|
let
|
|
bar0_guest="15";
|
|
bar2_guest="8";
|
|
bar0_host="15";
|
|
bar2_host="8";
|
|
|
|
VM_UUID = "ad2632db-0da0-4204-98b3-0592a185ebd0";
|
|
|
|
startedHook = ''
|
|
QEMU_PID=$(ps aux | grep qemu-system-x86_64 | grep "${VM_UUID}" | tr -s ' ' | cut -d " " -f 2)
|
|
|
|
for pid in $(cat /sys/fs/cgroup/cpu/machine.slice/machine-qemu*$1.scope/libvirt/vcpu*/tasks); do
|
|
renice -n "-15" -p "$pid";
|
|
done
|
|
renice -n "-10" -p "$QEMU_PID";
|
|
|
|
echo "${materusArg.materusPC.hostCoresMask}" > /proc/irq/default_smp_affinity
|
|
for irq in /proc/irq/[0-9]*/smp_affinity; do
|
|
if [ $(cat $irq) = "${materusArg.materusPC.allCoresMask}" ]; then
|
|
echo "${materusArg.materusPC.hostCoresMask}" > $irq 2> /dev/null
|
|
fi;
|
|
done;
|
|
for irq in $(cat /proc/interrupts | grep vfio | cut -d ":" -f 1); do
|
|
echo "${materusArg.materusPC.vmCoresMask}" > /proc/irq/$irq/smp_affinity;
|
|
done
|
|
|
|
|
|
'';
|
|
startHook = /*''
|
|
|
|
|
|
# Debugging
|
|
exec 19>/home/materus/startlogfile
|
|
BASH_XTRACEFD=19
|
|
set -x
|
|
|
|
exec 3>&1 4>&2
|
|
trap 'exec 2>&4 1>&3' 0 1 2 3
|
|
exec 1>/home/materus/startlogfile.out 2>&1
|
|
''
|
|
+*/
|
|
''
|
|
systemctl stop windows-share-mount.service
|
|
|
|
# Make sure nothing renders on gpu to prevent "sysfs: cannot create duplicate filename" after rebinding to amdgpu
|
|
chmod 0 /dev/dri/by-path/pci-$VIRSH_GPU_VIDEO-render
|
|
chmod 0 /dev/dri/by-path/pci-$VIRSH_GPU_VIDEO-card
|
|
fuser -k /dev/dri/by-path/pci-$VIRSH_GPU_VIDEO-render
|
|
pkill Xwayland
|
|
|
|
# Seems to fix reset bug for 7900 XTX
|
|
echo "0" > "/sys/bus/pci/devices/''${VIRSH_GPU_VIDEO}/d3cold_allowed"
|
|
|
|
#####################################################################
|
|
# Weird bug on kernel 6.7+, after changing bar sizes and binding to vfio driver, performance after returning to host will be lower than expected
|
|
# binding to amdgpu after changing bar sizes and binding after it to vfio will work as expected.
|
|
# I could skip changing bar sizes since I'm able to use full bar, but keeping it just in case
|
|
echo ''$VIRSH_GPU_VIDEO > "/sys/bus/pci/devices/''${VIRSH_GPU_VIDEO}/driver/unbind"
|
|
sleep 1s
|
|
echo "${bar0_host}" > "/sys/bus/pci/devices/''${VIRSH_GPU_VIDEO}/resource0_resize"
|
|
echo "${bar2_host}" > "/sys/bus/pci/devices/''${VIRSH_GPU_VIDEO}/resource2_resize"
|
|
|
|
echo ''$VIRSH_GPU_VIDEO > /sys/bus/pci/drivers/amdgpu/bind
|
|
|
|
sleep 1s
|
|
|
|
chmod 0 /dev/dri/renderD128
|
|
fuser -k /dev/dri/renderD128
|
|
#####################################################################
|
|
|
|
echo ''$VIRSH_GPU_VIDEO > "/sys/bus/pci/devices/''${VIRSH_GPU_VIDEO}/driver/unbind"
|
|
echo ''$VIRSH_GPU_AUDIO > "/sys/bus/pci/devices/''${VIRSH_GPU_AUDIO}/driver/unbind"
|
|
|
|
echo "${bar0_guest}" > "/sys/bus/pci/devices/''${VIRSH_GPU_VIDEO}/resource0_resize"
|
|
echo "${bar2_guest}" > "/sys/bus/pci/devices/''${VIRSH_GPU_VIDEO}/resource2_resize"
|
|
|
|
sync
|
|
echo "3" > /proc/sys/vm/drop_caches
|
|
sync
|
|
echo "1" > /proc/sys/vm/compact_memory
|
|
|
|
|
|
|
|
|
|
systemctl set-property --runtime -- user.slice AllowedCPUs=${materusArg.materusPC.hostCores}
|
|
systemctl set-property --runtime -- system.slice AllowedCPUs=${materusArg.materusPC.hostCores}
|
|
systemctl set-property --runtime -- init.scope AllowedCPUs=${materusArg.materusPC.hostCores}
|
|
echo "${materusArg.materusPC.hostCoresMask}" > /sys/bus/workqueue/devices/writeback/cpumask
|
|
echo performance | tee /sys/devices/system/cpu/cpu*/cpufreq/scaling_governor
|
|
|
|
sysctl vm.stat_interval=120
|
|
sysctl -w kernel.watchdog=0
|
|
|
|
|
|
|
|
|
|
|
|
'';
|
|
stopHook = ''
|
|
|
|
# Debugging
|
|
# exec 19>/home/materus/stoplogfile
|
|
# BASH_XTRACEFD=19
|
|
# set -x
|
|
|
|
# exec 3>&1 4>&2
|
|
# trap 'exec 2>&4 1>&3' 0 1 2 3
|
|
# exec 1>/home/materus/stoplogfile.out 2>&1
|
|
echo performance | tee /sys/devices/system/cpu/cpu*/cpufreq/scaling_governor
|
|
|
|
sysctl vm.stat_interval=1
|
|
sysctl -w kernel.watchdog=1
|
|
echo "${materusArg.materusPC.allCoresMask}" > /proc/irq/default_smp_affinity
|
|
for irq in /proc/irq/[0-9]*/smp_affinity; do
|
|
if [ $(cat $irq) = "${materusArg.materusPC.hostCoresMask}" ] || [ $(cat $irq) = "${materusArg.materusPC.vmCoresMask}" ]; then
|
|
echo "${materusArg.materusPC.allCoresMask}" > $irq 2> /dev/null
|
|
fi;
|
|
done;
|
|
|
|
|
|
sleep 1s
|
|
echo ''$VIRSH_GPU_VIDEO > "/sys/bus/pci/devices/''${VIRSH_GPU_VIDEO}/driver/unbind"
|
|
echo ''$VIRSH_GPU_AUDIO > "/sys/bus/pci/devices/''${VIRSH_GPU_AUDIO}/driver/unbind"
|
|
|
|
|
|
|
|
|
|
echo "${bar0_host}" > "/sys/bus/pci/devices/''${VIRSH_GPU_VIDEO}/resource0_resize"
|
|
echo "${bar2_host}" > "/sys/bus/pci/devices/''${VIRSH_GPU_VIDEO}/resource2_resize"
|
|
echo "1" > "/sys/bus/pci/devices/''${VIRSH_GPU_VIDEO}/d3cold_allowed"
|
|
|
|
|
|
echo ''$VIRSH_GPU_VIDEO > /sys/bus/pci/drivers/amdgpu/bind
|
|
echo ''$VIRSH_GPU_AUDIO > /sys/bus/pci/drivers/snd_hda_intel/bind
|
|
|
|
|
|
systemctl start windows-share-mount.service
|
|
|
|
systemctl set-property --runtime -- user.slice AllowedCPUs=${materusArg.materusPC.allCores}
|
|
systemctl set-property --runtime -- system.slice AllowedCPUs=${materusArg.materusPC.allCores}
|
|
systemctl set-property --runtime -- init.scope AllowedCPUs=${materusArg.materusPC.allCores}
|
|
echo "${materusArg.materusPC.allCoresMask}" > /sys/bus/workqueue/devices/writeback/cpumask
|
|
|
|
'';
|
|
in
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
virtualisation.libvirtd.hooks.qemu = {
|
|
"windows-vfio" = pkgs.writeShellScript "windows.sh" ''
|
|
VIRSH_GPU_VIDEO="0000:03:00.0"
|
|
VIRSH_GPU_AUDIO="0000:03:00.1"
|
|
VIRSH_USB1="0000:10:00.0"
|
|
|
|
if [ ''$1 = "windows-vfio" ]; then
|
|
if [ ''$2 = "prepare" ] && [ ''$3 = "begin" ]; then
|
|
${startHook}
|
|
fi
|
|
|
|
if [ ''$2 = "started" ] && [ ''$3 = "begin" ]; then
|
|
${startedHook}
|
|
fi
|
|
|
|
if [ ''$2 = "release" ] && [ ''$3 = "end" ]; then
|
|
${stopHook}
|
|
fi
|
|
|
|
fi
|
|
|
|
|
|
'';
|
|
};
|
|
|
|
systemd.services.windows-share-mount = {
|
|
wantedBy = [ "multi-user.target" ];
|
|
path = [ config.virtualisation.libvirtd.qemu.package pkgs.util-linux pkgs.kmod pkgs.coreutils ];
|
|
serviceConfig.Type = "oneshot";
|
|
serviceConfig.RemainAfterExit = true;
|
|
script = ''
|
|
modprobe nbd max_part=16
|
|
sleep 1
|
|
qemu-nbd -c /dev/nbd10 /materus/data/VM/data.qcow2 --discard=unmap
|
|
sleep 1
|
|
mount /dev/nbd10p1 /materus/data/Windows -o uid=1000,gid=100
|
|
'';
|
|
preStop = ''
|
|
umount -r /dev/nbd10p1
|
|
qemu-nbd -d /dev/nbd10
|
|
'';
|
|
};
|
|
}
|