Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update macOS system surface so it can work with multiple surfaces #267

Closed
wants to merge 1 commit into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion surfman/examples/chaos_game.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ use euclid::default::Point2D;
use rand::{self, Rng};
use surfman::{SurfaceAccess, SurfaceType};
use winit::dpi::PhysicalSize;
use winit::event::{DeviceEvent, Event, WindowEvent, KeyboardInput, VirtualKeyCode};
use winit::event::{DeviceEvent, Event, KeyboardInput, VirtualKeyCode, WindowEvent};
use winit::event_loop::{ControlFlow, EventLoop};
use winit::window::WindowBuilder;

Expand Down
9 changes: 5 additions & 4 deletions surfman/examples/threads.rs
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ use winit::{
dpi::PhysicalSize,
event::{DeviceEvent, Event, KeyboardInput, VirtualKeyCode, WindowEvent},
event_loop::{ControlFlow, EventLoop},
window::WindowBuilder
window::WindowBuilder,
};

pub mod common;
Expand Down Expand Up @@ -146,9 +146,11 @@ fn main() {
}),
..
} => *control_flow = ControlFlow::Exit,
_ => { app.tick(true); *control_flow = ControlFlow::Poll; }
_ => {
app.tick(true);
*control_flow = ControlFlow::Poll;
}
});

}
pub struct App {
main_from_worker_receiver: Receiver<Frame>,
Expand All @@ -169,7 +171,6 @@ impl Drop for App {
}

impl App {

pub fn new(
connection: Connection,
adapter: Adapter,
Expand Down
2 changes: 1 addition & 1 deletion surfman/src/chains.rs
Original file line number Diff line number Diff line change
Expand Up @@ -22,8 +22,8 @@

#![allow(missing_docs)]

use crate::{ContextID, Error, SurfaceAccess, SurfaceInfo, SurfaceType};
use crate::device::Device as DeviceAPI;
use crate::{ContextID, Error, SurfaceAccess, SurfaceInfo, SurfaceType};
use euclid::default::Size2D;
use fnv::{FnvHashMap, FnvHashSet};
use log::debug;
Expand Down
16 changes: 11 additions & 5 deletions surfman/src/platform/android/context.rs
Original file line number Diff line number Diff line change
Expand Up @@ -316,11 +316,17 @@ impl Device {
context::get_proc_address(symbol_name)
}

pub(crate) fn context_to_egl_config(
&self,
context: &Context,
) -> EGLConfig {
unsafe { context::egl_config_from_id(self.egl_display, context::get_context_attr(self.egl_display, context.egl_context, egl::CONFIG_ID as EGLint)) }
pub(crate) fn context_to_egl_config(&self, context: &Context) -> EGLConfig {
unsafe {
context::egl_config_from_id(
self.egl_display,
context::get_context_attr(
self.egl_display,
context.egl_context,
egl::CONFIG_ID as EGLint,
),
)
}
}

pub(crate) fn temporarily_make_context_current(
Expand Down
111 changes: 101 additions & 10 deletions surfman/src/platform/macos/system/surface.rs
Original file line number Diff line number Diff line change
Expand Up @@ -11,17 +11,19 @@ use crate::{Error, SurfaceAccess, SurfaceID, SurfaceType, SystemSurfaceInfo};
use cocoa::appkit::{NSScreen, NSView as NSViewMethods, NSWindow};
use cocoa::base::{id, YES};
use cocoa::foundation::{NSPoint, NSRect, NSSize};
use cocoa::quartzcore::{transaction, CALayer, CATransform3D};
use cocoa::quartzcore::{transaction, CATransform3D};
use core_foundation::array::{CFArray, CFArrayRef};
use core_foundation::base::TCFType;
use core_foundation::dictionary::CFDictionary;
use core_foundation::number::CFNumber;
use core_foundation::string::CFString;
use core_graphics::geometry::{CGRect, CGSize, CG_ZERO_POINT};
use servo_display_link::macos::cvdisplaylink::{CVDisplayLink, CVTimeStamp, DisplayLink};
use euclid::default::Size2D;
use io_surface::{self, kIOSurfaceBytesPerElement, kIOSurfaceBytesPerRow, IOSurface, IOSurfaceRef};
use io_surface::{kIOSurfaceCacheMode, kIOSurfaceHeight, kIOSurfacePixelFormat, kIOSurfaceWidth};
use mach::kern_return::KERN_SUCCESS;
use objc::runtime::BOOL;
use servo_display_link::macos::cvdisplaylink::{CVDisplayLink, CVTimeStamp, DisplayLink};
use std::fmt::{self, Debug, Formatter};
use std::mem;
use std::os::raw::c_void;
Expand Down Expand Up @@ -174,9 +176,10 @@ impl Device {
surface_access: SurfaceAccess,
native_widget: &NativeWidget,
) -> ViewInfo {
let view = native_widget.view.0;
let front_surface = self.create_io_surface(&size, surface_access);

let window: id = msg_send![native_widget.view.0, window];
let window: id = msg_send![view, window];
let device_description: CFDictionary<CFString, CFNumber> =
CFDictionary::wrap_under_get_rule(window.screen().deviceDescription() as *const _);
let description_key: CFString = CFString::from("NSScreenNumber");
Expand All @@ -195,20 +198,31 @@ impl Device {
transaction::begin();
transaction::set_disable_actions(true);

let superlayer = CALayer::new();
native_widget.view.0.setLayer(superlayer.id());
native_widget.view.0.setWantsLayer(YES);
let layer: id = msg_send![view, layer];
let superlayer = if layer.is_null() {
let layer = CALayer::new();
view.setLayer(layer.id());
view.setWantsLayer(YES);
layer
} else {
CALayer(layer)
};

let layer = if let Some(sublayers) = superlayer.sublayers() {
let layer = sublayers.get_all_values()[0] as id;
CALayer(layer)
} else {
CALayer::new()
};

// Compute logical size.
let window: id = msg_send![native_widget.view.0, window];
let window: id = msg_send![view, window];
let logical_rect: NSRect = msg_send![window, convertRectFromBacking:NSRect {
origin: NSPoint { x: 0.0, y: 0.0 },
size: NSSize { width: size.width as f64, height: size.height as f64 },
}];
let logical_size = logical_rect.size;

let opaque = native_widget.opaque;
let layer = CALayer::new();
let layer_size = CGSize::new(logical_size.width as f64, logical_size.height as f64);
layer.set_frame(&CGRect::new(&CG_ZERO_POINT, &layer_size));
layer.set_contents(front_surface.obj as id);
Expand Down Expand Up @@ -249,7 +263,11 @@ impl Device {
}

/// Resizes a widget surface
pub fn resize_surface(&self, surface: &mut Surface, mut size: Size2D<i32>) -> Result<(), Error> {
pub fn resize_surface(
&self,
surface: &mut Surface,
mut size: Size2D<i32>,
) -> Result<(), Error> {
// The surface will not appear if its width is not a multiple of 4 (i.e. stride is a
// multiple of 16 bytes). Enforce this.
let width = size.width as i32;
Expand Down Expand Up @@ -495,3 +513,76 @@ unsafe extern "C" fn display_link_output_callback(
mem::forget(next_vblank);
kCVReturnSuccess
}

pub struct CALayer(id);

unsafe impl Send for CALayer {}
unsafe impl Sync for CALayer {}

impl Clone for CALayer {
#[inline]
fn clone(&self) -> CALayer {
unsafe { CALayer(msg_send![self.id(), retain]) }
}
}

impl Drop for CALayer {
#[inline]
fn drop(&mut self) {
unsafe { msg_send![self.id(), release] }
}
}

impl CALayer {
#[inline]
pub fn id(&self) -> id {
self.0
}

#[inline]
pub fn new() -> CALayer {
unsafe { CALayer(msg_send![class!(CALayer), layer]) }
}

#[inline]
pub fn set_sublayer_transform(&self, sublayer_transform: CATransform3D) {
unsafe { msg_send![self.id(), setSublayerTransform: sublayer_transform] }
}

#[inline]
pub fn set_frame(&self, frame: &CGRect) {
unsafe { msg_send![self.id(), setFrame:*frame] }
}

#[inline]
pub unsafe fn set_contents(&self, contents: id) {
msg_send![self.id(), setContents: contents]
}

#[inline]
pub fn set_contents_opaque(&self, opaque: bool) {
unsafe { msg_send![self.id(), setContentsOpaque: opaque as BOOL] }
}

#[inline]
pub fn add_sublayer(&self, sublayer: &CALayer) {
unsafe { msg_send![self.id(), addSublayer:sublayer.id()] }
}

#[inline]
pub fn set_opaque(&self, opaque: bool) {
unsafe { msg_send![self.id(), setOpaque: opaque as BOOL] }
}

#[inline]
pub fn sublayers(&self) -> Option<CFArray<CALayer>> {
unsafe {
let sublayers: CFArrayRef = msg_send![self.id(), sublayers];
if sublayers.is_null() {
None
} else {
Some(TCFType::wrap_under_create_rule(sublayers))
}
}
}
}
31 changes: 15 additions & 16 deletions surfman/src/platform/windows/angle/device.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,11 @@

use super::connection::Connection;
use crate::egl;
use crate::egl::types::{EGLAttrib, EGLDisplay, EGLint, EGLDeviceEXT};
use crate::egl::types::{EGLAttrib, EGLDeviceEXT, EGLDisplay, EGLint};
use crate::platform::generic::egl::device::EGL_FUNCTIONS;
use crate::platform::generic::egl::ffi::EGL_DEVICE_EXT;
use crate::platform::generic::egl::ffi::{EGL_D3D11_DEVICE_ANGLE, EGL_EXTENSION_FUNCTIONS};
use crate::platform::generic::egl::ffi::{EGL_NO_DEVICE_EXT, EGL_PLATFORM_DEVICE_EXT};
use crate::platform::generic::egl::ffi::{EGL_DEVICE_EXT};
use crate::{Error, GLApi};

use std::cell::{RefCell, RefMut};
Expand Down Expand Up @@ -186,9 +186,9 @@ impl Device {
debug_assert!(d3d11_feature_level >= D3D_FEATURE_LEVEL_9_3);
let d3d11_device = ComPtr::from_raw(d3d11_device);

let eglCreateDeviceANGLE = EGL_EXTENSION_FUNCTIONS.CreateDeviceANGLE.expect(
"Where's the `EGL_ANGLE_device_creation` extension?",
);
let eglCreateDeviceANGLE = EGL_EXTENSION_FUNCTIONS
.CreateDeviceANGLE
.expect("Where's the `EGL_ANGLE_device_creation` extension?");
let egl_device = eglCreateDeviceANGLE(
EGL_D3D11_DEVICE_ANGLE as EGLint,
d3d11_device.as_raw() as *mut c_void,
Expand Down Expand Up @@ -234,27 +234,26 @@ impl Device {

#[allow(non_snake_case)]
pub(crate) fn from_egl_display(egl_display: EGLDisplay) -> Result<Device, Error> {

let eglQueryDisplayAttribEXT = EGL_EXTENSION_FUNCTIONS.QueryDisplayAttribEXT.expect(
"Where's the `EGL_EXT_device_query` extension?"
);
let eglQueryDeviceAttribEXT = EGL_EXTENSION_FUNCTIONS.QueryDeviceAttribEXT.expect(
"Where's the `EGL_EXT_device_query` extension?"
);
let mut angle_device : EGLAttrib = 0;
let eglQueryDisplayAttribEXT = EGL_EXTENSION_FUNCTIONS
.QueryDisplayAttribEXT
.expect("Where's the `EGL_EXT_device_query` extension?");
let eglQueryDeviceAttribEXT = EGL_EXTENSION_FUNCTIONS
.QueryDeviceAttribEXT
.expect("Where's the `EGL_EXT_device_query` extension?");
let mut angle_device: EGLAttrib = 0;
let result = eglQueryDisplayAttribEXT(
egl_display,
EGL_DEVICE_EXT as EGLint,
&mut angle_device as *mut EGLAttrib
&mut angle_device as *mut EGLAttrib,
);
if result == egl::FALSE {
return Err(Error::DeviceOpenFailed);
}
let mut device : EGLAttrib = 0;
let mut device: EGLAttrib = 0;
let result = eglQueryDeviceAttribEXT(
angle_device as EGLDeviceEXT,
EGL_D3D11_DEVICE_ANGLE as EGLint,
&mut device as *mut EGLAttrib
&mut device as *mut EGLAttrib,
);
if result == egl::FALSE {
return Err(Error::DeviceOpenFailed);
Expand Down