forked from sourcegraph/zoekt
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindexfile.go
95 lines (81 loc) · 2.45 KB
/
indexfile.go
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
// Copyright 2016 Google Inc. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package zoekt
import (
"fmt"
"log"
"os"
"runtime"
// cross-platform memory-mapped file package.
// Benchmarks the same speed as syscall/unix Mmap
// see https://github.com/peterguy/benchmark-mmap
mmap "github.com/edsrzf/mmap-go"
)
type mmapedIndexFile struct {
name string
size uint32
data mmap.MMap
}
func (f *mmapedIndexFile) Read(off, sz uint32) ([]byte, error) {
if off > off+sz || off+sz > uint32(len(f.data)) {
return nil, fmt.Errorf("out of bounds: %d, len %d, name %s", off+sz, len(f.data), f.name)
}
return f.data[off : off+sz], nil
}
func (f *mmapedIndexFile) Name() string {
return f.name
}
func (f *mmapedIndexFile) Size() (uint32, error) {
return f.size, nil
}
func (f *mmapedIndexFile) Close() {
if err := f.data.Unmap(); err != nil {
log.Printf("WARN failed to memory unmap %s: %v", f.name, err)
}
}
func bufferSize(f *mmapedIndexFile) int {
// On Unix/Linux, mmap likes to allocate memory in
// page-sized chunks, so round up to the OS page size.
// mmap will zero-fill the extra bytes.
// On Windows, the Windows API CreateFileMapping method
// requires a buffer the same size as the file.
bsize := int(f.size)
if runtime.GOOS != "windows" {
pagesize := os.Getpagesize() - 1
bsize = (bsize + pagesize) &^ pagesize
}
return bsize
}
// NewIndexFile returns a new index file. The index file takes
// ownership of the passed in file, and may close it.
func NewIndexFile(f *os.File) (IndexFile, error) {
defer f.Close()
fi, err := f.Stat()
if err != nil {
return nil, err
}
sz := fi.Size()
if sz >= maxUInt32 {
return nil, fmt.Errorf("file %s too large: %d", f.Name(), sz)
}
r := &mmapedIndexFile{
name: f.Name(),
size: uint32(sz),
}
r.data, err = mmap.MapRegion(f, bufferSize(r), mmap.RDONLY, 0, 0)
if err != nil {
return nil, fmt.Errorf("NewIndexFile: unable to memory map %s: %w", f.Name(), err)
}
return r, nil
}