csharpc-sharpdotnetxamlavaloniauicross-platformcross-platform-xamlavaloniaguimulti-platformuser-interfacedotnetcore
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
254 lines
9.7 KiB
254 lines
9.7 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using System.ComponentModel;
|
|
using System.Linq;
|
|
using System.Runtime.InteropServices;
|
|
using Avalonia.OpenGL;
|
|
using Avalonia.Platform.Interop;
|
|
using static Avalonia.LinuxFramebuffer.NativeUnsafeMethods;
|
|
using static Avalonia.LinuxFramebuffer.Output.LibDrm;
|
|
namespace Avalonia.LinuxFramebuffer.Output
|
|
{
|
|
public unsafe class DrmOutput : IOutputBackend, IGlPlatformSurface, IWindowingPlatformGlFeature
|
|
{
|
|
private DrmCard _card;
|
|
private readonly EglGlPlatformSurface _eglPlatformSurface;
|
|
public PixelSize PixelSize => _mode.Resolution;
|
|
public double Scaling { get; set; }
|
|
public DrmOutput(string path = null)
|
|
{
|
|
var card = new DrmCard(path);
|
|
|
|
var resources = card.GetResources();
|
|
|
|
|
|
var connector =
|
|
resources.Connectors.FirstOrDefault(x => x.Connection == DrmModeConnection.DRM_MODE_CONNECTED);
|
|
if(connector == null)
|
|
throw new InvalidOperationException("Unable to find connected DRM connector");
|
|
|
|
var mode = connector.Modes.OrderByDescending(x => x.IsPreferred)
|
|
.ThenByDescending(x => x.Resolution.Width * x.Resolution.Height)
|
|
//.OrderByDescending(x => x.Resolution.Width * x.Resolution.Height)
|
|
.FirstOrDefault();
|
|
if(mode == null)
|
|
throw new InvalidOperationException("Unable to find a usable DRM mode");
|
|
Init(card, resources, connector, mode);
|
|
}
|
|
|
|
public DrmOutput(DrmCard card, DrmResources resources, DrmConnector connector, DrmModeInfo modeInfo)
|
|
{
|
|
Init(card, resources, connector, modeInfo);
|
|
}
|
|
|
|
[DllImport("libEGL.so.1")]
|
|
static extern IntPtr eglGetProcAddress(Utf8Buffer proc);
|
|
|
|
private GbmBoUserDataDestroyCallbackDelegate FbDestroyDelegate;
|
|
private drmModeModeInfo _mode;
|
|
private EglDisplay _eglDisplay;
|
|
private EglSurface _eglSurface;
|
|
private EglContext _immediateContext;
|
|
private EglContext _deferredContext;
|
|
private IntPtr _currentBo;
|
|
private IntPtr _gbmTargetSurface;
|
|
private uint _crtcId;
|
|
|
|
void FbDestroyCallback(IntPtr bo, IntPtr userData)
|
|
{
|
|
drmModeRmFB(_card.Fd, userData.ToInt32());
|
|
}
|
|
|
|
uint GetFbIdForBo(IntPtr bo)
|
|
{
|
|
if (bo == IntPtr.Zero)
|
|
throw new ArgumentException("bo is 0");
|
|
var data = gbm_bo_get_user_data(bo);
|
|
if (data != IntPtr.Zero)
|
|
return (uint)data.ToInt32();
|
|
|
|
var w = gbm_bo_get_width(bo);
|
|
var h = gbm_bo_get_height(bo);
|
|
var stride = gbm_bo_get_stride(bo);
|
|
var handle = gbm_bo_get_handle(bo);
|
|
|
|
var ret = drmModeAddFB(_card.Fd, w, h, 24, 32, stride, (uint)handle, out var fbHandle);
|
|
if (ret != 0)
|
|
throw new Win32Exception(ret, "drmModeAddFb failed");
|
|
|
|
gbm_bo_set_user_data(bo, new IntPtr((int)fbHandle), FbDestroyDelegate);
|
|
|
|
|
|
return fbHandle;
|
|
}
|
|
|
|
|
|
void Init(DrmCard card, DrmResources resources, DrmConnector connector, DrmModeInfo modeInfo)
|
|
{
|
|
FbDestroyDelegate = FbDestroyCallback;
|
|
_card = card;
|
|
uint GetCrtc()
|
|
{
|
|
if (resources.Encoders.TryGetValue(connector.EncoderId, out var encoder))
|
|
{
|
|
// Not sure why that should work
|
|
return encoder.Encoder.crtc_id;
|
|
}
|
|
else
|
|
{
|
|
foreach (var encId in connector.EncoderIds)
|
|
{
|
|
if (resources.Encoders.TryGetValue(encId, out encoder)
|
|
&& encoder.PossibleCrtcs.Count>0)
|
|
return encoder.PossibleCrtcs.First().crtc_id;
|
|
}
|
|
|
|
throw new InvalidOperationException("Unable to find CRTC matching the desired mode");
|
|
}
|
|
}
|
|
|
|
_crtcId = GetCrtc();
|
|
var device = gbm_create_device(card.Fd);
|
|
_gbmTargetSurface = gbm_surface_create(device, modeInfo.Resolution.Width, modeInfo.Resolution.Height,
|
|
GbmColorFormats.GBM_FORMAT_XRGB8888, GbmBoFlags.GBM_BO_USE_SCANOUT | GbmBoFlags.GBM_BO_USE_RENDERING);
|
|
if(_gbmTargetSurface == null)
|
|
throw new InvalidOperationException("Unable to create GBM surface");
|
|
|
|
|
|
|
|
_eglDisplay = new EglDisplay(new EglInterface(eglGetProcAddress), 0x31D7, device, null);
|
|
_eglSurface = _eglDisplay.CreateWindowSurface(_gbmTargetSurface);
|
|
|
|
|
|
EglContext CreateContext(EglContext share)
|
|
{
|
|
var offSurf = gbm_surface_create(device, 1, 1, GbmColorFormats.GBM_FORMAT_XRGB8888,
|
|
GbmBoFlags.GBM_BO_USE_RENDERING);
|
|
if (offSurf == null)
|
|
throw new InvalidOperationException("Unable to create 1x1 sized GBM surface");
|
|
return _eglDisplay.CreateContext(share, _eglDisplay.CreateWindowSurface(offSurf));
|
|
}
|
|
|
|
_immediateContext = CreateContext(null);
|
|
_deferredContext = CreateContext(_immediateContext);
|
|
|
|
_immediateContext.MakeCurrent(_eglSurface);
|
|
_eglDisplay.GlInterface.ClearColor(0, 0, 0, 0);
|
|
_eglDisplay.GlInterface.Clear(GlConsts.GL_COLOR_BUFFER_BIT | GlConsts.GL_STENCIL_BUFFER_BIT);
|
|
_eglSurface.SwapBuffers();
|
|
var bo = gbm_surface_lock_front_buffer(_gbmTargetSurface);
|
|
var fbId = GetFbIdForBo(bo);
|
|
var connectorId = connector.Id;
|
|
var mode = modeInfo.Mode;
|
|
|
|
|
|
var res = drmModeSetCrtc(_card.Fd, _crtcId, fbId, 0, 0, &connectorId, 1, &mode);
|
|
if (res != 0)
|
|
throw new Win32Exception(res, "drmModeSetCrtc failed");
|
|
|
|
_mode = mode;
|
|
_currentBo = bo;
|
|
|
|
// Go trough two cycles of buffer swapping (there are render artifacts otherwise)
|
|
for(var c=0;c<2;c++)
|
|
using (CreateGlRenderTarget().BeginDraw())
|
|
{
|
|
_eglDisplay.GlInterface.ClearColor(0, 0, 0, 0);
|
|
_eglDisplay.GlInterface.Clear(GlConsts.GL_COLOR_BUFFER_BIT | GlConsts.GL_STENCIL_BUFFER_BIT);
|
|
}
|
|
}
|
|
|
|
public IGlPlatformSurfaceRenderTarget CreateGlRenderTarget()
|
|
{
|
|
return new RenderTarget(this);
|
|
}
|
|
|
|
class RenderTarget : IGlPlatformSurfaceRenderTarget
|
|
{
|
|
private readonly DrmOutput _parent;
|
|
|
|
public RenderTarget(DrmOutput parent)
|
|
{
|
|
_parent = parent;
|
|
}
|
|
public void Dispose()
|
|
{
|
|
// We are wrapping GBM buffer chain associated with CRTC, and don't free it on a whim
|
|
}
|
|
|
|
class RenderSession : IGlPlatformSurfaceRenderingSession
|
|
{
|
|
private readonly DrmOutput _parent;
|
|
|
|
public RenderSession(DrmOutput parent)
|
|
{
|
|
_parent = parent;
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
_parent._eglDisplay.GlInterface.Flush();
|
|
_parent._eglSurface.SwapBuffers();
|
|
|
|
var nextBo = gbm_surface_lock_front_buffer(_parent._gbmTargetSurface);
|
|
if (nextBo == IntPtr.Zero)
|
|
{
|
|
// Not sure what else can be done
|
|
Console.WriteLine("gbm_surface_lock_front_buffer failed");
|
|
}
|
|
else
|
|
{
|
|
|
|
var fb = _parent.GetFbIdForBo(nextBo);
|
|
bool waitingForFlip = true;
|
|
|
|
drmModePageFlip(_parent._card.Fd, _parent._crtcId, fb, DrmModePageFlip.Event, null);
|
|
|
|
DrmEventPageFlipHandlerDelegate flipCb =
|
|
(int fd, uint sequence, uint tv_sec, uint tv_usec, void* user_data) =>
|
|
{
|
|
waitingForFlip = false;
|
|
};
|
|
var cbHandle = GCHandle.Alloc(flipCb);
|
|
var ctx = new DrmEventContext
|
|
{
|
|
version = 4, page_flip_handler = Marshal.GetFunctionPointerForDelegate(flipCb)
|
|
};
|
|
while (waitingForFlip)
|
|
{
|
|
var pfd = new pollfd {events = 1, fd = _parent._card.Fd};
|
|
poll(&pfd, new IntPtr(1), -1);
|
|
drmHandleEvent(_parent._card.Fd, &ctx);
|
|
}
|
|
|
|
cbHandle.Free();
|
|
gbm_surface_release_buffer(_parent._gbmTargetSurface, _parent._currentBo);
|
|
_parent._currentBo = nextBo;
|
|
}
|
|
_parent._eglDisplay.ClearContext();
|
|
}
|
|
|
|
|
|
public IGlDisplay Display => _parent._eglDisplay;
|
|
|
|
public PixelSize Size => _parent._mode.Resolution;
|
|
|
|
public double Scaling => _parent.Scaling;
|
|
|
|
public bool IsYFlipped { get; }
|
|
}
|
|
|
|
public IGlPlatformSurfaceRenderingSession BeginDraw()
|
|
{
|
|
_parent._deferredContext.MakeCurrent(_parent._eglSurface);
|
|
return new RenderSession(_parent);
|
|
}
|
|
|
|
|
|
}
|
|
|
|
IGlContext IWindowingPlatformGlFeature.ImmediateContext => _immediateContext;
|
|
}
|
|
|
|
|
|
}
|
|
|