Source file
src/os/file_unix.go
1
2
3
4
5
6
7 package os
8
9 import (
10 "internal/poll"
11 "internal/syscall/unix"
12 "io/fs"
13 "runtime"
14 "sync/atomic"
15 "syscall"
16 _ "unsafe"
17 )
18
19 const _UTIME_OMIT = unix.UTIME_OMIT
20
21
22 func fixLongPath(path string) string {
23 return path
24 }
25
26 func rename(oldname, newname string) error {
27 fi, err := Lstat(newname)
28 if err == nil && fi.IsDir() {
29
30
31
32
33
34
35
36
37 if ofi, err := Lstat(oldname); err != nil {
38 if pe, ok := err.(*PathError); ok {
39 err = pe.Err
40 }
41 return &LinkError{"rename", oldname, newname, err}
42 } else if newname == oldname || !SameFile(fi, ofi) {
43 return &LinkError{"rename", oldname, newname, syscall.EEXIST}
44 }
45 }
46 err = ignoringEINTR(func() error {
47 return syscall.Rename(oldname, newname)
48 })
49 if err != nil {
50 return &LinkError{"rename", oldname, newname, err}
51 }
52 return nil
53 }
54
55
56
57
58
59 type file struct {
60 pfd poll.FD
61 name string
62 dirinfo atomic.Pointer[dirInfo]
63 nonblock bool
64 stdoutOrErr bool
65 appendMode bool
66 inRoot bool
67 }
68
69
70 func (f *File) fd() uintptr {
71 if f == nil {
72 return ^(uintptr(0))
73 }
74
75
76
77
78
79
80 if f.nonblock {
81 f.pfd.SetBlocking()
82 }
83
84 return uintptr(f.pfd.Sysfd)
85 }
86
87
88 func newFileFromNewFile(fd uintptr, name string) *File {
89 fdi := int(fd)
90 if fdi < 0 {
91 return nil
92 }
93
94 flags, err := unix.Fcntl(fdi, syscall.F_GETFL, 0)
95 if err != nil {
96 flags = 0
97 }
98 f := newFile(fdi, name, kindNewFile, unix.HasNonblockFlag(flags))
99 f.appendMode = flags&syscall.O_APPEND != 0
100 return f
101 }
102
103
104
105
106
107
108
109
110
111
112
113 func net_newUnixFile(fd int, name string) *File {
114 if fd < 0 {
115 panic("invalid FD")
116 }
117
118 return newFile(fd, name, kindSock, true)
119 }
120
121
122 type newFileKind int
123
124 const (
125
126 kindNewFile newFileKind = iota
127
128
129 kindOpenFile
130
131 kindPipe
132
133
134 kindSock
135
136
137
138 kindNoPoll
139 )
140
141
142
143
144 func newFile(fd int, name string, kind newFileKind, nonBlocking bool) *File {
145 f := &File{&file{
146 pfd: poll.FD{
147 Sysfd: fd,
148 IsStream: true,
149 ZeroReadIsEOF: true,
150 },
151 name: name,
152 stdoutOrErr: fd == 1 || fd == 2,
153 }}
154
155 pollable := kind == kindOpenFile || kind == kindPipe || kind == kindSock || nonBlocking
156
157
158
159
160
161
162
163
164
165 if kind == kindOpenFile {
166 switch runtime.GOOS {
167 case "darwin", "ios", "dragonfly", "freebsd", "netbsd", "openbsd":
168 var st syscall.Stat_t
169 err := ignoringEINTR(func() error {
170 return syscall.Fstat(fd, &st)
171 })
172 typ := st.Mode & syscall.S_IFMT
173
174
175
176
177
178
179
180 if err == nil && (typ == syscall.S_IFREG || typ == syscall.S_IFDIR) {
181 pollable = false
182 }
183
184
185
186
187
188 if (runtime.GOOS == "darwin" || runtime.GOOS == "ios") && typ == syscall.S_IFIFO {
189 pollable = false
190 }
191 }
192 }
193
194 clearNonBlock := false
195 if pollable {
196
197
198
199 if nonBlocking {
200
201 if kind == kindSock {
202 f.nonblock = true
203 }
204 } else if err := syscall.SetNonblock(fd, true); err == nil {
205 f.nonblock = true
206 clearNonBlock = true
207 } else {
208 pollable = false
209 }
210 }
211
212
213
214
215
216
217
218
219 if pollErr := f.pfd.Init("file", pollable); pollErr != nil && clearNonBlock {
220 if err := syscall.SetNonblock(fd, false); err == nil {
221 f.nonblock = false
222 }
223 }
224
225 runtime.SetFinalizer(f.file, (*file).close)
226 return f
227 }
228
229 func sigpipe()
230
231
232
233
234 func epipecheck(file *File, e error) {
235 if e == syscall.EPIPE && file.stdoutOrErr {
236 sigpipe()
237 }
238 }
239
240
241
242 const DevNull = "/dev/null"
243
244
245
246 func openFileNolog(name string, flag int, perm FileMode) (*File, error) {
247 setSticky := false
248 if !supportsCreateWithStickyBit && flag&O_CREATE != 0 && perm&ModeSticky != 0 {
249 if _, err := Stat(name); IsNotExist(err) {
250 setSticky = true
251 }
252 }
253
254 var (
255 r int
256 s poll.SysFile
257 e error
258 )
259
260 ignoringEINTR(func() error {
261 r, s, e = open(name, flag|syscall.O_CLOEXEC, syscallMode(perm))
262 return e
263 })
264 if e != nil {
265 return nil, &PathError{Op: "open", Path: name, Err: e}
266 }
267
268
269 if setSticky {
270 setStickyBit(name)
271 }
272
273
274
275 if !supportsCloseOnExec {
276 syscall.CloseOnExec(r)
277 }
278
279 f := newFile(r, name, kindOpenFile, unix.HasNonblockFlag(flag))
280 f.pfd.SysFile = s
281 return f, nil
282 }
283
284 func openDirNolog(name string) (*File, error) {
285 var (
286 r int
287 s poll.SysFile
288 e error
289 )
290 ignoringEINTR(func() error {
291 r, s, e = open(name, O_RDONLY|syscall.O_CLOEXEC|syscall.O_DIRECTORY, 0)
292 return e
293 })
294 if e != nil {
295 return nil, &PathError{Op: "open", Path: name, Err: e}
296 }
297
298 if !supportsCloseOnExec {
299 syscall.CloseOnExec(r)
300 }
301
302 f := newFile(r, name, kindNoPoll, false)
303 f.pfd.SysFile = s
304 return f, nil
305 }
306
307 func (file *file) close() error {
308 if file == nil {
309 return syscall.EINVAL
310 }
311 if info := file.dirinfo.Swap(nil); info != nil {
312 info.close()
313 }
314 var err error
315 if e := file.pfd.Close(); e != nil {
316 if e == poll.ErrFileClosing {
317 e = ErrClosed
318 }
319 err = &PathError{Op: "close", Path: file.name, Err: e}
320 }
321
322
323 runtime.SetFinalizer(file, nil)
324 return err
325 }
326
327
328
329
330
331 func (f *File) seek(offset int64, whence int) (ret int64, err error) {
332 if info := f.dirinfo.Swap(nil); info != nil {
333
334
335 info.close()
336 }
337 ret, err = f.pfd.Seek(offset, whence)
338 runtime.KeepAlive(f)
339 return ret, err
340 }
341
342
343
344
345 func Truncate(name string, size int64) error {
346 e := ignoringEINTR(func() error {
347 return syscall.Truncate(name, size)
348 })
349 if e != nil {
350 return &PathError{Op: "truncate", Path: name, Err: e}
351 }
352 return nil
353 }
354
355
356
357 func Remove(name string) error {
358
359
360
361
362 e := ignoringEINTR(func() error {
363 return syscall.Unlink(name)
364 })
365 if e == nil {
366 return nil
367 }
368 e1 := ignoringEINTR(func() error {
369 return syscall.Rmdir(name)
370 })
371 if e1 == nil {
372 return nil
373 }
374
375
376
377
378
379
380
381
382
383
384 if e1 != syscall.ENOTDIR {
385 e = e1
386 }
387 return &PathError{Op: "remove", Path: name, Err: e}
388 }
389
390 func tempDir() string {
391 dir := Getenv("TMPDIR")
392 if dir == "" {
393 if runtime.GOOS == "android" {
394 dir = "/data/local/tmp"
395 } else {
396 dir = "/tmp"
397 }
398 }
399 return dir
400 }
401
402
403
404 func Link(oldname, newname string) error {
405 e := ignoringEINTR(func() error {
406 return syscall.Link(oldname, newname)
407 })
408 if e != nil {
409 return &LinkError{"link", oldname, newname, e}
410 }
411 return nil
412 }
413
414
415
416
417
418 func Symlink(oldname, newname string) error {
419 e := ignoringEINTR(func() error {
420 return syscall.Symlink(oldname, newname)
421 })
422 if e != nil {
423 return &LinkError{"symlink", oldname, newname, e}
424 }
425 return nil
426 }
427
428 func readlink(name string) (string, error) {
429 for len := 128; ; len *= 2 {
430 b := make([]byte, len)
431 n, err := ignoringEINTR2(func() (int, error) {
432 return fixCount(syscall.Readlink(name, b))
433 })
434
435 if (runtime.GOOS == "aix" || runtime.GOOS == "wasip1") && err == syscall.ERANGE {
436 continue
437 }
438 if err != nil {
439 return "", &PathError{Op: "readlink", Path: name, Err: err}
440 }
441 if n < len {
442 return string(b[0:n]), nil
443 }
444 }
445 }
446
447 type unixDirent struct {
448 parent string
449 name string
450 typ FileMode
451 info FileInfo
452 }
453
454 func (d *unixDirent) Name() string { return d.name }
455 func (d *unixDirent) IsDir() bool { return d.typ.IsDir() }
456 func (d *unixDirent) Type() FileMode { return d.typ }
457
458 func (d *unixDirent) Info() (FileInfo, error) {
459 if d.info != nil {
460 return d.info, nil
461 }
462 return Lstat(d.parent + "/" + d.name)
463 }
464
465 func (d *unixDirent) String() string {
466 return fs.FormatDirEntry(d)
467 }
468
469 func newUnixDirent(parent *File, name string, typ FileMode) (DirEntry, error) {
470 ude := &unixDirent{
471 parent: parent.name,
472 name: name,
473 typ: typ,
474 }
475
476
477
478 if typ != ^FileMode(0) && !parent.inRoot {
479 return ude, nil
480 }
481
482 info, err := parent.lstatat(name)
483 if err != nil {
484 return nil, err
485 }
486
487 ude.typ = info.Mode().Type()
488 ude.info = info
489 return ude, nil
490 }
491
View as plain text