2011-04-07 19:09:10 +02:00
|
|
|
// Copyright 2009 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package os
|
|
|
|
|
|
|
|
import (
|
2011-12-03 03:17:34 +01:00
|
|
|
"io"
|
2011-04-07 19:09:10 +02:00
|
|
|
"syscall"
|
|
|
|
)
|
|
|
|
|
2012-12-13 00:13:29 +01:00
|
|
|
func (file *File) readdir(n int) ([]FileInfo, error) {
|
2011-04-07 19:09:10 +02:00
|
|
|
// If this file has no dirinfo, create one.
|
|
|
|
if file.dirinfo == nil {
|
2011-05-20 02:18:15 +02:00
|
|
|
file.dirinfo = new(dirInfo)
|
2011-04-07 19:09:10 +02:00
|
|
|
}
|
2011-05-20 02:18:15 +02:00
|
|
|
d := file.dirinfo
|
2011-09-16 17:47:21 +02:00
|
|
|
size := n
|
|
|
|
if size <= 0 {
|
2011-04-07 19:09:10 +02:00
|
|
|
size = 100
|
2011-09-16 17:47:21 +02:00
|
|
|
n = -1
|
2011-04-07 19:09:10 +02:00
|
|
|
}
|
2012-12-13 00:13:29 +01:00
|
|
|
fi := make([]FileInfo, 0, size) // Empty with room to grow.
|
2011-09-16 17:47:21 +02:00
|
|
|
for n != 0 {
|
2012-12-13 00:13:29 +01:00
|
|
|
// Refill the buffer if necessary.
|
2011-05-20 02:18:15 +02:00
|
|
|
if d.bufp >= d.nbuf {
|
2012-12-13 00:13:29 +01:00
|
|
|
nb, err := file.Read(d.buf[:])
|
|
|
|
|
|
|
|
// Update the buffer state before checking for errors.
|
|
|
|
d.bufp, d.nbuf = 0, nb
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
if err == io.EOF {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
return fi, &PathError{"readdir", file.name, err}
|
2011-04-07 19:09:10 +02:00
|
|
|
}
|
2012-12-13 00:13:29 +01:00
|
|
|
if nb < syscall.STATFIXLEN {
|
|
|
|
return fi, &PathError{"readdir", file.name, syscall.ErrShortStat}
|
2011-05-20 02:18:15 +02:00
|
|
|
}
|
2011-04-07 19:09:10 +02:00
|
|
|
}
|
|
|
|
|
2012-12-13 00:13:29 +01:00
|
|
|
// Get a record from the buffer.
|
|
|
|
b := d.buf[d.bufp:]
|
|
|
|
m := int(uint16(b[0])|uint16(b[1])<<8) + 2
|
2011-05-20 02:18:15 +02:00
|
|
|
if m < syscall.STATFIXLEN {
|
2012-12-13 00:13:29 +01:00
|
|
|
return fi, &PathError{"readdir", file.name, syscall.ErrShortStat}
|
2011-04-07 19:09:10 +02:00
|
|
|
}
|
2012-12-13 00:13:29 +01:00
|
|
|
|
|
|
|
dir, err := syscall.UnmarshalDir(b[:m])
|
|
|
|
if err != nil {
|
|
|
|
return fi, &PathError{"readdir", file.name, err}
|
2011-04-07 19:09:10 +02:00
|
|
|
}
|
2012-12-13 00:13:29 +01:00
|
|
|
fi = append(fi, fileInfoFromStat(dir))
|
2011-04-07 19:09:10 +02:00
|
|
|
|
2012-12-13 00:13:29 +01:00
|
|
|
d.bufp += m
|
2011-09-16 17:47:21 +02:00
|
|
|
n--
|
2011-05-20 02:18:15 +02:00
|
|
|
}
|
2011-04-07 19:09:10 +02:00
|
|
|
|
2012-12-13 00:13:29 +01:00
|
|
|
if n >= 0 && len(fi) == 0 {
|
|
|
|
return fi, io.EOF
|
2011-04-07 19:09:10 +02:00
|
|
|
}
|
2012-12-13 00:13:29 +01:00
|
|
|
return fi, nil
|
2011-09-16 17:47:21 +02:00
|
|
|
}
|
2011-04-07 19:09:10 +02:00
|
|
|
|
2012-01-25 22:54:22 +01:00
|
|
|
func (file *File) readdirnames(n int) (names []string, err error) {
|
2011-09-16 17:47:21 +02:00
|
|
|
fi, err := file.Readdir(n)
|
2011-04-07 19:09:10 +02:00
|
|
|
names = make([]string, len(fi))
|
2011-05-20 02:18:15 +02:00
|
|
|
for i := range fi {
|
2012-01-12 02:31:45 +01:00
|
|
|
names[i] = fi[i].Name()
|
2011-04-07 19:09:10 +02:00
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|