prometheusmetricshost-metricsmachine-metricsnode-metricsprocfsprometheus-exportersystem-informationsystem-metrics
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
120 lines
2.8 KiB
120 lines
2.8 KiB
// Copyright 2020 The Prometheus Authors |
|
// 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. |
|
|
|
//go:build !nointerrupts |
|
// +build !nointerrupts |
|
|
|
package collector |
|
|
|
import ( |
|
"fmt" |
|
"strconv" |
|
"unsafe" |
|
|
|
"github.com/prometheus/client_golang/prometheus" |
|
"golang.org/x/sys/unix" |
|
) |
|
|
|
const ( |
|
KERN_INTRCNT = 63 |
|
KERN_INTRCNT_NUM = 1 |
|
KERN_INTRCNT_CNT = 2 |
|
KERN_INTRCNT_NAME = 3 |
|
KERN_INTRCNT_VECTOR = 4 |
|
) |
|
|
|
func nintr() _C_int { |
|
mib := [3]_C_int{unix.CTL_KERN, KERN_INTRCNT, KERN_INTRCNT_NUM} |
|
buf, err := sysctl(mib[:]) |
|
if err != nil { |
|
return 0 |
|
} |
|
return *(*_C_int)(unsafe.Pointer(&buf[0])) |
|
} |
|
|
|
func intr(idx _C_int) (itr interrupt, err error) { |
|
mib := [4]_C_int{unix.CTL_KERN, KERN_INTRCNT, KERN_INTRCNT_NAME, idx} |
|
buf, err := sysctl(mib[:]) |
|
if err != nil { |
|
return |
|
} |
|
dev := *(*[128]byte)(unsafe.Pointer(&buf[0])) |
|
itr.device = string(dev[:]) |
|
|
|
mib[2] = KERN_INTRCNT_VECTOR |
|
buf, err = sysctl(mib[:]) |
|
if err != nil { |
|
return |
|
} |
|
itr.vector = *(*int)(unsafe.Pointer(&buf[0])) |
|
|
|
mib[2] = KERN_INTRCNT_CNT |
|
buf, err = sysctl(mib[:]) |
|
if err != nil { |
|
return |
|
} |
|
count := *(*uint64)(unsafe.Pointer(&buf[0])) |
|
itr.values = []float64{float64(count)} |
|
return |
|
} |
|
|
|
var interruptLabelNames = []string{"cpu", "type", "devices"} |
|
|
|
func (c *interruptsCollector) Update(ch chan<- prometheus.Metric) error { |
|
interrupts, err := getInterrupts() |
|
if err != nil { |
|
return fmt.Errorf("couldn't get interrupts: %s", err) |
|
} |
|
for dev, interrupt := range interrupts { |
|
for cpuNo, value := range interrupt.values { |
|
interruptType := fmt.Sprintf("%d", interrupt.vector) |
|
filterName := interruptType + ";" + dev |
|
if c.nameFilter.ignored(filterName) { |
|
c.logger.Debug("ignoring interrupt name", "filter_name", filterName) |
|
continue |
|
} |
|
if !c.includeZeros && value == 0.0 { |
|
c.logger.Debug("ignoring interrupt with zero value", "filter_name", filterName, "cpu", cpuNo) |
|
continue |
|
} |
|
ch <- c.desc.mustNewConstMetric( |
|
value, |
|
strconv.Itoa(cpuNo), |
|
interruptType, |
|
dev, |
|
) |
|
} |
|
} |
|
return nil |
|
} |
|
|
|
type interrupt struct { |
|
vector int |
|
device string |
|
values []float64 |
|
} |
|
|
|
func getInterrupts() (map[string]interrupt, error) { |
|
var interrupts = map[string]interrupt{} |
|
n := nintr() |
|
|
|
for i := _C_int(0); i < n; i++ { |
|
itr, err := intr(i) |
|
if err != nil { |
|
return nil, err |
|
} |
|
interrupts[itr.device] = itr |
|
} |
|
|
|
return interrupts, nil |
|
}
|
|
|