2014-04-29 14:16:34 +02:00
|
|
|
|
# This module provides the proprietary NVIDIA X11 / OpenGL drivers.
|
|
|
|
|
|
2018-11-03 06:55:57 +01:00
|
|
|
|
{ stdenv, config, lib, pkgs, ... }:
|
2014-04-29 14:16:34 +02:00
|
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
|
|
let
|
|
|
|
|
|
|
|
|
|
drivers = config.services.xserver.videoDrivers;
|
|
|
|
|
|
|
|
|
|
# FIXME: should introduce an option like
|
|
|
|
|
# ‘hardware.video.nvidia.package’ for overriding the default NVIDIA
|
|
|
|
|
# driver.
|
2015-02-21 16:29:36 +01:00
|
|
|
|
nvidiaForKernel = kernelPackages:
|
2014-04-29 14:16:34 +02:00
|
|
|
|
if elem "nvidia" drivers then
|
2015-04-04 06:53:34 +02:00
|
|
|
|
kernelPackages.nvidia_x11
|
2016-03-07 23:25:59 +01:00
|
|
|
|
else if elem "nvidiaBeta" drivers then
|
|
|
|
|
kernelPackages.nvidia_x11_beta
|
2014-05-13 18:35:57 +02:00
|
|
|
|
else if elem "nvidiaLegacy304" drivers then
|
2015-02-21 16:29:36 +01:00
|
|
|
|
kernelPackages.nvidia_x11_legacy304
|
2014-11-24 22:34:00 +01:00
|
|
|
|
else if elem "nvidiaLegacy340" drivers then
|
2015-02-21 16:29:36 +01:00
|
|
|
|
kernelPackages.nvidia_x11_legacy340
|
2018-11-30 19:50:47 +01:00
|
|
|
|
else if elem "nvidiaLegacy390" drivers then
|
|
|
|
|
kernelPackages.nvidia_x11_legacy390
|
2015-02-21 16:29:36 +01:00
|
|
|
|
else null;
|
|
|
|
|
|
|
|
|
|
nvidia_x11 = nvidiaForKernel config.boot.kernelPackages;
|
2018-08-11 17:32:00 +02:00
|
|
|
|
nvidia_libs32 =
|
|
|
|
|
if versionOlder nvidia_x11.version "391" then
|
2018-11-03 06:55:57 +01:00
|
|
|
|
((nvidiaForKernel pkgs.pkgsi686Linux.linuxPackages).override { libsOnly = true; kernel = null; }).out
|
2018-08-11 17:32:00 +02:00
|
|
|
|
else
|
|
|
|
|
(nvidiaForKernel config.boot.kernelPackages).lib32;
|
2014-04-29 14:16:34 +02:00
|
|
|
|
|
2015-02-21 16:29:36 +01:00
|
|
|
|
enabled = nvidia_x11 != null;
|
2018-06-30 09:33:45 +02:00
|
|
|
|
|
|
|
|
|
cfg = config.hardware.nvidia;
|
|
|
|
|
optimusCfg = cfg.optimus_prime;
|
2014-04-29 14:16:34 +02:00
|
|
|
|
in
|
|
|
|
|
|
|
|
|
|
{
|
2018-06-30 09:33:45 +02:00
|
|
|
|
options = {
|
|
|
|
|
hardware.nvidia.modesetting.enable = lib.mkOption {
|
|
|
|
|
type = lib.types.bool;
|
|
|
|
|
default = false;
|
|
|
|
|
description = ''
|
|
|
|
|
Enable kernel modesetting when using the NVIDIA proprietary driver.
|
|
|
|
|
|
|
|
|
|
Enabling this fixes screen tearing when using Optimus via PRIME (see
|
|
|
|
|
<option>hardware.nvidia.optimus_prime.enable</option>. This is not enabled
|
|
|
|
|
by default because it is not officially supported by NVIDIA and would not
|
|
|
|
|
work with SLI.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
hardware.nvidia.optimus_prime.enable = lib.mkOption {
|
|
|
|
|
type = lib.types.bool;
|
|
|
|
|
default = false;
|
|
|
|
|
description = ''
|
|
|
|
|
Enable NVIDIA Optimus support using the NVIDIA proprietary driver via PRIME.
|
|
|
|
|
If enabled, the NVIDIA GPU will be always on and used for all rendering,
|
|
|
|
|
while enabling output to displays attached only to the integrated Intel GPU
|
|
|
|
|
without a multiplexer.
|
|
|
|
|
|
|
|
|
|
Note that this option only has any effect if the "nvidia" driver is specified
|
|
|
|
|
in <option>services.xserver.videoDrivers</option>, and it should preferably
|
|
|
|
|
be the only driver there.
|
|
|
|
|
|
|
|
|
|
If this is enabled, then the bus IDs of the NVIDIA and Intel GPUs have to be
|
|
|
|
|
specified (<option>hardware.nvidia.optimus_prime.nvidiaBusId</option> and
|
|
|
|
|
<option>hardware.nvidia.optimus_prime.intelBusId</option>).
|
|
|
|
|
|
|
|
|
|
If you enable this, you may want to also enable kernel modesetting for the
|
|
|
|
|
NVIDIA driver (<option>hardware.nvidia.modesetting.enable</option>) in order
|
|
|
|
|
to prevent tearing.
|
|
|
|
|
|
|
|
|
|
Note that this configuration will only be successful when a display manager
|
|
|
|
|
for which the <option>services.xserver.displayManager.setupCommands</option>
|
|
|
|
|
option is supported is used; notably, SLiM is not supported.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
hardware.nvidia.optimus_prime.nvidiaBusId = lib.mkOption {
|
|
|
|
|
type = lib.types.string;
|
|
|
|
|
default = "";
|
|
|
|
|
example = "PCI:1:0:0";
|
|
|
|
|
description = ''
|
|
|
|
|
Bus ID of the NVIDIA GPU. You can find it using lspci; for example if lspci
|
|
|
|
|
shows the NVIDIA GPU at "01:00.0", set this option to "PCI:1:0:0".
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
hardware.nvidia.optimus_prime.intelBusId = lib.mkOption {
|
|
|
|
|
type = lib.types.string;
|
|
|
|
|
default = "";
|
|
|
|
|
example = "PCI:0:2:0";
|
|
|
|
|
description = ''
|
|
|
|
|
Bus ID of the Intel GPU. You can find it using lspci; for example if lspci
|
|
|
|
|
shows the Intel GPU at "00:02.0", set this option to "PCI:0:2:0".
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
};
|
2014-04-29 14:16:34 +02:00
|
|
|
|
|
|
|
|
|
config = mkIf enabled {
|
2017-05-07 22:27:02 +02:00
|
|
|
|
assertions = [
|
|
|
|
|
{
|
2018-12-12 00:20:52 +01:00
|
|
|
|
assertion = with config.services.xserver.displayManager; gdm.enable -> !gdm.wayland;
|
2018-12-12 16:25:39 +01:00
|
|
|
|
message = "NVIDIA drivers don't support wayland, set services.xserver.displayManager.gdm.wayland=false";
|
2017-05-07 22:27:02 +02:00
|
|
|
|
}
|
2018-06-30 09:33:45 +02:00
|
|
|
|
{
|
|
|
|
|
assertion = !optimusCfg.enable ||
|
|
|
|
|
(optimusCfg.nvidiaBusId != "" && optimusCfg.intelBusId != "");
|
|
|
|
|
message = ''
|
|
|
|
|
When NVIDIA Optimus via PRIME is enabled, the GPU bus IDs must configured.
|
|
|
|
|
'';
|
|
|
|
|
}
|
2017-05-07 22:27:02 +02:00
|
|
|
|
];
|
2014-04-29 14:16:34 +02:00
|
|
|
|
|
2018-06-30 09:33:45 +02:00
|
|
|
|
# If Optimus/PRIME is enabled, we:
|
|
|
|
|
# - Specify the configured NVIDIA GPU bus ID in the Device section for the
|
|
|
|
|
# "nvidia" driver.
|
|
|
|
|
# - Add the AllowEmptyInitialConfiguration option to the Screen section for the
|
|
|
|
|
# "nvidia" driver, in order to allow the X server to start without any outputs.
|
|
|
|
|
# - Add a separate Device section for the Intel GPU, using the "modesetting"
|
|
|
|
|
# driver and with the configured BusID.
|
|
|
|
|
# - Reference that Device section from the ServerLayout section as an inactive
|
|
|
|
|
# device.
|
|
|
|
|
# - Configure the display manager to run specific `xrandr` commands which will
|
|
|
|
|
# configure/enable displays connected to the Intel GPU.
|
|
|
|
|
|
|
|
|
|
services.xserver.drivers = singleton {
|
|
|
|
|
name = "nvidia";
|
|
|
|
|
modules = [ nvidia_x11.bin ];
|
|
|
|
|
libPath = [ nvidia_x11 ];
|
|
|
|
|
deviceSection = optionalString optimusCfg.enable
|
|
|
|
|
''
|
|
|
|
|
BusID "${optimusCfg.nvidiaBusId}"
|
|
|
|
|
'';
|
|
|
|
|
screenSection =
|
|
|
|
|
''
|
|
|
|
|
Option "RandRRotation" "on"
|
|
|
|
|
${optionalString optimusCfg.enable "Option \"AllowEmptyInitialConfiguration\""}
|
|
|
|
|
'';
|
|
|
|
|
};
|
2014-04-29 14:16:34 +02:00
|
|
|
|
|
2018-06-30 09:33:45 +02:00
|
|
|
|
services.xserver.extraConfig = optionalString optimusCfg.enable
|
|
|
|
|
''
|
|
|
|
|
Section "Device"
|
|
|
|
|
Identifier "nvidia-optimus-intel"
|
|
|
|
|
Driver "modesetting"
|
|
|
|
|
BusID "${optimusCfg.intelBusId}"
|
|
|
|
|
Option "AccelMethod" "none"
|
|
|
|
|
EndSection
|
|
|
|
|
'';
|
|
|
|
|
services.xserver.serverLayoutSection = optionalString optimusCfg.enable
|
2014-04-29 14:16:34 +02:00
|
|
|
|
''
|
2018-06-30 09:33:45 +02:00
|
|
|
|
Inactive "nvidia-optimus-intel"
|
2014-04-29 14:16:34 +02:00
|
|
|
|
'';
|
|
|
|
|
|
2018-06-30 09:33:45 +02:00
|
|
|
|
services.xserver.displayManager.setupCommands = optionalString optimusCfg.enable ''
|
|
|
|
|
# Added by nvidia configuration module for Optimus/PRIME.
|
|
|
|
|
${pkgs.xorg.xrandr}/bin/xrandr --setprovideroutputsource modesetting NVIDIA-0
|
|
|
|
|
${pkgs.xorg.xrandr}/bin/xrandr --auto
|
|
|
|
|
'';
|
|
|
|
|
|
2017-02-11 15:11:18 +01:00
|
|
|
|
environment.etc."nvidia/nvidia-application-profiles-rc" = mkIf nvidia_x11.useProfiles {
|
|
|
|
|
source = "${nvidia_x11.bin}/share/nvidia/nvidia-application-profiles-rc";
|
|
|
|
|
};
|
|
|
|
|
|
2018-03-17 16:52:42 +01:00
|
|
|
|
hardware.opengl.package = nvidia_x11.out;
|
2018-08-11 17:32:00 +02:00
|
|
|
|
hardware.opengl.package32 = nvidia_libs32;
|
2014-04-29 14:16:34 +02:00
|
|
|
|
|
2017-05-30 20:56:34 +02:00
|
|
|
|
environment.systemPackages = [ nvidia_x11.bin nvidia_x11.settings ]
|
|
|
|
|
++ lib.filter (p: p != null) [ nvidia_x11.persistenced ];
|
2014-04-29 14:16:34 +02:00
|
|
|
|
|
2017-01-29 22:29:39 +01:00
|
|
|
|
boot.extraModulePackages = [ nvidia_x11.bin ];
|
2014-04-29 14:16:34 +02:00
|
|
|
|
|
2015-03-03 20:15:38 +01:00
|
|
|
|
# nvidia-uvm is required by CUDA applications.
|
2017-05-07 21:37:56 +02:00
|
|
|
|
boot.kernelModules = [ "nvidia-uvm" ] ++
|
|
|
|
|
lib.optionals config.services.xserver.enable [ "nvidia" "nvidia_modeset" "nvidia_drm" ];
|
|
|
|
|
|
2018-06-30 09:33:45 +02:00
|
|
|
|
# If requested enable modesetting via kernel parameter.
|
|
|
|
|
boot.kernelParams = optional cfg.modesetting.enable "nvidia-drm.modeset=1";
|
2015-03-03 20:15:38 +01:00
|
|
|
|
|
|
|
|
|
# Create /dev/nvidia-uvm when the nvidia-uvm module is loaded.
|
|
|
|
|
services.udev.extraRules =
|
|
|
|
|
''
|
2018-03-01 20:38:53 +01:00
|
|
|
|
KERNEL=="nvidia", RUN+="${pkgs.runtimeShell} -c 'mknod -m 666 /dev/nvidiactl c $(grep nvidia-frontend /proc/devices | cut -d \ -f 1) 255'"
|
|
|
|
|
KERNEL=="nvidia_modeset", RUN+="${pkgs.runtimeShell} -c 'mknod -m 666 /dev/nvidia-modeset c $(grep nvidia-frontend /proc/devices | cut -d \ -f 1) 254'"
|
|
|
|
|
KERNEL=="card*", SUBSYSTEM=="drm", DRIVERS=="nvidia", RUN+="${pkgs.runtimeShell} -c 'mknod -m 666 /dev/nvidia%n c $(grep nvidia-frontend /proc/devices | cut -d \ -f 1) %n'"
|
|
|
|
|
KERNEL=="nvidia_uvm", RUN+="${pkgs.runtimeShell} -c 'mknod -m 666 /dev/nvidia-uvm c $(grep nvidia-uvm /proc/devices | cut -d \ -f 1) 0'"
|
2015-03-03 20:15:38 +01:00
|
|
|
|
'';
|
|
|
|
|
|
2014-04-29 14:16:34 +02:00
|
|
|
|
boot.blacklistedKernelModules = [ "nouveau" "nvidiafb" ];
|
|
|
|
|
|
|
|
|
|
services.acpid.enable = true;
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
}
|