Skip to content

Commit

Permalink
Merge branch 'nvidia-ctk-cdi-transform' into 'main'
Browse files Browse the repository at this point in the history
Add 'target-driver-root' option to 'nvidia-ctk cdi generate' to transform root...

See merge request nvidia/container-toolkit/container-toolkit!363
  • Loading branch information
Evan Lezar committed Mar 28, 2023
2 parents 8eef7e5 + 7f7fc35 commit f698396
Show file tree
Hide file tree
Showing 5 changed files with 250 additions and 3 deletions.
2 changes: 2 additions & 0 deletions cmd/nvidia-ctk/cdi/cdi.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ package cdi

import (
"github.com/NVIDIA/nvidia-container-toolkit/cmd/nvidia-ctk/cdi/generate"
"github.com/NVIDIA/nvidia-container-toolkit/cmd/nvidia-ctk/cdi/transform"
"github.com/sirupsen/logrus"
"github.com/urfave/cli/v2"
)
Expand All @@ -44,6 +45,7 @@ func (m command) build() *cli.Command {

hook.Subcommands = []*cli.Command{
generate.NewCommand(m.logger),
transform.NewCommand(m.logger),
}

return &hook
Expand Down
26 changes: 24 additions & 2 deletions cmd/nvidia-ctk/cdi/generate/generate.go
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,8 @@ type config struct {
driverRoot string
nvidiaCTKPath string
mode string
vendor string
class string
}

// NewCommand constructs a generate-cdi command with the specified logger
Expand Down Expand Up @@ -108,6 +110,20 @@ func (m command) build() *cli.Command {
Usage: "Specify the path to use for the nvidia-ctk in the generated CDI specification. If this is left empty, the path will be searched.",
Destination: &cfg.nvidiaCTKPath,
},
&cli.StringFlag{
Name: "vendor",
Aliases: []string{"cdi-vendor"},
Usage: "the vendor string to use for the generated CDI specification.",
Value: "nvidia.com",
Destination: &cfg.vendor,
},
&cli.StringFlag{
Name: "class",
Aliases: []string{"cdi-class"},
Usage: "the class string to use for the generated CDI specification.",
Value: "gpu",
Destination: &cfg.class,
},
}

return &c
Expand Down Expand Up @@ -149,6 +165,12 @@ func (m command) validateFlags(c *cli.Context, cfg *config) error {
}
}

if err := cdi.ValidateVendorName(cfg.vendor); err != nil {
return fmt.Errorf("invalid CDI vendor name: %v", err)
}
if err := cdi.ValidateClassName(cfg.class); err != nil {
return fmt.Errorf("invalid CDI class name: %v", err)
}
return nil
}

Expand Down Expand Up @@ -224,8 +246,8 @@ func (m command) generateSpec(cfg *config) (spec.Interface, error) {
}

