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
117
118
119
120
121
122
123
124
125
//! A wrapper around X11 Displays.

use crate::glx::types::Display as GlxDisplay;
use crate::{Error, GLApi};
use super::adapter::Adapter;
use super::connection::Connection;

use std::ffi::CStr;
use std::os::raw::c_int;
use std::ptr;
use x11::xlib::{self, Display, XCloseDisplay, XDisplayString, XOpenDisplay};

pub struct Device {
    pub(crate) native_display: Box<dyn NativeDisplay>,
    pub(crate) quirks: Quirks,
}

pub(crate) trait NativeDisplay {
    fn display(&self) -> *mut Display;
    fn is_destroyed(&self) -> bool;
    unsafe fn destroy(&mut self);
}

bitflags! {
    pub struct Quirks: u8 {
        const BROKEN_GLX_TEXTURE_FROM_PIXMAP = 0x01;
    }
}

impl Device {
    #[inline]
    pub fn new(connection: &Connection, _: &Adapter) -> Result<Device, Error> {
        unsafe {
            let display_name = match connection.display_name {
                None => ptr::null(),
                Some(ref display_name) => display_name.as_ptr(),
            };
            let display = XOpenDisplay(display_name);
            if display.is_null() {
                return Err(Error::DeviceOpenFailed);
            }
            Ok(Device {
                native_display: Box::new(OwnedDisplay { display }),
                quirks: Quirks::detect(),
            })
        }
    }

    #[inline]
    pub fn adapter(&self) -> Adapter {
        Adapter
    }

    #[inline]
    pub fn connection(&self) -> Connection {
        unsafe {
            let display_name = XDisplayString(self.native_display.display());
            assert!(!display_name.is_null());
            Connection { display_name: Some(CStr::from_ptr(display_name).to_owned()) }
        }
    }

    #[inline]
    pub fn gl_api() -> GLApi {
        GLApi::GL
    }

    pub(crate) fn glx_display(&self) -> *mut GlxDisplay {
        self.native_display.display() as *mut GlxDisplay
    }
}

pub(crate) struct OwnedDisplay {
    pub(crate) display: *mut Display,
}

impl NativeDisplay for OwnedDisplay {
    #[inline]
    fn display(&self) -> *mut Display {
        debug_assert!(!self.is_destroyed());
        self.display
    }

    #[inline]
    fn is_destroyed(&self) -> bool {
        self.display.is_null()
    }

    unsafe fn destroy(&mut self) {
        assert!(!self.is_destroyed());
        let result = XCloseDisplay(self.display);
        assert_eq!(result, xlib::Success as c_int);
        self.display = ptr::null_mut();
    }
}

pub(crate) struct UnsafeDisplayRef {
    pub(crate) display: *mut Display,
}

impl NativeDisplay for UnsafeDisplayRef {
    #[inline]
    fn display(&self) -> *mut Display {
        debug_assert!(!self.is_destroyed());
        self.display
    }

    #[inline]
    fn is_destroyed(&self) -> bool {
        self.display.is_null()
    }

    unsafe fn destroy(&mut self) {
        assert!(!self.is_destroyed());
        self.display = ptr::null_mut();
    }
}

impl Quirks {
    pub(crate) fn detect() -> Quirks {
        // TODO(pcwalton): Whitelist implementations with working `GLX_texture_from_pixmap`.
        Quirks::BROKEN_GLX_TEXTURE_FROM_PIXMAP
    }
}