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
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
use DrawError;
use uniforms::SamplerBehavior;
use gl;
use context::CommandContext;
use version::Version;
use version::Api;
use GlObject;
use ToGlEnum;
pub struct SamplerObject {
id: gl::types::GLuint,
destroyed: bool,
}
impl SamplerObject {
pub fn new(ctxt: &mut CommandContext, behavior: &SamplerBehavior) -> SamplerObject {
assert!(ctxt.version >= &Version(Api::Gl, 3, 2) ||
ctxt.version >= &Version(Api::GlEs, 3, 0) ||
ctxt.extensions.gl_arb_sampler_objects);
let sampler = unsafe {
let mut sampler: gl::types::GLuint = 0;
ctxt.gl.GenSamplers(1, &mut sampler);
sampler
};
unsafe {
ctxt.gl.SamplerParameteri(sampler, gl::TEXTURE_WRAP_S,
behavior.wrap_function.0.to_glenum() as gl::types::GLint);
ctxt.gl.SamplerParameteri(sampler, gl::TEXTURE_WRAP_T,
behavior.wrap_function.1.to_glenum() as gl::types::GLint);
ctxt.gl.SamplerParameteri(sampler, gl::TEXTURE_WRAP_R,
behavior.wrap_function.2.to_glenum() as gl::types::GLint);
ctxt.gl.SamplerParameteri(sampler, gl::TEXTURE_MIN_FILTER,
behavior.minify_filter.to_glenum() as gl::types::GLint);
ctxt.gl.SamplerParameteri(sampler, gl::TEXTURE_MAG_FILTER,
behavior.magnify_filter.to_glenum() as gl::types::GLint);
if let Some(dtc) = behavior.depth_texture_comparison {
ctxt.gl.SamplerParameteri(sampler, gl::TEXTURE_COMPARE_MODE,
gl::COMPARE_R_TO_TEXTURE as gl::types::GLint);
ctxt.gl.SamplerParameteri(sampler, gl::TEXTURE_COMPARE_FUNC,
dtc.to_glenum() as gl::types::GLint);
}
if let Some(max_value) = ctxt.capabilities.max_texture_max_anisotropy {
let value = if behavior.max_anisotropy as f32 > max_value {
max_value
} else {
behavior.max_anisotropy as f32
};
ctxt.gl.SamplerParameterf(sampler, gl::TEXTURE_MAX_ANISOTROPY_EXT, value);
}
}
SamplerObject {
id: sampler,
destroyed: false,
}
}
#[inline]
pub fn destroy(mut self, ctxt: &mut CommandContext) {
self.destroyed = true;
unsafe {
ctxt.gl.DeleteSamplers(1, [self.id].as_ptr());
}
}
}
impl GlObject for SamplerObject {
type Id = gl::types::GLuint;
#[inline]
fn get_id(&self) -> gl::types::GLuint {
self.id
}
}
impl Drop for SamplerObject {
#[inline]
fn drop(&mut self) {
assert!(self.destroyed);
}
}
pub fn get_sampler(ctxt: &mut CommandContext, behavior: &SamplerBehavior)
-> Result<gl::types::GLuint, DrawError>
{
if ctxt.version < &Version(Api::Gl, 3, 2) && !ctxt.extensions.gl_arb_sampler_objects {
return Err(DrawError::SamplersNotSupported);
}
match ctxt.samplers.get(behavior) {
Some(obj) => return Ok(obj.get_id()),
None => ()
};
let sampler = SamplerObject::new(ctxt, behavior);
let id = sampler.get_id();
ctxt.samplers.insert(behavior.clone(), sampler);
Ok(id)
}