From 683aeef2c43f4752e33e574504a6723ea6a983b4 Mon Sep 17 00:00:00 2001 From: catatsuy Date: Sat, 18 Sep 2021 14:04:08 +0900 Subject: [PATCH] call getdents multiple times Getdents can be called multiple times. Set the default size of the buffer to 5MB and call it multiple times. --- cli/cli.go | 65 ++++++++++++++++++++++++++++-------------------------- 1 file changed, 34 insertions(+), 31 deletions(-) diff --git a/cli/cli.go b/cli/cli.go index 03821a4..62a042e 100644 --- a/cli/cli.go +++ b/cli/cli.go @@ -19,6 +19,9 @@ const ( ExitCodeOK = 0 ExitCodeParseFlagError = 1 ExitCodeFail = 1 + + // 5MB + DefaultBufSize = 5 * 1024 * 1024 ) type CLI struct { @@ -47,15 +50,15 @@ func (c *CLI) Run(args []string) int { var ( version bool debug bool - bufSize int64 + bufSize int ) flags := flag.NewFlagSet("lls", flag.ContinueOnError) flags.SetOutput(c.errStream) - flags.Int64Var(&bufSize, "buf-size", 0, "specify buf size") + flags.IntVar(&bufSize, "buf-size", DefaultBufSize, "specify buf size") - flags.BoolVar(&version, "version", false, "Print version information and quit") + flags.BoolVar(&version, "version", false, "print version information and quit") flags.BoolVar(&debug, "debug", false, "debug mode") err := flags.Parse(args[1:]) @@ -88,7 +91,7 @@ func (c *CLI) Run(args []string) int { return c.run(target, debug, bufSize) } -func (c *CLI) run(target string, debug bool, bufSize int64) int { +func (c *CLI) run(target string, debug bool, bufSize int) int { f, err := os.Open(target) if err != nil { fmt.Fprintln(c.errStream, err) @@ -107,41 +110,41 @@ func (c *CLI) run(target string, debug bool, bufSize int64) int { return ExitCodeFail } - if bufSize == 0 { - bufSize = finfo.Size() - } - - // about actual size: 20 + filename - // ls -dl buf := make([]byte, bufSize) - n, err := syscall.Getdents(int(f.Fd()), buf) - if err != nil { - fmt.Fprintln(c.errStream, err) - return ExitCodeFail - } - if debug { - fmt.Fprintf(c.errStream, "bufSize: %d; getdents ret: %d\n", bufSize, n) - return ExitCodeOK - } + for { + n, err := syscall.Getdents(int(f.Fd()), buf) + if err != nil { + fmt.Fprintln(c.errStream, err) + return ExitCodeFail + } - for bufp := 0; bufp < n; { - dirent := (*syscall.Dirent)(unsafe.Pointer(&buf[bufp])) - bufp += int(dirent.Reclen) + if n == 0 { + break + } - // deleted file - if dirent.Ino == 0 { - continue + if debug { + fmt.Fprintf(c.errStream, "bufSize: %d; getdents ret: %d\n", bufSize, n) } - bb := (*[256]byte)(unsafe.Pointer(&dirent.Name[0])) - name := string(bb[0:blen(*bb)]) + for bufp := 0; bufp < n; { + dirent := (*syscall.Dirent)(unsafe.Pointer(&buf[bufp])) + bufp += int(dirent.Reclen) + + // deleted file + if dirent.Ino == 0 { + continue + } + + bb := (*[256]byte)(unsafe.Pointer(&dirent.Name[0])) + name := string(bb[0:blen(*bb)]) - if name == "." || name == ".." { - // ignore - continue + if name == "." || name == ".." { + // ignore + continue + } + fmt.Fprintln(c.outStream, name) } - fmt.Fprintln(c.outStream, name) } return ExitCodeOK