| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193 |
- //go:build freebsd
- // +build freebsd
- package disk
- import (
- "bufio"
- "bytes"
- "context"
- "encoding/binary"
- "fmt"
- "strconv"
- "strings"
- "golang.org/x/sys/unix"
- "github.com/shirou/gopsutil/v3/internal/common"
- )
- // PartitionsWithContext returns disk partition.
- // 'all' argument is ignored, see: https://github.com/giampaolo/psutil/issues/906
- func PartitionsWithContext(ctx context.Context, all bool) ([]PartitionStat, error) {
- var ret []PartitionStat
- // get length
- count, err := unix.Getfsstat(nil, unix.MNT_WAIT)
- 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
- }
- for _, stat := range fs {
- opts := []string{"rw"}
- if stat.Flags&unix.MNT_RDONLY != 0 {
- opts = []string{"ro"}
- }
- if stat.Flags&unix.MNT_SYNCHRONOUS != 0 {
- opts = append(opts, "sync")
- }
- if stat.Flags&unix.MNT_NOEXEC != 0 {
- opts = append(opts, "noexec")
- }
- if stat.Flags&unix.MNT_NOSUID != 0 {
- opts = append(opts, "nosuid")
- }
- if stat.Flags&unix.MNT_UNION != 0 {
- opts = append(opts, "union")
- }
- if stat.Flags&unix.MNT_ASYNC != 0 {
- opts = append(opts, "async")
- }
- if stat.Flags&unix.MNT_SUIDDIR != 0 {
- opts = append(opts, "suiddir")
- }
- if stat.Flags&unix.MNT_SOFTDEP != 0 {
- opts = append(opts, "softdep")
- }
- if stat.Flags&unix.MNT_NOSYMFOLLOW != 0 {
- opts = append(opts, "nosymfollow")
- }
- if stat.Flags&unix.MNT_GJOURNAL != 0 {
- opts = append(opts, "gjournal")
- }
- if stat.Flags&unix.MNT_MULTILABEL != 0 {
- opts = append(opts, "multilabel")
- }
- if stat.Flags&unix.MNT_ACLS != 0 {
- opts = append(opts, "acls")
- }
- if stat.Flags&unix.MNT_NOATIME != 0 {
- opts = append(opts, "noatime")
- }
- if stat.Flags&unix.MNT_NOCLUSTERR != 0 {
- opts = append(opts, "noclusterr")
- }
- if stat.Flags&unix.MNT_NOCLUSTERW != 0 {
- opts = append(opts, "noclusterw")
- }
- if stat.Flags&unix.MNT_NFS4ACLS != 0 {
- opts = append(opts, "nfsv4acls")
- }
- d := PartitionStat{
- Device: common.ByteToString(stat.Mntfromname[:]),
- Mountpoint: common.ByteToString(stat.Mntonname[:]),
- Fstype: common.ByteToString(stat.Fstypename[:]),
- Opts: opts,
- }
- ret = append(ret, d)
- }
- return ret, nil
- }
- func IOCountersWithContext(ctx context.Context, names ...string) (map[string]IOCountersStat, error) {
- // statinfo->devinfo->devstat
- // /usr/include/devinfo.h
- ret := make(map[string]IOCountersStat)
- r, err := unix.Sysctl("kern.devstat.all")
- if err != nil {
- return nil, err
- }
- buf := []byte(r)
- length := len(buf)
- 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),
- Name: name,
- }
- ds.SerialNumber, _ = SerialNumberWithContext(ctx, name)
- ret[name] = ds
- }
- return ret, nil
- }
- 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[:])
- }
- func SerialNumberWithContext(ctx context.Context, name string) (string, error) {
- geomOut, err := invoke.CommandWithContext(ctx, "geom", "disk", "list", name)
- if err != nil {
- return "", fmt.Errorf("exec geom: %w", err)
- }
- s := bufio.NewScanner(bytes.NewReader(geomOut))
- serial := ""
- for s.Scan() {
- flds := strings.Fields(s.Text())
- if len(flds) == 2 && flds[0] == "ident:" {
- if flds[1] != "(null)" {
- serial = flds[1]
- }
- break
- }
- }
- if err = s.Err(); err != nil {
- return "", err
- }
- return serial, nil
- }
- func LabelWithContext(ctx context.Context, name string) (string, error) {
- return "", common.ErrNotImplementedError
- }
|