use std::sync::Arc;
use buffer::BufferAccess;
use format::ClearValue;
use format::Format;
use format::FormatDesc;
use image::Dimensions;
use image::ImageInner;
use image::ImageLayout;
use image::ViewType;
use image::sys::UnsafeImageView;
use image::traits::ImageAccess;
use image::traits::ImageClearValue;
use image::traits::ImageContent;
use image::traits::ImageViewAccess;
use swapchain::Swapchain;
use sync::AccessError;
use OomError;
pub struct SwapchainImage<W> {
swapchain: Arc<Swapchain<W>>,
image_offset: usize,
view: UnsafeImageView,
}
impl<W> SwapchainImage<W> {
pub unsafe fn from_raw(swapchain: Arc<Swapchain<W>>, id: usize)
-> Result<Arc<SwapchainImage<W>>, OomError> {
let image = swapchain.raw_image(id).unwrap();
let view = UnsafeImageView::raw(&image.image, ViewType::Dim2d, 0 .. 1, 0 .. 1)?;
Ok(Arc::new(SwapchainImage {
swapchain: swapchain.clone(),
image_offset: id,
view: view,
}))
}
#[inline]
pub fn dimensions(&self) -> [u32; 2] {
let dims = self.my_image().image.dimensions();
[dims.width(), dims.height()]
}
#[inline]
pub fn swapchain(&self) -> &Arc<Swapchain<W>> {
&self.swapchain
}
#[inline]
fn my_image(&self) -> ImageInner {
self.swapchain.raw_image(self.image_offset).unwrap()
}
#[inline]
fn layout_initialized(&self) {
self.swapchain.image_layout_initialized(self.image_offset);
}
#[inline]
fn is_layout_initialized(&self) -> bool {
self.swapchain.is_image_layout_initialized(self.image_offset)
}
}
unsafe impl<W> ImageAccess for SwapchainImage<W> {
#[inline]
fn inner(&self) -> ImageInner {
self.my_image()
}
#[inline]
fn initial_layout_requirement(&self) -> ImageLayout {
ImageLayout::PresentSrc
}
#[inline]
fn final_layout_requirement(&self) -> ImageLayout {
ImageLayout::PresentSrc
}
#[inline]
fn conflicts_buffer(&self, other: &dyn BufferAccess) -> bool {
false
}
#[inline]
fn conflicts_image(&self, other: &dyn ImageAccess) -> bool {
self.my_image().image.key() == other.conflict_key() }
#[inline]
fn conflict_key(&self) -> u64 {
self.my_image().image.key()
}
#[inline]
fn try_gpu_lock(&self, _: bool, _: ImageLayout) -> Result<(), AccessError> {
Err(AccessError::SwapchainImageAcquireOnly)
}
#[inline]
unsafe fn layout_initialized(&self) {
self.layout_initialized();
}
#[inline]
fn is_layout_initialized(&self) -> bool{
self.is_layout_initialized()
}
#[inline]
unsafe fn increase_gpu_lock(&self) {
}
#[inline]
unsafe fn unlock(&self, _: Option<ImageLayout>) {
}
}
unsafe impl<W> ImageClearValue<<Format as FormatDesc>::ClearValue> for SwapchainImage<W> {
#[inline]
fn decode(&self, value: <Format as FormatDesc>::ClearValue) -> Option<ClearValue> {
Some(self.swapchain.format().decode_clear_value(value))
}
}
unsafe impl<P,W> ImageContent<P> for SwapchainImage<W> {
#[inline]
fn matches_format(&self) -> bool {
true }
}
unsafe impl<W> ImageViewAccess for SwapchainImage<W> {
#[inline]
fn parent(&self) -> &dyn ImageAccess {
self
}
#[inline]
fn dimensions(&self) -> Dimensions {
let dims = self.swapchain.dimensions();
Dimensions::Dim2d {
width: dims[0],
height: dims[1],
}
}
#[inline]
fn inner(&self) -> &UnsafeImageView {
&self.view
}
#[inline]
fn descriptor_set_storage_image_layout(&self) -> ImageLayout {
ImageLayout::ShaderReadOnlyOptimal
}
#[inline]
fn descriptor_set_combined_image_sampler_layout(&self) -> ImageLayout {
ImageLayout::ShaderReadOnlyOptimal
}
#[inline]
fn descriptor_set_sampled_image_layout(&self) -> ImageLayout {
ImageLayout::ShaderReadOnlyOptimal
}
#[inline]
fn descriptor_set_input_attachment_layout(&self) -> ImageLayout {
ImageLayout::ShaderReadOnlyOptimal
}
#[inline]
fn identity_swizzle(&self) -> bool {
true
}
}