mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-01-23 17:52:26 -05:00
79acb998e1
If Metal context and IOSurface are available, Skia painter will use Ganesh GPU backend on macOS, which is noticeably faster than the default CPU backend. Painting pipeline: 1. (WebContent) Allocate IOSurface for backing store 2. (WebContent) Allocate MTLTexture that wraps IOSurface 3. (WebContent) Paint into MTLTexture using Skia 4. (Browser) Wrap IOSurface into Gfx::Painter and use QPainter/CoreGraphics to blit backing store into viewport. Things we should improve in the future: 1. Upload textures for images in advance instead of doing that before every repaint. 2. Teach AppKit client to read directly from IOSurface instead of copying.
92 lines
2.3 KiB
Text
92 lines
2.3 KiB
Text
/*
|
|
* Copyright (c) 2024, Aliaksandr Kalenik <kalenik.aliaksandr@gmail.com>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <AK/OwnPtr.h>
|
|
#include <LibCore/MetalContext.h>
|
|
|
|
#define FixedPoint FixedPointMacOS
|
|
#define Duration DurationMacOS
|
|
#include <Metal/Metal.h>
|
|
#undef FixedPoint
|
|
#undef Duration
|
|
|
|
namespace Core {
|
|
|
|
class MetalTextureImpl final : public MetalTexture {
|
|
public:
|
|
MetalTextureImpl(id<MTLTexture> texture)
|
|
: m_texture(texture)
|
|
{
|
|
}
|
|
|
|
void const* texture() const override { return m_texture; }
|
|
size_t width() const override { return m_texture.width; }
|
|
size_t height() const override { return m_texture.height; }
|
|
|
|
virtual ~MetalTextureImpl()
|
|
{
|
|
[m_texture release];
|
|
}
|
|
|
|
private:
|
|
id<MTLTexture> m_texture;
|
|
};
|
|
|
|
class MetalContextImpl final : public MetalContext {
|
|
public:
|
|
MetalContextImpl(id<MTLDevice> device, id<MTLCommandQueue> queue)
|
|
: m_device(device)
|
|
, m_queue(queue)
|
|
{
|
|
}
|
|
|
|
void const* device() const override { return m_device; }
|
|
void const* queue() const override { return m_queue; }
|
|
|
|
OwnPtr<MetalTexture> create_texture_from_iosurface(IOSurfaceHandle const& iosurface) override
|
|
{
|
|
auto* const descriptor = [[MTLTextureDescriptor alloc] init];
|
|
descriptor.pixelFormat = MTLPixelFormatBGRA8Unorm;
|
|
descriptor.width = iosurface.width();
|
|
descriptor.height = iosurface.height();
|
|
descriptor.storageMode = MTLStorageModeShared;
|
|
descriptor.usage = MTLTextureUsageRenderTarget | MTLTextureUsageShaderRead;
|
|
|
|
id<MTLTexture> texture = [m_device newTextureWithDescriptor:descriptor iosurface:(IOSurfaceRef)iosurface.core_foundation_pointer() plane:0];
|
|
[descriptor release];
|
|
return make<MetalTextureImpl>(texture);
|
|
}
|
|
|
|
virtual ~MetalContextImpl() override
|
|
{
|
|
[m_queue release];
|
|
[m_device release];
|
|
}
|
|
|
|
private:
|
|
id<MTLDevice> m_device;
|
|
id<MTLCommandQueue> m_queue;
|
|
};
|
|
|
|
OwnPtr<MetalContext> get_metal_context()
|
|
{
|
|
auto device = MTLCreateSystemDefaultDevice();
|
|
if (!device) {
|
|
dbgln("Failed to create Metal device");
|
|
return {};
|
|
}
|
|
|
|
auto queue = [device newCommandQueue];
|
|
if (!queue) {
|
|
dbgln("Failed to create Metal command queue");
|
|
[device release];
|
|
return {};
|
|
}
|
|
|
|
return make<MetalContextImpl>(device, queue);
|
|
}
|
|
|
|
}
|