use std::collections::BTreeMap;
use crate::*;
use pretty_hex::PrettyHex;
pub struct PluginMemory {
    pub store: Option<Store<Internal>>,
    pub memory: Memory,
    pub live_blocks: BTreeMap<usize, usize>,
    pub free: Vec<MemoryBlock>,
    pub position: usize,
    pub manifest: Manifest,
}
pub trait ToMemoryBlock {
    fn to_memory_block(&self, mem: &PluginMemory) -> Result<MemoryBlock, Error>;
}
impl ToMemoryBlock for MemoryBlock {
    fn to_memory_block(&self, _mem: &PluginMemory) -> Result<MemoryBlock, Error> {
        Ok(*self)
    }
}
impl ToMemoryBlock for (usize, usize) {
    fn to_memory_block(&self, _mem: &PluginMemory) -> Result<MemoryBlock, Error> {
        Ok(MemoryBlock {
            offset: self.0,
            length: self.1,
        })
    }
}
impl ToMemoryBlock for usize {
    fn to_memory_block(&self, mem: &PluginMemory) -> Result<MemoryBlock, Error> {
        match mem.at_offset(*self) {
            Some(x) => Ok(x),
            None => Err(Error::msg(format!("Invalid memory offset: {}", self))),
        }
    }
}
const PAGE_SIZE: u32 = 65536;
const BLOCK_SIZE_THRESHOLD: usize = 32;
impl PluginMemory {
    pub fn new(store: Store<Internal>, memory: Memory, manifest: Manifest) -> Self {
        PluginMemory {
            free: Vec::new(),
            live_blocks: BTreeMap::new(),
            store: Some(store),
            memory,
            position: 1,
            manifest,
        }
    }
    pub fn store(&self) -> &Store<Internal> {
        self.store.as_ref().unwrap()
    }
    pub fn store_mut(&mut self) -> &mut Store<Internal> {
        self.store.as_mut().unwrap()
    }
    pub fn reinstantiate(&mut self) -> Result<(), Error> {
        if let Some(store) = self.store.take() {
            let engine = store.engine().clone();
            let internal = store.into_data();
            let mut store = Store::new(&engine, internal);
            store.epoch_deadline_callback(|_internal| Err(Error::msg("timeout")));
            self.memory = Memory::new(
                &mut store,
                MemoryType::new(4, self.manifest.as_ref().memory.max_pages),
            )?;
            self.store = Some(store);
        }
        self.reset();
        Ok(())
    }
    pub(crate) fn store_u8(&mut self, offs: usize, data: u8) -> Result<(), MemoryAccessError> {
        trace!("store_u8: offset={offs} data={data:#04x}");
        if offs >= self.size() {
            let buf = &mut [0];
            self.memory.read(&self.store.as_ref().unwrap(), offs, buf)?;
            return Ok(());
        }
        self.memory.data_mut(&mut self.store.as_mut().unwrap())[offs] = data;
        Ok(())
    }
    pub(crate) fn load_u8(&self, offs: usize) -> Result<u8, MemoryAccessError> {
        trace!("load_u8: offset={offs}");
        if offs >= self.size() {
            let buf = &mut [0];
            self.memory.read(&self.store.as_ref().unwrap(), offs, buf)?;
            return Ok(0);
        }
        Ok(self.memory.data(&self.store.as_ref().unwrap())[offs])
    }
    pub(crate) fn store_u64(&mut self, offs: usize, data: u64) -> Result<(), Error> {
        trace!("store_u64: offset={offs} data={data:#18x}");
        let handle = MemoryBlock {
            offset: offs,
            length: 8,
        };
        self.write(handle, data.to_ne_bytes())?;
        Ok(())
    }
    pub(crate) fn load_u64(&self, offs: usize) -> Result<u64, Error> {
        trace!("load_u64: offset={offs}");
        let mut buf = [0; 8];
        let handle = MemoryBlock {
            offset: offs,
            length: 8,
        };
        self.read(handle, &mut buf)?;
        Ok(u64::from_ne_bytes(buf))
    }
    pub fn write(&mut self, pos: impl ToMemoryBlock, data: impl AsRef<[u8]>) -> Result<(), Error> {
        let pos = pos.to_memory_block(self)?;
        assert!(data.as_ref().len() <= pos.length);
        self.memory
            .write(&mut self.store.as_mut().unwrap(), pos.offset, data.as_ref())?;
        Ok(())
    }
    pub fn read(&self, pos: impl ToMemoryBlock, mut data: impl AsMut<[u8]>) -> Result<(), Error> {
        let pos = pos.to_memory_block(self)?;
        assert!(data.as_mut().len() <= pos.length);
        self.memory
            .read(&self.store.as_ref().unwrap(), pos.offset, data.as_mut())?;
        Ok(())
    }
    pub fn size(&self) -> usize {
        self.memory.data_size(&self.store.as_ref().unwrap())
    }
    pub fn pages(&self) -> u32 {
        self.memory.size(&self.store.as_ref().unwrap()) as u32
    }
    pub fn alloc(&mut self, n: usize) -> Result<MemoryBlock, Error> {
        debug!("Allocating {n} bytes");
        for (i, block) in self.free.iter_mut().enumerate() {
            if block.length == n {
                let block = self.free.swap_remove(i);
                self.live_blocks.insert(block.offset, block.length);
                debug!("Found block with exact size at offset {}", block.offset);
                return Ok(block);
            } else if block.length.saturating_sub(n) >= BLOCK_SIZE_THRESHOLD {
                let handle = MemoryBlock {
                    offset: block.offset,
                    length: n,
                };
                debug!(
                    "Using block with size {} at offset {}",
                    block.length, block.offset
                );
                block.offset += n;
                block.length -= n;
                self.live_blocks.insert(handle.offset, handle.length);
                return Ok(handle);
            }
        }
        let new_offset = self.position.saturating_add(n);
        if new_offset >= self.size() {
            debug!("Need more memory");
            let bytes_needed = (new_offset as f64 - self.size() as f64) / PAGE_SIZE as f64;
            let mut pages_needed = bytes_needed.ceil() as u64;
            if pages_needed == 0 {
                pages_needed = 1
            }
            debug!("Requesting {pages_needed} more pages");
            self.memory
                .grow(&mut self.store.as_mut().unwrap(), pages_needed)?;
        }
        let mem = MemoryBlock {
            offset: self.position,
            length: n,
        };
        debug!(
            "Allocated new block: {} bytes at offset {}",
            mem.length, mem.offset
        );
        self.live_blocks.insert(mem.offset, mem.length);
        self.position += n;
        Ok(mem)
    }
    pub fn alloc_bytes(&mut self, data: impl AsRef<[u8]>) -> Result<MemoryBlock, Error> {
        let handle = self.alloc(data.as_ref().len())?;
        self.write(handle, data)?;
        Ok(handle)
    }
    pub fn free(&mut self, offset: usize) {
        debug!("Freeing block at {offset}");
        if let Some(length) = self.live_blocks.remove(&offset) {
            self.free.push(MemoryBlock { offset, length });
        } else {
            return;
        }
        let free_size: usize = self.free.iter().map(|x| x.length).sum();
        if free_size >= 1024 {
            let mut last: Option<MemoryBlock> = None;
            let mut free = Vec::new();
            for block in self.free.iter() {
                match last {
                    None => {
                        free.push(*block);
                    }
                    Some(last) => {
                        if last.offset + last.length == block.offset {
                            free.push(MemoryBlock {
                                offset: last.offset,
                                length: last.length + block.length,
                            });
                        }
                    }
                }
                last = Some(*block);
            }
            self.free = free;
        }
    }
    pub fn dump(&self) {
        let data = self.memory.data(self.store.as_ref().unwrap());
        trace!("{:?}", data[..self.position].hex_dump());
    }
    pub fn reset(&mut self) {
        self.free.clear();
        self.live_blocks.clear();
        self.position = 1;
    }
    pub fn data(&self) -> &[u8] {
        self.memory.data(self.store.as_ref().unwrap())
    }
    pub fn data_mut(&mut self) -> &mut [u8] {
        self.memory.data_mut(self.store.as_mut().unwrap())
    }
    pub fn get(&self, handle: impl ToMemoryBlock) -> Result<&[u8], Error> {
        let handle = handle.to_memory_block(self)?;
        Ok(&self.memory.data(self.store.as_ref().unwrap())
            [handle.offset..handle.offset + handle.length])
    }
    pub fn get_mut(&mut self, handle: impl ToMemoryBlock) -> Result<&mut [u8], Error> {
        let handle = handle.to_memory_block(self)?;
        Ok(&mut self.memory.data_mut(self.store.as_mut().unwrap())
            [handle.offset..handle.offset + handle.length])
    }
    pub fn get_str(&self, handle: impl ToMemoryBlock) -> Result<&str, Error> {
        let handle = handle.to_memory_block(self)?;
        Ok(std::str::from_utf8(
            &self.memory.data(self.store.as_ref().unwrap())
                [handle.offset..handle.offset + handle.length],
        )?)
    }
    pub fn get_mut_str(&mut self, handle: impl ToMemoryBlock) -> Result<&mut str, Error> {
        let handle = handle.to_memory_block(self)?;
        Ok(std::str::from_utf8_mut(
            &mut self.memory.data_mut(self.store.as_mut().unwrap())
                [handle.offset..handle.offset + handle.length],
        )?)
    }
    pub fn ptr(&self, handle: impl ToMemoryBlock) -> Result<*mut u8, Error> {
        let handle = handle.to_memory_block(self)?;
        Ok(unsafe {
            self.memory
                .data_ptr(&self.store.as_ref().unwrap())
                .add(handle.offset)
        })
    }
    pub fn block_length(&self, offs: usize) -> Option<usize> {
        self.live_blocks.get(&offs).cloned()
    }
    pub fn at_offset(&self, offset: usize) -> Option<MemoryBlock> {
        let block_length = self.block_length(offset);
        block_length.map(|length| MemoryBlock { offset, length })
    }
}
#[derive(Clone, Copy)]
pub struct MemoryBlock {
    pub offset: usize,
    pub length: usize,
}
impl From<(usize, usize)> for MemoryBlock {
    fn from(x: (usize, usize)) -> Self {
        MemoryBlock {
            offset: x.0,
            length: x.1,
        }
    }
}
impl MemoryBlock {
    pub fn new(offset: usize, length: usize) -> Self {
        MemoryBlock { offset, length }
    }
}