// 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. //go:build darwin && go1.12 // +build darwin,go1.12 package ld import ( "syscall" "unsafe" ) // Implemented in the syscall package. //go:linkname fcntl syscall.fcntl func fcntl(fd int, cmd int, arg int) (int, error) func (out *OutBuf) fallocate(size uint64) error { stat, err := out.f.Stat() if err != nil { return err } // F_PEOFPOSMODE allocates from the end of the file, so we want the size difference. // Apparently, it uses the end of the allocation, instead of the logical end of the // the file. cursize := uint64(stat.Sys().(*syscall.Stat_t).Blocks * 512) // allocated size if size <= cursize { return nil } store := &syscall.Fstore_t{ Flags: syscall.F_ALLOCATEALL, Posmode: syscall.F_PEOFPOSMODE, Offset: 0, Length: int64(size - cursize), } _, err = fcntl(int(out.f.Fd()), syscall.F_PREALLOCATE, int(uintptr(unsafe.Pointer(store)))) return err } func (out *OutBuf) purgeSignatureCache() { // Apparently, the Darwin kernel may cache the code signature at mmap. // When we mmap the output buffer, it doesn't have a code signature // (as we haven't generated one). Invalidate the kernel cache now that // we have generated the signature. See issue #42684. syscall.Syscall(syscall.SYS_MSYNC, uintptr(unsafe.Pointer(&out.buf[0])), uintptr(len(out.buf)), syscall.MS_INVALIDATE) // Best effort. Ignore error. }