1
0
mirror of https://github.com/shirou/gopsutil.git synced 2025-04-26 13:48:59 +08:00
shirou_gopsutil/disk/disk_freebsd.go

163 lines
3.7 KiB
Go
Raw Normal View History

2014-04-22 19:25:28 +09:00
// +build freebsd
2014-12-30 22:09:05 +09:00
package disk
2014-04-22 19:25:28 +09:00
import (
"bytes"
2017-12-31 15:25:49 +09:00
"context"
"encoding/binary"
"strconv"
"golang.org/x/sys/unix"
2015-10-20 00:04:57 +09:00
"github.com/shirou/gopsutil/internal/common"
2014-04-22 19:25:28 +09:00
)
// PartitionsWithContext returns disk partition.
// 'all' argument is ignored, see: https://github.com/giampaolo/psutil/issues/906
2017-12-31 15:25:49 +09:00
func PartitionsWithContext(ctx context.Context, all bool) ([]PartitionStat, error) {
var ret []PartitionStat
2014-04-22 19:25:28 +09:00
// get length
count, err := unix.Getfsstat(nil, unix.MNT_WAIT)
2014-04-22 19:25:28 +09:00
if err != nil {
return ret, err
}
fs := make([]unix.Statfs_t, count)
if _, err = unix.Getfsstat(fs, unix.MNT_WAIT); err != nil {
return ret, err
}
2014-04-22 19:25:28 +09:00
for _, stat := range fs {
opts := "rw"
if stat.Flags&unix.MNT_RDONLY != 0 {
2014-04-22 19:25:28 +09:00
opts = "ro"
}
if stat.Flags&unix.MNT_SYNCHRONOUS != 0 {
2014-04-22 19:25:28 +09:00
opts += ",sync"
}
if stat.Flags&unix.MNT_NOEXEC != 0 {
2014-04-22 19:25:28 +09:00
opts += ",noexec"
}
if stat.Flags&unix.MNT_NOSUID != 0 {
2014-04-22 19:25:28 +09:00
opts += ",nosuid"
}
if stat.Flags&unix.MNT_UNION != 0 {
2014-04-22 19:25:28 +09:00
opts += ",union"
}
if stat.Flags&unix.MNT_ASYNC != 0 {
2014-04-22 19:25:28 +09:00
opts += ",async"
}
if stat.Flags&unix.MNT_SUIDDIR != 0 {
2014-04-22 19:25:28 +09:00
opts += ",suiddir"
}
if stat.Flags&unix.MNT_SOFTDEP != 0 {
2014-04-22 19:25:28 +09:00
opts += ",softdep"
}
if stat.Flags&unix.MNT_NOSYMFOLLOW != 0 {
2014-04-22 19:25:28 +09:00
opts += ",nosymfollow"
}
if stat.Flags&unix.MNT_GJOURNAL != 0 {
opts += ",gjournal"
2014-04-22 19:25:28 +09:00
}
if stat.Flags&unix.MNT_MULTILABEL != 0 {
2014-04-22 19:25:28 +09:00
opts += ",multilabel"
}
if stat.Flags&unix.MNT_ACLS != 0 {
2014-04-22 19:25:28 +09:00
opts += ",acls"
}
if stat.Flags&unix.MNT_NOATIME != 0 {
opts += ",noatime"
2014-04-22 19:25:28 +09:00
}
if stat.Flags&unix.MNT_NOCLUSTERR != 0 {
opts += ",noclusterr"
2014-04-22 19:25:28 +09:00
}
if stat.Flags&unix.MNT_NOCLUSTERW != 0 {
2014-04-22 19:25:28 +09:00
opts += ",noclusterw"
}
if stat.Flags&unix.MNT_NFS4ACLS != 0 {
opts += ",nfsv4acls"
2014-04-22 19:25:28 +09:00
}
d := PartitionStat{
Device: common.ByteToString(stat.Mntfromname[:]),
Mountpoint: common.ByteToString(stat.Mntonname[:]),
Fstype: common.ByteToString(stat.Fstypename[:]),
2014-04-22 19:25:28 +09:00
Opts: opts,
}
2014-04-22 19:25:28 +09:00
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) {
2014-06-02 23:50:55 +09:00
// statinfo->devinfo->devstat
2014-06-10 14:35:59 +09:00
// /usr/include/devinfo.h
ret := make(map[string]IOCountersStat)
r, err := unix.Sysctl("kern.devstat.all")
2014-06-02 23:50:55 +09:00
if err != nil {
return nil, err
}
buf := []byte(r)
length := len(buf)
2014-06-02 23:50:55 +09:00
count := int(uint64(length) / uint64(sizeOfDevstat))
buf = buf[8:] // devstat.all has version in the head.
// parse buf to Devstat
for i := 0; i < count; i++ {
b := buf[i*sizeOfDevstat : i*sizeOfDevstat+sizeOfDevstat]
d, err := parseDevstat(b)
if err != nil {
continue
}
un := strconv.Itoa(int(d.Unit_number))
name := common.IntToString(d.Device_name[:]) + un
if len(names) > 0 && !common.StringsHas(names, name) {
continue
}
ds := IOCountersStat{
ReadCount: d.Operations[DEVSTAT_READ],
WriteCount: d.Operations[DEVSTAT_WRITE],
ReadBytes: d.Bytes[DEVSTAT_READ],
WriteBytes: d.Bytes[DEVSTAT_WRITE],
ReadTime: uint64(d.Duration[DEVSTAT_READ].Compute() * 1000),
WriteTime: uint64(d.Duration[DEVSTAT_WRITE].Compute() * 1000),
IoTime: uint64(d.Busy_time.Compute() * 1000),
2014-06-02 23:50:55 +09:00
Name: name,
}
ret[name] = ds
2014-06-02 23:50:55 +09:00
}
return ret, nil
2014-04-29 14:59:22 +09:00
}
func (b Bintime) Compute() float64 {
BINTIME_SCALE := 5.42101086242752217003726400434970855712890625e-20
return float64(b.Sec) + float64(b.Frac)*BINTIME_SCALE
}
// BT2LD(time) ((long double)(time).sec + (time).frac * BINTIME_SCALE)
func parseDevstat(buf []byte) (Devstat, error) {
var ds Devstat
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
}
func getFsType(stat unix.Statfs_t) string {
return common.ByteToString(stat.Fstypename[:])
}