Skip to content

Commit 94af800

Browse files
committed
Lchmod(): fix for Linux/Go 1.11
Linux does not support file mode for symlinks. It does not support flags argument for fchmodat() syscall either (the only flag described by POSIX is AT_SYMLINK_NOFOLLOW to allow changing mode on symlink). This is probably the reason why the standard os package from Go does not provide Lchmod(). Go < 1.11 errorneously assumed the flags argument is supported on Linux, and implemented syscall.Fchmodat() with flags being passed on to the kernel. The kernel ignored the flags (as the kernel syscall doesn't even have flags argument). The result was changing the mode on the file that symlink points to (i.e. ignoring AT_SYMLINK_NOFOLLOW). The above bug was fixed in Go-1.11beta1 (see [1]), and since the fix, if any flags are set, an error is returned. This rendered Lchmod() useless on Linux + Go 1.11, as now it always returns an error. The best possible fix I can think of, given the constraints of keeping the API intact, is implemented in this commit. In short, chown() is not called for symlinks on Linux. [1] https://go-review.googlesource.com/c/go/+/118658 Signed-off-by: Kir Kolyshkin <[email protected]>
1 parent 9ab0ec6 commit 94af800

File tree

3 files changed

+33
-6
lines changed

3 files changed

+33
-6
lines changed

driver/driver_unix.go

-6
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,6 @@ import (
1010

1111
"github.com/containerd/continuity/devices"
1212
"github.com/containerd/continuity/sysx"
13-
"golang.org/x/sys/unix"
1413
)
1514

1615
func (d *driver) Mknod(path string, mode os.FileMode, major, minor int) error {
@@ -26,11 +25,6 @@ func (d *driver) Mkfifo(path string, mode os.FileMode) error {
2625
return devices.Mknod(path, mode, 0, 0)
2726
}
2827

29-
// Lchmod changes the mode of a file not following symlinks.
30-
func (d *driver) Lchmod(path string, mode os.FileMode) error {
31-
return unix.Fchmodat(unix.AT_FDCWD, path, uint32(mode), unix.AT_SYMLINK_NOFOLLOW)
32-
}
33-
3428
// Getxattr returns all of the extended attributes for the file at path p.
3529
func (d *driver) Getxattr(p string) (map[string][]byte, error) {
3630
xattrs, err := sysx.Listxattr(p)

driver/lchmod_linux.go

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
package driver
2+
3+
import (
4+
"os"
5+
6+
"golang.org/x/sys/unix"
7+
)
8+
9+
// Lchmod changes the mode of a file not following symlinks.
10+
func (d *driver) Lchmod(path string, mode os.FileMode) error {
11+
// On Linux, file mode is not supported for symlinks,
12+
// and fchmodat() does not support AT_SYMLINK_NOFOLLOW,
13+
// so symlinks need to be skipped entirely.
14+
if st, err := os.Stat(path); err == nil && st.Mode()&os.ModeSymlink != 0 {
15+
return nil
16+
}
17+
18+
return unix.Fchmodat(unix.AT_FDCWD, path, uint32(mode), 0)
19+
}

driver/lchmod_unix.go

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
// +build darwin freebsd solaris
2+
3+
package driver
4+
5+
import (
6+
"os"
7+
8+
"golang.org/x/sys/unix"
9+
)
10+
11+
// Lchmod changes the mode of a file not following symlinks.
12+
func (d *driver) Lchmod(path string, mode os.FileMode) error {
13+
return unix.Fchmodat(unix.AT_FDCWD, path, uint32(mode), unix.AT_SYMLINK_NOFOLLOW)
14+
}

0 commit comments

Comments
 (0)