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
use memmap2::*;
use std::{
    fs::{OpenOptions,File,metadata}
    ,io::{prelude::*}
};

pub struct FileMmap{
    file:File
    ,mmap:MmapMut
    ,len:u64
}

impl FileMmap{
    pub fn new(path:&str,initial_size:u64) -> Result<FileMmap,std::io::Error>{
        let mut len = match metadata(&path){
            Ok(md)=>md.len()
            ,Err(_)=>0
        };
        match OpenOptions::new()
           .read(true)
           .write(true)
           .create(true)
           .open(&path)
        {
            Ok(mut file)=>{
                if len==0{
                    if let Err(e)=file.set_len(if initial_size==0{
                        1
                    }else{
                        initial_size
                    }){   //サイズが0の場合失敗するようなので0の場合はとりあえず1バイト指定しておく
                        return Err(e);
                    }
                    if let Err(e)=file.seek(std::io::SeekFrom::Start(initial_size)){
                        return Err(e);
                    }
                    len=initial_size;
                }else{
                    file.seek(std::io::SeekFrom::Start(len))?;
                }
                let mmap = unsafe {
                    MmapOptions::new().map_mut(&file).unwrap()
                };
                Ok(FileMmap{
                    file
                    ,mmap
                    ,len
                })
            }
            ,Err(e)=>{
                Err(e)
            }
        }
    }
    pub fn len(&self)->u64{
        self.len
    }
    pub fn as_ptr(&self)->*const i64{
        self.mmap.as_ptr() as *const i64
    }
    pub fn offset(&self,addr:isize)->*const i8{
        unsafe{
            self.mmap.as_ptr().offset(addr) as *const i8
        }
    }
    pub fn slice(&self,addr:isize,len:usize)->&[u8]{
        unsafe{
            std::slice::from_raw_parts(self.mmap.as_ptr().offset(addr),len)
        }
    }
    pub fn set_len(&mut self,len:u64)->std::io::Result<()>{
        self.len=len;
        match self.file.set_len(len){
            Err(e)=>{
                Err(e)
            }
            ,Ok(())=>{
                Ok(())
            }
        }
    }
    pub fn append(&mut self,bytes:&[u8])->Option<u64>{
        let addr=self.len;
        if let Ok(_)=self.set_len(self.len+bytes.len() as u64 + 1){
            self.write(addr,bytes);
            Some(addr)
        }else{
            None
        }
    }
    pub fn write(&mut self,addr:u64,bytes:&[u8]){
        let len=bytes.len();
        unsafe{
            std::ptr::copy(
                bytes.as_ptr()
                ,self.mmap.as_ptr().offset(addr as isize) as *mut u8
                ,len
            );
        }
        self.write_0(addr as isize+len as isize,1);
    }
    pub fn write_0(&mut self,addr:isize,len:u64){
        unsafe{
            std::ptr::write_bytes(
                self.mmap.as_ptr().offset(addr) as *mut u8
                ,0
                ,len as usize
            );
        }
    }
}