aboutsummaryrefslogtreecommitdiffstats
path: root/vendor/github.com/edsrzf/mmap-go/mmap.go
blob: 7bb4965ed52746641d4fc3d70077477449940371 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
// 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) {
    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
}