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
| // Copyright 2016 Tom Thorogood. All rights reserved.
| // Use of this source code is governed by a
| // Modified BSD License license that can be found in
| // the LICENSE file.
|
| package shm
|
| import (
| "sync/atomic"
| "unsafe"
|
| "golang.org/x/sys/unix"
| )
|
| func OpenSimplex(name string) (*ReadWriteCloser, error) {
| file, err := Open(name, unix.O_RDWR, 0)
| if err != nil {
| return nil, err
| }
|
| defer file.Close()
|
| data, err := unix.Mmap(int(file.Fd()), 0, sharedHeaderSize, unix.PROT_READ, unix.MAP_SHARED)
| if err != nil {
| return nil, err
| }
|
| shared := (*sharedMem)(unsafe.Pointer(&data[0]))
|
| if atomic.LoadUint32((*uint32)(&shared.Version)) != version {
| return nil, ErrInvalidSharedMemory
| }
|
| blockCount, blockSize := uint64(shared.BlockCount), uint64(shared.BlockSize)
|
| if err = unix.Munmap(data); err != nil {
| return nil, err
| }
|
| size := sharedHeaderSize + (blockHeaderSize+blockSize)*blockCount
|
| data, err = unix.Mmap(int(file.Fd()), 0, int(size), unix.PROT_READ|unix.PROT_WRITE, unix.MAP_SHARED)
| if err != nil {
| return nil, err
| }
|
| shared = (*sharedMem)(unsafe.Pointer(&data[0]))
| return &ReadWriteCloser{
| name: name,
|
| data: data,
| readShared: shared,
| writeShared: shared,
| size: size,
| fullBlockSize: blockHeaderSize + blockSize,
|
| Flags: (*[len(shared.Flags)]uint32)(unsafe.Pointer(&shared.Flags[0])),
| }, nil
| }
|
| func OpenDuplex(name string) (*ReadWriteCloser, error) {
| file, err := Open(name, unix.O_RDWR, 0)
| if err != nil {
| return nil, err
| }
|
| defer file.Close()
|
| data, err := unix.Mmap(int(file.Fd()), 0, sharedHeaderSize, unix.PROT_READ, unix.MAP_SHARED)
| if err != nil {
| return nil, err
| }
|
| shared := (*sharedMem)(unsafe.Pointer(&data[0]))
|
| if atomic.LoadUint32((*uint32)(&shared.Version)) != version {
| return nil, ErrInvalidSharedMemory
| }
|
| blockCount, blockSize := uint64(shared.BlockCount), uint64(shared.BlockSize)
|
| if err = unix.Munmap(data); err != nil {
| return nil, err
| }
|
| sharedSize := sharedHeaderSize + (blockHeaderSize+blockSize)*blockCount
| size := 2 * sharedSize
|
| data, err = unix.Mmap(int(file.Fd()), 0, int(size), unix.PROT_READ|unix.PROT_WRITE, unix.MAP_SHARED)
| if err != nil {
| return nil, err
| }
|
| writeShared := (*sharedMem)(unsafe.Pointer(&data[0]))
| return &ReadWriteCloser{
| name: name,
|
| data: data,
| readShared: (*sharedMem)(unsafe.Pointer(&data[sharedSize])),
| writeShared: writeShared,
| size: size,
| fullBlockSize: blockHeaderSize + blockSize,
|
| Flags: (*[len(writeShared.Flags)]uint32)(unsafe.Pointer(&writeShared.Flags[0])),
| }, nil
| }
|
|