Show a patch.

GET /api/patches/15580/?format=api
HTTP 200 OK
Allow: GET, PUT, PATCH, HEAD, OPTIONS
Content-Type: application/json
Vary: Accept

{
    "id": 15580,
    "url": "https://patchwork.libcamera.org/api/patches/15580/?format=api",
    "web_url": "https://patchwork.libcamera.org/patch/15580/",
    "project": {
        "id": 1,
        "url": "https://patchwork.libcamera.org/api/projects/1/?format=api",
        "name": "libcamera",
        "link_name": "libcamera",
        "list_id": "libcamera_core",
        "list_email": "libcamera-devel@lists.libcamera.org",
        "web_url": "",
        "scm_url": "",
        "webscm_url": ""
    },
    "msgid": "<20220329111725.24565-1-ecurtin@redhat.com>",
    "date": "2022-03-29T11:17:25",
    "name": "[libcamera-devel,v4] cam: sdl_sink: Add SDL sink",
    "commit_ref": null,
    "pull_url": null,
    "state": "superseded",
    "archived": false,
    "hash": "b44f018315b88030e76604ad6fcf5743997666cc",
    "submitter": {
        "id": 101,
        "url": "https://patchwork.libcamera.org/api/people/101/?format=api",
        "name": "Eric Curtin",
        "email": "ecurtin@redhat.com"
    },
    "delegate": null,
    "mbox": "https://patchwork.libcamera.org/patch/15580/mbox/",
    "series": [
        {
            "id": 3011,
            "url": "https://patchwork.libcamera.org/api/series/3011/?format=api",
            "web_url": "https://patchwork.libcamera.org/project/libcamera/list/?series=3011",
            "date": "2022-03-29T11:17:25",
            "name": "[libcamera-devel,v4] cam: sdl_sink: Add SDL sink",
            "version": 4,
            "mbox": "https://patchwork.libcamera.org/series/3011/mbox/"
        }
    ],
    "comments": "https://patchwork.libcamera.org/api/patches/15580/comments/",
    "check": "pending",
    "checks": "https://patchwork.libcamera.org/api/patches/15580/checks/",
    "tags": {},
    "headers": {
        "Return-Path": "<libcamera-devel-bounces@lists.libcamera.org>",
        "X-Original-To": "parsemail@patchwork.libcamera.org",
        "Delivered-To": "parsemail@patchwork.libcamera.org",
        "Received": [
            "from lancelot.ideasonboard.com (lancelot.ideasonboard.com\n\t[92.243.16.209])\n\tby patchwork.libcamera.org (Postfix) with ESMTPS id BBE64C0F1B\n\tfor <parsemail@patchwork.libcamera.org>;\n\tTue, 29 Mar 2022 11:17:36 +0000 (UTC)",
            "from lancelot.ideasonboard.com (localhost [IPv6:::1])\n\tby lancelot.ideasonboard.com (Postfix) with ESMTP id 0077465632;\n\tTue, 29 Mar 2022 13:17:35 +0200 (CEST)",
            "from us-smtp-delivery-124.mimecast.com\n\t(us-smtp-delivery-124.mimecast.com [170.10.129.124])\n\tby lancelot.ideasonboard.com (Postfix) with ESMTPS id 32A6F60135\n\tfor <libcamera-devel@lists.libcamera.org>;\n\tTue, 29 Mar 2022 13:17:34 +0200 (CEST)",
            "from mail-wr1-f70.google.com (mail-wr1-f70.google.com\n\t[209.85.221.70]) by relay.mimecast.com with ESMTP with STARTTLS\n\t(version=TLSv1.2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id\n\tus-mta-589-RrgTNtytM8C1IZ-l9rJGFQ-1; Tue, 29 Mar 2022 07:17:31 -0400",
            "by mail-wr1-f70.google.com with SMTP id\n\th11-20020a5d430b000000b001f01a35a86fso4924367wrq.4\n\tfor <libcamera-devel@lists.libcamera.org>;\n\tTue, 29 Mar 2022 04:17:30 -0700 (PDT)",
            "from p1.Home ([2001:8a0:6724:4500:a69c:e66f:828e:b340])\n\tby smtp.gmail.com with ESMTPSA id\n\tq6-20020adffec6000000b00205b60faeeesm8107591wrs.24.2022.03.29.04.17.28\n\t(version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256);\n\tTue, 29 Mar 2022 04:17:28 -0700 (PDT)"
        ],
        "DKIM-Signature": [
            "v=1; a=rsa-sha256; c=relaxed/simple; d=libcamera.org;\n\ts=mail; t=1648552656;\n\tbh=ylE6KRzyGbxBVszbgevvSzwM0kPna1RogbZ/qxptmdo=;\n\th=To:Date:Subject:List-Id:List-Unsubscribe:List-Archive:List-Post:\n\tList-Help:List-Subscribe:From:Reply-To:From;\n\tb=Jfnp8S5MrXKcnpnDkD3taH/ydZLyc5mvgKfglxhZSvCnVqkMzJ7pj+g1R1W1NDylT\n\t7Mu2UdyNCNjSWNFha7n8KVbhxv3voAR0WKAUBoVnpexnCBSh7/kuK7FkuXEIhke9Ik\n\thyYVQls2Om/gzSefgo0eWCZMhlbhnl3+XUdlIFWK0X5H4504ikZagW/G6z+i/nqoxq\n\t6Q0UURInTJ5/6T8NE9xnF8LIUqzVlzbkqeBs6jUHoxSueKl2etiei6XgFvszregudP\n\t9SDVWAHkC+UL8M2W4kXtr79upqkPiGwBEIe0gvi0PzMSW/Uuwsar4d/YgRNZ+qGmFJ\n\t1+h+gBkEcLlFg==",
            "v=1; a=rsa-sha256; c=relaxed/relaxed; d=redhat.com;\n\ts=mimecast20190719; t=1648552652;\n\th=from:from:reply-to:subject:subject:date:date:message-id:message-id:\n\tto:to:cc:cc:mime-version:mime-version:content-type:content-type:\n\tcontent-transfer-encoding:content-transfer-encoding;\n\tbh=2xxcOCFy8kVVunHVssTOocKJpmwLLDNxnHbPyEFQAVA=;\n\tb=hdB7GsmvAdHmlFmB26jG9anIjw/EwOscmaGqzL4YfzRJKHTmLMjzscfuvdG135BApBvx2C\n\tZRf4ccmppTSXyUdwFs09SJSJZjVVtbQdXHnNKFyQRd/NPCxQ4k04aPMmdTkC4dCPfjlv0v\n\t+twowyubGh5Eg6ZJQeiD+1j5Wf3XEwU="
        ],
        "Authentication-Results": [
            "lancelot.ideasonboard.com; dkim=pass (1024-bit key; \n\tunprotected) header.d=redhat.com\n\theader.i=@redhat.com header.b=\"hdB7Gsmv\"; \n\tdkim-atps=neutral",
            "relay.mimecast.com;\n\tauth=pass smtp.auth=CUSA124A263 smtp.mailfrom=ecurtin@redhat.com"
        ],
        "X-MC-Unique": "RrgTNtytM8C1IZ-l9rJGFQ-1",
        "X-Google-DKIM-Signature": "v=1; a=rsa-sha256; c=relaxed/relaxed;\n\td=1e100.net; s=20210112;\n\th=x-gm-message-state:from:to:cc:subject:date:message-id:mime-version\n\t:content-transfer-encoding;\n\tbh=2xxcOCFy8kVVunHVssTOocKJpmwLLDNxnHbPyEFQAVA=;\n\tb=yQxLiQBw6SGWusNPwEpVUl0eXyRrQS2FXCeqhvWRVzLuNNnUhEIEWZihPWWX2WnZiE\n\t7DQLey9shF2yz4L7Mtt38K4csC7PkC0/ACp0uWACxG78CB6mXYBrUpfz3ZBtT5YGy5MC\n\tuUMoudo5ZQ+M39ZU2MmWswvgjf8BJq/fL2BZacvmyoSObbyVDgk5ONTjOnvbkM1gwato\n\tgJqflkKmXPJDKD/HLTNVqplaXX+bzuFgP+rMaJXMQlGwI1ks+UECgGw/Qhsj1FT8Ep1F\n\t5xzMC70aAadVg9dXSDuGKteqwva58jFKZa1mz0MrniAkwItgelVFfwxGBEYZYUU/uVBX\n\t256Q==",
        "X-Gm-Message-State": "AOAM531ZuRYzbsAdUV6sNctV0hzQ3fNYdhISuvLjRxgkvrerp/Yufh8z\n\taGpHg6+NdV6Dbj1KBC7BhIG69fcxIL614n5Sp1iuhnzUdnFeJEmJPjy43VlHbwblVIzkBdxt20n\n\t8VZOiRWXT/facFdW22STxZzjh0wfzebhoujpWbeDFIQ46fe8La/reRBch4YNRCSMgEK5bfGwCfy\n\tLgZKsumT77",
        "X-Received": [
            "by 2002:a05:600c:3504:b0:38c:bcbc:f95e with SMTP id\n\th4-20020a05600c350400b0038cbcbcf95emr6032895wmq.170.1648552649655; \n\tTue, 29 Mar 2022 04:17:29 -0700 (PDT)",
            "by 2002:a05:600c:3504:b0:38c:bcbc:f95e with SMTP id\n\th4-20020a05600c350400b0038cbcbcf95emr6032860wmq.170.1648552649339; \n\tTue, 29 Mar 2022 04:17:29 -0700 (PDT)"
        ],
        "X-Google-Smtp-Source": "ABdhPJxcYCHfu3fHJaDrrcDw/AgZtRVAM602Q2HUOpzeg8+HmyhcjZ1TNNnZE2phYsFyTc0b5y7FSw==",
        "To": "libcamera-devel@lists.libcamera.org",
        "Date": "Tue, 29 Mar 2022 12:17:25 +0100",
        "Message-Id": "<20220329111725.24565-1-ecurtin@redhat.com>",
        "X-Mailer": "git-send-email 2.35.1",
        "MIME-Version": "1.0",
        "X-Mimecast-Spam-Score": "0",
        "X-Mimecast-Originator": "redhat.com",
        "Content-Transfer-Encoding": "8bit",
        "Content-Type": "text/plain; charset=\"US-ASCII\"; x-default=true",
        "Subject": "[libcamera-devel] [PATCH v4] cam: sdl_sink: Add SDL sink",
        "X-BeenThere": "libcamera-devel@lists.libcamera.org",
        "X-Mailman-Version": "2.1.29",
        "Precedence": "list",
        "List-Id": "<libcamera-devel.lists.libcamera.org>",
        "List-Unsubscribe": "<https://lists.libcamera.org/options/libcamera-devel>,\n\t<mailto:libcamera-devel-request@lists.libcamera.org?subject=unsubscribe>",
        "List-Archive": "<https://lists.libcamera.org/pipermail/libcamera-devel/>",
        "List-Post": "<mailto:libcamera-devel@lists.libcamera.org>",
        "List-Help": "<mailto:libcamera-devel-request@lists.libcamera.org?subject=help>",
        "List-Subscribe": "<https://lists.libcamera.org/listinfo/libcamera-devel>,\n\t<mailto:libcamera-devel-request@lists.libcamera.org?subject=subscribe>",
        "From": "Eric Curtin via libcamera-devel <libcamera-devel@lists.libcamera.org>",
        "Reply-To": "Eric Curtin <ecurtin@redhat.com>",
        "Errors-To": "libcamera-devel-bounces@lists.libcamera.org",
        "Sender": "\"libcamera-devel\" <libcamera-devel-bounces@lists.libcamera.org>"
    },
    "content": "This adds more portability to existing cam sinks. You can pass a\nYUYV camera buffer for example and SDL will handle the pixel buffer\nconversion, although SDL does not support decompression for pixelformats\nlike MJPEG. This allows cam reference implementation to display images\non VMs, Mac M1, Raspberry Pi, etc. This also enables cam reference\nimplementation, to run as a desktop application in wayland or x11.\nSDL also has support for Android and ChromeOS which I have not tested.\nAlso tested on simpledrm raspberry pi 4 framebuffer successfully where\nexisting kms sink did not work. Can also be used as kmsdrm sink.\n\nSigned-off-by: Eric Curtin <ecurtin@redhat.com>\n---\nChanges in v2:\n - Remove hardcoded pixel format from SDL_CreateTexture call\n\nChanges in v3:\n - Drop blank line\n - The contents of the if..endif are indented\n - Split configure function into start and configure\n - Add SDL_DestroyRenderer\n - Remove assign and test in the same statement\n\nChanges in v4:\n - Add processSDLEvents as a timed event in the evloop\n---\n src/cam/camera_session.cpp |   8 +++\n src/cam/event_loop.cpp     |  10 ++-\n src/cam/event_loop.h       |   1 +\n src/cam/main.cpp           |   5 ++\n src/cam/main.h             |   1 +\n src/cam/meson.build        |  10 +++\n src/cam/sdl_sink.cpp       | 142 +++++++++++++++++++++++++++++++++++++\n src/cam/sdl_sink.h         |  42 +++++++++++\n 8 files changed, 218 insertions(+), 1 deletion(-)\n create mode 100644 src/cam/sdl_sink.cpp\n create mode 100644 src/cam/sdl_sink.h",
    "diff": "diff --git a/src/cam/camera_session.cpp b/src/cam/camera_session.cpp\nindex 0428b538..30162dbd 100644\n--- a/src/cam/camera_session.cpp\n+++ b/src/cam/camera_session.cpp\n@@ -19,6 +19,9 @@\n #ifdef HAVE_KMS\n #include \"kms_sink.h\"\n #endif\n+#ifdef HAVE_SDL\n+#include \"sdl_sink.h\"\n+#endif\n #include \"main.h\"\n #include \"stream_options.h\"\n \n@@ -187,6 +190,11 @@ int CameraSession::start()\n \t\tsink_ = std::make_unique<KMSSink>(options_[OptDisplay].toString());\n #endif\n \n+#ifdef HAVE_SDL\n+\tif (options_.isSet(OptSDL))\n+\t\tsink_ = std::make_unique<SDLSink>();\n+#endif\n+\n \tif (options_.isSet(OptFile)) {\n \t\tif (!options_[OptFile].toString().empty())\n \t\t\tsink_ = std::make_unique<FileSink>(streamNames_,\ndiff --git a/src/cam/event_loop.cpp b/src/cam/event_loop.cpp\nindex e25784c0..41339d4e 100644\n--- a/src/cam/event_loop.cpp\n+++ b/src/cam/event_loop.cpp\n@@ -75,7 +75,15 @@ void EventLoop::addEvent(int fd, EventType type,\n \t\treturn;\n \t}\n \n-\tint ret = event_add(event->event_, nullptr);\n+\tstruct timeval *tp = nullptr;\n+\tstruct timeval tv;\n+\tif (events == EV_PERSIST) {\n+\t\ttp = &tv;\n+\t\ttv.tv_sec = 0;\n+\t\ttv.tv_usec = 10000; /* every 10 ms */\n+\t}\n+\n+\tint ret = event_add(event->event_, tp);\n \tif (ret < 0) {\n \t\tstd::cerr << \"Failed to add event for fd \" << fd << std::endl;\n \t\treturn;\ndiff --git a/src/cam/event_loop.h b/src/cam/event_loop.h\nindex a4613eb2..8fd9bd20 100644\n--- a/src/cam/event_loop.h\n+++ b/src/cam/event_loop.h\n@@ -20,6 +20,7 @@ class EventLoop\n {\n public:\n \tenum EventType {\n+\t\tDefault = 0, /* the event can be triggered only by a timeout or by manual activation */\n \t\tRead = 1,\n \t\tWrite = 2,\n \t};\ndiff --git a/src/cam/main.cpp b/src/cam/main.cpp\nindex c7f664b9..1d62a64a 100644\n--- a/src/cam/main.cpp\n+++ b/src/cam/main.cpp\n@@ -137,6 +137,11 @@ int CamApp::parseOptions(int argc, char *argv[])\n \t\t\t \"Display viewfinder through DRM/KMS on specified connector\",\n \t\t\t \"display\", ArgumentOptional, \"connector\", false,\n \t\t\t OptCamera);\n+#endif\n+#ifdef HAVE_SDL\n+\tparser.addOption(OptSDL, OptionNone,\n+\t\t\t \"Display viewfinder through SDL\",\n+\t\t\t \"sdl\", ArgumentNone, \"\", false, OptCamera);\n #endif\n \tparser.addOption(OptFile, OptionString,\n \t\t\t \"Write captured frames to disk\\n\"\ndiff --git a/src/cam/main.h b/src/cam/main.h\nindex 62f7bbc9..a64f95a0 100644\n--- a/src/cam/main.h\n+++ b/src/cam/main.h\n@@ -11,6 +11,7 @@ enum {\n \tOptCamera = 'c',\n \tOptCapture = 'C',\n \tOptDisplay = 'D',\n+\tOptSDL = 'S',\n \tOptFile = 'F',\n \tOptHelp = 'h',\n \tOptInfo = 'I',\ndiff --git a/src/cam/meson.build b/src/cam/meson.build\nindex e8e2ae57..bd536c5b 100644\n--- a/src/cam/meson.build\n+++ b/src/cam/meson.build\n@@ -32,11 +32,21 @@ cam_sources += files([\n ])\n endif\n \n+libsdl2 = dependency('SDL2', required : false)\n+\n+if libsdl2.found()\n+    cam_cpp_args += [ '-DHAVE_SDL' ]\n+    cam_sources += files([\n+        'sdl_sink.cpp'\n+    ])\n+endif\n+\n cam  = executable('cam', cam_sources,\n                   dependencies : [\n                       libatomic,\n                       libcamera_public,\n                       libdrm,\n+                      libsdl2,\n                       libevent,\n                   ],\n                   cpp_args : cam_cpp_args,\ndiff --git a/src/cam/sdl_sink.cpp b/src/cam/sdl_sink.cpp\nnew file mode 100644\nindex 00000000..6c6e37d0\n--- /dev/null\n+++ b/src/cam/sdl_sink.cpp\n@@ -0,0 +1,142 @@\n+/* SPDX-License-Identifier: GPL-2.0-or-later */\n+/*\n+ * sdl_sink.cpp - SDL Sink\n+ */\n+\n+#include \"sdl_sink.h\"\n+\n+#include <assert.h>\n+#include <fcntl.h>\n+#include <iomanip>\n+#include <iostream>\n+#include <signal.h>\n+#include <sstream>\n+#include <string.h>\n+#include <unistd.h>\n+\n+#include <libcamera/camera.h>\n+#include <libcamera/formats.h>\n+\n+#include \"event_loop.h\"\n+#include \"image.h\"\n+\n+using namespace libcamera;\n+\n+SDLSink::SDLSink()\n+\t: sdlRenderer_(0)\n+{\n+\tmemset(&sdlRect_, 0, sizeof(sdlRect_));\n+}\n+\n+SDLSink::~SDLSink()\n+{\n+\tif (sdlRenderer_)\n+\t\tSDL_DestroyRenderer(sdlRenderer_);\n+\tSDL_Quit();\n+}\n+\n+int SDLSink::configure(const libcamera::CameraConfiguration &cfg)\n+{\n+\tint ret = FrameSink::configure(cfg);\n+\tif (ret < 0)\n+\t\treturn ret;\n+\n+\tconst libcamera::StreamConfiguration &sCfg = cfg.at(0);\n+\tpf = (SDL_PixelFormatEnum)sCfg.pixelFormat.fourcc();\n+\tif (pf == SDL_DEFINE_PIXELFOURCC('Y', 'U', 'Y', 'V')) {\n+\t\tpf = SDL_PIXELFORMAT_YUY2;\n+\t} else if (int ne = strcmp(SDL_GetPixelFormatName(pf), \"SDL_PIXELFORMAT_UNKNOWN\"); !ne) {\n+\t\tstd::cerr << \"SDL_GetPixelFormatName error - exiting: SDL_PIXELFORMAT_UNKNOWN, no \" << sCfg.pixelFormat.toString() << \" support\\n\";\n+\t\treturn -EINVAL;\n+\t}\n+\n+\tsdlRect_.w = sCfg.size.width;\n+\tsdlRect_.h = sCfg.size.height;\n+\n+\treturn 0;\n+}\n+\n+int SDLSink::start()\n+{\n+\tint ret = SDL_Init(SDL_INIT_VIDEO);\n+\tif (ret) {\n+\t\tstd::cerr << \"SDL_Init error - exiting: \" << SDL_GetError() << std::endl;\n+\t\treturn ret;\n+\t}\n+\n+\tsdlScreen_ = SDL_CreateWindow(\"\", SDL_WINDOWPOS_UNDEFINED,\n+\t\t\t\t      SDL_WINDOWPOS_UNDEFINED, sdlRect_.w,\n+\t\t\t\t      sdlRect_.h, SDL_WINDOW_SHOWN | SDL_WINDOW_RESIZABLE);\n+\tif (!sdlScreen_) {\n+\t\tstd::cerr << \"SDL_CreateWindow error - exiting: \" << SDL_GetError() << std::endl;\n+\t\treturn -EINVAL;\n+\t}\n+\n+\tsdlRenderer_ = SDL_CreateRenderer(\n+\t\tsdlScreen_, -1, 0);\n+\tif (!sdlRenderer_) {\n+\t\tstd::cerr << \"SDL_CreateRenderer error - exiting: \" << SDL_GetError() << std::endl;\n+\t\treturn -EINVAL;\n+\t}\n+\n+\tSDL_RenderSetLogicalSize(sdlRenderer_, sdlRect_.w,\n+\t\t\t\t sdlRect_.h);\n+\n+\tsdlTexture_ =\n+\t\tSDL_CreateTexture(sdlRenderer_, pf,\n+\t\t\t\t  SDL_TEXTUREACCESS_STREAMING, sdlRect_.w,\n+\t\t\t\t  sdlRect_.h);\n+\tEventLoop::instance()->addEvent(-1, EventLoop::Default,\n+\t\t\t\t\tstd::bind(&SDLSink::processSDLEvents, this));\n+\n+\treturn 0;\n+}\n+\n+void SDLSink::mapBuffer(FrameBuffer *buffer)\n+{\n+\tstd::unique_ptr<Image> image =\n+\t\tImage::fromFrameBuffer(buffer, Image::MapMode::ReadOnly);\n+\tassert(image != nullptr);\n+\n+\tmappedBuffers_[buffer] = std::move(image);\n+}\n+\n+bool SDLSink::processRequest(Request *request)\n+{\n+\tfor (auto [stream, buffer] : request->buffers())\n+\t\twriteBuffer(buffer);\n+\n+\treturn true;\n+}\n+\n+/*\n+ * Process SDL events, required for things like window resize and quit button\n+ */\n+void SDLSink::processSDLEvents()\n+{\n+\tfor (SDL_Event e; SDL_PollEvent(&e);) {\n+\t\tif (e.type == SDL_QUIT) { /* click close icon then quit */\n+\t\t\tkill(getpid(), SIGINT);\n+\t\t}\n+\t}\n+}\n+\n+void SDLSink::writeBuffer(FrameBuffer *buffer)\n+{\n+\tImage *image = mappedBuffers_[buffer].get();\n+\n+\tfor (unsigned int i = 0; i < buffer->planes().size(); ++i) {\n+\t\tconst FrameMetadata::Plane &meta = buffer->metadata().planes()[i];\n+\n+\t\tSpan<uint8_t> data = image->data(i);\n+\t\tif (meta.bytesused > data.size())\n+\t\t\tstd::cerr << \"payload size \" << meta.bytesused\n+\t\t\t\t  << \" larger than plane size \" << data.size()\n+\t\t\t\t  << std::endl;\n+\n+\t\tSDL_UpdateTexture(sdlTexture_, &sdlRect_, data.data(), sdlRect_.w * 2);\n+\t\tSDL_RenderClear(sdlRenderer_);\n+\t\tSDL_RenderCopy(sdlRenderer_, sdlTexture_, NULL, NULL);\n+\t\tSDL_RenderPresent(sdlRenderer_);\n+\t}\n+}\ndiff --git a/src/cam/sdl_sink.h b/src/cam/sdl_sink.h\nnew file mode 100644\nindex 00000000..6f20e2d6\n--- /dev/null\n+++ b/src/cam/sdl_sink.h\n@@ -0,0 +1,42 @@\n+/* SPDX-License-Identifier: GPL-2.0-or-later */\n+/*\n+ * sdl_sink.h - SDL Sink\n+ */\n+\n+#pragma once\n+\n+#include <map>\n+#include <memory>\n+#include <string>\n+\n+#include <libcamera/stream.h>\n+\n+#include <SDL2/SDL.h>\n+\n+#include \"frame_sink.h\"\n+\n+class Image;\n+\n+class SDLSink : public FrameSink\n+{\n+public:\n+\tSDLSink();\n+\t~SDLSink();\n+\n+\tint configure(const libcamera::CameraConfiguration &cfg) override;\n+\tint start() override;\n+\tvoid mapBuffer(libcamera::FrameBuffer *buffer) override;\n+\n+\tbool processRequest(libcamera::Request *request) override;\n+\n+private:\n+\tvoid writeBuffer(libcamera::FrameBuffer *buffer);\n+\tvoid processSDLEvents();\n+\n+\tstd::map<libcamera::FrameBuffer *, std::unique_ptr<Image>> mappedBuffers_;\n+\tSDL_Window *sdlScreen_;\n+\tSDL_Renderer *sdlRenderer_;\n+\tSDL_Texture *sdlTexture_;\n+\tSDL_Rect sdlRect_;\n+\tSDL_PixelFormatEnum pf;\n+};\n",
    "prefixes": [
        "libcamera-devel",
        "v4"
    ]
}