return spec.New(
spec.WithVendor("nvidia.com"),
spec.WithClass("gpu"),
spec.WithVendor(cfg.vendor),
spec.WithClass(cfg.class),
spec.WithDeviceSpecs(deviceSpecs),
spec.WithEdits(*commonEdits.ContainerEdits),
spec.WithFormat(cfg.format),
Expand Down
159 changes: 159 additions & 0 deletions cmd/nvidia-ctk/cdi/transform/root/root.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,159 @@
/**
# 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 root

import (
"fmt"
"io"
"os"

"github.com/NVIDIA/nvidia-container-toolkit/pkg/nvcdi/spec"
"github.com/NVIDIA/nvidia-container-toolkit/pkg/nvcdi/transform"
"github.com/container-orchestrated-devices/container-device-interface/pkg/cdi"
"github.com/sirupsen/logrus"
"github.com/urfave/cli/v2"
)

type loadSaver interface {
Load() (spec.Interface, error)
Save(spec.Interface) error
}

type command struct {
logger *logrus.Logger
}

type transformOptions struct {
input string
output string
}

type options struct {
transformOptions
from string
to string
}

// NewCommand constructs a generate-cdi command with the specified logger
func NewCommand(logger *logrus.Logger) *cli.Command {
c := command{
logger: logger,
}
return c.build()
}

// build creates the CLI command
func (m command) build() *cli.Command {
opts := options{}

c := cli.Command{
Name: "root",
Usage: "Apply a root transform to a CDI specification",
Before: func(c *cli.Context) error {
return m.validateFlags(c, &opts)
},
Action: func(c *cli.Context) error {
return m.run(c, &opts)
},
}

c.Flags = []cli.Flag{
&cli.StringFlag{
Name: "input",
Usage: "Specify the file to read the CDI specification from. If this is '-' the specification is read from STDIN",
Value: "-",
Destination: &opts.input,
},
&cli.StringFlag{
Name: "output",
Usage: "Specify the file to output the generated CDI specification to. If this is '' the specification is output to STDOUT",
Destination: &opts.output,
},
&cli.StringFlag{
Name: "from",
Usage: "specify the root to be transformed",
Destination: &opts.from,
},
&cli.StringFlag{
Name: "to",
Usage: "specify the replacement root. If this is the same as the from root, the transform is a no-op.",
Value: "",
Destination: &opts.to,
},
}

return &c
}

func (m command) validateFlags(c *cli.Context, opts *options) error {
return nil
}

func (m command) run(c *cli.Context, opts *options) error {
spec, err := opts.Load()
if err != nil {
return fmt.Errorf("failed to load CDI specification: %w", err)
}

err = transform.NewRootTransformer(
opts.from,
opts.to,
).Transform(spec.Raw())
if err != nil {
return fmt.Errorf("failed to transform CDI specification: %w", err)
}

return opts.Save(spec)
}

// Load lodas the input CDI specification
func (o transformOptions) Load() (spec.Interface, error) {
contents, err := o.getContents()
if err != nil {
return nil, fmt.Errorf("failed to read spec contents: %v", err)
}

raw, err := cdi.ParseSpec(contents)
if err != nil {
return nil, fmt.Errorf("failed to parse CDI spec: %v", err)
}

return spec.New(
spec.WithRawSpec(raw),
)
}

func (o transformOptions) getContents() ([]byte, error) {
if o.input == "-" {
return io.ReadAll(os.Stdin)
}

return os.ReadFile(o.input)
}

// Save saves the CDI specification to the output file
func (o transformOptions) Save(s spec.Interface) error {
if o.output == "" {
_, err := s.WriteTo(os.Stdout)
if err != nil {
return fmt.Errorf("failed to write CDI spec to STDOUT: %v", err)
}
return nil
}

return s.Save(o.output)
}
51 changes: 51 additions & 0 deletions cmd/nvidia-ctk/cdi/transform/transform.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
/**
# 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 transform

import (
"github.com/NVIDIA/nvidia-container-toolkit/cmd/nvidia-ctk/cdi/transform/root"
"github.com/sirupsen/logrus"
"github.com/urfave/cli/v2"
)

type command struct {
logger *logrus.Logger
}

// NewCommand constructs a command with the specified logger
func NewCommand(logger *logrus.Logger) *cli.Command {
c := command{
logger: logger,
}
return c.build()
}

// build creates the CLI command
func (m command) build() *cli.Command {
c := cli.Command{
Name: "transform",
Usage: "Apply a transform to a CDI specification",
}

c.Flags = []cli.Flag{}

c.Subcommands = []*cli.Command{
root.NewCommand(m.logger),
}

return &c
}
15 changes: 14 additions & 1 deletion pkg/nvcdi/spec/builder.go
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,13 @@ func newBuilder(opts ...Option) *builder {
for _, opt := range opts {
opt(s)
}
if s.raw != nil {
s.noSimplify = true
vendor, class := cdi.ParseQualifier(s.raw.Kind)
s.vendor = vendor
s.class = class
}

if s.version == "" {
s.version = DetectMinimumVersion
}
Expand All @@ -60,7 +67,6 @@ func newBuilder(opts ...Option) *builder {
// Build builds a CDI spec form the spec builder.
func (o *builder) Build() (*spec, error) {
raw := o.raw

if raw == nil {
raw = &specs.Spec{
Version: o.version,
Expand Down Expand Up @@ -144,3 +150,10 @@ func WithNoSimplify(noSimplify bool) Option {
o.noSimplify = noSimplify
}
}

// WithRawSpec sets the raw spec for the spec builder
func WithRawSpec(raw *specs.Spec) Option {
return func(o *builder) {
o.raw = raw
}
}

0 comments on commit f698396

Please sign in to comment.