📄 sdl_wsconsvideo.c
字号:
/* SDL - Simple DirectMedia Layer Copyright (C) 1997-2006 Sam Lantinga This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. This library 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA Sam Lantinga slouken@libsdl.org*/#include "SDL_config.h"#include <sys/time.h>#include <sys/mman.h>#include <sys/ioctl.h>#include <dev/wscons/wsdisplay_usl_io.h>#include <fcntl.h>#include <unistd.h>#include <errno.h>#include "SDL_video.h"#include "SDL_mouse.h"#include "../SDL_sysvideo.h"#include "../SDL_pixels_c.h"#include "../../events/SDL_events_c.h"#include "SDL_wsconsvideo.h"#include "SDL_wsconsevents_c.h"#include "SDL_wsconsmouse_c.h"#define WSCONSVID_DRIVER_NAME "wscons"enum { WSCONS_ROTATE_NONE = 0, WSCONS_ROTATE_CCW = 90, WSCONS_ROTATE_UD = 180, WSCONS_ROTATE_CW = 270};#define min(a,b) ((a)<(b)?(a):(b))/* Initialization/Query functions */static int WSCONS_VideoInit(_THIS, SDL_PixelFormat *vformat);static SDL_Rect **WSCONS_ListModes(_THIS, SDL_PixelFormat *format, Uint32 flags);static SDL_Surface *WSCONS_SetVideoMode(_THIS, SDL_Surface *current, int width, int height, int bpp, Uint32 flags);static int WSCONS_SetColors(_THIS, int firstcolor, int ncolors, SDL_Color *colors);static void WSCONS_VideoQuit(_THIS);/* Hardware surface functions */static int WSCONS_AllocHWSurface(_THIS, SDL_Surface *surface);static int WSCONS_LockHWSurface(_THIS, SDL_Surface *surface);static void WSCONS_UnlockHWSurface(_THIS, SDL_Surface *surface);static void WSCONS_FreeHWSurface(_THIS, SDL_Surface *surface);/* etc. */static WSCONS_bitBlit WSCONS_blit16;static WSCONS_bitBlit WSCONS_blit16blocked;static void WSCONS_UpdateRects(_THIS, int numrects, SDL_Rect *rects);void WSCONS_ReportError(char *fmt, ...){ char message[200]; va_list vaArgs; message[199] = '\0'; va_start(vaArgs, fmt); vsnprintf(message, 199, fmt, vaArgs); va_end(vaArgs); SDL_SetError(message); fprintf(stderr, "WSCONS error: %s\n", message);}/* WSCONS driver bootstrap functions */static int WSCONS_Available(void){ return 1;}static void WSCONS_DeleteDevice(SDL_VideoDevice *device){ SDL_free(device->hidden); SDL_free(device);}static SDL_VideoDevice *WSCONS_CreateDevice(int devindex){ SDL_VideoDevice *device; /* Initialize all variables that we clean on shutdown */ device = (SDL_VideoDevice *)SDL_malloc(sizeof(SDL_VideoDevice)); if (device == NULL) { SDL_OutOfMemory(); return 0; } SDL_memset(device, 0, (sizeof *device)); device->hidden = (struct SDL_PrivateVideoData *)SDL_malloc((sizeof *device->hidden)); if (device->hidden == NULL) { SDL_OutOfMemory(); SDL_free(device); return(0); } SDL_memset(device->hidden, 0, (sizeof *device->hidden)); device->hidden->fd = -1; /* Set the function pointers */ device->VideoInit = WSCONS_VideoInit; device->ListModes = WSCONS_ListModes; device->SetVideoMode = WSCONS_SetVideoMode; device->SetColors = WSCONS_SetColors; device->UpdateRects = WSCONS_UpdateRects; device->VideoQuit = WSCONS_VideoQuit; device->AllocHWSurface = WSCONS_AllocHWSurface; device->LockHWSurface = WSCONS_LockHWSurface; device->UnlockHWSurface = WSCONS_UnlockHWSurface; device->FreeHWSurface = WSCONS_FreeHWSurface; device->InitOSKeymap = WSCONS_InitOSKeymap; device->PumpEvents = WSCONS_PumpEvents; device->free = WSCONS_DeleteDevice; return device;}VideoBootStrap WSCONS_bootstrap = { WSCONSVID_DRIVER_NAME, "SDL wscons video driver", WSCONS_Available, WSCONS_CreateDevice};#define WSCONSDEV_FORMAT "/dev/ttyC%01x"int WSCONS_VideoInit(_THIS, SDL_PixelFormat *vformat){ char devnamebuf[30]; char *devname; char *rotation; int wstype; int wsmode = WSDISPLAYIO_MODE_DUMBFB; size_t len, mapsize; int pagemask; int width, height; devname = SDL_getenv("SDL_WSCONSDEV"); if (devname == NULL) { int activeVT; if (ioctl(STDIN_FILENO, VT_GETACTIVE, &activeVT) == -1) { WSCONS_ReportError("Unable to determine active terminal: %s", strerror(errno)); return -1; } SDL_snprintf(devnamebuf, sizeof(devnamebuf), WSCONSDEV_FORMAT, activeVT - 1); devname = devnamebuf; } private->fd = open(devname, O_RDWR | O_NONBLOCK, 0); if (private->fd == -1) { WSCONS_ReportError("open %s: %s", devname, strerror(errno)); return -1; } if (ioctl(private->fd, WSDISPLAYIO_GINFO, &private->info) == -1) { WSCONS_ReportError("ioctl WSDISPLAY_GINFO: %s", strerror(errno)); return -1; } if (ioctl(private->fd, WSDISPLAYIO_GTYPE, &wstype) == -1) { WSCONS_ReportError("ioctl WSDISPLAY_GTYPE: %s", strerror(errno)); return -1; } if (ioctl(private->fd, WSDISPLAYIO_LINEBYTES, &private->physlinebytes) == -1) { WSCONS_ReportError("ioctl WSDISPLAYIO_LINEBYTES: %s", strerror(errno)); return -1; } if (private->info.depth > 8) { if (wstype == WSDISPLAY_TYPE_SUN24 || wstype == WSDISPLAY_TYPE_SUNCG12 || wstype == WSDISPLAY_TYPE_SUNCG14 || wstype == WSDISPLAY_TYPE_SUNTCX || wstype == WSDISPLAY_TYPE_SUNFFB) { private->redMask = 0x0000ff; private->greenMask = 0x00ff00; private->blueMask = 0xff0000;#ifdef WSDISPLAY_TYPE_PXALCD } else if (wstype == WSDISPLAY_TYPE_PXALCD) { private->redMask = 0x1f << 11; private->greenMask = 0x3f << 5; private->blueMask = 0x1f;#endif } else { WSCONS_ReportError("Unknown video hardware"); return -1; } } else { WSCONS_ReportError("Displays with 8 bpp or less are not supported"); return -1; } private->rotate = WSCONS_ROTATE_NONE; rotation = SDL_getenv("SDL_VIDEO_WSCONS_ROTATION"); if (rotation != NULL) { if (SDL_strlen(rotation) == 0) { private->shadowFB = 0; private->rotate = WSCONS_ROTATE_NONE; printf("Not rotating, no shadow\n"); } else if (!SDL_strcmp(rotation, "NONE")) { private->shadowFB = 1; private->rotate = WSCONS_ROTATE_NONE; printf("Not rotating, but still using shadow\n"); } else if (!SDL_strcmp(rotation, "CW")) { private->shadowFB = 1; private->rotate = WSCONS_ROTATE_CW; printf("Rotating screen clockwise\n"); } else if (!SDL_strcmp(rotation, "CCW")) { private->shadowFB = 1; private->rotate = WSCONS_ROTATE_CCW; printf("Rotating screen counter clockwise\n"); } else if (!SDL_strcmp(rotation, "UD")) { private->shadowFB = 1; private->rotate = WSCONS_ROTATE_UD; printf("Rotating screen upside down\n"); } else { WSCONS_ReportError("\"%s\" is not a valid value for " "SDL_VIDEO_WSCONS_ROTATION", rotation); return -1; } } switch (private->info.depth) { case 1: case 4: case 8: len = private->physlinebytes * private->info.height; break; case 16: if (private->physlinebytes == private->info.width) { len = private->info.width * private->info.height * sizeof(short); } else { len = private->physlinebytes * private->info.height; } if (private->rotate == WSCONS_ROTATE_NONE || private->rotate == WSCONS_ROTATE_UD) { private->blitFunc = WSCONS_blit16; } else { private->blitFunc = WSCONS_blit16blocked; } break; case 32: if (private->physlinebytes == private->info.width) { len = private->info.width * private->info.height * sizeof(int); } else { len = private->physlinebytes * private->info.height; } break; default: WSCONS_ReportError("unsupported depth %d", private->info.depth); return -1; } if (private->shadowFB && private->blitFunc == NULL) { WSCONS_ReportError("Using software buffer, but no blitter function is " "available for this %d bpp.", private->info.depth); return -1; } if (ioctl(private->fd, WSDISPLAYIO_SMODE, &wsmode) == -1) { WSCONS_ReportError("ioctl SMODE"); return -1; } pagemask = getpagesize() - 1; mapsize = ((int)len + pagemask) & ~pagemask; private->physmem = (Uint8 *)mmap(NULL, mapsize, PROT_READ | PROT_WRITE, MAP_SHARED, private->fd, (off_t)0); if (private->physmem == (Uint8 *)MAP_FAILED) { private->physmem = NULL; WSCONS_ReportError("mmap: %s", strerror(errno)); return -1; } private->fbmem_len = len; if (private->rotate == WSCONS_ROTATE_CW || private->rotate == WSCONS_ROTATE_CCW) { width = private->info.height; height = private->info.width; } else { width = private->info.width; height = private->info.height; } this->info.current_w = width; this->info.current_h = height;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -