Skip to content

Commit

Permalink
Add IntelRdt to CDI spec
Browse files Browse the repository at this point in the history
Add support for specifying the OCI Linux.IntelRdt configuration that
is the control point for cache and memory bandwidth allocation
technologies sucn as Intel RDT (Resource Director Technology). There can
only be one IntelRdt configuration per container so in case multiple
specs happened to specify it the last one applied prevails. Also, the
IntelRdt configuration is always applied as a whole - editing specific
sub-fields is not supported.

Signed-off-by: Markus Lehtonen <[email protected]>
  • Loading branch information
marquiz committed Sep 29, 2023
1 parent 1f83d84 commit c4b03ce
Show file tree
Hide file tree
Showing 6 changed files with 133 additions and 4 deletions.
21 changes: 18 additions & 3 deletions SPEC.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@

## Version

This is CDI **spec** version **0.6.0**.
This is CDI **spec** version **0.7.0**.

### Update policy

Expand All @@ -27,7 +27,7 @@ Released versions of the spec are available as Git tags.
| v0.4.0 | | Added `type` field to Mount specification |
| v0.5.0 | | Add `HostPath` to `DeviceNodes` |
| v0.6.0 | | Add `Annotations` field to `Spec` and `Device` specifications |
| | | Allow dots (`.`) in name segment of `Kind` field |
| | | Allow dots (`.`) in name segment of `Kind` field. Add `IntelRdt`field. |

*Note*: The initial release of a **spec** with version `v0.x.0` will be tagged as
`v0.x.0` with subsequent changes to the API applicable to this version tagged as `v0.x.y`.
Expand Down Expand Up @@ -82,7 +82,7 @@ The keywords "must", "must not", "required", "shall", "shall not", "should", "sh

