/** # 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" "path/filepath" "strings" "github.com/NVIDIA/nvidia-container-toolkit/internal/lookup" "github.com/container-orchestrated-devices/container-device-interface/pkg/cdi" "github.com/sirupsen/logrus" ) // NewLDCacheUpdateHook creates a discoverer that updates the ldcache for the specified mounts. A logger can also be specified func NewLDCacheUpdateHook(logger *logrus.Logger, mounts Discover, cfg *Config) (Discover, error) { d := ldconfig{ logger: logger, mountsFrom: mounts, lookup: lookup.NewExecutableLocator(logger, cfg.Root), nvidiaCTKExecutablePath: cfg.NVIDIAContainerToolkitCLIExecutablePath, } return &d, nil } const ( nvidiaCTKDefaultFilePath = "/usr/bin/nvidia-ctk" ) type ldconfig struct { None logger *logrus.Logger mountsFrom Discover lookup lookup.Locator nvidiaCTKExecutablePath string } // Hooks checks the required mounts for libraries and returns a hook to update the LDcache for the discovered paths. func (d ldconfig) Hooks() ([]Hook, error) { mounts, err := d.mountsFrom.Mounts() if err != nil { return nil, fmt.Errorf("failed to discover mounts for ldcache update: %v", err) } h := CreateLDCacheUpdateHook( d.nvidiaCTKExecutablePath, getLibraryPaths(mounts), ) return []Hook{h}, nil } // CreateLDCacheUpdateHook locates the NVIDIA Container Toolkit CLI and creates a hook for updating the LD Cache func CreateLDCacheUpdateHook(executable string, libraries []string) Hook { var args []string for _, f := range uniqueFolders(libraries) { args = append(args, "--folder", f) } hook := CreateNvidiaCTKHook( executable, "update-ldcache", args..., ) return hook } // CreateNvidiaCTKHook creates a hook which invokes the NVIDIA Container CLI hook subcommand. func CreateNvidiaCTKHook(executable string, hookName string, additionalArgs ...string) Hook { return Hook{ Lifecycle: cdi.CreateContainerHook, Path: executable, Args: append([]string{filepath.Base(executable), "hook", hookName}, additionalArgs...), } } // getLibraryPaths extracts the library dirs from the specified mounts func getLibraryPaths(mounts []Mount) []string { var paths []string for _, m := range mounts { if !isLibName(m.Path) { continue } paths = append(paths, m.Path) } return paths } // isLibName checks if the specified filename is a library (i.e. ends in `.so*`) func isLibName(filename string) bool { base := filepath.Base(filename) isLib, err := filepath.Match("lib?*.so*", base) if !isLib || err != nil { return false } parts := strings.Split(base, ".so") if len(parts) == 1 { return true } return parts[len(parts)-1] == "" || strings.HasPrefix(parts[len(parts)-1], ".") } // uniqueFolders returns the unique set of folders for the specified files func uniqueFolders(libraries []string) []string { var paths []string checked := make(map[string]bool) for _, l := range libraries { dir := filepath.Dir(l) if dir == "" { continue } if checked[dir] { continue } checked[dir] = true paths = append(paths, dir) } return paths }