Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
116 lines
3.6 KiB
116 lines
3.6 KiB
// Copyright 2011 Evan Shaw. All rights reserved. |
|
// Use of this source code is governed by a BSD-style |
|
// license that can be found in the LICENSE file. |
|
|
|
// This file defines the common package interface and contains a little bit of |
|
// factored out logic. |
|
|
|
// Package mmap allows mapping files into memory. It tries to provide a simple, reasonably portable interface, |
|
// but doesn't go out of its way to abstract away every little platform detail. |
|
// This specifically means: |
|
// * forked processes may or may not inherit mappings |
|
// * a file's timestamp may or may not be updated by writes through mappings |
|
// * specifying a size larger than the file's actual size can increase the file's size |
|
// * If the mapped file is being modified by another process while your program's running, don't expect consistent results between platforms |
|
package mmap |
|
|
|
import ( |
|
"errors" |
|
"os" |
|
"reflect" |
|
"unsafe" |
|
) |
|
|
|
const ( |
|
// RDONLY maps the memory read-only. |
|
// Attempts to write to the MMap object will result in undefined behavior. |
|
RDONLY = 0 |
|
// RDWR maps the memory as read-write. Writes to the MMap object will update the |
|
// underlying file. |
|
RDWR = 1 << iota |
|
// COPY maps the memory as copy-on-write. Writes to the MMap object will affect |
|
// memory, but the underlying file will remain unchanged. |
|
COPY |
|
// If EXEC is set, the mapped memory is marked as executable. |
|
EXEC |
|
) |
|
|
|
const ( |
|
// If the ANON flag is set, the mapped memory will not be backed by a file. |
|
ANON = 1 << iota |
|
) |
|
|
|
// MMap represents a file mapped into memory. |
|
type MMap []byte |
|
|
|
// Map maps an entire file into memory. |
|
// If ANON is set in flags, f is ignored. |
|
func Map(f *os.File, prot, flags int) (MMap, error) { |
|
return MapRegion(f, -1, prot, flags, 0) |
|
} |
|
|
|
// MapRegion maps part of a file into memory. |
|
// The offset parameter must be a multiple of the system's page size. |
|
// If length < 0, the entire file will be mapped. |
|
// If ANON is set in flags, f is ignored. |
|
func MapRegion(f *os.File, length int, prot, flags int, offset int64) (MMap, error) { |
|
if offset%int64(os.Getpagesize()) != 0 { |
|
return nil, errors.New("offset parameter must be a multiple of the system's page size") |
|
} |
|
|
|
var fd uintptr |
|
if flags&ANON == 0 { |
|
fd = uintptr(f.Fd()) |
|
if length < 0 { |
|
fi, err := f.Stat() |
|
if err != nil { |
|
return nil, err |
|
} |
|
length = int(fi.Size()) |
|
} |
|
} else { |
|
if length <= 0 { |
|
return nil, errors.New("anonymous mapping requires non-zero length") |
|
} |
|
fd = ^uintptr(0) |
|
} |
|
return mmap(length, uintptr(prot), uintptr(flags), fd, offset) |
|
} |
|
|
|
func (m *MMap) header() *reflect.SliceHeader { |
|
return (*reflect.SliceHeader)(unsafe.Pointer(m)) |
|
} |
|
|
|
// Lock keeps the mapped region in physical memory, ensuring that it will not be |
|
// swapped out. |
|
func (m MMap) Lock() error { |
|
dh := m.header() |
|
return lock(dh.Data, uintptr(dh.Len)) |
|
} |
|
|
|
// Unlock reverses the effect of Lock, allowing the mapped region to potentially |
|
// be swapped out. |
|
// If m is already unlocked, aan error will result. |
|
func (m MMap) Unlock() error { |
|
dh := m.header() |
|
return unlock(dh.Data, uintptr(dh.Len)) |
|
} |
|
|
|
// Flush synchronizes the mapping's contents to the file's contents on disk. |
|
func (m MMap) Flush() error { |
|
dh := m.header() |
|
return flush(dh.Data, uintptr(dh.Len)) |
|
} |
|
|
|
// Unmap deletes the memory mapped region, flushes any remaining changes, and sets |
|
// m to nil. |
|
// Trying to read or write any remaining references to m after Unmap is called will |
|
// result in undefined behavior. |
|
// Unmap should only be called on the slice value that was originally returned from |
|
// a call to Map. Calling Unmap on a derived slice may cause errors. |
|
func (m *MMap) Unmap() error { |
|
dh := m.header() |
|
err := unmap(dh.Data, uintptr(dh.Len)) |
|
*m = nil |
|
return err |
|
}
|
|
|