1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
use std::error;
use std::fmt;
use device::Device;
pub fn check_dispatch(device: &Device, dimensions: [u32; 3]) -> Result<(), CheckDispatchError> {
let max = device
.physical_device()
.limits()
.max_compute_work_group_count();
if dimensions[0] > max[0] || dimensions[1] > max[1] || dimensions[2] > max[2] {
return Err(CheckDispatchError::UnsupportedDimensions {
requested: dimensions,
max_supported: max,
});
}
Ok(())
}
#[derive(Debug, Copy, Clone)]
pub enum CheckDispatchError {
UnsupportedDimensions {
requested: [u32; 3],
max_supported: [u32; 3],
},
}
impl error::Error for CheckDispatchError {
#[inline]
fn description(&self) -> &str {
match *self {
CheckDispatchError::UnsupportedDimensions { .. } => {
"the dimensions are too large for the device's limits"
},
}
}
}
impl fmt::Display for CheckDispatchError {
#[inline]
fn fmt(&self, fmt: &mut fmt::Formatter) -> Result<(), fmt::Error> {
write!(fmt, "{}", error::Error::description(self))
}
}
#[cfg(test)]
mod tests {
use command_buffer::validity;
#[test]
fn max_checked() {
let (device, _) = gfx_dev_and_queue!();
let attempted = [u32::max_value(), u32::max_value(), u32::max_value()];
if device
.physical_device()
.limits()
.max_compute_work_group_count() == attempted
{
return;
}
match validity::check_dispatch(&device, attempted) {
Err(validity::CheckDispatchError::UnsupportedDimensions { requested, .. }) => {
assert_eq!(requested, attempted);
},
_ => panic!(),
}
}
}