1016 lines
24 KiB
C
1016 lines
24 KiB
C
/*
|
|
* Copyright (c) 2015 Intel Corporation
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
* to deal in the Software without restriction, including without limitation
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
* Software is furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice (including the next
|
|
* paragraph) shall be included in all copies or substantial portions of the
|
|
* Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
* SOFTWARE.
|
|
*
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <X11/Xlib.h>
|
|
#include <X11/Xatom.h>
|
|
#include <X11/Xlib-xcb.h>
|
|
#include <X11/xshmfence.h>
|
|
#include <X11/Xutil.h>
|
|
#include <X11/Xlibint.h>
|
|
#include <X11/extensions/Xcomposite.h>
|
|
#include <X11/extensions/Xdamage.h>
|
|
#include <X11/extensions/dpms.h>
|
|
#include <X11/extensions/randr.h>
|
|
#include <X11/extensions/Xrandr.h>
|
|
#include <xcb/xcb.h>
|
|
#include <xcb/present.h>
|
|
#include <xcb/dri3.h>
|
|
#include <xcb/xfixes.h>
|
|
#include <xf86drm.h>
|
|
#include <i915_drm.h>
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <assert.h>
|
|
#include <errno.h>
|
|
#include <setjmp.h>
|
|
#include <signal.h>
|
|
#include <sys/wait.h>
|
|
|
|
#include "dri3.h"
|
|
|
|
static int _x_error_occurred;
|
|
static uint32_t stamp;
|
|
|
|
struct list {
|
|
struct list *next, *prev;
|
|
};
|
|
|
|
static void
|
|
list_init(struct list *list)
|
|
{
|
|
list->next = list->prev = list;
|
|
}
|
|
|
|
static inline void
|
|
__list_add(struct list *entry,
|
|
struct list *prev,
|
|
struct list *next)
|
|
{
|
|
next->prev = entry;
|
|
entry->next = next;
|
|
entry->prev = prev;
|
|
prev->next = entry;
|
|
}
|
|
|
|
static inline void
|
|
list_add(struct list *entry, struct list *head)
|
|
{
|
|
__list_add(entry, head, head->next);
|
|
}
|
|
|
|
static inline void
|
|
__list_del(struct list *prev, struct list *next)
|
|
{
|
|
next->prev = prev;
|
|
prev->next = next;
|
|
}
|
|
|
|
static inline void
|
|
_list_del(struct list *entry)
|
|
{
|
|
__list_del(entry->prev, entry->next);
|
|
}
|
|
|
|
static inline void
|
|
list_move(struct list *list, struct list *head)
|
|
{
|
|
if (list->prev != head) {
|
|
_list_del(list);
|
|
list_add(list, head);
|
|
}
|
|
}
|
|
|
|
#define __container_of(ptr, sample, member) \
|
|
(void *)((char *)(ptr) - ((char *)&(sample)->member - (char *)(sample)))
|
|
|
|
#define list_for_each_entry(pos, head, member) \
|
|
for (pos = __container_of((head)->next, pos, member); \
|
|
&pos->member != (head); \
|
|
pos = __container_of(pos->member.next, pos, member))
|
|
|
|
static int
|
|
_check_error_handler(Display *display,
|
|
XErrorEvent *event)
|
|
{
|
|
if (_x_error_occurred < 0)
|
|
return True;
|
|
|
|
printf("X11 error from display %s, serial=%ld, error=%d, req=%d.%d\n",
|
|
DisplayString(display),
|
|
event->serial,
|
|
event->error_code,
|
|
event->request_code,
|
|
event->minor_code);
|
|
_x_error_occurred++;
|
|
return False; /* ignored */
|
|
}
|
|
|
|
static double elapsed(const struct timespec *start,
|
|
const struct timespec *end)
|
|
{
|
|
return 1e6*(end->tv_sec - start->tv_sec) + (end->tv_nsec - start->tv_nsec)/1000;
|
|
}
|
|
|
|
struct buffer {
|
|
struct list link;
|
|
Pixmap pixmap;
|
|
struct dri3_fence fence;
|
|
int fd;
|
|
int busy;
|
|
int id;
|
|
};
|
|
|
|
#define DRI3 1
|
|
#define NOCOPY 2
|
|
#define ASYNC 4
|
|
static void run(Display *dpy, Window win, const char *name, unsigned options)
|
|
{
|
|
xcb_connection_t *c = XGetXCBConnection(dpy);
|
|
struct timespec start, end;
|
|
#define N_BACK 8
|
|
char test_name[128];
|
|
struct buffer buffer[N_BACK];
|
|
struct list mru;
|
|
Window root;
|
|
unsigned int width, height;
|
|
unsigned border, depth;
|
|
unsigned present_flags = 0;
|
|
xcb_xfixes_region_t update = 0;
|
|
int completed = 0;
|
|
int queued = 0;
|
|
uint32_t eid = 0;
|
|
void *Q = NULL;
|
|
int i, n;
|
|
|
|
list_init(&mru);
|
|
|
|
XGetGeometry(dpy, win,
|
|
&root, &i, &n, &width, &height, &border, &depth);
|
|
|
|
_x_error_occurred = 0;
|
|
|
|
for (n = 0; n < N_BACK; n++) {
|
|
buffer[n].pixmap = xcb_generate_id(c);
|
|
xcb_create_pixmap(c, depth, buffer[n].pixmap, win,
|
|
width, height);
|
|
buffer[n].fence.xid = 0;
|
|
buffer[n].fd = -1;
|
|
buffer[n].id = n;
|
|
if (options & DRI3) {
|
|
xcb_dri3_buffer_from_pixmap_reply_t *reply;
|
|
int *fds;
|
|
|
|
if (dri3_create_fence(dpy, win, &buffer[n].fence))
|
|
return;
|
|
|
|
reply = xcb_dri3_buffer_from_pixmap_reply (c,
|
|
xcb_dri3_buffer_from_pixmap(c, buffer[n].pixmap),
|
|
NULL);
|
|
if (reply == NULL)
|
|
return;
|
|
|
|
fds = xcb_dri3_buffer_from_pixmap_reply_fds (c, reply);
|
|
buffer[n].fd = fds[0];
|
|
free(reply);
|
|
|
|
/* start idle */
|
|
xshmfence_trigger(buffer[n].fence.addr);
|
|
}
|
|
buffer[n].busy = 0;
|
|
list_add(&buffer[n].link, &mru);
|
|
}
|
|
if (options & ASYNC)
|
|
present_flags |= XCB_PRESENT_OPTION_ASYNC;
|
|
if (options & NOCOPY) {
|
|
update = xcb_generate_id(c);
|
|
xcb_xfixes_create_region(c, update, 0, NULL);
|
|
present_flags |= XCB_PRESENT_OPTION_COPY;
|
|
}
|
|
|
|
if (!(options & DRI3)) {
|
|
eid = xcb_generate_id(c);
|
|
xcb_present_select_input(c, eid, win,
|
|
(options & NOCOPY ? 0 : XCB_PRESENT_EVENT_MASK_IDLE_NOTIFY) |
|
|
XCB_PRESENT_EVENT_MASK_COMPLETE_NOTIFY);
|
|
Q = xcb_register_for_special_xge(c, &xcb_present_id, eid, &stamp);
|
|
}
|
|
|
|
clock_gettime(CLOCK_MONOTONIC, &start);
|
|
do {
|
|
for (n = 0; n < 1000; n++) {
|
|
struct buffer *tmp, *b = NULL;
|
|
retry:
|
|
list_for_each_entry(tmp, &mru, link) {
|
|
if (tmp->fence.xid)
|
|
tmp->busy = !xshmfence_query(tmp->fence.addr);
|
|
if (!tmp->busy) {
|
|
b = tmp;
|
|
break;
|
|
}
|
|
}
|
|
if (options & DRI3) {
|
|
if (b == NULL)
|
|
goto retry;
|
|
|
|
xshmfence_reset(b->fence.addr);
|
|
queued--;
|
|
completed++;
|
|
} else while (b == NULL) {
|
|
xcb_present_generic_event_t *ev;
|
|
|
|
ev = (xcb_present_generic_event_t *)
|
|
xcb_wait_for_special_event(c, Q);
|
|
if (ev == NULL)
|
|
abort();
|
|
|
|
do {
|
|
switch (ev->evtype) {
|
|
case XCB_PRESENT_COMPLETE_NOTIFY:
|
|
completed++;
|
|
queued--;
|
|
break;
|
|
|
|
case XCB_PRESENT_EVENT_IDLE_NOTIFY:
|
|
{
|
|
xcb_present_idle_notify_event_t *ie = (xcb_present_idle_notify_event_t *)ev;
|
|
assert(ie->serial < N_BACK);
|
|
buffer[ie->serial].busy = 0;
|
|
if (b == NULL)
|
|
b = &buffer[ie->serial];
|
|
break;
|
|
}
|
|
}
|
|
free(ev);
|
|
} while ((ev = (xcb_present_generic_event_t *)xcb_poll_for_special_event(c, Q)));
|
|
}
|
|
|
|
b->busy = (options & NOCOPY) == 0;
|
|
xcb_present_pixmap(c, win, b->pixmap, b->id,
|
|
0, /* valid */
|
|
update, /* update */
|
|
0, /* x_off */
|
|
0, /* y_off */
|
|
None,
|
|
None, /* wait fence */
|
|
b->fence.xid,
|
|
present_flags,
|
|
0, /* target msc */
|
|
0, /* divisor */
|
|
0, /* remainder */
|
|
0, NULL);
|
|
list_move(&b->link, &mru);
|
|
queued++;
|
|
xcb_flush(c);
|
|
}
|
|
clock_gettime(CLOCK_MONOTONIC, &end);
|
|
} while (end.tv_sec < start.tv_sec + 10);
|
|
|
|
if (options & DRI3) {
|
|
struct buffer *b;
|
|
XID pixmap;
|
|
|
|
pixmap = xcb_generate_id(c);
|
|
xcb_create_pixmap(c, depth, pixmap, win, width, height);
|
|
xcb_present_pixmap(c, win, pixmap, 0xdeadbeef,
|
|
0, /* valid */
|
|
None, /* update */
|
|
0, /* x_off */
|
|
0, /* y_off */
|
|
None,
|
|
None, /* wait fence */
|
|
None,
|
|
0,
|
|
0, /* target msc */
|
|
0, /* divisor */
|
|
0, /* remainder */
|
|
0, NULL);
|
|
xcb_flush(c);
|
|
|
|
list_for_each_entry(b, &mru, link)
|
|
xshmfence_await(b->fence.addr);
|
|
|
|
xcb_free_pixmap(c, pixmap);
|
|
completed += queued;
|
|
} else while (queued) {
|
|
xcb_present_generic_event_t *ev;
|
|
|
|
ev = (xcb_present_generic_event_t *)
|
|
xcb_wait_for_special_event(c, Q);
|
|
if (ev == NULL)
|
|
abort();
|
|
|
|
do {
|
|
switch (ev->evtype) {
|
|
case XCB_PRESENT_COMPLETE_NOTIFY:
|
|
completed++;
|
|
queued--;
|
|
break;
|
|
|
|
case XCB_PRESENT_EVENT_IDLE_NOTIFY:
|
|
break;
|
|
}
|
|
free(ev);
|
|
} while ((ev = (xcb_present_generic_event_t *)xcb_poll_for_special_event(c, Q)));
|
|
}
|
|
clock_gettime(CLOCK_MONOTONIC, &end);
|
|
|
|
if (update)
|
|
xcb_xfixes_destroy_region(c, update);
|
|
for (n = 0; n < N_BACK; n++) {
|
|
if (buffer[n].fence.xid)
|
|
dri3_fence_free(dpy, &buffer[n].fence);
|
|
if (buffer[n].fd != -1)
|
|
close(buffer[n].fd);
|
|
xcb_free_pixmap(c, buffer[n].pixmap);
|
|
}
|
|
|
|
if (Q) {
|
|
xcb_discard_reply(c, xcb_present_select_input_checked(c, eid, win, 0).sequence);
|
|
XSync(dpy, True);
|
|
xcb_unregister_for_special_event(c, Q);
|
|
}
|
|
|
|
test_name[0] = '\0';
|
|
if (options) {
|
|
snprintf(test_name, sizeof(test_name), "(%s%s%s )",
|
|
options & NOCOPY ? " no-copy" : "",
|
|
options & DRI3 ? " dri3" : "",
|
|
options & ASYNC ? " async" : "");
|
|
}
|
|
printf("%s%s: Completed %d presents in %.1fs, %.3fus each (%.1f FPS)\n",
|
|
name, test_name,
|
|
completed, elapsed(&start, &end) / 1000000,
|
|
elapsed(&start, &end) / completed,
|
|
completed / (elapsed(&start, &end) / 1000000));
|
|
}
|
|
|
|
struct perpixel {
|
|
Window win;
|
|
struct buffer buffer[N_BACK];
|
|
struct list mru;
|
|
uint32_t eid;
|
|
void *Q;
|
|
int queued;
|
|
};
|
|
|
|
static void perpixel(Display *dpy,
|
|
int max_width, int max_height, unsigned options)
|
|
{
|
|
//const int sz = max_width * max_height;
|
|
const int sz = 1048;
|
|
struct perpixel *pp;
|
|
xcb_connection_t *c = XGetXCBConnection(dpy);
|
|
struct timespec start, end;
|
|
char test_name[128];
|
|
unsigned present_flags = 0;
|
|
xcb_xfixes_region_t update = 0;
|
|
int completed = 0;
|
|
int i, n;
|
|
|
|
pp = calloc(sz, sizeof(*pp));
|
|
if (!pp)
|
|
return;
|
|
|
|
for (i = 0; i < sz; i++) {
|
|
XSetWindowAttributes attr = { .override_redirect = 1 };
|
|
int depth = DefaultDepth(dpy, DefaultScreen(dpy));
|
|
pp[i].win = XCreateWindow(dpy, DefaultRootWindow(dpy),
|
|
i % max_width, i / max_width, 1, 1, 0, depth,
|
|
InputOutput,
|
|
DefaultVisual(dpy, DefaultScreen(dpy)),
|
|
CWOverrideRedirect, &attr);
|
|
XMapWindow(dpy, pp[i].win);
|
|
list_init(&pp[i].mru);
|
|
for (n = 0; n < N_BACK; n++) {
|
|
pp[i].buffer[n].pixmap = xcb_generate_id(c);
|
|
xcb_create_pixmap(c, depth, pp[i].buffer[n].pixmap,
|
|
pp[i].win, 1, 1);
|
|
pp[i].buffer[n].fence.xid = 0;
|
|
pp[i].buffer[n].fd = -1;
|
|
pp[i].buffer[n].id = n;
|
|
if (options & DRI3) {
|
|
xcb_dri3_buffer_from_pixmap_reply_t *reply;
|
|
int *fds;
|
|
|
|
if (dri3_create_fence(dpy, pp[i].win, &pp[i].buffer[n].fence))
|
|
return;
|
|
|
|
reply = xcb_dri3_buffer_from_pixmap_reply(c,
|
|
xcb_dri3_buffer_from_pixmap(c, pp[i].buffer[n].pixmap),
|
|
NULL);
|
|
if (reply == NULL)
|
|
return;
|
|
|
|
fds = xcb_dri3_buffer_from_pixmap_reply_fds(c, reply);
|
|
pp[i].buffer[n].fd = fds[0];
|
|
free(reply);
|
|
|
|
/* start idle */
|
|
xshmfence_trigger(pp[i].buffer[n].fence.addr);
|
|
}
|
|
pp[i].buffer[n].busy = 0;
|
|
list_add(&pp[i].buffer[n].link, &pp[i].mru);
|
|
}
|
|
|
|
if (!(options & DRI3)) {
|
|
pp[i].eid = xcb_generate_id(c);
|
|
xcb_present_select_input(c, pp[i].eid, pp[i].win,
|
|
(options & NOCOPY ? 0 : XCB_PRESENT_EVENT_MASK_IDLE_NOTIFY) |
|
|
XCB_PRESENT_EVENT_MASK_COMPLETE_NOTIFY);
|
|
pp[i].Q = xcb_register_for_special_xge(c, &xcb_present_id, pp[i].eid, &stamp);
|
|
}
|
|
pp[i].queued = 0;
|
|
}
|
|
|
|
XSync(dpy, True);
|
|
_x_error_occurred = 0;
|
|
|
|
if (options & ASYNC)
|
|
present_flags |= XCB_PRESENT_OPTION_ASYNC;
|
|
if (options & NOCOPY) {
|
|
update = xcb_generate_id(c);
|
|
xcb_xfixes_create_region(c, update, 0, NULL);
|
|
present_flags |= XCB_PRESENT_OPTION_COPY;
|
|
}
|
|
|
|
clock_gettime(CLOCK_MONOTONIC, &start);
|
|
do {
|
|
for (i = 0; i < sz; i++) {
|
|
struct buffer *tmp, *b = NULL;
|
|
retry:
|
|
list_for_each_entry(tmp, &pp[i].mru, link) {
|
|
if (tmp->fence.xid)
|
|
tmp->busy = !xshmfence_query(tmp->fence.addr);
|
|
if (!tmp->busy) {
|
|
b = tmp;
|
|
break;
|
|
}
|
|
}
|
|
if (options & DRI3) {
|
|
if (b == NULL)
|
|
goto retry;
|
|
|
|
xshmfence_reset(b->fence.addr);
|
|
pp[i].queued--;
|
|
completed++;
|
|
} else while (b == NULL) {
|
|
xcb_present_generic_event_t *ev;
|
|
|
|
ev = (xcb_present_generic_event_t *)
|
|
xcb_wait_for_special_event(c, pp[i].Q);
|
|
if (ev == NULL)
|
|
abort();
|
|
|
|
do {
|
|
switch (ev->evtype) {
|
|
case XCB_PRESENT_COMPLETE_NOTIFY:
|
|
completed++;
|
|
pp[i].queued--;
|
|
break;
|
|
|
|
case XCB_PRESENT_EVENT_IDLE_NOTIFY:
|
|
{
|
|
xcb_present_idle_notify_event_t *ie = (xcb_present_idle_notify_event_t *)ev;
|
|
assert(ie->serial < N_BACK);
|
|
pp[i].buffer[ie->serial].busy = 0;
|
|
if (b == NULL)
|
|
b = &pp[i].buffer[ie->serial];
|
|
break;
|
|
}
|
|
}
|
|
free(ev);
|
|
} while ((ev = (xcb_present_generic_event_t *)xcb_poll_for_special_event(c, pp[i].Q)));
|
|
}
|
|
|
|
b->busy = (options & NOCOPY) == 0;
|
|
xcb_present_pixmap(c, pp[i].win, b->pixmap, b->id,
|
|
0, /* valid */
|
|
update, /* update */
|
|
0, /* x_off */
|
|
0, /* y_off */
|
|
None,
|
|
None, /* wait fence */
|
|
b->fence.xid,
|
|
present_flags,
|
|
0, /* target msc */
|
|
0, /* divisor */
|
|
0, /* remainder */
|
|
0, NULL);
|
|
list_move(&b->link, &pp[i].mru);
|
|
pp[i].queued++;
|
|
}
|
|
xcb_flush(c);
|
|
clock_gettime(CLOCK_MONOTONIC, &end);
|
|
} while (end.tv_sec < start.tv_sec + 10);
|
|
|
|
for (i = 0; i < sz; i++) {
|
|
if (options & DRI3) {
|
|
int depth = DefaultDepth(dpy, DefaultScreen(dpy));
|
|
struct buffer *b;
|
|
XID pixmap;
|
|
|
|
pixmap = xcb_generate_id(c);
|
|
xcb_create_pixmap(c, depth, pixmap, pp[i].win, 1, 1);
|
|
xcb_present_pixmap(c, pp[i].win, pixmap, 0xdeadbeef,
|
|
0, /* valid */
|
|
None, /* update */
|
|
0, /* x_off */
|
|
0, /* y_off */
|
|
None,
|
|
None, /* wait fence */
|
|
None,
|
|
0,
|
|
0, /* target msc */
|
|
0, /* divisor */
|
|
0, /* remainder */
|
|
0, NULL);
|
|
xcb_flush(c);
|
|
|
|
list_for_each_entry(b, &pp[i].mru, link)
|
|
xshmfence_await(b->fence.addr);
|
|
|
|
xcb_free_pixmap(c, pixmap);
|
|
completed += pp[i].queued;
|
|
} else while (pp[i].queued) {
|
|
xcb_present_generic_event_t *ev;
|
|
|
|
ev = (xcb_present_generic_event_t *)
|
|
xcb_wait_for_special_event(c, pp[i].Q);
|
|
if (ev == NULL)
|
|
abort();
|
|
|
|
do {
|
|
switch (ev->evtype) {
|
|
case XCB_PRESENT_COMPLETE_NOTIFY:
|
|
completed++;
|
|
pp[i].queued--;
|
|
break;
|
|
|
|
case XCB_PRESENT_EVENT_IDLE_NOTIFY:
|
|
break;
|
|
}
|
|
free(ev);
|
|
} while ((ev = (xcb_present_generic_event_t *)xcb_poll_for_special_event(c, pp[i].Q)));
|
|
}
|
|
}
|
|
clock_gettime(CLOCK_MONOTONIC, &end);
|
|
|
|
if (update)
|
|
xcb_xfixes_destroy_region(c, update);
|
|
|
|
for (i = 0; i < sz; i++) {
|
|
for (n = 0; n < N_BACK; n++) {
|
|
if (pp[i].buffer[n].fence.xid)
|
|
dri3_fence_free(dpy, &pp[i].buffer[n].fence);
|
|
if (pp[i].buffer[n].fd != -1)
|
|
close(pp[i].buffer[n].fd);
|
|
xcb_free_pixmap(c, pp[i].buffer[n].pixmap);
|
|
}
|
|
|
|
if (pp[i].Q) {
|
|
xcb_discard_reply(c, xcb_present_select_input_checked(c, pp[i].eid, pp[i].win, 0).sequence);
|
|
XSync(dpy, True);
|
|
xcb_unregister_for_special_event(c, pp[i].Q);
|
|
}
|
|
|
|
XDestroyWindow(dpy, pp[i].win);
|
|
}
|
|
free(pp);
|
|
|
|
test_name[0] = '\0';
|
|
if (options) {
|
|
snprintf(test_name, sizeof(test_name), "(%s%s%s )",
|
|
options & NOCOPY ? " no-copy" : "",
|
|
options & DRI3 ? " dri3" : "",
|
|
options & ASYNC ? " async" : "");
|
|
}
|
|
printf("%s%s: Completed %d presents in %.1fs, %.3fus each (%.1f FPS)\n",
|
|
__func__, test_name,
|
|
completed, elapsed(&start, &end) / 1000000,
|
|
elapsed(&start, &end) / completed,
|
|
completed / (elapsed(&start, &end) / 1000000));
|
|
}
|
|
|
|
static int isqrt(int x)
|
|
{
|
|
int i;
|
|
|
|
for (i = 2; i*i < x; i++)
|
|
;
|
|
return i;
|
|
}
|
|
|
|
struct sibling {
|
|
pthread_t thread;
|
|
Display *dpy;
|
|
int x, y;
|
|
int width, height;
|
|
unsigned options;
|
|
};
|
|
|
|
static void *sibling(void *arg)
|
|
{
|
|
struct sibling *s = arg;
|
|
XSetWindowAttributes attr = { .override_redirect = 1 };
|
|
Window win = XCreateWindow(s->dpy, DefaultRootWindow(s->dpy),
|
|
s->x, s->y, s->width, s->height, 0,
|
|
DefaultDepth(s->dpy, DefaultScreen(s->dpy)),
|
|
InputOutput,
|
|
DefaultVisual(s->dpy, DefaultScreen(s->dpy)),
|
|
CWOverrideRedirect, &attr);
|
|
XMapWindow(s->dpy, win);
|
|
run(s->dpy, win, "sibling", s->options);
|
|
return NULL;
|
|
}
|
|
|
|
static void siblings(Display *dpy,
|
|
int max_width, int max_height, int ncpus, unsigned options)
|
|
{
|
|
int sq_ncpus = isqrt(ncpus);
|
|
int width = max_width / sq_ncpus;
|
|
int height = max_height/ sq_ncpus;
|
|
struct sibling s[ncpus];
|
|
int child;
|
|
|
|
if (ncpus <= 1)
|
|
return;
|
|
|
|
for (child = 0; child < ncpus; child++) {
|
|
s[child].dpy = dpy;
|
|
s[child].x = (child % sq_ncpus) * width;
|
|
s[child].y = (child / sq_ncpus) * height;
|
|
s[child].width = width;
|
|
s[child].height = height;
|
|
s[child].options = options;
|
|
pthread_create(&s[child].thread, NULL, sibling, &s[child]);
|
|
}
|
|
|
|
for (child = 0; child < ncpus; child++)
|
|
pthread_join(s[child].thread, NULL);
|
|
}
|
|
|
|
static void cousins(int max_width, int max_height, int ncpus, unsigned options)
|
|
{
|
|
int sq_ncpus = isqrt(ncpus);
|
|
int width = max_width / sq_ncpus;
|
|
int height = max_height/ sq_ncpus;
|
|
int child;
|
|
|
|
if (ncpus <= 1)
|
|
return;
|
|
|
|
for (child = 0; child < ncpus; child++) {
|
|
for (; fork() == 0; exit(0)) {
|
|
int x = (child % sq_ncpus) * width;
|
|
int y = (child / sq_ncpus) * height;
|
|
XSetWindowAttributes attr = { .override_redirect = 1 };
|
|
Display *dpy = XOpenDisplay(NULL);
|
|
Window win = XCreateWindow(dpy, DefaultRootWindow(dpy),
|
|
x, y, width, height, 0,
|
|
DefaultDepth(dpy, DefaultScreen(dpy)),
|
|
InputOutput,
|
|
DefaultVisual(dpy, DefaultScreen(dpy)),
|
|
CWOverrideRedirect, &attr);
|
|
XMapWindow(dpy, win);
|
|
run(dpy, win, "cousin", options);
|
|
}
|
|
}
|
|
|
|
while (child) {
|
|
int status = -1;
|
|
pid_t pid = wait(&status);
|
|
if (pid == -1)
|
|
continue;
|
|
child--;
|
|
}
|
|
}
|
|
|
|
static int has_present(Display *dpy)
|
|
{
|
|
xcb_connection_t *c = XGetXCBConnection(dpy);
|
|
xcb_generic_error_t *error = NULL;
|
|
void *reply;
|
|
|
|
reply = xcb_present_query_version_reply(c,
|
|
xcb_present_query_version(c,
|
|
XCB_PRESENT_MAJOR_VERSION,
|
|
XCB_PRESENT_MINOR_VERSION),
|
|
&error);
|
|
|
|
free(reply);
|
|
free(error);
|
|
if (reply == NULL) {
|
|
fprintf(stderr, "Present not supported on %s\n", DisplayString(dpy));
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int has_composite(Display *dpy)
|
|
{
|
|
int event, error;
|
|
int major, minor;
|
|
|
|
if (!XDamageQueryExtension (dpy, &event, &error))
|
|
return 0;
|
|
|
|
if (!XCompositeQueryExtension(dpy, &event, &error))
|
|
return 0;
|
|
|
|
XCompositeQueryVersion(dpy, &major, &minor);
|
|
|
|
return major > 0 || minor >= 4;
|
|
}
|
|
|
|
static int dri3_query_version(Display *dpy, int *major, int *minor)
|
|
{
|
|
xcb_connection_t *c = XGetXCBConnection(dpy);
|
|
xcb_dri3_query_version_reply_t *reply;
|
|
xcb_generic_error_t *error;
|
|
|
|
*major = *minor = -1;
|
|
|
|
reply = xcb_dri3_query_version_reply(c,
|
|
xcb_dri3_query_version(c,
|
|
XCB_DRI3_MAJOR_VERSION,
|
|
XCB_DRI3_MINOR_VERSION),
|
|
&error);
|
|
free(error);
|
|
if (reply == NULL)
|
|
return -1;
|
|
|
|
*major = reply->major_version;
|
|
*minor = reply->minor_version;
|
|
free(reply);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int has_dri3(Display *dpy)
|
|
{
|
|
const xcb_query_extension_reply_t *ext;
|
|
int major, minor;
|
|
|
|
ext = xcb_get_extension_data(XGetXCBConnection(dpy), &xcb_dri3_id);
|
|
if (ext == NULL || !ext->present)
|
|
return 0;
|
|
|
|
if (dri3_query_version(dpy, &major, &minor) < 0)
|
|
return 0;
|
|
|
|
return major >= 0;
|
|
}
|
|
|
|
static int has_xfixes(Display *dpy)
|
|
{
|
|
xcb_connection_t *c = XGetXCBConnection(dpy);
|
|
const xcb_query_extension_reply_t *ext;
|
|
void *reply;
|
|
|
|
ext = xcb_get_extension_data(c, &xcb_xfixes_id);
|
|
if (ext == NULL || !ext->present)
|
|
return 0;
|
|
|
|
reply = xcb_xfixes_query_version_reply(c,
|
|
xcb_xfixes_query_version(c,
|
|
XCB_XFIXES_MAJOR_VERSION,
|
|
XCB_XFIXES_MINOR_VERSION),
|
|
NULL);
|
|
free(reply);
|
|
|
|
return reply != NULL;
|
|
}
|
|
|
|
static inline XRRScreenResources *_XRRGetScreenResourcesCurrent(Display *dpy, Window window)
|
|
{
|
|
XRRScreenResources *res;
|
|
|
|
res = XRRGetScreenResourcesCurrent(dpy, window);
|
|
if (res == NULL)
|
|
res = XRRGetScreenResources(dpy, window);
|
|
|
|
return res;
|
|
}
|
|
|
|
static XRRModeInfo *lookup_mode(XRRScreenResources *res, int id)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < res->nmode; i++) {
|
|
if (res->modes[i].id == id)
|
|
return &res->modes[i];
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void fullscreen(Display *dpy, Window win)
|
|
{
|
|
Atom atom = XInternAtom(dpy, "_NET_WM_STATE_FULLSCREEN", False);
|
|
XChangeProperty(dpy, win,
|
|
XInternAtom(dpy, "_NET_WM_STATE", False),
|
|
XA_ATOM, 32, PropModeReplace,
|
|
(unsigned char *)&atom, 1);
|
|
}
|
|
|
|
static void loop(Display *dpy, XRRScreenResources *res, unsigned options)
|
|
{
|
|
Window root = DefaultRootWindow(dpy);
|
|
Window win;
|
|
XSetWindowAttributes attr;
|
|
int i, j;
|
|
|
|
attr.override_redirect = 1;
|
|
|
|
run(dpy, root, "off", options);
|
|
XSync(dpy, True);
|
|
|
|
for (i = 0; i < res->noutput; i++) {
|
|
XRROutputInfo *output;
|
|
XRRModeInfo *mode;
|
|
|
|
output = XRRGetOutputInfo(dpy, res, res->outputs[i]);
|
|
if (output == NULL)
|
|
continue;
|
|
|
|
mode = NULL;
|
|
if (res->nmode)
|
|
mode = lookup_mode(res, output->modes[0]);
|
|
|
|
for (j = 0; mode && j < 2*output->ncrtc; j++) {
|
|
int c = j;
|
|
if (c >= output->ncrtc)
|
|
c = 2*output->ncrtc - j - 1;
|
|
|
|
printf("[%d, %d] -- OUTPUT:%ld, CRTC:%ld: %dx%d\n",
|
|
i, c, (long)res->outputs[i], (long)output->crtcs[c],
|
|
mode->width, mode->height);
|
|
XRRSetCrtcConfig(dpy, res, output->crtcs[c], CurrentTime,
|
|
0, 0, output->modes[0], RR_Rotate_0, &res->outputs[i], 1);
|
|
|
|
run(dpy, root, "root", options);
|
|
XSync(dpy, True);
|
|
|
|
win = XCreateWindow(dpy, root,
|
|
0, 0, mode->width, mode->height, 0,
|
|
DefaultDepth(dpy, DefaultScreen(dpy)),
|
|
InputOutput,
|
|
DefaultVisual(dpy, DefaultScreen(dpy)),
|
|
CWOverrideRedirect, &attr);
|
|
fullscreen(dpy, win);
|
|
XMapWindow(dpy, win);
|
|
run(dpy, win, "fullscreen", options);
|
|
XDestroyWindow(dpy, win);
|
|
XSync(dpy, True);
|
|
|
|
win = XCreateWindow(dpy, root,
|
|
0, 0, mode->width, mode->height, 0,
|
|
DefaultDepth(dpy, DefaultScreen(dpy)),
|
|
InputOutput,
|
|
DefaultVisual(dpy, DefaultScreen(dpy)),
|
|
CWOverrideRedirect, &attr);
|
|
XMapWindow(dpy, win);
|
|
run(dpy, win, "windowed", options);
|
|
XDestroyWindow(dpy, win);
|
|
XSync(dpy, True);
|
|
|
|
if (has_composite(dpy)) {
|
|
Damage damage;
|
|
|
|
_x_error_occurred = 0;
|
|
win = XCreateWindow(dpy, root,
|
|
0, 0, mode->width, mode->height, 0,
|
|
DefaultDepth(dpy, DefaultScreen(dpy)),
|
|
InputOutput,
|
|
DefaultVisual(dpy, DefaultScreen(dpy)),
|
|
CWOverrideRedirect, &attr);
|
|
XCompositeRedirectWindow(dpy, win, CompositeRedirectManual);
|
|
damage = XDamageCreate(dpy, win, XDamageReportNonEmpty);
|
|
XMapWindow(dpy, win);
|
|
XSync(dpy, True);
|
|
if (!_x_error_occurred)
|
|
run(dpy, win, "composited", options);
|
|
XDamageDestroy(dpy, damage);
|
|
XDestroyWindow(dpy, win);
|
|
XSync(dpy, True);
|
|
}
|
|
|
|
win = XCreateWindow(dpy, root,
|
|
0, 0, mode->width/2, mode->height/2, 0,
|
|
DefaultDepth(dpy, DefaultScreen(dpy)),
|
|
InputOutput,
|
|
DefaultVisual(dpy, DefaultScreen(dpy)),
|
|
CWOverrideRedirect, &attr);
|
|
XMapWindow(dpy, win);
|
|
run(dpy, win, "half", options);
|
|
XDestroyWindow(dpy, win);
|
|
XSync(dpy, True);
|
|
|
|
perpixel(dpy, mode->width, mode->height, options);
|
|
|
|
siblings(dpy, mode->width, mode->height,
|
|
sysconf(_SC_NPROCESSORS_ONLN),
|
|
options);
|
|
|
|
cousins(mode->width, mode->height,
|
|
sysconf(_SC_NPROCESSORS_ONLN),
|
|
options);
|
|
|
|
XRRSetCrtcConfig(dpy, res, output->crtcs[c], CurrentTime,
|
|
0, 0, None, RR_Rotate_0, NULL, 0);
|
|
}
|
|
|
|
XRRFreeOutputInfo(output);
|
|
}
|
|
|
|
}
|
|
|
|
int main(void)
|
|
{
|
|
Display *dpy;
|
|
XRRScreenResources *res;
|
|
XRRCrtcInfo **original_crtc;
|
|
int i;
|
|
|
|
XInitThreads();
|
|
|
|
dpy = XOpenDisplay(NULL);
|
|
if (dpy == NULL)
|
|
return 77;
|
|
|
|
if (!has_present(dpy))
|
|
return 77;
|
|
|
|
if (DPMSQueryExtension(dpy, &i, &i))
|
|
DPMSDisable(dpy);
|
|
|
|
signal(SIGALRM, SIG_IGN);
|
|
XSetErrorHandler(_check_error_handler);
|
|
|
|
res = NULL;
|
|
if (XRRQueryVersion(dpy, &i, &i))
|
|
res = _XRRGetScreenResourcesCurrent(dpy, DefaultRootWindow(dpy));
|
|
if (res == NULL)
|
|
return 77;
|
|
|
|
original_crtc = malloc(sizeof(XRRCrtcInfo *)*res->ncrtc);
|
|
for (i = 0; i < res->ncrtc; i++)
|
|
original_crtc[i] = XRRGetCrtcInfo(dpy, res, res->crtcs[i]);
|
|
|
|
printf("noutput=%d, ncrtc=%d\n", res->noutput, res->ncrtc);
|
|
for (i = 0; i < res->ncrtc; i++)
|
|
XRRSetCrtcConfig(dpy, res, res->crtcs[i], CurrentTime,
|
|
0, 0, None, RR_Rotate_0, NULL, 0);
|
|
|
|
loop(dpy, res, 0);
|
|
loop(dpy, res, ASYNC);
|
|
if (has_xfixes(dpy))
|
|
loop(dpy, res, NOCOPY);
|
|
if (has_dri3(dpy)) {
|
|
loop(dpy, res, DRI3);
|
|
loop(dpy, res, DRI3 | ASYNC);
|
|
}
|
|
|
|
for (i = 0; i < res->ncrtc; i++)
|
|
XRRSetCrtcConfig(dpy, res, res->crtcs[i], CurrentTime,
|
|
original_crtc[i]->x,
|
|
original_crtc[i]->y,
|
|
original_crtc[i]->mode,
|
|
original_crtc[i]->rotation,
|
|
original_crtc[i]->outputs,
|
|
original_crtc[i]->noutput);
|
|
|
|
if (DPMSQueryExtension(dpy, &i, &i))
|
|
DPMSEnable(dpy);
|
|
return 0;
|
|
}
|