diff --git a/pkg/nvcdi/lib-imex.go b/pkg/nvcdi/lib-imex.go new file mode 100644 index 00000000..3c375d56 --- /dev/null +++ b/pkg/nvcdi/lib-imex.go @@ -0,0 +1,118 @@ +/** +# Copyright 2024 NVIDIA CORPORATION +# +# 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 nvcdi + +import ( + "fmt" + "path/filepath" + "strconv" + "strings" + + "tags.cncf.io/container-device-interface/pkg/cdi" + "tags.cncf.io/container-device-interface/specs-go" + + "github.com/NVIDIA/go-nvlib/pkg/nvlib/device" + + "github.com/NVIDIA/nvidia-container-toolkit/internal/discover" + "github.com/NVIDIA/nvidia-container-toolkit/internal/edits" + "github.com/NVIDIA/nvidia-container-toolkit/pkg/nvcdi/spec" +) + +type imexlib nvcdilib + +var _ Interface = (*imexlib)(nil) + +const ( + classImexChannel = "imex-channel" +) + +// GetSpec should not be called for imexlib. +func (l *imexlib) GetSpec() (spec.Interface, error) { + return nil, fmt.Errorf("unexpected call to imexlib.GetSpec()") +} + +// GetAllDeviceSpecs returns the device specs for all available devices. +func (l *imexlib) GetAllDeviceSpecs() ([]specs.Device, error) { + channelsDiscoverer := discover.NewCharDeviceDiscoverer( + l.logger, + l.devRoot, + []string{"/dev/nvidia-caps-imex-channels/channel*"}, + ) + + channels, err := channelsDiscoverer.Devices() + if err != nil { + return nil, err + } + + var channelIDs []string + for _, channel := range channels { + channelIDs = append(channelIDs, filepath.Base(channel.Path)) + } + + return l.GetDeviceSpecsByID(channelIDs...) +} + +// GetCommonEdits returns an empty set of edits for IMEX devices. +func (l *imexlib) GetCommonEdits() (*cdi.ContainerEdits, error) { + return edits.FromDiscoverer(discover.None{}) +} + +// GetDeviceSpecsByID returns the CDI device specs for the IMEX channels specified. +func (l *imexlib) GetDeviceSpecsByID(ids ...string) ([]specs.Device, error) { + var deviceSpecs []specs.Device + for _, id := range ids { + trimmed := strings.TrimPrefix(id, "channel") + _, err := strconv.ParseUint(trimmed, 10, 64) + if err != nil { + return nil, fmt.Errorf("invalid channel ID %v: %w", id, err) + } + path := "/dev/nvidia-caps-imex-channels/channel" + trimmed + deviceSpec := specs.Device{ + Name: trimmed, + ContainerEdits: specs.ContainerEdits{ + DeviceNodes: []*specs.DeviceNode{ + { + Path: path, + HostPath: filepath.Join(l.devRoot, path), + }, + }, + }, + } + deviceSpecs = append(deviceSpecs, deviceSpec) + } + return deviceSpecs, nil +} + +// GetGPUDeviceEdits is unsupported for the imexlib specs +func (l *imexlib) GetGPUDeviceEdits(device.Device) (*cdi.ContainerEdits, error) { + return nil, fmt.Errorf("GetGPUDeviceEdits is not supported") +} + +// GetGPUDeviceSpecs is unsupported for the imexlib specs +func (l *imexlib) GetGPUDeviceSpecs(int, device.Device) ([]specs.Device, error) { + return nil, fmt.Errorf("GetGPUDeviceSpecs is not supported") +} + +// GetMIGDeviceEdits is unsupported for the imexlib specs +func (l *imexlib) GetMIGDeviceEdits(device.Device, device.MigDevice) (*cdi.ContainerEdits, error) { + return nil, fmt.Errorf("GetMIGDeviceEdits is not supported") +} + +// GetMIGDeviceSpecs is unsupported for the imexlib specs +func (l *imexlib) GetMIGDeviceSpecs(int, device.Device, int, device.MigDevice) ([]specs.Device, error) { + return nil, fmt.Errorf("GetMIGDeviceSpecs is not supported") +} diff --git a/pkg/nvcdi/lib-imex_test.go b/pkg/nvcdi/lib-imex_test.go new file mode 100644 index 00000000..c6d099c9 --- /dev/null +++ b/pkg/nvcdi/lib-imex_test.go @@ -0,0 +1,80 @@ +/** +# Copyright 2024 NVIDIA CORPORATION +# +# 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 nvcdi + +import ( + "bytes" + "path/filepath" + "strings" + "testing" + + testlog "github.com/sirupsen/logrus/hooks/test" + "github.com/stretchr/testify/require" + + "github.com/NVIDIA/nvidia-container-toolkit/internal/test" +) + +func TestImexMode(t *testing.T) { + t.Setenv("__NVCT_TESTING_DEVICES_ARE_FILES", "true") + + logger, _ := testlog.NewNullLogger() + + moduleRoot, err := test.GetModuleRoot() + require.NoError(t, err) + hostRoot := filepath.Join(moduleRoot, "testdata", "lookup", "rootfs-1") + + expectedSpec := `--- +cdiVersion: 0.5.0 +containerEdits: + env: + - NVIDIA_VISIBLE_DEVICES=void +devices: +- containerEdits: + deviceNodes: + - hostPath: {{ .hostRoot }}/dev/nvidia-caps-imex-channels/channel0 + path: /dev/nvidia-caps-imex-channels/channel0 + name: "0" +- containerEdits: + deviceNodes: + - hostPath: {{ .hostRoot }}/dev/nvidia-caps-imex-channels/channel1 + path: /dev/nvidia-caps-imex-channels/channel1 + name: "1" +- containerEdits: + deviceNodes: + - hostPath: {{ .hostRoot }}/dev/nvidia-caps-imex-channels/channel2047 + path: /dev/nvidia-caps-imex-channels/channel2047 + name: "2047" +kind: nvidia.com/imex-channel +` + expectedSpec = strings.ReplaceAll(expectedSpec, "{{ .hostRoot }}", hostRoot) + + lib, err := New( + WithLogger(logger), + WithMode(ModeImex), + WithDriverRoot(hostRoot), + ) + require.NoError(t, err) + + spec, err := lib.GetSpec() + require.NoError(t, err) + + var b bytes.Buffer + + _, err = spec.WriteTo(&b) + require.NoError(t, err) + require.Equal(t, expectedSpec, b.String()) +} diff --git a/pkg/nvcdi/lib.go b/pkg/nvcdi/lib.go index ef72efc3..91c837fa 100644 --- a/pkg/nvcdi/lib.go +++ b/pkg/nvcdi/lib.go @@ -161,6 +161,11 @@ func New(opts ...Option) (Interface, error) { l.class = "mofed" } lib = (*mofedlib)(l) + case ModeImex: + if l.class == "" { + l.class = classImexChannel + } + lib = (*imexlib)(l) default: return nil, fmt.Errorf("unknown mode %q", l.mode) } diff --git a/pkg/nvcdi/mode.go b/pkg/nvcdi/mode.go index ad08fa57..5b8f0369 100644 --- a/pkg/nvcdi/mode.go +++ b/pkg/nvcdi/mode.go @@ -40,6 +40,8 @@ const ( // ModeCSV configures the CDI spec generator to generate a spec based on the contents of CSV // mountspec files. ModeCSV = Mode("csv") + // ModeImex configures the CDI spec generated to generate a spec for the available IMEX channels. + ModeImex = Mode("imex") ) type modeConstraint interface { diff --git a/testdata/lookup/rootfs-1/dev/nvidia-caps-imex-channels/channel0 b/testdata/lookup/rootfs-1/dev/nvidia-caps-imex-channels/channel0 new file mode 100644 index 00000000..e69de29b diff --git a/testdata/lookup/rootfs-1/dev/nvidia-caps-imex-channels/channel1 b/testdata/lookup/rootfs-1/dev/nvidia-caps-imex-channels/channel1 new file mode 100644 index 00000000..e69de29b diff --git a/testdata/lookup/rootfs-1/dev/nvidia-caps-imex-channels/channel2047 b/testdata/lookup/rootfs-1/dev/nvidia-caps-imex-channels/channel2047 new file mode 100644 index 00000000..e69de29b diff --git a/tools/container/toolkit/toolkit_test.go b/tools/container/toolkit/toolkit_test.go index bab94c4a..07a3a03a 100644 --- a/tools/container/toolkit/toolkit_test.go +++ b/tools/container/toolkit/toolkit_test.go @@ -100,6 +100,12 @@ devices: path: /dev/nvidia0 - hostPath: /host/driver/root/dev/nvidiactl path: /dev/nvidiactl + - hostPath: /host/driver/root/dev/nvidia-caps-imex-channels/channel0 + path: /dev/nvidia-caps-imex-channels/channel0 + - hostPath: /host/driver/root/dev/nvidia-caps-imex-channels/channel1 + path: /dev/nvidia-caps-imex-channels/channel1 + - hostPath: /host/driver/root/dev/nvidia-caps-imex-channels/channel2047 + path: /dev/nvidia-caps-imex-channels/channel2047 name: all kind: example.com/class `,