2014-06-28 00:53:20 +04:00
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
// $Id$
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
2017-01-03 23:59:59 +03:00
|
|
|
// Copyright (C) 2014-2017 The Bochs Project
|
2014-06-28 00:53:20 +04:00
|
|
|
//
|
|
|
|
// 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
|
|
|
|
|
|
|
|
#define _MULTI_THREAD
|
|
|
|
|
|
|
|
// 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 "bochs.h"
|
|
|
|
#include "param_names.h"
|
2014-06-28 17:25:52 +04:00
|
|
|
#include "keymap.h"
|
2014-06-28 00:53:20 +04:00
|
|
|
#include "iodev.h"
|
|
|
|
#if BX_WITH_SDL2
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <SDL.h>
|
|
|
|
#include <SDL_endian.h>
|
|
|
|
|
|
|
|
#include "icon_bochs.h"
|
|
|
|
#include "sdl.h"
|
|
|
|
#ifdef WIN32
|
|
|
|
#include "win32dialog.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
class bx_sdl2_gui_c : public bx_gui_c {
|
|
|
|
public:
|
2014-07-08 19:46:46 +04:00
|
|
|
bx_sdl2_gui_c();
|
2014-06-28 00:53:20 +04:00
|
|
|
DECLARE_GUI_VIRTUAL_METHODS()
|
|
|
|
DECLARE_GUI_NEW_VIRTUAL_METHODS()
|
2014-07-04 00:17:19 +04:00
|
|
|
virtual void set_display_mode(disp_mode_t newmode);
|
2014-06-28 00:53:20 +04:00
|
|
|
virtual void statusbar_setitem_specific(int element, bx_bool active, bx_bool w);
|
2014-07-04 00:17:19 +04:00
|
|
|
virtual void get_capabilities(Bit16u *xres, Bit16u *yres, Bit16u *bpp);
|
2014-06-28 00:53:20 +04:00
|
|
|
virtual void set_mouse_mode_absxy(bx_bool mode);
|
|
|
|
#if BX_SHOW_IPS
|
|
|
|
virtual void show_ips(Bit32u ips_count);
|
|
|
|
#endif
|
2017-01-14 00:37:06 +03:00
|
|
|
virtual void set_console_edit_mode(bx_bool mode);
|
2014-06-28 00:53:20 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
// declare one instance of the gui object and call macro to insert the
|
|
|
|
// plugin code
|
|
|
|
static bx_sdl2_gui_c *theGui = NULL;
|
|
|
|
IMPLEMENT_GUI_PLUGIN_CODE(sdl2)
|
|
|
|
|
|
|
|
#define LOG_THIS theGui->
|
|
|
|
|
|
|
|
#if SDL_BYTEORDER == SDL_BIG_ENDIAN
|
|
|
|
const Uint32 status_led_green = 0x00ff0000;
|
|
|
|
const Uint32 status_gray_text = 0x80808000;
|
|
|
|
const Uint32 status_led_red = 0x0040ff00;
|
|
|
|
#else
|
|
|
|
const Uint32 status_led_green = 0x0000ff00;
|
|
|
|
const Uint32 status_gray_text = 0x00808080;
|
|
|
|
const Uint32 status_led_red = 0x00ff4000;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static unsigned prev_cursor_x=0;
|
|
|
|
static unsigned prev_cursor_y=0;
|
2014-06-28 17:25:52 +04:00
|
|
|
static Bit32u convertStringToSDLKey(const char *string);
|
2014-06-28 00:53:20 +04:00
|
|
|
|
|
|
|
#define MAX_SDL_BITMAPS 32
|
|
|
|
struct bitmaps {
|
|
|
|
SDL_Surface *surface;
|
2017-01-14 16:10:20 +03:00
|
|
|
SDL_Rect src, dst;
|
2014-06-28 00:53:20 +04:00
|
|
|
};
|
|
|
|
|
2014-06-28 12:18:26 +04:00
|
|
|
static SDL_Window *window;
|
2014-07-04 00:17:19 +04:00
|
|
|
SDL_Surface *sdl_screen, *sdl_fullscreen;
|
|
|
|
SDL_DisplayMode sdl_maxres;
|
|
|
|
int sdl_fullscreen_toggle;
|
2014-06-28 00:53:20 +04:00
|
|
|
int sdl_grab;
|
|
|
|
unsigned res_x, res_y;
|
|
|
|
unsigned half_res_x, half_res_y;
|
|
|
|
int headerbar_height;
|
|
|
|
static unsigned bx_bitmap_left_xorigin = 0; // pixels from left
|
|
|
|
static unsigned bx_bitmap_right_xorigin = 0; // pixels from right
|
|
|
|
static unsigned int text_rows = 25, text_cols = 80;
|
|
|
|
Bit8u h_panning = 0, v_panning = 0;
|
|
|
|
Bit16u line_compare = 1023;
|
|
|
|
int fontwidth = 8, fontheight = 16;
|
|
|
|
static unsigned disp_bpp=8;
|
|
|
|
unsigned char menufont[256][8];
|
|
|
|
Uint32 sdl_palette[256];
|
|
|
|
Uint32 headerbar_fg, headerbar_bg;
|
|
|
|
Bit8u old_mousebuttons=0, new_mousebuttons=0;
|
|
|
|
int old_mousex=0, new_mousex=0;
|
|
|
|
int old_mousey=0, new_mousey=0;
|
|
|
|
bx_bool just_warped = 0;
|
|
|
|
bx_bool sdl_mouse_mode_absxy = 0;
|
2014-06-28 12:18:26 +04:00
|
|
|
bx_bool sdl_nokeyrepeat = 0;
|
2014-06-28 00:53:20 +04:00
|
|
|
bitmaps *sdl_bitmaps[MAX_SDL_BITMAPS];
|
|
|
|
int n_sdl_bitmaps = 0;
|
|
|
|
int statusbar_height = 18;
|
|
|
|
static unsigned statusitem_pos[12] = {
|
2015-10-23 21:39:38 +03:00
|
|
|
0, 170, 220, 270, 320, 370, 420, 470, 520, 570, 620, 670
|
2014-06-28 00:53:20 +04:00
|
|
|
};
|
|
|
|
static bx_bool statusitem_active[12];
|
|
|
|
#if BX_SHOW_IPS
|
2014-06-29 11:37:58 +04:00
|
|
|
SDL_TimerID timer_id;
|
2014-06-28 00:53:20 +04:00
|
|
|
static bx_bool sdl_hide_ips = 0;
|
|
|
|
static bx_bool sdl_ips_update = 0;
|
|
|
|
static char sdl_ips_text[20];
|
|
|
|
#endif
|
2014-06-29 11:37:58 +04:00
|
|
|
#ifndef WIN32
|
|
|
|
BxEvent *sdl2_notify_callback(void *unused, BxEvent *event);
|
|
|
|
static bxevent_handler old_callback = NULL;
|
|
|
|
static void *old_callback_arg = NULL;
|
|
|
|
#endif
|
2014-06-28 00:53:20 +04:00
|
|
|
|
|
|
|
|
|
|
|
static void sdl_set_status_text(int element, const char *text, bx_bool active, bx_bool w = 0)
|
|
|
|
{
|
|
|
|
Uint32 *buf, *buf_row;
|
|
|
|
Uint32 disp, fgcolor, bgcolor;
|
|
|
|
unsigned char *pfont_row, font_row;
|
|
|
|
int rowsleft = statusbar_height - 2;
|
|
|
|
int colsleft, textlen;
|
|
|
|
int x, xleft, xsize;
|
2014-07-14 11:58:42 +04:00
|
|
|
SDL_Rect item;
|
2014-06-28 00:53:20 +04:00
|
|
|
|
|
|
|
statusitem_active[element] = active;
|
2014-07-04 00:17:19 +04:00
|
|
|
if (!sdl_screen) return;
|
2014-06-28 00:53:20 +04:00
|
|
|
disp = sdl_screen->pitch/4;
|
|
|
|
xleft = statusitem_pos[element] + 2;
|
|
|
|
xsize = statusitem_pos[element+1] - xleft - 1;
|
|
|
|
buf = (Uint32 *)sdl_screen->pixels + (res_y + headerbar_height + 1) * disp + xleft;
|
|
|
|
rowsleft = statusbar_height - 2;
|
|
|
|
fgcolor = active?headerbar_fg:status_gray_text;
|
|
|
|
if (element > 0) {
|
|
|
|
bgcolor = active?(w?status_led_red:status_led_green):headerbar_bg;
|
|
|
|
} else {
|
|
|
|
bgcolor = headerbar_bg;
|
|
|
|
}
|
|
|
|
do {
|
|
|
|
colsleft = xsize;
|
|
|
|
buf_row = buf;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
*buf++ = bgcolor;
|
|
|
|
} while(--colsleft);
|
|
|
|
buf = buf_row + disp;
|
|
|
|
} while(--rowsleft);
|
2015-10-23 21:39:38 +03:00
|
|
|
if ((element > 0) && (strlen(text) > 6)) {
|
|
|
|
textlen = 6;
|
2014-06-28 00:53:20 +04:00
|
|
|
} else {
|
|
|
|
textlen = strlen(text);
|
|
|
|
}
|
|
|
|
buf = (Uint32 *)sdl_screen->pixels + (res_y + headerbar_height + 5) * disp + xleft;
|
|
|
|
x = 0;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
pfont_row = &menufont[(unsigned)text[x]][0];
|
|
|
|
buf_row = buf;
|
|
|
|
rowsleft = 8;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
font_row = *pfont_row++;
|
|
|
|
colsleft = 8;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if((font_row & 0x80) != 0x00)
|
|
|
|
*buf++ = fgcolor;
|
|
|
|
else
|
|
|
|
buf++;
|
|
|
|
font_row <<= 1;
|
|
|
|
} while(--colsleft);
|
|
|
|
buf += (disp - 8);
|
|
|
|
} while(--rowsleft);
|
|
|
|
buf = buf_row + 8;
|
|
|
|
x++;
|
|
|
|
} while (--textlen);
|
2014-07-14 11:58:42 +04:00
|
|
|
item.x = xleft;
|
|
|
|
item.y = res_y + headerbar_height + 1;
|
|
|
|
item.w = xsize;
|
|
|
|
item.h = statusbar_height - 2;
|
|
|
|
SDL_UpdateWindowSurfaceRects(window, &item, 1);
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static Bit32u sdl_sym_to_bx_key(SDL_Keycode sym)
|
|
|
|
{
|
|
|
|
switch (sym) {
|
|
|
|
case SDLK_UNKNOWN: return BX_KEY_UNHANDLED;
|
|
|
|
case SDLK_BACKSPACE: return BX_KEY_BACKSPACE;
|
|
|
|
case SDLK_TAB: return BX_KEY_TAB;
|
|
|
|
case SDLK_RETURN: return BX_KEY_ENTER;
|
|
|
|
case SDLK_PAUSE: return BX_KEY_PAUSE;
|
|
|
|
case SDLK_ESCAPE: return BX_KEY_ESC;
|
|
|
|
case SDLK_SPACE: return BX_KEY_SPACE;
|
|
|
|
case SDLK_QUOTE: return BX_KEY_SINGLE_QUOTE;
|
|
|
|
case SDLK_COMMA: return BX_KEY_COMMA;
|
|
|
|
case SDLK_MINUS: return BX_KEY_MINUS;
|
|
|
|
case SDLK_PERIOD: return BX_KEY_PERIOD;
|
|
|
|
case SDLK_SLASH: return BX_KEY_SLASH;
|
|
|
|
case SDLK_0: return BX_KEY_0;
|
|
|
|
case SDLK_1: return BX_KEY_1;
|
|
|
|
case SDLK_2: return BX_KEY_2;
|
|
|
|
case SDLK_3: return BX_KEY_3;
|
|
|
|
case SDLK_4: return BX_KEY_4;
|
|
|
|
case SDLK_5: return BX_KEY_5;
|
|
|
|
case SDLK_6: return BX_KEY_6;
|
|
|
|
case SDLK_7: return BX_KEY_7;
|
|
|
|
case SDLK_8: return BX_KEY_8;
|
|
|
|
case SDLK_9: return BX_KEY_9;
|
|
|
|
case SDLK_SEMICOLON: return BX_KEY_SEMICOLON;
|
|
|
|
case SDLK_EQUALS: return BX_KEY_EQUALS;
|
|
|
|
/*
|
|
|
|
Skip uppercase letters
|
|
|
|
*/
|
|
|
|
case SDLK_LEFTBRACKET: return BX_KEY_LEFT_BRACKET;
|
|
|
|
case SDLK_BACKSLASH: return BX_KEY_BACKSLASH;
|
|
|
|
case SDLK_RIGHTBRACKET: return BX_KEY_RIGHT_BRACKET;
|
|
|
|
case SDLK_BACKQUOTE: return BX_KEY_GRAVE;
|
|
|
|
case SDLK_a: return BX_KEY_A;
|
|
|
|
case SDLK_b: return BX_KEY_B;
|
|
|
|
case SDLK_c: return BX_KEY_C;
|
|
|
|
case SDLK_d: return BX_KEY_D;
|
|
|
|
case SDLK_e: return BX_KEY_E;
|
|
|
|
case SDLK_f: return BX_KEY_F;
|
|
|
|
case SDLK_g: return BX_KEY_G;
|
|
|
|
case SDLK_h: return BX_KEY_H;
|
|
|
|
case SDLK_i: return BX_KEY_I;
|
|
|
|
case SDLK_j: return BX_KEY_J;
|
|
|
|
case SDLK_k: return BX_KEY_K;
|
|
|
|
case SDLK_l: return BX_KEY_L;
|
|
|
|
case SDLK_m: return BX_KEY_M;
|
|
|
|
case SDLK_n: return BX_KEY_N;
|
|
|
|
case SDLK_o: return BX_KEY_O;
|
|
|
|
case SDLK_p: return BX_KEY_P;
|
|
|
|
case SDLK_q: return BX_KEY_Q;
|
|
|
|
case SDLK_r: return BX_KEY_R;
|
|
|
|
case SDLK_s: return BX_KEY_S;
|
|
|
|
case SDLK_t: return BX_KEY_T;
|
|
|
|
case SDLK_u: return BX_KEY_U;
|
|
|
|
case SDLK_v: return BX_KEY_V;
|
|
|
|
case SDLK_w: return BX_KEY_W;
|
|
|
|
case SDLK_x: return BX_KEY_X;
|
|
|
|
case SDLK_y: return BX_KEY_Y;
|
|
|
|
case SDLK_z: return BX_KEY_Z;
|
|
|
|
case SDLK_DELETE: return BX_KEY_DELETE;
|
|
|
|
/* End of ASCII mapped keysyms */
|
|
|
|
|
|
|
|
/* Numeric keypad */
|
|
|
|
case SDLK_KP_0: return BX_KEY_KP_INSERT;
|
|
|
|
case SDLK_KP_1: return BX_KEY_KP_END;
|
|
|
|
case SDLK_KP_2: return BX_KEY_KP_DOWN;
|
|
|
|
case SDLK_KP_3: return BX_KEY_KP_PAGE_DOWN;
|
|
|
|
case SDLK_KP_4: return BX_KEY_KP_LEFT;
|
|
|
|
case SDLK_KP_5: return BX_KEY_KP_5;
|
|
|
|
case SDLK_KP_6: return BX_KEY_KP_RIGHT;
|
|
|
|
case SDLK_KP_7: return BX_KEY_KP_HOME;
|
|
|
|
case SDLK_KP_8: return BX_KEY_KP_UP;
|
|
|
|
case SDLK_KP_9: return BX_KEY_KP_PAGE_UP;
|
|
|
|
case SDLK_KP_PERIOD: return BX_KEY_KP_DELETE;
|
|
|
|
case SDLK_KP_DIVIDE: return BX_KEY_KP_DIVIDE;
|
|
|
|
case SDLK_KP_MULTIPLY: return BX_KEY_KP_MULTIPLY;
|
|
|
|
case SDLK_KP_MINUS: return BX_KEY_KP_SUBTRACT;
|
|
|
|
case SDLK_KP_PLUS: return BX_KEY_KP_ADD;
|
|
|
|
case SDLK_KP_ENTER: return BX_KEY_KP_ENTER;
|
|
|
|
|
|
|
|
/* Arrows + Home/End pad */
|
|
|
|
case SDLK_UP: return BX_KEY_UP;
|
|
|
|
case SDLK_DOWN: return BX_KEY_DOWN;
|
|
|
|
case SDLK_RIGHT: return BX_KEY_RIGHT;
|
|
|
|
case SDLK_LEFT: return BX_KEY_LEFT;
|
|
|
|
case SDLK_INSERT: return BX_KEY_INSERT;
|
|
|
|
case SDLK_HOME: return BX_KEY_HOME;
|
|
|
|
case SDLK_END: return BX_KEY_END;
|
|
|
|
case SDLK_PAGEUP: return BX_KEY_PAGE_UP;
|
|
|
|
case SDLK_PAGEDOWN: return BX_KEY_PAGE_DOWN;
|
|
|
|
|
|
|
|
/* Function keys */
|
|
|
|
case SDLK_F1: return BX_KEY_F1;
|
|
|
|
case SDLK_F2: return BX_KEY_F2;
|
|
|
|
case SDLK_F3: return BX_KEY_F3;
|
|
|
|
case SDLK_F4: return BX_KEY_F4;
|
|
|
|
case SDLK_F5: return BX_KEY_F5;
|
|
|
|
case SDLK_F6: return BX_KEY_F6;
|
|
|
|
case SDLK_F7: return BX_KEY_F7;
|
|
|
|
case SDLK_F8: return BX_KEY_F8;
|
|
|
|
case SDLK_F9: return BX_KEY_F9;
|
|
|
|
case SDLK_F10: return BX_KEY_F10;
|
|
|
|
case SDLK_F11: return BX_KEY_F11;
|
|
|
|
case SDLK_F12: return BX_KEY_F12;
|
|
|
|
|
|
|
|
/* Key state modifier keys */
|
|
|
|
case SDLK_NUMLOCKCLEAR: return BX_KEY_NUM_LOCK;
|
|
|
|
case SDLK_CAPSLOCK: return BX_KEY_CAPS_LOCK;
|
|
|
|
case SDLK_SCROLLLOCK: return BX_KEY_SCRL_LOCK;
|
|
|
|
case SDLK_RSHIFT: return BX_KEY_SHIFT_R;
|
|
|
|
case SDLK_LSHIFT: return BX_KEY_SHIFT_L;
|
|
|
|
case SDLK_RCTRL: return BX_KEY_CTRL_R;
|
|
|
|
case SDLK_LCTRL: return BX_KEY_CTRL_L;
|
|
|
|
case SDLK_RALT: return BX_KEY_ALT_R;
|
|
|
|
case SDLK_LALT: return BX_KEY_ALT_L;
|
|
|
|
case SDLK_RGUI: return BX_KEY_WIN_R;
|
|
|
|
case SDLK_LGUI: return BX_KEY_WIN_L;
|
|
|
|
|
|
|
|
/* Miscellaneous function keys */
|
|
|
|
case SDLK_PRINTSCREEN: return BX_KEY_PRINT;
|
|
|
|
case SDLK_MENU: return BX_KEY_MENU;
|
|
|
|
|
|
|
|
default:
|
|
|
|
BX_ERROR(("sdl keysym %d not mapped", (int)sym));
|
|
|
|
return BX_KEY_UNHANDLED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-07-04 00:17:19 +04:00
|
|
|
void switch_to_windowed(void)
|
|
|
|
{
|
|
|
|
SDL_SetWindowFullscreen(window, 0);
|
|
|
|
SDL_SetWindowSize(window, res_x, res_y + headerbar_height + statusbar_height);
|
|
|
|
sdl_screen = SDL_GetWindowSurface(window);
|
|
|
|
sdl_fullscreen = NULL;
|
|
|
|
bx_gui->show_headerbar();
|
|
|
|
DEV_vga_redraw_area(0, 0, res_x, res_y);
|
2014-07-05 12:00:01 +04:00
|
|
|
if (sdl_grab) {
|
|
|
|
bx_gui->toggle_mouse_enable();
|
|
|
|
}
|
2014-07-04 00:17:19 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void switch_to_fullscreen(void)
|
|
|
|
{
|
2014-07-05 12:00:01 +04:00
|
|
|
if (!sdl_grab) {
|
|
|
|
bx_gui->toggle_mouse_enable();
|
|
|
|
}
|
2014-07-04 00:17:19 +04:00
|
|
|
SDL_SetWindowSize(window, res_x, res_y);
|
|
|
|
SDL_SetWindowFullscreen(window, SDL_WINDOW_FULLSCREEN);
|
|
|
|
sdl_fullscreen = SDL_GetWindowSurface(window);
|
|
|
|
sdl_screen = NULL;
|
|
|
|
DEV_vga_redraw_area(0, 0, res_x, res_y);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-28 00:53:20 +04:00
|
|
|
#if BX_SHOW_IPS
|
|
|
|
#if defined(__MINGW32__) || defined(_MSC_VER)
|
2014-06-29 03:49:10 +04:00
|
|
|
Uint32 sdlTimer(Uint32 interval, void *param)
|
2014-06-28 00:53:20 +04:00
|
|
|
{
|
|
|
|
bx_show_ips_handler();
|
|
|
|
return interval;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#if defined(WIN32) && BX_DEBUGGER && BX_DEBUGGER_GUI
|
|
|
|
DWORD WINAPI DebugGuiThread(LPVOID)
|
|
|
|
{
|
|
|
|
MSG msg;
|
|
|
|
|
|
|
|
bx_gui->init_debug_dialog();
|
|
|
|
while (GetMessage(&msg, NULL, 0, 0)) {
|
|
|
|
TranslateMessage(&msg);
|
|
|
|
DispatchMessage(&msg);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2014-07-08 19:46:46 +04:00
|
|
|
bx_sdl2_gui_c::bx_sdl2_gui_c()
|
|
|
|
{
|
|
|
|
Uint32 flags;
|
|
|
|
|
|
|
|
put("SDL2");
|
|
|
|
flags = SDL_INIT_VIDEO;
|
|
|
|
#if BX_SHOW_IPS
|
|
|
|
#if defined(__MINGW32__) || defined(_MSC_VER)
|
|
|
|
flags |= SDL_INIT_TIMER;
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
if (SDL_Init(flags) < 0) {
|
2017-01-25 00:52:19 +03:00
|
|
|
BX_FATAL(("Unable to initialize SDL2 libraries"));
|
2014-07-08 19:46:46 +04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
atexit(SDL_Quit);
|
|
|
|
SDL_GetDisplayMode(0, 0, &sdl_maxres);
|
2014-12-27 18:57:30 +03:00
|
|
|
info("maximum host resolution: x=%d y=%d", sdl_maxres.w, sdl_maxres.h);
|
2014-07-08 19:46:46 +04:00
|
|
|
}
|
|
|
|
|
2017-01-15 14:44:43 +03:00
|
|
|
// SDL2 implementation of the bx_gui_c methods (see nogui.cc for details)
|
|
|
|
|
2014-06-28 00:53:20 +04:00
|
|
|
void bx_sdl2_gui_c::specific_init(int argc, char **argv, unsigned headerbar_y)
|
|
|
|
{
|
|
|
|
int i, j;
|
2014-06-28 17:25:52 +04:00
|
|
|
unsigned icon_id;
|
2017-01-03 23:59:59 +03:00
|
|
|
bx_bool gui_ci = 0;
|
2014-06-28 00:53:20 +04:00
|
|
|
|
2017-01-03 23:59:59 +03:00
|
|
|
#ifdef WIN32
|
2014-06-28 00:53:20 +04:00
|
|
|
gui_ci = !strcmp(SIM->get_param_enum(BXPN_SEL_CONFIG_INTERFACE)->get_selected(), "win32config");
|
|
|
|
#endif
|
|
|
|
put("SDL2");
|
|
|
|
|
|
|
|
headerbar_height = headerbar_y;
|
|
|
|
|
|
|
|
for(i=0;i<256;i++)
|
|
|
|
for(j=0;j<16;j++)
|
|
|
|
vga_charmap[i*32+j] = sdl_font8x16[i][j];
|
|
|
|
|
|
|
|
for(i=0;i<256;i++)
|
|
|
|
for(j=0;j<8;j++)
|
|
|
|
menufont[i][j] = sdl_font8x8[i][j];
|
|
|
|
|
2014-07-04 00:17:19 +04:00
|
|
|
window = SDL_CreateWindow(
|
|
|
|
BOCHS_WINDOW_NAME,
|
|
|
|
SDL_WINDOWPOS_UNDEFINED,
|
|
|
|
SDL_WINDOWPOS_UNDEFINED,
|
|
|
|
640,
|
|
|
|
480,
|
|
|
|
SDL_WINDOW_SHOWN
|
|
|
|
);
|
|
|
|
if (window == NULL) {
|
2017-01-25 00:52:19 +03:00
|
|
|
BX_FATAL(("Unable to create SDL2 window"));
|
2014-07-04 00:17:19 +04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
sdl_screen = NULL;
|
|
|
|
sdl_fullscreen_toggle = 0;
|
|
|
|
dimension_update(640, 480);
|
|
|
|
|
|
|
|
SDL_WarpMouseInWindow(window, half_res_x, half_res_y);
|
|
|
|
icon_id = create_bitmap(bochs_icon_bits, bochs_icon_width, bochs_icon_height);
|
|
|
|
SDL_SetWindowIcon(window, sdl_bitmaps[icon_id]->surface);
|
2014-06-28 00:53:20 +04:00
|
|
|
|
2014-06-29 11:37:58 +04:00
|
|
|
#ifndef WIN32
|
|
|
|
// redirect notify callback to SDL2 specific code
|
|
|
|
SIM->get_notify_callback(&old_callback, &old_callback_arg);
|
|
|
|
assert(old_callback != NULL);
|
|
|
|
SIM->set_notify_callback(sdl2_notify_callback, NULL);
|
|
|
|
#endif
|
|
|
|
|
2014-06-28 17:25:52 +04:00
|
|
|
// load keymap for sdl
|
|
|
|
if (SIM->get_param_bool(BXPN_KBD_USEMAPPING)->get()) {
|
|
|
|
bx_keymap.loadKeymap(convertStringToSDLKey);
|
|
|
|
}
|
|
|
|
|
2014-06-28 00:53:20 +04:00
|
|
|
// parse sdl specific options
|
|
|
|
if (argc > 1) {
|
|
|
|
for (i = 1; i < argc; i++) {
|
2014-07-04 00:17:19 +04:00
|
|
|
if (!strcmp(argv[i], "fullscreen")) {
|
|
|
|
sdl_fullscreen_toggle = 1;
|
|
|
|
switch_to_fullscreen();
|
|
|
|
} else if (!strcmp(argv[i], "nokeyrepeat")) {
|
2014-06-28 12:18:26 +04:00
|
|
|
BX_INFO(("disabled host keyboard repeat"));
|
|
|
|
sdl_nokeyrepeat = 1;
|
2014-06-28 00:53:20 +04:00
|
|
|
#if BX_DEBUGGER && BX_DEBUGGER_GUI
|
|
|
|
} else if (!strcmp(argv[i], "gui_debug")) {
|
|
|
|
SIM->set_debug_gui(1);
|
|
|
|
#ifdef WIN32
|
|
|
|
if (gui_ci) {
|
|
|
|
// on Windows the debugger gui must run in a separate thread
|
|
|
|
DWORD threadID;
|
|
|
|
CreateThread(NULL, 0, DebugGuiThread, NULL, 0, &threadID);
|
|
|
|
} else {
|
|
|
|
BX_PANIC(("Config interface 'win32config' is required for gui debugger"));
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
init_debug_dialog();
|
2017-01-03 23:59:59 +03:00
|
|
|
#endif
|
2014-06-28 00:53:20 +04:00
|
|
|
#endif
|
|
|
|
#if BX_SHOW_IPS
|
|
|
|
} else if (!strcmp(argv[i], "hideIPS")) {
|
|
|
|
BX_INFO(("hide IPS display in status bar"));
|
|
|
|
sdl_hide_ips = 1;
|
|
|
|
#endif
|
|
|
|
} else {
|
|
|
|
BX_PANIC(("Unknown sdl option '%s'", argv[i]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
new_gfx_api = 1;
|
2017-01-03 23:59:59 +03:00
|
|
|
#if defined(WIN32) && BX_SHOW_IPS
|
|
|
|
timer_id = SDL_AddTimer(1000, sdlTimer, NULL);
|
|
|
|
#endif
|
2014-06-28 00:53:20 +04:00
|
|
|
if (gui_ci) {
|
|
|
|
dialog_caps = BX_GUI_DLG_ALL;
|
2017-01-03 23:59:59 +03:00
|
|
|
} else {
|
2017-01-14 00:37:06 +03:00
|
|
|
console.present = 1;
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void bx_sdl2_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 *pfont_row, *old_line, *new_line, *text_base;
|
|
|
|
unsigned int cs_y, i, x, y;
|
|
|
|
unsigned int curs, hchars, offset;
|
|
|
|
Bit8u fontline, fontpixels, fontrows;
|
|
|
|
int rows;
|
|
|
|
Uint32 fgcolor, bgcolor;
|
|
|
|
Uint32 *buf, *buf_row, *buf_char;
|
|
|
|
Uint32 disp;
|
|
|
|
Bit16u font_row, mask;
|
|
|
|
Bit8u cfstart, cfwidth, cfheight, split_fontrows, split_textrow;
|
|
|
|
bx_bool cursor_visible, gfxcharw9, invert, forceUpdate, split_screen;
|
2015-10-25 01:03:51 +03:00
|
|
|
bx_bool blink_mode, blink_state, dwidth;
|
2014-06-28 00:53:20 +04:00
|
|
|
Uint32 text_palette[16];
|
|
|
|
|
|
|
|
forceUpdate = 0;
|
|
|
|
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)
|
|
|
|
forceUpdate = 1;
|
|
|
|
}
|
2015-10-25 01:03:51 +03:00
|
|
|
dwidth = (fontwidth > 9);
|
2014-06-28 00:53:20 +04:00
|
|
|
if (charmap_updated) {
|
|
|
|
forceUpdate = 1;
|
|
|
|
charmap_updated = 0;
|
|
|
|
}
|
|
|
|
for (i=0; i<16; i++) {
|
|
|
|
text_palette[i] = sdl_palette[tm_info->actl_palette[i]];
|
|
|
|
}
|
|
|
|
if ((tm_info->h_panning != h_panning) || (tm_info->v_panning != v_panning)) {
|
|
|
|
forceUpdate = 1;
|
|
|
|
h_panning = tm_info->h_panning;
|
|
|
|
v_panning = tm_info->v_panning;
|
|
|
|
}
|
|
|
|
if (tm_info->line_compare != line_compare) {
|
|
|
|
forceUpdate = 1;
|
|
|
|
line_compare = tm_info->line_compare;
|
|
|
|
}
|
2014-07-04 00:17:19 +04:00
|
|
|
if (sdl_screen) {
|
|
|
|
disp = sdl_screen->pitch/4;
|
|
|
|
buf_row = (Uint32 *)sdl_screen->pixels + headerbar_height * disp;
|
|
|
|
} else {
|
|
|
|
disp = sdl_fullscreen->pitch/4;
|
|
|
|
buf_row = (Uint32 *)sdl_fullscreen->pixels;
|
|
|
|
}
|
2014-06-28 00:53:20 +04:00
|
|
|
// first invalidate character at previous and new cursor location
|
|
|
|
if ((prev_cursor_y < text_rows) && (prev_cursor_x < text_cols)) {
|
|
|
|
curs = prev_cursor_y * tm_info->line_offset + prev_cursor_x * 2;
|
|
|
|
old_text[curs] = ~new_text[curs];
|
|
|
|
}
|
|
|
|
cursor_visible = ((tm_info->cs_start <= tm_info->cs_end) && (tm_info->cs_start < fontheight));
|
|
|
|
if((cursor_visible) && (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;
|
|
|
|
if (v_panning) rows++;
|
|
|
|
y = 0;
|
|
|
|
cs_y = 0;
|
|
|
|
text_base = new_text - tm_info->start_address;
|
|
|
|
if (line_compare < res_y) {
|
|
|
|
split_textrow = (line_compare + v_panning) / fontheight;
|
|
|
|
split_fontrows = ((line_compare + v_panning) % fontheight) + 1;
|
|
|
|
} else {
|
|
|
|
split_textrow = rows + 1;
|
|
|
|
split_fontrows = 0;
|
|
|
|
}
|
|
|
|
split_screen = 0;
|
|
|
|
|
|
|
|
do {
|
|
|
|
buf = buf_row;
|
|
|
|
hchars = text_cols;
|
|
|
|
if (h_panning) hchars++;
|
|
|
|
cfheight = fontheight;
|
|
|
|
cfstart = 0;
|
|
|
|
if (split_screen) {
|
|
|
|
if (rows == 1) {
|
|
|
|
cfheight = (res_y - line_compare - 1) % fontheight;
|
|
|
|
if (cfheight == 0) cfheight = fontheight;
|
|
|
|
}
|
|
|
|
} else if (v_panning) {
|
|
|
|
if (y == 0) {
|
|
|
|
cfheight -= v_panning;
|
|
|
|
cfstart = v_panning;
|
|
|
|
} else if (rows == 1) {
|
|
|
|
cfheight = v_panning;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!split_screen && (y == split_textrow)) {
|
|
|
|
if ((split_fontrows - cfstart) < cfheight) {
|
|
|
|
cfheight = split_fontrows - cfstart;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
new_line = new_text;
|
|
|
|
old_line = old_text;
|
|
|
|
x = 0;
|
|
|
|
offset = cs_y * tm_info->line_offset;
|
|
|
|
do {
|
|
|
|
cfwidth = fontwidth;
|
|
|
|
if (h_panning) {
|
|
|
|
if (hchars > text_cols) {
|
|
|
|
cfwidth -= h_panning;
|
|
|
|
} else if (hchars == 1) {
|
|
|
|
cfwidth = h_panning;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// check if char needs to be updated
|
|
|
|
if(forceUpdate || (old_text[0] != new_text[0])
|
|
|
|
|| (old_text[1] != new_text[1])) {
|
|
|
|
// Get Foreground/Background pixel colors
|
|
|
|
fgcolor = text_palette[new_text[1] & 0x0F];
|
|
|
|
if (blink_mode) {
|
|
|
|
bgcolor = text_palette[(new_text[1] >> 4) & 0x07];
|
|
|
|
if (!blink_state && (new_text[1] & 0x80))
|
|
|
|
fgcolor = bgcolor;
|
|
|
|
} else {
|
|
|
|
bgcolor = text_palette[(new_text[1] >> 4) & 0x0F];
|
|
|
|
}
|
|
|
|
invert = ((offset == curs) && (cursor_visible));
|
|
|
|
gfxcharw9 = ((tm_info->line_graphics) && ((new_text[0] & 0xE0) == 0xC0));
|
|
|
|
|
|
|
|
// Display this one char
|
|
|
|
fontrows = cfheight;
|
|
|
|
fontline = cfstart;
|
|
|
|
if (y > 0) {
|
|
|
|
pfont_row = &vga_charmap[(new_text[0] << 5)];
|
|
|
|
} else {
|
|
|
|
pfont_row = &vga_charmap[(new_text[0] << 5) + cfstart];
|
|
|
|
}
|
|
|
|
buf_char = buf;
|
|
|
|
do {
|
|
|
|
font_row = *pfont_row++;
|
|
|
|
if (gfxcharw9) {
|
|
|
|
font_row = (font_row << 1) | (font_row & 0x01);
|
|
|
|
} else {
|
|
|
|
font_row <<= 1;
|
|
|
|
}
|
|
|
|
if (hchars > text_cols) {
|
|
|
|
font_row <<= h_panning;
|
|
|
|
}
|
|
|
|
fontpixels = cfwidth;
|
|
|
|
if ((invert) && (fontline >= tm_info->cs_start) && (fontline <= tm_info->cs_end))
|
|
|
|
mask = 0x100;
|
|
|
|
else
|
|
|
|
mask = 0x00;
|
|
|
|
do {
|
|
|
|
if ((font_row & 0x100) == mask)
|
|
|
|
*buf = bgcolor;
|
|
|
|
else
|
|
|
|
*buf = fgcolor;
|
|
|
|
buf++;
|
2015-10-25 01:03:51 +03:00
|
|
|
if (!dwidth || (fontpixels & 1)) font_row <<= 1;
|
2014-06-28 00:53:20 +04:00
|
|
|
} while (--fontpixels);
|
|
|
|
buf -= cfwidth;
|
|
|
|
buf += disp;
|
|
|
|
fontline++;
|
|
|
|
} while (--fontrows);
|
|
|
|
|
|
|
|
// restore output buffer ptr to start of this char
|
|
|
|
buf = buf_char;
|
|
|
|
}
|
|
|
|
// move to next char location on screen
|
|
|
|
buf += cfwidth;
|
|
|
|
|
|
|
|
// select next char in old/new text
|
|
|
|
new_text+=2;
|
|
|
|
old_text+=2;
|
|
|
|
offset+=2;
|
|
|
|
x++;
|
|
|
|
|
|
|
|
// process one entire horizontal row
|
|
|
|
} while (--hchars);
|
|
|
|
|
|
|
|
// go to next character row location
|
|
|
|
buf_row += disp * cfheight;
|
|
|
|
if (!split_screen && (y == split_textrow)) {
|
|
|
|
new_text = text_base;
|
|
|
|
forceUpdate = 1;
|
|
|
|
cs_y = 0;
|
|
|
|
if (tm_info->split_hpanning) h_panning = 0;
|
|
|
|
rows = ((res_y - line_compare + fontheight - 2) / fontheight) + 1;
|
|
|
|
split_screen = 1;
|
|
|
|
} else {
|
|
|
|
new_text = new_line + tm_info->line_offset;
|
|
|
|
old_text = old_line + tm_info->line_offset;
|
|
|
|
cs_y++;
|
|
|
|
y++;
|
|
|
|
}
|
|
|
|
} while (--rows);
|
|
|
|
h_panning = tm_info->h_panning;
|
|
|
|
prev_cursor_x = cursor_x;
|
|
|
|
prev_cursor_y = cursor_y;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void bx_sdl2_gui_c::graphics_tile_update(Bit8u *snapshot, unsigned x, unsigned y)
|
|
|
|
{
|
|
|
|
Uint32 *buf, disp;
|
|
|
|
Uint32 *buf_row;
|
2014-07-04 00:17:19 +04:00
|
|
|
int i, j;
|
2014-06-28 00:53:20 +04:00
|
|
|
|
2014-07-04 00:17:19 +04:00
|
|
|
if (sdl_screen) {
|
|
|
|
disp = sdl_screen->pitch/4;
|
|
|
|
buf = (Uint32 *)sdl_screen->pixels + (headerbar_height + y) * disp + x;
|
|
|
|
} else {
|
|
|
|
disp = sdl_fullscreen->pitch/4;
|
|
|
|
buf = (Uint32 *)sdl_fullscreen->pixels + y * disp + x;
|
|
|
|
}
|
2014-06-28 00:53:20 +04:00
|
|
|
|
|
|
|
i = y_tilesize;
|
|
|
|
if(i + y > res_y) i = res_y - y;
|
|
|
|
|
|
|
|
// FIXME
|
|
|
|
if (i<=0) return;
|
|
|
|
|
|
|
|
switch (disp_bpp) {
|
|
|
|
case 8: /* 8 bpp */
|
|
|
|
do {
|
|
|
|
buf_row = buf;
|
|
|
|
j = x_tilesize;
|
|
|
|
do {
|
|
|
|
*buf++ = sdl_palette[*snapshot++];
|
|
|
|
} while(--j);
|
|
|
|
buf = buf_row + disp;
|
|
|
|
} while(--i);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
BX_PANIC(("%u bpp modes handled by new graphics API", disp_bpp));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void bx_sdl2_gui_c::handle_events(void)
|
|
|
|
{
|
|
|
|
SDL_Event sdl_event;
|
|
|
|
Bit32u key_event;
|
|
|
|
Bit8u mouse_state;
|
|
|
|
int dx, dy, wheel_status;
|
|
|
|
bx_bool mouse_toggle = 0;
|
|
|
|
|
|
|
|
while (SDL_PollEvent(&sdl_event)) {
|
|
|
|
switch (sdl_event.type) {
|
|
|
|
case SDL_WINDOWEVENT:
|
|
|
|
if (sdl_event.window.event == SDL_WINDOWEVENT_EXPOSED) {
|
|
|
|
SDL_UpdateWindowSurface(window);
|
|
|
|
}
|
2014-12-27 18:57:30 +03:00
|
|
|
if (sdl_event.window.event == SDL_WINDOWEVENT_FOCUS_LOST) {
|
|
|
|
DEV_kbd_release_keys();
|
2014-12-26 23:25:27 +03:00
|
|
|
}
|
2014-06-28 00:53:20 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL_MOUSEMOTION:
|
2017-01-05 01:24:33 +03:00
|
|
|
if (!sdl_grab || console_running()) {
|
2014-06-28 00:53:20 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (just_warped
|
|
|
|
&& sdl_event.motion.x == (int)half_res_x
|
|
|
|
&& sdl_event.motion.y == (int)half_res_y) {
|
|
|
|
// This event was generated as a side effect of the WarpMouse,
|
|
|
|
// and it must be ignored.
|
|
|
|
just_warped = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
new_mousebuttons = ((sdl_event.motion.state & 0x01)|((sdl_event.motion.state>>1)&0x02)
|
|
|
|
|((sdl_event.motion.state<<1)&0x04));
|
|
|
|
if (sdl_mouse_mode_absxy) {
|
2014-07-05 12:00:01 +04:00
|
|
|
dx = sdl_event.motion.x * 0x7fff / res_x;
|
|
|
|
if (sdl_fullscreen_toggle) {
|
|
|
|
dy = sdl_event.motion.y * 0x7fff / res_y;
|
|
|
|
DEV_mouse_motion(dx, dy, 0, new_mousebuttons, 1);
|
|
|
|
} else if ((sdl_event.motion.y >= headerbar_height) && (sdl_event.motion.y < ((int)res_y + headerbar_height))) {
|
2014-06-28 00:53:20 +04:00
|
|
|
dy = (sdl_event.motion.y - headerbar_height) * 0x7fff / res_y;
|
2014-07-05 12:00:01 +04:00
|
|
|
DEV_mouse_motion(dx, dy, 0, new_mousebuttons, 1);
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
} else {
|
2014-07-05 12:00:01 +04:00
|
|
|
DEV_mouse_motion(sdl_event.motion.xrel, -sdl_event.motion.yrel, 0,
|
|
|
|
new_mousebuttons, 0);
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
old_mousebuttons = new_mousebuttons;
|
|
|
|
old_mousex = (int)(sdl_event.motion.x);
|
|
|
|
old_mousey = (int)(sdl_event.motion.y);
|
|
|
|
if (!sdl_mouse_mode_absxy) {
|
|
|
|
SDL_WarpMouseInWindow(window, half_res_x, half_res_y);
|
|
|
|
just_warped = 1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL_MOUSEBUTTONDOWN:
|
|
|
|
// mouse capture toggle-check
|
2014-07-05 12:00:01 +04:00
|
|
|
if ((sdl_event.button.button == SDL_BUTTON_MIDDLE)
|
|
|
|
&& (sdl_fullscreen_toggle == 0)) {
|
2014-06-28 00:53:20 +04:00
|
|
|
if (mouse_toggle_check(BX_MT_MBUTTON, 1)) {
|
|
|
|
if (sdl_grab == 0) {
|
|
|
|
SDL_ShowCursor(0);
|
|
|
|
SDL_SetWindowGrab(window, SDL_TRUE);
|
|
|
|
} else {
|
|
|
|
SDL_ShowCursor(1);
|
|
|
|
SDL_SetWindowGrab(window, SDL_FALSE);
|
|
|
|
}
|
|
|
|
sdl_grab = ~sdl_grab;
|
|
|
|
toggle_mouse_enable();
|
|
|
|
}
|
|
|
|
break;
|
2014-07-04 00:17:19 +04:00
|
|
|
} else if (!sdl_fullscreen_toggle && (sdl_event.button.y < headerbar_height)) {
|
2014-06-28 00:53:20 +04:00
|
|
|
headerbar_click(sdl_event.button.x);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case SDL_MOUSEBUTTONUP:
|
2014-07-05 12:00:01 +04:00
|
|
|
if ((sdl_event.button.button == SDL_BUTTON_MIDDLE)
|
|
|
|
&& (sdl_fullscreen_toggle == 0)) {
|
2014-06-28 00:53:20 +04:00
|
|
|
mouse_toggle_check(BX_MT_MBUTTON, 0);
|
|
|
|
}
|
2017-01-05 01:24:33 +03:00
|
|
|
if (console_running()) {
|
|
|
|
break;
|
|
|
|
}
|
2014-06-28 00:53:20 +04:00
|
|
|
// figure out mouse state
|
|
|
|
new_mousex = (int)(sdl_event.button.x);
|
|
|
|
new_mousey = (int)(sdl_event.button.y);
|
|
|
|
// SDL_GetMouseState() returns the state of all buttons
|
|
|
|
mouse_state = SDL_GetMouseState(NULL, NULL);
|
|
|
|
new_mousebuttons =
|
|
|
|
(mouse_state & 0x01) |
|
|
|
|
((mouse_state>>1)&0x02) |
|
|
|
|
((mouse_state<<1)&0x04);
|
|
|
|
// send motion information
|
|
|
|
if (sdl_mouse_mode_absxy) {
|
2014-07-05 12:00:01 +04:00
|
|
|
dx = new_mousex * 0x7fff / res_x;
|
|
|
|
if (sdl_fullscreen_toggle) {
|
|
|
|
dy = new_mousey * 0x7fff / res_y;
|
|
|
|
DEV_mouse_motion(dx, dy, 0, new_mousebuttons, 1);
|
|
|
|
} else if ((new_mousey >= headerbar_height) && (new_mousey < (int)(res_y + headerbar_height))) {
|
2014-06-28 00:53:20 +04:00
|
|
|
dy = (new_mousey - headerbar_height) * 0x7fff / res_y;
|
2014-07-05 12:00:01 +04:00
|
|
|
DEV_mouse_motion(dx, dy, 0, new_mousebuttons, 1);
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
DEV_mouse_motion(new_mousex - old_mousex, -(new_mousey - old_mousey),
|
2014-07-05 12:00:01 +04:00
|
|
|
0, new_mousebuttons, 0);
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
// mark current state to diff with next packet
|
|
|
|
old_mousebuttons = new_mousebuttons;
|
|
|
|
old_mousex = new_mousex;
|
|
|
|
old_mousey = new_mousey;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL_MOUSEWHEEL:
|
2017-01-05 01:24:33 +03:00
|
|
|
if (sdl_grab && !console_running()) {
|
2014-06-28 12:18:26 +04:00
|
|
|
wheel_status = sdl_event.wheel.y;
|
|
|
|
if (sdl_mouse_mode_absxy) {
|
2014-06-28 17:25:52 +04:00
|
|
|
DEV_mouse_motion(old_mousex, old_mousey, wheel_status, old_mousebuttons, 1);
|
2014-06-28 12:18:26 +04:00
|
|
|
} else {
|
|
|
|
DEV_mouse_motion(0, 0, wheel_status, old_mousebuttons, 0);
|
|
|
|
}
|
|
|
|
}
|
2014-06-28 00:53:20 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL_KEYDOWN:
|
2017-01-05 01:24:33 +03:00
|
|
|
if (console_running()) {
|
|
|
|
if ((sdl_event.key.keysym.sym & (1 << 30)) == 0) {
|
2017-01-14 00:37:06 +03:00
|
|
|
Bit8u ascii = (Bit8u)sdl_event.key.keysym.sym;
|
|
|
|
if ((ascii == SDLK_RETURN) || (ascii == SDLK_BACKSPACE)) {
|
|
|
|
console_key_enq(ascii);
|
|
|
|
}
|
2017-01-05 01:24:33 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2014-06-28 00:53:20 +04:00
|
|
|
// mouse capture toggle-check
|
2014-07-05 12:00:01 +04:00
|
|
|
if (sdl_fullscreen_toggle == 0) {
|
|
|
|
if ((sdl_event.key.keysym.sym == SDLK_LCTRL) ||
|
|
|
|
(sdl_event.key.keysym.sym == SDLK_RCTRL)) {
|
|
|
|
mouse_toggle = mouse_toggle_check(BX_MT_KEY_CTRL, 1);
|
|
|
|
} else if (sdl_event.key.keysym.sym == SDLK_LALT) {
|
|
|
|
mouse_toggle = mouse_toggle_check(BX_MT_KEY_ALT, 1);
|
|
|
|
} else if (sdl_event.key.keysym.sym == SDLK_F10) {
|
|
|
|
mouse_toggle = mouse_toggle_check(BX_MT_KEY_F10, 1);
|
|
|
|
} else if (sdl_event.key.keysym.sym == SDLK_F12) {
|
|
|
|
mouse_toggle = mouse_toggle_check(BX_MT_KEY_F12, 1);
|
|
|
|
}
|
|
|
|
if (mouse_toggle) {
|
|
|
|
toggle_mouse_enable();
|
|
|
|
}
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
|
2014-07-04 00:17:19 +04:00
|
|
|
// Window/Fullscreen toggle-check
|
|
|
|
if (sdl_event.key.keysym.sym == SDLK_SCROLLLOCK) {
|
|
|
|
sdl_fullscreen_toggle = ~sdl_fullscreen_toggle;
|
|
|
|
if (sdl_fullscreen_toggle == 0) {
|
|
|
|
switch_to_windowed();
|
|
|
|
} else {
|
|
|
|
switch_to_fullscreen();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2014-06-28 12:18:26 +04:00
|
|
|
if (sdl_nokeyrepeat && sdl_event.key.repeat) {
|
|
|
|
break;
|
|
|
|
}
|
2014-06-28 17:25:52 +04:00
|
|
|
// convert sym->bochs code
|
|
|
|
if (!SIM->get_param_bool(BXPN_KBD_USEMAPPING)->get()) {
|
|
|
|
key_event = sdl_sym_to_bx_key(sdl_event.key.keysym.sym);
|
|
|
|
BX_DEBUG(("keypress scancode=%d, sym=%d, bx_key = %d", sdl_event.key.keysym.scancode, sdl_event.key.keysym.sym, key_event));
|
|
|
|
} else {
|
|
|
|
/* use mapping */
|
|
|
|
BXKeyEntry *entry = bx_keymap.findHostKey(sdl_event.key.keysym.sym);
|
|
|
|
if (!entry) {
|
|
|
|
BX_ERROR(("host key %d (0x%x) not mapped!",
|
|
|
|
(unsigned) sdl_event.key.keysym.sym,
|
2014-07-05 12:00:01 +04:00
|
|
|
(unsigned) sdl_event.key.keysym.sym));
|
2014-06-28 17:25:52 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
key_event = entry->baseKey;
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
2014-06-28 17:25:52 +04:00
|
|
|
if (key_event == BX_KEY_UNHANDLED) break;
|
|
|
|
DEV_kbd_gen_scancode(key_event);
|
2014-06-28 00:53:20 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SDL_KEYUP:
|
|
|
|
// mouse capture toggle-check
|
|
|
|
if ((sdl_event.key.keysym.sym == SDLK_LCTRL) ||
|
|
|
|
(sdl_event.key.keysym.sym == SDLK_RCTRL)) {
|
|
|
|
mouse_toggle_check(BX_MT_KEY_CTRL, 0);
|
|
|
|
} else if (sdl_event.key.keysym.sym == SDLK_LALT) {
|
|
|
|
mouse_toggle_check(BX_MT_KEY_ALT, 0);
|
|
|
|
} else if (sdl_event.key.keysym.sym == SDLK_F10) {
|
|
|
|
mouse_toggle_check(BX_MT_KEY_F10, 0);
|
|
|
|
} else if (sdl_event.key.keysym.sym == SDLK_F12) {
|
|
|
|
mouse_toggle_check(BX_MT_KEY_F12, 0);
|
|
|
|
}
|
|
|
|
|
2014-07-04 00:17:19 +04:00
|
|
|
// filter out release of Windows/Fullscreen toggle
|
|
|
|
if (sdl_event.key.keysym.sym != SDLK_SCROLLLOCK) {
|
|
|
|
// convert sym->bochs code
|
|
|
|
if (!SIM->get_param_bool(BXPN_KBD_USEMAPPING)->get()) {
|
|
|
|
key_event = sdl_sym_to_bx_key(sdl_event.key.keysym.sym);
|
|
|
|
} else {
|
|
|
|
/* use mapping */
|
|
|
|
BXKeyEntry *entry = bx_keymap.findHostKey(sdl_event.key.keysym.sym);
|
|
|
|
if (!entry) {
|
|
|
|
BX_ERROR(("host key %d (0x%x) not mapped!",
|
|
|
|
(unsigned) sdl_event.key.keysym.sym,
|
|
|
|
(unsigned) sdl_event.key.keysym.sym));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
key_event = entry->baseKey;
|
2014-06-28 17:25:52 +04:00
|
|
|
}
|
2014-07-04 00:17:19 +04:00
|
|
|
if (key_event == BX_KEY_UNHANDLED) break;
|
|
|
|
DEV_kbd_gen_scancode(key_event | BX_KEY_RELEASED);
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2017-01-14 00:37:06 +03:00
|
|
|
case SDL_TEXTINPUT:
|
|
|
|
if (console_running()) {
|
|
|
|
console_key_enq(sdl_event.text.text[0]);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2014-06-28 00:53:20 +04:00
|
|
|
case SDL_QUIT:
|
2017-01-25 00:52:19 +03:00
|
|
|
BX_FATAL(("User requested shutdown."));
|
2014-06-28 00:53:20 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#if BX_SHOW_IPS
|
|
|
|
if (sdl_ips_update) {
|
|
|
|
sdl_ips_update = 0;
|
|
|
|
sdl_set_status_text(0, sdl_ips_text, 1);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void bx_sdl2_gui_c::flush(void)
|
|
|
|
{
|
|
|
|
SDL_UpdateWindowSurface(window);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void bx_sdl2_gui_c::clear_screen(void)
|
|
|
|
{
|
2014-07-14 11:58:42 +04:00
|
|
|
SDL_Rect rect;
|
2014-06-28 00:53:20 +04:00
|
|
|
|
2014-07-04 00:17:19 +04:00
|
|
|
if (sdl_screen) {
|
2014-07-14 11:58:42 +04:00
|
|
|
rect.x = 0;
|
|
|
|
rect.y = headerbar_height;
|
|
|
|
rect.w = res_x;
|
|
|
|
rect.h = res_y;
|
|
|
|
SDL_FillRect(sdl_screen, &rect, SDL_MapRGB(sdl_screen->format, 0, 0, 0));
|
2014-07-04 00:17:19 +04:00
|
|
|
} else if (sdl_fullscreen) {
|
2014-07-14 11:58:42 +04:00
|
|
|
rect.x = 0;
|
|
|
|
rect.y = 0;
|
|
|
|
rect.w = res_x;
|
|
|
|
rect.h = res_y;
|
|
|
|
SDL_FillRect(sdl_fullscreen, &rect, SDL_MapRGB(sdl_fullscreen->format, 0, 0, 0));
|
2014-07-04 00:17:19 +04:00
|
|
|
}
|
|
|
|
else return;
|
2014-06-28 00:53:20 +04:00
|
|
|
|
2014-07-14 11:58:42 +04:00
|
|
|
SDL_UpdateWindowSurfaceRects(window, &rect, 1);
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bx_bool bx_sdl2_gui_c::palette_change(Bit8u index, Bit8u red, Bit8u green, Bit8u blue)
|
|
|
|
{
|
2014-07-04 00:17:19 +04:00
|
|
|
if (sdl_screen)
|
2014-06-28 00:53:20 +04:00
|
|
|
sdl_palette[index] = SDL_MapRGB(sdl_screen->format, red, green, blue);
|
2014-07-04 00:17:19 +04:00
|
|
|
else if (sdl_fullscreen)
|
|
|
|
sdl_palette[index] = SDL_MapRGB(sdl_fullscreen->format, red, green, blue);
|
|
|
|
|
2014-06-28 00:53:20 +04:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void bx_sdl2_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) {
|
|
|
|
disp_bpp = guest_bpp = bpp;
|
|
|
|
} else {
|
|
|
|
BX_PANIC(("%d bpp graphics mode not supported", bpp));
|
|
|
|
}
|
|
|
|
guest_textmode = (fheight > 0);
|
2017-01-03 23:59:59 +03:00
|
|
|
guest_fsize = (fheight << 4) | fwidth;
|
2014-06-28 00:53:20 +04:00
|
|
|
guest_xres = x;
|
|
|
|
guest_yres = y;
|
|
|
|
if (guest_textmode) {
|
|
|
|
fontheight = fheight;
|
|
|
|
fontwidth = fwidth;
|
|
|
|
text_cols = x / fontwidth;
|
|
|
|
text_rows = y / fontheight;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((x == res_x) && (y == res_y)) return;
|
|
|
|
|
2016-08-12 20:06:14 +03:00
|
|
|
#ifndef ANDROID
|
|
|
|
// This is not needed on Android
|
2014-07-04 00:17:19 +04:00
|
|
|
if (((int)x > sdl_maxres.w) || ((int)y > sdl_maxres.h)) {
|
|
|
|
BX_PANIC(("dimension_update(): resolution of out of display bounds"));
|
|
|
|
return;
|
|
|
|
}
|
2016-08-12 20:06:14 +03:00
|
|
|
#endif
|
2014-07-04 00:17:19 +04:00
|
|
|
if (sdl_fullscreen_toggle == 0) {
|
|
|
|
SDL_SetWindowSize(window, x, y + headerbar_height + statusbar_height);
|
|
|
|
sdl_screen = SDL_GetWindowSurface(window);
|
|
|
|
headerbar_fg = SDL_MapRGB(
|
2014-07-05 12:00:01 +04:00
|
|
|
sdl_screen->format,
|
|
|
|
BX_HEADERBAR_FG_RED,
|
|
|
|
BX_HEADERBAR_FG_GREEN,
|
|
|
|
BX_HEADERBAR_FG_BLUE);
|
2014-07-04 00:17:19 +04:00
|
|
|
headerbar_bg = SDL_MapRGB(
|
2014-07-05 12:00:01 +04:00
|
|
|
sdl_screen->format,
|
|
|
|
BX_HEADERBAR_BG_RED,
|
|
|
|
BX_HEADERBAR_BG_GREEN,
|
|
|
|
BX_HEADERBAR_BG_BLUE);
|
2014-07-04 00:17:19 +04:00
|
|
|
} else {
|
|
|
|
SDL_SetWindowSize(window, x, y);
|
2014-07-14 11:58:42 +04:00
|
|
|
// BUG: SDL2 does not update the surface here
|
2014-07-04 00:17:19 +04:00
|
|
|
sdl_fullscreen = SDL_GetWindowSurface(window);
|
|
|
|
}
|
2014-06-28 00:53:20 +04:00
|
|
|
|
|
|
|
res_x = x;
|
|
|
|
res_y = y;
|
|
|
|
half_res_x = x/2;
|
|
|
|
half_res_y = y/2;
|
|
|
|
bx_gui->show_headerbar();
|
|
|
|
host_xres = x;
|
|
|
|
host_yres = y;
|
|
|
|
host_bpp = 32;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
unsigned bx_sdl2_gui_c::create_bitmap(const unsigned char *bmap, unsigned xdim, unsigned ydim)
|
|
|
|
{
|
|
|
|
Uint32 *buf, *buf_row;
|
|
|
|
Uint32 disp;
|
|
|
|
unsigned char pixels;
|
|
|
|
|
|
|
|
if (n_sdl_bitmaps >= MAX_SDL_BITMAPS) {
|
|
|
|
BX_PANIC(("too many SDL bitmaps. To fix, increase MAX_SDL_BITMAPS"));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bitmaps *tmp = new bitmaps;
|
|
|
|
|
|
|
|
tmp->surface = SDL_CreateRGBSurface(
|
|
|
|
SDL_SWSURFACE,
|
|
|
|
xdim,
|
|
|
|
ydim,
|
|
|
|
32,
|
|
|
|
#if SDL_BYTEORDER == SDL_BIG_ENDIAN
|
|
|
|
0xff000000,
|
|
|
|
0x00ff0000,
|
|
|
|
0x0000ff00,
|
|
|
|
0x00000000
|
|
|
|
#else
|
|
|
|
0x000000ff,
|
|
|
|
0x0000ff00,
|
|
|
|
0x00ff0000,
|
|
|
|
0x00000000
|
|
|
|
#endif
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!tmp->surface) {
|
|
|
|
delete tmp;
|
|
|
|
bx_gui->exit();
|
2017-01-25 00:52:19 +03:00
|
|
|
BX_FATAL(("Unable to create requested bitmap"));
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
tmp->src.w = xdim;
|
|
|
|
tmp->src.h = ydim;
|
|
|
|
tmp->src.x = 0;
|
|
|
|
tmp->src.y = 0;
|
|
|
|
tmp->dst.x = -1;
|
|
|
|
tmp->dst.y = 0;
|
|
|
|
tmp->dst.w = xdim;
|
|
|
|
tmp->dst.h = ydim;
|
|
|
|
buf = (Uint32 *)tmp->surface->pixels;
|
|
|
|
disp = tmp->surface->pitch/4;
|
|
|
|
|
|
|
|
do {
|
|
|
|
buf_row = buf;
|
|
|
|
xdim = tmp->src.w / 8;
|
|
|
|
do {
|
|
|
|
pixels = *bmap++;
|
|
|
|
for (unsigned i=0; i<8; i++) {
|
|
|
|
if ((pixels & 0x01) == 0)
|
|
|
|
*buf++ = headerbar_bg;
|
|
|
|
else
|
|
|
|
*buf++ = headerbar_fg;
|
|
|
|
pixels = pixels >> 1;
|
|
|
|
}
|
|
|
|
} while (--xdim);
|
|
|
|
buf = buf_row + disp;
|
|
|
|
} while (--ydim);
|
|
|
|
|
|
|
|
sdl_bitmaps[n_sdl_bitmaps] = tmp;
|
|
|
|
|
|
|
|
return n_sdl_bitmaps++;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
unsigned bx_sdl2_gui_c::headerbar_bitmap(unsigned bmap_id, unsigned alignment,
|
|
|
|
void (*f)(void))
|
|
|
|
{
|
|
|
|
unsigned hb_index;
|
|
|
|
|
|
|
|
if (bmap_id >= (unsigned)n_sdl_bitmaps) return 0;
|
|
|
|
|
|
|
|
if ((bx_headerbar_entries+1) > BX_MAX_HEADERBAR_ENTRIES)
|
|
|
|
BX_PANIC(("too many headerbar entries, increase BX_MAX_HEADERBAR_ENTRIES"));
|
|
|
|
|
2017-01-14 16:10:20 +03:00
|
|
|
hb_index = bx_headerbar_entries++;
|
2014-06-28 00:53:20 +04:00
|
|
|
|
2017-01-14 16:10:20 +03:00
|
|
|
bx_headerbar_entry[hb_index].bmap_id = bmap_id;
|
|
|
|
bx_headerbar_entry[hb_index].xdim = sdl_bitmaps[bmap_id]->src.w;
|
|
|
|
bx_headerbar_entry[hb_index].ydim = sdl_bitmaps[bmap_id]->src.h;
|
|
|
|
bx_headerbar_entry[hb_index].alignment = alignment;
|
|
|
|
bx_headerbar_entry[hb_index].f = f;
|
2014-06-28 00:53:20 +04:00
|
|
|
|
|
|
|
if (alignment == BX_GRAVITY_LEFT) {
|
|
|
|
sdl_bitmaps[bmap_id]->dst.x = bx_bitmap_left_xorigin;
|
|
|
|
bx_bitmap_left_xorigin += sdl_bitmaps[bmap_id]->src.w;
|
|
|
|
} else {
|
|
|
|
bx_bitmap_right_xorigin += sdl_bitmaps[bmap_id]->src.w;
|
|
|
|
sdl_bitmaps[bmap_id]->dst.x = bx_bitmap_right_xorigin;
|
|
|
|
}
|
2017-01-14 16:10:20 +03:00
|
|
|
bx_headerbar_entry[hb_index].xorigin = sdl_bitmaps[bmap_id]->dst.x;
|
2014-06-28 00:53:20 +04:00
|
|
|
|
|
|
|
return hb_index;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void bx_sdl2_gui_c::replace_bitmap(unsigned hbar_id, unsigned bmap_id)
|
|
|
|
{
|
|
|
|
SDL_Rect hb_dst;
|
|
|
|
unsigned old_id;
|
|
|
|
|
2014-07-04 00:17:19 +04:00
|
|
|
if (!sdl_screen) return;
|
2017-01-14 16:10:20 +03:00
|
|
|
old_id = bx_headerbar_entry[hbar_id].bmap_id;
|
2014-06-28 00:53:20 +04:00
|
|
|
hb_dst = sdl_bitmaps[old_id]->dst;
|
|
|
|
sdl_bitmaps[old_id]->dst.x = -1;
|
2017-01-14 16:10:20 +03:00
|
|
|
bx_headerbar_entry[hbar_id].bmap_id = bmap_id;
|
2014-06-28 00:53:20 +04:00
|
|
|
sdl_bitmaps[bmap_id]->dst.x = hb_dst.x;
|
|
|
|
if (sdl_bitmaps[bmap_id]->dst.x != -1) {
|
2017-01-14 16:10:20 +03:00
|
|
|
if (bx_headerbar_entry[hbar_id].alignment == BX_GRAVITY_RIGHT) {
|
2014-06-28 00:53:20 +04:00
|
|
|
hb_dst.x = res_x - hb_dst.x;
|
|
|
|
}
|
|
|
|
SDL_BlitSurface(
|
|
|
|
sdl_bitmaps[bmap_id]->surface,
|
|
|
|
&sdl_bitmaps[bmap_id]->src,
|
|
|
|
sdl_screen,
|
|
|
|
&hb_dst);
|
2014-07-14 11:58:42 +04:00
|
|
|
SDL_UpdateWindowSurfaceRects(window, &hb_dst, 1);
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void bx_sdl2_gui_c::show_headerbar(void)
|
|
|
|
{
|
|
|
|
Uint32 *buf;
|
|
|
|
Uint32 *buf_row;
|
|
|
|
Uint32 disp;
|
|
|
|
int rowsleft = headerbar_height;
|
|
|
|
int colsleft, sb_item;
|
|
|
|
int bitmapscount = bx_headerbar_entries;
|
|
|
|
unsigned current_bmp, pos_x;
|
2014-07-14 11:58:42 +04:00
|
|
|
SDL_Rect hb_dst, hb_rect;
|
2014-06-28 00:53:20 +04:00
|
|
|
|
|
|
|
if (!sdl_screen) return;
|
|
|
|
disp = sdl_screen->pitch/4;
|
|
|
|
buf = (Uint32 *)sdl_screen->pixels;
|
|
|
|
|
|
|
|
// draw headerbar background
|
2014-07-14 11:58:42 +04:00
|
|
|
hb_rect.x = 0;
|
|
|
|
hb_rect.y = 0;
|
|
|
|
hb_rect.w = res_x;
|
|
|
|
hb_rect.h = headerbar_height;
|
|
|
|
SDL_FillRect(sdl_screen, &hb_rect, headerbar_bg);
|
2014-06-28 00:53:20 +04:00
|
|
|
|
|
|
|
// go thru the bitmaps and display the active ones
|
|
|
|
while (bitmapscount--) {
|
2017-01-14 16:10:20 +03:00
|
|
|
current_bmp = bx_headerbar_entry[bitmapscount].bmap_id;
|
2014-06-28 00:53:20 +04:00
|
|
|
if(sdl_bitmaps[current_bmp]->dst.x != -1) {
|
|
|
|
hb_dst = sdl_bitmaps[current_bmp]->dst;
|
2017-01-14 16:10:20 +03:00
|
|
|
if (bx_headerbar_entry[bitmapscount].alignment == BX_GRAVITY_RIGHT) {
|
2014-06-28 00:53:20 +04:00
|
|
|
hb_dst.x = res_x - hb_dst.x;
|
|
|
|
}
|
|
|
|
SDL_BlitSurface(
|
|
|
|
sdl_bitmaps[current_bmp]->surface,
|
|
|
|
&sdl_bitmaps[current_bmp]->src,
|
|
|
|
sdl_screen,
|
|
|
|
&hb_dst);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// draw statusbar background
|
|
|
|
rowsleft = statusbar_height;
|
|
|
|
buf = (Uint32 *)sdl_screen->pixels + (res_y + headerbar_height) * disp;
|
|
|
|
do {
|
|
|
|
colsleft = res_x;
|
|
|
|
buf_row = buf;
|
|
|
|
sb_item = 1;
|
|
|
|
pos_x = 0;
|
|
|
|
do {
|
|
|
|
if (pos_x == statusitem_pos[sb_item]) {
|
|
|
|
*buf++ = headerbar_fg;
|
|
|
|
if (sb_item < 11) sb_item++;
|
|
|
|
} else {
|
|
|
|
*buf++ = headerbar_bg;
|
|
|
|
}
|
|
|
|
pos_x++;
|
|
|
|
} while (--colsleft);
|
|
|
|
buf = buf_row + disp;
|
|
|
|
} while (--rowsleft);
|
2014-07-14 11:58:42 +04:00
|
|
|
SDL_UpdateWindowSurfaceRects(window, &hb_rect, 1);
|
2014-06-28 00:53:20 +04:00
|
|
|
for (unsigned i=0; i<statusitem_count; i++) {
|
|
|
|
sdl_set_status_text(i+1, statusitem[i].text, statusitem_active[i+1]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int bx_sdl2_gui_c::get_clipboard_text(Bit8u **bytes, Bit32s *nbytes)
|
|
|
|
{
|
2017-01-05 01:24:33 +03:00
|
|
|
char *tmp = SDL_GetClipboardText();
|
|
|
|
int len = strlen(tmp) + 1;
|
|
|
|
Bit8u *buf = new Bit8u[len];
|
|
|
|
memcpy(buf, tmp, len);
|
|
|
|
*bytes = buf;
|
|
|
|
*nbytes = len;
|
|
|
|
SDL_free(tmp);
|
|
|
|
return 1;
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int bx_sdl2_gui_c::set_clipboard_text(char *text_snapshot, Bit32u len)
|
|
|
|
{
|
2017-01-05 01:24:33 +03:00
|
|
|
return (SDL_SetClipboardText(text_snapshot) == 0);
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void bx_sdl2_gui_c::mouse_enabled_changed_specific(bx_bool val)
|
|
|
|
{
|
|
|
|
if (val == 1) {
|
|
|
|
SDL_ShowCursor(0);
|
|
|
|
SDL_SetWindowGrab(window, SDL_TRUE);
|
|
|
|
} else {
|
|
|
|
SDL_ShowCursor(1);
|
|
|
|
SDL_SetWindowGrab(window, SDL_FALSE);
|
|
|
|
}
|
|
|
|
sdl_grab = val;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void bx_sdl2_gui_c::exit(void)
|
|
|
|
{
|
2014-07-04 00:17:19 +04:00
|
|
|
SDL_ShowCursor(1);
|
|
|
|
SDL_SetWindowGrab(window, SDL_FALSE);
|
2014-06-28 00:53:20 +04:00
|
|
|
while (n_sdl_bitmaps) {
|
|
|
|
SDL_FreeSurface(sdl_bitmaps[n_sdl_bitmaps-1]->surface);
|
|
|
|
n_sdl_bitmaps--;
|
|
|
|
}
|
2014-08-17 16:48:05 +04:00
|
|
|
|
|
|
|
#if BX_DEBUGGER && BX_DEBUGGER_GUI
|
|
|
|
if (SIM->has_debug_gui()) {
|
|
|
|
close_debug_dialog();
|
|
|
|
}
|
|
|
|
#endif
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// New graphics API methods
|
|
|
|
|
|
|
|
bx_svga_tileinfo_t *bx_sdl2_gui_c::graphics_tile_info(bx_svga_tileinfo_t *info)
|
|
|
|
{
|
2014-07-04 00:17:19 +04:00
|
|
|
if (sdl_screen) {
|
|
|
|
info->bpp = sdl_screen->format->BitsPerPixel;
|
|
|
|
info->pitch = sdl_screen->pitch;
|
|
|
|
info->red_shift = sdl_screen->format->Rshift + 8 - sdl_screen->format->Rloss;
|
|
|
|
info->green_shift = sdl_screen->format->Gshift + 8 - sdl_screen->format->Gloss;
|
|
|
|
info->blue_shift = sdl_screen->format->Bshift + 8 - sdl_screen->format->Bloss;
|
|
|
|
info->red_mask = sdl_screen->format->Rmask;
|
|
|
|
info->green_mask = sdl_screen->format->Gmask;
|
|
|
|
info->blue_mask = sdl_screen->format->Bmask;
|
|
|
|
info->is_indexed = (sdl_screen->format->palette != NULL);
|
|
|
|
} else {
|
|
|
|
info->bpp = sdl_fullscreen->format->BitsPerPixel;
|
|
|
|
info->pitch = sdl_fullscreen->pitch;
|
|
|
|
info->red_shift = sdl_fullscreen->format->Rshift + 8 - sdl_fullscreen->format->Rloss;
|
|
|
|
info->green_shift = sdl_fullscreen->format->Gshift + 8 - sdl_fullscreen->format->Gloss;
|
|
|
|
info->blue_shift = sdl_fullscreen->format->Bshift + 8 - sdl_fullscreen->format->Bloss;
|
|
|
|
info->red_mask = sdl_fullscreen->format->Rmask;
|
|
|
|
info->green_mask = sdl_fullscreen->format->Gmask;
|
|
|
|
info->blue_mask = sdl_fullscreen->format->Bmask;
|
|
|
|
info->is_indexed = (sdl_fullscreen->format->palette != NULL);
|
|
|
|
}
|
2014-06-28 00:53:20 +04:00
|
|
|
#ifdef BX_LITTLE_ENDIAN
|
|
|
|
info->is_little_endian = 1;
|
|
|
|
#else
|
|
|
|
info->is_little_endian = 0;
|
|
|
|
#endif
|
|
|
|
return info;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Bit8u *bx_sdl2_gui_c::graphics_tile_get(unsigned x0, unsigned y0, unsigned *w, unsigned *h)
|
|
|
|
{
|
|
|
|
if (x0+x_tilesize > res_x) {
|
|
|
|
*w = res_x - x0;
|
|
|
|
} else {
|
|
|
|
*w = x_tilesize;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (y0+y_tilesize > res_y) {
|
|
|
|
*h = res_y - y0;
|
|
|
|
} else {
|
|
|
|
*h = y_tilesize;
|
|
|
|
}
|
|
|
|
|
2014-07-04 00:17:19 +04:00
|
|
|
if (sdl_screen) {
|
|
|
|
return (Bit8u *)sdl_screen->pixels +
|
2014-07-05 12:00:01 +04:00
|
|
|
sdl_screen->pitch * (headerbar_height + y0) +
|
|
|
|
sdl_screen->format->BytesPerPixel * x0;
|
2014-07-04 00:17:19 +04:00
|
|
|
} else {
|
|
|
|
return (Bit8u *)sdl_fullscreen->pixels +
|
|
|
|
sdl_fullscreen->pitch * y0 +
|
|
|
|
sdl_fullscreen->format->BytesPerPixel * x0;
|
|
|
|
}
|
2014-06-28 00:53:20 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void bx_sdl2_gui_c::graphics_tile_update_in_place(unsigned x0, unsigned y0,
|
|
|
|
unsigned w, unsigned h)
|
|
|
|
{
|
|
|
|
// Nothing to do here
|
|
|
|
}
|
|
|
|
|
|
|
|
// Optional bx_gui_c methods
|
|
|
|
|
2014-07-04 00:17:19 +04:00
|
|
|
void bx_sdl2_gui_c::set_display_mode(disp_mode_t newmode)
|
|
|
|
{
|
|
|
|
// if no mode change, do nothing.
|
|
|
|
if (disp_mode == newmode) return;
|
|
|
|
// remember the display mode for next time
|
|
|
|
disp_mode = newmode;
|
2017-01-05 01:24:33 +03:00
|
|
|
if ((newmode == DISP_MODE_SIM) && console_running()) {
|
|
|
|
console_cleanup();
|
|
|
|
return;
|
|
|
|
}
|
2014-07-04 00:17:19 +04:00
|
|
|
// If fullscreen mode is on, we must switch back to windowed mode if
|
|
|
|
// the user needs to see the text console.
|
|
|
|
if (sdl_fullscreen_toggle) {
|
|
|
|
switch (newmode) {
|
|
|
|
case DISP_MODE_CONFIG:
|
|
|
|
BX_DEBUG(("switch to configuration mode (windowed)"));
|
|
|
|
switch_to_windowed();
|
|
|
|
break;
|
|
|
|
case DISP_MODE_SIM:
|
|
|
|
BX_DEBUG(("switch to simulation mode (fullscreen)"));
|
|
|
|
switch_to_fullscreen();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-28 00:53:20 +04:00
|
|
|
void bx_sdl2_gui_c::statusbar_setitem_specific(int element, bx_bool active, bx_bool w)
|
|
|
|
{
|
|
|
|
sdl_set_status_text(element+1, statusitem[element].text, active, w);
|
|
|
|
}
|
|
|
|
|
2014-07-04 00:17:19 +04:00
|
|
|
|
|
|
|
void bx_sdl2_gui_c::get_capabilities(Bit16u *xres, Bit16u *yres, Bit16u *bpp)
|
|
|
|
{
|
|
|
|
*xres = sdl_maxres.w;
|
|
|
|
*yres = sdl_maxres.h;
|
|
|
|
*bpp = 32;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-28 00:53:20 +04:00
|
|
|
void bx_sdl2_gui_c::set_mouse_mode_absxy(bx_bool mode)
|
|
|
|
{
|
|
|
|
sdl_mouse_mode_absxy = mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#if BX_SHOW_IPS
|
|
|
|
void bx_sdl2_gui_c::show_ips(Bit32u ips_count)
|
|
|
|
{
|
|
|
|
if (!sdl_hide_ips && !sdl_ips_update) {
|
|
|
|
ips_count /= 1000;
|
|
|
|
sprintf(sdl_ips_text, "IPS: %u.%3.3uM", ips_count / 1000, ips_count % 1000);
|
|
|
|
sdl_ips_update = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-01-14 00:37:06 +03:00
|
|
|
void bx_sdl2_gui_c::set_console_edit_mode(bx_bool mode)
|
|
|
|
{
|
|
|
|
if (mode) {
|
|
|
|
SDL_StartTextInput();
|
|
|
|
} else {
|
|
|
|
SDL_StopTextInput();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-29 11:37:58 +04:00
|
|
|
/// key mapping code for SDL2
|
|
|
|
|
2014-06-28 17:25:52 +04:00
|
|
|
typedef struct {
|
|
|
|
const char *name;
|
|
|
|
Bit32u value;
|
|
|
|
} keyTableEntry;
|
|
|
|
|
|
|
|
#define DEF_SDL_KEY(key) \
|
|
|
|
{ #key, key },
|
|
|
|
|
|
|
|
keyTableEntry keytable[] = {
|
|
|
|
// this include provides all the entries.
|
|
|
|
#include "sdlkeys.h"
|
|
|
|
// one final entry to mark the end
|
|
|
|
{ NULL, 0 }
|
|
|
|
};
|
|
|
|
|
|
|
|
// function to convert key names into SDLKey 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 convertStringToSDLKey (const char *string)
|
|
|
|
{
|
|
|
|
keyTableEntry *ptr;
|
|
|
|
for (ptr = &keytable[0]; ptr->name != NULL; ptr++) {
|
|
|
|
//BX_DEBUG (("comparing string '%s' to SDL key '%s'", string, ptr->name));
|
|
|
|
if (!strcmp(string, ptr->name))
|
|
|
|
return ptr->value;
|
|
|
|
}
|
|
|
|
return BX_KEYMAP_UNKNOWN;
|
|
|
|
}
|
|
|
|
|
2014-06-29 11:37:58 +04:00
|
|
|
// SDL2 ask dialog
|
|
|
|
|
|
|
|
#ifndef WIN32
|
|
|
|
int sdl2_ask_dialog(BxEvent *event)
|
|
|
|
{
|
|
|
|
SDL_MessageBoxData msgboxdata;
|
|
|
|
SDL_MessageBoxButtonData buttondata[4];
|
2016-12-29 12:47:07 +03:00
|
|
|
int i = 0, level, mode, retcode;
|
2014-06-29 11:37:58 +04:00
|
|
|
char message[512];
|
|
|
|
|
|
|
|
level = event->u.logmsg.level;
|
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/bochs.h ./bochs.h
--- /home/volker/bochs/bochs/bochs.h 2016-08-12 19:06:18.803209189 +0200
+++ ./bochs.h 2016-12-28 00:41:20.000627252 +0100
@@ -2,7 +2,7 @@
// $Id: bochs.h 12935 2016-08-12 17:06:14Z vruppert $
/////////////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2001-2015 The Bochs Project
+// Copyright (C) 2001-2016 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
@@ -276,8 +276,9 @@
void error(const char *fmt, ...) BX_CPP_AttrPrintf(2, 3);
void panic(const char *fmt, ...) BX_CPP_AttrPrintf(2, 3);
void ldebug(const char *fmt, ...) BX_CPP_AttrPrintf(2, 3);
- void fatal (const char *prefix, const char *fmt, va_list ap, int exit_status);
- void ask (int level, const char *prefix, const char *fmt, va_list ap);
+ void fatal(const char *prefix, const char *fmt, va_list ap, int exit_status);
+ void warn(int level, const char *prefix, const char *fmt, va_list ap);
+ void ask(int level, const char *prefix, const char *fmt, va_list ap);
void put(const char *p);
void put(const char *n, const char *p);
void setio(class iofunctions *);
@@ -334,7 +335,8 @@
void set_log_action(int loglevel, int action);
const char *getlevel(int i) const;
const char *getaction(int i) const;
-
+ int isaction(const char *val) const;
+
protected:
int n_logfn;
#define MAX_LOGFNS 512
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/CHANGES ./CHANGES
--- /home/volker/bochs/bochs/CHANGES 2016-12-26 10:45:44.000000000 +0100
+++ ./CHANGES 2016-12-28 15:54:25.127088081 +0100
@@ -1,5 +1,8 @@
Changes after 2.6.8 release:
+- General
+ - Added new log action "warn", designed to show a message box on error events.
+
- Configure and compile
- Added Android host platform support.
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/config.cc ./config.cc
--- /home/volker/bochs/bochs/config.cc 2016-05-03 21:15:09.158016000 +0200
+++ ./config.cc 2016-12-27 19:53:10.461420368 +0100
@@ -2062,15 +2062,8 @@
actstr = strtok(NULL, "");
if (actstr != NULL) {
def_action = !strcmp(module, "action");
- if (!strcmp(actstr, "fatal"))
- action = ACT_FATAL;
- else if (!strcmp (actstr, "report"))
- action = ACT_REPORT;
- else if (!strcmp (actstr, "ignore"))
- action = ACT_IGNORE;
- else if (!strcmp (actstr, "ask"))
- action = ACT_ASK;
- else {
+ action = SIM->is_action_name(actstr);
+ if (action < ACT_IGNORE) {
PARSE_ERR(("%s: %s directive malformed.", context, params[0]));
free(param);
return -1;
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/sdl2.cc ./gui/sdl2.cc
--- /home/volker/bochs/bochs/gui/sdl2.cc 2016-08-12 19:06:18.811209142 +0200
+++ ./gui/sdl2.cc 2016-12-28 12:33:39.534288819 +0100
@@ -2,7 +2,7 @@
// $Id: sdl2.cc 12935 2016-08-12 17:06:14Z vruppert $
/////////////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2014-2015 The Bochs Project
+// Copyright (C) 2014-2016 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
@@ -1478,20 +1478,16 @@
SDL_MessageBoxData msgboxdata;
SDL_MessageBoxButtonData buttondata[4];
int level, retcode;
-#if BX_DEBUGGER || BX_GDBSTUB
- int defbtn = 3;
-#else
- int defbtn = 2;
-#endif
char message[512];
level = event->u.logmsg.level;
- sprintf(message, "%s %s", event->u.logmsg.prefix, event->u.logmsg.msg);
+ sprintf(message, "Device: %s\nMessage: %s", event->u.logmsg.prefix,
+ event->u.logmsg.msg);
msgboxdata.flags = SDL_MESSAGEBOX_ERROR;
msgboxdata.window = window;
msgboxdata.title = SIM->get_log_level_name(level);
msgboxdata.message = message;
- msgboxdata.numbuttons = defbtn + 1;
+ msgboxdata.numbuttons = 2;
msgboxdata.buttons = buttondata;
msgboxdata.colorScheme = NULL;
buttondata[0].flags = 0;
@@ -1500,14 +1496,18 @@
buttondata[1].flags = 0;
buttondata[1].buttonid = BX_LOG_ASK_CHOICE_CONTINUE_ALWAYS;
buttondata[1].text = "Alwayscont";
+ if (event->u.logmsg.flag == BX_LOG_ASK_ASKDLG) {
+ msgboxdata.numbuttons = 3;
+ buttondata[2].flags = SDL_MESSAGEBOX_BUTTON_ESCAPEKEY_DEFAULT;
+ buttondata[2].buttonid = BX_LOG_ASK_CHOICE_DIE;
+ buttondata[2].text = "Quit";
#if BX_DEBUGGER || BX_GDBSTUB
- buttondata[2].flags = 0;
- buttondata[2].buttonid = BX_LOG_ASK_CHOICE_ENTER_DEBUG;
- buttondata[2].text = "Debugger";
-#endif
- buttondata[defbtn].flags = SDL_MESSAGEBOX_BUTTON_ESCAPEKEY_DEFAULT;
- buttondata[defbtn].buttonid = BX_LOG_ASK_CHOICE_DIE;
- buttondata[defbtn].text = "Quit";
+ msgboxdata.numbuttons = 4;
+ buttondata[3].flags = 0;
+ buttondata[3].buttonid = BX_LOG_ASK_CHOICE_ENTER_DEBUG;
+ buttondata[3].text = "Debugger";
+#endif
+ }
if (SDL_ShowMessageBox(&msgboxdata, &retcode) < 0) {
return -1;
} else {
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/siminterface.cc ./gui/siminterface.cc
--- /home/volker/bochs/bochs/gui/siminterface.cc 2016-12-05 19:56:56.729685000 +0100
+++ ./gui/siminterface.cc 2016-12-28 11:14:02.004075717 +0100
@@ -2,7 +2,7 @@
// $Id: siminterface.cc 12981 2016-12-05 18:56:56Z sshwarts $
/////////////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2002-2015 The Bochs Project
+// Copyright (C) 2002-2016 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
@@ -100,6 +100,7 @@
virtual int get_log_action(int mod, int level);
virtual void set_log_action(int mod, int level, int action);
virtual const char *get_action_name(int action);
+ virtual int is_action_name(const char *val);
virtual int get_default_log_action(int level) {
return logfunctions::get_default_action(level);
}
@@ -123,6 +124,7 @@
virtual void set_notify_callback(bxevent_handler func, void *arg);
virtual void get_notify_callback(bxevent_handler *func, void **arg);
virtual BxEvent* sim_to_ci_event(BxEvent *event);
+ virtual int log_warn(const char *prefix, int level, const char *msg);
virtual int log_ask(const char *prefix, int level, const char *msg);
virtual void log_msg(const char *prefix, int level, const char *msg);
virtual void set_log_viewer(bx_bool val) { bx_log_viewer = val; }
@@ -420,6 +422,11 @@
return io->getaction(action);
}
+int bx_real_sim_c::is_action_name(const char *val)
+{
+ return io->isaction(val);
+}
+
const char *bx_real_sim_c::get_log_level_name(int level)
{
return io->getlevel(level);
@@ -575,6 +582,21 @@
}
}
+int bx_real_sim_c::log_warn(const char *prefix, int level, const char *msg)
+{
+ BxEvent be;
+ be.type = BX_SYNC_EVT_LOG_ASK;
+ be.u.logmsg.prefix = prefix;
+ be.u.logmsg.level = level;
+ be.u.logmsg.msg = msg;
+ be.u.logmsg.flag = BX_LOG_ASK_MSGBOX_WARN;
+ // default return value in case something goes wrong.
+ be.retcode = BX_LOG_NOTIFY_FAILED;
+ // calling notify
+ sim_to_ci_event(&be);
+ return be.retcode;
+}
+
// returns 0 for continue, 1 for alwayscontinue, 2 for die.
int bx_real_sim_c::log_ask(const char *prefix, int level, const char *msg)
{
@@ -583,6 +605,7 @@
be.u.logmsg.prefix = prefix;
be.u.logmsg.level = level;
be.u.logmsg.msg = msg;
+ be.u.logmsg.flag = BX_LOG_ASK_ASKDLG;
// default return value in case something goes wrong.
be.retcode = BX_LOG_NOTIFY_FAILED;
// calling notify
@@ -1157,16 +1180,10 @@
} else if (!strncmp(string, "PANIC=", 6)) {
type = LOGLEV_PANIC;
}
- if (!strcmp(string+j, "ignore")) {
- action = ACT_IGNORE;
- } else if (!strcmp(string+j, "report")) {
- action = ACT_REPORT;
- } else if (!strcmp(string+j, "ask")) {
- action = ACT_ASK;
- } else if (!strcmp(string+j, "fatal")) {
- action = ACT_FATAL;
+ action = is_action_name(string+j);
+ if (action >= ACT_IGNORE) {
+ set_log_action(dev, type, action);
}
- set_log_action(dev, type, action);
} else {
if (i == 1) {
BX_ERROR(("restore_logopts(): log module '%s' not found", devname));
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/siminterface.h ./gui/siminterface.h
--- /home/volker/bochs/bochs/gui/siminterface.h 2016-03-31 19:24:37.451025427 +0200
+++ ./gui/siminterface.h 2016-12-28 11:11:21.036683362 +0100
@@ -168,6 +168,7 @@
typedef enum {
ACT_IGNORE = 0,
ACT_REPORT,
+ ACT_WARN,
ACT_ASK,
ACT_FATAL,
N_ACT
@@ -178,11 +179,11 @@
// normally all action choices are available for all event types. The exclude
// expression allows some choices to be eliminated if they don't make any
// sense. For example, it would be stupid to ignore a panic.
-#define BX_LOG_OPTS_EXCLUDE(type, choice) ( \
- /* can't die or ask, on debug or info events */ \
- (type <= LOGLEV_INFO && (choice == ACT_ASK || choice == ACT_FATAL)) \
- /* can't ignore panics */ \
- || (type == LOGLEV_PANIC && choice == ACT_IGNORE) \
+#define BX_LOG_OPTS_EXCLUDE(type, choice) ( \
+ /* can't die, ask or warn, on debug or info events */ \
+ (type <= LOGLEV_INFO && (choice >= ACT_WARN)) \
+ /* can't ignore panics */ \
+ || (type == LOGLEV_PANIC && choice == ACT_IGNORE) \
)
// floppy / cdrom media status
@@ -392,6 +393,7 @@
// synchronizing threads, etc. for each.
typedef struct {
Bit8u level;
+ Bit8u flag;
const char *prefix;
const char *msg;
} BxLogMsgEvent;
@@ -419,6 +421,12 @@
BX_LOG_NOTIFY_FAILED
};
+enum {
+ BX_LOG_ASK_ASKDLG,
+ BX_LOG_ASK_MSGBOX_WARN,
+ BX_LOG_ASK_MSGBOX_QUIT
+};
+
// Event type: BX_SYNC_EVT_GET_DBG_COMMAND
//
// This is a synchronous event sent from the simulator to the debugger
@@ -675,6 +683,7 @@
virtual int get_default_log_action(int level) {return -1;}
virtual void set_default_log_action(int level, int action) {}
virtual const char *get_action_name(int action) {return NULL;}
+ virtual int is_action_name(const char *val) {return -1;}
virtual const char *get_log_level_name(int level) {return NULL;}
virtual int get_max_log_level() {return -1;}
@@ -715,6 +724,9 @@
// send an event from the simulator to the CI.
virtual BxEvent* sim_to_ci_event(BxEvent *event) {return NULL;}
+ // called from simulator when it hits errors, to warn the user
+ // before continuing simulation
+ virtual int log_warn(const char *prefix, int level, const char *msg) {return -1;}
// called from simulator when it hits serious errors, to ask if the user
// wants to continue or not
virtual int log_ask(const char *prefix, int level, const char *msg) {return -1;}
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/textconfig.cc ./gui/textconfig.cc
--- /home/volker/bochs/bochs/gui/textconfig.cc 2016-12-05 20:15:59.112637000 +0100
+++ ./gui/textconfig.cc 2016-12-28 12:44:43.079411258 +0100
@@ -2,7 +2,7 @@
// $Id: textconfig.cc 12983 2016-12-05 19:15:59Z sshwarts $
/////////////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2002-2013 The Bochs Project
+// Copyright (C) 2002-2016 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
@@ -552,8 +552,8 @@
}
static const char *log_options_prompt1 = "Enter the ID of the device to edit, or -1 to return: [-1] ";
-static const char *log_level_choices[] = { "ignore", "report", "ask", "fatal", "no change" };
-static int log_level_n_choices_normal = 4;
+static const char *log_level_choices[N_ACT+1] = { "ignore", "report", "warn", "ask", "fatal", "no change" };
+static int log_level_n_choices_normal = N_ACT;
void bx_log_options(int individual)
{
@@ -589,7 +589,7 @@
bx_print_log_action_table();
for (int level=0; level<SIM->get_max_log_level(); level++) {
char prompt[1024];
- int action, default_action = 4; // default to no change
+ int action, default_action = N_ACT; // default to no change
sprintf(prompt, "Enter action for %s event on all devices: [no change] ", SIM->get_log_level_name(level));
// do show the no change choice (choices=4)
if (ask_menu(prompt, "", log_level_n_choices_normal+1, log_level_choices, default_action, &action)<0)
@@ -718,47 +718,50 @@
event->retcode = event->u.param.param->text_ask(stdin, stderr);
return event;
case BX_SYNC_EVT_LOG_ASK:
- {
- int level = event->u.logmsg.level;
- fprintf(stderr, "========================================================================\n");
- fprintf(stderr, "Event type: %s\n", SIM->get_log_level_name (level));
- fprintf(stderr, "Device: %s\n", event->u.logmsg.prefix);
- fprintf(stderr, "Message: %s\n\n", event->u.logmsg.msg);
- fprintf(stderr, "A %s has occurred. Do you want to:\n", SIM->get_log_level_name (level));
- fprintf(stderr, " cont - continue execution\n");
- fprintf(stderr, " alwayscont - continue execution, and don't ask again.\n");
- fprintf(stderr, " This affects only %s events from device %s\n", SIM->get_log_level_name (level), event->u.logmsg.prefix);
- fprintf(stderr, " die - stop execution now\n");
- fprintf(stderr, " abort - dump core %s\n",
- BX_HAVE_ABORT ? "" : "(Disabled)");
+ if (event->u.logmsg.flag == BX_LOG_ASK_ASKDLG) {
+ int level = event->u.logmsg.level;
+ fprintf(stderr, "========================================================================\n");
+ fprintf(stderr, "Event type: %s\n", SIM->get_log_level_name (level));
+ fprintf(stderr, "Device: %s\n", event->u.logmsg.prefix);
+ fprintf(stderr, "Message: %s\n\n", event->u.logmsg.msg);
+ fprintf(stderr, "A %s has occurred. Do you want to:\n", SIM->get_log_level_name (level));
+ fprintf(stderr, " cont - continue execution\n");
+ fprintf(stderr, " alwayscont - continue execution, and don't ask again.\n");
+ fprintf(stderr, " This affects only %s events from device %s\n", SIM->get_log_level_name (level), event->u.logmsg.prefix);
+ fprintf(stderr, " die - stop execution now\n");
+ fprintf(stderr, " abort - dump core %s\n",
+ BX_HAVE_ABORT ? "" : "(Disabled)");
#if BX_DEBUGGER
- fprintf(stderr, " debug - continue and return to bochs debugger\n");
+ fprintf(stderr, " debug - continue and return to bochs debugger\n");
#endif
#if BX_GDBSTUB
- fprintf(stderr, " debug - hand control to gdb\n");
+ fprintf(stderr, " debug - hand control to gdb\n");
#endif
- int choice;
+ int choice;
ask:
- if (ask_menu("Choose one of the actions above: [%s] ", "",
- log_action_n_choices, log_action_ask_choices, 2, &choice) < 0)
- event->retcode = -1;
- // return 0 for continue, 1 for alwayscontinue, 2 for die, 3 for debug.
- if (!BX_HAVE_ABORT && choice==BX_LOG_ASK_CHOICE_DUMP_CORE) goto ask;
- fflush(stdout);
- fflush(stderr);
- event->retcode = choice;
- }
- return event;
- case BX_ASYNC_EVT_REFRESH:
- case BX_ASYNC_EVT_DBG_MSG:
- case BX_ASYNC_EVT_LOG_MSG:
- // The text mode interface does not use these events, so just ignore
- // them.
- return event;
- default:
- fprintf(stderr, "textconfig: notify callback called with event type %04x\n", event->type);
- return event;
+ if (ask_menu("Choose one of the actions above: [%s] ", "",
+ log_action_n_choices, log_action_ask_choices, 2, &choice) < 0)
+ event->retcode = -1;
+ // return 0 for continue, 1 for alwayscontinue, 2 for die, 3 for debug.
+ if (!BX_HAVE_ABORT && choice==BX_LOG_ASK_CHOICE_DUMP_CORE) goto ask;
+ fflush(stdout);
+ fflush(stderr);
+ event->retcode = choice;
+ } else {
+ // warning prompt not implemented
+ event->retcode = 0;
+ }
+ return event;
+ case BX_ASYNC_EVT_REFRESH:
+ case BX_ASYNC_EVT_DBG_MSG:
+ case BX_ASYNC_EVT_LOG_MSG:
+ // The text mode interface does not use these events, so just ignore
+ // them.
+ return event;
+ default:
+ fprintf(stderr, "textconfig: notify callback called with event type %04x\n", event->type);
+ return event;
}
assert(0); // switch statement should return
}
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/win32dialog.cc ./gui/win32dialog.cc
--- /home/volker/bochs/bochs/gui/win32dialog.cc 2014-06-20 11:32:02.034026376 +0200
+++ ./gui/win32dialog.cc 2016-12-28 12:50:14.148888740 +0100
@@ -2,7 +2,7 @@
// $Id: win32dialog.cc 12381 2014-06-20 09:31:56Z vruppert $
/////////////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2003-2014 The Bochs Project
+// Copyright (C) 2003-2016 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
@@ -27,7 +27,7 @@
#include "win32res.h"
#include "win32paramdlg.h"
-const char log_choices[5][16] = {"ignore", "log", "ask user", "end simulation", "no change"};
+const char log_choices[N_ACT+1][16] = {"ignore", "log", "warn user", "ask user", "end simulation", "no change"};
HWND GetBochsWindow()
{
@@ -97,12 +97,16 @@
SetWindowText(GetDlgItem(hDlg, IDASKMSG), event->u.logmsg.msg);
SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Continue");
SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Continue and don't ask again");
- SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Kill simulation");
- SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Abort (dump core)");
+ if (event->u.logmsg.flag == BX_LOG_ASK_ASKDLG) {
+ SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Kill simulation");
+ SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Abort (dump core)");
#if BX_DEBUGGER
- SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Continue and return to debugger");
+ SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Continue and return to debugger");
#endif
- SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_SETCURSEL, 2, 0);
+ SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_SETCURSEL, 2, 0);
+ } else {
+ SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_SETCURSEL, 0, 0);
+ }
SetFocus(GetDlgItem(hDlg, IDASKLIST));
return FALSE;
case WM_CLOSE:
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/wxdialog.cc ./gui/wxdialog.cc
--- /home/volker/bochs/bochs/gui/wxdialog.cc 2015-01-07 17:17:40.447882000 +0100
+++ ./gui/wxdialog.cc 2016-12-27 20:30:44.997609007 +0100
@@ -2,7 +2,7 @@
// $Id: wxdialog.cc 12594 2015-01-07 16:17:40Z sshwarts $
/////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2002-2014 The Bochs Project
+// Copyright (C) 2002-2016 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
@@ -208,7 +208,6 @@
: wxDialog(parent, id, wxT(""), wxDefaultPosition, wxDefaultSize,
wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER)
{
- //static int integers[LOG_OPTS_N_CHOICES_NORMAL] = {0, 1, 2, 3};
static wxString names[] = ADVLOG_OPTS_TYPE_NAMES;
SetTitle(ADVLOG_OPTS_TITLE);
vertSizer = new wxBoxSizer(wxVERTICAL);
@@ -1563,7 +1562,7 @@
bool includeNoChange)
{
static wxString choices[] = LOG_OPTS_CHOICES;
- static int integers[LOG_OPTS_N_CHOICES] = {0, 1, 2, 3, 4};
+ static int integers[LOG_OPTS_N_CHOICES] = {0, 1, 2, 3, 4, 5};
wxChoice *control = new wxChoice(parent, id, wxDefaultPosition, wxDefaultSize);
int lastChoice = 0; // remember index of last choice
int nchoice = includeNoChange? LOG_OPTS_N_CHOICES : LOG_OPTS_N_CHOICES_NORMAL;
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/wxdialog.h ./gui/wxdialog.h
--- /home/volker/bochs/bochs/gui/wxdialog.h 2014-12-23 20:30:12.896090221 +0100
+++ ./gui/wxdialog.h 2016-12-27 20:34:28.518389938 +0100
@@ -2,7 +2,7 @@
// $Id: wxdialog.h 12576 2014-12-23 19:30:03Z vruppert $
////////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2002-2014 The Bochs Project
+// Copyright (C) 2002-2016 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
@@ -354,10 +354,10 @@
#define LOG_OPTS_PROMPT wxT("How should Bochs respond to each type of event?")
#define LOG_OPTS_TYPE_NAMES { wxT("Debug events"), wxT("Info events"), wxT("Error events"), wxT("Panic events") }
#define LOG_OPTS_N_TYPES 4
-#define LOG_OPTS_CHOICES { wxT("ignore"), wxT("log"), wxT("ask user"), wxT("end simulation"), wxT("no change") }
-#define LOG_OPTS_N_CHOICES_NORMAL 4
-#define LOG_OPTS_N_CHOICES 5 // number of choices, including "no change"
-#define LOG_OPTS_NO_CHANGE 4 // index of "no change"
+#define LOG_OPTS_CHOICES { wxT("ignore"), wxT("log"), wxT("warn user"), wxT("ask user"), wxT("end simulation"), wxT("no change") }
+#define LOG_OPTS_N_CHOICES_NORMAL 5
+#define LOG_OPTS_N_CHOICES 6 // number of choices, including "no change"
+#define LOG_OPTS_NO_CHANGE 5 // index of "no change"
#define LOG_OPTS_ADV wxT("For additional control over how each device responds to events, use the menu option \"Log ... By Device\".")
wxFlexGridSizer *gridSizer;
wxChoice *action[LOG_OPTS_N_TYPES];
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/wxmain.cc ./gui/wxmain.cc
--- /home/volker/bochs/bochs/gui/wxmain.cc 2016-12-26 17:12:57.470174541 +0100
+++ ./gui/wxmain.cc 2016-12-28 12:15:26.035961463 +0100
@@ -2,7 +2,7 @@
// $Id: wxmain.cc 13006 2016-12-26 16:12:54Z vruppert $
/////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2002-2014 The Bochs Project
+// Copyright (C) 2002-2016 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
@@ -1158,6 +1158,10 @@
#if !BX_DEBUGGER && !BX_GDBSTUB
dlg.EnableButton(dlg.DEBUG, FALSE);
#endif
+ if (be->u.logmsg.flag != BX_LOG_ASK_ASKDLG) {
+ dlg.EnableButton(dlg.DIE, FALSE);
+ dlg.EnableButton(dlg.DUMP, FALSE);
+ }
dlg.SetContext(wxString(be->u.logmsg.prefix, wxConvUTF8));
dlg.SetMessage(wxString(be->u.logmsg.msg, wxConvUTF8));
int n = dlg.ShowModal();
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/x.cc ./gui/x.cc
--- /home/volker/bochs/bochs/gui/x.cc 2016-12-27 17:26:59.622665119 +0100
+++ ./gui/x.cc 2016-12-28 12:03:10.963351647 +0100
@@ -2687,11 +2687,7 @@
} else {
size_x = 30 + maxlen * 6;
}
- if (lines < 3) {
- size_y = 90;
- } else {
- size_y = 60 + lines * 15;
- }
+ size_y = 70 + lines * 15;
x11_dialog_c *xdlg = new x11_dialog_c(name, size_x, size_y,
(mode == XDLG_SIMPLE) ? 1 : 2);
ypos = 34;
@@ -2729,11 +2725,21 @@
bx_param_string_c *sparam;
bx_param_enum_c *eparam;
bx_list_c *list;
+ char message[256];
switch (event->type)
{
case BX_SYNC_EVT_LOG_ASK:
- event->retcode = x11_ask_dialog(event);
+ if (event->u.logmsg.flag == BX_LOG_ASK_ASKDLG) {
+ event->retcode = x11_ask_dialog(event);
+ } else if (event->u.logmsg.flag == BX_LOG_ASK_MSGBOX_WARN) {
+ const char *title = SIM->get_log_level_name(event->u.logmsg.level);
+ sprintf(message, "Device: %s\n\nMessage: %s", event->u.logmsg.prefix,
+ event->u.logmsg.msg);
+ bx_param_bool_c bparam(NULL, "warn", title, message, 1);
+ x11_message_box(&bparam, XDLG_SIMPLE);
+ event->retcode = 0;
+ }
return event;
case BX_SYNC_EVT_ASK_PARAM:
param = event->u.param.param;
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/logio.cc ./logio.cc
--- /home/volker/bochs/bochs/logio.cc 2015-05-10 08:55:18.678940963 +0200
+++ ./logio.cc 2016-12-28 00:40:40.395736643 +0100
@@ -2,7 +2,7 @@
// $Id: logio.cc 12759 2015-05-10 06:55:16Z vruppert $
/////////////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2001-2014 The Bochs Project
+// Copyright (C) 2001-2016 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
@@ -50,11 +50,25 @@
else return "?";
}
+static const char *act_name[N_ACT] = { "ignore", "report", "warn", "ask", "fatal" };
+
const char* iofunctions::getaction(int i) const
{
- static const char *name[] = { "ignore", "report", "ask", "fatal" };
assert (i>=ACT_IGNORE && i<N_ACT);
- return name[i];
+ return act_name[i];
+}
+
+int iofunctions::isaction(const char *val) const
+{
+ int action = -1;
+
+ for (int i = 0; i < N_ACT; i++) {
+ if (!strcmp(val, act_name[i])) {
+ action = ACT_IGNORE + i;
+ break;
+ }
+ }
+ return action;
}
void iofunctions::flush(void)
@@ -414,6 +428,11 @@
logio->out(LOGLEV_ERROR, prefix, fmt, ap);
va_end(ap);
+ if (onoff[LOGLEV_ERROR] == ACT_WARN) {
+ va_start(ap, fmt);
+ warn(LOGLEV_ERROR, prefix, fmt, ap);
+ va_end(ap);
+ }
if (onoff[LOGLEV_ERROR] == ACT_ASK) {
va_start(ap, fmt);
ask(LOGLEV_ERROR, prefix, fmt, ap);
@@ -438,6 +457,11 @@
logio->out(LOGLEV_PANIC, prefix, fmt, ap);
va_end(ap);
+ if (onoff[LOGLEV_PANIC] == ACT_WARN) {
+ va_start(ap, fmt);
+ warn(LOGLEV_PANIC, prefix, fmt, ap);
+ va_end(ap);
+ }
if (onoff[LOGLEV_PANIC] == ACT_ASK) {
va_start(ap, fmt);
ask(LOGLEV_PANIC, prefix, fmt, ap);
@@ -465,6 +489,36 @@
// the actions ask() and fatal() are not supported here
}
+void logfunctions::warn(int level, const char *prefix, const char *fmt, va_list ap)
+{
+ // Guard against reentry on warn() function. The danger is that some
+ // function that's called within warn() could trigger another
+ // BX_ERROR that could call warn() again, leading to infinite
+ // recursion and infinite asks.
+ static char in_warn_already = 0;
+ char buf1[1024];
+ if (in_warn_already) {
+ fprintf(stderr, "logfunctions::warn() should not reenter!!\n");
+ return;
+ }
+ in_warn_already = 1;
+ vsnprintf(buf1, sizeof(buf1), fmt, ap);
+ // FIXME: facility set to 0 because it's unknown.
+
+ // update vga screen. This is useful because sometimes useful messages
+ // are printed on the screen just before a panic. It's also potentially
+ // dangerous if this function calls ask again... That's why I added
+ // the reentry check above.
+ SIM->refresh_vga();
+
+ // ensure the text screen is showing
+ SIM->set_display_mode(DISP_MODE_CONFIG);
+ SIM->log_warn(prefix, level, buf1);
+ // return to simulation mode
+ SIM->set_display_mode(DISP_MODE_SIM);
+ in_warn_already = 0;
+}
+
void logfunctions::ask(int level, const char *prefix, const char *fmt, va_list ap)
{
// Guard against reentry on ask() function. The danger is that some
2016-12-28 18:06:34 +03:00
|
|
|
sprintf(message, "Device: %s\nMessage: %s", event->u.logmsg.prefix,
|
|
|
|
event->u.logmsg.msg);
|
2014-06-29 11:37:58 +04:00
|
|
|
msgboxdata.flags = SDL_MESSAGEBOX_ERROR;
|
|
|
|
msgboxdata.window = window;
|
|
|
|
msgboxdata.title = SIM->get_log_level_name(level);
|
|
|
|
msgboxdata.message = message;
|
|
|
|
msgboxdata.buttons = buttondata;
|
|
|
|
msgboxdata.colorScheme = NULL;
|
2016-12-29 12:47:07 +03:00
|
|
|
mode = event->u.logmsg.mode;
|
|
|
|
if ((mode == BX_LOG_DLG_ASK) || (mode == BX_LOG_DLG_WARN)) {
|
|
|
|
buttondata[0].flags = 0;
|
|
|
|
buttondata[0].buttonid = BX_LOG_ASK_CHOICE_CONTINUE;
|
|
|
|
buttondata[0].text = "Continue";
|
|
|
|
buttondata[1].flags = 0;
|
|
|
|
buttondata[1].buttonid = BX_LOG_ASK_CHOICE_CONTINUE_ALWAYS;
|
|
|
|
buttondata[1].text = "Alwayscont";
|
|
|
|
i = 2;
|
|
|
|
}
|
2014-06-29 11:37:58 +04:00
|
|
|
#if BX_DEBUGGER || BX_GDBSTUB
|
2016-12-29 12:47:07 +03:00
|
|
|
if (mode == BX_LOG_DLG_ASK) {
|
|
|
|
buttondata[i].flags = 0;
|
|
|
|
buttondata[i].buttonid = BX_LOG_ASK_CHOICE_ENTER_DEBUG;
|
|
|
|
buttondata[i].text = "Debugger";
|
|
|
|
i++;
|
|
|
|
}
|
2014-06-29 11:37:58 +04:00
|
|
|
#endif
|
2016-12-29 12:47:07 +03:00
|
|
|
if ((mode == BX_LOG_DLG_ASK) || (mode == BX_LOG_DLG_QUIT)) {
|
|
|
|
buttondata[i].flags = SDL_MESSAGEBOX_BUTTON_ESCAPEKEY_DEFAULT;
|
|
|
|
buttondata[i].buttonid = BX_LOG_ASK_CHOICE_DIE;
|
|
|
|
buttondata[i].text = "Quit";
|
|
|
|
i++;
|
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/bochs.h ./bochs.h
--- /home/volker/bochs/bochs/bochs.h 2016-08-12 19:06:18.803209189 +0200
+++ ./bochs.h 2016-12-28 00:41:20.000627252 +0100
@@ -2,7 +2,7 @@
// $Id: bochs.h 12935 2016-08-12 17:06:14Z vruppert $
/////////////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2001-2015 The Bochs Project
+// Copyright (C) 2001-2016 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
@@ -276,8 +276,9 @@
void error(const char *fmt, ...) BX_CPP_AttrPrintf(2, 3);
void panic(const char *fmt, ...) BX_CPP_AttrPrintf(2, 3);
void ldebug(const char *fmt, ...) BX_CPP_AttrPrintf(2, 3);
- void fatal (const char *prefix, const char *fmt, va_list ap, int exit_status);
- void ask (int level, const char *prefix, const char *fmt, va_list ap);
+ void fatal(const char *prefix, const char *fmt, va_list ap, int exit_status);
+ void warn(int level, const char *prefix, const char *fmt, va_list ap);
+ void ask(int level, const char *prefix, const char *fmt, va_list ap);
void put(const char *p);
void put(const char *n, const char *p);
void setio(class iofunctions *);
@@ -334,7 +335,8 @@
void set_log_action(int loglevel, int action);
const char *getlevel(int i) const;
const char *getaction(int i) const;
-
+ int isaction(const char *val) const;
+
protected:
int n_logfn;
#define MAX_LOGFNS 512
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/CHANGES ./CHANGES
--- /home/volker/bochs/bochs/CHANGES 2016-12-26 10:45:44.000000000 +0100
+++ ./CHANGES 2016-12-28 15:54:25.127088081 +0100
@@ -1,5 +1,8 @@
Changes after 2.6.8 release:
+- General
+ - Added new log action "warn", designed to show a message box on error events.
+
- Configure and compile
- Added Android host platform support.
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/config.cc ./config.cc
--- /home/volker/bochs/bochs/config.cc 2016-05-03 21:15:09.158016000 +0200
+++ ./config.cc 2016-12-27 19:53:10.461420368 +0100
@@ -2062,15 +2062,8 @@
actstr = strtok(NULL, "");
if (actstr != NULL) {
def_action = !strcmp(module, "action");
- if (!strcmp(actstr, "fatal"))
- action = ACT_FATAL;
- else if (!strcmp (actstr, "report"))
- action = ACT_REPORT;
- else if (!strcmp (actstr, "ignore"))
- action = ACT_IGNORE;
- else if (!strcmp (actstr, "ask"))
- action = ACT_ASK;
- else {
+ action = SIM->is_action_name(actstr);
+ if (action < ACT_IGNORE) {
PARSE_ERR(("%s: %s directive malformed.", context, params[0]));
free(param);
return -1;
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/sdl2.cc ./gui/sdl2.cc
--- /home/volker/bochs/bochs/gui/sdl2.cc 2016-08-12 19:06:18.811209142 +0200
+++ ./gui/sdl2.cc 2016-12-28 12:33:39.534288819 +0100
@@ -2,7 +2,7 @@
// $Id: sdl2.cc 12935 2016-08-12 17:06:14Z vruppert $
/////////////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2014-2015 The Bochs Project
+// Copyright (C) 2014-2016 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
@@ -1478,20 +1478,16 @@
SDL_MessageBoxData msgboxdata;
SDL_MessageBoxButtonData buttondata[4];
int level, retcode;
-#if BX_DEBUGGER || BX_GDBSTUB
- int defbtn = 3;
-#else
- int defbtn = 2;
-#endif
char message[512];
level = event->u.logmsg.level;
- sprintf(message, "%s %s", event->u.logmsg.prefix, event->u.logmsg.msg);
+ sprintf(message, "Device: %s\nMessage: %s", event->u.logmsg.prefix,
+ event->u.logmsg.msg);
msgboxdata.flags = SDL_MESSAGEBOX_ERROR;
msgboxdata.window = window;
msgboxdata.title = SIM->get_log_level_name(level);
msgboxdata.message = message;
- msgboxdata.numbuttons = defbtn + 1;
+ msgboxdata.numbuttons = 2;
msgboxdata.buttons = buttondata;
msgboxdata.colorScheme = NULL;
buttondata[0].flags = 0;
@@ -1500,14 +1496,18 @@
buttondata[1].flags = 0;
buttondata[1].buttonid = BX_LOG_ASK_CHOICE_CONTINUE_ALWAYS;
buttondata[1].text = "Alwayscont";
+ if (event->u.logmsg.flag == BX_LOG_ASK_ASKDLG) {
+ msgboxdata.numbuttons = 3;
+ buttondata[2].flags = SDL_MESSAGEBOX_BUTTON_ESCAPEKEY_DEFAULT;
+ buttondata[2].buttonid = BX_LOG_ASK_CHOICE_DIE;
+ buttondata[2].text = "Quit";
#if BX_DEBUGGER || BX_GDBSTUB
- buttondata[2].flags = 0;
- buttondata[2].buttonid = BX_LOG_ASK_CHOICE_ENTER_DEBUG;
- buttondata[2].text = "Debugger";
-#endif
- buttondata[defbtn].flags = SDL_MESSAGEBOX_BUTTON_ESCAPEKEY_DEFAULT;
- buttondata[defbtn].buttonid = BX_LOG_ASK_CHOICE_DIE;
- buttondata[defbtn].text = "Quit";
+ msgboxdata.numbuttons = 4;
+ buttondata[3].flags = 0;
+ buttondata[3].buttonid = BX_LOG_ASK_CHOICE_ENTER_DEBUG;
+ buttondata[3].text = "Debugger";
+#endif
+ }
if (SDL_ShowMessageBox(&msgboxdata, &retcode) < 0) {
return -1;
} else {
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/siminterface.cc ./gui/siminterface.cc
--- /home/volker/bochs/bochs/gui/siminterface.cc 2016-12-05 19:56:56.729685000 +0100
+++ ./gui/siminterface.cc 2016-12-28 11:14:02.004075717 +0100
@@ -2,7 +2,7 @@
// $Id: siminterface.cc 12981 2016-12-05 18:56:56Z sshwarts $
/////////////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2002-2015 The Bochs Project
+// Copyright (C) 2002-2016 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
@@ -100,6 +100,7 @@
virtual int get_log_action(int mod, int level);
virtual void set_log_action(int mod, int level, int action);
virtual const char *get_action_name(int action);
+ virtual int is_action_name(const char *val);
virtual int get_default_log_action(int level) {
return logfunctions::get_default_action(level);
}
@@ -123,6 +124,7 @@
virtual void set_notify_callback(bxevent_handler func, void *arg);
virtual void get_notify_callback(bxevent_handler *func, void **arg);
virtual BxEvent* sim_to_ci_event(BxEvent *event);
+ virtual int log_warn(const char *prefix, int level, const char *msg);
virtual int log_ask(const char *prefix, int level, const char *msg);
virtual void log_msg(const char *prefix, int level, const char *msg);
virtual void set_log_viewer(bx_bool val) { bx_log_viewer = val; }
@@ -420,6 +422,11 @@
return io->getaction(action);
}
+int bx_real_sim_c::is_action_name(const char *val)
+{
+ return io->isaction(val);
+}
+
const char *bx_real_sim_c::get_log_level_name(int level)
{
return io->getlevel(level);
@@ -575,6 +582,21 @@
}
}
+int bx_real_sim_c::log_warn(const char *prefix, int level, const char *msg)
+{
+ BxEvent be;
+ be.type = BX_SYNC_EVT_LOG_ASK;
+ be.u.logmsg.prefix = prefix;
+ be.u.logmsg.level = level;
+ be.u.logmsg.msg = msg;
+ be.u.logmsg.flag = BX_LOG_ASK_MSGBOX_WARN;
+ // default return value in case something goes wrong.
+ be.retcode = BX_LOG_NOTIFY_FAILED;
+ // calling notify
+ sim_to_ci_event(&be);
+ return be.retcode;
+}
+
// returns 0 for continue, 1 for alwayscontinue, 2 for die.
int bx_real_sim_c::log_ask(const char *prefix, int level, const char *msg)
{
@@ -583,6 +605,7 @@
be.u.logmsg.prefix = prefix;
be.u.logmsg.level = level;
be.u.logmsg.msg = msg;
+ be.u.logmsg.flag = BX_LOG_ASK_ASKDLG;
// default return value in case something goes wrong.
be.retcode = BX_LOG_NOTIFY_FAILED;
// calling notify
@@ -1157,16 +1180,10 @@
} else if (!strncmp(string, "PANIC=", 6)) {
type = LOGLEV_PANIC;
}
- if (!strcmp(string+j, "ignore")) {
- action = ACT_IGNORE;
- } else if (!strcmp(string+j, "report")) {
- action = ACT_REPORT;
- } else if (!strcmp(string+j, "ask")) {
- action = ACT_ASK;
- } else if (!strcmp(string+j, "fatal")) {
- action = ACT_FATAL;
+ action = is_action_name(string+j);
+ if (action >= ACT_IGNORE) {
+ set_log_action(dev, type, action);
}
- set_log_action(dev, type, action);
} else {
if (i == 1) {
BX_ERROR(("restore_logopts(): log module '%s' not found", devname));
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/siminterface.h ./gui/siminterface.h
--- /home/volker/bochs/bochs/gui/siminterface.h 2016-03-31 19:24:37.451025427 +0200
+++ ./gui/siminterface.h 2016-12-28 11:11:21.036683362 +0100
@@ -168,6 +168,7 @@
typedef enum {
ACT_IGNORE = 0,
ACT_REPORT,
+ ACT_WARN,
ACT_ASK,
ACT_FATAL,
N_ACT
@@ -178,11 +179,11 @@
// normally all action choices are available for all event types. The exclude
// expression allows some choices to be eliminated if they don't make any
// sense. For example, it would be stupid to ignore a panic.
-#define BX_LOG_OPTS_EXCLUDE(type, choice) ( \
- /* can't die or ask, on debug or info events */ \
- (type <= LOGLEV_INFO && (choice == ACT_ASK || choice == ACT_FATAL)) \
- /* can't ignore panics */ \
- || (type == LOGLEV_PANIC && choice == ACT_IGNORE) \
+#define BX_LOG_OPTS_EXCLUDE(type, choice) ( \
+ /* can't die, ask or warn, on debug or info events */ \
+ (type <= LOGLEV_INFO && (choice >= ACT_WARN)) \
+ /* can't ignore panics */ \
+ || (type == LOGLEV_PANIC && choice == ACT_IGNORE) \
)
// floppy / cdrom media status
@@ -392,6 +393,7 @@
// synchronizing threads, etc. for each.
typedef struct {
Bit8u level;
+ Bit8u flag;
const char *prefix;
const char *msg;
} BxLogMsgEvent;
@@ -419,6 +421,12 @@
BX_LOG_NOTIFY_FAILED
};
+enum {
+ BX_LOG_ASK_ASKDLG,
+ BX_LOG_ASK_MSGBOX_WARN,
+ BX_LOG_ASK_MSGBOX_QUIT
+};
+
// Event type: BX_SYNC_EVT_GET_DBG_COMMAND
//
// This is a synchronous event sent from the simulator to the debugger
@@ -675,6 +683,7 @@
virtual int get_default_log_action(int level) {return -1;}
virtual void set_default_log_action(int level, int action) {}
virtual const char *get_action_name(int action) {return NULL;}
+ virtual int is_action_name(const char *val) {return -1;}
virtual const char *get_log_level_name(int level) {return NULL;}
virtual int get_max_log_level() {return -1;}
@@ -715,6 +724,9 @@
// send an event from the simulator to the CI.
virtual BxEvent* sim_to_ci_event(BxEvent *event) {return NULL;}
+ // called from simulator when it hits errors, to warn the user
+ // before continuing simulation
+ virtual int log_warn(const char *prefix, int level, const char *msg) {return -1;}
// called from simulator when it hits serious errors, to ask if the user
// wants to continue or not
virtual int log_ask(const char *prefix, int level, const char *msg) {return -1;}
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/textconfig.cc ./gui/textconfig.cc
--- /home/volker/bochs/bochs/gui/textconfig.cc 2016-12-05 20:15:59.112637000 +0100
+++ ./gui/textconfig.cc 2016-12-28 12:44:43.079411258 +0100
@@ -2,7 +2,7 @@
// $Id: textconfig.cc 12983 2016-12-05 19:15:59Z sshwarts $
/////////////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2002-2013 The Bochs Project
+// Copyright (C) 2002-2016 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
@@ -552,8 +552,8 @@
}
static const char *log_options_prompt1 = "Enter the ID of the device to edit, or -1 to return: [-1] ";
-static const char *log_level_choices[] = { "ignore", "report", "ask", "fatal", "no change" };
-static int log_level_n_choices_normal = 4;
+static const char *log_level_choices[N_ACT+1] = { "ignore", "report", "warn", "ask", "fatal", "no change" };
+static int log_level_n_choices_normal = N_ACT;
void bx_log_options(int individual)
{
@@ -589,7 +589,7 @@
bx_print_log_action_table();
for (int level=0; level<SIM->get_max_log_level(); level++) {
char prompt[1024];
- int action, default_action = 4; // default to no change
+ int action, default_action = N_ACT; // default to no change
sprintf(prompt, "Enter action for %s event on all devices: [no change] ", SIM->get_log_level_name(level));
// do show the no change choice (choices=4)
if (ask_menu(prompt, "", log_level_n_choices_normal+1, log_level_choices, default_action, &action)<0)
@@ -718,47 +718,50 @@
event->retcode = event->u.param.param->text_ask(stdin, stderr);
return event;
case BX_SYNC_EVT_LOG_ASK:
- {
- int level = event->u.logmsg.level;
- fprintf(stderr, "========================================================================\n");
- fprintf(stderr, "Event type: %s\n", SIM->get_log_level_name (level));
- fprintf(stderr, "Device: %s\n", event->u.logmsg.prefix);
- fprintf(stderr, "Message: %s\n\n", event->u.logmsg.msg);
- fprintf(stderr, "A %s has occurred. Do you want to:\n", SIM->get_log_level_name (level));
- fprintf(stderr, " cont - continue execution\n");
- fprintf(stderr, " alwayscont - continue execution, and don't ask again.\n");
- fprintf(stderr, " This affects only %s events from device %s\n", SIM->get_log_level_name (level), event->u.logmsg.prefix);
- fprintf(stderr, " die - stop execution now\n");
- fprintf(stderr, " abort - dump core %s\n",
- BX_HAVE_ABORT ? "" : "(Disabled)");
+ if (event->u.logmsg.flag == BX_LOG_ASK_ASKDLG) {
+ int level = event->u.logmsg.level;
+ fprintf(stderr, "========================================================================\n");
+ fprintf(stderr, "Event type: %s\n", SIM->get_log_level_name (level));
+ fprintf(stderr, "Device: %s\n", event->u.logmsg.prefix);
+ fprintf(stderr, "Message: %s\n\n", event->u.logmsg.msg);
+ fprintf(stderr, "A %s has occurred. Do you want to:\n", SIM->get_log_level_name (level));
+ fprintf(stderr, " cont - continue execution\n");
+ fprintf(stderr, " alwayscont - continue execution, and don't ask again.\n");
+ fprintf(stderr, " This affects only %s events from device %s\n", SIM->get_log_level_name (level), event->u.logmsg.prefix);
+ fprintf(stderr, " die - stop execution now\n");
+ fprintf(stderr, " abort - dump core %s\n",
+ BX_HAVE_ABORT ? "" : "(Disabled)");
#if BX_DEBUGGER
- fprintf(stderr, " debug - continue and return to bochs debugger\n");
+ fprintf(stderr, " debug - continue and return to bochs debugger\n");
#endif
#if BX_GDBSTUB
- fprintf(stderr, " debug - hand control to gdb\n");
+ fprintf(stderr, " debug - hand control to gdb\n");
#endif
- int choice;
+ int choice;
ask:
- if (ask_menu("Choose one of the actions above: [%s] ", "",
- log_action_n_choices, log_action_ask_choices, 2, &choice) < 0)
- event->retcode = -1;
- // return 0 for continue, 1 for alwayscontinue, 2 for die, 3 for debug.
- if (!BX_HAVE_ABORT && choice==BX_LOG_ASK_CHOICE_DUMP_CORE) goto ask;
- fflush(stdout);
- fflush(stderr);
- event->retcode = choice;
- }
- return event;
- case BX_ASYNC_EVT_REFRESH:
- case BX_ASYNC_EVT_DBG_MSG:
- case BX_ASYNC_EVT_LOG_MSG:
- // The text mode interface does not use these events, so just ignore
- // them.
- return event;
- default:
- fprintf(stderr, "textconfig: notify callback called with event type %04x\n", event->type);
- return event;
+ if (ask_menu("Choose one of the actions above: [%s] ", "",
+ log_action_n_choices, log_action_ask_choices, 2, &choice) < 0)
+ event->retcode = -1;
+ // return 0 for continue, 1 for alwayscontinue, 2 for die, 3 for debug.
+ if (!BX_HAVE_ABORT && choice==BX_LOG_ASK_CHOICE_DUMP_CORE) goto ask;
+ fflush(stdout);
+ fflush(stderr);
+ event->retcode = choice;
+ } else {
+ // warning prompt not implemented
+ event->retcode = 0;
+ }
+ return event;
+ case BX_ASYNC_EVT_REFRESH:
+ case BX_ASYNC_EVT_DBG_MSG:
+ case BX_ASYNC_EVT_LOG_MSG:
+ // The text mode interface does not use these events, so just ignore
+ // them.
+ return event;
+ default:
+ fprintf(stderr, "textconfig: notify callback called with event type %04x\n", event->type);
+ return event;
}
assert(0); // switch statement should return
}
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/win32dialog.cc ./gui/win32dialog.cc
--- /home/volker/bochs/bochs/gui/win32dialog.cc 2014-06-20 11:32:02.034026376 +0200
+++ ./gui/win32dialog.cc 2016-12-28 12:50:14.148888740 +0100
@@ -2,7 +2,7 @@
// $Id: win32dialog.cc 12381 2014-06-20 09:31:56Z vruppert $
/////////////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2003-2014 The Bochs Project
+// Copyright (C) 2003-2016 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
@@ -27,7 +27,7 @@
#include "win32res.h"
#include "win32paramdlg.h"
-const char log_choices[5][16] = {"ignore", "log", "ask user", "end simulation", "no change"};
+const char log_choices[N_ACT+1][16] = {"ignore", "log", "warn user", "ask user", "end simulation", "no change"};
HWND GetBochsWindow()
{
@@ -97,12 +97,16 @@
SetWindowText(GetDlgItem(hDlg, IDASKMSG), event->u.logmsg.msg);
SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Continue");
SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Continue and don't ask again");
- SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Kill simulation");
- SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Abort (dump core)");
+ if (event->u.logmsg.flag == BX_LOG_ASK_ASKDLG) {
+ SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Kill simulation");
+ SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Abort (dump core)");
#if BX_DEBUGGER
- SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Continue and return to debugger");
+ SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_ADDSTRING, 0, (LPARAM)"Continue and return to debugger");
#endif
- SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_SETCURSEL, 2, 0);
+ SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_SETCURSEL, 2, 0);
+ } else {
+ SendMessage(GetDlgItem(hDlg, IDASKLIST), LB_SETCURSEL, 0, 0);
+ }
SetFocus(GetDlgItem(hDlg, IDASKLIST));
return FALSE;
case WM_CLOSE:
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/wxdialog.cc ./gui/wxdialog.cc
--- /home/volker/bochs/bochs/gui/wxdialog.cc 2015-01-07 17:17:40.447882000 +0100
+++ ./gui/wxdialog.cc 2016-12-27 20:30:44.997609007 +0100
@@ -2,7 +2,7 @@
// $Id: wxdialog.cc 12594 2015-01-07 16:17:40Z sshwarts $
/////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2002-2014 The Bochs Project
+// Copyright (C) 2002-2016 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
@@ -208,7 +208,6 @@
: wxDialog(parent, id, wxT(""), wxDefaultPosition, wxDefaultSize,
wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER)
{
- //static int integers[LOG_OPTS_N_CHOICES_NORMAL] = {0, 1, 2, 3};
static wxString names[] = ADVLOG_OPTS_TYPE_NAMES;
SetTitle(ADVLOG_OPTS_TITLE);
vertSizer = new wxBoxSizer(wxVERTICAL);
@@ -1563,7 +1562,7 @@
bool includeNoChange)
{
static wxString choices[] = LOG_OPTS_CHOICES;
- static int integers[LOG_OPTS_N_CHOICES] = {0, 1, 2, 3, 4};
+ static int integers[LOG_OPTS_N_CHOICES] = {0, 1, 2, 3, 4, 5};
wxChoice *control = new wxChoice(parent, id, wxDefaultPosition, wxDefaultSize);
int lastChoice = 0; // remember index of last choice
int nchoice = includeNoChange? LOG_OPTS_N_CHOICES : LOG_OPTS_N_CHOICES_NORMAL;
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/wxdialog.h ./gui/wxdialog.h
--- /home/volker/bochs/bochs/gui/wxdialog.h 2014-12-23 20:30:12.896090221 +0100
+++ ./gui/wxdialog.h 2016-12-27 20:34:28.518389938 +0100
@@ -2,7 +2,7 @@
// $Id: wxdialog.h 12576 2014-12-23 19:30:03Z vruppert $
////////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2002-2014 The Bochs Project
+// Copyright (C) 2002-2016 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
@@ -354,10 +354,10 @@
#define LOG_OPTS_PROMPT wxT("How should Bochs respond to each type of event?")
#define LOG_OPTS_TYPE_NAMES { wxT("Debug events"), wxT("Info events"), wxT("Error events"), wxT("Panic events") }
#define LOG_OPTS_N_TYPES 4
-#define LOG_OPTS_CHOICES { wxT("ignore"), wxT("log"), wxT("ask user"), wxT("end simulation"), wxT("no change") }
-#define LOG_OPTS_N_CHOICES_NORMAL 4
-#define LOG_OPTS_N_CHOICES 5 // number of choices, including "no change"
-#define LOG_OPTS_NO_CHANGE 4 // index of "no change"
+#define LOG_OPTS_CHOICES { wxT("ignore"), wxT("log"), wxT("warn user"), wxT("ask user"), wxT("end simulation"), wxT("no change") }
+#define LOG_OPTS_N_CHOICES_NORMAL 5
+#define LOG_OPTS_N_CHOICES 6 // number of choices, including "no change"
+#define LOG_OPTS_NO_CHANGE 5 // index of "no change"
#define LOG_OPTS_ADV wxT("For additional control over how each device responds to events, use the menu option \"Log ... By Device\".")
wxFlexGridSizer *gridSizer;
wxChoice *action[LOG_OPTS_N_TYPES];
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/wxmain.cc ./gui/wxmain.cc
--- /home/volker/bochs/bochs/gui/wxmain.cc 2016-12-26 17:12:57.470174541 +0100
+++ ./gui/wxmain.cc 2016-12-28 12:15:26.035961463 +0100
@@ -2,7 +2,7 @@
// $Id: wxmain.cc 13006 2016-12-26 16:12:54Z vruppert $
/////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2002-2014 The Bochs Project
+// Copyright (C) 2002-2016 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
@@ -1158,6 +1158,10 @@
#if !BX_DEBUGGER && !BX_GDBSTUB
dlg.EnableButton(dlg.DEBUG, FALSE);
#endif
+ if (be->u.logmsg.flag != BX_LOG_ASK_ASKDLG) {
+ dlg.EnableButton(dlg.DIE, FALSE);
+ dlg.EnableButton(dlg.DUMP, FALSE);
+ }
dlg.SetContext(wxString(be->u.logmsg.prefix, wxConvUTF8));
dlg.SetMessage(wxString(be->u.logmsg.msg, wxConvUTF8));
int n = dlg.ShowModal();
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/gui/x.cc ./gui/x.cc
--- /home/volker/bochs/bochs/gui/x.cc 2016-12-27 17:26:59.622665119 +0100
+++ ./gui/x.cc 2016-12-28 12:03:10.963351647 +0100
@@ -2687,11 +2687,7 @@
} else {
size_x = 30 + maxlen * 6;
}
- if (lines < 3) {
- size_y = 90;
- } else {
- size_y = 60 + lines * 15;
- }
+ size_y = 70 + lines * 15;
x11_dialog_c *xdlg = new x11_dialog_c(name, size_x, size_y,
(mode == XDLG_SIMPLE) ? 1 : 2);
ypos = 34;
@@ -2729,11 +2725,21 @@
bx_param_string_c *sparam;
bx_param_enum_c *eparam;
bx_list_c *list;
+ char message[256];
switch (event->type)
{
case BX_SYNC_EVT_LOG_ASK:
- event->retcode = x11_ask_dialog(event);
+ if (event->u.logmsg.flag == BX_LOG_ASK_ASKDLG) {
+ event->retcode = x11_ask_dialog(event);
+ } else if (event->u.logmsg.flag == BX_LOG_ASK_MSGBOX_WARN) {
+ const char *title = SIM->get_log_level_name(event->u.logmsg.level);
+ sprintf(message, "Device: %s\n\nMessage: %s", event->u.logmsg.prefix,
+ event->u.logmsg.msg);
+ bx_param_bool_c bparam(NULL, "warn", title, message, 1);
+ x11_message_box(&bparam, XDLG_SIMPLE);
+ event->retcode = 0;
+ }
return event;
case BX_SYNC_EVT_ASK_PARAM:
param = event->u.param.param;
diff -urNX /home/volker/exclude-bochs /home/volker/bochs/bochs/logio.cc ./logio.cc
--- /home/volker/bochs/bochs/logio.cc 2015-05-10 08:55:18.678940963 +0200
+++ ./logio.cc 2016-12-28 00:40:40.395736643 +0100
@@ -2,7 +2,7 @@
// $Id: logio.cc 12759 2015-05-10 06:55:16Z vruppert $
/////////////////////////////////////////////////////////////////////////
//
-// Copyright (C) 2001-2014 The Bochs Project
+// Copyright (C) 2001-2016 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
@@ -50,11 +50,25 @@
else return "?";
}
+static const char *act_name[N_ACT] = { "ignore", "report", "warn", "ask", "fatal" };
+
const char* iofunctions::getaction(int i) const
{
- static const char *name[] = { "ignore", "report", "ask", "fatal" };
assert (i>=ACT_IGNORE && i<N_ACT);
- return name[i];
+ return act_name[i];
+}
+
+int iofunctions::isaction(const char *val) const
+{
+ int action = -1;
+
+ for (int i = 0; i < N_ACT; i++) {
+ if (!strcmp(val, act_name[i])) {
+ action = ACT_IGNORE + i;
+ break;
+ }
+ }
+ return action;
}
void iofunctions::flush(void)
@@ -414,6 +428,11 @@
logio->out(LOGLEV_ERROR, prefix, fmt, ap);
va_end(ap);
+ if (onoff[LOGLEV_ERROR] == ACT_WARN) {
+ va_start(ap, fmt);
+ warn(LOGLEV_ERROR, prefix, fmt, ap);
+ va_end(ap);
+ }
if (onoff[LOGLEV_ERROR] == ACT_ASK) {
va_start(ap, fmt);
ask(LOGLEV_ERROR, prefix, fmt, ap);
@@ -438,6 +457,11 @@
logio->out(LOGLEV_PANIC, prefix, fmt, ap);
va_end(ap);
+ if (onoff[LOGLEV_PANIC] == ACT_WARN) {
+ va_start(ap, fmt);
+ warn(LOGLEV_PANIC, prefix, fmt, ap);
+ va_end(ap);
+ }
if (onoff[LOGLEV_PANIC] == ACT_ASK) {
va_start(ap, fmt);
ask(LOGLEV_PANIC, prefix, fmt, ap);
@@ -465,6 +489,36 @@
// the actions ask() and fatal() are not supported here
}
+void logfunctions::warn(int level, const char *prefix, const char *fmt, va_list ap)
+{
+ // Guard against reentry on warn() function. The danger is that some
+ // function that's called within warn() could trigger another
+ // BX_ERROR that could call warn() again, leading to infinite
+ // recursion and infinite asks.
+ static char in_warn_already = 0;
+ char buf1[1024];
+ if (in_warn_already) {
+ fprintf(stderr, "logfunctions::warn() should not reenter!!\n");
+ return;
+ }
+ in_warn_already = 1;
+ vsnprintf(buf1, sizeof(buf1), fmt, ap);
+ // FIXME: facility set to 0 because it's unknown.
+
+ // update vga screen. This is useful because sometimes useful messages
+ // are printed on the screen just before a panic. It's also potentially
+ // dangerous if this function calls ask again... That's why I added
+ // the reentry check above.
+ SIM->refresh_vga();
+
+ // ensure the text screen is showing
+ SIM->set_display_mode(DISP_MODE_CONFIG);
+ SIM->log_warn(prefix, level, buf1);
+ // return to simulation mode
+ SIM->set_display_mode(DISP_MODE_SIM);
+ in_warn_already = 0;
+}
+
void logfunctions::ask(int level, const char *prefix, const char *fmt, va_list ap)
{
// Guard against reentry on ask() function. The danger is that some
2016-12-28 18:06:34 +03:00
|
|
|
}
|
2016-12-29 12:47:07 +03:00
|
|
|
msgboxdata.numbuttons = i;
|
2014-06-29 11:37:58 +04:00
|
|
|
if (SDL_ShowMessageBox(&msgboxdata, &retcode) < 0) {
|
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
return retcode;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-19 23:32:20 +03:00
|
|
|
int sdl2_yesno_dialog(bx_param_bool_c *bparam)
|
|
|
|
{
|
|
|
|
SDL_MessageBoxData msgboxdata;
|
|
|
|
SDL_MessageBoxButtonData buttondata[2];
|
|
|
|
int retcode;
|
|
|
|
|
|
|
|
msgboxdata.flags = SDL_MESSAGEBOX_ERROR;
|
|
|
|
msgboxdata.window = window;
|
|
|
|
msgboxdata.title = bparam->get_label();
|
|
|
|
msgboxdata.message = bparam->get_description();
|
|
|
|
msgboxdata.numbuttons = 2;
|
|
|
|
msgboxdata.buttons = buttondata;
|
|
|
|
msgboxdata.colorScheme = NULL;
|
|
|
|
buttondata[0].flags = 0;
|
|
|
|
buttondata[0].buttonid = 1;
|
|
|
|
buttondata[0].text = "Yes";
|
|
|
|
buttondata[1].flags = SDL_MESSAGEBOX_BUTTON_ESCAPEKEY_DEFAULT;
|
|
|
|
buttondata[1].buttonid = 0;
|
|
|
|
buttondata[1].text = "No";
|
|
|
|
if (SDL_ShowMessageBox(&msgboxdata, &retcode) < 0) {
|
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
bparam->set(retcode);
|
|
|
|
return retcode;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-29 11:37:58 +04:00
|
|
|
BxEvent *sdl2_notify_callback(void *unused, BxEvent *event)
|
|
|
|
{
|
2015-01-19 23:32:20 +03:00
|
|
|
bx_param_c *param;
|
|
|
|
|
2014-06-29 11:37:58 +04:00
|
|
|
switch (event->type) {
|
2016-12-29 12:47:07 +03:00
|
|
|
case BX_SYNC_EVT_LOG_DLG:
|
2014-06-29 11:37:58 +04:00
|
|
|
event->retcode = sdl2_ask_dialog(event);
|
|
|
|
return event;
|
|
|
|
case BX_SYNC_EVT_ASK_PARAM:
|
2015-01-19 23:32:20 +03:00
|
|
|
param = event->u.param.param;
|
|
|
|
if (param->get_type() == BXT_PARAM_BOOL) {
|
|
|
|
event->retcode = sdl2_yesno_dialog((bx_param_bool_c*)param);
|
|
|
|
return event;
|
|
|
|
}
|
2014-06-29 11:37:58 +04:00
|
|
|
case BX_SYNC_EVT_TICK: // called periodically by siminterface.
|
|
|
|
case BX_ASYNC_EVT_REFRESH: // called when some bx_param_c parameters have changed.
|
|
|
|
// fall into default case
|
|
|
|
default:
|
|
|
|
return (*old_callback)(old_callback_arg, event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2014-06-28 00:53:20 +04:00
|
|
|
#endif /* if BX_WITH_SDL2 */
|