Skip to content

Commit

Permalink
Switch EGL to use only pf_reqs and opengl
Browse files Browse the repository at this point in the history
  • Loading branch information
tomaka committed Sep 21, 2015
1 parent 50365b4 commit 85ac7fd
Show file tree
Hide file tree
Showing 7 changed files with 57 additions and 48 deletions.
5 changes: 3 additions & 2 deletions src/api/android/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -112,7 +112,8 @@ impl Window {
return Err(OsError(format!("Android's native window is null")));
}

let context = try!(EglContext::new(egl::ffi::egl::Egl, &builder, egl::NativeDisplay::Android)
let context = try!(EglContext::new(egl::ffi::egl::Egl, &builder.pf_reqs, &builder.opengl,
egl::NativeDisplay::Android)
.and_then(|p| p.finish(native_window as *const _)));

let (tx, rx) = channel();
Expand Down Expand Up @@ -256,7 +257,7 @@ impl HeadlessContext {
/// See the docs in the crate root file.
pub fn new(builder: BuilderAttribs) -> Result<HeadlessContext, CreationError> {
let context = try!(EglContext::new(egl::ffi::egl::Egl, &builder, egl::NativeDisplay::Android));
let context = try!(context.finish_pbuffer());
let context = try!(context.finish_pbuffer(builder.window.dimensions.unwrap_or((800, 600)))); // TODO:
Ok(HeadlessContext(context))
}
}
Expand Down
39 changes: 19 additions & 20 deletions src/api/egl/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,12 +2,13 @@
target_os = "dragonfly", target_os = "freebsd"))]
#![allow(unused_variables)]

use BuilderAttribs;
use ContextError;
use CreationError;
use GlAttributes;
use GlContext;
use GlRequest;
use PixelFormat;
use PixelFormatRequirements;
use Robustness;
use Api;

