mirror of
https://github.com/minio/minio.git
synced 2025-01-03 19:13:22 -05:00
30c2596512
Currently, we read from `/proc/diskstats` which is found to be un-reliable in k8s environments. We can read from `sysfs` instead. Also, cache the latest drive io stats to find the diff and update the metrics.
174 lines
4.8 KiB
Go
174 lines
4.8 KiB
Go
//go:build linux && !s390x && !arm && !386
|
|
// +build linux,!s390x,!arm,!386
|
|
|
|
// Copyright (c) 2015-2023 MinIO, Inc.
|
|
//
|
|
// This file is part of MinIO Object Storage stack
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Affero General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package disk
|
|
|
|
import (
|
|
"bufio"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
"syscall"
|
|
|
|
"github.com/prometheus/procfs/blockdevice"
|
|
"golang.org/x/sys/unix"
|
|
)
|
|
|
|
// GetInfo returns total and free bytes available in a directory, e.g. `/`.
|
|
func GetInfo(path string, firstTime bool) (info Info, err error) {
|
|
s := syscall.Statfs_t{}
|
|
err = syscall.Statfs(path, &s)
|
|
if err != nil {
|
|
return Info{}, err
|
|
}
|
|
reservedBlocks := s.Bfree - s.Bavail
|
|
info = Info{
|
|
Total: uint64(s.Frsize) * (s.Blocks - reservedBlocks),
|
|
Free: uint64(s.Frsize) * s.Bavail,
|
|
Files: s.Files,
|
|
Ffree: s.Ffree,
|
|
//nolint:unconvert
|
|
FSType: getFSType(int64(s.Type)),
|
|
}
|
|
|
|
st := syscall.Stat_t{}
|
|
err = syscall.Stat(path, &st)
|
|
if err != nil {
|
|
return Info{}, err
|
|
}
|
|
//nolint:unconvert
|
|
devID := uint64(st.Dev) // Needed to support multiple GOARCHs
|
|
info.Major = unix.Major(devID)
|
|
info.Minor = unix.Minor(devID)
|
|
|
|
// Check for overflows.
|
|
// https://github.com/minio/minio/issues/8035
|
|
// XFS can show wrong values at times error out
|
|
// in such scenarios.
|
|
if info.Free > info.Total {
|
|
return info, fmt.Errorf("detected free space (%d) > total drive space (%d), fs corruption at (%s). please run 'fsck'", info.Free, info.Total, path)
|
|
}
|
|
info.Used = info.Total - info.Free
|
|
|
|
if firstTime {
|
|
bfs, err := blockdevice.NewDefaultFS()
|
|
if err == nil {
|
|
devName := ""
|
|
diskstats, _ := bfs.ProcDiskstats()
|
|
for _, dstat := range diskstats {
|
|
// ignore all loop devices
|
|
if strings.HasPrefix(dstat.DeviceName, "loop") {
|
|
continue
|
|
}
|
|
if dstat.MajorNumber == info.Major && dstat.MinorNumber == info.Minor {
|
|
devName = dstat.DeviceName
|
|
break
|
|
}
|
|
}
|
|
if devName != "" {
|
|
info.Name = devName
|
|
qst, err := bfs.SysBlockDeviceQueueStats(devName)
|
|
if err != nil { // Mostly not found error
|
|
// Check if there is a parent device:
|
|
// e.g. if the mount is based on /dev/nvme0n1p1, let's calculate the
|
|
// real device name (nvme0n1) to get its sysfs information
|
|
parentDevPath, e := os.Readlink("/sys/class/block/" + devName)
|
|
if e == nil {
|
|
parentDev := filepath.Base(filepath.Dir(parentDevPath))
|
|
qst, err = bfs.SysBlockDeviceQueueStats(parentDev)
|
|
}
|
|
}
|
|
if err == nil {
|
|
info.NRRequests = qst.NRRequests
|
|
rot := qst.Rotational == 1 // Rotational is '1' if the device is HDD
|
|
info.Rotational = &rot
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return info, nil
|
|
}
|
|
|
|
// GetDriveStats returns IO stats of the drive by its major:minor
|
|
func GetDriveStats(major, minor uint32) (iostats IOStats, err error) {
|
|
return readDriveStats(fmt.Sprintf("/sys/dev/block/%v:%v/stat", major, minor))
|
|
}
|
|
|
|
func readDriveStats(statsFile string) (iostats IOStats, err error) {
|
|
stats, err := readStat(statsFile)
|
|
if err != nil {
|
|
return IOStats{}, err
|
|
}
|
|
if len(stats) < 11 {
|
|
return IOStats{}, fmt.Errorf("found invalid format while reading %v", statsFile)
|
|
}
|
|
// refer https://www.kernel.org/doc/Documentation/block/stat.txt
|
|
iostats = IOStats{
|
|
ReadIOs: stats[0],
|
|
ReadMerges: stats[1],
|
|
ReadSectors: stats[2],
|
|
ReadTicks: stats[3],
|
|
WriteIOs: stats[4],
|
|
WriteMerges: stats[5],
|
|
WriteSectors: stats[6],
|
|
WriteTicks: stats[7],
|
|
CurrentIOs: stats[8],
|
|
TotalTicks: stats[9],
|
|
ReqTicks: stats[10],
|
|
}
|
|
// as per the doc, only 11 fields are guaranteed
|
|
// only set if available
|
|
if len(stats) > 14 {
|
|
iostats.DiscardIOs = stats[11]
|
|
iostats.DiscardMerges = stats[12]
|
|
iostats.DiscardSectors = stats[13]
|
|
iostats.DiscardTicks = stats[14]
|
|
}
|
|
return
|
|
}
|
|
|
|
func readStat(fileName string) (stats []uint64, err error) {
|
|
file, err := os.Open(fileName)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer file.Close()
|
|
|
|
s, err := bufio.NewReader(file).ReadString('\n')
|
|
if err != nil && !errors.Is(err, io.EOF) {
|
|
return nil, err
|
|
}
|
|
statLine := strings.TrimSpace(s)
|
|
for _, token := range strings.Fields(statLine) {
|
|
ui64, err := strconv.ParseUint(token, 10, 64)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
stats = append(stats, ui64)
|
|
}
|
|
|
|
return stats, nil
|
|
}
|