use crate::{HeaderTagFlag, HeaderTagHeader, HeaderTagType};
use core::fmt;
use core::fmt::{Debug, Formatter};
use core::mem;
use multiboot2_common::{MaybeDynSized, Tag};
#[repr(u32)]
#[derive(Copy, Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum RelocatableHeaderTagPreference {
    None = 0,
    Low = 1,
    High = 2,
}
#[derive(Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
#[repr(C, align(8))]
pub struct RelocatableHeaderTag {
    header: HeaderTagHeader,
    min_addr: u32,
    max_addr: u32,
    align: u32,
    preference: RelocatableHeaderTagPreference,
}
impl RelocatableHeaderTag {
    #[must_use]
    pub const fn new(
        flags: HeaderTagFlag,
        min_addr: u32,
        max_addr: u32,
        align: u32,
        preference: RelocatableHeaderTagPreference,
    ) -> Self {
        let header = HeaderTagHeader::new(
            HeaderTagType::Relocatable,
            flags,
            mem::size_of::<Self>() as u32,
        );
        Self {
            header,
            min_addr,
            max_addr,
            align,
            preference,
        }
    }
    #[must_use]
    pub const fn typ(&self) -> HeaderTagType {
        self.header.typ()
    }
    #[must_use]
    pub const fn flags(&self) -> HeaderTagFlag {
        self.header.flags()
    }
    #[must_use]
    pub const fn size(&self) -> u32 {
        self.header.size()
    }
    #[must_use]
    pub const fn min_addr(&self) -> u32 {
        self.min_addr
    }
    #[must_use]
    pub const fn max_addr(&self) -> u32 {
        self.max_addr
    }
    #[must_use]
    pub const fn align(&self) -> u32 {
        self.align
    }
    #[must_use]
    pub const fn preference(&self) -> RelocatableHeaderTagPreference {
        self.preference
    }
}
impl Debug for RelocatableHeaderTag {
    fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
        f.debug_struct("RelocatableHeaderTag")
            .field("type", &self.typ())
            .field("flags", &self.flags())
            .field("size", &self.size())
            .field("min_addr", &(self.min_addr as *const u32))
            .field("max_addr", &(self.max_addr as *const u32))
            .field("align", &{ self.align })
            .field("preference", &{ self.preference })
            .finish()
    }
}
impl MaybeDynSized for RelocatableHeaderTag {
    type Header = HeaderTagHeader;
    const BASE_SIZE: usize = mem::size_of::<Self>();
    fn dst_len(_header: &Self::Header) -> Self::Metadata {}
}
impl Tag for RelocatableHeaderTag {
    type IDType = HeaderTagType;
    const ID: HeaderTagType = HeaderTagType::Relocatable;
}
#[cfg(test)]
mod tests {
    use crate::RelocatableHeaderTag;
    #[test]
    fn test_assert_size() {
        assert_eq!(
            core::mem::size_of::<RelocatableHeaderTag>(),
            2 + 2 + 4 + 4 + 4 + 4 + 4
        );
    }
}