1503 lines
42 KiB
C++
1503 lines
42 KiB
C++
/////////////////////////////////////////////////////////////////////////
|
|
// $Id$
|
|
/////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Copyright (C) 2000 Psyon.Org!
|
|
//
|
|
// Donald Becker
|
|
// http://www.psyon.org
|
|
//
|
|
// Copyright (C) 2001-2013 The Bochs Project
|
|
//
|
|
// 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 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
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
// New RFB(VNC) gui implementation using the LibVNCServer library to support
|
|
// higher resolutions and > 8 bpp graphics modes. This code is based on a patch
|
|
// sent by "HH LL" <superminililg@gmail.com>.
|
|
// To let mouse work properly in VNC client, you should use USB tablet
|
|
// emulation as the input source in bochsrc:
|
|
// usb_uhci: enabled=1, port1=tablet
|
|
// BTW, Thanks libvncserver, you also can use noVNC client(HTML5 VNC client) to
|
|
// view the screen in web browser.
|
|
|
|
// TODO:
|
|
// fix random segfaults in dimension_update()
|
|
// fix Windows (MinGW) support
|
|
// fixes for clients not supporting 'rfbEncodingNewFBSize'
|
|
// fix cursor shape after dimension update()
|
|
|
|
|
|
// Define BX_PLUGGABLE in files that can be compiled into plugins. For
|
|
// platforms that require a special tag on exported symbols, BX_PLUGGABLE
|
|
// is used to know when we are exporting symbols and when we are importing.
|
|
#define BX_PLUGGABLE
|
|
|
|
#include "param_names.h"
|
|
#include "iodev.h"
|
|
#include "keymap.h"
|
|
#if BX_WITH_VNCSRV
|
|
|
|
#include "icon_bochs.h"
|
|
#include "font/vga.bitmap.h"
|
|
#if (BX_WITH_SDL || BX_WITH_RFB) && !BX_PLUGINS
|
|
extern unsigned char sdl_font8x8[256][8];
|
|
#else
|
|
#include "sdl.h" // 8x8 font for status text
|
|
#endif
|
|
|
|
#define HAVE_LIBVNCSERVER
|
|
|
|
#include "rfb.h"
|
|
#include "rfbkeys.h"
|
|
|
|
class bx_vncsrv_gui_c: public bx_gui_c {
|
|
public:
|
|
bx_vncsrv_gui_c(void) : screen(NULL) {}
|
|
DECLARE_GUI_VIRTUAL_METHODS()
|
|
DECLARE_GUI_NEW_VIRTUAL_METHODS()
|
|
void get_capabilities(Bit16u *xres, Bit16u *yres, Bit16u *bpp);
|
|
void statusbar_setitem_specific(int element, bx_bool active, bx_bool w);
|
|
virtual void set_mouse_mode_absxy(bx_bool mode);
|
|
#if BX_SHOW_IPS
|
|
void show_ips(Bit32u ips_count);
|
|
#endif
|
|
rfbScreenInfoPtr screen;
|
|
};
|
|
|
|
// declare one instance of the gui object and call macro to insert the
|
|
// plugin code
|
|
static bx_vncsrv_gui_c *theGui = NULL;
|
|
IMPLEMENT_GUI_PLUGIN_CODE(vncsrv)
|
|
|
|
#define LOG_THIS theGui->
|
|
|
|
#ifdef WIN32
|
|
|
|
#include <winsock.h>
|
|
#include <process.h>
|
|
|
|
#undef LOCK
|
|
#undef UNLOCK
|
|
#undef MUTEX
|
|
#undef INIT_MUTEX
|
|
#undef TINI_MUTEX
|
|
#define LOCK(mutex) EnterCriticalSection(&(mutex))
|
|
#define UNLOCK(mutex) LeaveCriticalSection(&(mutex))
|
|
#define MUTEX(mutex) CRITICAL_SECTION (mutex)
|
|
#define INIT_MUTEX(mutex) InitializeCriticalSection(&(mutex))
|
|
#define TINI_MUTEX(mutex) DeleteCriticalSection(&(mutex))
|
|
|
|
#else
|
|
|
|
#include <sys/socket.h>
|
|
#include <netinet/tcp.h>
|
|
#include <netinet/in.h>
|
|
#include <unistd.h>
|
|
#ifndef __QNXNTO__
|
|
#include <sys/errno.h>
|
|
#else
|
|
#include <errno.h>
|
|
#endif
|
|
#include <pthread.h>
|
|
|
|
#endif
|
|
|
|
static bx_bool client_connected;
|
|
static bx_bool desktop_resizable = 1;
|
|
#if BX_SHOW_IPS
|
|
static bx_bool rfbHideIPS = 0;
|
|
static bx_bool rfbIPSupdate = 0;
|
|
static char rfbIPStext[40];
|
|
#endif
|
|
|
|
// Headerbar stuff
|
|
static unsigned rfbBitmapCount = 0;
|
|
static struct _rfbBitmaps {
|
|
char *bmap;
|
|
unsigned xdim;
|
|
unsigned ydim;
|
|
} rfbBitmaps[BX_MAX_PIXMAPS];
|
|
|
|
static unsigned rfbHeaderbarBitmapCount = 0;
|
|
static struct _rfbHeaderbarBitmaps {
|
|
unsigned int index;
|
|
unsigned int xorigin;
|
|
unsigned int yorigin;
|
|
unsigned int alignment;
|
|
void (*f)(void);
|
|
} rfbHeaderbarBitmaps[BX_MAX_HEADERBAR_ENTRIES];
|
|
|
|
//Keyboard stuff
|
|
#define KEYBOARD 1
|
|
#define MOUSE 0
|
|
#define MAX_KEY_EVENTS 512
|
|
static struct _rfbKeyboardEvent {
|
|
bx_bool type;
|
|
int key;
|
|
int down;
|
|
int x;
|
|
int y;
|
|
int z;
|
|
} rfbKeyboardEvent[MAX_KEY_EVENTS];
|
|
static unsigned long rfbKeyboardEvents = 0;
|
|
static MUTEX(bKeyboardInUse);
|
|
|
|
// Misc Stuff
|
|
|
|
#define BX_RFB_MAX_XDIM 1280
|
|
#define BX_RFB_MAX_YDIM 1024
|
|
#define BX_RFB_DEF_XDIM 720
|
|
#define BX_RFB_DEF_YDIM 480
|
|
|
|
#ifdef BX_LITTLE_ENDIAN
|
|
const rfbPixel status_led_green = 0x0000ff00;
|
|
const rfbPixel status_gray_text = 0x00808080;
|
|
const rfbPixel status_led_red = 0x000040ff;
|
|
const rfbPixel headerbar_bg = 0x00d0d0d0;
|
|
const rfbPixel headerbar_fg = 0x00101010;
|
|
#else
|
|
const rfbPixel status_led_green = 0x00ff0000;
|
|
const rfbPixel status_gray_text = 0x80808000;
|
|
const rfbPixel status_led_red = 0x00ff4000;
|
|
const rfbPixel headerbar_bg = 0xd0d0d000;
|
|
const rfbPixel headerbar_fg = 0x10101000;
|
|
#endif
|
|
|
|
static rfbPixel rfbPalette[256];
|
|
|
|
static bool rfbServerDown = false;
|
|
|
|
static unsigned rfbWindowX, rfbWindowY;
|
|
static unsigned rfbDimensionX, rfbDimensionY;
|
|
static long rfbHeaderbarY;
|
|
static long rfbTileX = 0;
|
|
static long rfbTileY = 0;
|
|
static unsigned long rfbCursorX = 0;
|
|
static unsigned long rfbCursorY = 0;
|
|
static unsigned long rfbOriginLeft = 0;
|
|
static unsigned long rfbOriginRight = 0;
|
|
static bx_bool rfbMouseModeAbsXY = 0;
|
|
static unsigned rfbStatusbarY = 18;
|
|
static unsigned rfbStatusitemPos[12] = {
|
|
0, 170, 210, 250, 290, 330, 370, 410, 450, 490, 530, 570
|
|
};
|
|
static bx_bool rfbStatusitemActive[12];
|
|
|
|
static unsigned int text_rows = 25, text_cols = 80;
|
|
static unsigned int font_height = 16, font_width = 8;
|
|
|
|
inline rfbPixel rfbMapRGB(U32 red, U32 green, U32 blue) {
|
|
U16 redMax = theGui->screen->serverFormat.redMax;
|
|
U16 greenMax = theGui->screen->serverFormat.greenMax;
|
|
U16 blueMax = theGui->screen->serverFormat.blueMax;
|
|
U16 redShift = theGui->screen->serverFormat.redShift;
|
|
U16 greenShift = theGui->screen->serverFormat.greenShift;
|
|
U16 blueShift = theGui->screen->serverFormat.blueShift;
|
|
rfbPixel pixel_value = ((red & redMax) << redShift)
|
|
| ((green & greenMax) << greenShift)
|
|
| ((blue & blueMax) << blueShift);
|
|
return pixel_value;
|
|
}
|
|
|
|
void vncStartThread();
|
|
void DrawBitmap(int x, int y, int width, int height, char *bmap, rfbPixel fg,
|
|
rfbPixel bg);
|
|
void DrawChar(int x, int y, int width, int height, int fonty, char *bmap,
|
|
rfbPixel fg, rfbPixel bg, bx_bool gfxchar);
|
|
void UpdateScreen(rfbPixel *newBits, int x, int y, int width, int height);
|
|
void SendUpdate(int x, int y, int width, int height);
|
|
void vncSetStatusText(int element, const char *text, bx_bool active, bx_bool w = 0);
|
|
static Bit32u convertStringToRfbKey(const char *string);
|
|
void vncKeyPressed(Bit32u key, int press_release);
|
|
void vncMouseMove(int x, int y, int z, int bmask);
|
|
|
|
void clientgone(rfbClientPtr cl);
|
|
enum rfbNewClientAction newclient(rfbClientPtr cl);
|
|
void dokey(rfbBool down, rfbKeySym key, rfbClientPtr cl);
|
|
void doptr(int buttonMask, int x, int y, rfbClientPtr cl);
|
|
void newframebuffer(rfbScreenInfoPtr screen, int width, int height);
|
|
|
|
#if BX_SHOW_IPS && defined(WIN32)
|
|
DWORD WINAPI vncShowIPSthread(LPVOID);
|
|
#endif
|
|
|
|
|
|
// VNCSRV implementation of the bx_gui_c methods
|
|
|
|
// ::SPECIFIC_INIT()
|
|
//
|
|
// Called from gui.cc, once upon program startup, to allow for the
|
|
// specific GUI code (X11, Win32, ...) to be initialized.
|
|
//
|
|
// argc, argv: used to pass display library specific options to the init code
|
|
// (X11 options, Win32 options,...)
|
|
//
|
|
// headerbar_y: A headerbar (toolbar) is display on the top of the
|
|
// VGA window, showing floppy status, and other information. It
|
|
// always assumes the width of the current VGA mode width, but
|
|
// it's height is defined by this parameter.
|
|
|
|
void bx_vncsrv_gui_c::specific_init(int argc, char **argv, unsigned headerbar_y)
|
|
{
|
|
unsigned char fc, vc;
|
|
int i, timeout = 30;
|
|
|
|
put("VNCSV");
|
|
UNUSED(bochs_icon_bits);
|
|
|
|
rfbHeaderbarY = headerbar_y;
|
|
rfbDimensionX = BX_RFB_DEF_XDIM;
|
|
rfbDimensionY = BX_RFB_DEF_YDIM;
|
|
rfbWindowX = rfbDimensionX;
|
|
rfbWindowY = rfbDimensionY + rfbHeaderbarY + rfbStatusbarY;
|
|
rfbTileX = x_tilesize;
|
|
rfbTileY = y_tilesize;
|
|
|
|
max_xres = BX_RFB_MAX_XDIM;
|
|
max_yres = BX_RFB_DEF_YDIM;
|
|
|
|
for (i = 0; i < 256; i++) {
|
|
for (int j = 0; j < 16; j++) {
|
|
vc = bx_vgafont[i].data[j];
|
|
fc = 0;
|
|
for (int b = 0; b < 8; b++) {
|
|
fc |= (vc & 0x01) << (7 - b);
|
|
vc >>= 1;
|
|
}
|
|
vga_charmap[i * 32 + j] = fc;
|
|
}
|
|
}
|
|
|
|
// parse vncsrv specific options
|
|
if (argc > 1) {
|
|
for (i = 1; i < argc; i++) {
|
|
if (!strncmp(argv[i], "timeout=", 8)) {
|
|
timeout = atoi(&argv[i][8]);
|
|
if (timeout < 0) {
|
|
BX_PANIC(("invalid timeout value: %d", timeout));
|
|
} else {
|
|
BX_INFO(("connection timeout set to %d", timeout));
|
|
}
|
|
#if BX_SHOW_IPS
|
|
} else if (!strcmp(argv[i], "hideIPS")) {
|
|
BX_INFO(("hide IPS display in status bar"));
|
|
rfbHideIPS = 1;
|
|
#endif
|
|
} else {
|
|
BX_PANIC(("Unknown rfb option '%s'", argv[i]));
|
|
}
|
|
}
|
|
}
|
|
|
|
if (SIM->get_param_bool(BXPN_PRIVATE_COLORMAP)->get()) {
|
|
BX_ERROR(("private_colormap option ignored."));
|
|
}
|
|
|
|
screen = rfbGetScreen(&argc, argv, rfbWindowX, rfbWindowY, 8, 3, sizeof(rfbPixel));
|
|
if (!screen)
|
|
BX_PANIC(("Create VNC screen failed!"));
|
|
|
|
screen->desktopName = "Bochs VNC Screen";
|
|
screen->frameBuffer = (char *) malloc(rfbWindowX * rfbWindowY * sizeof(rfbPixel));
|
|
screen->alwaysShared = TRUE;
|
|
screen->ptrAddEvent = doptr;
|
|
screen->kbdAddEvent = dokey;
|
|
screen->newClientHook = newclient;
|
|
screen->httpDir = (char*)"./webclients";
|
|
screen->httpEnableProxyConnect = TRUE;
|
|
|
|
memset(screen->frameBuffer, 0, rfbWindowX * rfbWindowY * sizeof(rfbPixel));
|
|
|
|
/* initialize the server */
|
|
rfbInitServer(screen);
|
|
|
|
client_connected = 0;
|
|
INIT_MUTEX(bKeyboardInUse);
|
|
vncStartThread();
|
|
|
|
#ifdef WIN32
|
|
Sleep(1000);
|
|
SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_BELOW_NORMAL);
|
|
#endif
|
|
|
|
// load keymap for vncsrv
|
|
if (SIM->get_param_bool(BXPN_KBD_USEMAPPING)->get()) {
|
|
bx_keymap.loadKeymap(convertStringToRfbKey);
|
|
}
|
|
|
|
// the ask menu doesn't work on the client side
|
|
io->set_log_action(LOGLEV_PANIC, ACT_FATAL);
|
|
|
|
if (timeout > 0) {
|
|
while ((!client_connected) && (timeout--)) {
|
|
fprintf(stderr, "Bochs VNC server waiting for client: %2d\r", timeout+1);
|
|
#ifdef WIN32
|
|
Sleep(1000);
|
|
#else
|
|
sleep(1);
|
|
#endif
|
|
}
|
|
if ((timeout < 0) && (!client_connected)) {
|
|
BX_PANIC(("timeout! no client present"));
|
|
} else {
|
|
fprintf(stderr, "VNC client connected \r");
|
|
}
|
|
}
|
|
|
|
#if BX_SHOW_IPS && defined(WIN32)
|
|
if (!rfbHideIPS) {
|
|
DWORD threadID;
|
|
CreateThread(NULL, 0, vncShowIPSthread, NULL, 0, &threadID);
|
|
}
|
|
#endif
|
|
|
|
new_gfx_api = 1;
|
|
dialog_caps = 0;
|
|
}
|
|
|
|
// ::HANDLE_EVENTS()
|
|
//
|
|
// Called periodically (vga_update_interval in .bochsrc) so the
|
|
// the gui code can poll for keyboard, mouse, and other
|
|
// relevant events.
|
|
|
|
void bx_vncsrv_gui_c::handle_events(void)
|
|
{
|
|
LOCK(bKeyboardInUse);
|
|
|
|
if (rfbKeyboardEvents > 0) {
|
|
for (unsigned i = 0; i < rfbKeyboardEvents; i++) {
|
|
if (rfbKeyboardEvent[i].type == KEYBOARD) {
|
|
vncKeyPressed(rfbKeyboardEvent[i].key, rfbKeyboardEvent[i].down);
|
|
} else { //type == MOUSE;
|
|
vncMouseMove(rfbKeyboardEvent[i].x, rfbKeyboardEvent[i].y, rfbKeyboardEvent[i].z, rfbKeyboardEvent[i].down);
|
|
}
|
|
}
|
|
rfbKeyboardEvents = 0;
|
|
}
|
|
UNLOCK(bKeyboardInUse);
|
|
|
|
#if BX_SHOW_IPS
|
|
if (rfbIPSupdate) {
|
|
rfbIPSupdate = 0;
|
|
vncSetStatusText(0, rfbIPStext, 1);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
// ::FLUSH()
|
|
//
|
|
// Called periodically, requesting that the gui code flush all pending
|
|
// screen update requests.
|
|
|
|
void bx_vncsrv_gui_c::flush(void)
|
|
{
|
|
}
|
|
|
|
// ::CLEAR_SCREEN()
|
|
//
|
|
// Called to request that the VGA region is cleared. Don't
|
|
// clear the area that defines the headerbar.
|
|
void bx_vncsrv_gui_c::clear_screen(void)
|
|
{
|
|
memset(&(screen->frameBuffer[rfbWindowX * rfbHeaderbarY * sizeof(rfbPixel)]),
|
|
0, rfbWindowX * rfbDimensionY * sizeof(rfbPixel));
|
|
SendUpdate(0, rfbHeaderbarY, rfbDimensionX, rfbDimensionY);
|
|
}
|
|
|
|
// ::TEXT_UPDATE()
|
|
//
|
|
// Called in a VGA text mode, to update the screen with
|
|
// new content.
|
|
//
|
|
// old_text: array of character/attributes making up the contents
|
|
// of the screen from the last call. See below
|
|
// new_text: array of character/attributes making up the current
|
|
// contents, which should now be displayed. See below
|
|
//
|
|
// format of old_text & new_text: each is tm_info->line_offset*text_rows
|
|
// bytes long. Each character consists of 2 bytes. The first by is
|
|
// the character value, the second is the attribute byte.
|
|
//
|
|
// cursor_x: new x location of cursor
|
|
// cursor_y: new y location of cursor
|
|
// tm_info: this structure contains information for additional
|
|
// features in text mode (cursor shape, line offset,...)
|
|
|
|
void bx_vncsrv_gui_c::text_update(Bit8u *old_text, Bit8u *new_text,
|
|
unsigned long cursor_x, unsigned long cursor_y,
|
|
bx_vga_tminfo_t *tm_info)
|
|
{
|
|
Bit8u *old_line, *new_line;
|
|
Bit8u cAttr, cChar;
|
|
unsigned int curs, hchars, offset, rows, x, y, xc, yc, i;
|
|
bx_bool force_update = 0, gfxchar, blink_state, blink_mode;
|
|
rfbPixel text_palette[16];
|
|
rfbPixel fgcolor, bgcolor;
|
|
|
|
for (i = 0; i < 16; i++) {
|
|
text_palette[i] = rfbPalette[tm_info->actl_palette[i]];
|
|
}
|
|
|
|
blink_mode = (tm_info->blink_flags & BX_TEXT_BLINK_MODE) > 0;
|
|
blink_state = (tm_info->blink_flags & BX_TEXT_BLINK_STATE) > 0;
|
|
if (blink_mode) {
|
|
if (tm_info->blink_flags & BX_TEXT_BLINK_TOGGLE)
|
|
force_update = 1;
|
|
}
|
|
if (charmap_updated) {
|
|
force_update = 1;
|
|
charmap_updated = 0;
|
|
}
|
|
|
|
// first invalidate character at previous and new cursor location
|
|
if ((rfbCursorY < text_rows) && (rfbCursorX < text_cols)) {
|
|
curs = rfbCursorY * tm_info->line_offset + rfbCursorX * 2;
|
|
old_text[curs] = ~new_text[curs];
|
|
}
|
|
if ((tm_info->cs_start <= tm_info->cs_end) && (tm_info->cs_start < font_height)
|
|
&& (cursor_y < text_rows) && (cursor_x < text_cols)) {
|
|
curs = cursor_y * tm_info->line_offset + cursor_x * 2;
|
|
old_text[curs] = ~new_text[curs];
|
|
} else {
|
|
curs = 0xffff;
|
|
}
|
|
|
|
rows = text_rows;
|
|
y = 0;
|
|
do {
|
|
hchars = text_cols;
|
|
new_line = new_text;
|
|
old_line = old_text;
|
|
offset = y * tm_info->line_offset;
|
|
yc = y * font_height + rfbHeaderbarY;
|
|
x = 0;
|
|
do {
|
|
if (force_update || (old_text[0] != new_text[0])
|
|
|| (old_text[1] != new_text[1])) {
|
|
cChar = new_text[0];
|
|
|
|
if (blink_mode) {
|
|
cAttr = new_text[1] & 0x7F;
|
|
if (!blink_state && (new_text[1] & 0x80))
|
|
cAttr = (cAttr & 0x70) | (cAttr >> 4);
|
|
} else {
|
|
cAttr = new_text[1];
|
|
}
|
|
fgcolor = text_palette[cAttr & 0x0F];
|
|
bgcolor = text_palette[cAttr >> 4];
|
|
|
|
gfxchar = tm_info->line_graphics && ((cChar & 0xE0) == 0xC0);
|
|
xc = x * font_width;
|
|
DrawChar(xc, yc, font_width, font_height, 0,
|
|
(char *) &vga_charmap[cChar << 5], fgcolor, bgcolor,
|
|
gfxchar);
|
|
|
|
SendUpdate(xc, yc, font_width, font_height);
|
|
|
|
if (offset == curs) {
|
|
DrawChar(xc, yc + tm_info->cs_start, font_width,
|
|
tm_info->cs_end - tm_info->cs_start + 1,
|
|
tm_info->cs_start,
|
|
(char *) &vga_charmap[cChar << 5], bgcolor, fgcolor,
|
|
gfxchar);
|
|
}
|
|
}
|
|
x++;
|
|
new_text += 2;
|
|
old_text += 2;
|
|
offset += 2;
|
|
} while (--hchars);
|
|
y++;
|
|
new_text = new_line + tm_info->line_offset;
|
|
old_text = old_line + tm_info->line_offset;
|
|
} while (--rows);
|
|
|
|
rfbCursorX = cursor_x;
|
|
rfbCursorY = cursor_y;
|
|
}
|
|
|
|
int bx_vncsrv_gui_c::get_clipboard_text(Bit8u **bytes, Bit32s *nbytes)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int bx_vncsrv_gui_c::set_clipboard_text(char *text_snapshot, Bit32u len)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
// ::PALETTE_CHANGE()
|
|
//
|
|
// Allocate a color in the native GUI, for this color, and put
|
|
// it in the colormap location 'index'.
|
|
// returns: 0=no screen update needed (color map change has direct effect)
|
|
// 1=screen updated needed (redraw using current colormap)
|
|
|
|
bx_bool bx_vncsrv_gui_c::palette_change(U8 index, U8 red,
|
|
U8 green, U8 blue)
|
|
{
|
|
rfbPalette[index] = rfbMapRGB(red, green, blue);
|
|
return 1;
|
|
}
|
|
|
|
// ::GRAPHICS_TILE_UPDATE()
|
|
//
|
|
// Called to request that a tile of graphics be drawn to the
|
|
// screen, since info in this region has changed.
|
|
//
|
|
// tile: array of 8bit values representing a block of pixels with
|
|
// dimension equal to the 'x_tilesize' & 'y_tilesize' members.
|
|
// Each value specifies an index into the
|
|
// array of colors you allocated for ::palette_change()
|
|
// x0: x origin of tile
|
|
// y0: y origin of tile
|
|
//
|
|
// note: origin of tile and of window based on (0,0) being in the upper
|
|
// left of the window.
|
|
void bx_vncsrv_gui_c::graphics_tile_update(Bit8u *tile, unsigned x0, unsigned y0)
|
|
{
|
|
rfbPixel *buf;
|
|
rfbPixel *buf_row;
|
|
int i, j;
|
|
|
|
buf = ((rfbPixel*) (screen->frameBuffer))
|
|
+ (rfbHeaderbarY + y0) * rfbWindowX + x0;
|
|
|
|
i = rfbTileY;
|
|
if (i + y0 > rfbDimensionY)
|
|
i = rfbDimensionY - y0;
|
|
|
|
// FIXME
|
|
if (i <= 0)
|
|
return;
|
|
|
|
switch (guest_bpp) {
|
|
case 8: /* 8 bpp */
|
|
do {
|
|
buf_row = buf;
|
|
j = rfbTileX;
|
|
do {
|
|
*buf++ = rfbPalette[*tile++];
|
|
} while (--j);
|
|
buf = buf_row + rfbWindowX;
|
|
} while (--i);
|
|
break;
|
|
default:
|
|
BX_PANIC(("%u bpp modes handled by new graphics API", guest_bpp));
|
|
return;
|
|
}
|
|
SendUpdate(x0, y0 + rfbHeaderbarY, rfbTileX, rfbTileY);
|
|
}
|
|
|
|
// ::DIMENSION_UPDATE()
|
|
//
|
|
// Called when the VGA mode changes it's X,Y dimensions.
|
|
// Resize the window to this size, but you need to add on
|
|
// the height of the headerbar to the Y value.
|
|
//
|
|
// x: new VGA x size
|
|
// y: new VGA y size (add headerbar_y parameter from ::specific_init().
|
|
// fheight: new VGA character height in text mode
|
|
// fwidth : new VGA character width in text mode
|
|
// bpp : bits per pixel in graphics mode
|
|
|
|
void bx_vncsrv_gui_c::dimension_update(unsigned x, unsigned y, unsigned fheight,
|
|
unsigned fwidth, unsigned bpp)
|
|
{
|
|
if (bpp == 8 || bpp == 15 || bpp == 16 || bpp == 24 || bpp == 32) {
|
|
guest_bpp = bpp;
|
|
} else {
|
|
BX_PANIC(("%d bpp graphics mode not supported", bpp));
|
|
}
|
|
guest_textmode = (fheight > 0);
|
|
guest_xres = x;
|
|
guest_yres = y;
|
|
if (guest_textmode) {
|
|
font_height = fheight;
|
|
font_width = fwidth;
|
|
text_cols = x / fwidth;
|
|
text_rows = y / fheight;
|
|
}
|
|
if ((x != rfbDimensionX) || (y != rfbDimensionY)) {
|
|
if (desktop_resizable) {
|
|
if ((x > BX_RFB_MAX_XDIM) || (y > BX_RFB_MAX_YDIM)) {
|
|
BX_PANIC(("dimension_update(): VNC doesn't support graphics mode %dx%d", x, y));
|
|
}
|
|
rfbDimensionX = x;
|
|
rfbDimensionY = y;
|
|
rfbWindowX = rfbDimensionX;
|
|
rfbWindowY = rfbDimensionY + rfbHeaderbarY + rfbStatusbarY;
|
|
newframebuffer(this->screen, rfbWindowX, rfbWindowY);
|
|
SendUpdate(0, 0, rfbWindowX, rfbWindowY);
|
|
bx_gui->show_headerbar();
|
|
} else {
|
|
if ((x > BX_RFB_DEF_XDIM) || (y > BX_RFB_DEF_YDIM)) {
|
|
BX_PANIC(("dimension_update(): VNC doesn't support graphics mode %dx%d", x, y));
|
|
}
|
|
clear_screen();
|
|
SendUpdate(0, rfbHeaderbarY, rfbDimensionX, rfbDimensionY);
|
|
rfbDimensionX = x;
|
|
rfbDimensionY = y;
|
|
}
|
|
}
|
|
}
|
|
|
|
// ::CREATE_BITMAP()
|
|
//
|
|
// Create a monochrome bitmap of size 'xdim' by 'ydim', which will
|
|
// be drawn in the headerbar. Return an integer ID to the bitmap,
|
|
// with which the bitmap can be referenced later.
|
|
//
|
|
// bmap: packed 8 pixels-per-byte bitmap. The pixel order is:
|
|
// bit0 is the left most pixel, bit7 is the right most pixel.
|
|
// xdim: x dimension of bitmap
|
|
// ydim: y dimension of bitmap
|
|
|
|
unsigned bx_vncsrv_gui_c::create_bitmap(const unsigned char *bmap, unsigned xdim,
|
|
unsigned ydim)
|
|
{
|
|
if (rfbBitmapCount >= BX_MAX_PIXMAPS) {
|
|
BX_ERROR(("too many pixmaps."));
|
|
return 0;
|
|
}
|
|
rfbBitmaps[rfbBitmapCount].bmap = (char *) malloc((xdim * ydim) / 8);
|
|
rfbBitmaps[rfbBitmapCount].xdim = xdim;
|
|
rfbBitmaps[rfbBitmapCount].ydim = ydim;
|
|
memcpy(rfbBitmaps[rfbBitmapCount].bmap, bmap, (xdim * ydim) / 8);
|
|
|
|
rfbBitmapCount++;
|
|
return (rfbBitmapCount - 1);
|
|
}
|
|
|
|
// ::HEADERBAR_BITMAP()
|
|
//
|
|
// Called to install a bitmap in the bochs headerbar (toolbar).
|
|
//
|
|
// bmap_id: will correspond to an ID returned from
|
|
// ::create_bitmap(). 'alignment' is either BX_GRAVITY_LEFT
|
|
// or BX_GRAVITY_RIGHT, meaning install the bitmap in the next
|
|
// available leftmost or rightmost space.
|
|
// alignment: is either BX_GRAVITY_LEFT or BX_GRAVITY_RIGHT,
|
|
// meaning install the bitmap in the next
|
|
// available leftmost or rightmost space.
|
|
// f: a 'C' function pointer to callback when the mouse is clicked in
|
|
// the boundaries of this bitmap.
|
|
|
|
unsigned bx_vncsrv_gui_c::headerbar_bitmap(unsigned bmap_id, unsigned alignment,
|
|
void (*f)(void))
|
|
{
|
|
int hb_index;
|
|
|
|
if ((rfbHeaderbarBitmapCount + 1) > BX_MAX_HEADERBAR_ENTRIES) {
|
|
return 0;
|
|
}
|
|
|
|
rfbHeaderbarBitmapCount++;
|
|
hb_index = rfbHeaderbarBitmapCount - 1;
|
|
rfbHeaderbarBitmaps[hb_index].index = bmap_id;
|
|
rfbHeaderbarBitmaps[hb_index].alignment = alignment;
|
|
rfbHeaderbarBitmaps[hb_index].f = f;
|
|
if (alignment == BX_GRAVITY_LEFT) {
|
|
rfbHeaderbarBitmaps[hb_index].xorigin = rfbOriginLeft;
|
|
rfbHeaderbarBitmaps[hb_index].yorigin = 0;
|
|
rfbOriginLeft += rfbBitmaps[bmap_id].xdim;
|
|
} else { // BX_GRAVITY_RIGHT
|
|
rfbOriginRight += rfbBitmaps[bmap_id].xdim;
|
|
rfbHeaderbarBitmaps[hb_index].xorigin = rfbOriginRight;
|
|
rfbHeaderbarBitmaps[hb_index].yorigin = 0;
|
|
}
|
|
return hb_index;
|
|
}
|
|
|
|
// ::SHOW_HEADERBAR()
|
|
//
|
|
// Show (redraw) the current headerbar, which is composed of
|
|
// currently installed bitmaps.
|
|
|
|
void bx_vncsrv_gui_c::show_headerbar(void)
|
|
{
|
|
char *newBits, value;
|
|
unsigned int i, xorigin, addr;
|
|
|
|
newBits = (char *) malloc(rfbWindowX * rfbHeaderbarY);
|
|
memset(newBits, 0, (rfbWindowX * rfbHeaderbarY));
|
|
DrawBitmap(0, 0, rfbWindowX, rfbHeaderbarY, newBits, headerbar_fg,
|
|
headerbar_bg);
|
|
for (i = 0; i < rfbHeaderbarBitmapCount; i++) {
|
|
if (rfbHeaderbarBitmaps[i].alignment == BX_GRAVITY_LEFT) {
|
|
xorigin = rfbHeaderbarBitmaps[i].xorigin;
|
|
} else {
|
|
xorigin = rfbWindowX - rfbHeaderbarBitmaps[i].xorigin;
|
|
}
|
|
DrawBitmap(xorigin, 0, rfbBitmaps[rfbHeaderbarBitmaps[i].index].xdim,
|
|
rfbBitmaps[rfbHeaderbarBitmaps[i].index].ydim,
|
|
rfbBitmaps[rfbHeaderbarBitmaps[i].index].bmap, headerbar_fg,
|
|
headerbar_bg);
|
|
}
|
|
free(newBits);
|
|
newBits = (char *) malloc(rfbWindowX * rfbStatusbarY / 8);
|
|
memset(newBits, 0, (rfbWindowX * rfbStatusbarY / 8));
|
|
for (i = 1; i < 12; i++) {
|
|
addr = rfbStatusitemPos[i] / 8;
|
|
value = 1 << (rfbStatusitemPos[i] % 8);
|
|
for (unsigned j = 1; j < rfbStatusbarY; j++) {
|
|
newBits[(rfbWindowX * j / 8) + addr] = value;
|
|
}
|
|
}
|
|
DrawBitmap(0, rfbWindowY - rfbStatusbarY, rfbWindowX, rfbStatusbarY,
|
|
newBits, headerbar_fg, headerbar_bg);
|
|
free(newBits);
|
|
for (i = 1; i <= statusitem_count; i++) {
|
|
vncSetStatusText(i, statusitem[i - 1].text, rfbStatusitemActive[i]);
|
|
}
|
|
}
|
|
|
|
// ::REPLACE_BITMAP()
|
|
//
|
|
// Replace the bitmap installed in the headerbar ID slot 'hbar_id',
|
|
// with the one specified by 'bmap_id'. 'bmap_id' will have
|
|
// been generated by ::create_bitmap(). The old and new bitmap
|
|
// must be of the same size. This allows the bitmap the user
|
|
// sees to change, when some action occurs. For example when
|
|
// the user presses on the floppy icon, it then displays
|
|
// the ejected status.
|
|
//
|
|
// hbar_id: headerbar slot ID
|
|
// bmap_id: bitmap ID
|
|
|
|
void bx_vncsrv_gui_c::replace_bitmap(unsigned hbar_id, unsigned bmap_id)
|
|
{
|
|
unsigned int xorigin;
|
|
|
|
if (bmap_id == rfbHeaderbarBitmaps[hbar_id].index)
|
|
return;
|
|
rfbHeaderbarBitmaps[hbar_id].index = bmap_id;
|
|
if (rfbHeaderbarBitmaps[hbar_id].alignment == BX_GRAVITY_LEFT) {
|
|
xorigin = rfbHeaderbarBitmaps[hbar_id].xorigin;
|
|
} else {
|
|
xorigin = rfbWindowX - rfbHeaderbarBitmaps[hbar_id].xorigin;
|
|
}
|
|
DrawBitmap(xorigin, 0, rfbBitmaps[rfbHeaderbarBitmaps[hbar_id].index].xdim,
|
|
rfbBitmaps[rfbHeaderbarBitmaps[hbar_id].index].ydim,
|
|
rfbBitmaps[rfbHeaderbarBitmaps[hbar_id].index].bmap, headerbar_fg,
|
|
headerbar_bg);
|
|
}
|
|
|
|
// ::EXIT()
|
|
//
|
|
// Called before bochs terminates, to allow for a graceful
|
|
// exit from the native GUI mechanism.
|
|
void bx_vncsrv_gui_c::exit(void)
|
|
{
|
|
unsigned int i;
|
|
|
|
rfbShutdownServer(theGui->screen, TRUE);
|
|
|
|
while (!rfbServerDown) {
|
|
usleep(10000); //10ms
|
|
}
|
|
TINI_MUTEX(bKeyboardInUse);
|
|
|
|
for (i = 0; i < rfbBitmapCount; i++) {
|
|
free(rfbBitmaps[i].bmap);
|
|
}
|
|
|
|
BX_DEBUG(("bx_vncsrv_gui_c::exit()"));
|
|
}
|
|
|
|
void bx_vncsrv_gui_c::mouse_enabled_changed_specific(bx_bool val)
|
|
{
|
|
}
|
|
|
|
bx_svga_tileinfo_t *bx_vncsrv_gui_c::graphics_tile_info(bx_svga_tileinfo_t *info)
|
|
{
|
|
info->bpp = 32;
|
|
info->pitch = rfbWindowX * 4;
|
|
info->red_shift = 8;
|
|
info->green_shift = 16;
|
|
info->blue_shift = 24;
|
|
info->red_mask = 0x0000ff;
|
|
info->green_mask = 0x00ff00;
|
|
info->blue_mask = 0xff0000;
|
|
info->is_indexed = 0;
|
|
#ifdef BX_LITTLE_ENDIAN
|
|
info->is_little_endian = 1;
|
|
#else
|
|
info->is_little_endian = 0;
|
|
#endif
|
|
|
|
return info;
|
|
}
|
|
|
|
Bit8u *bx_vncsrv_gui_c::graphics_tile_get(unsigned x0, unsigned y0, unsigned *w,
|
|
unsigned *h)
|
|
{
|
|
if (x0 + rfbTileX > rfbDimensionX) {
|
|
*w = rfbDimensionX - x0;
|
|
} else {
|
|
*w = rfbTileX;
|
|
}
|
|
|
|
if (y0 + rfbTileY > rfbDimensionY) {
|
|
*h = rfbDimensionY - y0;
|
|
} else {
|
|
*h = rfbTileY;
|
|
}
|
|
|
|
return (Bit8u *) (screen->frameBuffer)
|
|
+ (rfbHeaderbarY + y0) * rfbWindowX * sizeof(rfbPixel)
|
|
+ x0 * sizeof(rfbPixel);
|
|
}
|
|
|
|
void bx_vncsrv_gui_c::graphics_tile_update_in_place(unsigned x0, unsigned y0,
|
|
unsigned w, unsigned h)
|
|
{
|
|
SendUpdate(x0, y0 + rfbHeaderbarY, w, h);
|
|
}
|
|
void bx_vncsrv_gui_c::get_capabilities(Bit16u *xres, Bit16u *yres, Bit16u *bpp)
|
|
{
|
|
if (desktop_resizable) {
|
|
*xres = BX_RFB_MAX_XDIM;
|
|
*yres = BX_RFB_MAX_YDIM;
|
|
} else {
|
|
*xres = BX_RFB_DEF_XDIM;
|
|
*yres = BX_RFB_DEF_YDIM;
|
|
}
|
|
*bpp = 32;
|
|
}
|
|
|
|
void bx_vncsrv_gui_c::statusbar_setitem_specific(int element, bx_bool active,
|
|
bx_bool w)
|
|
{
|
|
vncSetStatusText(element + 1, statusitem[element].text, active, w);
|
|
}
|
|
|
|
void bx_vncsrv_gui_c::set_mouse_mode_absxy(bx_bool mode)
|
|
{
|
|
rfbMouseModeAbsXY = mode;
|
|
}
|
|
|
|
#if BX_SHOW_IPS
|
|
void bx_vncsrv_gui_c::show_ips(Bit32u ips_count)
|
|
{
|
|
if (!rfbIPSupdate && !rfbHideIPS) {
|
|
ips_count /= 1000;
|
|
sprintf(rfbIPStext, "IPS: %u.%3.3uM", ips_count / 1000, ips_count % 1000);
|
|
rfbIPSupdate = 1;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// VNCSRV specific functions
|
|
|
|
void CDECL vncServerThreadInit(void *indata)
|
|
{
|
|
/* this is the blocking event loop, i.e. it never returns */
|
|
/* 40000 are the microseconds to wait on select(), i.e. 0.04 seconds*/
|
|
rfbRunEventLoop(theGui->screen, 40000, FALSE);
|
|
|
|
free(theGui->screen->frameBuffer);
|
|
rfbScreenCleanup(theGui->screen);
|
|
|
|
rfbServerDown = true;
|
|
|
|
return;
|
|
}
|
|
|
|
void vncStartThread()
|
|
{
|
|
#ifdef WIN32
|
|
_beginthread(vncServerThreadInit, 0, NULL);
|
|
#else
|
|
pthread_t thread;
|
|
pthread_create(&thread, NULL, (void *(*)(void *))&vncServerThreadInit, NULL);
|
|
#endif
|
|
}
|
|
|
|
void DrawBitmap(int x, int y, int width, int height, char *bmap,
|
|
rfbPixel fgcolor, rfbPixel bgcolor)
|
|
{
|
|
rfbPixel *newBits;
|
|
|
|
newBits = (rfbPixel *) malloc(width * height * sizeof(rfbPixel));
|
|
memset(newBits, 0, (width * height * sizeof(rfbPixel)));
|
|
for (int i = 0; i < (width * height) / 8; i++) {
|
|
newBits[i * 8 + 0] = (bmap[i] & 0x01) ? fgcolor : bgcolor;
|
|
newBits[i * 8 + 1] = (bmap[i] & 0x02) ? fgcolor : bgcolor;
|
|
newBits[i * 8 + 2] = (bmap[i] & 0x04) ? fgcolor : bgcolor;
|
|
newBits[i * 8 + 3] = (bmap[i] & 0x08) ? fgcolor : bgcolor;
|
|
newBits[i * 8 + 4] = (bmap[i] & 0x10) ? fgcolor : bgcolor;
|
|
newBits[i * 8 + 5] = (bmap[i] & 0x20) ? fgcolor : bgcolor;
|
|
newBits[i * 8 + 6] = (bmap[i] & 0x40) ? fgcolor : bgcolor;
|
|
newBits[i * 8 + 7] = (bmap[i] & 0x80) ? fgcolor : bgcolor;
|
|
}
|
|
UpdateScreen(newBits, x, y, width, height);
|
|
free(newBits);
|
|
}
|
|
|
|
void DrawChar(int x, int y, int width, int height, int fonty, char *bmap,
|
|
rfbPixel fgcolor, rfbPixel bgcolor, bx_bool gfxchar)
|
|
{
|
|
static rfbPixel newBits[9 * 32];
|
|
unsigned char mask;
|
|
int bytes = width * height;
|
|
|
|
for (int i = 0; i < bytes; i += width) {
|
|
mask = 0x80;
|
|
for (int j = 0; j < width; j++) {
|
|
if (mask > 0) {
|
|
newBits[i + j] = (bmap[fonty] & mask) ? fgcolor : bgcolor;
|
|
} else {
|
|
if (gfxchar) {
|
|
newBits[i + j] = (bmap[fonty] & 0x01) ? fgcolor : bgcolor;
|
|
} else {
|
|
newBits[i + j] = bgcolor;
|
|
}
|
|
}
|
|
mask >>= 1;
|
|
}
|
|
fonty++;
|
|
}
|
|
UpdateScreen(newBits, x, y, width, height);
|
|
}
|
|
|
|
void UpdateScreen(rfbPixel *newBits, int x, int y, int width, int height)
|
|
{
|
|
int i, x0, y0;
|
|
x0 = x;
|
|
y0 = y;
|
|
if ((unsigned)(x + width - 1) >= rfbWindowX) {
|
|
width = rfbWindowX - x + 1;
|
|
}
|
|
if ((unsigned)(y + height - 1) >= rfbWindowY) {
|
|
height = rfbWindowY - y + 1;
|
|
}
|
|
for (i = 0; i < height; i++) {
|
|
memcpy(&(theGui->screen->frameBuffer[y * rfbWindowX * sizeof(rfbPixel)
|
|
+ x * sizeof(rfbPixel)]),
|
|
&newBits[i * width], width * sizeof(rfbPixel));
|
|
y++;
|
|
}
|
|
SendUpdate(x0, y0, width, height);
|
|
}
|
|
|
|
void SendUpdate(int x, int y, int width, int height)
|
|
{
|
|
rfbMarkRectAsModified(theGui->screen, x, y, x + width, y + height);
|
|
}
|
|
|
|
void vncSetStatusText(int element, const char *text, bx_bool active, bx_bool w)
|
|
{
|
|
char *newBits;
|
|
unsigned xleft, xsize, i, len;
|
|
|
|
rfbStatusitemActive[element] = active;
|
|
xleft = rfbStatusitemPos[element] + 2;
|
|
xsize = rfbStatusitemPos[element + 1] - xleft - 1;
|
|
newBits = (char *) malloc(((xsize / 8) + 1) * (rfbStatusbarY - 2));
|
|
memset(newBits, 0, ((xsize / 8) + 1) * (rfbStatusbarY - 2));
|
|
for (i = 0; i < (rfbStatusbarY - 2); i++) {
|
|
newBits[((xsize / 8) + 1) * i] = 0;
|
|
}
|
|
|
|
rfbPixel fgcolor = active ? headerbar_fg : status_gray_text;
|
|
rfbPixel bgcolor = 0;
|
|
if (element > 0) {
|
|
bgcolor = active ? (w ? status_led_red : status_led_green) : headerbar_bg;
|
|
} else {
|
|
bgcolor = headerbar_bg;
|
|
}
|
|
DrawBitmap(xleft, rfbWindowY - rfbStatusbarY + 1, xsize, rfbStatusbarY - 2,
|
|
newBits, fgcolor, bgcolor);
|
|
|
|
free(newBits);
|
|
len = ((element > 0) && (strlen(text) > 4)) ? 4 : strlen(text);
|
|
for (i = 0; i < len; i++) {
|
|
DrawChar(xleft + i * 8 + 2, rfbWindowY - rfbStatusbarY + 5, 8, 8, 0,
|
|
(char *) &sdl_font8x8[(unsigned) text[i]][0], fgcolor, bgcolor, 0);
|
|
}
|
|
|
|
rfbMarkRectAsModified(theGui->screen, xleft, rfbWindowY - rfbStatusbarY + 1,
|
|
xleft + xsize, rfbWindowY - 2);
|
|
}
|
|
|
|
// function to convert key names into rfb key values.
|
|
// This first try will be horribly inefficient, but it only has
|
|
// to be done while loading a keymap. Once the simulation starts,
|
|
// this function won't be called.
|
|
static Bit32u convertStringToRfbKey(const char *string)
|
|
{
|
|
rfbKeyTabEntry *ptr;
|
|
for (ptr = &rfb_keytable[0]; ptr->name != NULL; ptr++) {
|
|
if (!strcmp(string, ptr->name))
|
|
return ptr->value;
|
|
}
|
|
return BX_KEYMAP_UNKNOWN;
|
|
}
|
|
|
|
static Bit32u rfb_ascii_to_key_event[0x5f] = {
|
|
// !"#$%&'
|
|
BX_KEY_SPACE, BX_KEY_1, BX_KEY_SINGLE_QUOTE, BX_KEY_3, BX_KEY_4,
|
|
BX_KEY_5, BX_KEY_7, BX_KEY_SINGLE_QUOTE,
|
|
|
|
// ()*+,-./
|
|
BX_KEY_9, BX_KEY_0, BX_KEY_8, BX_KEY_EQUALS, BX_KEY_COMMA, BX_KEY_MINUS,
|
|
BX_KEY_PERIOD, BX_KEY_SLASH,
|
|
|
|
// 01234567
|
|
BX_KEY_0, BX_KEY_1, BX_KEY_2, BX_KEY_3, BX_KEY_4, BX_KEY_5, BX_KEY_6,
|
|
BX_KEY_7,
|
|
|
|
// 89:;<=>?
|
|
BX_KEY_8, BX_KEY_9, BX_KEY_SEMICOLON, BX_KEY_SEMICOLON, BX_KEY_COMMA,
|
|
BX_KEY_EQUALS, BX_KEY_PERIOD, BX_KEY_SLASH,
|
|
|
|
// @ABCDEFG
|
|
BX_KEY_2, BX_KEY_A, BX_KEY_B, BX_KEY_C, BX_KEY_D, BX_KEY_E, BX_KEY_F,
|
|
BX_KEY_G,
|
|
|
|
// HIJKLMNO
|
|
BX_KEY_H, BX_KEY_I, BX_KEY_J, BX_KEY_K, BX_KEY_L, BX_KEY_M, BX_KEY_N,
|
|
BX_KEY_O,
|
|
|
|
// PQRSTUVW
|
|
BX_KEY_P, BX_KEY_Q, BX_KEY_R, BX_KEY_S, BX_KEY_T, BX_KEY_U, BX_KEY_V,
|
|
BX_KEY_W,
|
|
|
|
// XYZ[\]^_
|
|
BX_KEY_X, BX_KEY_Y, BX_KEY_Z, BX_KEY_LEFT_BRACKET, BX_KEY_BACKSLASH,
|
|
BX_KEY_RIGHT_BRACKET, BX_KEY_6, BX_KEY_MINUS,
|
|
|
|
// `abcdefg
|
|
BX_KEY_GRAVE, BX_KEY_A, BX_KEY_B, BX_KEY_C, BX_KEY_D, BX_KEY_E,
|
|
BX_KEY_F, BX_KEY_G,
|
|
|
|
// hijklmno
|
|
BX_KEY_H, BX_KEY_I, BX_KEY_J, BX_KEY_K, BX_KEY_L, BX_KEY_M, BX_KEY_N,
|
|
BX_KEY_O,
|
|
|
|
// pqrstuvw
|
|
BX_KEY_P, BX_KEY_Q, BX_KEY_R, BX_KEY_S, BX_KEY_T, BX_KEY_U, BX_KEY_V,
|
|
BX_KEY_W,
|
|
|
|
// xyz{|}~
|
|
BX_KEY_X, BX_KEY_Y, BX_KEY_Z, BX_KEY_LEFT_BRACKET, BX_KEY_BACKSLASH,
|
|
BX_KEY_RIGHT_BRACKET, BX_KEY_GRAVE
|
|
};
|
|
|
|
void vncKeyPressed(Bit32u key, int press_release)
|
|
{
|
|
Bit32u key_event;
|
|
|
|
if (!SIM->get_param_bool(BXPN_KBD_USEMAPPING)->get()) {
|
|
if ((key >= XK_space) && (key <= XK_asciitilde)) {
|
|
key_event = rfb_ascii_to_key_event[key - XK_space];
|
|
} else {
|
|
switch (key) {
|
|
case XK_KP_1:
|
|
#ifdef XK_KP_End
|
|
case XK_KP_End:
|
|
#endif
|
|
key_event = BX_KEY_KP_END;
|
|
break;
|
|
|
|
case XK_KP_2:
|
|
#ifdef XK_KP_Down
|
|
case XK_KP_Down:
|
|
#endif
|
|
key_event = BX_KEY_KP_DOWN;
|
|
break;
|
|
|
|
case XK_KP_3:
|
|
#ifdef XK_KP_Page_Down
|
|
case XK_KP_Page_Down:
|
|
#endif
|
|
key_event = BX_KEY_KP_PAGE_DOWN;
|
|
break;
|
|
|
|
case XK_KP_4:
|
|
#ifdef XK_KP_Left
|
|
case XK_KP_Left:
|
|
#endif
|
|
key_event = BX_KEY_KP_LEFT;
|
|
break;
|
|
|
|
case XK_KP_5:
|
|
#ifdef XK_KP_Begin
|
|
case XK_KP_Begin:
|
|
#endif
|
|
key_event = BX_KEY_KP_5;
|
|
break;
|
|
|
|
case XK_KP_6:
|
|
#ifdef XK_KP_Right
|
|
case XK_KP_Right:
|
|
#endif
|
|
key_event = BX_KEY_KP_RIGHT;
|
|
break;
|
|
|
|
case XK_KP_7:
|
|
#ifdef XK_KP_Home
|
|
case XK_KP_Home:
|
|
#endif
|
|
key_event = BX_KEY_KP_HOME;
|
|
break;
|
|
|
|
case XK_KP_8:
|
|
#ifdef XK_KP_Up
|
|
case XK_KP_Up:
|
|
#endif
|
|
key_event = BX_KEY_KP_UP;
|
|
break;
|
|
|
|
case XK_KP_9:
|
|
#ifdef XK_KP_Page_Up
|
|
case XK_KP_Page_Up:
|
|
#endif
|
|
key_event = BX_KEY_KP_PAGE_UP;
|
|
break;
|
|
|
|
case XK_KP_0:
|
|
#ifdef XK_KP_Insert
|
|
case XK_KP_Insert:
|
|
#endif
|
|
key_event = BX_KEY_KP_INSERT;
|
|
break;
|
|
|
|
case XK_KP_Decimal:
|
|
#ifdef XK_KP_Delete
|
|
case XK_KP_Delete:
|
|
#endif
|
|
key_event = BX_KEY_KP_DELETE;
|
|
break;
|
|
|
|
#ifdef XK_KP_Enter
|
|
case XK_KP_Enter:
|
|
key_event = BX_KEY_KP_ENTER;
|
|
break;
|
|
#endif
|
|
|
|
case XK_KP_Subtract:
|
|
key_event = BX_KEY_KP_SUBTRACT;
|
|
break;
|
|
case XK_KP_Add:
|
|
key_event = BX_KEY_KP_ADD;
|
|
break;
|
|
|
|
case XK_KP_Multiply:
|
|
key_event = BX_KEY_KP_MULTIPLY;
|
|
break;
|
|
case XK_KP_Divide:
|
|
key_event = BX_KEY_KP_DIVIDE;
|
|
break;
|
|
|
|
case XK_Up:
|
|
key_event = BX_KEY_UP;
|
|
break;
|
|
case XK_Down:
|
|
key_event = BX_KEY_DOWN;
|
|
break;
|
|
case XK_Left:
|
|
key_event = BX_KEY_LEFT;
|
|
break;
|
|
case XK_Right:
|
|
key_event = BX_KEY_RIGHT;
|
|
break;
|
|
|
|
case XK_Delete:
|
|
key_event = BX_KEY_DELETE;
|
|
break;
|
|
case XK_BackSpace:
|
|
key_event = BX_KEY_BACKSPACE;
|
|
break;
|
|
case XK_Tab:
|
|
key_event = BX_KEY_TAB;
|
|
break;
|
|
#ifdef XK_ISO_Left_Tab
|
|
case XK_ISO_Left_Tab:
|
|
key_event = BX_KEY_TAB;
|
|
break;
|
|
#endif
|
|
case XK_Return:
|
|
key_event = BX_KEY_ENTER;
|
|
break;
|
|
case XK_Escape:
|
|
key_event = BX_KEY_ESC;
|
|
break;
|
|
case XK_F1:
|
|
key_event = BX_KEY_F1;
|
|
break;
|
|
case XK_F2:
|
|
key_event = BX_KEY_F2;
|
|
break;
|
|
case XK_F3:
|
|
key_event = BX_KEY_F3;
|
|
break;
|
|
case XK_F4:
|
|
key_event = BX_KEY_F4;
|
|
break;
|
|
case XK_F5:
|
|
key_event = BX_KEY_F5;
|
|
break;
|
|
case XK_F6:
|
|
key_event = BX_KEY_F6;
|
|
break;
|
|
case XK_F7:
|
|
key_event = BX_KEY_F7;
|
|
break;
|
|
case XK_F8:
|
|
key_event = BX_KEY_F8;
|
|
break;
|
|
case XK_F9:
|
|
key_event = BX_KEY_F9;
|
|
break;
|
|
case XK_F10:
|
|
key_event = BX_KEY_F10;
|
|
break;
|
|
case XK_F11:
|
|
key_event = BX_KEY_F11;
|
|
break;
|
|
case XK_F12:
|
|
key_event = BX_KEY_F12;
|
|
break;
|
|
case XK_Control_L:
|
|
key_event = BX_KEY_CTRL_L;
|
|
break;
|
|
#ifdef XK_Control_R
|
|
case XK_Control_R:
|
|
key_event = BX_KEY_CTRL_R;
|
|
break;
|
|
#endif
|
|
case XK_Shift_L:
|
|
key_event = BX_KEY_SHIFT_L;
|
|
break;
|
|
case XK_Shift_R:
|
|
key_event = BX_KEY_SHIFT_R;
|
|
break;
|
|
case XK_Alt_L:
|
|
key_event = BX_KEY_ALT_L;
|
|
break;
|
|
#ifdef XK_Alt_R
|
|
case XK_Alt_R:
|
|
key_event = BX_KEY_ALT_R;
|
|
break;
|
|
#endif
|
|
case XK_Caps_Lock:
|
|
key_event = BX_KEY_CAPS_LOCK;
|
|
break;
|
|
case XK_Num_Lock:
|
|
key_event = BX_KEY_NUM_LOCK;
|
|
break;
|
|
#ifdef XK_Scroll_Lock
|
|
case XK_Scroll_Lock:
|
|
key_event = BX_KEY_SCRL_LOCK;
|
|
break;
|
|
#endif
|
|
#ifdef XK_Print
|
|
case XK_Print:
|
|
key_event = BX_KEY_PRINT;
|
|
break;
|
|
#endif
|
|
#ifdef XK_Pause
|
|
case XK_Pause:
|
|
key_event = BX_KEY_PAUSE;
|
|
break;
|
|
#endif
|
|
|
|
case XK_Insert:
|
|
key_event = BX_KEY_INSERT;
|
|
break;
|
|
case XK_Home:
|
|
key_event = BX_KEY_HOME;
|
|
break;
|
|
case XK_End:
|
|
key_event = BX_KEY_END;
|
|
break;
|
|
case XK_Page_Up:
|
|
key_event = BX_KEY_PAGE_UP;
|
|
break;
|
|
case XK_Page_Down:
|
|
key_event = BX_KEY_PAGE_DOWN;
|
|
break;
|
|
|
|
default:
|
|
BX_ERROR(("rfbKeyPress(): key %04x unhandled!", key));
|
|
return;
|
|
}
|
|
}
|
|
} else {
|
|
BXKeyEntry *entry = bx_keymap.findHostKey(key);
|
|
if (!entry) {
|
|
BX_ERROR(("vncKeyPressed(): key %x unhandled!", (unsigned) key));
|
|
return;
|
|
}
|
|
key_event = entry->baseKey;
|
|
}
|
|
|
|
if (!press_release)
|
|
key_event |= BX_KEY_RELEASED;
|
|
DEV_kbd_gen_scancode(key_event);
|
|
}
|
|
|
|
void vncMouseMove(int x, int y, int z, int bmask)
|
|
{
|
|
static int oldx = -1;
|
|
static int oldy = -1;
|
|
int dx, dy, xorigin;
|
|
|
|
if ((oldx == 1) && (oldy == -1)) {
|
|
oldx = x;
|
|
oldy = y;
|
|
return;
|
|
}
|
|
if (y > rfbHeaderbarY) {
|
|
if (rfbMouseModeAbsXY) {
|
|
if ((y >= rfbHeaderbarY) && (y < (int)(rfbDimensionY + rfbHeaderbarY))) {
|
|
dx = x * 0x7fff / rfbDimensionX;
|
|
dy = (y - rfbHeaderbarY) * 0x7fff / rfbDimensionY;
|
|
DEV_mouse_motion(dx, dy, z, bmask, 1);
|
|
}
|
|
} else {
|
|
DEV_mouse_motion(x - oldx, oldy - y, z, bmask, 0);
|
|
}
|
|
oldx = x;
|
|
oldy = y;
|
|
} else {
|
|
if (bmask == 1) {
|
|
for (unsigned i = 0; i < rfbHeaderbarBitmapCount; i++) {
|
|
if (rfbHeaderbarBitmaps[i].alignment == BX_GRAVITY_LEFT)
|
|
xorigin = rfbHeaderbarBitmaps[i].xorigin;
|
|
else
|
|
xorigin = rfbWindowX - rfbHeaderbarBitmaps[i].xorigin;
|
|
if ((x >= xorigin) &&
|
|
(x < (xorigin + int(rfbBitmaps[rfbHeaderbarBitmaps[i].index].xdim)))) {
|
|
rfbHeaderbarBitmaps[i].f();
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void newframebuffer(rfbScreenInfoPtr screen, int width, int height)
|
|
{
|
|
unsigned char *oldfb, *newfb;
|
|
|
|
oldfb = (unsigned char*) screen->frameBuffer;
|
|
newfb = (unsigned char*) malloc(width * height * sizeof(rfbPixel));
|
|
rfbNewFramebuffer(screen, (char*) newfb, width, height, 8, 3,
|
|
sizeof(rfbPixel));
|
|
free(oldfb);
|
|
|
|
/*** FIXME: Re-install cursor. ***/
|
|
}
|
|
|
|
/* Here the key events are handled */
|
|
|
|
void dokey(rfbBool down, rfbKeySym key, rfbClientPtr cl)
|
|
{
|
|
bx_bool mouse_toggle = 0;
|
|
|
|
if ((key == XK_Control_L) || (key == XK_Control_R)) {
|
|
mouse_toggle = bx_gui->mouse_toggle_check(BX_MT_KEY_CTRL, down);
|
|
} else if (key == XK_Alt_L) {
|
|
mouse_toggle = bx_gui->mouse_toggle_check(BX_MT_KEY_ALT, down);
|
|
} else if (key == XK_F10) {
|
|
mouse_toggle = bx_gui->mouse_toggle_check(BX_MT_KEY_F10, down);
|
|
} else if (key == XK_F12) {
|
|
mouse_toggle = bx_gui->mouse_toggle_check(BX_MT_KEY_F12, down);
|
|
}
|
|
if (mouse_toggle) {
|
|
bx_gui->toggle_mouse_enable();
|
|
} else {
|
|
LOCK(bKeyboardInUse);
|
|
if (rfbKeyboardEvents >= MAX_KEY_EVENTS)
|
|
return;
|
|
rfbKeyboardEvent[rfbKeyboardEvents].type = KEYBOARD;
|
|
rfbKeyboardEvent[rfbKeyboardEvents].key = key;
|
|
rfbKeyboardEvent[rfbKeyboardEvents].down = down;
|
|
rfbKeyboardEvents++;
|
|
UNLOCK(bKeyboardInUse);
|
|
}
|
|
}
|
|
|
|
void doptr(int buttonMask, int x, int y, rfbClientPtr cl)
|
|
{
|
|
static Bit8u wheel_status = 0;
|
|
|
|
if (bx_gui->mouse_toggle_check(BX_MT_MBUTTON, (buttonMask & 0x02) > 0)) {
|
|
bx_gui->toggle_mouse_enable();
|
|
} else {
|
|
LOCK(bKeyboardInUse);
|
|
if (rfbKeyboardEvents >= MAX_KEY_EVENTS)
|
|
return;
|
|
rfbKeyboardEvent[rfbKeyboardEvents].type = MOUSE;
|
|
rfbKeyboardEvent[rfbKeyboardEvents].x = x;
|
|
rfbKeyboardEvent[rfbKeyboardEvents].y = y;
|
|
rfbKeyboardEvent[rfbKeyboardEvents].z = 0;
|
|
rfbKeyboardEvent[rfbKeyboardEvents].down = (buttonMask & 0x01)
|
|
| ((buttonMask >> 1) & 0x02) | ((buttonMask << 1) & 0x04);
|
|
|
|
if ((buttonMask & 0x18) != wheel_status) {
|
|
if (buttonMask & 0x10) {
|
|
rfbKeyboardEvent[rfbKeyboardEvents].z = -1;
|
|
} else if (buttonMask & 0x08) {
|
|
rfbKeyboardEvent[rfbKeyboardEvents].z = 1;
|
|
}
|
|
wheel_status = buttonMask & 0x18;
|
|
}
|
|
rfbKeyboardEvents++;
|
|
UNLOCK(bKeyboardInUse);
|
|
}
|
|
}
|
|
|
|
void clientgone(rfbClientPtr cl)
|
|
{
|
|
client_connected = 0;
|
|
}
|
|
|
|
enum rfbNewClientAction newclient(rfbClientPtr cl)
|
|
{
|
|
cl->clientData = NULL;
|
|
cl->clientGoneHook = clientgone;
|
|
client_connected = 1;
|
|
return RFB_CLIENT_ACCEPT;
|
|
}
|
|
|
|
#if BX_SHOW_IPS && defined(WIN32)
|
|
VOID CALLBACK vncIPSTimerProc(HWND hWnd, UINT nMsg, UINT_PTR nIDEvent, DWORD dwTime)
|
|
{
|
|
if (!rfbServerDown) {
|
|
bx_show_ips_handler();
|
|
}
|
|
}
|
|
|
|
DWORD WINAPI vncShowIPSthread(LPVOID)
|
|
{
|
|
MSG msg;
|
|
|
|
UINT TimerId = SetTimer(NULL, 0, 1000, &vncIPSTimerProc);
|
|
while (!rfbServerDown && GetMessage(&msg, NULL, 0, 0)) {
|
|
DispatchMessage(&msg);
|
|
}
|
|
KillTimer(NULL, TimerId);
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
#endif /* if BX_WITH_VNCSRV */
|