```
{
"cdiVersion": "0.6.0",
"cdiVersion": "0.7.0",
"kind": "<name>",
// This field contains a set of key-value pairs that may be used to provide
Expand Down Expand Up @@ -150,6 +150,13 @@ The keywords "must", "must not", "required", "shall", "shall not", "should", "sh
"timeout": <int> (optional)
}
]
"intelRdt": { (optional)
"closID": "<name>", (optional)
"l3CacheSchema": "string" (optional)
"memBwSchema": "string" (optional)
"enableCMT": "<boolean>" (optional)
"enableMBM": "<boolean>" (optional)
}
}
]
}
Expand Down Expand Up @@ -220,6 +227,12 @@ The `containerEdits` field has the following definition:
* `args` (array of strings, OPTIONAL) with the same semantics as IEEE Std 1003.1-2008 execv's argv.
* `env` (array of strings, OPTIONAL) with the same semantics as IEEE Std 1003.1-2008's environ.
* `timeout` (int, OPTIONAL) is the number of seconds before aborting the hook. If set, timeout MUST be greater than zero. If not set container runtime will wait for the hook to return.
* `intelRdt` (object, OPTIONAL) describes the Linux [resctrl][resctrl] settings for the container (object, OPTIONAL)
* `closID` (string, OPTIONAL) name of the CLOS (Class of Service).
* `l3CacheSchema` (string, OPTIONAL) L3 cache allocation schema for the CLOS.
* `memBwSchema` (string, OPTIONAL) memory bandwidth allocation schema for the CLOS.
* `enableCMT` (boolean, OPTIONAL) whether to enable cache monitoring
* `enableMBM` (boolean, OPTIONAL) whether to enable memory bandwidth monitoring

## Error Handling
* Kind requested is not present in any CDI file.
Expand All @@ -231,3 +244,5 @@ The `containerEdits` field has the following definition:
This is because a resource does not need to exist when the spec is written, but it needs to exist when the container is created.
* Hook fails to execute.
Container runtimes should surface an error when hooks fails to execute.

[resctrl]: https://docs.kernel.org/arch/x86/resctrl.html
31 changes: 30 additions & 1 deletion pkg/cdi/container-edits.go
Original file line number Diff line number Diff line change
Expand Up @@ -144,6 +144,13 @@ func (e *ContainerEdits) Apply(spec *oci.Spec) error {
}
}

if e.IntelRdt != nil {
// The specgen is missing functionality to set all parameters so we
// just piggy-back on it to initialize all structs and the copy over.
specgen.SetLinuxIntelRdtClosID(e.IntelRdt.ClosID)
spec.Linux.IntelRdt = e.IntelRdt.ToOCI()
}

return nil
}

Expand Down Expand Up @@ -171,6 +178,11 @@ func (e *ContainerEdits) Validate() error {
return err
}
}
if e.IntelRdt != nil {
if err := ValidateIntelRdt(e.IntelRdt); err != nil {
return err
}
}

return nil
}
Expand All @@ -192,6 +204,9 @@ func (e *ContainerEdits) Append(o *ContainerEdits) *ContainerEdits {
e.DeviceNodes = append(e.DeviceNodes, o.DeviceNodes...)
e.Hooks = append(e.Hooks, o.Hooks...)
e.Mounts = append(e.Mounts, o.Mounts...)
if o.IntelRdt != nil {
e.IntelRdt = o.IntelRdt
}

return e
}
Expand All @@ -202,7 +217,7 @@ func (e *ContainerEdits) isEmpty() bool {
if e == nil {
return false
}
return len(e.Env)+len(e.DeviceNodes)+len(e.Hooks)+len(e.Mounts) == 0
return len(e.Env)+len(e.DeviceNodes)+len(e.Hooks)+len(e.Mounts) == 0 && e.IntelRdt == nil
}

// ValidateEnv validates the given environment variables.
Expand Down Expand Up @@ -280,6 +295,20 @@ func (m *Mount) Validate() error {
return nil
}

// IntelRdt a wrapper used for validating IntelRdt configuration.
type IntelRdt struct {
*specs.IntelRdt
}

// ValidateIntelRdt validates the IntelRdt configuration.
func ValidateIntelRdt(i *specs.IntelRdt) error {
// ClosID must be a valid Linux filename
if len(i.ClosID) >= 4096 || i.ClosID == "." || i.ClosID == ".." || strings.ContainsAny(i.ClosID, "/\n") {
return errors.New("invalid ClosID")
}
return nil
}

// Ensure OCI Spec hooks are not nil so we can add hooks.
func ensureOCIHooks(spec *oci.Spec) {
if spec.Hooks == nil {
Expand Down
41 changes: 41 additions & 0 deletions pkg/cdi/container-edits_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -245,6 +245,23 @@ func TestValidateContainerEdits(t *testing.T) {
},
invalid: true,
},
{
name: "valid rdt config",
edits: &cdi.ContainerEdits{
IntelRdt: &cdi.IntelRdt{
ClosID: "foo.bar",
},
},
},
{
name: "invalid rdt config, invalid closID",
edits: &cdi.ContainerEdits{
IntelRdt: &cdi.IntelRdt{
ClosID: "foo/bar",
},
},
invalid: true,
},
} {
t.Run(tc.name, func(t *testing.T) {
edits := ContainerEdits{tc.edits}
Expand Down Expand Up @@ -467,6 +484,30 @@ func TestApplyContainerEdits(t *testing.T) {
},
},
},
{
name: "empty spec, rdt",
spec: &oci.Spec{},
edits: &cdi.ContainerEdits{
IntelRdt: &cdi.IntelRdt{
ClosID: "clos-1",
L3CacheSchema: "L3:0=ff;1=ff",
MemBwSchema: "MB:0=50;1=50",
EnableCMT: true,
EnableMBM: true,
},
},
result: &oci.Spec{
Linux: &oci.Linux{
IntelRdt: &oci.LinuxIntelRdt{
ClosID: "clos-1",
L3CacheSchema: "L3:0=ff;1=ff",
MemBwSchema: "MB:0=50;1=50",
EnableCMT: true,
EnableMBM: true,
},
},
},
},
} {
t.Run(tc.name, func(t *testing.T) {
edits := ContainerEdits{tc.edits}
Expand Down
23 changes: 23 additions & 0 deletions schema/defs.json
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,9 @@
"type": "string"
}
},
"FileName": {
"type": "string"
},
"FilePath": {
"type": "string"
},
Expand Down Expand Up @@ -134,6 +137,26 @@
"items": {
"$ref": "#/definitions/Hook"
}
},
"intelRdt": {
"type": "object",
"properties": {
"closID": {
"$ref": "#/definitions/FileName"
},
"l3CacheSchema": {
"type": "string"
},
"memBwSchema": {
"type": "string"
},
"enableCMT": {
"type": "boolean"
},
"enableMBM": {
"type": "boolean"
}
}
}
}
},
Expand Down
10 changes: 10 additions & 0 deletions specs-go/config.go
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ type ContainerEdits struct {
DeviceNodes []*DeviceNode `json:"deviceNodes,omitempty"`
Hooks []*Hook `json:"hooks,omitempty"`
Mounts []*Mount `json:"mounts,omitempty"`
IntelRdt *IntelRdt `json:"intelRdt,omitempty"`
}

// DeviceNode represents a device node that needs to be added to the OCI spec.
Expand Down Expand Up @@ -60,3 +61,12 @@ type Hook struct {
Env []string `json:"env,omitempty"`
Timeout *int `json:"timeout,omitempty"`
}

// IntelRdt describes the Linux IntelRdt parameters to set in the OCI spec.
type IntelRdt struct {
ClosID string `json:"closID,omitempty"`
L3CacheSchema string `json:"l3CacheSchema,omitempty"`
MemBwSchema string `json:"memBwSchema,omitempty"`
EnableCMT bool `json:"enableCMT,omitempty"`
EnableMBM bool `json:"enableMBM,omitempty"`
}
11 changes: 11 additions & 0 deletions specs-go/oci.go
Original file line number Diff line number Diff line change
Expand Up @@ -36,3 +36,14 @@ func (d *DeviceNode) ToOCI() spec.LinuxDevice {
GID: d.GID,
}
}

// ToOCI returns the opencontainers runtime Spec LinuxIntelRdt for this IntelRdt config.
func (i *IntelRdt) ToOCI() *spec.LinuxIntelRdt {
return &spec.LinuxIntelRdt{
ClosID: i.ClosID,
L3CacheSchema: i.L3CacheSchema,
MemBwSchema: i.MemBwSchema,
EnableCMT: i.EnableCMT,
EnableMBM: i.EnableMBM,
}
}

0 comments on commit c4b03ce

Please sign in to comment.