?? rw_x11.c
字號:
/*****************************************************************************/
/*
** SWimp_Init
**
** This routine is responsible for initializing the implementation
** specific stuff in a software rendering subsystem.
*/
int SWimp_Init( void *hInstance, void *wndProc )
{
vid_xpos = ri.Cvar_Get ("vid_xpos", "3", CVAR_ARCHIVE);
vid_ypos = ri.Cvar_Get ("vid_ypos", "22", CVAR_ARCHIVE);
// open the display
dpy = XOpenDisplay(0);
if (!dpy)
{
if (getenv("DISPLAY"))
Sys_Error("VID: Could not open display [%s]\n",
getenv("DISPLAY"));
else
Sys_Error("VID: Could not open local display\n");
}
// catch signals so i can turn on auto-repeat
{
struct sigaction sa;
sigaction(SIGINT, 0, &sa);
sa.sa_handler = TragicDeath;
sigaction(SIGINT, &sa, 0);
sigaction(SIGTERM, &sa, 0);
}
return true;
}
/*
** SWimp_InitGraphics
**
** This initializes the software refresh's implementation specific
** graphics subsystem. In the case of Windows it creates DIB or
** DDRAW surfaces.
**
** The necessary width and height parameters are grabbed from
** vid.width and vid.height.
*/
static qboolean SWimp_InitGraphics( qboolean fullscreen )
{
int pnum, i;
XVisualInfo template;
int num_visuals;
int template_mask;
Window root;
srandom(getpid());
// free resources in use
SWimp_Shutdown ();
// let the sound and input subsystems know about the new window
ri.Vid_NewWindow (vid.width, vid.height);
// XAutoRepeatOff(dpy);
// for debugging only
XSynchronize(dpy, True);
// check for command-line window size
template_mask = 0;
#if 0
// specify a visual id
if ((pnum=COM_CheckParm("-visualid")))
{
if (pnum >= com_argc-1)
Sys_Error("VID: -visualid <id#>\n");
template.visualid = Q_atoi(com_argv[pnum+1]);
template_mask = VisualIDMask;
}
// If not specified, use default visual
else
#endif
{
int screen;
screen = XDefaultScreen(dpy);
template.visualid =
XVisualIDFromVisual(XDefaultVisual(dpy, screen));
template_mask = VisualIDMask;
}
// pick a visual- warn if more than one was available
x_visinfo = XGetVisualInfo(dpy, template_mask, &template, &num_visuals);
if (num_visuals > 1)
{
printf("Found more than one visual id at depth %d:\n", template.depth);
for (i=0 ; i<num_visuals ; i++)
printf(" -visualid %d\n", (int)(x_visinfo[i].visualid));
}
else if (num_visuals == 0)
{
if (template_mask == VisualIDMask)
Sys_Error("VID: Bad visual id %d\n", template.visualid);
else
Sys_Error("VID: No visuals at depth %d\n", template.depth);
}
#if 0
printf("Using visualid %d:\n", (int)(x_visinfo->visualid));
printf(" screen %d\n", x_visinfo->screen);
printf(" red_mask 0x%x\n", (int)(x_visinfo->red_mask));
printf(" green_mask 0x%x\n", (int)(x_visinfo->green_mask));
printf(" blue_mask 0x%x\n", (int)(x_visinfo->blue_mask));
printf(" colormap_size %d\n", x_visinfo->colormap_size);
printf(" bits_per_rgb %d\n", x_visinfo->bits_per_rgb);
#endif
x_vis = x_visinfo->visual;
root = XRootWindow(dpy, x_visinfo->screen);
// setup attributes for main window
{
int attribmask = CWEventMask | CWColormap | CWBorderPixel;
XSetWindowAttributes attribs;
Colormap tmpcmap;
tmpcmap = XCreateColormap(dpy, root, x_vis, AllocNone);
attribs.event_mask = X_MASK;
attribs.border_pixel = 0;
attribs.colormap = tmpcmap;
// create the main window
win = XCreateWindow(dpy, root, (int)vid_xpos->value, (int)vid_ypos->value,
vid.width, vid.height, 0, x_visinfo->depth, InputOutput, x_vis,
attribmask, &attribs );
XStoreName(dpy, win, "Quake II");
if (x_visinfo->class != TrueColor)
XFreeColormap(dpy, tmpcmap);
}
if (x_visinfo->depth == 8)
{
// create and upload the palette
if (x_visinfo->class == PseudoColor)
{
x_cmap = XCreateColormap(dpy, win, x_vis, AllocAll);
XSetWindowColormap(dpy, win, x_cmap);
}
}
// create the GC
{
XGCValues xgcvalues;
int valuemask = GCGraphicsExposures;
xgcvalues.graphics_exposures = False;
x_gc = XCreateGC(dpy, win, valuemask, &xgcvalues );
}
XMapWindow(dpy, win);
XMoveWindow(dpy, win, (int)vid_xpos->value, (int)vid_ypos->value);
// wait for first exposure event
{
XEvent event;
exposureflag = false;
do
{
HandleEvents();
} while (!exposureflag);
}
// now safe to draw
// even if MITSHM is available, make sure it's a local connection
if (XShmQueryExtension(dpy))
{
char *displayname;
doShm = true;
displayname = (char *) getenv("DISPLAY");
if (displayname)
{
char *d = displayname;
while (*d && (*d != ':')) d++;
if (*d) *d = 0;
if (!(!strcasecmp(displayname, "unix") || !*displayname))
doShm = false;
}
}
if (doShm)
{
x_shmeventtype = XShmGetEventBase(dpy) + ShmCompletion;
ResetSharedFrameBuffers();
}
else
ResetFrameBuffer();
current_framebuffer = 0;
vid.rowbytes = x_framebuffer[0]->bytes_per_line;
vid.buffer = x_framebuffer[0]->data;
// XSynchronize(dpy, False);
X11_active = true;
return true;
}
/*
** SWimp_EndFrame
**
** This does an implementation specific copy from the backbuffer to the
** front buffer. In the Win32 case it uses BitBlt or BltFast depending
** on whether we're using DIB sections/GDI or DDRAW.
*/
void SWimp_EndFrame (void)
{
// if the window changes dimension, skip this frame
#if 0
if (config_notify)
{
fprintf(stderr, "config notify\n");
config_notify = 0;
vid.width = config_notify_width & ~7;
vid.height = config_notify_height;
if (doShm)
ResetSharedFrameBuffers();
else
ResetFrameBuffer();
vid.rowbytes = x_framebuffer[0]->bytes_per_line;
vid.buffer = x_framebuffer[current_framebuffer]->data;
vid.recalc_refdef = 1; // force a surface cache flush
Con_CheckResize();
Con_Clear_f();
return;
}
#endif
if (doShm)
{
if (x_visinfo->depth == 16)
st2_fixup( x_framebuffer[current_framebuffer], 0, 0, vid.width, vid.height);
else if (x_visinfo->depth == 24)
st3_fixup( x_framebuffer[current_framebuffer], 0, 0, vid.width, vid.height);
if (!XShmPutImage(dpy, win, x_gc,
x_framebuffer[current_framebuffer], 0, 0, 0, 0, vid.width, vid.height, True))
Sys_Error("VID_Update: XShmPutImage failed\n");
oktodraw = false;
while (!oktodraw)
HandleEvents();
current_framebuffer = !current_framebuffer;
vid.buffer = x_framebuffer[current_framebuffer]->data;
XSync(dpy, False);
}
else
{
if (x_visinfo->depth == 16)
st2_fixup( x_framebuffer[current_framebuffer], 0, 0, vid.width, vid.height);
else if (x_visinfo->depth == 24)
st3_fixup( x_framebuffer[current_framebuffer], 0, 0, vid.width, vid.height);
XPutImage(dpy, win, x_gc, x_framebuffer[0], 0, 0, 0, 0, vid.width, vid.height);
XSync(dpy, False);
}
}
/*
** SWimp_SetMode
*/
rserr_t SWimp_SetMode( int *pwidth, int *pheight, int mode, qboolean fullscreen )
{
rserr_t retval = rserr_ok;
ri.Con_Printf (PRINT_ALL, "setting mode %d:", mode );
if ( !ri.Vid_GetModeInfo( pwidth, pheight, mode ) )
{
ri.Con_Printf( PRINT_ALL, " invalid mode\n" );
return rserr_invalid_mode;
}
ri.Con_Printf( PRINT_ALL, " %d %d\n", *pwidth, *pheight);
if ( !SWimp_InitGraphics( false ) ) {
// failed to set a valid mode in windowed mode
return rserr_invalid_mode;
}
R_GammaCorrectAndSetPalette( ( const unsigned char * ) d_8to24table );
return retval;
}
/*
** SWimp_SetPalette
**
** System specific palette setting routine. A NULL palette means
** to use the existing palette. The palette is expected to be in
** a padded 4-byte xRGB format.
*/
void SWimp_SetPalette( const unsigned char *palette )
{
int i;
XColor colors[256];
if (!X11_active)
return;
if ( !palette )
palette = ( const unsigned char * ) sw_state.currentpalette;
for(i=0;i<256;i++) {
st2d_8to16table[i]= xlib_rgb16(palette[i*4], palette[i*4+1],palette[i*4+2]);
st2d_8to24table[i]= xlib_rgb24(palette[i*4], palette[i*4+1],palette[i*4+2]);
}
if (x_visinfo->class == PseudoColor && x_visinfo->depth == 8)
{
for (i=0 ; i<256 ; i++)
{
colors[i].pixel = i;
colors[i].flags = DoRed|DoGreen|DoBlue;
colors[i].red = palette[i*4] * 257;
colors[i].green = palette[i*4+1] * 257;
colors[i].blue = palette[i*4+2] * 257;
}
XStoreColors(dpy, x_cmap, colors, 256);
}
}
/*
** SWimp_Shutdown
**
** System specific graphics subsystem shutdown routine. Destroys
** DIBs or DDRAW surfaces as appropriate.
*/
void SWimp_Shutdown( void )
{
int i;
if (!X11_active)
return;
if (doShm) {
for (i = 0; i < 2; i++)
if (x_framebuffer[i]) {
XShmDetach(dpy, &x_shminfo[i]);
free(x_framebuffer[i]);
shmdt(x_shminfo[i].shmaddr);
x_framebuffer[i] = NULL;
}
} else if (x_framebuffer[0]) {
free(x_framebuffer[0]->data);
free(x_framebuffer[0]);
x_framebuffer[0] = NULL;
}
XDestroyWindow( dpy, win );
// XAutoRepeatOn(dpy);
// XCloseDisplay(dpy);
X11_active = false;
}
/*
** SWimp_AppActivate
*/
void SWimp_AppActivate( qboolean active )
{
}
//===============================================================================
/*
================
Sys_MakeCodeWriteable
================
*/
void Sys_MakeCodeWriteable (unsigned long startaddr, unsigned long length)
{
int r;
unsigned long addr;
int psize = getpagesize();
addr = (startaddr & ~(psize-1)) - psize;
// fprintf(stderr, "writable code %lx(%lx)-%lx, length=%lx\n", startaddr,
// addr, startaddr+length, length);
r = mprotect((char*)addr, length + startaddr - addr + psize, 7);
if (r < 0)
Sys_Error("Protection change failed\n");
}
/*****************************************************************************/
/* KEYBOARD */
/*****************************************************************************/
Key_Event_fp_t Key_Event_fp;
void KBD_Init(Key_Event_fp_t fp)
{
Key_Event_fp = fp;
}
void KBD_Update(void)
{
// get events from x server
HandleEvents();
}
void KBD_Close(void)
{
}
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -