2021-08-18 18:35:22 -07:00
|
|
|
//go:build linux && !s390x && !arm && !386
|
2020-08-24 12:11:20 -07:00
|
|
|
// +build linux,!s390x,!arm,!386
|
2015-10-17 16:45:42 -07:00
|
|
|
|
2023-07-24 09:30:19 -07:00
|
|
|
// Copyright (c) 2015-2023 MinIO, Inc.
|
2021-04-18 12:41:13 -07:00
|
|
|
//
|
|
|
|
// 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/>.
|
2015-10-17 16:45:42 -07:00
|
|
|
|
|
|
|
package disk
|
|
|
|
|
|
|
|
import (
|
2022-08-16 15:13:49 +01:00
|
|
|
"bufio"
|
2019-08-13 02:58:43 -07:00
|
|
|
"fmt"
|
2022-08-16 15:13:49 +01:00
|
|
|
"os"
|
2023-09-06 20:37:57 +01:00
|
|
|
"path/filepath"
|
2022-08-16 15:13:49 +01:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
2015-10-17 16:45:42 -07:00
|
|
|
"syscall"
|
2022-08-16 15:13:49 +01:00
|
|
|
|
2023-07-24 09:30:19 -07:00
|
|
|
"github.com/prometheus/procfs/blockdevice"
|
2022-08-16 15:13:49 +01:00
|
|
|
"golang.org/x/sys/unix"
|
2015-10-17 16:45:42 -07:00
|
|
|
)
|
|
|
|
|
2016-01-24 23:03:38 -08:00
|
|
|
// GetInfo returns total and free bytes available in a directory, e.g. `/`.
|
2023-07-31 15:20:48 -07:00
|
|
|
func GetInfo(path string, firstTime bool) (info Info, err error) {
|
2015-10-17 16:45:42 -07:00
|
|
|
s := syscall.Statfs_t{}
|
|
|
|
err = syscall.Statfs(path, &s)
|
|
|
|
if err != nil {
|
2016-01-24 23:03:38 -08:00
|
|
|
return Info{}, err
|
2015-10-17 16:45:42 -07:00
|
|
|
}
|
2020-08-24 12:11:20 -07:00
|
|
|
reservedBlocks := s.Bfree - s.Bavail
|
2017-09-26 03:46:19 +02:00
|
|
|
info = Info{
|
2021-05-24 09:28:19 -07:00
|
|
|
Total: uint64(s.Frsize) * (s.Blocks - reservedBlocks),
|
|
|
|
Free: uint64(s.Frsize) * s.Bavail,
|
|
|
|
Files: s.Files,
|
|
|
|
Ffree: s.Ffree,
|
|
|
|
//nolint:unconvert
|
2021-03-15 06:02:09 -07:00
|
|
|
FSType: getFSType(int64(s.Type)),
|
2017-09-26 03:46:19 +02:00
|
|
|
}
|
2022-08-16 15:13:49 +01:00
|
|
|
|
|
|
|
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)
|
2023-07-24 09:30:19 -07:00
|
|
|
|
|
|
|
// 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
|
|
|
|
|
2023-07-31 15:20:48 -07:00
|
|
|
if firstTime {
|
|
|
|
bfs, err := blockdevice.NewDefaultFS()
|
|
|
|
if err == nil {
|
2023-09-06 20:37:57 +01:00
|
|
|
devName := ""
|
2023-07-31 15:20:48 -07:00
|
|
|
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 {
|
2023-09-06 20:37:57 +01:00
|
|
|
devName = dstat.DeviceName
|
2023-07-31 15:20:48 -07:00
|
|
|
break
|
|
|
|
}
|
2023-07-24 09:30:19 -07:00
|
|
|
}
|
2023-09-06 20:37:57 +01:00
|
|
|
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 {
|
2023-09-11 14:48:54 -07:00
|
|
|
info.NRRequests = qst.NRRequests
|
2023-09-06 20:37:57 +01:00
|
|
|
rot := qst.Rotational == 1 // Rotational is '1' if the device is HDD
|
|
|
|
info.Rotational = &rot
|
|
|
|
}
|
|
|
|
}
|
2023-07-24 09:30:19 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-24 23:03:38 -08:00
|
|
|
return info, nil
|
2015-10-17 16:45:42 -07:00
|
|
|
}
|
2022-08-16 15:13:49 +01:00
|
|
|
|
|
|
|
const (
|
|
|
|
statsPath = "/proc/diskstats"
|
|
|
|
)
|
|
|
|
|
|
|
|
// GetAllDrivesIOStats returns IO stats of all drives found in the machine
|
|
|
|
func GetAllDrivesIOStats() (info AllDrivesIOStats, err error) {
|
|
|
|
proc, err := os.Open(statsPath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer proc.Close()
|
|
|
|
|
|
|
|
ret := make(AllDrivesIOStats)
|
|
|
|
|
|
|
|
sc := bufio.NewScanner(proc)
|
|
|
|
for sc.Scan() {
|
|
|
|
line := sc.Text()
|
|
|
|
fields := strings.Fields(line)
|
|
|
|
if len(fields) < 11 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
var err error
|
|
|
|
var ds IOStats
|
|
|
|
|
|
|
|
ds.ReadIOs, err = strconv.ParseUint((fields[3]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
ds.ReadMerges, err = strconv.ParseUint((fields[4]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
ds.ReadSectors, err = strconv.ParseUint((fields[5]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
ds.ReadTicks, err = strconv.ParseUint((fields[6]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
ds.WriteIOs, err = strconv.ParseUint((fields[7]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
ds.WriteMerges, err = strconv.ParseUint((fields[8]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
ds.WriteSectors, err = strconv.ParseUint((fields[9]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
ds.WriteTicks, err = strconv.ParseUint((fields[10]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(fields) > 11 {
|
|
|
|
ds.CurrentIOs, err = strconv.ParseUint((fields[11]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ds.TotalTicks, err = strconv.ParseUint((fields[12]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
ds.ReqTicks, err = strconv.ParseUint((fields[13]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(fields) > 14 {
|
|
|
|
ds.DiscardIOs, err = strconv.ParseUint((fields[14]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
ds.DiscardMerges, err = strconv.ParseUint((fields[15]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
ds.DiscardSectors, err = strconv.ParseUint((fields[16]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
ds.DiscardTicks, err = strconv.ParseUint((fields[17]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(fields) > 18 {
|
|
|
|
ds.FlushIOs, err = strconv.ParseUint((fields[18]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
ds.FlushTicks, err = strconv.ParseUint((fields[19]), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
major, err := strconv.ParseUint((fields[0]), 10, 32)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
|
|
|
|
minor, err := strconv.ParseUint((fields[1]), 10, 32)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
ret[DevID{uint32(major), uint32(minor)}] = ds
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := sc.Err(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret, nil
|
|
|
|
}
|