2022-07-20 08:52:43 +00:00
/ * *
# Copyright ( c ) 2022 , NVIDIA CORPORATION . All rights reserved .
#
# Licensed under the Apache License , Version 2.0 ( the "License" ) ;
# you may not use this file except in compliance with the License .
# You may obtain a copy of the License at
#
# http : //www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing , software
# distributed under the License is distributed on an "AS IS" BASIS ,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND , either express or implied .
# See the License for the specific language governing permissions and
# limitations under the License .
* * /
package discover
import (
"fmt"
2022-09-29 12:47:04 +00:00
"os"
"path/filepath"
2022-07-20 08:52:43 +00:00
2022-09-29 12:47:04 +00:00
"github.com/NVIDIA/nvidia-container-toolkit/internal/config/image"
"github.com/NVIDIA/nvidia-container-toolkit/internal/info/drm"
"github.com/NVIDIA/nvidia-container-toolkit/internal/info/proc"
2022-07-20 08:52:43 +00:00
" github . com / NVIDIA / nvidia - container - toolkit / internal / lookup "
"github.com/sirupsen/logrus"
)
// NewGraphicsDiscoverer returns the discoverer for graphics tools such as Vulkan.
2022-09-29 12:47:04 +00:00
func NewGraphicsDiscoverer ( logger * logrus . Logger , devices image . VisibleDevices , cfg * Config ) ( Discover , error ) {
2023-02-02 14:57:15 +00:00
driverRoot := cfg . DriverRoot
2022-09-29 12:47:04 +00:00
2023-02-02 14:57:15 +00:00
mounts , err := NewGraphicsMountsDiscoverer ( logger , driverRoot )
2022-11-23 15:26:30 +00:00
if err != nil {
return nil , fmt . Errorf ( "failed to create mounts discoverer: %v" , err )
}
2023-02-02 14:57:15 +00:00
drmDeviceNodes , err := newDRMDeviceDiscoverer ( logger , devices , driverRoot )
2022-11-23 15:26:30 +00:00
if err != nil {
return nil , fmt . Errorf ( "failed to create DRM device discoverer: %v" , err )
}
drmByPathSymlinks := newCreateDRMByPathSymlinks ( logger , drmDeviceNodes , cfg )
discover := Merge (
Merge ( drmDeviceNodes , drmByPathSymlinks ) ,
mounts ,
)
return discover , nil
}
// NewGraphicsMountsDiscoverer creates a discoverer for the mounts required by graphics tools such as vulkan.
2023-02-02 14:57:15 +00:00
func NewGraphicsMountsDiscoverer ( logger * logrus . Logger , driverRoot string ) ( Discover , error ) {
locator , err := lookup . NewLibraryLocator ( logger , driverRoot )
2022-07-20 08:52:43 +00:00
if err != nil {
return nil , fmt . Errorf ( "failed to construct library locator: %v" , err )
}
libraries := NewMounts (
logger ,
locator ,
2023-02-02 14:57:15 +00:00
driverRoot ,
2022-07-20 08:52:43 +00:00
[ ] string {
"libnvidia-egl-gbm.so" ,
} ,
)
jsonMounts := NewMounts (
logger ,
2022-12-02 10:38:40 +00:00
lookup . NewFileLocator (
lookup . WithLogger ( logger ) ,
2023-02-02 14:57:15 +00:00
lookup . WithRoot ( driverRoot ) ,
2022-12-02 10:38:40 +00:00
lookup . WithSearchPaths ( "/etc" , "/usr/share" ) ,
) ,
2023-02-02 14:57:15 +00:00
driverRoot ,
2022-07-20 08:52:43 +00:00
[ ] string {
2022-11-25 13:19:59 +00:00
"glvnd/egl_vendor.d/10_nvidia.json" ,
"vulkan/icd.d/nvidia_icd.json" ,
"vulkan/implicit_layer.d/nvidia_layers.json" ,
"egl/egl_external_platform.d/15_nvidia_gbm.json" ,
"egl/egl_external_platform.d/10_nvidia_wayland.json" ,
2022-07-20 08:52:43 +00:00
} ,
)
discover := Merge (
libraries ,
jsonMounts ,
)
return discover , nil
}
2022-09-29 12:47:04 +00:00
type drmDevicesByPath struct {
None
2023-01-18 10:43:05 +00:00
logger * logrus . Logger
nvidiaCTKPath string
2023-02-02 14:57:15 +00:00
driverRoot string
2023-01-18 10:43:05 +00:00
devicesFrom Discover
2022-09-29 12:47:04 +00:00
}
// newCreateDRMByPathSymlinks creates a discoverer for a hook to create the by-path symlinks for DRM devices discovered by the specified devices discoverer
func newCreateDRMByPathSymlinks ( logger * logrus . Logger , devices Discover , cfg * Config ) Discover {
d := drmDevicesByPath {
2023-01-18 10:43:05 +00:00
logger : logger ,
2023-01-19 09:31:42 +00:00
nvidiaCTKPath : FindNvidiaCTK ( logger , cfg . NvidiaCTKPath ) ,
2023-02-02 14:57:15 +00:00
driverRoot : cfg . DriverRoot ,
2023-01-18 10:43:05 +00:00
devicesFrom : devices ,
2022-09-29 12:47:04 +00:00
}
return & d
}
// Hooks returns a hook to create the symlinks from the required CSV files
func ( d drmDevicesByPath ) Hooks ( ) ( [ ] Hook , error ) {
devices , err := d . devicesFrom . Devices ( )
if err != nil {
return nil , fmt . Errorf ( "failed to discover devices for by-path symlinks: %v" , err )
}
if len ( devices ) == 0 {
return nil , nil
}
2022-11-23 20:49:02 +00:00
links , err := d . getSpecificLinkArgs ( devices )
if err != nil {
return nil , fmt . Errorf ( "failed to determine specific links: %v" , err )
}
if len ( links ) == 0 {
return nil , nil
}
2022-09-29 12:47:04 +00:00
2023-01-19 09:37:10 +00:00
var args [ ] string
2022-09-29 12:47:04 +00:00
for _ , l := range links {
args = append ( args , "--link" , l )
}
2023-01-19 09:37:10 +00:00
hook := CreateNvidiaCTKHook (
d . nvidiaCTKPath ,
"create-symlinks" ,
args ... ,
)
2022-09-29 12:47:04 +00:00
2023-01-19 09:37:10 +00:00
return [ ] Hook { hook } , nil
2022-09-29 12:47:04 +00:00
}
// getSpecificLinkArgs returns the required specic links that need to be created
func ( d drmDevicesByPath ) getSpecificLinkArgs ( devices [ ] Device ) ( [ ] string , error ) {
selectedDevices := make ( map [ string ] bool )
for _ , d := range devices {
selectedDevices [ filepath . Base ( d . HostPath ) ] = true
}
2022-12-02 10:38:40 +00:00
linkLocator := lookup . NewFileLocator (
lookup . WithLogger ( d . logger ) ,
2023-02-02 14:57:15 +00:00
lookup . WithRoot ( d . driverRoot ) ,
2022-12-02 10:38:40 +00:00
)
2022-09-29 12:47:04 +00:00
candidates , err := linkLocator . Locate ( "/dev/dri/by-path/pci-*-*" )
if err != nil {
2022-11-23 20:49:02 +00:00
d . logger . Warningf ( "Failed to locate by-path links: %v; ignoring" , err )
return nil , nil
2022-09-29 12:47:04 +00:00
}
var links [ ] string
for _ , c := range candidates {
device , err := os . Readlink ( c )
if err != nil {
d . logger . Warningf ( "Failed to evaluate symlink %v; ignoring" , c )
continue
}
if selectedDevices [ filepath . Base ( device ) ] {
d . logger . Debugf ( "adding device symlink %v -> %v" , c , device )
links = append ( links , fmt . Sprintf ( "%v::%v" , device , c ) )
}
}
return links , nil
}
// newDRMDeviceDiscoverer creates a discoverer for the DRM devices associated with the requested devices.
2023-02-02 14:57:15 +00:00
func newDRMDeviceDiscoverer ( logger * logrus . Logger , devices image . VisibleDevices , driverRoot string ) ( Discover , error ) {
2022-09-29 12:47:04 +00:00
allDevices := NewDeviceDiscoverer (
logger ,
2023-01-23 15:01:29 +00:00
lookup . NewCharDeviceLocator (
lookup . WithLogger ( logger ) ,
2023-02-02 14:57:15 +00:00
lookup . WithRoot ( driverRoot ) ,
2023-01-23 15:01:29 +00:00
) ,
2023-02-02 14:57:15 +00:00
driverRoot ,
2022-09-29 12:47:04 +00:00
[ ] string {
"/dev/dri/card*" ,
"/dev/dri/renderD*" ,
} ,
)
2023-02-02 14:57:15 +00:00
filter , err := newDRMDeviceFilter ( logger , devices , driverRoot )
2022-09-29 12:47:04 +00:00
if err != nil {
return nil , fmt . Errorf ( "failed to construct DRM device filter: %v" , err )
}
// We return a discoverer that applies the DRM device filter created above to all discovered DRM device nodes.
d := newFilteredDisoverer (
logger ,
allDevices ,
filter ,
)
return d , err
}
// newDRMDeviceFilter creates a filter that matches DRM devices nodes for the visible devices.
2023-02-02 14:57:15 +00:00
func newDRMDeviceFilter ( logger * logrus . Logger , devices image . VisibleDevices , driverRoot string ) ( Filter , error ) {
gpuInformationPaths , err := proc . GetInformationFilePaths ( driverRoot )
2022-09-29 12:47:04 +00:00
if err != nil {
return nil , fmt . Errorf ( "failed to read GPU information: %v" , err )
}
var selectedBusIds [ ] string
for _ , f := range gpuInformationPaths {
info , err := proc . ParseGPUInformationFile ( f )
if err != nil {
return nil , fmt . Errorf ( "failed to parse %v: %v" , f , err )
}
uuid := info [ proc . GPUInfoGPUUUID ]
busID := info [ proc . GPUInfoBusLocation ]
minor := info [ proc . GPUInfoDeviceMinor ]
if devices . Has ( minor ) || devices . Has ( uuid ) || devices . Has ( busID ) {
selectedBusIds = append ( selectedBusIds , busID )
}
}
filter := make ( selectDeviceByPath )
for _ , busID := range selectedBusIds {
drmDeviceNodes , err := drm . GetDeviceNodesByBusID ( busID )
if err != nil {
return nil , fmt . Errorf ( "failed to determine DRM devices for %v: %v" , busID , err )
}
for _ , drmDeviceNode := range drmDeviceNodes {
filter [ filepath . Join ( drmDeviceNode ) ] = true
}
}
return filter , nil
}
// selectDeviceByPath is a filter that allows devices to be selected by the path
type selectDeviceByPath map [ string ] bool
var _ Filter = ( * selectDeviceByPath ) ( nil )
// DeviceIsSelected determines whether the device's path has been selected
func ( s selectDeviceByPath ) DeviceIsSelected ( device Device ) bool {
return s [ device . Path ]
}
// MountIsSelected is always true
func ( s selectDeviceByPath ) MountIsSelected ( Mount ) bool {
return true
}
// HookIsSelected is always true
func ( s selectDeviceByPath ) HookIsSelected ( Hook ) bool {
return true
}