Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

gpu monitor on dcgm doc #2292

Open
wants to merge 2 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
79 changes: 79 additions & 0 deletions cmd/koordlet/main.go
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
/*

Check failure on line 1 in cmd/koordlet/main.go

View workflow job for this annotation

GitHub Actions / golangci-lint

: # github.com/koordinator-sh/koordinator/cmd/koordlet
Copyright 2022 The Koordinator Authors.

Licensed under the Apache License, Version 2.0 (the "License");
Expand All @@ -18,8 +18,12 @@

import (
"flag"
"google.golang.org/grpc"
pb "k8s.io/kubelet/pkg/apis/podresources/v1alpha1"
"net"
"net/http"
_ "net/http/pprof"
"os"
"time"

"github.com/prometheus/client_golang/prometheus/promhttp"
Expand All @@ -36,6 +40,8 @@
metricsutil "github.com/koordinator-sh/koordinator/pkg/util/metrics"
)

const ServerPodResourcesKubeletSocket = "/pod-resources/koordlet.sock"

func main() {
cfg := config.NewConfiguration()
cfg.InitFlags(flag.CommandLine)
Expand Down Expand Up @@ -101,3 +107,76 @@
// http.HandleFunc("/healthz", d.HealthzHandler())
klog.Fatalf("Prometheus monitoring failed: %v", http.ListenAndServe(*options.ServerAddr, mux))
}

type PodResourcesServer struct{}

var nodePodResources = []*pb.PodResources{
{
Name: "pod-1",
Namespace: "default",
Containers: []*pb.ContainerResources{
{
Name: "container-1",
Devices: []*pb.ContainerDevices{
{
ResourceName: "nvidia.com/gpu",
DeviceIds: []string{"GPU-32e51276-4ddd-5d40-b63a-7bf69ea08b2e", "GPU-6638b0e5-0708-3bef-74d1-5b75b85f6e75"},
},
},
},
},
},
{
Name: "pod-2",
Namespace: "kube-system",
Containers: []*pb.ContainerResources{
{
Name: "container-2",
Devices: []*pb.ContainerDevices{
{
ResourceName: "nvidia.com/gpu",
DeviceIds: []string{"GPU-68d4072a-f4b8-46e5-c76f-66ce3d02cc38"},
},
},
},
},
},
}

func (s *PodResourcesServer) List(ctx context.Context, req *pb.ListPodResourcesRequest) (*pb.ListPodResourcesResponse, error) {

Check failure on line 146 in cmd/koordlet/main.go

View workflow job for this annotation

GitHub Actions / golangci-lint

undefined: context (typecheck)

Check failure on line 146 in cmd/koordlet/main.go

View workflow job for this annotation

GitHub Actions / unit-tests(Run Go build)

undefined: context

Check failure on line 146 in cmd/koordlet/main.go

View workflow job for this annotation

GitHub Actions / unit-tests(Run Go test)

undefined: context
//var pods []*pb.PodResources
klog.V(1).Infof("List(): start to list pod")

return &pb.ListPodResourcesResponse{PodResources: nodePodResources}, nil
}

func startGrpc() error {
lis, err := net.Listen("unix", ServerPodResourcesKubeletSocket)
if err != nil {
klog.Errorf("failed to listen: %v", err)
return err
}
klog.V(1).Infof("setSocketPermissions...")
if err := setSocketPermissions(ServerPodResourcesKubeletSocket); err != nil {
klog.Errorf("failed to set socket permissions: %v", err)
return err
}
klog.V(1).Infof("NewServer...")
server := grpc.NewServer()
klog.V(1).Infof("RegisterPodResourcesListerServer...")
pb.RegisterPodResourcesListerServer(server, &PodResourcesServer{})

klog.V(1).Infof("Starting gRPC server on %s", ServerPodResourcesKubeletSocket)
if err := server.Serve(lis); err != nil {
klog.Errorf("failed to serve: %v", err)
return err
}
return nil
}

func setSocketPermissions(socketPath string) error {
// In a real application, you would set the correct permissions here.
// For example:
return os.Chmod(socketPath, 0660)
//return nil
}
Binary file added docs/images/dcgm-gpuhook-architecture.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added docs/images/dcgm-gpuhook-readwrite.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
2 changes: 1 addition & 1 deletion docs/proposals/koordlet/20220615-qos-manager.md
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@ Currently, plugins from resmanager in Koordlet are mixed together, they should b
## Proposal
### Design

![image](../../images/qos-manager.svg)
_![image](../../images/qos-manager.svg)_
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

这个改动是为啥


The below is directory tree of qos-manager inside koordlet, all existing dynamic plugins(as built-in plugins) will be move into sub-directory `plugins`.

Expand Down
172 changes: 172 additions & 0 deletions docs/proposals/koordlet/20241210-GPU monitor on DCGM.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,172 @@
# GPU Monitor on DCGM

## Summary

To support koordinator's GPU monitoring capabilities, koordinator needs smooth integration with the GPU monitoring component dcgm. In order to achieve this goal, a joint adaptation scheme of koordinator and dcgm is proposed to collect GPU indicators on a single node.


## Motivation

dcgm is a GPU monitoring component, which reads the kubelet.sock on the side to obtain pod and allocate GPU information, and then associate GPU monitoring indicators. The GPU allocation of Koordlet framework is a centralized allocation architecture, and kublete does not have GPU allocation information on the end side, so adaptation modification is required.

### Goals
- The GPU NRI HOOK writes the Pod list and corresponding GPU list information to the mount file on the GPU NRI maintenance terminal.
- Adjust the logic of the DCGM component reading Pod and GPU: switch from reading from kubelet.sock to reading from the above mount file
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

we shouldn't modify the dcgm code

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It should be possible to implement through configuration and is being analyzed


## Proposal

### User Stories

#### Story 1

As a user, I want to observe the performance metrics of GPU at runtime.

#### Story 2

As a cluster administrator, I want to monitor the GPU resource usage of the cluster.

### Design

#### GPU NRI Hook

The GPU NRI Hook maintains a list of all Pods on the node and a list of corresponding allocated GPUs in the gpu.go code each time a Pod's GPU request is processed.

The path of the gpu.go code file is:

```go
pkg/koordlet/runtimehooks/hooks/gpu
- gpu.go
```

Define a function in the gpu.go code file: maintainPodList().

Maintain PodList logic is implemented in the function maintainPodList(), and function named InjectContainerGPUEnv() in the Gpu.go code file actively calls the function to dynamically maintain the latest Pod list each time the GPU environment variable is set.The function definition and function call are as follows in the gpu.go code file:

```go
func maintainPodList(request ContainerRequest) error {
...

}

func (p *gpuPlugin) InjectContainerGPUEnv(proto protocol.HooksProtocol) error {
...

maintainPodList(request)
}
```

In order to be compatible with dcgm's pod and GPU mapping data structures, a structure named PodInfo is defined in the NRI HOOK to maintain Pod information, including the following properties: Pod's name, namespace, container name, and list of assigned GPUs.

The structure PodInfo is defined as follows:

```go
type PodInfo struct {
Name string `json:"name"`
Namespace string `json:"namespace"`
Container string `json:"container"`
GPUList []string `json:"gpu_list"`
}
```

Then koordlet mounts the directory file of the host, such as /var/lib/kubelet/pod-resources/podlist.json, which is used to write Pod and GPU information. Mount information is edited at koordlet.yaml:

```yaml
volumeMounts:
- mountPath: /podlist.json
name: podInfo
......
volumes:
- hostPath:
path: /var/lib/kubelet/pod-resources/podlist.json
type: ""
name: podInfo
```

The GPU NRI Hook is then responsible for converting the maintained podInfo list information into a json string and writing it to a file named podlist.json.

<img src="/docs/images/dcgm-gpuhook-architecture.png" style="zoom:25%;" />

#### DCGM

In the DCGM native project, the logic to extract the list of Pods and the list of Gpus is shown in the following code:

```go
#dcgm\pkg\dcgmexporter\kubernetes.go

func (p *PodMapper) Process(metrics MetricsByCounter, sysInfo SystemInfo) error {
socketPath := p.Config.PodResourcesKubeletSocket
_, err := os.Stat(socketPath)
if os.IsNotExist(err) {
logrus.Info("No Kubelet socket, ignoring")
return nil
}

c, cleanup, err := connectToServer(socketPath)
if err != nil {
return err
}
defer cleanup()

pods, err := p.listPods(c)
if err != nil {
return err
}

deviceToPod := p.toDeviceToPod(pods, sysInfo)

......

return nil
}
```

The connectToServer(socketPath) above indicates: PodResourcesKubeletSocket pointing Kubernetes is a particular node in the cluster on the Unix domain socket file, through this socket can query scheduling to the Pod of this node and its information resources.That is, use gRPC to query Pod resource information from kubelet.

Because the architecture of the koordinator framework allocates GPU devices centrally on the central side, rather than by Kubelet end-to-side, kubelet does not save Pod and GPU allocation information. Therefore, the logic of DCGM querying pod and GPU devices needs to be adjusted, from the previous query kubelet to read the file podlist.json to query. Since dcgm is deployed in DaemonSet, the dcgm component on each node must be mounted to the directory of the host machine: /var/lib/kubelet/pod-resources/podlist.json, so that the file can be accessed directly from inside the dcgm container.The architecture diagram of dcgm is as follows:

<img src="/docs/images/dcgm-gpuhook-readwrite.png" style="zoom:25%;" />

Defines a function that reads a json file to construct a pod and gpu mapping.

```go
func (p *PodMapper) listPods(string filePath) ([]PodInfo, error) {}
```

Define another function to handle the listPods return value: the PodInfo array. Walk through the set, converted to map[string]PodInfo

```go
func (p *PodMapper) toDeviceToPod([]PodInfo, sysInfo SystemInfo) map[string]PodInfo {}
```

Finally, the function Process of dcgm is changed to:

```
#dcgm\pkg\dcgmexporter\kubernetes.go

func (p *PodMapper) Process(metrics MetricsByCounter, sysInfo SystemInfo) error {
socketPath := p.Config.PodResourcesKubeletSocket
_, err := os.Stat(socketPath)
if os.IsNotExist(err) {
logrus.Info("No Kubelet socket, ignoring")
return nil
}

c, cleanup, err := connectToServer(socketPath)
if err != nil {
return err
}
defer cleanup()

pods, err := p.listPods(/podlist.json)
if err != nil {
return err
}

deviceToPod := p.toDeviceToPod(pods, sysInfo)

......

return nil
}
```

Loading