From 8674d0599fbc5bd9cc85a0fc4f36bbf426ccdfd8 Mon Sep 17 00:00:00 2001 From: Vasiliy Stelmachenok <92667539+ventureoo@users.noreply.github.com> Date: Fri, 4 Oct 2024 20:18:52 +0000 Subject: [PATCH] Fix NVIDIA GPUs detection (#139) * profiles/graphic_drivers: Remove PRIME profiles in favor of detection within hooks Signed-off-by: Vasiliy Stelmachenok * Add dumped PCI IDs for NVIDIA GPUs Signed-off-by: Vasiliy Stelmachenok * profiles/graphic_drivers: Use dumped PCI IDs for older NVIDIA profiles instead of device name patterns Guarantees greater accuracy in profile selection Signed-off-by: Vasiliy Stelmachenok * scripts/nvidia-pci-ids-dumper: Refactor error handling Signed-off-by: Vasiliy Stelmachenok * scripts/nvidia-pci-ids-dumper: Support dumping PCI ids for closed modules Signed-off-by: Vasiliy Stelmachenok * Do PCI ids sorting and add dump for closed modules Since dumper adds the PCI ID as a key to the hash table to avoid duplicates, we must sort the ID lists so that they are reproducible and also for easy check against hwdata. Signed-off-by: Vasiliy Stelmachenok * profiles/graphic_drivers: Use dumped list for closed modules instead of pattern matching Signed-off-by: Vasiliy Stelmachenok * scripts/nvidia-pci-ids-dumper: Remove redundant line Signed-off-by: Vasiliy Stelmachenok * scripts/nvidia-pci-ids-dumper: Rename err to die function Signed-off-by: Vasiliy Stelmachenok --------- Signed-off-by: Vasiliy Stelmachenok --- ids/nouveau.ids | 1 + ids/nvidia-390.ids | 1 + ids/nvidia-470.ids | 1 + ids/nvidia-latest-closed.ids | 1 + profiles/pci/graphic_drivers/profiles.toml | 130 ++++++++---------- .../nvidia-pci-ids-dumper | 120 ++++++++++++++++ 6 files changed, 184 insertions(+), 70 deletions(-) create mode 100644 ids/nouveau.ids create mode 100644 ids/nvidia-390.ids create mode 100644 ids/nvidia-470.ids create mode 100644 ids/nvidia-latest-closed.ids create mode 100755 scripts/nvidia-pci-ids-dumper/nvidia-pci-ids-dumper diff --git a/ids/nouveau.ids b/ids/nouveau.ids new file mode 100644 index 0000000..f5b104e --- /dev/null +++ b/ids/nouveau.ids @@ -0,0 +1 @@ +0020 0028 0029 002c 002d 0040 0041 0042 0043 0044 0045 0046 0047 0048 004e 0090 0091 0092 0093 0095 0098 0099 009d 00a0 00c0 00c1 00c2 00c3 00c8 00c9 00cc 00cd 00ce 00f1 00f2 00f3 00f4 00f5 00f6 00f8 00f9 00fa 00fb 00fc 00fd 00fe 0100 0101 0103 0110 0111 0112 0113 0140 0141 0142 0143 0144 0145 0146 0147 0148 0149 014a 014c 014d 014e 014f 0150 0151 0152 0153 0160 0161 0162 0163 0164 0165 0166 0167 0168 0169 016a 0170 0171 0172 0173 0174 0175 0176 0177 0178 0179 017a 017c 017d 0181 0182 0183 0185 0188 018a 018b 018c 0191 0193 0194 0197 019d 019e 01a0 01d0 01d1 01d2 01d3 01d6 01d7 01d8 01da 01db 01dc 01dd 01de 01df 01f0 0200 0201 0202 0203 0211 0212 0215 0218 0221 0222 0240 0241 0242 0244 0245 0247 0250 0251 0253 0258 0259 025b 0280 0281 0282 0286 0288 0289 028c 0290 0291 0292 0293 0294 0295 0297 0298 0299 029a 029b 029c 029d 029e 029f 02e0 02e1 02e2 02e3 02e4 0301 0302 0308 0309 0311 0312 0314 031a 031b 031c 0320 0321 0322 0323 0324 0325 0326 0327 0328 032a 032b 032c 032d 0330 0331 0332 0333 0334 0338 033f 0341 0342 0343 0344 0347 0348 034c 034e 038b 0390 0391 0392 0393 0394 0395 0397 0398 0399 039c 039e 03d0 03d1 03d2 03d5 03d6 0400 0401 0402 0403 0404 0405 0406 0407 0408 0409 040a 040b 040c 040d 040e 040f 0410 0420 0421 0422 0423 0424 0425 0426 0427 0428 0429 042a 042b 042c 042d 042e 042f 0531 0533 053a 053b 053e 05e0 05e1 05e2 05e3 05e6 05e7 05ea 05eb 05ed 05f8 05f9 05fd 05fe 05ff 0600 0601 0602 0603 0604 0605 0606 0607 0608 0609 060a 060b 060c 060d 060f 0610 0611 0612 0613 0614 0615 0617 0618 0619 061a 061b 061c 061d 061e 061f 0621 0622 0623 0625 0626 0627 0628 062a 062b 062c 062d 062e 0630 0631 0632 0635 0637 0638 063a 0640 0641 0643 0644 0645 0646 0647 0648 0649 064a 064b 064c 0651 0652 0653 0654 0655 0656 0658 0659 065a 065b 065c 06e0 06e1 06e2 06e3 06e4 06e5 06e6 06e7 06e8 06e9 06ea 06eb 06ec 06ef 06f1 06f8 06f9 06fa 06fb 06fd 06ff 07e0 07e1 07e2 07e3 07e5 0840 0844 0845 0846 0847 0848 0849 084a 084b 084c 084d 084f 0860 0861 0862 0863 0864 0865 0866 0867 0868 0869 086a 086c 086d 086e 086f 0870 0871 0872 0873 0874 0876 087a 087d 087e 087f 08a0 08a2 08a3 08a4 08a5 0a20 0a22 0a23 0a26 0a27 0a28 0a29 0a2a 0a2b 0a2c 0a2d 0a32 0a34 0a35 0a38 0a3c 0a60 0a62 0a63 0a64 0a65 0a66 0a67 0a68 0a69 0a6a 0a6c 0a6e 0a6f 0a70 0a71 0a72 0a73 0a74 0a75 0a76 0a78 0a7a 0a7c 0ca0 0ca2 0ca3 0ca4 0ca5 0ca7 0ca8 0ca9 0cac 0caf 0cb0 0cb1 0cbc 0fef 0ff2 10c0 10c3 10c5 10d8 11bf diff --git a/ids/nvidia-390.ids b/ids/nvidia-390.ids new file mode 100644 index 0000000..6dbda0f --- /dev/null +++ b/ids/nvidia-390.ids @@ -0,0 +1 @@ +06c0 06c4 06ca 06cd 06d1 06d2 06d8 06d9 06da 06dc 06dd 06de 06df 0dc0 0dc4 0dc5 0dc6 0dcd 0dce 0dd1 0dd2 0dd3 0dd6 0dd8 0dda 0de0 0de1 0de2 0de3 0de4 0de5 0de7 0de8 0de9 0dea 0deb 0dec 0ded 0dee 0def 0df0 0df1 0df2 0df3 0df4 0df5 0df6 0df7 0df8 0df9 0dfa 0dfc 0e22 0e23 0e24 0e30 0e31 0e3a 0e3b 0f00 0f01 0f02 0f03 1040 1042 1048 1049 104a 104b 104c 1050 1051 1052 1054 1055 1056 1057 1058 1059 105a 105b 107c 107d 1080 1081 1082 1084 1086 1087 1088 1089 108b 1091 1094 1096 109a 109b 1140 1200 1201 1203 1205 1206 1207 1208 1210 1211 1212 1213 1241 1243 1244 1245 1246 1247 1248 1249 124b 124d 1251 diff --git a/ids/nvidia-470.ids b/ids/nvidia-470.ids new file mode 100644 index 0000000..32d32f8 --- /dev/null +++ b/ids/nvidia-470.ids @@ -0,0 +1 @@ +0fc6 0fc8 0fc9 0fcd 0fce 0fd1 0fd2 0fd3 0fd4 0fd5 0fd8 0fd9 0fdf 0fe0 0fe1 0fe2 0fe3 0fe4 0fe9 0fea 0fec 0fed 0fee 0ff6 0ff8 0ff9 0ffa 0ffb 0ffc 0ffd 0ffe 0fff 1001 1004 1005 1007 1008 100a 100c 1021 1022 1023 1024 1026 1027 1028 1029 102a 102d 103a 103c 1180 1183 1184 1185 1187 1188 1189 118a 118e 118f 1193 1194 1195 1198 1199 119a 119d 119e 119f 11a0 11a1 11a2 11a3 11a7 11b4 11b6 11b7 11b8 11ba 11bc 11bd 11be 11c0 11c2 11c3 11c4 11c5 11c6 11c8 11cb 11e0 11e1 11e2 11e3 11fa 11fc 1280 1281 1282 1284 1286 1287 1288 1289 128b 1290 1291 1292 1293 1295 1296 1298 1299 129a 12b9 12ba diff --git a/ids/nvidia-latest-closed.ids b/ids/nvidia-latest-closed.ids new file mode 100644 index 0000000..e991b0a --- /dev/null +++ b/ids/nvidia-latest-closed.ids @@ -0,0 +1 @@ +1340 1341 1344 1346 1347 1348 1349 134b 134d 134e 134f 137a 137b 1380 1381 1382 1390 1391 1392 1393 1398 1399 139a 139b 139c 139d 13b0 13b1 13b2 13b3 13b4 13b6 13b9 13ba 13bb 13bc 13c0 13c2 13d7 13d8 13d9 13da 13f0 13f1 13f2 13f3 13f8 13f9 13fa 13fb 1401 1402 1406 1407 1427 1430 1431 1436 15f0 15f7 15f8 15f9 1617 1618 1619 161a 1667 174d 174e 179c 17c2 17c8 17f0 17f1 17fd 1b00 1b02 1b06 1b30 1b38 1b80 1b81 1b82 1b83 1b84 1b87 1ba0 1ba1 1ba2 1bb0 1bb1 1bb4 1bb5 1bb6 1bb7 1bb8 1bb9 1bbb 1bc7 1be0 1be1 1c02 1c03 1c04 1c06 1c07 1c09 1c20 1c21 1c22 1c23 1c30 1c31 1c60 1c61 1c62 1c81 1c82 1c83 1c8c 1c8d 1c8f 1c90 1c91 1c92 1c94 1c96 1cb1 1cb2 1cb3 1cb6 1cba 1cbb 1cbc 1cbd 1cfa 1cfb 1d01 1d02 1d10 1d11 1d12 1d13 1d16 1d33 1d34 1d52 1d81 1db1 1db3 1db4 1db5 1db6 1db7 1db8 1dba 1df0 1df2 1df6 1e09 1f0b 2189 diff --git a/profiles/pci/graphic_drivers/profiles.toml b/profiles/pci/graphic_drivers/profiles.toml index ef59f60..c4e85dc 100644 --- a/profiles/pci/graphic_drivers/profiles.toml +++ b/profiles/pci/graphic_drivers/profiles.toml @@ -20,24 +20,31 @@ desc = 'Open source NVIDIA drivers for Linux (Latest)' class_ids = "0300 0302 0380" vendor_ids = "10de" +device_ids = "*" priority = 10 packages = 'nvidia-utils egl-wayland nvidia-settings opencl-nvidia lib32-opencl-nvidia lib32-nvidia-utils libva-nvidia-driver vulkan-icd-loader lib32-vulkan-icd-loader' conditional_packages = """ kernels="$(pacman -Qqs "^linux-cachyos")" - modules="" + packages="" for kernel in $kernels; do case "$kernel" in *-nvidia-open) modules+=" ${kernel}";; *-headers|*-zfs|*-nvidia|*-dbg);; - *) modules+=" ${kernel}-nvidia-open";; + *) packages+=" ${kernel}-nvidia-open";; esac done # Fallback if there are no kernels with pre-built modules - [ -z "$modules" ] && modules="nvidia-open-dkms" + [ -z "$packages" ] && packages="nvidia-open-dkms" + + # Trying to determine the laptop + device_type="$(cat /sys/devices/virtual/dmi/id/chassis_type)" + if ((device_type >= 8 && device_type <= 11)); then + packages+=" nvidia-prime switcheroo-control" + fi - echo "$modules" + echo "$packages" """ post_install = """ @@ -47,35 +54,20 @@ MODULES+=(nvidia nvidia_modeset nvidia_uvm nvidia_drm) EOF mkinitcpio -P - # Add libva-nvidia-driver to profile - echo "export LIBVA_DRIVER_NAME=nvidia" > /etc/profile.d/nvidia-vaapi.sh + # Trying to determine the laptop + device_type="$(cat /sys/devices/virtual/dmi/id/chassis_type)" + if ((device_type >= 8 && device_type <= 11)); then + systemctl enable switcheroo-control + else + # Add libva-nvidia-driver to profile + echo "export LIBVA_DRIVER_NAME=nvidia" > /etc/profile.d/nvidia-vaapi.sh + fi """ post_remove = """ rm -f /etc/mkinitcpio.conf.d/10-chwd.conf rm -f /etc/profile.d/nvidia-vaapi.sh mkinitcpio -P """ -device_name_pattern = '([A-Z]+[0-9]+[^M]*)\s.*' - -[nvidia-open-dkms.prime] -desc = 'Open source NVIDIA drivers for Linux (Latest)' -class_ids = "0300 0302 0380" -device_name_pattern = '([A-Z]+[0-9]+[A-Z]*M)\s.*' -priority = 11 -packages = 'nvidia-utils egl-wayland nvidia-settings opencl-nvidia lib32-opencl-nvidia lib32-nvidia-utils vulkan-icd-loader lib32-vulkan-icd-loader nvidia-prime switcheroo-control' -post_install = """ - cat </etc/mkinitcpio.conf.d/10-chwd.conf -# This file is automatically generated by chwd. PLEASE DO NOT EDIT IT. -MODULES+=(nvidia nvidia_modeset nvidia_uvm nvidia_drm) -EOF - mkinitcpio -P - systemctl enable switcheroo-control -""" -post_remove = """ - rm -f /etc/mkinitcpio.conf.d/10-chwd.conf - mkinitcpio -P -""" - [nvidia-dkms] desc = 'Closed source NVIDIA drivers for Linux (Latest)' @@ -83,22 +75,30 @@ class_ids = "0300 0302 0380" vendor_ids = "10de" priority = 12 packages = 'nvidia-utils egl-wayland nvidia-settings opencl-nvidia lib32-opencl-nvidia lib32-nvidia-utils libva-nvidia-driver vulkan-icd-loader lib32-vulkan-icd-loader' +device_ids = '>/var/lib/chwd/ids/nvidia-latest-closed.ids' conditional_packages = """ kernels="$(pacman -Qqs "^linux-cachyos")" - modules="" + packages="" for kernel in $kernels; do case "$kernel" in *-headers|*-zfs|*-nvidia-open|*-dbg);; - *-nvidia) modules+=" ${kernel}";; - *) modules+=" ${kernel}-nvidia";; + *-nvidia) packages+=" ${kernel}";; + *) packages+=" ${kernel}-nvidia";; esac done # Fallback if there are no kernels with pre-built modules - [ -z "$modules" ] && modules="nvidia-dkms" + [ -z "$packages" ] && packages="nvidia-dkms" + + + # Trying to determine the laptop + device_type="$(cat /sys/devices/virtual/dmi/id/chassis_type)" + if [[ "$device_type" -ge 8 && "$device_type" -le 11 ]]; then + packages+=" nvidia-prime switcheroo-control" + fi - echo "$modules" + echo "$packages" """ post_install = """ @@ -108,73 +108,63 @@ MODULES+=(nvidia nvidia_modeset nvidia_uvm nvidia_drm) EOF mkinitcpio -P - # Add libva-nvidia-driver to profile - echo "export LIBVA_DRIVER_NAME=nvidia" > /etc/profile.d/nvidia-vaapi.sh + # Trying to determine the laptop + device_type="$(cat /sys/devices/virtual/dmi/id/chassis_type)" + if ((device_type >= 8 && device_type <= 11)); then + systemctl enable switcheroo-control + else + # Add libva-nvidia-driver to profile + echo "export LIBVA_DRIVER_NAME=nvidia" > /etc/profile.d/nvidia-vaapi.sh + fi """ post_remove = """ rm -f /etc/mkinitcpio.conf.d/10-chwd.conf rm -f /etc/profile.d/nvidia-vaapi.sh mkinitcpio -P """ -device_name_pattern = '((GM|GP)+[0-9]+[^M]*\s.*)' - -[nvidia-dkms.prime] -desc = 'Closed source NVIDIA drivers for Linux (Latest)' -class_ids = "0300 0302 0380" -device_name_pattern = '((GM|GP)+[0-9]+[A-Z]*M\s.*)' -priority = 13 -packages = 'nvidia-utils egl-wayland nvidia-settings opencl-nvidia lib32-opencl-nvidia lib32-nvidia-utils vulkan-icd-loader lib32-vulkan-icd-loader nvidia-prime switcheroo-control' -post_install = """ - cat </etc/mkinitcpio.conf.d/10-chwd.conf -# This file is automatically generated by chwd. PLEASE DO NOT EDIT IT. -MODULES+=(nvidia nvidia_modeset nvidia_uvm nvidia_drm) -EOF - mkinitcpio -P - systemctl enable switcheroo-control -""" -post_remove = """ - rm -f /etc/mkinitcpio.conf.d/10-chwd.conf - mkinitcpio -P -""" [nvidia-dkms-470xx] desc = 'Closed source NVIDIA drivers for Linux (470xx branch, only for Kepler GPUs)' priority = 14 -class_ids = "0300 0380" +class_ids = "0300 0302 0380" vendor_ids = "10de" +device_ids = '>/var/lib/chwd/ids/nvidia-470.ids' packages = 'nvidia-470xx-dkms nvidia-470xx-utils nvidia-470xx-settings opencl-nvidia-470xx vulkan-icd-loader lib32-nvidia-470xx-utils lib32-opencl-nvidia-470xx lib32-vulkan-icd-loader libva-nvidia-driver' -device_name_pattern = '(GK)\w+' +conditional_packages = """ + # Trying to determine the laptop + device_type="$(cat /sys/devices/virtual/dmi/id/chassis_type)" + if ((device_type >= 8 && device_type <= 11)); then + echo "nvidia-prime switcheroo-control" + fi +""" post_install = """ - # Add libva-nvidia-driver to profile - echo "export LIBVA_DRIVER_NAME=nvidia" > /etc/profile.d/nvidia-vaapi.sh + # Trying to determine the laptop + device_type="$(cat /sys/devices/virtual/dmi/id/chassis_type)" + if ((device_type >= 8 && device_type <= 11)); then + systemctl enable switcheroo-control + else + # Add libva-nvidia-driver to profile + echo "export LIBVA_DRIVER_NAME=nvidia" > /etc/profile.d/nvidia-vaapi.sh + fi """ post_remove = """ rm -f /etc/profile.d/nvidia-vaapi.sh """ -[nvidia-dkms-470xx.prime] -desc = 'Closed source NVIDIA drivers for Linux (470xx branch, only for Kepler GPUs)' -priority = 15 -class_ids = "0302" -packages = 'nvidia-470xx-dkms nvidia-470xx-utils nvidia-470xx-settings opencl-nvidia-470xx vulkan-icd-loader lib32-nvidia-470xx-utils lib32-opencl-nvidia-470xx lib32-vulkan-icd-loader switcheroo-control nvidia-prime' -device_name_pattern = '(GK)\w+' -post_install = """ - systemctl enable switcheroo-control -""" - [nvidia-dkms-390xx] desc = 'Closed source NVIDIA drivers for Linux (390xx branch, only for Fermi GPUs)' priority = 16 vendor_ids = "10de" class_ids = "0300 0302 0380" packages = 'nvidia-390xx-dkms nvidia-390xx-utils nvidia-390xx-settings opencl-nvidia-390xx nvidia-390xx-utils nvidia-390xx-settings opencl-nvidia-390xx lib32-nvidia-390xx-utils lib32-opencl-nvidia-390xx' -device_name_pattern = '(GF)\w+' +device_ids = '>/var/lib/chwd/ids/nvidia-390.ids' [nouveau] desc = "Open Nouveau driver for NVIDIA graphics cards (Only for old GPUs)" class_ids = "0300 0302" vendor_ids = "10de" -priority = 9 +device_ids = '>/var/lib/chwd/ids/nouveau.ids' +priority = 18 packages = 'mesa lib32-mesa libva-mesa-driver mesa-vdpau lib32-libva-mesa-driver vulkan-nouveau' conditional_packages = """ if [ -z "$(lspci -d "10de:*:030x")" ]; then diff --git a/scripts/nvidia-pci-ids-dumper/nvidia-pci-ids-dumper b/scripts/nvidia-pci-ids-dumper/nvidia-pci-ids-dumper new file mode 100755 index 0000000..502c824 --- /dev/null +++ b/scripts/nvidia-pci-ids-dumper/nvidia-pci-ids-dumper @@ -0,0 +1,120 @@ +#!/usr/bin/env lua +--[[ +Copyright (C) 2024 Vasiliy Stelmachenok for CachyOS + +This program is free software; you can redistribute it and/or modify +it under the terms of the GNU General Public License as published by +the Free Software Foundation; either version 2 of the License, or +(at your option) any later version. + +This program is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +You should have received a copy of the GNU General Public License along +with this program; if not, write to the Free Software Foundation, Inc., +51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. +--]] +local found, json = pcall(require, 'dkjson') + +local function die(msg, ...) + print(msg:format(...)) + os.exit(1) +end + +if not found then + die("lua-dkjson dependency is missing, please install it from repositories") +end + +local function write(generations) + for branch, ids in pairs(generations) do + local fname = branch .. ".ids" + local devices = {} + + for id in pairs(ids) do + devices[#devices+1] = id + end + + devices = table.sort(devices) or devices + + local hwdb, errmsg = io.open(fname, "w+") + if hwdb then + local content = table.concat(devices, " ") + hwdb:write(content .. "\n") + hwdb:close() + else + die("Failed to write PCI ids: %s", errmsg) + end + end +end + +local function parse(chips) + local drivers = { + ["nvidia-390"] = {}, + ["nvidia-470"] = {}, + ['nvidia-latest-closed'] = {}, + ["nouveau"] = {} + } + for _, chip in ipairs(chips) do + local id = chip.devid:gsub("0x", ""):lower() + if chip.legacybranch then + local branch = chip.legacybranch + if branch == "470.xx" then + drivers["nvidia-470"][id] = true + elseif branch == "390.xx" then + drivers["nvidia-390"][id] = true + else + drivers["nouveau"][id] = true + end + else + local features = chip.features + local kernelOpen = false + + for _, feature in ipairs(features) do + if feature == "kernelopen" then + kernelOpen = true + break + end + end + + if not kernelOpen then + drivers['nvidia-latest-closed'][id] = true + end + end + end + + return drivers +end + +local function main() + if #arg < 1 then + die("Specify path to the supported-gpu.json file") + end + + local path = arg[1] + local file, errmsg = io.open(path, "r") + + if not file then + die("Failed to open %s file: %s", path, errmsg) + return + end + + local raw = file:read() + file:close() + + if not raw then + die("Failed to read file %s", path) + end + + local supported_gpus = json.decode(raw) + + if not supported_gpus.chips then + die("Failed to parse supported_gpus.json") + end + + local generations = parse(supported_gpus.chips) + write(generations) +end + +main()