Source file
src/os/root_unix.go
1
2
3
4
5
6
7 package os
8
9 import (
10 "errors"
11 "internal/syscall/unix"
12 "runtime"
13 "syscall"
14 "time"
15 )
16
17 type sysfdType = int
18
19
20 func openRootNolog(name string) (*Root, error) {
21 var fd int
22 err := ignoringEINTR(func() error {
23 var err error
24 fd, _, err = open(name, syscall.O_CLOEXEC, 0)
25 return err
26 })
27 if err != nil {
28 return nil, &PathError{Op: "open", Path: name, Err: err}
29 }
30 return newRoot(fd, name)
31 }
32
33
34
35 func newRoot(fd int, name string) (*Root, error) {
36 var fs fileStat
37 err := ignoringEINTR(func() error {
38 return syscall.Fstat(fd, &fs.sys)
39 })
40 fillFileStatFromSys(&fs, name)
41 if err == nil && !fs.IsDir() {
42 syscall.Close(fd)
43 return nil, &PathError{Op: "open", Path: name, Err: errors.New("not a directory")}
44 }
45
46
47
48 if !supportsCloseOnExec {
49 syscall.CloseOnExec(fd)
50 }
51
52 r := &Root{&root{
53 fd: fd,
54 name: name,
55 }}
56 r.root.cleanup = runtime.AddCleanup(r, func(f *root) { f.Close() }, r.root)
57 return r, nil
58 }
59
60
61 func openRootInRoot(r *Root, name string) (*Root, error) {
62 fd, err := doInRoot(r, name, func(parent int, name string) (fd int, err error) {
63 ignoringEINTR(func() error {
64 fd, err = unix.Openat(parent, name, syscall.O_NOFOLLOW|syscall.O_CLOEXEC, 0)
65 if isNoFollowErr(err) {
66 err = checkSymlink(parent, name, err)
67 }
68 return err
69 })
70 return fd, err
71 })
72 if err != nil {
73 return nil, &PathError{Op: "openat", Path: name, Err: err}
74 }
75 return newRoot(fd, name)
76 }
77
78
79 func rootOpenFileNolog(root *Root, name string, flag int, perm FileMode) (*File, error) {
80 fd, err := doInRoot(root, name, func(parent int, name string) (fd int, err error) {
81 ignoringEINTR(func() error {
82 fd, err = unix.Openat(parent, name, syscall.O_NOFOLLOW|syscall.O_CLOEXEC|flag, uint32(perm))
83 if isNoFollowErr(err) || err == syscall.ENOTDIR {
84 err = checkSymlink(parent, name, err)
85 }
86 return err
87 })
88 return fd, err
89 })
90 if err != nil {
91 return nil, &PathError{Op: "openat", Path: name, Err: err}
92 }
93 f := newFile(fd, joinPath(root.Name(), name), kindOpenFile, unix.HasNonblockFlag(flag))
94 return f, nil
95 }
96
97 func rootOpenDir(parent int, name string) (int, error) {
98 var (
99 fd int
100 err error
101 )
102 ignoringEINTR(func() error {
103 fd, err = unix.Openat(parent, name, syscall.O_NOFOLLOW|syscall.O_CLOEXEC|syscall.O_DIRECTORY, 0)
104 if isNoFollowErr(err) || err == syscall.ENOTDIR {
105 err = checkSymlink(parent, name, err)
106 } else if err == syscall.ENOTSUP || err == syscall.EOPNOTSUPP {
107
108
109
110 err = syscall.ENOTDIR
111 }
112 return err
113 })
114 return fd, err
115 }
116
117 func rootStat(r *Root, name string, lstat bool) (FileInfo, error) {
118 fi, err := doInRoot(r, name, func(parent sysfdType, n string) (FileInfo, error) {
119 var fs fileStat
120 if err := unix.Fstatat(parent, n, &fs.sys, unix.AT_SYMLINK_NOFOLLOW); err != nil {
121 return nil, err
122 }
123 fillFileStatFromSys(&fs, name)
124 if !lstat && fs.Mode()&ModeSymlink != 0 {
125 return nil, checkSymlink(parent, n, syscall.ELOOP)
126 }
127 return &fs, nil
128 })
129 if err != nil {
130 return nil, &PathError{Op: "statat", Path: name, Err: err}
131 }
132 return fi, nil
133 }
134
135 func rootSymlink(r *Root, oldname, newname string) error {
136 _, err := doInRoot(r, newname, func(parent sysfdType, name string) (struct{}, error) {
137 return struct{}{}, symlinkat(oldname, parent, name)
138 })
139 if err != nil {
140 return &LinkError{"symlinkat", oldname, newname, err}
141 }
142 return nil
143 }
144
145
146
147
148
149
150
151
152
153
154
155
156 func afterResolvingSymlink(parent int, name string, f func() error) error {
157 if err := checkSymlink(parent, name, nil); err != nil {
158 return err
159 }
160 return f()
161 }
162
163 func chmodat(parent int, name string, mode FileMode) error {
164 return afterResolvingSymlink(parent, name, func() error {
165 return ignoringEINTR(func() error {
166 return unix.Fchmodat(parent, name, syscallMode(mode), unix.AT_SYMLINK_NOFOLLOW)
167 })
168 })
169 }
170
171 func chownat(parent int, name string, uid, gid int) error {
172 return afterResolvingSymlink(parent, name, func() error {
173 return ignoringEINTR(func() error {
174 return unix.Fchownat(parent, name, uid, gid, unix.AT_SYMLINK_NOFOLLOW)
175 })
176 })
177 }
178
179 func lchownat(parent int, name string, uid, gid int) error {
180 return ignoringEINTR(func() error {
181 return unix.Fchownat(parent, name, uid, gid, unix.AT_SYMLINK_NOFOLLOW)
182 })
183 }
184
185 func chtimesat(parent int, name string, atime time.Time, mtime time.Time) error {
186 return afterResolvingSymlink(parent, name, func() error {
187 return ignoringEINTR(func() error {
188 utimes := chtimesUtimes(atime, mtime)
189 return unix.Utimensat(parent, name, &utimes, unix.AT_SYMLINK_NOFOLLOW)
190 })
191 })
192 }
193
194 func mkdirat(fd int, name string, perm FileMode) error {
195 return ignoringEINTR(func() error {
196 return unix.Mkdirat(fd, name, syscallMode(perm))
197 })
198 }
199
200 func removeat(fd int, name string) error {
201
202
203 e := ignoringEINTR(func() error {
204 return unix.Unlinkat(fd, name, 0)
205 })
206 if e == nil {
207 return nil
208 }
209 e1 := ignoringEINTR(func() error {
210 return unix.Unlinkat(fd, name, unix.AT_REMOVEDIR)
211 })
212 if e1 == nil {
213 return nil
214 }
215
216 if e1 != syscall.ENOTDIR {
217 return e1
218 }
219 return e
220 }
221
222 func removefileat(fd int, name string) error {
223 return ignoringEINTR(func() error {
224 return unix.Unlinkat(fd, name, 0)
225 })
226 }
227
228 func removedirat(fd int, name string) error {
229 return ignoringEINTR(func() error {
230 return unix.Unlinkat(fd, name, unix.AT_REMOVEDIR)
231 })
232 }
233
234 func renameat(oldfd int, oldname string, newfd int, newname string) error {
235 return unix.Renameat(oldfd, oldname, newfd, newname)
236 }
237
238 func linkat(oldfd int, oldname string, newfd int, newname string) error {
239 return unix.Linkat(oldfd, oldname, newfd, newname, 0)
240 }
241
242 func symlinkat(oldname string, newfd int, newname string) error {
243 return unix.Symlinkat(oldname, newfd, newname)
244 }
245
246
247
248
249
250 func checkSymlink(parent int, name string, origError error) error {
251 link, err := readlinkat(parent, name)
252 if err != nil {
253 return origError
254 }
255 return errSymlink(link)
256 }
257
258 func readlinkat(fd int, name string) (string, error) {
259 for len := 128; ; len *= 2 {
260 b := make([]byte, len)
261 var (
262 n int
263 e error
264 )
265 ignoringEINTR(func() error {
266 n, e = unix.Readlinkat(fd, name, b)
267 return e
268 })
269 if e == syscall.ERANGE {
270 continue
271 }
272 if e != nil {
273 return "", e
274 }
275 if n < 0 {
276 n = 0
277 }
278 if n < len {
279 return string(b[0:n]), nil
280 }
281 }
282 }
283
View as plain text