Во-первых, исходя из моего понимания, я бы призвал вас быть крайне подозрительным к этому ящику .Это позволяет вам делать вещи в безопасном Rust, что вы должны не .
Например, если у вас есть mmap с файловой поддержкой, то любой процесс на вашем компьютере с правами доступа к файлу может изменить его.Это означает, что:
- Никогда не допустимо, чтобы файл mmapped считался неизменным фрагментом байтов (
&[u8]
), поскольку он может быть изменен! - Он никогда не действителен дляmmapped-файл, который будет рассматриваться как изменяемый фрагмент байтов (
&mut [u8]
), поскольку изменяемая ссылка подразумевает исключительного владельца, который может изменять эти данные, но у вас его нет.
The документация для этого ящика охватывает нет этих проблем и не обсуждает, как вы должны безопасно использовать небольшую горстку unsafe
функций.Для меня это признаки того, что вы можете вводить неопределенное поведение в свой код, что очень плохо.
Например:
use memmap;
use std::{fs, io::prelude::*};
fn main() -> Result<(), Box<std::error::Error>> {
let mut backing_file = fs::OpenOptions::new()
.read(true)
.append(true)
.create(true)
.write(true)
.open("foo.txt")?;
backing_file.write_all(b"initial")?;
let mut mmap_mut = unsafe { memmap::MmapMut::map_mut(&backing_file)? };
let mmap_immut = unsafe { memmap::Mmap::map(&backing_file)? };
// Code after here violates the rules of references, but doesn't use `unsafe`
let a_str: &str = std::str::from_utf8(&mmap_immut)?;
println!("{}", a_str); // initial
mmap_mut[0] = b'x';
// Look, we just changed an "immutable reference"!
println!("{}", a_str); // xnitial
Ok(())
}
Поскольку люди обычно некак если бы вам сказали «нет, не делайте этого, это плохая идея», вот как заставить ваш код «работать»: напрямую добавьте файл и затем заново создайте mmap:
use memmap;
use std::{fs, io::prelude::*, thread, time::Duration};
fn main() -> Result<(), Box<std::error::Error>> {
let mut backing_file = fs::OpenOptions::new()
.read(true)
.append(true)
.create(true)
.write(true)
.open("foo.txt")?;
// mmap requires that the initial mapping be non-zero
backing_file.write_all(b"initial")?;
for _ in 0..3 {
let mmap = unsafe { memmap::MmapMut::map_mut(&backing_file)? };
// I think this line can introduce memory unsafety
println!("{}", std::str::from_utf8(&mmap[..])?);
thread::sleep(Duration::from_secs(1));
backing_file.write_all(b"somestring")?;
}
Ok(())
}
Возможно, вы захотите предварительно выделить «большой» кусок пространства в этом файле, чтобы вы могли просто открыть его и начать писать, вместо того, чтобы заново отображать его.
Я бы не стал использовать этот код ни для чегогде важно, чтобы данные были правильными, сам.
См. также: