| 
									
										
										
										
											2018-05-29 19:29:23 +03:00
										 |  |  | { config, lib, pkgs, ... }: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | with lib; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | let | 
					
						
							|  |  |  |   cfg = config.virtualisation.kvmgt; | 
					
						
							| 
									
										
										
										
											2019-07-07 23:39:44 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-29 19:29:23 +03:00
										 |  |  |   kernelPackages = config.boot.kernelPackages; | 
					
						
							| 
									
										
										
										
											2019-07-07 23:39:44 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-29 19:29:23 +03:00
										 |  |  |   vgpuOptions = { | 
					
						
							|  |  |  |     uuid = mkOption { | 
					
						
							| 
									
										
										
										
											2020-05-30 11:56:45 -04:00
										 |  |  |       type = with types; listOf str; | 
					
						
							|  |  |  |       description = "UUID(s) of VGPU device. You can generate one with <package>libossp_uuid</package>."; | 
					
						
							| 
									
										
										
										
											2018-05-29 19:29:23 +03:00
										 |  |  |     }; | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2019-07-07 23:39:44 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-29 19:29:23 +03:00
										 |  |  | in { | 
					
						
							|  |  |  |   options = { | 
					
						
							|  |  |  |     virtualisation.kvmgt = { | 
					
						
							|  |  |  |       enable = mkEnableOption ''
 | 
					
						
							|  |  |  |         KVMGT (iGVT-g) VGPU support. Allows Qemu/KVM guests to share host's Intel integrated graphics card. | 
					
						
							| 
									
										
										
										
											2020-03-13 07:00:17 +00:00
										 |  |  |         Currently only one graphical device can be shared. To allow users to access the device without root add them | 
					
						
							|  |  |  |         to the kvm group: <literal>users.extraUsers.<yourusername>.extraGroups = [ "kvm" ];</literal> | 
					
						
							| 
									
										
										
										
											2018-05-29 19:29:23 +03:00
										 |  |  |       '';
 | 
					
						
							|  |  |  |       # multi GPU support is under the question | 
					
						
							|  |  |  |       device = mkOption { | 
					
						
							| 
									
										
										
										
											2019-08-08 22:48:27 +02:00
										 |  |  |         type = types.str; | 
					
						
							| 
									
										
										
										
											2018-05-29 19:29:23 +03:00
										 |  |  |         default = "0000:00:02.0"; | 
					
						
							|  |  |  |         description = "PCI ID of graphics card. You can figure it with <command>ls /sys/class/mdev_bus</command>."; | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  |       vgpus = mkOption { | 
					
						
							|  |  |  |         default = {}; | 
					
						
							|  |  |  |         type = with types; attrsOf (submodule [ { options = vgpuOptions; } ]); | 
					
						
							|  |  |  |         description = ''
 | 
					
						
							|  |  |  |           Virtual GPUs to be used in Qemu. You can find devices via <command>ls /sys/bus/pci/devices/*/mdev_supported_types</command>
 | 
					
						
							|  |  |  |           and find info about device via <command>cat /sys/bus/pci/devices/*/mdev_supported_types/i915-GVTg_V5_4/description</command>
 | 
					
						
							|  |  |  |         '';
 | 
					
						
							|  |  |  |         example = { | 
					
						
							| 
									
										
										
										
											2020-05-30 11:56:45 -04:00
										 |  |  |           i915-GVTg_V5_8.uuid = [ "a297db4a-f4c2-11e6-90f6-d3b88d6c9525" ]; | 
					
						
							| 
									
										
										
										
											2018-05-29 19:29:23 +03:00
										 |  |  |         }; | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   config = mkIf cfg.enable { | 
					
						
							|  |  |  |     assertions = singleton { | 
					
						
							|  |  |  |       assertion = versionAtLeast kernelPackages.kernel.version "4.16"; | 
					
						
							|  |  |  |       message = "KVMGT is not properly supported for kernels older than 4.16"; | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2019-07-07 23:39:44 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     boot.kernelModules = [ "kvmgt" ]; | 
					
						
							| 
									
										
										
										
											2020-03-13 06:58:29 +00:00
										 |  |  |     boot.kernelParams = [ "i915.enable_gvt=1" ]; | 
					
						
							| 
									
										
										
										
											2019-07-07 23:39:44 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-13 07:00:17 +00:00
										 |  |  |     services.udev.extraRules = ''
 | 
					
						
							|  |  |  |       SUBSYSTEM=="vfio", OWNER="root", GROUP="kvm" | 
					
						
							|  |  |  |     '';
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-30 11:56:45 -04:00
										 |  |  |     systemd = let | 
					
						
							|  |  |  |       vgpus = listToAttrs (flatten (mapAttrsToList | 
					
						
							|  |  |  |         (mdev: opt: map (id: nameValuePair "kvmgt-${id}" { inherit mdev; uuid = id; }) opt.uuid) | 
					
						
							|  |  |  |         cfg.vgpus)); | 
					
						
							|  |  |  |     in { | 
					
						
							|  |  |  |       paths = mapAttrs (_: opt: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           description = "KVMGT VGPU ${opt.uuid} path"; | 
					
						
							|  |  |  |           wantedBy = [ "multi-user.target" ]; | 
					
						
							|  |  |  |           pathConfig = { | 
					
						
							|  |  |  |             PathExists = "/sys/bus/pci/devices/${cfg.device}/mdev_supported_types/${opt.mdev}/create"; | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  |         }) vgpus; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       services = mapAttrs (_: opt: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           description = "KVMGT VGPU ${opt.uuid}"; | 
					
						
							|  |  |  |           serviceConfig = { | 
					
						
							|  |  |  |             Type = "oneshot"; | 
					
						
							|  |  |  |             RemainAfterExit = true; | 
					
						
							|  |  |  |             ExecStart = "${pkgs.runtimeShell} -c 'echo ${opt.uuid} > /sys/bus/pci/devices/${cfg.device}/mdev_supported_types/${opt.mdev}/create'"; | 
					
						
							|  |  |  |             ExecStop = "${pkgs.runtimeShell} -c 'echo 1 > /sys/bus/pci/devices/${cfg.device}/${opt.uuid}/remove'"; | 
					
						
							|  |  |  |           }; | 
					
						
							|  |  |  |         }) vgpus; | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2018-05-29 19:29:23 +03:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   meta.maintainers = with maintainers; [ gnidorah ]; | 
					
						
							|  |  |  | } |