Codebase list golang-github-pkg-xattr / lintian-fixes/main xattr_linux.go
lintian-fixes/main

Tree @lintian-fixes/main (Download .tar.gz)

xattr_linux.go @lintian-fixes/mainraw · history · blame

//go:build linux
// +build linux

package xattr

import (
	"os"
	"syscall"

	"golang.org/x/sys/unix"
)

const (
	// XATTR_SUPPORTED will be true if the current platform is supported
	XATTR_SUPPORTED = true

	XATTR_CREATE  = unix.XATTR_CREATE
	XATTR_REPLACE = unix.XATTR_REPLACE

	// ENOATTR is not exported by the syscall package on Linux, because it is
	// an alias for ENODATA. We export it here so it is available on all
	// our supported platforms.
	ENOATTR = syscall.ENODATA
)

// On Linux, FUSE and CIFS filesystems can return EINTR for interrupted system
// calls. This function works around this by retrying system calls until they
// stop returning EINTR.
//
// See https://github.com/golang/go/commit/6b420169d798c7ebe733487b56ea5c3fa4aab5ce.
func ignoringEINTR(fn func() error) (err error) {
	for {
		err = fn()
		if err != unix.EINTR {
			break
		}
	}
	return err
}

func getxattr(path string, name string, data []byte) (int, error) {
	var r int
	err := ignoringEINTR(func() (err error) {
		r, err = unix.Getxattr(path, name, data)
		return err
	})
	return r, err
}

func lgetxattr(path string, name string, data []byte) (int, error) {
	var r int
	err := ignoringEINTR(func() (err error) {
		r, err = unix.Lgetxattr(path, name, data)
		return err
	})
	return r, err
}

func fgetxattr(f *os.File, name string, data []byte) (int, error) {
	var r int
	err := ignoringEINTR(func() (err error) {
		r, err = unix.Fgetxattr(int(f.Fd()), name, data)
		return err
	})
	return r, err
}

func setxattr(path string, name string, data []byte, flags int) error {
	return ignoringEINTR(func() (err error) {
		return unix.Setxattr(path, name, data, flags)
	})
}

func lsetxattr(path string, name string, data []byte, flags int) error {
	return ignoringEINTR(func() (err error) {
		return unix.Lsetxattr(path, name, data, flags)
	})
}

func fsetxattr(f *os.File, name string, data []byte, flags int) error {
	return ignoringEINTR(func() (err error) {
		return unix.Fsetxattr(int(f.Fd()), name, data, flags)
	})
}

func removexattr(path string, name string) error {
	return ignoringEINTR(func() (err error) {
		return unix.Removexattr(path, name)
	})
}

func lremovexattr(path string, name string) error {
	return ignoringEINTR(func() (err error) {
		return unix.Lremovexattr(path, name)
	})
}

func fremovexattr(f *os.File, name string) error {
	return ignoringEINTR(func() (err error) {
		return unix.Fremovexattr(int(f.Fd()), name)
	})
}

func listxattr(path string, data []byte) (int, error) {
	var r int
	err := ignoringEINTR(func() (err error) {
		r, err = unix.Listxattr(path, data)
		return err
	})
	return r, err
}

func llistxattr(path string, data []byte) (int, error) {
	var r int
	err := ignoringEINTR(func() (err error) {
		r, err = unix.Llistxattr(path, data)
		return err
	})
	return r, err
}

func flistxattr(f *os.File, data []byte) (int, error) {
	var r int
	err := ignoringEINTR(func() (err error) {
		r, err = unix.Flistxattr(int(f.Fd()), data)
		return err
	})
	return r, err
}

// stringsFromByteSlice converts a sequence of attributes to a []string.
// On Darwin and Linux, each entry is a NULL-terminated string.
func stringsFromByteSlice(buf []byte) (result []string) {
	offset := 0
	for index, b := range buf {
		if b == 0 {
			result = append(result, string(buf[offset:index]))
			offset = index + 1
		}
	}
	return
}