mirror of
https://github.com/golang/sys.git
synced 2026-01-29 07:02:06 +03:00
Same as CL 587655 and CL 690475 did for package syscall tests. Change-Id: Ie3c8726a4e2c859b5d7992ea1baec14083b9fdba Reviewed-on: https://go-review.googlesource.com/c/sys/+/708558 Reviewed-by: Junyang Shao <shaojunyang@google.com> Reviewed-by: Florian Lehner <lehner.florian86@gmail.com> Reviewed-by: Carlos Amedee <carlos@golang.org> Auto-Submit: Tobias Klauser <tobias.klauser@gmail.com> LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com> Reviewed-by: Sean Liao <sean@liao.dev>
154 lines
3.5 KiB
Go
154 lines
3.5 KiB
Go
// Copyright 2019 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.
|
|
|
|
//go:build aix || darwin || dragonfly || freebsd || linux || netbsd || openbsd || solaris || zos
|
|
|
|
package unix_test
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"runtime"
|
|
"slices"
|
|
"strconv"
|
|
"strings"
|
|
"testing"
|
|
"unsafe"
|
|
|
|
"golang.org/x/sys/unix"
|
|
)
|
|
|
|
func TestDirent(t *testing.T) {
|
|
const (
|
|
direntBufSize = 2048 // arbitrary? See https://go.dev/issue/37323.
|
|
filenameMinSize = 11
|
|
)
|
|
|
|
d := t.TempDir()
|
|
t.Logf("tmpdir: %s", d)
|
|
|
|
for i, c := range []byte("0123456789") {
|
|
name := string(bytes.Repeat([]byte{c}, filenameMinSize+i))
|
|
err := os.WriteFile(filepath.Join(d, name), nil, 0644)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
names := make([]string, 0, 10)
|
|
|
|
fd, err := unix.Open(d, unix.O_RDONLY, 0)
|
|
if err != nil {
|
|
t.Fatalf("Open: %v", err)
|
|
}
|
|
defer unix.Close(fd)
|
|
|
|
buf := bytes.Repeat([]byte{0xCD}, direntBufSize)
|
|
for {
|
|
n, err := unix.ReadDirent(fd, buf)
|
|
if err == unix.EINVAL {
|
|
// On linux, 'man getdents64' says that EINVAL indicates result buffer is too small.
|
|
// Try a bigger buffer.
|
|
t.Logf("ReadDirent: %v; retrying with larger buffer", err)
|
|
buf = bytes.Repeat([]byte{0xCD}, len(buf)*2)
|
|
continue
|
|
}
|
|
if err != nil {
|
|
t.Fatalf("ReadDirent: %v", err)
|
|
}
|
|
t.Logf("ReadDirent: read %d bytes", n)
|
|
if n == 0 {
|
|
break
|
|
}
|
|
|
|
var consumed, count int
|
|
consumed, count, names = unix.ParseDirent(buf[:n], -1, names)
|
|
t.Logf("ParseDirent: %d new name(s)", count)
|
|
if consumed != n {
|
|
t.Fatalf("ParseDirent: consumed %d bytes; expected %d", consumed, n)
|
|
}
|
|
}
|
|
|
|
slices.Sort(names)
|
|
t.Logf("names: %q", names)
|
|
|
|
if len(names) != 10 {
|
|
t.Errorf("got %d names; expected 10", len(names))
|
|
}
|
|
for i, name := range names {
|
|
ord, err := strconv.Atoi(name[:1])
|
|
if err != nil {
|
|
t.Fatalf("names[%d] is non-integer %q: %v", i, names[i], err)
|
|
}
|
|
if expected := string(strings.Repeat(name[:1], filenameMinSize+ord)); name != expected {
|
|
t.Errorf("names[%d] is %q (len %d); expected %q (len %d)", i, name, len(name), expected, len(expected))
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestDirentRepeat(t *testing.T) {
|
|
const N = 100
|
|
// Note: the size of the buffer is small enough that the loop
|
|
// below will need to execute multiple times. See issue #31368.
|
|
size := N * unsafe.Offsetof(unix.Dirent{}.Name) / 4
|
|
if runtime.GOOS == "freebsd" || runtime.GOOS == "netbsd" {
|
|
if size < 1024 {
|
|
size = 1024 // DIRBLKSIZ, see issue 31403.
|
|
}
|
|
if runtime.GOOS == "freebsd" {
|
|
t.Skip("need to fix issue 31416 first")
|
|
}
|
|
}
|
|
|
|
// Make a directory containing N files
|
|
d := t.TempDir()
|
|
|
|
var files []string
|
|
for i := range N {
|
|
files = append(files, fmt.Sprintf("file%d", i))
|
|
}
|
|
for _, file := range files {
|
|
err := os.WriteFile(filepath.Join(d, file), []byte("contents"), 0644)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
// Read the directory entries using ReadDirent.
|
|
fd, err := unix.Open(d, unix.O_RDONLY, 0)
|
|
if err != nil {
|
|
t.Fatalf("Open: %v", err)
|
|
}
|
|
defer unix.Close(fd)
|
|
var files2 []string
|
|
for {
|
|
buf := make([]byte, size)
|
|
n, err := unix.ReadDirent(fd, buf)
|
|
if err != nil {
|
|
t.Fatalf("ReadDirent: %v", err)
|
|
}
|
|
if n == 0 {
|
|
break
|
|
}
|
|
buf = buf[:n]
|
|
for len(buf) > 0 {
|
|
var consumed int
|
|
consumed, _, files2 = unix.ParseDirent(buf, -1, files2)
|
|
if consumed == 0 && len(buf) > 0 {
|
|
t.Fatal("no progress")
|
|
}
|
|
buf = buf[consumed:]
|
|
}
|
|
}
|
|
|
|
// Check results
|
|
slices.Sort(files)
|
|
slices.Sort(files2)
|
|
if !slices.Equal(files, files2) {
|
|
t.Errorf("bad file list: want\n%q\ngot\n%q", files, files2)
|
|
}
|
|
}
|