#![deny(missing_docs)]
#[macro_use]
extern crate bitflags;
extern crate libc;
pub use error::{Error, Result};
pub use lock::{lock, unlock, LockGuard};
pub use protect::{protect, protect_with_handle, ProtectGuard, Protection};
mod error;
mod lock;
mod os;
pub mod page;
mod protect;
#[derive(Debug, Clone, Copy)]
pub struct Region {
pub base: *const u8,
pub guarded: bool,
pub protection: Protection,
pub shared: bool,
pub size: usize,
}
impl Region {
pub fn lower(&self) -> usize {
self.base as usize
}
pub fn upper(&self) -> usize {
self.lower() + self.size
}
}
unsafe impl Send for Region {}
unsafe impl Sync for Region {}
pub fn query(address: *const u8) -> Result<Region> {
if address.is_null() {
Err(Error::NullAddress)?;
}
os::get_region(page::floor(address as usize) as *const u8)
}
pub fn query_range(address: *const u8, size: usize) -> Result<Vec<Region>> {
if size == 0 {
Err(Error::EmptyRange)?;
}
let mut result = Vec::new();
let mut base = page::floor(address as usize);
let limit = address as usize + size;
loop {
let region = query(base as *const u8)?;
result.push(region);
base = region.upper();
if limit <= region.upper() {
break;
}
}
Ok(result)
}
#[cfg(test)]
mod tests {
extern crate memmap;
use self::memmap::MmapMut;
use super::*;
pub fn alloc_pages(prots: &[Protection]) -> MmapMut {
let pz = page::size();
let map = MmapMut::map_anon(pz * prots.len()).unwrap();
let mut base = map.as_ptr();
for protection in prots {
unsafe {
protect(base, pz, *protection).unwrap();
base = base.offset(pz as isize);
}
}
map
}
#[test]
fn query_null() {
assert!(query(::std::ptr::null()).is_err());
}
#[test]
#[cfg(unix)]
fn query_code() {
let region = query(&query_code as *const _ as *const u8).unwrap();
assert_eq!(region.guarded, false);
if cfg!(not(target_os = "freebsd")) { assert_eq!(region.protection, Protection::ReadExecute);
}
assert_eq!(region.shared, false);
}
#[test]
fn query_alloc() {
let size = page::size() * 2;
let mut map = alloc_pages(&[Protection::ReadExecute, Protection::ReadExecute]);
let region = query(map.as_ptr()).unwrap();
assert_eq!(region.guarded, false);
assert_eq!(region.protection, Protection::ReadExecute);
assert!(!region.base.is_null() && region.base <= map.as_mut_ptr());
assert!(region.size >= size);
}
#[test]
fn query_area_zero() {
assert!(query_range(&query_area_zero as *const _ as *const u8, 0).is_err());
}
#[test]
fn query_area_overlap() {
let pz = page::size();
let prots = [Protection::ReadExecute, Protection::ReadWrite];
let map = alloc_pages(&prots);
let address = unsafe { map.as_ptr().offset(pz as isize - 1) };
let result = query_range(address, 2).unwrap();
assert_eq!(result.len(), prots.len());
for i in 0..prots.len() {
assert_eq!(result[i].protection, prots[i]);
}
}
#[test]
fn query_area_alloc() {
let pz = page::size();
let prots = [
Protection::Read,
Protection::ReadWrite,
Protection::ReadExecute,
];
let map = alloc_pages(&prots);
let result = query_range(map.as_ptr(), pz).unwrap();
assert_eq!(result.len(), 1);
assert_eq!(result[0].protection, prots[0]);
let result = query_range(map.as_ptr(), pz * prots.len()).unwrap();
assert_eq!(result.len(), prots.len());
assert_eq!(result[1].size, pz);
for i in 0..prots.len() {
assert_eq!(result[i].protection, prots[i]);
}
}
}