# GPU acceleration {#sec-gpu-accel} NixOS provides various APIs that benefit from GPU hardware acceleration, such as VA-API and VDPAU for video playback; OpenGL and Vulkan for 3D graphics; and OpenCL for general-purpose computing. This chapter describes how to set up GPU hardware acceleration (as far as this is not done automatically) and how to verify that hardware acceleration is indeed used. Most of the aforementioned APIs are agnostic with regards to which display server is used. Consequently, these instructions should apply both to the X Window System and Wayland compositors. ## OpenCL {#sec-gpu-accel-opencl} [OpenCL](https://en.wikipedia.org/wiki/OpenCL) is a general compute API. It is used by various applications such as Blender and Darktable to accelerate certain operations. OpenCL applications load drivers through the *Installable Client Driver* (ICD) mechanism. In this mechanism, an ICD file specifies the path to the OpenCL driver for a particular GPU family. In NixOS, there are two ways to make ICD files visible to the ICD loader. The first is through the `OCL_ICD_VENDORS` environment variable. This variable can contain a directory which is scanned by the ICL loader for ICD files. For example: ```ShellSession $ export \ OCL_ICD_VENDORS=`nix-build '' --no-out-link -A rocmPackages.clr.icd`/etc/OpenCL/vendors/ ``` The second mechanism is to add the OpenCL driver package to [](#opt-hardware.graphics.extraPackages). This links the ICD file under `/run/opengl-driver`, where it will be visible to the ICD loader. The proper installation of OpenCL drivers can be verified through the `clinfo` command of the clinfo package. This command will report the number of hardware devices that is found and give detailed information for each device: ```ShellSession $ clinfo | head -n3 Number of platforms 1 Platform Name AMD Accelerated Parallel Processing Platform Vendor Advanced Micro Devices, Inc. ``` ### AMD {#sec-gpu-accel-opencl-amd} Modern AMD [Graphics Core Next](https://en.wikipedia.org/wiki/Graphics_Core_Next) (GCN) GPUs are supported through the rocmPackages.clr.icd package. Adding this package to [](#opt-hardware.graphics.extraPackages) enables OpenCL support: ```nix { hardware.graphics.extraPackages = [ rocmPackages.clr.icd ]; } ``` ### Intel {#sec-gpu-accel-opencl-intel} [Intel Gen8 and later GPUs](https://en.wikipedia.org/wiki/List_of_Intel_graphics_processing_units#Gen8) are supported by the Intel NEO OpenCL runtime that is provided by the intel-compute-runtime package. The proprietary Intel OpenCL runtime, in the intel-ocl package, is an alternative for Gen7 GPUs. The intel-compute-runtime or intel-ocl package can be added to [](#opt-hardware.graphics.extraPackages) to enable OpenCL support. For example, for Gen8 and later GPUs, the following configuration can be used: ```nix { hardware.graphics.extraPackages = [ intel-compute-runtime ]; } ``` ## Vulkan {#sec-gpu-accel-vulkan} [Vulkan](https://en.wikipedia.org/wiki/Vulkan_(API)) is a graphics and compute API for GPUs. It is used directly by games or indirectly though compatibility layers like [DXVK](https://github.com/doitsujin/dxvk/wiki). By default, if [](#opt-hardware.graphics.enable) is enabled, Mesa is installed and provides Vulkan for supported hardware. Similar to OpenCL, Vulkan drivers are loaded through the *Installable Client Driver* (ICD) mechanism. ICD files for Vulkan are JSON files that specify the path to the driver library and the supported Vulkan version. All successfully loaded drivers are exposed to the application as different GPUs. In NixOS, there are two ways to make ICD files visible to Vulkan applications: an environment variable and a module option. The first option is through the `VK_ICD_FILENAMES` environment variable. This variable can contain multiple JSON files, separated by `:`. For example: ```ShellSession $ export \ VK_ICD_FILENAMES=`nix-build '' --no-out-link -A amdvlk`/share/vulkan/icd.d/amd_icd64.json ``` The second mechanism is to add the Vulkan driver package to [](#opt-hardware.graphics.extraPackages). This links the ICD file under `/run/opengl-driver`, where it will be visible to the ICD loader. The proper installation of Vulkan drivers can be verified through the `vulkaninfo` command of the vulkan-tools package. This command will report the hardware devices and drivers found, in this example output amdvlk and radv: ```ShellSession $ vulkaninfo | grep GPU GPU id : 0 (Unknown AMD GPU) GPU id : 1 (AMD RADV NAVI10 (LLVM 9.0.1)) ... GPU0: deviceType = PHYSICAL_DEVICE_TYPE_DISCRETE_GPU deviceName = Unknown AMD GPU GPU1: deviceType = PHYSICAL_DEVICE_TYPE_DISCRETE_GPU ``` A simple graphical application that uses Vulkan is `vkcube` from the vulkan-tools package. ### AMD {#sec-gpu-accel-vulkan-amd} Modern AMD [Graphics Core Next](https://en.wikipedia.org/wiki/Graphics_Core_Next) (GCN) GPUs are supported through either radv, which is part of mesa, or the amdvlk package. Adding the amdvlk package to [](#opt-hardware.graphics.extraPackages) makes amdvlk the default driver and hides radv and lavapipe from the device list. A specific driver can be forced as follows: ```nix { hardware.graphics.extraPackages = [ pkgs.amdvlk ]; # To enable Vulkan support for 32-bit applications, also add: hardware.graphics.extraPackages32 = [ pkgs.driversi686Linux.amdvlk ]; # Force radv environment.variables.AMD_VULKAN_ICD = "RADV"; # Or environment.variables.VK_ICD_FILENAMES = "/run/opengl-driver/share/vulkan/icd.d/radeon_icd.x86_64.json"; } ``` ## VA-API {#sec-gpu-accel-va-api} [VA-API (Video Acceleration API)](https://www.intel.com/content/www/us/en/developer/articles/technical/linuxmedia-vaapi.html) is an open-source library and API specification, which provides access to graphics hardware acceleration capabilities for video processing. VA-API drivers are loaded by `libva`. The version in nixpkgs is built to search the opengl driver path, so drivers can be installed in [](#opt-hardware.graphics.extraPackages). VA-API can be tested using: ```ShellSession $ nix-shell -p libva-utils --run vainfo ``` ### Intel {#sec-gpu-accel-va-api-intel} Modern Intel GPUs use the iHD driver, which can be installed with: ```nix { hardware.graphics.extraPackages = [ intel-media-driver ]; } ``` Older Intel GPUs use the i965 driver, which can be installed with: ```nix { hardware.graphics.extraPackages = [ intel-vaapi-driver ]; } ``` ## Common issues {#sec-gpu-accel-common-issues} ### User permissions {#sec-gpu-accel-common-issues-permissions} Except where noted explicitly, it should not be necessary to adjust user permissions to use these acceleration APIs. In the default configuration, GPU devices have world-read/write permissions (`/dev/dri/renderD*`) or are tagged as `uaccess` (`/dev/dri/card*`). The access control lists of devices with the `uaccess` tag will be updated automatically when a user logs in through `systemd-logind`. For example, if the user *alice* is logged in, the access control list should look as follows: ```ShellSession $ getfacl /dev/dri/card0 # file: dev/dri/card0 # owner: root # group: video user::rw- user:alice:rw- group::rw- mask::rw- other::--- ``` If you disabled (this functionality of) `systemd-logind`, you may need to add the user to the `video` group and log in again. ### Mixing different versions of nixpkgs {#sec-gpu-accel-common-issues-mixing-nixpkgs} The *Installable Client Driver* (ICD) mechanism used by OpenCL and Vulkan loads runtimes into its address space using `dlopen`. Mixing an ICD loader mechanism and runtimes from different version of nixpkgs may not work. For example, if the ICD loader uses an older version of glibc than the runtime, the runtime may not be loadable due to missing symbols. Unfortunately, the loader will generally be quiet about such issues. If you suspect that you are running into library version mismatches between an ICL loader and a runtime, you could run an application with the `LD_DEBUG` variable set to get more diagnostic information. For example, OpenCL can be tested with `LD_DEBUG=files clinfo`, which should report missing symbols.