use std::cmp;
use std::error;
use std::fmt;
use VulkanObject;
use buffer::TypedBufferAccess;
use device::Device;
use device::DeviceOwned;
pub fn check_copy_buffer<S, D, T>(device: &Device, source: &S, destination: &D)
-> Result<CheckCopyBuffer, CheckCopyBufferError>
where S: ?Sized + TypedBufferAccess<Content = T>,
D: ?Sized + TypedBufferAccess<Content = T>,
T: ?Sized,
{
assert_eq!(source.inner().buffer.device().internal_object(),
device.internal_object());
assert_eq!(destination.inner().buffer.device().internal_object(),
device.internal_object());
if !source.inner().buffer.usage_transfer_source() {
return Err(CheckCopyBufferError::SourceMissingTransferUsage);
}
if !destination.inner().buffer.usage_transfer_destination() {
return Err(CheckCopyBufferError::DestinationMissingTransferUsage);
}
let copy_size = cmp::min(source.size(), destination.size());
if source.conflicts_buffer(0, copy_size, &destination, 0, copy_size) {
return Err(CheckCopyBufferError::OverlappingRanges);
} else {
debug_assert!(!destination.conflicts_buffer(0, copy_size, &source, 0, copy_size));
}
Ok(CheckCopyBuffer { copy_size })
}
pub struct CheckCopyBuffer {
pub copy_size: usize,
}
#[derive(Debug, Copy, Clone)]
pub enum CheckCopyBufferError {
SourceMissingTransferUsage,
DestinationMissingTransferUsage,
OverlappingRanges,
}
impl error::Error for CheckCopyBufferError {
#[inline]
fn description(&self) -> &str {
match *self {
CheckCopyBufferError::SourceMissingTransferUsage => {
"the source buffer is missing the transfer source usage"
},
CheckCopyBufferError::DestinationMissingTransferUsage => {
"the destination buffer is missing the transfer destination usage"
},
CheckCopyBufferError::OverlappingRanges => {
"the source and destination are overlapping"
},
}
}
}
impl fmt::Display for CheckCopyBufferError {
#[inline]
fn fmt(&self, fmt: &mut fmt::Formatter) -> Result<(), fmt::Error> {
write!(fmt, "{}", error::Error::description(self))
}
}