sandbox/vfs: count mountinfo entries
All checks were successful
Test / Create distribution (push) Successful in 25s
Test / Fortify (push) Successful in 2m39s
Test / Fpkg (push) Successful in 3m37s
Test / Data race detector (push) Successful in 4m10s
Test / Flake checks (push) Successful in 52s

Signed-off-by: Ophestra <cat@gensokyo.uk>
This commit is contained in:
Ophestra 2025-03-21 12:14:33 +09:00
parent 9ddf5794dd
commit 5098b12e4a
Signed by: cat
SSH Key Fingerprint: SHA256:gQ67O0enBZ7UdZypgtspB2FDM1g3GVw8nX0XSdcFw8Q
2 changed files with 25 additions and 13 deletions

View File

@ -52,10 +52,14 @@ type (
) )
// ParseMountInfo parses a mountinfo file according to proc_pid_mountinfo(5). // ParseMountInfo parses a mountinfo file according to proc_pid_mountinfo(5).
func ParseMountInfo(r io.Reader) (*MountInfo, error) { func ParseMountInfo(r io.Reader) (*MountInfo, int, error) {
var m, cur *MountInfo var m, cur *MountInfo
s := bufio.NewScanner(r) s := bufio.NewScanner(r)
var n int
for s.Scan() { for s.Scan() {
n++
if cur == nil { if cur == nil {
m = new(MountInfo) m = new(MountInfo)
cur = m cur = m
@ -67,7 +71,7 @@ func ParseMountInfo(r io.Reader) (*MountInfo, error) {
// prevent proceeding with misaligned fields due to optional fields // prevent proceeding with misaligned fields due to optional fields
f := strings.Split(s.Text(), " ") f := strings.Split(s.Text(), " ")
if len(f) < 10 { if len(f) < 10 {
return nil, ErrMountInfoFields return nil, -1, ErrMountInfoFields
} }
// 36 35 98:0 /mnt1 /mnt2 rw,noatime master:1 - ext3 /dev/root rw,errors=continue // 36 35 98:0 /mnt1 /mnt2 rw,noatime master:1 - ext3 /dev/root rw,errors=continue
@ -75,42 +79,42 @@ func ParseMountInfo(r io.Reader) (*MountInfo, error) {
// (1) id // (1) id
if id, err := strconv.Atoi(f[0]); err != nil { // 0 if id, err := strconv.Atoi(f[0]); err != nil { // 0
return nil, err return nil, -1, err
} else { } else {
cur.ID = id cur.ID = id
} }
// (2) parent // (2) parent
if parent, err := strconv.Atoi(f[1]); err != nil { // 1 if parent, err := strconv.Atoi(f[1]); err != nil { // 1
return nil, err return nil, -1, err
} else { } else {
cur.Parent = parent cur.Parent = parent
} }
// (3) maj:min // (3) maj:min
if n, err := fmt.Sscanf(f[2], "%d:%d", &cur.Devno[0], &cur.Devno[1]); err != nil { if n, err := fmt.Sscanf(f[2], "%d:%d", &cur.Devno[0], &cur.Devno[1]); err != nil {
return nil, err return nil, -1, err
} else if n != 2 { } else if n != 2 {
// unreachable // unreachable
return nil, ErrMountInfoDevno return nil, -1, ErrMountInfoDevno
} }
// (4) mountroot // (4) mountroot
cur.Root = Unmangle(f[3]) cur.Root = Unmangle(f[3])
if cur.Root == "" { if cur.Root == "" {
return nil, ErrMountInfoEmpty return nil, -1, ErrMountInfoEmpty
} }
// (5) target // (5) target
cur.Target = Unmangle(f[4]) cur.Target = Unmangle(f[4])
if cur.Target == "" { if cur.Target == "" {
return nil, ErrMountInfoEmpty return nil, -1, ErrMountInfoEmpty
} }
// (6) vfs options (fs-independent) // (6) vfs options (fs-independent)
cur.VfsOptstr = Unmangle(f[5]) cur.VfsOptstr = Unmangle(f[5])
if cur.VfsOptstr == "" { if cur.VfsOptstr == "" {
return nil, ErrMountInfoEmpty return nil, -1, ErrMountInfoEmpty
} }
// (7) optional fields, terminated by " - " // (7) optional fields, terminated by " - "
@ -119,14 +123,14 @@ func ParseMountInfo(r io.Reader) (*MountInfo, error) {
// (8) optional fields end marker // (8) optional fields end marker
if f[i] != "-" { if f[i] != "-" {
return nil, ErrMountInfoSep return nil, -1, ErrMountInfoSep
} }
i++ i++
// (9) FS type // (9) FS type
cur.FsType = Unmangle(f[i]) cur.FsType = Unmangle(f[i])
if cur.FsType == "" { if cur.FsType == "" {
return nil, ErrMountInfoEmpty return nil, -1, ErrMountInfoEmpty
} }
i++ i++
@ -137,5 +141,5 @@ func ParseMountInfo(r io.Reader) (*MountInfo, error) {
// (11) fs options (fs specific) // (11) fs options (fs specific)
cur.FsOptstr = Unmangle(f[i]) cur.FsOptstr = Unmangle(f[i])
} }
return m, s.Err() return m, n, s.Err()
} }

View File

@ -120,7 +120,7 @@ id 20 0:53 / /mnt/test rw,relatime shared:212 - tmpfs rw
for _, tc := range testCases { for _, tc := range testCases {
t.Run(tc.name, func(t *testing.T) { t.Run(tc.name, func(t *testing.T) {
got, err := vfs.ParseMountInfo(strings.NewReader(tc.sample)) got, n, err := vfs.ParseMountInfo(strings.NewReader(tc.sample))
if !errors.Is(err, tc.wantErr) { if !errors.Is(err, tc.wantErr) {
if tc.wantError == "" { if tc.wantError == "" {
t.Errorf("ParseMountInfo: error = %v, wantErr %v", t.Errorf("ParseMountInfo: error = %v, wantErr %v",
@ -131,6 +131,14 @@ id 20 0:53 / /mnt/test rw,relatime shared:212 - tmpfs rw
} }
} }
wantCount := len(tc.want)
if tc.wantErr != nil || tc.wantError != "" {
wantCount = -1
}
if n != wantCount {
t.Errorf("ParseMountInfo: got %d entries, want %d", n, wantCount)
}
i := 0 i := 0
for cur := got; cur != nil; cur = cur.Next { for cur := got; cur != nil; cur = cur.Next {
if i == len(tc.want) { if i == len(tc.want) {