Expand Down Expand Up @@ -158,11 +159,11 @@ impl Context {
/// This function initializes some things and chooses the pixel format.
///
/// To finish the process, you must call `.finish(window)` on the `ContextPrototype`.
pub fn new<'a>(egl: ffi::egl::Egl, builder: &'a BuilderAttribs<'a>,
native_display: NativeDisplay)
pub fn new<'a>(egl: ffi::egl::Egl, pf_reqs: &PixelFormatRequirements,
opengl: &'a GlAttributes<&'a Context>, native_display: NativeDisplay)
-> Result<ContextPrototype<'a>, CreationError>
{
if builder.opengl.sharing.is_some() {
if opengl.sharing.is_some() {
unimplemented!()
}

Expand Down Expand Up @@ -197,7 +198,7 @@ impl Context {

// binding the right API and choosing the version
let (version, api) = unsafe {
match builder.opengl.version {
match opengl.version {
GlRequest::Latest => {
if egl_version >= (1, 4) {
if egl.BindAPI(ffi::egl::OPENGL_API) != 0 {
Expand Down Expand Up @@ -246,10 +247,10 @@ impl Context {
};

let configs = unsafe { try!(enumerate_configs(&egl, display, &egl_version, api, version)) };
let (config_id, pixel_format) = try!(builder.choose_pixel_format(configs.into_iter()));
let (config_id, pixel_format) = try!(pf_reqs.choose_pixel_format(configs.into_iter()));

Ok(ContextPrototype {
builder: builder,
opengl: opengl,
egl: egl,
display: display,
egl_version: egl_version,
Expand Down Expand Up @@ -330,7 +331,7 @@ impl Drop for Context {
}

pub struct ContextPrototype<'a> {
builder: &'a BuilderAttribs<'a>,
opengl: &'a GlAttributes<&'a Context>,
egl: ffi::egl::Egl,
display: ffi::egl::types::EGLDisplay,
egl_version: (ffi::egl::types::EGLint, ffi::egl::types::EGLint),
Expand Down Expand Up @@ -366,9 +367,7 @@ impl<'a> ContextPrototype<'a> {
self.finish_impl(surface)
}

pub fn finish_pbuffer(self) -> Result<Context, CreationError> {
let dimensions = self.builder.window.dimensions.unwrap_or((800, 600));

pub fn finish_pbuffer(self, dimensions: (u32, u32)) -> Result<Context, CreationError> {
let attrs = &[
ffi::egl::WIDTH as libc::c_int, dimensions.0 as libc::c_int,
ffi::egl::HEIGHT as libc::c_int, dimensions.1 as libc::c_int,
Expand All @@ -394,18 +393,18 @@ impl<'a> ContextPrototype<'a> {
if let Some(version) = self.version {
try!(create_context(&self.egl, self.display, &self.egl_version,
&self.extensions, self.api, version, self.config_id,
self.builder.opengl.debug, self.builder.opengl.robustness))
self.opengl.debug, self.opengl.robustness))

} else if self.api == Api::OpenGlEs {
if let Ok(ctxt) = create_context(&self.egl, self.display, &self.egl_version,
&self.extensions, self.api, (2, 0), self.config_id,
self.builder.opengl.debug, self.builder.opengl.robustness)
self.opengl.debug, self.opengl.robustness)
{
ctxt
} else if let Ok(ctxt) = create_context(&self.egl, self.display, &self.egl_version,
&self.extensions, self.api, (1, 0),
self.config_id, self.builder.opengl.debug,
self.builder.opengl.robustness)
self.config_id, self.opengl.debug,
self.opengl.robustness)
{
ctxt
} else {
Expand All @@ -415,19 +414,19 @@ impl<'a> ContextPrototype<'a> {
} else {
if let Ok(ctxt) = create_context(&self.egl, self.display, &self.egl_version,
&self.extensions, self.api, (3, 2), self.config_id,
self.builder.opengl.debug, self.builder.opengl.robustness)
self.opengl.debug, self.opengl.robustness)
{
ctxt
} else if let Ok(ctxt) = create_context(&self.egl, self.display, &self.egl_version,
&self.extensions, self.api, (3, 1),
self.config_id, self.builder.opengl.debug,
self.builder.opengl.robustness)
self.config_id, self.opengl.debug,
self.opengl.robustness)
{
ctxt
} else if let Ok(ctxt) = create_context(&self.egl, self.display, &self.egl_version,
&self.extensions, self.api, (1, 0),
self.config_id, self.builder.opengl.debug,
self.builder.opengl.robustness)
self.config_id, self.opengl.debug,
self.opengl.robustness)
{
ctxt
} else {
Expand Down
2 changes: 1 addition & 1 deletion src/api/wayland/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -291,7 +291,7 @@ impl Window {
});
try!(EglContext::new(
egl,
&builder,
&builder.pf_reqs, &builder.opengl.clone().map_sharing(|_| unimplemented!()), // TODO:
egl::NativeDisplay::Wayland(Some(wayland_context.display.ptr() as *const _)))
.and_then(|p| p.finish((**shell_window.get_shell()).ptr() as *const _))
)
Expand Down
2 changes: 1 addition & 1 deletion src/api/win32/init.rs
Original file line number Diff line number Diff line change
Expand Up @@ -162,7 +162,7 @@ unsafe fn init(title: Vec<u16>, builder: BuilderAttribs<'static>,
let context = match builder.opengl.version {
GlRequest::Specific(Api::OpenGlEs, (_major, _minor)) => {
if let Some(egl) = egl {
if let Ok(c) = EglContext::new(egl, &builder,
if let Ok(c) = EglContext::new(egl, &builder.pf_reqs, &builder.opengl.clone().map_sharing(|_| unimplemented!()),
egl::NativeDisplay::Other(Some(ptr::null())))
.and_then(|p| p.finish(real_window.0))
{
Expand Down
4 changes: 2 additions & 2 deletions src/api/x11/window.rs
Original file line number Diff line number Diff line change
Expand Up @@ -356,14 +356,14 @@ impl Window {
if let Some(ref glx) = display.glx {
Prototype::Glx(try!(GlxContext::new(glx.clone(), &display.xlib, &builder_clone, display.display)))
} else if let Some(ref egl) = display.egl {
Prototype::Egl(try!(EglContext::new(egl.clone(), &builder_clone, egl::NativeDisplay::X11(Some(display.display as *const _)))))
Prototype::Egl(try!(EglContext::new(egl.clone(), &builder_clone.pf_reqs, &builder_clone.opengl, egl::NativeDisplay::X11(Some(display.display as *const _)))))
} else {
return Err(CreationError::NotSupported);
}
},
GlRequest::Specific(Api::OpenGlEs, _) => {
if let Some(ref egl) = display.egl {
Prototype::Egl(try!(EglContext::new(egl.clone(), &builder_clone, egl::NativeDisplay::X11(Some(display.display as *const _)))))
Prototype::Egl(try!(EglContext::new(egl.clone(), &builder_clone.pf_reqs, &builder_clone.opengl, egl::NativeDisplay::X11(Some(display.display as *const _)))))
} else {
return Err(CreationError::NotSupported);
}
Expand Down
48 changes: 28 additions & 20 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -402,32 +402,53 @@ impl<'a> BuilderAttribs<'a> {
(new_attribs, sharing)
}

fn choose_pixel_format<T, I>(&self, iter: I) -> Result<(T, PixelFormat), CreationError>
where I: IntoIterator<Item=(T, PixelFormat)>, T: Clone
{
self.pf_reqs.choose_pixel_format(iter)
}
}

/// VERY UNSTABLE! Describes how the backend should choose a pixel format.
#[derive(Clone, Debug)]
#[allow(missing_docs)]
pub struct PixelFormatRequirements {
pub multisampling: Option<u16>,
pub depth_bits: Option<u8>,
pub stencil_bits: Option<u8>,
pub color_bits: Option<u8>,
pub alpha_bits: Option<u8>,
pub stereoscopy: bool,
pub srgb: Option<bool>,
}

impl PixelFormatRequirements {
fn choose_pixel_format<T, I>(&self, iter: I) -> Result<(T, PixelFormat), CreationError>
where I: IntoIterator<Item=(T, PixelFormat)>, T: Clone
{
// filtering formats that don't match the requirements
let iter = iter.into_iter().filter(|&(_, ref format)| {
if format.color_bits < self.pf_reqs.color_bits.unwrap_or(0) {
if format.color_bits < self.color_bits.unwrap_or(0) {
return false;
}

if format.alpha_bits < self.pf_reqs.alpha_bits.unwrap_or(0) {
if format.alpha_bits < self.alpha_bits.unwrap_or(0) {
return false;
}

if format.depth_bits < self.pf_reqs.depth_bits.unwrap_or(0) {
if format.depth_bits < self.depth_bits.unwrap_or(0) {
return false;
}

if format.stencil_bits < self.pf_reqs.stencil_bits.unwrap_or(0) {
if format.stencil_bits < self.stencil_bits.unwrap_or(0) {
return false;
}

if !format.stereoscopy && self.pf_reqs.stereoscopy {
if !format.stereoscopy && self.stereoscopy {
return false;
}

if let Some(req_ms) = self.pf_reqs.multisampling {
if let Some(req_ms) = self.multisampling {
match format.multisampling {
Some(val) if val >= req_ms => (),
_ => return false
Expand All @@ -438,7 +459,7 @@ impl<'a> BuilderAttribs<'a> {
}
}

if let Some(srgb) = self.pf_reqs.srgb {
if let Some(srgb) = self.srgb {
if srgb != format.srgb {
return false;
}
Expand Down Expand Up @@ -502,19 +523,6 @@ impl<'a> BuilderAttribs<'a> {
}
}

/// VERY UNSTABLE! Describes how the backend should choose a pixel format.
#[derive(Clone, Debug)]
#[allow(missing_docs)]
pub struct PixelFormatRequirements {
pub multisampling: Option<u16>,
pub depth_bits: Option<u8>,
pub stencil_bits: Option<u8>,
pub color_bits: Option<u8>,
pub alpha_bits: Option<u8>,
pub stereoscopy: bool,
pub srgb: Option<bool>,
}

impl Default for PixelFormatRequirements {
fn default() -> PixelFormatRequirements {
PixelFormatRequirements {
Expand Down
5 changes: 3 additions & 2 deletions src/platform/windows/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -91,8 +91,9 @@ impl HeadlessContext {
// if EGL is available, we try using EGL first
// if EGL returns an error, we try the hidden window method
if let &Some(ref egl) = &*EGL {
let context = EglContext::new(egl.0.clone(), &builder, egl::NativeDisplay::Other(None))
.and_then(|prototype| prototype.finish_pbuffer())
let context = EglContext::new(egl.0.clone(), &builder.pf_reqs, &builder.opengl.clone().map_sharing(|_| unimplemented!()), // TODO:
egl::NativeDisplay::Other(None))
.and_then(|prototype| prototype.finish_pbuffer(builder.window.dimensions.unwrap_or((800, 600)))) // TODO:
.map(|ctxt| HeadlessContext::EglPbuffer(ctxt));

if let Ok(context) = context {
Expand Down

0 comments on commit 85ac7fd

Please sign in to comment.