/* Wayland compositor running on top of an X server.
Copyright (C) 2022 to various contributors.
This file is part of 12to11.
12to11 is free software: you can redistribute it and/or modify it
under the terms of the GNU General Public License as published by the
Free Software Foundation, either version 3 of the License, or (at your
option) any later version.
12to11 is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
for more details.
You should have received a copy of the GNU General Public License
along with 12to11. If not, see . */
#include
#include
#include "compositor.h"
#include
#include
#include
struct _Fence
{
/* The xshmfence. */
struct xshmfence *fence;
/* The sync fence. */
XSyncFence fence_id;
/* The number of references to this fence. Incremented by
FenceRetain, decremented by FenceRelease. */
int refcount;
};
Fence *
GetFence (void)
{
Fence *fence;
int fd;
Window drawable;
drawable = DefaultRootWindow (compositor.display);
/* Allocate a new fence. */
fence = XLCalloc (1, sizeof *fence);
fd = xshmfence_alloc_shm ();
if (fd < 0)
{
perror ("xshmfence_alloc_shm");
abort ();
}
/* Map it. */
fence->fence = xshmfence_map_shm (fd);
if (!fence->fence)
{
perror ("xshmfence_map_shm");
abort ();
}
/* Upload the fence to the X server. XCB will close the file
descriptor. */
fence->fence_id = xcb_generate_id (compositor.conn);
/* Make the file descriptor CLOEXEC, since it isn't closed
immediately. */
XLAddFdFlag (fd, FD_CLOEXEC, False);
xcb_dri3_fence_from_fd (compositor.conn, drawable,
fence->fence_id, 0, fd);
/* Retain the fence. */
FenceRetain (fence);
/* Return the fence. */
return fence;
}
void
FenceAwait (Fence *fence)
{
/* Wait for the fence to be triggered. */
xshmfence_await (fence->fence);
/* Reset the fence. */
xshmfence_reset (fence->fence);
}
void
FenceRelease (Fence *fence)
{
if (--fence->refcount)
return;
/* Unmap the fence. */
xshmfence_unmap_shm (fence->fence);
/* Destroy the fence. */
XSyncDestroyFence (compositor.display, fence->fence_id);
/* Free the fence. */
XLFree (fence);
}
void
FenceRetain (Fence *fence)
{
fence->refcount++;
}
XSyncFence
FenceToXFence (Fence *fence)
{
return fence->fence_id;
}