2018-10-17 20:20:35 +00:00
|
|
|
// +build darwin
|
|
|
|
|
|
|
|
package disk
|
|
|
|
|
|
|
|
import (
|
2018-10-19 18:33:23 +00:00
|
|
|
"context"
|
2018-10-17 20:20:35 +00:00
|
|
|
"path"
|
|
|
|
"unsafe"
|
|
|
|
|
|
|
|
"github.com/shirou/gopsutil/internal/common"
|
2018-10-19 18:33:23 +00:00
|
|
|
"golang.org/x/sys/unix"
|
2018-10-17 20:20:35 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func Partitions(all bool) ([]PartitionStat, error) {
|
2018-10-19 18:33:23 +00:00
|
|
|
return PartitionsWithContext(context.Background(), all)
|
|
|
|
}
|
|
|
|
|
|
|
|
func PartitionsWithContext(ctx context.Context, all bool) ([]PartitionStat, error) {
|
2018-10-17 20:20:35 +00:00
|
|
|
var ret []PartitionStat
|
|
|
|
|
|
|
|
count, err := Getfsstat(nil, MntWait)
|
|
|
|
if err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
2018-10-19 18:33:23 +00:00
|
|
|
fs := make([]Statfs, count)
|
|
|
|
if _, err = Getfsstat(fs, MntWait); err != nil {
|
|
|
|
return ret, err
|
|
|
|
}
|
2018-10-17 20:20:35 +00:00
|
|
|
for _, stat := range fs {
|
|
|
|
opts := "rw"
|
|
|
|
if stat.Flags&MntReadOnly != 0 {
|
|
|
|
opts = "ro"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntSynchronous != 0 {
|
|
|
|
opts += ",sync"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntNoExec != 0 {
|
|
|
|
opts += ",noexec"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntNoSuid != 0 {
|
|
|
|
opts += ",nosuid"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntUnion != 0 {
|
|
|
|
opts += ",union"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntAsync != 0 {
|
|
|
|
opts += ",async"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntSuidDir != 0 {
|
|
|
|
opts += ",suiddir"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntSoftDep != 0 {
|
|
|
|
opts += ",softdep"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntNoSymFollow != 0 {
|
|
|
|
opts += ",nosymfollow"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntGEOMJournal != 0 {
|
|
|
|
opts += ",gjounalc"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntMultilabel != 0 {
|
|
|
|
opts += ",multilabel"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntACLs != 0 {
|
|
|
|
opts += ",acls"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntNoATime != 0 {
|
|
|
|
opts += ",noattime"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntClusterRead != 0 {
|
|
|
|
opts += ",nocluster"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntClusterWrite != 0 {
|
|
|
|
opts += ",noclusterw"
|
|
|
|
}
|
|
|
|
if stat.Flags&MntNFS4ACLs != 0 {
|
|
|
|
opts += ",nfs4acls"
|
|
|
|
}
|
|
|
|
d := PartitionStat{
|
|
|
|
Device: common.IntToString(stat.Mntfromname[:]),
|
|
|
|
Mountpoint: common.IntToString(stat.Mntonname[:]),
|
|
|
|
Fstype: common.IntToString(stat.Fstypename[:]),
|
|
|
|
Opts: opts,
|
|
|
|
}
|
|
|
|
if all == false {
|
|
|
|
if !path.IsAbs(d.Device) || !common.PathExists(d.Device) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = append(ret, d)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
|
2018-10-19 18:33:23 +00:00
|
|
|
func Getfsstat(buf []Statfs, flags int) (n int, err error) {
|
|
|
|
return GetfsstatWithContext(context.Background(), buf, flags)
|
2018-10-17 20:20:35 +00:00
|
|
|
}
|
|
|
|
|
2018-10-19 18:33:23 +00:00
|
|
|
func GetfsstatWithContext(ctx context.Context, buf []Statfs, flags int) (n int, err error) {
|
2018-10-17 20:20:35 +00:00
|
|
|
var _p0 unsafe.Pointer
|
|
|
|
var bufsize uintptr
|
|
|
|
if len(buf) > 0 {
|
|
|
|
_p0 = unsafe.Pointer(&buf[0])
|
2018-10-19 18:33:23 +00:00
|
|
|
bufsize = unsafe.Sizeof(Statfs{}) * uintptr(len(buf))
|
2018-10-17 20:20:35 +00:00
|
|
|
}
|
2018-10-19 18:33:23 +00:00
|
|
|
r0, _, e1 := unix.Syscall(SYS_GETFSSTAT64, uintptr(_p0), bufsize, uintptr(flags))
|
2018-10-17 20:20:35 +00:00
|
|
|
n = int(r0)
|
|
|
|
if e1 != 0 {
|
|
|
|
err = e1
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-10-19 18:33:23 +00:00
|
|
|
func getFsType(stat unix.Statfs_t) string {
|
2018-10-17 20:20:35 +00:00
|
|
|
return common.IntToString(stat.Fstypename[:])
|
|
|
|
}
|