| // 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 |
| } |