gitee.com/quant1x/gox@v1.21.2/cache/mmap.go (about) 1 // Copyright 2011 Evan Shaw. All rights reserved. 2 // Use of this source code is governed by a BSD-style 3 // license that can be found in the LICENSE file. 4 5 // This file defines the common package interface and contains a little bit of 6 // factored out logic. 7 8 // Package mmap allows mapping files into memory. It tries to provide a simple, reasonably portable interface, 9 // but doesn't go out of its way to abstract away every little platform detail. 10 // This specifically means: 11 // - forked processes may or may not inherit mappings 12 // - a file's timestamp may or may not be updated by writes through mappings 13 // - specifying a size larger than the file's actual size can increase the file's size 14 // - If the mapped file is being modified by another process while your program's running, don't expect consistent results between platforms 15 package cache 16 17 import ( 18 "errors" 19 "os" 20 "reflect" 21 "unsafe" 22 ) 23 24 const ( 25 // RDONLY maps the memory read-only. 26 // Attempts to write to the MMap object will result in undefined behavior. 27 RDONLY = 0 28 // RDWR maps the memory as read-write. Writes to the MMap object will update the 29 // underlying file. 30 RDWR = 1 << iota 31 // COPY maps the memory as copy-on-write. Writes to the MMap object will affect 32 // memory, but the underlying file will remain unchanged. 33 COPY 34 // If EXEC is set, the mapped memory is marked as executable. 35 EXEC 36 ) 37 38 const ( 39 // If the ANON flag is set, the mapped memory will not be backed by a file. 40 ANON = 1 << iota 41 ) 42 43 // MMap represents a file mapped into memory. 44 type MMap []byte 45 46 // FileMap maps an entire file into memory. 47 // If ANON is set in flags, f is ignored. 48 func FileMap(f *os.File, prot, flags int) (MMap, error) { 49 return MapRegion(f, -1, prot, flags, 0) 50 } 51 52 // MapRegion maps part of a file into memory. 53 // The offset parameter must be a multiple of the system's page size. 54 // If length < 0, the entire file will be mapped. 55 // If ANON is set in flags, f is ignored. 56 func MapRegion(f *os.File, length int, prot, flags int, offset int64) (MMap, error) { 57 if offset%int64(os.Getpagesize()) != 0 { 58 return nil, errors.New("offset parameter must be a multiple of the system's page size") 59 } 60 61 var fd uintptr 62 if flags&ANON == 0 { 63 fd = uintptr(f.Fd()) 64 if length < 0 { 65 fi, err := f.Stat() 66 if err != nil { 67 return nil, err 68 } 69 length = int(fi.Size()) 70 } 71 } else { 72 if length <= 0 { 73 return nil, errors.New("anonymous mapping requires non-zero length") 74 } 75 fd = ^uintptr(0) 76 } 77 return mmap(length, uintptr(prot), uintptr(flags), fd, offset) 78 } 79 80 func (m *MMap) header() *reflect.SliceHeader { 81 return (*reflect.SliceHeader)(unsafe.Pointer(m)) 82 } 83 84 func (m *MMap) addrLen() (uintptr, uintptr) { 85 header := m.header() 86 return header.Data, uintptr(header.Len) 87 } 88 89 // Lock keeps the mapped region in physical memory, ensuring that it will not be 90 // swapped out. 91 func (m MMap) Lock() error { 92 return m.lock() 93 } 94 95 // Unlock reverses the effect of Lock, allowing the mapped region to potentially 96 // be swapped out. 97 // If m is already unlocked, aan error will result. 98 func (m MMap) Unlock() error { 99 return m.unlock() 100 } 101 102 // Flush synchronizes the mapping's contents to the file's contents on disk. 103 func (m MMap) Flush() error { 104 return m.flush() 105 } 106 107 // Unmap deletes the memory mapped region, flushes any remaining changes, and sets 108 // m to nil. 109 // Trying to read or write any remaining references to m after Unmap is called will 110 // result in undefined behavior. 111 // Unmap should only be called on the slice value that was originally returned from 112 // a call to FileMap. Calling Unmap on a derived slice may cause errors. 113 func (m *MMap) Unmap() error { 114 err := m.unmap() 115 *m = nil 116 return err 117 }