blob: 2c88be1fcbe291e2e84aad1f84d2e9a67ad6f463 [file] [log] [blame]
// Copyright 2020 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 cache
import (
"bytes"
"fmt"
"os"
"path/filepath"
"strings"
"syscall"
"unsafe"
)
func init() {
checkPathCase = darwinCheckPathCase
}
func darwinCheckPathCase(path string) error {
// Darwin provides fcntl(F_GETPATH) to get a path for an arbitrary FD.
// Conveniently for our purposes, it gives the canonical case back. But
// there's no guarantee that it will follow the same route through the
// filesystem that the original path did.
path, err := filepath.Abs(path)
if err != nil {
return err
}
fd, err := syscall.Open(path, os.O_RDONLY, 0)
if err != nil {
return err
}
defer syscall.Close(fd)
buf := make([]byte, 4096) // No MAXPATHLEN in syscall, I think it's 1024, this is bigger.
// Wheeee! syscall doesn't expose a way to call Fcntl except FcntlFlock.
// As of writing, it just passes the pointer through, so we can just lie.
if err := syscall.FcntlFlock(uintptr(fd), syscall.F_GETPATH, (*syscall.Flock_t)(unsafe.Pointer(&buf[0]))); err != nil {
return err
}
buf = buf[:bytes.IndexByte(buf, 0)]
isRoot := func(p string) bool {
return p[len(p)-1] == filepath.Separator
}
// Darwin seems to like having multiple names for the same folder. Match as much of the suffix as we can.
for got, want := path, string(buf); !isRoot(got) && !isRoot(want); got, want = filepath.Dir(got), filepath.Dir(want) {
g, w := filepath.Base(got), filepath.Base(want)
if !strings.EqualFold(g, w) {
break
}
if g != w {
return fmt.Errorf("case mismatch in path %q: component %q is listed by macOS as %q", path, g, w)
}
}
return nil
}