Refactor netlink implementation.

This allows us to plug in a scheddebug based interface.
This commit is contained in:
Rohit Jnagal 2015-01-23 23:03:41 +00:00
parent 2c1da9e1be
commit 1375f451b2
8 changed files with 111 additions and 59 deletions

View File

@ -44,7 +44,7 @@ type containerData struct {
info containerInfo info containerInfo
storageDriver storage.StorageDriver storageDriver storage.StorageDriver
lock sync.Mutex lock sync.Mutex
loadReader *cpuload.CpuLoadReader loadReader cpuload.CpuLoadReader
housekeepingInterval time.Duration housekeepingInterval time.Duration
lastUpdatedTime time.Time lastUpdatedTime time.Time
lastErrorTime time.Time lastErrorTime time.Time
@ -93,7 +93,7 @@ func (c *containerData) GetInfo() (*containerInfo, error) {
return &c.info, nil return &c.info, nil
} }
func newContainerData(containerName string, driver storage.StorageDriver, handler container.ContainerHandler, loadReader *cpuload.CpuLoadReader, logUsage bool) (*containerData, error) { func newContainerData(containerName string, driver storage.StorageDriver, handler container.ContainerHandler, loadReader cpuload.CpuLoadReader, logUsage bool) (*containerData, error) {
if driver == nil { if driver == nil {
return nil, fmt.Errorf("nil storage driver") return nil, fmt.Errorf("nil storage driver")
} }

View File

@ -120,7 +120,7 @@ type manager struct {
quitChannels []chan error quitChannels []chan error
cadvisorContainer string cadvisorContainer string
dockerContainersRegexp *regexp.Regexp dockerContainersRegexp *regexp.Regexp
loadReader *cpuload.CpuLoadReader loadReader cpuload.CpuLoadReader
} }
// Start the container manager. // Start the container manager.
@ -129,9 +129,14 @@ func (self *manager) Start() error {
cpuLoadReader, err := cpuload.New() cpuLoadReader, err := cpuload.New()
if err != nil { if err != nil {
// TODO(rjnagal): Promote to warning once we support cpu load inside namespaces. // TODO(rjnagal): Promote to warning once we support cpu load inside namespaces.
glog.Infof("could not initialize cpu load reader: %s", err) glog.Infof("Could not initialize cpu load reader: %s", err)
} else { } else {
self.loadReader = cpuLoadReader err = cpuLoadReader.Start()
if err != nil {
glog.Warning("Could not start cpu load stat collector: %s", err)
} else {
self.loadReader = cpuLoadReader
}
} }
// Create root and then recover all containers. // Create root and then recover all containers.
@ -176,7 +181,7 @@ func (self *manager) Stop() error {
} }
self.quitChannels = make([]chan error, 0, 2) self.quitChannels = make([]chan error, 0, 2)
if self.loadReader != nil { if self.loadReader != nil {
self.loadReader.Close() self.loadReader.Stop()
self.loadReader = nil self.loadReader = nil
} }
return nil return nil

View File

@ -16,58 +16,27 @@ package cpuload
import ( import (
"fmt" "fmt"
"os"
"github.com/golang/glog"
"github.com/google/cadvisor/info" "github.com/google/cadvisor/info"
"github.com/google/cadvisor/utils/cpuload/netlink"
) )
type CpuLoadReader struct { type CpuLoadReader interface {
familyId uint16 // Start the reader.
conn *Connection Start() error
// Stop the reader and clean up internal state.
Stop()
// Retrieve Cpu load for a given group.
// Path is an absolute filesystem path for a container under CPU cgroup hierarchy.
GetCpuLoad(path string) (info.LoadStats, error)
} }
func New() (*CpuLoadReader, error) { func New() (CpuLoadReader, error) {
conn, err := newConnection() reader, err := netlink.New()
if err != nil { if err != nil {
return nil, fmt.Errorf("failed to create a new connection: %s", err) return nil, fmt.Errorf("failed to create a netlink-based cpu load reader: %s", err)
} }
return reader, nil
id, err := getFamilyId(conn)
if err != nil {
return nil, fmt.Errorf("failed to get netlink family id for task stats: %s", err)
}
glog.V(2).Infof("Family id for taskstats: %d", id)
return &CpuLoadReader{
familyId: id,
conn: conn,
}, nil
}
func (self *CpuLoadReader) Close() {
if self.conn != nil {
self.conn.Close()
}
}
// Returns instantaneous number of running tasks in a group.
// Caller can use historical data to calculate cpu load.
// path is an absolute filesystem path for a container under the CPU cgroup hierarchy.
// NOTE: non-hierarchical load is returned. It does not include load for subcontainers.
func (self *CpuLoadReader) GetCpuLoad(path string) (info.LoadStats, error) {
if len(path) == 0 {
return info.LoadStats{}, fmt.Errorf("cgroup path can not be empty!")
}
cfd, err := os.Open(path)
if err != nil {
return info.LoadStats{}, fmt.Errorf("failed to open cgroup path %s: %q", path, err)
}
stats, err := getLoadStats(self.familyId, cfd.Fd(), self.conn)
if err != nil {
return info.LoadStats{}, err
}
glog.V(1).Infof("Task stats for %q: %+v", path, stats)
return stats, nil
} }

View File

@ -12,7 +12,7 @@
// See the License for the specific language governing permissions and // See the License for the specific language governing permissions and
// limitations under the License. // limitations under the License.
package cpuload package netlink
import ( import (
"bufio" "bufio"

View File

@ -12,7 +12,7 @@
// See the License for the specific language governing permissions and // See the License for the specific language governing permissions and
// limitations under the License. // limitations under the License.
package cpuload package netlink
/* /*
#include <linux/taskstats.h> #include <linux/taskstats.h>

View File

@ -17,20 +17,20 @@ package main
import ( import (
"log" "log"
"github.com/google/cadvisor/utils/cpuload" "github.com/google/cadvisor/utils/cpuload/netlink"
) )
func main() { func main() {
c, err := cpuload.New() n, err := netlink.New()
if err != nil { if err != nil {
log.Printf("Failed to create cpu load util: %s", err) log.Printf("Failed to create cpu load util: %s", err)
return return
} }
defer c.Close() defer n.Stop()
paths := []string{"/sys/fs/cgroup/cpu", "/sys/fs/cgroup/cpu/docker"} paths := []string{"/sys/fs/cgroup/cpu", "/sys/fs/cgroup/cpu/docker"}
for _, path := range paths { for _, path := range paths {
stats, err := c.GetCpuLoad(path) stats, err := n.GetCpuLoad(path)
if err != nil { if err != nil {
log.Printf("Error getting cpu load for %q: %s", path, err) log.Printf("Error getting cpu load for %q: %s", path, err)
} }

View File

@ -12,7 +12,7 @@
// See the License for the specific language governing permissions and // See the License for the specific language governing permissions and
// limitations under the License. // limitations under the License.
package cpuload package netlink
import ( import (
"bytes" "bytes"

View File

@ -0,0 +1,78 @@
// Copyright 2015 Google Inc. 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 netlink
import (
"fmt"
"os"
"github.com/golang/glog"
"github.com/google/cadvisor/info"
)
type NetlinkReader struct {
familyId uint16
conn *Connection
}
func New() (*NetlinkReader, error) {
conn, err := newConnection()
if err != nil {
return nil, fmt.Errorf("failed to create a new connection: %s", err)
}
id, err := getFamilyId(conn)
if err != nil {
return nil, fmt.Errorf("failed to get netlink family id for task stats: %s", err)
}
glog.V(2).Infof("Family id for taskstats: %d", id)
return &NetlinkReader{
familyId: id,
conn: conn,
}, nil
}
func (self *NetlinkReader) Stop() {
if self.conn != nil {
self.conn.Close()
}
}
func (self *NetlinkReader) Start() error {
// We do the start setup for netlink in New(). Nothing to do here.
return nil
}
// Returns instantaneous number of running tasks in a group.
// Caller can use historical data to calculate cpu load.
// path is an absolute filesystem path for a container under the CPU cgroup hierarchy.
// NOTE: non-hierarchical load is returned. It does not include load for subcontainers.
func (self *NetlinkReader) GetCpuLoad(path string) (info.LoadStats, error) {
if len(path) == 0 {
return info.LoadStats{}, fmt.Errorf("cgroup path can not be empty!")
}
cfd, err := os.Open(path)
if err != nil {
return info.LoadStats{}, fmt.Errorf("failed to open cgroup path %s: %q", path, err)
}
stats, err := getLoadStats(self.familyId, cfd.Fd(), self.conn)
if err != nil {
return info.LoadStats{}, err
}
glog.V(1).Infof("Task stats for %q: %+v", path, stats)
return stats, nil
}