quakeforge/libs/video/targets/vid_x11_sw.c
Bill Currie 0c437492b4 [renderer] Move to using dynamic frame buffers
For now, OpenGL and Vulkan renderers are broken as I focused on getting
the software renderer working (which was quite tricky to get right).

This fixes a couple of issues: the segfault when warping the screen (due
to the scene rendering move invalidating the warp buffer), and warp
always having 320x200 resolution. There's still the problem of the
effect being too subtle at high resolution, but that's just a matter of
updating the tables and tweaking the code in D_WarpScreen.

Another issue is the Draw functions should probably write directly to
the main frame buffer or even one passed in as a parameter. This would
remove the need for binding the main buffer at the beginning and end of
the frame.
2022-03-24 12:56:29 +09:00

710 lines
17 KiB
C

/*
vid_x11_sw.c
Software X11 video driver (8/32 bit)
Copyright (C) 1996-1997 Id Software, Inc.
Copyright (C) 1999-2000 contributors of the QuakeForge project
Copyright (C) 2000 Marcus Sundberg [mackan@stacken.kth.se]
Copyright (C) 1999,2000 contributors of the QuakeForge project
Please see the file "AUTHORS" for a list of contributors
This program 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 2
of the License, or (at your option) any later version.
This program 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 this program; if not, write to:
Free Software Foundation, Inc.
59 Temple Place - Suite 330
Boston, MA 02111-1307, USA
*/
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
#ifdef HAVE_STRING_H
# include <string.h>
#endif
#ifdef HAVE_STRINGS_H
# include <strings.h>
#endif
#ifdef HAVE_UNISTD_H
# include <unistd.h>
#endif
#include <ctype.h>
#include <errno.h>
#include <limits.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/time.h>
#include <sys/types.h>
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/extensions/XShm.h>
#include "QF/console.h"
#include "QF/cvar.h"
#include "QF/qargs.h"
#include "QF/sys.h"
#include "QF/vid.h"
#include "QF/ui/view.h"
#include "context_x11.h"
#include "r_internal.h"
#include "vid_internal.h"
#include "vid_sw.h"
int XShmGetEventBase (Display *x); // for broken X11 headers
static GC x_gc;
static qboolean doShm;
static XShmSegmentInfo x_shminfo[2];
static int current_framebuffer;
static XImage *x_framebuffer[2] = { 0, 0 };
typedef unsigned char PIXEL8;
typedef unsigned short PIXEL16;
typedef unsigned int PIXEL24;
static PIXEL16 st2d_8to16table[256];
static PIXEL24 st2d_8to24table[256];
static byte current_palette[768];
static int shiftmask_fl = 0;
static long r_shift, g_shift, b_shift;
static unsigned long r_mask, g_mask, b_mask;
static void
shiftmask_init (void)
{
unsigned long long x;
r_mask = x_vis->red_mask;
g_mask = x_vis->green_mask;
b_mask = x_vis->blue_mask;
for (r_shift = -8, x = 1; x < r_mask; x <<= 1)
r_shift++;
for (g_shift = -8, x = 1; x < g_mask; x <<= 1)
g_shift++;
for (b_shift = -8, x = 1; x < b_mask; x <<= 1)
b_shift++;
shiftmask_fl = 1;
}
static PIXEL16
xlib_rgb16 (int r, int g, int b)
{
PIXEL16 p = 0;
if (!shiftmask_fl)
shiftmask_init ();
if (r_shift > 0) {
p = (r << (r_shift)) & r_mask;
} else {
if (r_shift < 0) {
p = (r >> (-r_shift)) & r_mask;
} else {
p |= (r & r_mask);
}
}
if (g_shift > 0) {
p |= (g << (g_shift)) & g_mask;
} else {
if (g_shift < 0) {
p |= (g >> (-g_shift)) & g_mask;
} else {
p |= (g & g_mask);
}
}
if (b_shift > 0) {
p |= (b << (b_shift)) & b_mask;
} else {
if (b_shift < 0) {
p |= (b >> (-b_shift)) & b_mask;
} else {
p |= (b & b_mask);
}
}
return p;
}
static PIXEL24
xlib_rgb24 (int r, int g, int b)
{
PIXEL24 p = 0;
if (!shiftmask_fl)
shiftmask_init ();
if (r_shift > 0) {
p = (r << (r_shift)) & r_mask;
} else {
if (r_shift < 0) {
p = (r >> (-r_shift)) & r_mask;
} else {
p |= (r & r_mask);
}
}
if (g_shift > 0) {
p |= (g << (g_shift)) & g_mask;
} else {
if (g_shift < 0) {
p |= (g >> (-g_shift)) & g_mask;
} else {
p |= (g & g_mask);
}
}
if (b_shift > 0) {
p |= (b << (b_shift)) & b_mask;
} else {
if (b_shift < 0) {
p |= (b >> (-b_shift)) & b_mask;
} else {
p |= (b & b_mask);
}
}
return p;
}
static void
x11_set_palette (sw_ctx_t *ctx, const byte *palette)
{
int i;
XColor colors[256];
for (i = 0; i < 256; i++) {
st2d_8to16table[i] = xlib_rgb16 (palette[i * 3], palette[i * 3 + 1],
palette[i * 3 + 2]);
st2d_8to24table[i] = xlib_rgb24 (palette[i * 3], palette[i * 3 + 1],
palette[i * 3 + 2]);
}
if (x_visinfo->class == PseudoColor && x_visinfo->depth == 8) {
if (palette != current_palette) {
memcpy (current_palette, palette, 768);
}
for (i = 0; i < 256; i++) {
colors[i].pixel = i;
colors[i].flags = DoRed | DoGreen | DoBlue;
colors[i].red = palette[(i * 3)] << 8;
colors[i].green = palette[(i * 3) + 1] << 8;
colors[i].blue = palette[(i * 3) + 2] << 8;
}
XStoreColors (x_disp, x_cmap, colors, 256);
}
}
static void
st2_fixup (sw_ctx_t *ctx, XImage *framebuf, int x, int y, int width, int height)
{
int xi, yi;
unsigned char *src;
PIXEL16 *dest;
sw_framebuffer_t *fb = ctx->framebuffer->buffer;
if (x < 0 || y < 0)
return;
for (yi = y; yi < (y + height); yi++) {
src = &fb->color[yi * fb->rowbytes];
dest = (PIXEL16 *) &framebuf->data[yi * framebuf->bytes_per_line];
for (xi = x; xi < x + width; xi++) {
dest[xi] = st2d_8to16table[src[xi]];
}
}
}
static void
st3_fixup (sw_ctx_t *ctx, XImage *framebuf, int x, int y, int width, int height)
{
int yi;
unsigned char *src;
PIXEL24 *dest;
sw_framebuffer_t *fb = ctx->framebuffer->buffer;
register int count, n;
if (x < 0 || y < 0)
return;
for (yi = y; yi < (y + height); yi++) {
src = &fb->color[yi * fb->rowbytes + x];
dest = (PIXEL24 *) &framebuf->data[yi * framebuf->bytes_per_line + x];
// Duff's Device
count = width;
n = (count + 7) / 8;
switch (count % 8) {
case 0:
do {
*dest++ = st2d_8to24table[*src++];
case 7:
*dest++ = st2d_8to24table[*src++];
case 6:
*dest++ = st2d_8to24table[*src++];
case 5:
*dest++ = st2d_8to24table[*src++];
case 4:
*dest++ = st2d_8to24table[*src++];
case 3:
*dest++ = st2d_8to24table[*src++];
case 2:
*dest++ = st2d_8to24table[*src++];
case 1:
*dest++ = st2d_8to24table[*src++];
} while (--n > 0);
}
}
}
static void
x11_put_image (vrect_t *rect)
{
if (doShm) {
if (!XShmPutImage (x_disp, x_win, x_gc,
x_framebuffer[current_framebuffer],
rect->x, rect->y, rect->x, rect->y,
rect->width, rect->height, True)) {
Sys_Error ("VID_Update: XShmPutImage failed");
}
oktodraw = false;
while (!oktodraw)
X11_ProcessEvent ();
current_framebuffer = !current_framebuffer;
} else {
if (XPutImage (x_disp, x_win, x_gc, x_framebuffer[0],
rect->x, rect->y, rect->x, rect->y,
rect->width, rect->height)) {
Sys_Error ("VID_Update: XPutImage failed");
}
}
}
/*
Flush the given rectangles from the view buffer to the screen.
*/
static void
x11_sw8_8_update (sw_ctx_t *ctx, vrect_t *rects)
{
vrect_t urect = *rects;
while (rects->next) {
rects = rects->next;
int minx = min (VRect_MinX (&urect), VRect_MinX (rects));
int miny = min (VRect_MinY (&urect), VRect_MinY (rects));
int maxx = max (VRect_MaxX (&urect), VRect_MaxX (rects));
int maxy = max (VRect_MaxY (&urect), VRect_MaxY (rects));
urect.x = minx;
urect.y = miny;
urect.width = maxx - minx;
urect.height = maxy - miny;
}
x11_put_image (&urect);
XSync (x_disp, False);
r_data->scr_fullupdate = 0;
sw_framebuffer_t *fb = ctx->framebuffer->buffer;
fb->color = (byte *) x_framebuffer[current_framebuffer]->data;
}
static void
x11_sw8_16_update (sw_ctx_t *ctx, vrect_t *rects)
{
vrect_t urect = *rects;
st2_fixup (ctx, x_framebuffer[current_framebuffer],
rects->x, rects->y, rects->width, rects->height);
while (rects->next) {
rects = rects->next;
st2_fixup (ctx, x_framebuffer[current_framebuffer],
rects->x, rects->y, rects->width, rects->height);
int minx = min (VRect_MinX (&urect), VRect_MinX (rects));
int miny = min (VRect_MinY (&urect), VRect_MinY (rects));
int maxx = max (VRect_MaxX (&urect), VRect_MaxX (rects));
int maxy = max (VRect_MaxY (&urect), VRect_MaxY (rects));
urect.x = minx;
urect.y = miny;
urect.width = maxx - minx;
urect.height = maxy - miny;
}
x11_put_image (&urect);
XSync (x_disp, False);
r_data->scr_fullupdate = 0;
}
static void
x11_sw8_24_update (sw_ctx_t *ctx, vrect_t *rects)
{
vrect_t urect = *rects;
st3_fixup (ctx, x_framebuffer[current_framebuffer],
rects->x, rects->y, rects->width, rects->height);
while (rects->next) {
rects = rects->next;
st3_fixup (ctx, x_framebuffer[current_framebuffer],
rects->x, rects->y, rects->width, rects->height);
int minx = min (VRect_MinX (&urect), VRect_MinX (rects));
int miny = min (VRect_MinY (&urect), VRect_MinY (rects));
int maxx = max (VRect_MaxX (&urect), VRect_MaxX (rects));
int maxy = max (VRect_MaxY (&urect), VRect_MaxY (rects));
urect.x = minx;
urect.y = miny;
urect.width = maxx - minx;
urect.height = maxy - miny;
}
x11_put_image (&urect);
XSync (x_disp, False);
r_data->scr_fullupdate = 0;
}
#if 0
static void
x11_sw16_16_update (sw_ctx_t *ctx, vrect_t *rects)
{
uint16_t *buffer = (uint16_t *) viddef.buffer;
XImage *framebuf = x_framebuffer[current_framebuffer];
int rowbytes = framebuf->bytes_per_line;
while (rects) {
int x = rects->x;
int y = rects->y;
int width = rects->width;
int height = rects->height;
for (int yi = y; yi < (y + height); yi++) {
uint16_t *src = buffer + yi * viddef.width + x;
PIXEL16 *dest = (PIXEL16 *) (framebuf->data + yi * rowbytes) + x;
for (int count = width; count-- > 0; ) {
*dest++ = *src++;
}
}
x11_put_image (rects);
rects = rects->next;
}
XSync (x_disp, False);
r_data->scr_fullupdate = 0;
}
static void
x11_sw16_32_update (sw_ctx_t *ctx, vrect_t *rects)
{
uint16_t *buffer = (uint16_t *) viddef.buffer;
XImage *framebuf = x_framebuffer[current_framebuffer];
int rowbytes = framebuf->bytes_per_line;
while (rects) {
int x = rects->x;
int y = rects->y;
int width = rects->width;
int height = rects->height;
for (int yi = y; yi < (y + height); yi++) {
uint16_t *src = buffer + yi * viddef.width + x;
PIXEL24 *dest = (PIXEL24 *) (framebuf->data + yi * rowbytes) + x;
for (int count = width; count-- > 0; ) {
uint32_t c = *src++;
*dest++ = 0xff000000
| ((c & 0xf800) << 8)
| ((c & 0x07e0) << 5)
| ((c & 0x001f) << 3);
}
}
x11_put_image (rects);
rects = rects->next;
}
XSync (x_disp, False);
r_data->scr_fullupdate = 0;
}
static void
x11_sw32_update (sw_ctx_t *ctx, vrect_t *rects)
{
uint32_t *buffer = (uint32_t *) viddef.buffer;
XImage *framebuf = x_framebuffer[current_framebuffer];
int rowbytes = framebuf->bytes_per_line;
while (rects) {
int x = rects->x;
int y = rects->y;
int width = rects->width;
int height = rects->height;
for (int yi = y; yi < (y + height); yi++) {
uint32_t *src = buffer + yi * viddef.width + x;
PIXEL24 *dest = (PIXEL24 *) (framebuf->data + yi * rowbytes) + x;
for (int count = width; count-- > 0; ) {
*dest++ = *src++;
}
}
x11_put_image (rects);
rects = rects->next;
}
XSync (x_disp, False);
r_data->scr_fullupdate = 0;
}
#endif
static void
x11_choose_visual (sw_ctx_t *ctx)
{
int pnum, i;
XVisualInfo template;
int num_visuals;
int template_mask;
// specify a visual id
if ((pnum = COM_CheckParm ("-visualid"))) {
if (pnum >= com_argc - 1)
Sys_Error ("VID: -visualid <id#>");
template.visualid = atoi (com_argv[pnum + 1]);
template_mask = VisualIDMask;
} else { // If not specified, use default
// visual
template.visualid =
XVisualIDFromVisual (XDefaultVisual (x_disp, x_screen));
template_mask = VisualIDMask;
}
// pick a visual -- warn if more than one was available
x_visinfo = XGetVisualInfo (x_disp, template_mask, &template,
&num_visuals);
if (x_visinfo->depth == 8 && x_visinfo->class == PseudoColor)
x_cmap = XCreateColormap (x_disp, x_win, x_vis, AllocAll);
x_vis = x_visinfo->visual;
ctx->update = x11_sw8_8_update;
switch (x_visinfo->depth) {
case 8:
ctx->update = x11_sw8_8_update;
break;
case 16:
ctx->update = x11_sw8_16_update;
break;
case 24:
ctx->update = x11_sw8_24_update;
break;
}
if (num_visuals > 1) {
Sys_MaskPrintf (SYS_vid,
"Found more than one visual id at depth %d:\n",
template.depth);
for (i = 0; i < num_visuals; i++)
Sys_MaskPrintf (SYS_vid, " -visualid %d\n",
(int) x_visinfo[i].visualid);
} else {
if (num_visuals == 0) {
if (template_mask == VisualIDMask) {
Sys_Error ("VID: Bad visual ID %ld", template.visualid);
} else {
Sys_Error ("VID: No visuals at depth %d", template.depth);
}
}
}
Sys_MaskPrintf (SYS_vid, "Using visualid %d:\n",
(int) x_visinfo->visualid);
Sys_MaskPrintf (SYS_vid, " class %d\n", x_visinfo->class);
Sys_MaskPrintf (SYS_vid, " screen %d\n", x_visinfo->screen);
Sys_MaskPrintf (SYS_vid, " depth %d\n", x_visinfo->depth);
Sys_MaskPrintf (SYS_vid, " red_mask 0x%x\n",
(int) x_visinfo->red_mask);
Sys_MaskPrintf (SYS_vid, " green_mask 0x%x\n",
(int) x_visinfo->green_mask);
Sys_MaskPrintf (SYS_vid, " blue_mask 0x%x\n",
(int) x_visinfo->blue_mask);
Sys_MaskPrintf (SYS_vid, " colormap_size %d\n",
x_visinfo->colormap_size);
Sys_MaskPrintf (SYS_vid, " bits_per_rgb %d\n",
x_visinfo->bits_per_rgb);
}
static void
ResetFrameBuffer (void)
{
int mem, pwidth;
char *buf;
if (x_framebuffer[0]) {
XDestroyImage (x_framebuffer[0]);
}
pwidth = x_visinfo->depth / 8;
if (pwidth == 3)
pwidth = 4;
mem = ((viddef.width * pwidth + 7) & ~7) * viddef.height;
buf = malloc (mem);
SYS_CHECKMEM (buf);
// allocate new screen buffer
x_framebuffer[0] = XCreateImage (x_disp, x_vis, x_visinfo->depth,
ZPixmap, 0, buf, viddef.width,
viddef.height, 32, 0);
if (!x_framebuffer[0]) {
Sys_Error ("VID: XCreateImage failed");
}
}
static void
ResetSharedFrameBuffers (void)
{
int size;
int key;
int minsize = getpagesize ();
int frm;
for (frm = 0; frm < 2; frm++) {
// free up old frame buffer memory
if (x_framebuffer[frm]) {
XShmDetach (x_disp, &x_shminfo[frm]);
free (x_framebuffer[frm]);
shmdt (x_shminfo[frm].shmaddr);
}
// create the image
x_framebuffer[frm] = XShmCreateImage (x_disp, x_vis, x_visinfo->depth,
ZPixmap, 0, &x_shminfo[frm],
viddef.width, viddef.height);
// grab shared memory
size = x_framebuffer[frm]->bytes_per_line * x_framebuffer[frm]->height;
if (size < minsize)
Sys_Error ("VID: Window must use at least %d bytes", minsize);
key = random ();
x_shminfo[frm].shmid = shmget ((key_t) key, size, IPC_CREAT | 0777);
if (x_shminfo[frm].shmid == -1)
Sys_Error ("VID: Could not get any shared memory (%s)",
strerror (errno));
// attach to the shared memory segment
x_shminfo[frm].shmaddr = (void *) shmat (x_shminfo[frm].shmid, 0, 0);
Sys_MaskPrintf (SYS_vid, "VID: shared memory id=%d, addr=0x%lx\n",
x_shminfo[frm].shmid, (long) x_shminfo[frm].shmaddr);
x_framebuffer[frm]->data = x_shminfo[frm].shmaddr;
// get the X server to attach to it
if (!XShmAttach (x_disp, &x_shminfo[frm]))
Sys_Error ("VID: XShmAttach() failed");
XSync (x_disp, 0);
shmctl (x_shminfo[frm].shmid, IPC_RMID, 0);
}
}
static sw_framebuffer_t swfb;
static framebuffer_t fb = { .buffer = &swfb };
static void
x11_init_buffers (void *data)
{
sw_ctx_t *ctx = data;
ctx->framebuffer = &fb;
if (doShm)
ResetSharedFrameBuffers ();
else
ResetFrameBuffer ();
current_framebuffer = 0;
fb.width = viddef.width;
fb.height = viddef.height;
if (x_visinfo->depth != 8) {
if (swfb.color)
free (swfb.color);
swfb.rowbytes = viddef.width;
swfb.color = calloc (swfb.rowbytes, viddef.height);
if (!swfb.color)
Sys_Error ("Not enough memory for video mode");
} else {
swfb.rowbytes = x_framebuffer[current_framebuffer]->bytes_per_line;
swfb.color = (byte *) x_framebuffer[current_framebuffer]->data;
}
}
static void
x11_create_context (sw_ctx_t *ctx)
{
// create the GC
{
XGCValues xgcvalues;
int valuemask = GCGraphicsExposures;
xgcvalues.graphics_exposures = False;
x_gc = XCreateGC (x_disp, x_win, valuemask, &xgcvalues);
}
// even if MITSHM is available, make sure it's a local connection
if (XShmQueryExtension (x_disp)) {
char *displayname;
char *d;
doShm = true;
if ((displayname = XDisplayName (NULL))) {
if ((d = strchr (displayname, ':')))
*d = '\0';
if (!(!strcasecmp (displayname, "unix") || !*displayname))
doShm = false;
}
}
if (doShm) {
x_shmeventtype = XShmGetEventBase (x_disp) + ShmCompletion;
}
// FIXME this really shouldn't be here (ideally, scale console in sw)
// No console scaling in the sw renderer
viddef.conview->xlen = viddef.width;
viddef.conview->ylen = viddef.height;
Con_CheckResize ();
viddef.vid_internal->init_buffers = x11_init_buffers;
// XSynchronize (x_disp, False);
// X11_AddEvent (x_shmeventtype, event_shm);
}
sw_ctx_t *
X11_SW_Context (void)
{
sw_ctx_t *ctx = calloc (1, sizeof (sw_ctx_t));
ctx->set_palette = x11_set_palette;
ctx->choose_visual = x11_choose_visual;
ctx->create_context = x11_create_context;
ctx->update = x11_sw8_8_update;
return ctx;
}
void
X11_SW_Init_Cvars (void)
{
}