vdr-plugin-softhddevice-drm-gles 1.6.7
drmbuffer.cpp
Go to the documentation of this file.
1// SPDX-License-Identifier: AGPL-3.0-or-later
2
17#include <cstdint>
18#include <fcntl.h>
19#include <sys/mman.h>
20
21extern "C" {
22#include <libavutil/frame.h>
23#include <libavutil/hwcontext_drm.h>
24}
25
26#include <drm_fourcc.h>
27#include <xf86drm.h>
28#include <xf86drmMode.h>
29
30#include "drmbuffer.h"
31#include "logger.h"
32#include "pool.h"
33
34/*****************************************************************************
35 * Drm Buffer
36 ****************************************************************************/
37
44{
45 m_dirty = false;
46 m_dmaBufHandle[0] = 0;
47
48 m_numPlanes = 0;
49 for (int i = 0; i < 4; i++) {
50 m_pPlane[i] = nullptr;
51 }
52}
53
54#ifdef USE_GLES
67 : m_width(width),
68 m_height(height),
69 m_pixFmt(pixFmt),
70 m_drmDeviceFd(fdDrm),
71 m_pBo(bo)
72{
73 m_numPlanes = 0;
75
76 for (int i = 0; i < 4; i++) {
77 m_pPlane[i] = nullptr;
79 m_offset[i] = 0;
80 m_pitch[i] = 0;
81 m_size[i] = 0;
82 }
83 m_dirty = false;
84}
85#endif
86
91{
93 LOGDEBUG2(L_DRM, "drmbuffer: %s: destroy FB %d DMA-BUF handle %d", __FUNCTION__, m_fbId, m_dmaBufHandle[0]);
94
95 for (int i = 0; i < m_numPlanes; i++) {
96 if (m_pPlane[i]) {
97 if (munmap(m_pPlane[i], m_size[i]))
98 LOGERROR("drmbuffer: %s: failed unmap FB (%d): %m", __FUNCTION__, errno);
99 }
100 }
101
103 LOGERROR("drmbuffer: %s: cannot rm FB (%d): %m", __FUNCTION__, errno);
104
105 if (m_closeHandleOnDestroy &&m_dmaBufHandle[0] && fcntl(m_dmaBufHandle[0], F_GETFD) != -1) { // the handle can be invalid in reverse trickspeed, because the decoder is rapidly reopened
106 if (close(m_dmaBufHandle[0]))
107 LOGERROR("drmbuffer: %s: error closing DMA-BUF handle %d (%d): %m", __FUNCTION__, m_dmaBufHandle[0], errno);
108 }
109
110 for (int i = 0; i < m_numPlanes; i++) {
111 if (m_pPlane[i]) {
112 memset(&dreq, 0, sizeof(dreq));
113 dreq.handle = m_planePrimeHandle[i];
114
116 LOGERROR("drmbuffer: %s: cannot destroy dumb buffer (%d): %m", __FUNCTION__, errno);
118
119 }
120
121 m_pPlane[i] = 0;
122 m_size[i] = 0;
123 m_pitch[i] = 0;
124 m_offset[i] = 0;
125 m_objIdx[i] = 0;
126 }
127
128 for (int i = 0; i < m_numObjects; i++) {
130 LOGERROR("drmbuffer: %s: cannot close handle %d FB %d GEM (%d): %m", __FUNCTION__, m_objectPrimeHandle[i], m_fbId, errno);
131 }
132
133 m_width = 0;
134 m_height = 0;
135 m_fbId = 0;
136 m_dirty = false;
137 m_numPlanes = 0;
138 m_destroyAfterUse = false;
140}
141
160static const struct format_info format_info_array[] = {
161 { DRM_FORMAT_NV12, "NV12", 2, { { 8, 1, 1 }, { 16, 2, 2 } }, },
162 { DRM_FORMAT_YUV420, "YU12", 3, { { 8, 1, 1 }, { 8, 2, 2 }, {8, 2, 2 } }, },
163 { DRM_FORMAT_ARGB8888, "AR24", 1, { { 32, 1, 1 } }, },
164};
165
166#define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0]))
167
176{
177 for (int i = 0; i < (int)ARRAY_SIZE(format_info_array); i++) {
179 return &format_info_array[i];
180 }
181 return NULL;
182}
183
194{
195 uint64_t modifier[4] = { 0, 0, 0, 0 };
198 m_pitch[0] = m_pitch[1] = m_pitch[2] = m_pitch[3] = 0;
199 m_offset[0] = m_offset[1] = m_offset[2] = m_offset[3] = 0;
201
202 m_width = width;
203 m_height = height;
207
208 if (primedata) {
209 // we have no DRM objects yet, so return
210 if (!primedata->nb_objects)
211 LOGFATAL("drmbuffer: %s: No primedata objects available!", __FUNCTION__);
212
213 AVDRMLayerDescriptor *layer = &primedata->layers[0];
214
215 m_pixFmt = layer->format;
216 m_numPlanes = layer->nb_planes;
217 m_numObjects = primedata->nb_objects;
218
219// LOGDEBUG2(L_DRM, "drmbuffer: %s: PRIMEDATA %d x %d, pix_fmt %4.4s nb_planes %d nb_objects %d", __FUNCTION__,
220// m_width, m_height, (char *)&m_pixFmt, m_numPlanes, m_numObjects);
221
222 // create handles for PrimeFDs
223 for (int object = 0; object < primedata->nb_objects; object++) {
224 if (drmPrimeFDToHandle(drmDeviceFd, primedata->objects[object].fd, &m_objectPrimeHandle[object])) {
225 LOGFATAL("drmbuffer: %s: PRIMEDATA Failed to retrieve the Prime Handle %i size %zu (%d): %m", __FUNCTION__,
226 primedata->objects[object].fd,
227 primedata->objects[object].size, errno);
228 }
229
230 m_dmaBufHandle[object] = primedata->objects[object].fd;
231 m_size[object] = primedata->objects[object].size;
232// LOGDEBUG2(L_DRM, "drmbuffer: %s: PRIMEDATA create handle for PrimeFD (%d|%i): PrimeFD %i ToHandle %i size %zu modifier %" PRIx64 "",
233// __FUNCTION__, object, primedata->nb_objects, primedata->objects[object].fd, m_primehandle[object],
234// primedata->objects[object].size, primedata->objects[object].format_modifier);
235 }
236
237 // fill the planes
238 for (int plane = 0; plane < layer->nb_planes; plane++) {
239 int object = layer->planes[plane].object_index;
241 if (handle) {
243 m_pitch[plane] = layer->planes[plane].pitch;
244 m_offset[plane] = layer->planes[plane].offset;
246 if (primedata->objects[object].format_modifier)
247 modifier[plane] = primedata->objects[object].format_modifier;
248
249 // LOGDEBUG2(L_DRM, "drmbuffer: %s: PRIMEDATA fill plane %d: handle %d object_index %i pitch %d offset %d size %d modifier %" PRIx64 " (plane not mapped!)",
250 // __FUNCTION__, plane, m_planePrimeHandle[plane], m_objIdx[plane], m_pitch[plane], m_offset[plane], m_size[plane], modifier[plane]);
251 }
252 }
255
256 m_modifier = modifier[0];
257 } else {
259 if (!format_info)
260 LOGFATAL("drmbuffer: %s: No suitable format found!", __FUNCTION__);
261
263
264 // LOGDEBUG2(L_DRM, "drmbuffer: %s: %d x %d, pix_fmt %4.4s nb_planes %d", __FUNCTION__,
265 // m_width, m_height, (char *)&m_pixFmt, m_numPlanes);
266
267 for (int plane = 0; plane < format_info->num_planes; plane++) {
269
271 creq.height = m_height / plane_info->ysub;
272 creq.width = m_width / plane_info->xsub;
273 creq.bpp = plane_info->bitspp;
274 creq.flags = 0;
275 creq.handle = 0;
276 creq.pitch = 0;
277 creq.size = 0;
278
280 LOGFATAL("drmbuffer: %s: cannot create dumb buffer %dx%d@%d (%d): %m", __FUNCTION__,
281 creq.width, creq.height, creq.bpp, errno);
282
283 m_planePrimeHandle[plane] = creq.handle;
284 m_pitch[plane] = creq.pitch;
285 m_size[plane] = creq.size;
286
287 struct drm_mode_map_dumb mreq;
288 memset(&mreq, 0, sizeof(struct drm_mode_map_dumb));
289 mreq.handle = m_planePrimeHandle[plane];
290
292 LOGFATAL("drmbuffer: %s: cannot prepare dumb buffer for mapping (%d): %m", __FUNCTION__, errno);
293
295
296 if (m_pPlane[plane] == MAP_FAILED)
297 LOGFATAL("drmbuffer: %s: cannot map dumb buffer (%d): %m", __FUNCTION__, errno);
298
300
301 // LOGDEBUG2(L_DRM, "drmbuffer: %s: fill plane %d: prime handle %d pitch %d offset %d size %d address %p", __FUNCTION__,
302 // plane, m_planePrimeHandle[plane], m_pitch[plane], m_offset[plane], m_size[plane], m_pPlane[plane]);
303 }
304 }
305
306 int ret = -1;
308
309 if (ret) {
310 if (mod_flags)
311 LOGERROR("drmbuffer: %s: cannot create modifiers framebuffer (%d): %m", __FUNCTION__, errno);
312
315 }
316
317 if (ret)
318 LOGFATAL("drmbuffer: %s: cannot create framebuffer (%d): %m", __FUNCTION__, errno);
319
320 LOGDEBUG2(L_DRM, "drmbuffer: %s: Added %sFB fb_id %d width %d height %d pix_fmt %4.4s", __FUNCTION__,
321 primedata ? "primedata " : "", m_fbId, m_width, m_height, (char *)&m_pixFmt);
322
323 m_dirty = true;
324}
325
330{
331 for (uint32_t i = 0; i < m_width * m_height; ++i) {
332 m_pPlane[0][i] = 0x10;
333 if (i < m_width * m_height / 2)
334 m_pPlane[1][i] = 0x80;
335 }
336}
337
344{
346 m_presentationPending = false;
347
349 Destroy();
350}
351
352/*****************************************************************************
353 * Drm Buffer Pool
354 ****************************************************************************/
355
360{
361 for (const auto &buf : buffer) {
362 if (buf->IsDirty() && buf->DmaBufHandle() == primeFd)
363 return buf.get();
364 }
365
366 return nullptr;
367}
368
373{
374 for (const auto &buf : buffer) {
375 if (!buf->IsDirty())
376 return buf.get();
377 }
378
379 return nullptr;
380}
381
386{
387 for (const auto &buf : buffer) {
388 if (buf->IsDirty() && !buf->IsPresentationPending())
389 return buf.get();
390 }
391
392 return nullptr;
393}
394
402{
403 for (const auto &buf : buffer) {
404 if (buf.get() != exceptBuf && buf->IsDirty()) {
405 av_frame_free(&buf->frame);
406 buf->Destroy();
407 }
408 }
409}
cDrmBuffer * FindUninitilized(void)
Find a clean drm buffer from the buffer pool.
cDrmBuffer * FindByDmaBufHandle(int)
Find a drm buffer from the buffer pool by a given prime handle.
cDrmBuffer * FindNoPresentationPending(void)
Find a dirty drm buffer from the buffer pool which presentation has finished.
void DestroyAllExcept(cDrmBuffer *)
Destroy all drm buffers except the given one.
DRM Buffer.
Definition drmbuffer.h:48
bool m_dirty
true, if the buffer is dirty (it was written to)
Definition drmbuffer.h:112
cDrmBuffer(void)
Create a new drm buffer.
Definition drmbuffer.cpp:43
bool m_destroyAfterUse
true, if buffer should be destroyed after use
Definition drmbuffer.h:132
int m_objIdx[4]
index of the objects
Definition drmbuffer.h:121
bool m_closeHandleOnDestroy
true, if DMA-BUF handle should be closed on destroy
Definition drmbuffer.h:133
void Setup(int, uint32_t, uint32_t, uint32_t, AVDRMFrameDescriptor *, bool)
Setup the buffer.
uint64_t m_modifier
Definition drmbuffer.h:129
int m_numObjects
number of prime objects in the buffer
Definition drmbuffer.h:120
uint32_t m_fbId
framebuffer id
Definition drmbuffer.h:114
uint32_t m_width
buffer width
Definition drmbuffer.h:106
uint32_t m_offset[4]
array of the plane offset
Definition drmbuffer.h:126
uint32_t m_pitch[4]
array of the plane pitch
Definition drmbuffer.h:127
void FillBlack(void)
Color the buffer black.
AVFrame * frame
associated AVFrame
Definition drmbuffer.h:99
uint32_t m_objectPrimeHandle[4]
primedata objects prime handles (count is numObjects, index is objIdx)
Definition drmbuffer.h:122
int m_dmaBufHandle[4]
DMA-BUF file descriptor.
Definition drmbuffer.h:119
uint32_t m_pixFmt
buffer pixel format
Definition drmbuffer.h:108
int m_drmDeviceFd
drm device file descriptor
Definition drmbuffer.h:115
uint32_t m_height
buffer height
Definition drmbuffer.h:107
void PresentationFinished(void)
The presentation of this buffer has finished.
bool m_presentationPending
true, if buffer presentation is pending
Definition drmbuffer.h:131
uint8_t * m_pPlane[4]
array of the plane data
Definition drmbuffer.h:124
void Destroy(void)
Clear and destroy the buffer object and its parameters.
Definition drmbuffer.cpp:90
uint32_t m_size[4]
array of the plane size
Definition drmbuffer.h:128
uint32_t m_planePrimeHandle[4]
array of the plane handles
Definition drmbuffer.h:125
int m_numPlanes
number of planes in the buffer
Definition drmbuffer.h:117
std::vector< std::unique_ptr< cDrmBuffer > > buffer
Definition pool.h:24
#define ARRAY_SIZE(arr)
const struct format_info * FindFormat(uint32_t format)
Find infos for the given pixel format.
static const struct format_info format_info_array[]
Holds the infos of a pixel format.
DRM Buffer Header File.
int plane
#define LOGDEBUG2
log to LOG_DEBUG and add a prefix
Definition logger.h:47
#define LOGERROR
log to LOG_ERR
Definition logger.h:39
#define LOGFATAL
log to LOG_ERR and abort
Definition logger.h:37
@ L_DRM
drm logs
Definition logger.h:60
Logger Header File.
Pool Implementation.
uint32_t format
Definition drmbuffer.h:37
uint8_t num_planes
Definition drmbuffer.h:39
struct format_plane_info planes[4]
Definition drmbuffer.h:40