2024-02-17 03:48:29 +00:00
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
2021-12-22 21:54:41 +00:00
|
|
|
//go:build openbsd
|
2016-11-21 23:18:52 +01:00
|
|
|
|
|
|
|
package disk
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2017-12-31 15:25:49 +09:00
|
|
|
"context"
|
2016-11-21 23:18:52 +01:00
|
|
|
"encoding/binary"
|
|
|
|
|
2024-02-17 03:48:29 +00:00
|
|
|
"github.com/shirou/gopsutil/v4/internal/common"
|
2017-06-02 13:51:00 -07:00
|
|
|
"golang.org/x/sys/unix"
|
2016-11-21 23:18:52 +01:00
|
|
|
)
|
|
|
|
|
2017-12-31 15:25:49 +09:00
|
|
|
func PartitionsWithContext(ctx context.Context, all bool) ([]PartitionStat, error) {
|
2016-11-21 23:18:52 +01:00
|
|
|
var ret []PartitionStat
|
|
|
|
|
|
|
|
// get length
|
2019-12-23 13:25:57 +01:00
|
|
|
count, err := unix.Getfsstat(nil, unix.MNT_WAIT)
|
2016-11-21 23:18:52 +01:00
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
|
|
|
|
2020-01-08 09:39:05 +00:00
|
|
|
fs := make([]unix.Statfs_t, count)
|
|
|
|
if _, err = unix.Getfsstat(fs, unix.MNT_WAIT); err != nil {
|
2018-06-21 16:48:16 +02:00
|
|
|
return ret, err
|
|
|
|
}
|
2016-11-21 23:18:52 +01:00
|
|
|
|
|
|
|
for _, stat := range fs {
|
2021-11-06 09:53:56 +00:00
|
|
|
opts := []string{"rw"}
|
2019-12-23 13:25:57 +01:00
|
|
|
if stat.F_flags&unix.MNT_RDONLY != 0 {
|
2021-11-06 09:53:56 +00:00
|
|
|
opts = []string{"rw"}
|
2016-11-21 23:18:52 +01:00
|
|
|
}
|
2019-12-23 13:25:57 +01:00
|
|
|
if stat.F_flags&unix.MNT_SYNCHRONOUS != 0 {
|
2021-11-06 09:53:56 +00:00
|
|
|
opts = append(opts, "sync")
|
2016-11-21 23:18:52 +01:00
|
|
|
}
|
2019-12-23 13:25:57 +01:00
|
|
|
if stat.F_flags&unix.MNT_NOEXEC != 0 {
|
2021-11-06 09:53:56 +00:00
|
|
|
opts = append(opts, "noexec")
|
2016-11-21 23:18:52 +01:00
|
|
|
}
|
2019-12-23 13:25:57 +01:00
|
|
|
if stat.F_flags&unix.MNT_NOSUID != 0 {
|
2021-11-06 09:53:56 +00:00
|
|
|
opts = append(opts, "nosuid")
|
2016-11-21 23:18:52 +01:00
|
|
|
}
|
2019-12-23 13:25:57 +01:00
|
|
|
if stat.F_flags&unix.MNT_NODEV != 0 {
|
2021-11-06 09:53:56 +00:00
|
|
|
opts = append(opts, "nodev")
|
2016-11-21 23:18:52 +01:00
|
|
|
}
|
2019-12-23 13:25:57 +01:00
|
|
|
if stat.F_flags&unix.MNT_ASYNC != 0 {
|
2021-11-06 09:53:56 +00:00
|
|
|
opts = append(opts, "async")
|
2016-11-21 23:18:52 +01:00
|
|
|
}
|
2019-12-23 13:42:44 +01:00
|
|
|
if stat.F_flags&unix.MNT_SOFTDEP != 0 {
|
2021-11-06 09:53:56 +00:00
|
|
|
opts = append(opts, "softdep")
|
2019-12-23 13:42:44 +01:00
|
|
|
}
|
|
|
|
if stat.F_flags&unix.MNT_NOATIME != 0 {
|
2021-11-06 09:53:56 +00:00
|
|
|
opts = append(opts, "noatime")
|
2019-12-23 13:42:44 +01:00
|
|
|
}
|
|
|
|
if stat.F_flags&unix.MNT_WXALLOWED != 0 {
|
2021-11-06 09:53:56 +00:00
|
|
|
opts = append(opts, "wxallowed")
|
2019-12-23 13:42:44 +01:00
|
|
|
}
|
2016-11-21 23:18:52 +01:00
|
|
|
|
|
|
|
d := PartitionStat{
|
2022-06-06 14:05:41 -04:00
|
|
|
Device: common.ByteToString(stat.F_mntfromname[:]),
|
|
|
|
Mountpoint: common.ByteToString(stat.F_mntonname[:]),
|
|
|
|
Fstype: common.ByteToString(stat.F_fstypename[:]),
|
2016-11-21 23:18:52 +01:00
|
|
|
Opts: opts,
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = append(ret, d)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
|
2017-12-31 15:25:49 +09:00
|
|
|
func IOCountersWithContext(ctx context.Context, names ...string) (map[string]IOCountersStat, error) {
|
2016-11-21 23:18:52 +01:00
|
|
|
ret := make(map[string]IOCountersStat)
|
|
|
|
|
2017-11-08 20:52:36 +01:00
|
|
|
r, err := unix.SysctlRaw("hw.diskstats")
|
2016-11-21 23:18:52 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
buf := []byte(r)
|
|
|
|
length := len(buf)
|
|
|
|
|
|
|
|
count := int(uint64(length) / uint64(sizeOfDiskstats))
|
|
|
|
|
|
|
|
// parse buf to Diskstats
|
|
|
|
for i := 0; i < count; i++ {
|
|
|
|
b := buf[i*sizeOfDiskstats : i*sizeOfDiskstats+sizeOfDiskstats]
|
|
|
|
d, err := parseDiskstats(b)
|
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
name := common.IntToString(d.Name[:])
|
|
|
|
|
2017-04-06 17:54:50 -07:00
|
|
|
if len(names) > 0 && !common.StringsHas(names, name) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2016-11-21 23:18:52 +01:00
|
|
|
ds := IOCountersStat{
|
|
|
|
ReadCount: d.Rxfer,
|
|
|
|
WriteCount: d.Wxfer,
|
|
|
|
ReadBytes: d.Rbytes,
|
|
|
|
WriteBytes: d.Wbytes,
|
|
|
|
Name: name,
|
|
|
|
}
|
|
|
|
ret[name] = ds
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// BT2LD(time) ((long double)(time).sec + (time).frac * BINTIME_SCALE)
|
|
|
|
|
|
|
|
func parseDiskstats(buf []byte) (Diskstats, error) {
|
|
|
|
var ds Diskstats
|
|
|
|
br := bytes.NewReader(buf)
|
|
|
|
// err := binary.Read(br, binary.LittleEndian, &ds)
|
|
|
|
err := common.Read(br, binary.LittleEndian, &ds)
|
|
|
|
if err != nil {
|
|
|
|
return ds, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return ds, nil
|
|
|
|
}
|
|
|
|
|
2017-12-31 15:25:49 +09:00
|
|
|
func UsageWithContext(ctx context.Context, path string) (*UsageStat, error) {
|
2017-06-02 13:51:00 -07:00
|
|
|
stat := unix.Statfs_t{}
|
|
|
|
err := unix.Statfs(path, &stat)
|
2016-11-21 23:18:52 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
bsize := stat.F_bsize
|
|
|
|
|
|
|
|
ret := &UsageStat{
|
|
|
|
Path: path,
|
|
|
|
Fstype: getFsType(stat),
|
|
|
|
Total: (uint64(stat.F_blocks) * uint64(bsize)),
|
|
|
|
Free: (uint64(stat.F_bavail) * uint64(bsize)),
|
|
|
|
InodesTotal: (uint64(stat.F_files)),
|
|
|
|
InodesFree: (uint64(stat.F_ffree)),
|
|
|
|
}
|
|
|
|
|
|
|
|
ret.InodesUsed = (ret.InodesTotal - ret.InodesFree)
|
|
|
|
ret.InodesUsedPercent = (float64(ret.InodesUsed) / float64(ret.InodesTotal)) * 100.0
|
|
|
|
ret.Used = (uint64(stat.F_blocks) - uint64(stat.F_bfree)) * uint64(bsize)
|
|
|
|
ret.UsedPercent = (float64(ret.Used) / float64(ret.Total)) * 100.0
|
|
|
|
|
|
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
|
2017-06-02 13:51:00 -07:00
|
|
|
func getFsType(stat unix.Statfs_t) string {
|
2022-06-06 14:05:41 -04:00
|
|
|
return common.ByteToString(stat.F_fstypename[:])
|
2016-11-21 23:18:52 +01:00
|
|
|
}
|
2021-11-06 09:53:56 +00:00
|
|
|
|
|
|
|
func SerialNumberWithContext(ctx context.Context, name string) (string, error) {
|
|
|
|
return "", common.ErrNotImplementedError
|
|
|
|
}
|
|
|
|
|
|
|
|
func LabelWithContext(ctx context.Context, name string) (string, error) {
|
|
|
|
return "", common.ErrNotImplementedError
|
|
|
|
}
|