slint/internal/backends/linuxkms/renderer/femtovg.rs
Simon Hausmann 98dc4a8657 FemtoVG: Abstract the rest of the FemtoVGRenderer type to be independent from OpenGL
The main visible external change is that FemtoVGRenderer is now a generic type, but in the public Slint API the type alias that uses the OpenGL backend is re-exported under the old name. This looks a little different in rustdoc
but appears to be source compatible.
2025-04-25 10:08:38 +02:00

202 lines
7.2 KiB
Rust

// Copyright © SixtyFPS GmbH <info@slint.dev>
// SPDX-License-Identifier: GPL-3.0-only OR LicenseRef-Slint-Royalty-free-2.0 OR LicenseRef-Slint-Software-3.0
use std::{num::NonZeroU32, rc::Rc};
use i_slint_core::item_rendering::ItemRenderer;
use i_slint_core::platform::PlatformError;
use i_slint_renderer_femtovg::FemtoVGRendererExt;
use raw_window_handle::{HasDisplayHandle, HasWindowHandle};
use glutin::{
context::{ContextApi, ContextAttributesBuilder},
display::GetGlDisplay,
prelude::*,
surface::{SurfaceAttributesBuilder, WindowSurface},
};
use crate::display::{gbmdisplay::GbmDisplay, Presenter, RenderingRotation};
use crate::drmoutput::DrmOutput;
pub struct FemtoVGRendererAdapter {
renderer:
i_slint_renderer_femtovg::FemtoVGRenderer<i_slint_renderer_femtovg::opengl::OpenGLBackend>,
gbm_display: Rc<GbmDisplay>,
}
struct GlContextWrapper {
glutin_context: glutin::context::PossiblyCurrentContext,
glutin_surface: glutin::surface::Surface<glutin::surface::WindowSurface>,
gbm_display: Rc<GbmDisplay>,
}
impl GlContextWrapper {
fn new(gbm_display: &Rc<GbmDisplay>) -> Result<Self, PlatformError> {
let (width, height) = gbm_display.drm_output.size();
let width: std::num::NonZeroU32 = width.try_into().map_err(|_| {
format!("Attempting to create window surface with an invalid width: {}", width)
})?;
let height: std::num::NonZeroU32 = height.try_into().map_err(|_| {
format!("Attempting to create window surface with an invalid height: {}", height)
})?;
let display_handle = gbm_display.display_handle().unwrap();
let window_handle = gbm_display.window_handle().unwrap();
let gl_display = unsafe {
glutin::display::Display::new(
display_handle.as_raw(),
glutin::display::DisplayApiPreference::Egl,
)
.map_err(|e| format!("Error creating EGL display: {e}"))?
};
let config_template = gbm_display.config_template_builder().build();
let config = unsafe {
gl_display
.find_configs(config_template)
.map_err(|e| format!("Error locating EGL configs: {e}"))?
.filter(|config| gbm_display.filter_gl_config(config))
.reduce(|accum, config| {
let transparency_check = config.supports_transparency().unwrap_or(false)
& !accum.supports_transparency().unwrap_or(false);
if transparency_check || config.num_samples() < accum.num_samples() {
config
} else {
accum
}
})
.ok_or("Unable to find suitable GL config")?
};
let gles_context_attributes = ContextAttributesBuilder::new()
.with_context_api(ContextApi::Gles(Some(glutin::context::Version {
major: 2,
minor: 0,
})))
.build(Some(window_handle.as_raw()));
let fallback_context_attributes =
ContextAttributesBuilder::new().build(Some(window_handle.as_raw()));
let not_current_gl_context = unsafe {
gl_display
.create_context(&config, &gles_context_attributes)
.or_else(|_| gl_display.create_context(&config, &fallback_context_attributes))
.map_err(|e| format!("Error creating EGL context: {e}"))?
};
let attrs = SurfaceAttributesBuilder::<WindowSurface>::new().build(
window_handle.as_raw(),
width,
height,
);
let surface = unsafe {
config
.display()
.create_window_surface(&config, &attrs)
.map_err(|e| format!("Error creating EGL window surface: {e}"))?
};
let context = not_current_gl_context.make_current(&surface)
.map_err(|glutin_error: glutin::error::Error| -> PlatformError {
format!("FemtoVG Renderer: Failed to make newly created OpenGL context current: {glutin_error}")
.into()
})?;
Ok(Self {
glutin_context: context,
glutin_surface: surface,
gbm_display: gbm_display.clone(),
})
}
}
unsafe impl i_slint_renderer_femtovg::opengl::OpenGLInterface for GlContextWrapper {
fn ensure_current(&self) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
if !self.glutin_context.is_current() {
self.glutin_context.make_current(&self.glutin_surface).map_err(
|glutin_error| -> PlatformError {
format!("FemtoVG: Error making context current: {glutin_error}").into()
},
)?;
}
Ok(())
}
fn swap_buffers(&self) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
// Make sure the in-flight font-buffer from the previous swap_buffers call has been
// posted to the screen.
self.gbm_display.drm_output.wait_for_page_flip();
self.glutin_surface.swap_buffers(&self.glutin_context).map_err(
|glutin_error| -> PlatformError {
format!("FemtoVG: Error swapping buffers: {glutin_error}").into()
},
)?;
Ok(())
}
fn resize(
&self,
_width: NonZeroU32,
_height: NonZeroU32,
) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
// Ignore resize requests
Ok(())
}
fn get_proc_address(&self, name: &std::ffi::CStr) -> *const std::ffi::c_void {
self.glutin_context.display().get_proc_address(name)
}
}
impl FemtoVGRendererAdapter {
pub fn new(
device_opener: &crate::DeviceOpener,
) -> Result<Box<dyn crate::fullscreenwindowadapter::FullscreenRenderer>, PlatformError> {
let drm_output = DrmOutput::new(device_opener)?;
let egl_display = Rc::new(crate::display::gbmdisplay::GbmDisplay::new(drm_output)?);
let renderer = Box::new(Self {
renderer: i_slint_renderer_femtovg::FemtoVGRenderer::new(GlContextWrapper::new(
&egl_display,
)?)?,
gbm_display: egl_display,
});
eprintln!("Using FemtoVG OpenGL renderer");
Ok(renderer)
}
}
impl crate::fullscreenwindowadapter::FullscreenRenderer for FemtoVGRendererAdapter {
fn as_core_renderer(&self) -> &dyn i_slint_core::renderer::Renderer {
&self.renderer
}
fn render_and_present(
&self,
rotation: RenderingRotation,
draw_mouse_cursor_callback: &dyn Fn(&mut dyn ItemRenderer),
) -> Result<(), PlatformError> {
let size = self.size();
self.renderer.render_transformed_with_post_callback(
rotation.degrees(),
rotation.translation_after_rotation(size),
size,
Some(&|item_renderer| {
draw_mouse_cursor_callback(item_renderer);
}),
)?;
self.gbm_display.present()?;
Ok(())
}
fn size(&self) -> i_slint_core::api::PhysicalSize {
let (width, height) = self.gbm_display.drm_output.size();
i_slint_core::api::PhysicalSize::new(width, height)
}
}