zhangmeng
2019-08-26 a0123f163eddcea3e6b9f9d36f1f3fb3aa2c835a
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
}