2018-05-06 19:35:47 +02:00
|
|
|
#include "modeset.h"
|
|
|
|
|
2020-06-02 23:06:55 +02:00
|
|
|
#include "fifo.h"
|
|
|
|
|
2020-06-03 23:12:27 +02:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <pthread.h>
|
2020-06-04 22:33:49 +02:00
|
|
|
#include <semaphore.h>
|
|
|
|
|
2020-06-02 23:06:55 +02:00
|
|
|
typedef struct vsyncData
|
|
|
|
{
|
|
|
|
_image* i;
|
|
|
|
modeset_display_surface* s;
|
2020-06-03 23:12:27 +02:00
|
|
|
uint32_t flipPending;
|
|
|
|
uint64_t seqno;
|
2020-06-02 23:06:55 +02:00
|
|
|
} vsyncData;
|
|
|
|
|
2020-06-03 23:12:27 +02:00
|
|
|
#define FLIP_FIFO_SIZE 2
|
|
|
|
|
|
|
|
static uint32_t refCount = 0;
|
|
|
|
static pthread_t flipQueueThread = 0;
|
2020-06-03 23:40:58 +02:00
|
|
|
static Fifo flipQueueFifo;
|
2020-06-03 23:12:27 +02:00
|
|
|
static vsyncData dataMem[FLIP_FIFO_SIZE];
|
|
|
|
static FifoElem fifoMem[FLIP_FIFO_SIZE];
|
2020-06-04 22:33:49 +02:00
|
|
|
static sem_t flipQueueSem;
|
2020-06-03 23:12:27 +02:00
|
|
|
|
2020-06-07 20:06:44 +02:00
|
|
|
static sem_t savedStateSem;
|
|
|
|
|
2020-06-03 23:40:58 +02:00
|
|
|
static void* flipQueueThreadFunction(void* vargp)
|
2020-06-03 23:12:27 +02:00
|
|
|
{
|
|
|
|
uint32_t run = 1;
|
|
|
|
uint64_t lastFinishedSeqno = 0;
|
2020-06-03 23:40:58 +02:00
|
|
|
int threadFD = *(int*)vargp;
|
2020-06-03 23:12:27 +02:00
|
|
|
|
|
|
|
while(run)
|
|
|
|
{
|
|
|
|
|
|
|
|
|
2020-06-03 23:40:58 +02:00
|
|
|
uint64_t seqno = 0;
|
2020-06-03 23:12:27 +02:00
|
|
|
|
2020-06-04 22:33:49 +02:00
|
|
|
sem_wait(&flipQueueSem);
|
2020-06-03 23:40:58 +02:00
|
|
|
{
|
|
|
|
vsyncData* d = fifoGetLast(&flipQueueFifo);
|
|
|
|
if(d)
|
|
|
|
{
|
|
|
|
seqno = d->seqno;
|
|
|
|
}
|
2020-06-03 23:12:27 +02:00
|
|
|
|
2020-06-03 23:40:58 +02:00
|
|
|
run = refCount;
|
|
|
|
}
|
2020-06-04 22:33:49 +02:00
|
|
|
sem_post(&flipQueueSem);
|
2020-06-03 23:40:58 +02:00
|
|
|
|
|
|
|
if(seqno)
|
2020-06-03 23:12:27 +02:00
|
|
|
{
|
|
|
|
uint64_t timeOut = 1000000; //1 ms in ns
|
2020-06-03 23:40:58 +02:00
|
|
|
int ret = vc4_seqno_wait(threadFD, &lastFinishedSeqno, seqno, &timeOut);
|
2020-06-03 23:12:27 +02:00
|
|
|
|
|
|
|
if(ret > 0)
|
|
|
|
{
|
2020-06-04 22:33:49 +02:00
|
|
|
sem_wait(&flipQueueSem);
|
2020-06-03 23:12:27 +02:00
|
|
|
{
|
2020-06-04 22:33:49 +02:00
|
|
|
vsyncData* d = fifoGetLast(&flipQueueFifo);
|
|
|
|
|
|
|
|
d->seqno = 0; //done!
|
|
|
|
|
|
|
|
if(d->i->presentMode == VK_PRESENT_MODE_FIFO_KHR)
|
|
|
|
{
|
|
|
|
d->flipPending = 1;
|
|
|
|
drmModePageFlip(threadFD, d->s->crtc->crtc_id, d->i->fb, DRM_MODE_PAGE_FLIP_EVENT, d);
|
|
|
|
}
|
|
|
|
else if(d->i->presentMode == VK_PRESENT_MODE_IMMEDIATE_KHR)
|
|
|
|
{
|
|
|
|
drmModePageFlip(threadFD, d->s->crtc->crtc_id, d->i->fb, DRM_MODE_PAGE_FLIP_ASYNC, 0);
|
|
|
|
}
|
2020-06-03 23:12:27 +02:00
|
|
|
}
|
2020-06-04 22:33:49 +02:00
|
|
|
sem_post(&flipQueueSem);
|
2020-06-03 23:12:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-06-03 23:40:58 +02:00
|
|
|
|
|
|
|
return 0;
|
2020-06-03 23:12:27 +02:00
|
|
|
}
|
2020-06-02 23:06:55 +02:00
|
|
|
|
2020-04-16 01:34:51 +02:00
|
|
|
void modeset_enum_displays(int fd, uint32_t* numDisplays, modeset_display* displays)
|
2020-04-16 01:04:41 +02:00
|
|
|
{
|
|
|
|
drmModeResPtr resPtr = drmModeGetResources(fd);
|
|
|
|
|
|
|
|
uint32_t tmpNumDisplays = 0;
|
|
|
|
modeset_display tmpDisplays[16];
|
|
|
|
|
|
|
|
for(uint32_t c = 0; c < resPtr->count_connectors; ++c)
|
|
|
|
{
|
|
|
|
drmModeConnectorPtr connPtr = drmModeGetConnector(fd, resPtr->connectors[c]);
|
|
|
|
|
|
|
|
if(connPtr->connection != DRM_MODE_CONNECTED)
|
|
|
|
{
|
|
|
|
continue; //skip unused connector
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!connPtr->count_modes)
|
|
|
|
{
|
|
|
|
continue; //skip connectors with no valid modes
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(tmpDisplays[tmpNumDisplays].name, connPtr->modes[0].name, 32);
|
|
|
|
|
|
|
|
tmpDisplays[tmpNumDisplays].mmWidth = connPtr->mmWidth;
|
|
|
|
tmpDisplays[tmpNumDisplays].mmHeight = connPtr->mmHeight;
|
|
|
|
tmpDisplays[tmpNumDisplays].resWidth = connPtr->modes[0].hdisplay;
|
|
|
|
tmpDisplays[tmpNumDisplays].resHeight = connPtr->modes[0].vdisplay;
|
|
|
|
tmpDisplays[tmpNumDisplays].connectorID = connPtr->connector_id;
|
|
|
|
|
|
|
|
tmpNumDisplays++;
|
|
|
|
|
|
|
|
assert(tmpNumDisplays < 16);
|
|
|
|
|
|
|
|
drmModeFreeConnector(connPtr);
|
|
|
|
}
|
|
|
|
|
|
|
|
drmModeFreeResources(resPtr);
|
|
|
|
|
|
|
|
*numDisplays = tmpNumDisplays;
|
|
|
|
|
2020-04-16 01:34:51 +02:00
|
|
|
memcpy(displays, tmpDisplays, tmpNumDisplays * sizeof(modeset_display));
|
2020-04-16 01:04:41 +02:00
|
|
|
}
|
|
|
|
|
2020-04-16 01:34:51 +02:00
|
|
|
void modeset_enum_modes_for_display(int fd, uint32_t display, uint32_t* numModes, modeset_display_mode* modes)
|
2020-04-16 01:04:41 +02:00
|
|
|
{
|
|
|
|
drmModeResPtr resPtr = drmModeGetResources(fd);
|
|
|
|
|
|
|
|
drmModeConnectorPtr connPtr = drmModeGetConnector(fd, display);
|
|
|
|
|
2020-04-16 18:33:46 +02:00
|
|
|
if(!connPtr)
|
|
|
|
{
|
|
|
|
*numModes = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-04-16 01:04:41 +02:00
|
|
|
uint32_t tmpNumModes = 0;
|
|
|
|
modeset_display_mode tmpModes[1024];
|
|
|
|
|
|
|
|
for(uint32_t c = 0; c < connPtr->count_modes; ++c)
|
|
|
|
{
|
2020-05-31 15:00:19 +02:00
|
|
|
uint32_t found = 0;
|
|
|
|
for(uint32_t d = 0; d < tmpNumModes; ++d)
|
|
|
|
{
|
|
|
|
if(tmpModes[d].refreshRate == connPtr->modes[c].vrefresh &&
|
|
|
|
tmpModes[d].resWidth == connPtr->modes[c].hdisplay &&
|
|
|
|
tmpModes[d].resHeight == connPtr->modes[c].vdisplay)
|
|
|
|
{
|
|
|
|
found = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(found)
|
|
|
|
{
|
|
|
|
//skip modes that have lower "clock"
|
|
|
|
//but otherwise same resolution and vrefresh
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-04-16 01:04:41 +02:00
|
|
|
tmpModes[tmpNumModes].connectorID = display;
|
|
|
|
tmpModes[tmpNumModes].modeID = c;
|
|
|
|
tmpModes[tmpNumModes].refreshRate = connPtr->modes[c].vrefresh;
|
|
|
|
tmpModes[tmpNumModes].resWidth = connPtr->modes[c].hdisplay;
|
|
|
|
tmpModes[tmpNumModes].resHeight = connPtr->modes[c].vdisplay;
|
|
|
|
|
2020-05-31 15:00:19 +02:00
|
|
|
//explained here:
|
|
|
|
// https://01.org/linuxgraphics/gfx-docs/drm/API-struct-drm-display-mode.html
|
|
|
|
|
2020-05-19 18:42:57 +02:00
|
|
|
// char* typestr = 0;
|
|
|
|
// switch(connPtr->modes[c].type)
|
|
|
|
// {
|
|
|
|
// case DRM_MODE_TYPE_BUILTIN:
|
|
|
|
// typestr = "DRM_MODE_TYPE_BUILTIN";
|
|
|
|
// break;
|
|
|
|
// case DRM_MODE_TYPE_CLOCK_C:
|
|
|
|
// typestr = "DRM_MODE_TYPE_CLOCK_C";
|
|
|
|
// break;
|
|
|
|
// case DRM_MODE_TYPE_CRTC_C:
|
|
|
|
// typestr = "DRM_MODE_TYPE_CRTC_C";
|
|
|
|
// break;
|
|
|
|
// case DRM_MODE_TYPE_PREFERRED:
|
|
|
|
// typestr = "DRM_MODE_TYPE_PREFERRED";
|
|
|
|
// break;
|
|
|
|
// case DRM_MODE_TYPE_DEFAULT:
|
|
|
|
// typestr = "DRM_MODE_TYPE_DEFAULT";
|
|
|
|
// break;
|
|
|
|
// case DRM_MODE_TYPE_USERDEF:
|
|
|
|
// typestr = "DRM_MODE_TYPE_USERDEF";
|
|
|
|
// break;
|
|
|
|
// case DRM_MODE_TYPE_DRIVER:
|
|
|
|
// typestr = "DRM_MODE_TYPE_DRIVER";
|
|
|
|
// break;
|
|
|
|
// default:
|
|
|
|
// typestr = "UNKNOWN";
|
|
|
|
// break;
|
|
|
|
// }
|
|
|
|
|
|
|
|
// fprintf(stderr, "\nflags ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_PHSYNC) fprintf(stderr, "DRM_MODE_FLAG_PHSYNC ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_NHSYNC) fprintf(stderr, "DRM_MODE_FLAG_NHSYNC ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_PVSYNC) fprintf(stderr, "DRM_MODE_FLAG_PVSYNC ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_NVSYNC) fprintf(stderr, "DRM_MODE_FLAG_NVSYNC ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_INTERLACE) fprintf(stderr, "DRM_MODE_FLAG_INTERLACE ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_DBLSCAN) fprintf(stderr, "DRM_MODE_FLAG_DBLSCAN ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_CSYNC) fprintf(stderr, "DRM_MODE_FLAG_CSYNC ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_PCSYNC) fprintf(stderr, "DRM_MODE_FLAG_PCSYNC ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_NCSYNC) fprintf(stderr, "DRM_MODE_FLAG_NCSYNC ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_HSKEW) fprintf(stderr, "DRM_MODE_FLAG_HSKEW ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_BCAST) fprintf(stderr, "DRM_MODE_FLAG_BCAST ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_PIXMUX) fprintf(stderr, "DRM_MODE_FLAG_PIXMUX ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_DBLCLK) fprintf(stderr, "DRM_MODE_FLAG_DBLCLK ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_CLKDIV2) fprintf(stderr, "DRM_MODE_FLAG_CLKDIV2 ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_3D_MASK) fprintf(stderr, "DRM_MODE_FLAG_3D_MASK ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_3D_NONE) fprintf(stderr, "DRM_MODE_FLAG_3D_NONE ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_3D_FRAME_PACKING) fprintf(stderr, "DRM_MODE_FLAG_3D_FRAME_PACKING ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_3D_FIELD_ALTERNATIVE) fprintf(stderr, "DRM_MODE_FLAG_3D_FIELD_ALTERNATIVE ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_3D_LINE_ALTERNATIVE) fprintf(stderr, "DRM_MODE_FLAG_3D_LINE_ALTERNATIVE ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_3D_SIDE_BY_SIDE_FULL) fprintf(stderr, "DRM_MODE_FLAG_3D_SIDE_BY_SIDE_FULL ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_3D_L_DEPTH) fprintf(stderr, "DRM_MODE_FLAG_3D_L_DEPTH ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_3D_L_DEPTH_GFX_GFX_DEPTH) fprintf(stderr, "DRM_MODE_FLAG_3D_L_DEPTH_GFX_GFX_DEPTH ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_3D_TOP_AND_BOTTOM) fprintf(stderr, "DRM_MODE_FLAG_3D_TOP_AND_BOTTOM ");
|
|
|
|
// if(connPtr->modes[c].flags & DRM_MODE_FLAG_3D_SIDE_BY_SIDE_HALF) fprintf(stderr, "DRM_MODE_FLAG_3D_SIDE_BY_SIDE_HALF ");
|
|
|
|
|
|
|
|
// fprintf(stderr, "\nclock %u vrefresh %u type %s\n", connPtr->modes[c].clock, connPtr->modes[c].vrefresh, typestr);
|
|
|
|
// fprintf(stderr, "hdisplay %u hsync_start %u hsync_end %u htotal %u hskew %u\n", connPtr->modes[c].hdisplay, connPtr->modes[c].hsync_start, connPtr->modes[c].hsync_end, connPtr->modes[c].htotal, connPtr->modes[c].hskew);
|
|
|
|
// fprintf(stderr, "vdisplay %u vsync_start %u vsync_end %u vtotal %u vscan %u\n", connPtr->modes[c].vdisplay, connPtr->modes[c].vsync_start, connPtr->modes[c].vsync_end, connPtr->modes[c].vtotal, connPtr->modes[c].vscan);
|
|
|
|
|
2020-04-16 01:04:41 +02:00
|
|
|
tmpNumModes++;
|
|
|
|
|
|
|
|
assert(tmpNumModes < 1024);
|
|
|
|
}
|
|
|
|
|
|
|
|
drmModeFreeConnector(connPtr);
|
|
|
|
drmModeFreeResources(resPtr);
|
|
|
|
|
|
|
|
*numModes = tmpNumModes;
|
2020-04-16 01:34:51 +02:00
|
|
|
memcpy(modes, tmpModes, tmpNumModes * sizeof(modeset_display_mode));
|
2020-04-16 01:04:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void modeset_create_surface_for_mode(int fd, uint32_t display, uint32_t mode, modeset_display_surface* surface)
|
|
|
|
{
|
2020-04-16 19:07:45 +02:00
|
|
|
// modeset_debug_print(fd);
|
2020-04-16 18:33:46 +02:00
|
|
|
|
2020-06-03 23:12:27 +02:00
|
|
|
if(!refCount)
|
2020-06-02 23:06:55 +02:00
|
|
|
{
|
2020-06-03 23:12:27 +02:00
|
|
|
flipQueueFifo = createFifo(dataMem, fifoMem, FLIP_FIFO_SIZE, sizeof(vsyncData));
|
|
|
|
pthread_create(&flipQueueThread, 0, flipQueueThreadFunction, &fd);
|
2020-06-04 22:33:49 +02:00
|
|
|
sem_init(&flipQueueSem, 0, 0);
|
|
|
|
sem_post(&flipQueueSem);
|
2020-06-07 20:06:44 +02:00
|
|
|
|
|
|
|
sem_init(&savedStateSem, 0, 0);
|
|
|
|
sem_post(&savedStateSem);
|
2020-06-02 23:06:55 +02:00
|
|
|
}
|
|
|
|
|
2020-06-03 23:12:27 +02:00
|
|
|
refCount++;
|
|
|
|
|
2020-04-16 18:33:46 +02:00
|
|
|
surface->savedState = 0;
|
|
|
|
|
2020-04-16 01:04:41 +02:00
|
|
|
drmModeResPtr resPtr = drmModeGetResources(fd);
|
|
|
|
|
|
|
|
drmModeConnectorPtr connPtr = drmModeGetConnector(fd, display);
|
|
|
|
|
2020-04-16 18:33:46 +02:00
|
|
|
if(!connPtr)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-04-16 19:17:35 +02:00
|
|
|
uint32_t found = 0;
|
2020-04-16 18:33:46 +02:00
|
|
|
|
2020-04-16 01:04:41 +02:00
|
|
|
//if current encoder is valid, try to use that
|
|
|
|
if(connPtr->encoder_id)
|
|
|
|
{
|
2020-04-16 19:17:35 +02:00
|
|
|
drmModeEncoderPtr encPtr = drmModeGetEncoder(fd, connPtr->encoder_id);
|
2020-04-16 01:04:41 +02:00
|
|
|
|
2020-04-16 19:17:35 +02:00
|
|
|
if(encPtr && encPtr->crtc_id)
|
2020-04-16 01:04:41 +02:00
|
|
|
{
|
2020-04-16 18:33:46 +02:00
|
|
|
surface->connector = connPtr;
|
2020-04-16 01:04:41 +02:00
|
|
|
surface->modeID = mode;
|
2020-04-16 18:33:46 +02:00
|
|
|
surface->crtc = drmModeGetCrtc(fd, encPtr->crtc_id);
|
2020-04-16 19:07:45 +02:00
|
|
|
|
2020-04-16 19:17:35 +02:00
|
|
|
found = 1;
|
2020-04-16 18:33:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
drmModeFreeEncoder(encPtr);
|
|
|
|
}
|
2020-04-16 19:07:45 +02:00
|
|
|
|
2020-04-16 19:17:35 +02:00
|
|
|
if(!found)
|
|
|
|
{
|
|
|
|
// To find a suitable CRTC, you need to iterate over the list of encoders that are available
|
|
|
|
// for each connector. Each encoder contains a list of CRTCs that it can work with and you
|
|
|
|
// simply select one of these CRTCs. If you later program the CRTC to control a connector,
|
|
|
|
// it automatically selects the best encoder
|
|
|
|
|
|
|
|
// TODO if we were to output to multiple displays, we'd need to make sure we don't use a CRTC
|
|
|
|
// that we'd be using to drive the other screen
|
|
|
|
|
|
|
|
for(uint32_t c = 0; c < connPtr->count_encoders; ++c)
|
|
|
|
{
|
|
|
|
drmModeEncoderPtr encPtr = drmModeGetEncoder(fd, connPtr->encoders[c]);
|
|
|
|
|
|
|
|
for(uint32_t d = 0; d < 32; ++d)
|
|
|
|
{
|
|
|
|
if(encPtr->possible_crtcs & (1 << d))
|
|
|
|
{
|
|
|
|
surface->connector = connPtr;
|
|
|
|
surface->modeID = mode;
|
|
|
|
surface->crtc = drmModeGetCrtc(fd, resPtr->crtcs[d]);
|
|
|
|
|
|
|
|
found = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(found)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-16 19:07:45 +02:00
|
|
|
drmModeFreeResources(resPtr);
|
2020-04-16 19:17:35 +02:00
|
|
|
|
|
|
|
//fprintf(stderr, "connector id %i, crtc id %i\n", connPtr->connector_id, encPtr->crtc_id);
|
2020-04-16 18:33:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void modeset_create_fb_for_surface(int fd, _image* buf, modeset_display_surface* surface)
|
|
|
|
{
|
2020-05-31 15:18:32 +02:00
|
|
|
int bpp = buf->format == VK_FORMAT_B8G8R8A8_UNORM ? 32 : 16;
|
|
|
|
int ret = drmModeAddFB(fd, buf->width, buf->height, bpp, bpp, buf->stride, buf->boundMem->bo, &buf->fb);
|
2020-04-16 18:33:46 +02:00
|
|
|
|
|
|
|
if(ret)
|
|
|
|
{
|
|
|
|
buf->fb = 0;
|
|
|
|
fprintf(stderr, "cannot create framebuffer (%d): %m\n", errno);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void modeset_destroy_fb(int fd, _image* buf)
|
|
|
|
{
|
|
|
|
// delete framebuffer
|
|
|
|
drmModeRmFB(fd, buf->fb);
|
|
|
|
}
|
|
|
|
|
2020-05-31 16:00:45 +02:00
|
|
|
static void modeset_page_flip_event(int fd, unsigned int frame,
|
|
|
|
unsigned int sec, unsigned int usec,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
if(data)
|
|
|
|
{
|
2020-06-04 22:33:49 +02:00
|
|
|
sem_wait(&flipQueueSem);
|
2020-06-02 23:06:55 +02:00
|
|
|
{
|
2020-06-04 22:33:49 +02:00
|
|
|
if(data)
|
|
|
|
{
|
|
|
|
vsyncData* d = data;
|
2020-06-02 23:06:55 +02:00
|
|
|
|
2020-06-04 22:33:49 +02:00
|
|
|
d->flipPending = 0;
|
|
|
|
}
|
2020-06-02 23:06:55 +02:00
|
|
|
}
|
2020-06-04 22:33:49 +02:00
|
|
|
sem_post(&flipQueueSem);
|
2020-05-31 16:00:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-02 23:06:55 +02:00
|
|
|
void modeset_acquire_image(int fd, _image** buf, modeset_display_surface** surface)
|
2020-06-01 23:52:36 +02:00
|
|
|
{
|
2020-06-03 23:12:27 +02:00
|
|
|
uint32_t pending = 1, gpuprocessing = 1;
|
2020-06-02 23:06:55 +02:00
|
|
|
|
2020-06-04 22:33:49 +02:00
|
|
|
sem_wait(&flipQueueSem);
|
2020-06-02 23:06:55 +02:00
|
|
|
|
|
|
|
vsyncData* last = fifoGetLast(&flipQueueFifo);
|
|
|
|
|
|
|
|
if(last)
|
|
|
|
{
|
2020-06-03 23:12:27 +02:00
|
|
|
pending = last->flipPending;
|
|
|
|
gpuprocessing = last->seqno > 0;
|
2020-06-02 23:06:55 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
//fifo empty, just use any image
|
|
|
|
*buf = 0;
|
|
|
|
*surface = 0;
|
|
|
|
|
2020-06-04 22:33:49 +02:00
|
|
|
sem_post(&flipQueueSem);
|
2020-06-02 23:06:55 +02:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-06-04 22:33:49 +02:00
|
|
|
sem_post(&flipQueueSem);
|
2020-06-02 23:06:55 +02:00
|
|
|
|
2020-06-03 23:12:27 +02:00
|
|
|
while(pending || gpuprocessing)
|
2020-06-02 23:06:55 +02:00
|
|
|
{
|
2020-06-04 22:33:49 +02:00
|
|
|
if(pending)
|
|
|
|
{
|
|
|
|
drmEventContext ev;
|
|
|
|
memset(&ev, 0, sizeof(ev));
|
|
|
|
ev.version = 2;
|
|
|
|
ev.page_flip_handler = modeset_page_flip_event;
|
|
|
|
drmHandleEvent(fd, &ev);
|
|
|
|
}
|
2020-06-02 23:06:55 +02:00
|
|
|
|
2020-06-04 22:33:49 +02:00
|
|
|
sem_wait(&flipQueueSem);
|
|
|
|
{
|
|
|
|
vsyncData* d = fifoGetLast(&flipQueueFifo);
|
2020-06-02 23:06:55 +02:00
|
|
|
|
2020-06-04 22:33:49 +02:00
|
|
|
//a frame must be in flight
|
|
|
|
//so fifo must contain something
|
|
|
|
assert(d);
|
2020-06-02 23:06:55 +02:00
|
|
|
|
2020-06-04 22:33:49 +02:00
|
|
|
pending = d->flipPending;
|
|
|
|
gpuprocessing = d->seqno > 0;
|
|
|
|
}
|
|
|
|
sem_post(&flipQueueSem);
|
2020-06-02 23:06:55 +02:00
|
|
|
}
|
|
|
|
|
2020-06-04 22:33:49 +02:00
|
|
|
sem_wait(&flipQueueSem);
|
|
|
|
{
|
|
|
|
vsyncData d;
|
2020-06-03 23:12:27 +02:00
|
|
|
|
2020-06-04 22:33:49 +02:00
|
|
|
fifoRemove(&flipQueueFifo, &d);
|
2020-06-02 23:06:55 +02:00
|
|
|
|
2020-06-04 22:33:49 +02:00
|
|
|
*buf = d.i;
|
|
|
|
*surface = d.s;
|
|
|
|
}
|
|
|
|
sem_post(&flipQueueSem);
|
2020-06-01 23:52:36 +02:00
|
|
|
}
|
|
|
|
|
2020-06-03 23:12:27 +02:00
|
|
|
void modeset_present(int fd, _image *buf, modeset_display_surface* surface, uint64_t seqno)
|
2020-04-16 18:33:46 +02:00
|
|
|
{
|
|
|
|
if(!surface->savedState)
|
|
|
|
{
|
2020-06-07 20:06:44 +02:00
|
|
|
sem_wait(&savedStateSem);
|
2020-04-16 18:33:46 +02:00
|
|
|
{
|
2020-06-07 20:06:44 +02:00
|
|
|
for(uint32_t c = 0; c < 32; ++c)
|
2020-04-16 18:33:46 +02:00
|
|
|
{
|
2020-06-07 20:06:44 +02:00
|
|
|
if(!modeset_saved_states[c].used)
|
|
|
|
{
|
|
|
|
drmModeConnectorPtr tmpConnPtr = drmModeGetConnector(fd, surface->connector->connector_id);
|
|
|
|
drmModeCrtcPtr tmpCrtcPtr = drmModeGetCrtc(fd, surface->crtc->crtc_id);
|
|
|
|
modeset_saved_states[c].used = 1;
|
|
|
|
modeset_saved_states[c].conn = tmpConnPtr;
|
|
|
|
modeset_saved_states[c].crtc = tmpCrtcPtr;
|
|
|
|
surface->savedState = c;
|
|
|
|
break;
|
|
|
|
}
|
2020-04-16 18:33:46 +02:00
|
|
|
}
|
|
|
|
|
2020-06-07 20:06:44 +02:00
|
|
|
int ret = drmModeSetCrtc(fd, surface->crtc->crtc_id, buf->fb, 0, 0, &surface->connector->connector_id, 1, &surface->connector->modes[surface->modeID]);
|
|
|
|
if(ret)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "cannot set CRTC for connector %u: %m\n",
|
|
|
|
surface->connector->connector_id, errno);
|
|
|
|
}
|
2020-05-19 18:42:57 +02:00
|
|
|
}
|
2020-06-07 20:06:44 +02:00
|
|
|
sem_post(&savedStateSem);
|
2020-04-16 01:04:41 +02:00
|
|
|
}
|
2020-04-16 18:33:46 +02:00
|
|
|
|
2020-06-03 01:06:26 +02:00
|
|
|
//TODO presenting needs to happen *after* the gpu is done with rendering to an image
|
|
|
|
//then immediate mode flips the page immediately, but fifo will post it for next vblank
|
|
|
|
//otherwise fifo would tear too
|
|
|
|
//so we should add images to the flip queue
|
|
|
|
//then wait for the first submitted (first out) seqno to finish
|
|
|
|
//then perform pageflip for that image
|
|
|
|
//
|
2020-06-03 23:12:27 +02:00
|
|
|
//drmModePageFlip(fd, surface->crtc->crtc_id, buf->fb, DRM_MODE_PAGE_FLIP_EVENT, first);
|
|
|
|
//drmModePageFlip(fd, surface->crtc->crtc_id, buf->fb, DRM_MODE_PAGE_FLIP_ASYNC, 0);
|
2020-05-31 16:00:45 +02:00
|
|
|
|
2020-06-03 23:12:27 +02:00
|
|
|
vsyncData d;
|
|
|
|
d.i = buf;
|
|
|
|
d.s = surface;
|
|
|
|
d.flipPending = 0;
|
|
|
|
d.seqno = seqno;
|
2020-06-03 01:06:26 +02:00
|
|
|
|
2020-06-03 23:12:27 +02:00
|
|
|
uint32_t added = 0;
|
2020-06-03 01:06:26 +02:00
|
|
|
|
2020-06-03 23:12:27 +02:00
|
|
|
while(!added)
|
2020-05-31 16:00:45 +02:00
|
|
|
{
|
2020-06-04 22:33:49 +02:00
|
|
|
sem_wait(&flipQueueSem);
|
|
|
|
{
|
|
|
|
//try to add request to queue
|
|
|
|
added = fifoAdd(&flipQueueFifo, &d);
|
|
|
|
}
|
|
|
|
sem_post(&flipQueueSem);
|
2020-05-31 16:00:45 +02:00
|
|
|
}
|
2020-04-16 18:33:46 +02:00
|
|
|
|
|
|
|
//modeset_debug_print(fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
void modeset_destroy_surface(int fd, modeset_display_surface *surface)
|
|
|
|
{
|
|
|
|
//restore old state
|
|
|
|
drmModeSetCrtc(fd, modeset_saved_states[surface->savedState].crtc->crtc_id,
|
|
|
|
modeset_saved_states[surface->savedState].crtc->buffer_id,
|
|
|
|
modeset_saved_states[surface->savedState].crtc->x,
|
|
|
|
modeset_saved_states[surface->savedState].crtc->y,
|
|
|
|
&modeset_saved_states[surface->savedState].conn->connector_id,
|
|
|
|
1,
|
|
|
|
&modeset_saved_states[surface->savedState].crtc->mode);
|
|
|
|
|
|
|
|
{
|
2020-06-03 23:12:27 +02:00
|
|
|
refCount--;
|
|
|
|
|
2020-06-07 20:06:44 +02:00
|
|
|
sem_wait(&savedStateSem);
|
|
|
|
{
|
|
|
|
drmModeFreeConnector(modeset_saved_states[surface->savedState].conn);
|
|
|
|
drmModeFreeCrtc(modeset_saved_states[surface->savedState].crtc);
|
|
|
|
modeset_saved_states[surface->savedState].used = 0;
|
|
|
|
}
|
|
|
|
sem_post(&savedStateSem);
|
2020-04-16 18:33:46 +02:00
|
|
|
|
2020-06-03 23:12:27 +02:00
|
|
|
if(!refCount)
|
|
|
|
{
|
|
|
|
destroyFifo(&flipQueueFifo);
|
|
|
|
pthread_join(flipQueueThread, 0);
|
|
|
|
}
|
2020-04-16 18:33:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
drmModeFreeConnector(surface->connector);
|
|
|
|
drmModeFreeCrtc(surface->crtc);
|
|
|
|
}
|
|
|
|
|
|
|
|
void modeset_debug_print(int fd)
|
|
|
|
{
|
|
|
|
drmModeResPtr resPtr = drmModeGetResources(fd);
|
|
|
|
printf("res min width %i height %i\n", resPtr->min_width, resPtr->min_height);
|
|
|
|
printf("res max width %i height %i\n", resPtr->max_width, resPtr->max_height);
|
|
|
|
|
|
|
|
printf("\ncrtc count %i\n", resPtr->count_crtcs);
|
|
|
|
for(uint32_t c = 0; c < resPtr->count_crtcs; ++c)
|
|
|
|
{
|
|
|
|
drmModeCrtcPtr tmpCrtcPtr = drmModeGetCrtc(fd, resPtr->crtcs[c]);
|
|
|
|
printf("crtc id %i, buffer id %i\n", tmpCrtcPtr->crtc_id, tmpCrtcPtr->buffer_id);
|
|
|
|
drmModeFreeCrtc(tmpCrtcPtr);
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("\nfb count %i\n", resPtr->count_fbs);
|
|
|
|
for(uint32_t c = 0; c < resPtr->count_fbs; ++c)
|
|
|
|
{
|
|
|
|
drmModeFBPtr tmpFBptr = drmModeGetFB(fd, resPtr->fbs[c]);
|
|
|
|
printf("fb id %i, handle %i\n", tmpFBptr->fb_id, tmpFBptr->handle);
|
|
|
|
drmModeFreeFB(tmpFBptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("\nencoder count %i\n", resPtr->count_encoders);
|
|
|
|
for(uint32_t c = 0; c < resPtr->count_encoders; ++c)
|
|
|
|
{
|
|
|
|
drmModeEncoderPtr tmpEncoderPtr = drmModeGetEncoder(fd, resPtr->encoders[c]);
|
|
|
|
printf("encoder id %i, crtc id %i\n", tmpEncoderPtr->encoder_id, tmpEncoderPtr->crtc_id);
|
|
|
|
printf("possible crtcs: ");
|
|
|
|
for(uint32_t c = 0; c < 32; ++c)
|
|
|
|
{
|
|
|
|
if(tmpEncoderPtr->possible_crtcs & (1 << c))
|
|
|
|
{
|
|
|
|
printf("%i, ", c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
|
|
|
|
printf("possible clones: ");
|
|
|
|
for(uint32_t c = 0; c < 32; ++c)
|
|
|
|
{
|
|
|
|
if(tmpEncoderPtr->possible_clones & (1 << c))
|
|
|
|
{
|
|
|
|
printf("%i, ", c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
|
|
|
|
drmModeFreeEncoder(tmpEncoderPtr);
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("\nconnector count %i\n", resPtr->count_connectors);
|
|
|
|
for(uint32_t c = 0; c < resPtr->count_connectors; ++c)
|
|
|
|
{
|
|
|
|
drmModeConnectorPtr tmpConnPtr = drmModeGetConnector(fd, resPtr->connectors[c]);
|
|
|
|
printf("connector id %i, encoder id %i\n", tmpConnPtr->connector_id, tmpConnPtr->encoder_id);
|
|
|
|
drmModeFreeConnector(tmpConnPtr);
|
|
|
|
}
|
|
|
|
|
|
|
|
drmModeFreeResources(resPtr);
|
2020-04-16 01:04:41 +02:00
|
|
|
}
|