1999-10-03 14:16:26 +00:00
|
|
|
/*
|
|
|
|
* prefs_editor_gtk.cpp - Preferences editor, Unix implementation using GTK+
|
|
|
|
*
|
2008-01-01 09:40:36 +00:00
|
|
|
* Basilisk II (C) 1997-2008 Christian Bauer
|
1999-10-03 14:16:26 +00:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "sysdeps.h"
|
|
|
|
|
|
|
|
#include <gtk/gtk.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <dirent.h>
|
2022-05-02 14:40:55 +00:00
|
|
|
#include <unistd.h>
|
1999-10-03 14:16:26 +00:00
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <net/if.h>
|
|
|
|
#include <net/if_arp.h>
|
2022-10-03 18:36:46 +00:00
|
|
|
#include <sys/stat.h>
|
2022-06-03 18:40:40 +00:00
|
|
|
#include <cerrno>
|
|
|
|
|
2022-10-03 18:36:46 +00:00
|
|
|
#ifdef HAfVE_SYS_STAT_H
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#endif
|
|
|
|
|
2002-01-22 17:15:10 +00:00
|
|
|
#ifdef HAVE_GNOMEUI
|
|
|
|
#include <gnome.h>
|
|
|
|
#endif
|
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
#include "user_strings.h"
|
|
|
|
#include "version.h"
|
|
|
|
#include "cdrom.h"
|
|
|
|
#include "xpram.h"
|
|
|
|
#include "prefs.h"
|
|
|
|
#include "prefs_editor.h"
|
|
|
|
|
2006-04-16 21:25:41 +00:00
|
|
|
#define DEBUG 0
|
|
|
|
#include "debug.h"
|
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
// Global variables
|
|
|
|
static GtkWidget *win; // Preferences window
|
2006-04-17 21:22:01 +00:00
|
|
|
static bool start_clicked = false; // Return value of PrefsEditor() function
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Prototypes
|
|
|
|
static void create_volumes_pane(GtkWidget *top);
|
|
|
|
static void create_scsi_pane(GtkWidget *top);
|
|
|
|
static void create_graphics_pane(GtkWidget *top);
|
2000-07-13 17:26:17 +00:00
|
|
|
static void create_input_pane(GtkWidget *top);
|
1999-10-03 14:16:26 +00:00
|
|
|
static void create_serial_pane(GtkWidget *top);
|
|
|
|
static void create_memory_pane(GtkWidget *top);
|
2002-09-17 16:06:37 +00:00
|
|
|
static void create_jit_pane(GtkWidget *top);
|
1999-10-03 14:16:26 +00:00
|
|
|
static void read_settings(void);
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Utility functions
|
|
|
|
*/
|
|
|
|
|
2005-11-22 21:53:24 +00:00
|
|
|
#if ! GLIB_CHECK_VERSION(2,0,0)
|
|
|
|
#define G_OBJECT(obj) GTK_OBJECT(obj)
|
|
|
|
#define g_object_get_data(obj, key) gtk_object_get_data((obj), (key))
|
|
|
|
#define g_object_set_data(obj, key, data) gtk_object_set_data((obj), (key), (data))
|
|
|
|
#endif
|
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
struct opt_desc {
|
|
|
|
int label_id;
|
|
|
|
GtkSignalFunc func;
|
|
|
|
};
|
|
|
|
|
2002-09-17 16:06:37 +00:00
|
|
|
struct combo_desc {
|
|
|
|
int label_id;
|
|
|
|
};
|
|
|
|
|
2005-11-21 21:39:08 +00:00
|
|
|
struct file_req_assoc {
|
|
|
|
file_req_assoc(GtkWidget *r, GtkWidget *e) : req(r), entry(e) {}
|
|
|
|
GtkWidget *req;
|
|
|
|
GtkWidget *entry;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void cb_browse_ok(GtkWidget *button, file_req_assoc *assoc)
|
|
|
|
{
|
|
|
|
gchar *file = (char *)gtk_file_selection_get_filename(GTK_FILE_SELECTION(assoc->req));
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(assoc->entry), file);
|
|
|
|
gtk_widget_destroy(assoc->req);
|
|
|
|
delete assoc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cb_browse(GtkWidget *widget, void *user_data)
|
|
|
|
{
|
|
|
|
GtkWidget *req = gtk_file_selection_new(GetString(STR_BROWSE_TITLE));
|
|
|
|
gtk_signal_connect_object(GTK_OBJECT(req), "delete_event", GTK_SIGNAL_FUNC(gtk_widget_destroy), GTK_OBJECT(req));
|
|
|
|
gtk_signal_connect(GTK_OBJECT(GTK_FILE_SELECTION(req)->ok_button), "clicked", GTK_SIGNAL_FUNC(cb_browse_ok), new file_req_assoc(req, (GtkWidget *)user_data));
|
|
|
|
gtk_signal_connect_object(GTK_OBJECT(GTK_FILE_SELECTION(req)->cancel_button), "clicked", GTK_SIGNAL_FUNC(gtk_widget_destroy), GTK_OBJECT(req));
|
|
|
|
gtk_widget_show(req);
|
|
|
|
}
|
|
|
|
|
|
|
|
static GtkWidget *make_browse_button(GtkWidget *entry)
|
|
|
|
{
|
|
|
|
GtkWidget *button;
|
|
|
|
|
|
|
|
button = gtk_button_new_with_label(GetString(STR_BROWSE_CTRL));
|
|
|
|
gtk_widget_show(button);
|
|
|
|
gtk_signal_connect(GTK_OBJECT(button), "clicked", (GtkSignalFunc)cb_browse, (void *)entry);
|
|
|
|
return button;
|
|
|
|
}
|
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
static void add_menu_item(GtkWidget *menu, int label_id, GtkSignalFunc func)
|
|
|
|
{
|
|
|
|
GtkWidget *item = gtk_menu_item_new_with_label(GetString(label_id));
|
|
|
|
gtk_widget_show(item);
|
|
|
|
gtk_signal_connect(GTK_OBJECT(item), "activate", func, NULL);
|
|
|
|
gtk_menu_append(GTK_MENU(menu), item);
|
|
|
|
}
|
|
|
|
|
|
|
|
static GtkWidget *make_pane(GtkWidget *notebook, int title_id)
|
|
|
|
{
|
|
|
|
GtkWidget *frame, *label, *box;
|
|
|
|
|
|
|
|
frame = gtk_frame_new(NULL);
|
2006-04-17 14:25:57 +00:00
|
|
|
gtk_container_set_border_width(GTK_CONTAINER(frame), 4);
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
box = gtk_vbox_new(FALSE, 4);
|
|
|
|
gtk_container_set_border_width(GTK_CONTAINER(box), 4);
|
|
|
|
gtk_container_add(GTK_CONTAINER(frame), box);
|
2006-04-17 14:25:57 +00:00
|
|
|
|
|
|
|
gtk_widget_show_all(frame);
|
|
|
|
|
|
|
|
label = gtk_label_new(GetString(title_id));
|
|
|
|
gtk_notebook_append_page(GTK_NOTEBOOK(notebook), frame, label);
|
1999-10-03 14:16:26 +00:00
|
|
|
return box;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GtkWidget *make_button_box(GtkWidget *top, int border, const opt_desc *buttons)
|
|
|
|
{
|
|
|
|
GtkWidget *bb, *button;
|
|
|
|
|
|
|
|
bb = gtk_hbutton_box_new();
|
|
|
|
gtk_widget_show(bb);
|
|
|
|
gtk_container_set_border_width(GTK_CONTAINER(bb), border);
|
|
|
|
gtk_button_box_set_layout(GTK_BUTTON_BOX(bb), GTK_BUTTONBOX_DEFAULT_STYLE);
|
|
|
|
gtk_button_box_set_spacing(GTK_BUTTON_BOX(bb), 4);
|
|
|
|
gtk_box_pack_start(GTK_BOX(top), bb, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
while (buttons->label_id) {
|
|
|
|
button = gtk_button_new_with_label(GetString(buttons->label_id));
|
|
|
|
gtk_widget_show(button);
|
|
|
|
gtk_signal_connect_object(GTK_OBJECT(button), "clicked", buttons->func, NULL);
|
|
|
|
gtk_box_pack_start(GTK_BOX(bb), button, TRUE, TRUE, 0);
|
|
|
|
buttons++;
|
|
|
|
}
|
|
|
|
return bb;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GtkWidget *make_separator(GtkWidget *top)
|
|
|
|
{
|
|
|
|
GtkWidget *sep = gtk_hseparator_new();
|
|
|
|
gtk_box_pack_start(GTK_BOX(top), sep, FALSE, FALSE, 0);
|
|
|
|
gtk_widget_show(sep);
|
|
|
|
return sep;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GtkWidget *make_table(GtkWidget *top, int x, int y)
|
|
|
|
{
|
|
|
|
GtkWidget *table = gtk_table_new(x, y, FALSE);
|
|
|
|
gtk_widget_show(table);
|
|
|
|
gtk_box_pack_start(GTK_BOX(top), table, FALSE, FALSE, 0);
|
|
|
|
return table;
|
|
|
|
}
|
|
|
|
|
2005-11-21 21:39:08 +00:00
|
|
|
static GtkWidget *table_make_option_menu(GtkWidget *table, int row, int label_id, const opt_desc *options, int active)
|
|
|
|
{
|
|
|
|
GtkWidget *label, *opt, *menu;
|
|
|
|
|
|
|
|
label = gtk_label_new(GetString(label_id));
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), label, 0, 1, row, row + 1, (GtkAttachOptions)0, (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
|
|
|
opt = gtk_option_menu_new();
|
|
|
|
gtk_widget_show(opt);
|
|
|
|
menu = gtk_menu_new();
|
|
|
|
|
|
|
|
while (options->label_id) {
|
|
|
|
add_menu_item(menu, options->label_id, options->func);
|
|
|
|
options++;
|
|
|
|
}
|
|
|
|
gtk_menu_set_active(GTK_MENU(menu), active);
|
|
|
|
|
|
|
|
gtk_option_menu_set_menu(GTK_OPTION_MENU(opt), menu);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), opt, 1, 2, row, row + 1, (GtkAttachOptions)(GTK_FILL | GTK_EXPAND), (GtkAttachOptions)0, 4, 4);
|
|
|
|
return menu;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GtkWidget *table_make_combobox(GtkWidget *table, int row, int label_id, const char *default_value, GList *glist)
|
|
|
|
{
|
|
|
|
GtkWidget *label, *combo;
|
|
|
|
|
|
|
|
label = gtk_label_new(GetString(label_id));
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), label, 0, 1, row, row + 1, (GtkAttachOptions)0, (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
|
|
|
combo = gtk_combo_new();
|
|
|
|
gtk_widget_show(combo);
|
|
|
|
gtk_combo_set_popdown_strings(GTK_COMBO(combo), glist);
|
|
|
|
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(GTK_COMBO(combo)->entry), default_value);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), combo, 1, 2, row, row + 1, (GtkAttachOptions)(GTK_FILL | GTK_EXPAND), (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
|
|
|
return combo;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GtkWidget *table_make_combobox(GtkWidget *table, int row, int label_id, const char *default_value, const combo_desc *options)
|
|
|
|
{
|
|
|
|
GList *glist = NULL;
|
|
|
|
while (options->label_id) {
|
|
|
|
glist = g_list_append(glist, (void *)GetString(options->label_id));
|
|
|
|
options++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return table_make_combobox(table, row, label_id, default_value, glist);
|
|
|
|
}
|
|
|
|
|
|
|
|
static GtkWidget *table_make_file_entry(GtkWidget *table, int row, int label_id, const char *prefs_item, bool only_dirs = false)
|
|
|
|
{
|
|
|
|
GtkWidget *box, *label, *entry, *button;
|
|
|
|
|
|
|
|
label = gtk_label_new(GetString(label_id));
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), label, 0, 1, row, row + 1, (GtkAttachOptions)0, (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
|
|
|
const char *str = PrefsFindString(prefs_item);
|
|
|
|
if (str == NULL)
|
|
|
|
str = "";
|
|
|
|
|
|
|
|
box = gtk_hbox_new(FALSE, 4);
|
|
|
|
gtk_widget_show(box);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), box, 1, 2, row, row + 1, (GtkAttachOptions)(GTK_FILL | GTK_EXPAND), (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
|
|
|
entry = gtk_entry_new();
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(entry), str);
|
|
|
|
gtk_widget_show(entry);
|
|
|
|
gtk_box_pack_start(GTK_BOX(box), entry, TRUE, TRUE, 0);
|
|
|
|
|
|
|
|
button = make_browse_button(entry);
|
|
|
|
gtk_box_pack_start(GTK_BOX(box), button, FALSE, FALSE, 0);
|
|
|
|
g_object_set_data(G_OBJECT(entry), "chooser_button", button);
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
static GtkWidget *make_option_menu(GtkWidget *top, int label_id, const opt_desc *options, int active)
|
|
|
|
{
|
|
|
|
GtkWidget *box, *label, *opt, *menu;
|
|
|
|
|
|
|
|
box = gtk_hbox_new(FALSE, 4);
|
|
|
|
gtk_widget_show(box);
|
|
|
|
gtk_box_pack_start(GTK_BOX(top), box, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
label = gtk_label_new(GetString(label_id));
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_box_pack_start(GTK_BOX(box), label, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
opt = gtk_option_menu_new();
|
|
|
|
gtk_widget_show(opt);
|
|
|
|
menu = gtk_menu_new();
|
|
|
|
|
|
|
|
while (options->label_id) {
|
|
|
|
add_menu_item(menu, options->label_id, options->func);
|
|
|
|
options++;
|
|
|
|
}
|
|
|
|
gtk_menu_set_active(GTK_MENU(menu), active);
|
|
|
|
|
|
|
|
gtk_option_menu_set_menu(GTK_OPTION_MENU(opt), menu);
|
|
|
|
gtk_box_pack_start(GTK_BOX(box), opt, FALSE, FALSE, 0);
|
|
|
|
return menu;
|
|
|
|
}
|
|
|
|
|
2002-01-22 17:15:10 +00:00
|
|
|
static GtkWidget *make_file_entry(GtkWidget *top, int label_id, const char *prefs_item, bool only_dirs = false)
|
1999-10-03 14:16:26 +00:00
|
|
|
{
|
|
|
|
GtkWidget *box, *label, *entry;
|
|
|
|
|
|
|
|
box = gtk_hbox_new(FALSE, 4);
|
|
|
|
gtk_widget_show(box);
|
|
|
|
gtk_box_pack_start(GTK_BOX(top), box, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
label = gtk_label_new(GetString(label_id));
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_box_pack_start(GTK_BOX(box), label, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
const char *str = PrefsFindString(prefs_item);
|
|
|
|
if (str == NULL)
|
|
|
|
str = "";
|
2002-01-22 17:15:10 +00:00
|
|
|
|
|
|
|
#ifdef HAVE_GNOMEUI
|
|
|
|
entry = gnome_file_entry_new(NULL, GetString(label_id));
|
|
|
|
if (only_dirs)
|
|
|
|
gnome_file_entry_set_directory(GNOME_FILE_ENTRY(entry), true);
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(gnome_file_entry_gtk_entry(GNOME_FILE_ENTRY(entry))), str);
|
|
|
|
#else
|
|
|
|
entry = gtk_entry_new();
|
1999-10-03 14:16:26 +00:00
|
|
|
gtk_entry_set_text(GTK_ENTRY(entry), str);
|
2002-01-22 17:15:10 +00:00
|
|
|
#endif
|
|
|
|
gtk_widget_show(entry);
|
1999-10-03 14:16:26 +00:00
|
|
|
gtk_box_pack_start(GTK_BOX(box), entry, TRUE, TRUE, 0);
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
|
2005-06-19 15:52:09 +00:00
|
|
|
static const gchar *get_file_entry_path(GtkWidget *entry)
|
2002-01-22 17:15:10 +00:00
|
|
|
{
|
|
|
|
#ifdef HAVE_GNOMEUI
|
|
|
|
return gnome_file_entry_get_full_path(GNOME_FILE_ENTRY(entry), false);
|
|
|
|
#else
|
|
|
|
return gtk_entry_get_text(GTK_ENTRY(entry));
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
static GtkWidget *make_checkbox(GtkWidget *top, int label_id, const char *prefs_item, GtkSignalFunc func)
|
|
|
|
{
|
|
|
|
GtkWidget *button = gtk_check_button_new_with_label(GetString(label_id));
|
|
|
|
gtk_widget_show(button);
|
|
|
|
gtk_toggle_button_set_state(GTK_TOGGLE_BUTTON(button), PrefsFindBool(prefs_item));
|
|
|
|
gtk_signal_connect(GTK_OBJECT(button), "toggled", func, button);
|
|
|
|
gtk_box_pack_start(GTK_BOX(top), button, FALSE, FALSE, 0);
|
|
|
|
return button;
|
|
|
|
}
|
|
|
|
|
2002-09-17 16:06:37 +00:00
|
|
|
static GtkWidget *make_combobox(GtkWidget *top, int label_id, const char *prefs_item, const combo_desc *options)
|
|
|
|
{
|
|
|
|
GtkWidget *box, *label, *combo;
|
|
|
|
char str[32];
|
|
|
|
|
|
|
|
box = gtk_hbox_new(FALSE, 4);
|
|
|
|
gtk_widget_show(box);
|
|
|
|
gtk_box_pack_start(GTK_BOX(top), box, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
label = gtk_label_new(GetString(label_id));
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_box_pack_start(GTK_BOX(box), label, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
GList *glist = NULL;
|
|
|
|
while (options->label_id) {
|
|
|
|
glist = g_list_append(glist, (void *)GetString(options->label_id));
|
|
|
|
options++;
|
|
|
|
}
|
|
|
|
|
|
|
|
combo = gtk_combo_new();
|
|
|
|
gtk_widget_show(combo);
|
|
|
|
gtk_combo_set_popdown_strings(GTK_COMBO(combo), glist);
|
|
|
|
|
|
|
|
sprintf(str, "%d", PrefsFindInt32(prefs_item));
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(GTK_COMBO(combo)->entry), str);
|
|
|
|
gtk_box_pack_start(GTK_BOX(box), combo, TRUE, TRUE, 0);
|
|
|
|
|
|
|
|
return combo;
|
|
|
|
}
|
1999-10-03 14:16:26 +00:00
|
|
|
|
2002-09-17 16:06:37 +00:00
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
/*
|
|
|
|
* Show preferences editor
|
|
|
|
* Returns true when user clicked on "Start", false otherwise
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Window closed
|
|
|
|
static gint window_closed(void)
|
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Window destroyed
|
|
|
|
static void window_destroyed(void)
|
|
|
|
{
|
|
|
|
gtk_main_quit();
|
|
|
|
}
|
|
|
|
|
|
|
|
// "Start" button clicked
|
|
|
|
static void cb_start(...)
|
|
|
|
{
|
|
|
|
start_clicked = true;
|
|
|
|
read_settings();
|
|
|
|
SavePrefs();
|
|
|
|
gtk_widget_destroy(win);
|
|
|
|
}
|
|
|
|
|
|
|
|
// "Quit" button clicked
|
|
|
|
static void cb_quit(...)
|
|
|
|
{
|
|
|
|
start_clicked = false;
|
|
|
|
gtk_widget_destroy(win);
|
|
|
|
}
|
|
|
|
|
|
|
|
// "OK" button of "About" dialog clicked
|
|
|
|
static void dl_quit(GtkWidget *dialog)
|
|
|
|
{
|
|
|
|
gtk_widget_destroy(dialog);
|
|
|
|
}
|
|
|
|
|
|
|
|
// "About" selected
|
|
|
|
static void mn_about(...)
|
|
|
|
{
|
2002-01-22 17:15:10 +00:00
|
|
|
GtkWidget *dialog;
|
|
|
|
|
|
|
|
#ifdef HAVE_GNOMEUI
|
|
|
|
|
|
|
|
char version[32];
|
|
|
|
sprintf(version, "Version %d.%d", VERSION_MAJOR, VERSION_MINOR);
|
|
|
|
const char *authors[] = {
|
|
|
|
"Christian Bauer",
|
|
|
|
"Orlando Bassotto",
|
2022-10-03 18:36:46 +00:00
|
|
|
"Gwenolé Beauchesne",
|
2002-01-22 17:15:10 +00:00
|
|
|
"Marc Chabanas",
|
|
|
|
"Marc Hellwig",
|
|
|
|
"Biill Huey",
|
|
|
|
"Brian J. Johnson",
|
2022-10-03 18:36:46 +00:00
|
|
|
"Jürgen Lachmann",
|
2002-01-22 17:15:10 +00:00
|
|
|
"Samuel Lander",
|
|
|
|
"David Lawrence",
|
|
|
|
"Lauri Pesonen",
|
|
|
|
"Bernd Schmidt",
|
|
|
|
"and others",
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
dialog = gnome_about_new(
|
|
|
|
"Basilisk II",
|
|
|
|
version,
|
2008-01-01 09:40:36 +00:00
|
|
|
"Copyright (C) 1997-2008 Christian Bauer",
|
2002-01-22 17:15:10 +00:00
|
|
|
authors,
|
|
|
|
"Basilisk II comes with ABSOLUTELY NO WARRANTY."
|
|
|
|
"This is free software, and you are welcome to redistribute it"
|
|
|
|
"under the terms of the GNU General Public License.",
|
|
|
|
NULL
|
|
|
|
);
|
|
|
|
gnome_dialog_set_parent(GNOME_DIALOG(dialog), GTK_WINDOW(win));
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
GtkWidget *label, *button;
|
1999-10-03 14:16:26 +00:00
|
|
|
|
2001-06-30 20:18:36 +00:00
|
|
|
char str[512];
|
|
|
|
sprintf(str,
|
|
|
|
"Basilisk II\nVersion %d.%d\n\n"
|
2008-01-01 09:40:36 +00:00
|
|
|
"Copyright (C) 1997-2008 Christian Bauer et al.\n"
|
2001-06-30 20:18:36 +00:00
|
|
|
"E-mail: Christian.Bauer@uni-mainz.de\n"
|
|
|
|
"http://www.uni-mainz.de/~bauec002/B2Main.html\n\n"
|
|
|
|
"Basilisk II comes with ABSOLUTELY NO\n"
|
|
|
|
"WARRANTY. This is free software, and\n"
|
|
|
|
"you are welcome to redistribute it\n"
|
|
|
|
"under the terms of the GNU General\n"
|
|
|
|
"Public License.\n",
|
|
|
|
VERSION_MAJOR, VERSION_MINOR
|
|
|
|
);
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
dialog = gtk_dialog_new();
|
|
|
|
gtk_window_set_title(GTK_WINDOW(dialog), GetString(STR_ABOUT_TITLE));
|
|
|
|
gtk_container_border_width(GTK_CONTAINER(dialog), 5);
|
|
|
|
gtk_widget_set_uposition(GTK_WIDGET(dialog), 100, 150);
|
|
|
|
|
|
|
|
label = gtk_label_new(str);
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), label, TRUE, TRUE, 0);
|
|
|
|
|
|
|
|
button = gtk_button_new_with_label(GetString(STR_OK_BUTTON));
|
|
|
|
gtk_widget_show(button);
|
|
|
|
gtk_signal_connect_object(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(dl_quit), GTK_OBJECT(dialog));
|
|
|
|
gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->action_area), button, FALSE, FALSE, 0);
|
|
|
|
GTK_WIDGET_SET_FLAGS(button, GTK_CAN_DEFAULT);
|
|
|
|
gtk_widget_grab_default(button);
|
2002-01-22 17:15:10 +00:00
|
|
|
|
|
|
|
#endif
|
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
gtk_widget_show(dialog);
|
|
|
|
}
|
|
|
|
|
|
|
|
// "Zap PRAM" selected
|
|
|
|
static void mn_zap_pram(...)
|
|
|
|
{
|
|
|
|
ZapPRAM();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Menu item descriptions
|
|
|
|
static GtkItemFactoryEntry menu_items[] = {
|
1999-10-14 11:37:47 +00:00
|
|
|
{(gchar *)GetString(STR_PREFS_MENU_FILE_GTK), NULL, NULL, 0, "<Branch>"},
|
2005-11-21 21:39:08 +00:00
|
|
|
{(gchar *)GetString(STR_PREFS_ITEM_START_GTK), "<control>S", GTK_SIGNAL_FUNC(cb_start), 0, NULL},
|
1999-10-14 11:37:47 +00:00
|
|
|
{(gchar *)GetString(STR_PREFS_ITEM_ZAP_PRAM_GTK), NULL, GTK_SIGNAL_FUNC(mn_zap_pram), 0, NULL},
|
|
|
|
{(gchar *)GetString(STR_PREFS_ITEM_SEPL_GTK), NULL, NULL, 0, "<Separator>"},
|
|
|
|
{(gchar *)GetString(STR_PREFS_ITEM_QUIT_GTK), "<control>Q", GTK_SIGNAL_FUNC(cb_quit), 0, NULL},
|
|
|
|
{(gchar *)GetString(STR_HELP_MENU_GTK), NULL, NULL, 0, "<LastBranch>"},
|
|
|
|
{(gchar *)GetString(STR_HELP_ITEM_ABOUT_GTK), NULL, GTK_SIGNAL_FUNC(mn_about), 0, NULL}
|
1999-10-03 14:16:26 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
bool PrefsEditor(void)
|
|
|
|
{
|
|
|
|
// Create window
|
|
|
|
win = gtk_window_new(GTK_WINDOW_TOPLEVEL);
|
|
|
|
gtk_window_set_title(GTK_WINDOW(win), GetString(STR_PREFS_TITLE));
|
|
|
|
gtk_signal_connect(GTK_OBJECT(win), "delete_event", GTK_SIGNAL_FUNC(window_closed), NULL);
|
|
|
|
gtk_signal_connect(GTK_OBJECT(win), "destroy", GTK_SIGNAL_FUNC(window_destroyed), NULL);
|
|
|
|
|
|
|
|
// Create window contents
|
|
|
|
GtkWidget *box = gtk_vbox_new(FALSE, 4);
|
|
|
|
gtk_widget_show(box);
|
|
|
|
gtk_container_add(GTK_CONTAINER(win), box);
|
|
|
|
|
|
|
|
GtkAccelGroup *accel_group = gtk_accel_group_new();
|
|
|
|
GtkItemFactory *item_factory = gtk_item_factory_new(GTK_TYPE_MENU_BAR, "<main>", accel_group);
|
|
|
|
gtk_item_factory_create_items(item_factory, sizeof(menu_items) / sizeof(menu_items[0]), menu_items, NULL);
|
2005-06-19 15:52:09 +00:00
|
|
|
#if GTK_CHECK_VERSION(1,3,15)
|
|
|
|
gtk_window_add_accel_group(GTK_WINDOW(win), accel_group);
|
|
|
|
#else
|
1999-10-03 14:16:26 +00:00
|
|
|
gtk_accel_group_attach(accel_group, GTK_OBJECT(win));
|
2005-06-19 15:52:09 +00:00
|
|
|
#endif
|
1999-10-03 14:16:26 +00:00
|
|
|
GtkWidget *menu_bar = gtk_item_factory_get_widget(item_factory, "<main>");
|
|
|
|
gtk_widget_show(menu_bar);
|
|
|
|
gtk_box_pack_start(GTK_BOX(box), menu_bar, FALSE, TRUE, 0);
|
|
|
|
|
|
|
|
GtkWidget *notebook = gtk_notebook_new();
|
|
|
|
gtk_notebook_set_tab_pos(GTK_NOTEBOOK(notebook), GTK_POS_TOP);
|
1999-10-27 16:59:54 +00:00
|
|
|
gtk_notebook_set_scrollable(GTK_NOTEBOOK(notebook), FALSE);
|
1999-10-03 14:16:26 +00:00
|
|
|
gtk_box_pack_start(GTK_BOX(box), notebook, TRUE, TRUE, 0);
|
2006-04-17 14:25:57 +00:00
|
|
|
gtk_widget_realize(notebook);
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
create_volumes_pane(notebook);
|
|
|
|
create_scsi_pane(notebook);
|
|
|
|
create_graphics_pane(notebook);
|
2000-07-13 17:26:17 +00:00
|
|
|
create_input_pane(notebook);
|
1999-10-03 14:16:26 +00:00
|
|
|
create_serial_pane(notebook);
|
|
|
|
create_memory_pane(notebook);
|
2002-09-17 16:06:37 +00:00
|
|
|
create_jit_pane(notebook);
|
2006-04-17 14:25:57 +00:00
|
|
|
gtk_widget_show(notebook);
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
static const opt_desc buttons[] = {
|
|
|
|
{STR_START_BUTTON, GTK_SIGNAL_FUNC(cb_start)},
|
|
|
|
{STR_QUIT_BUTTON, GTK_SIGNAL_FUNC(cb_quit)},
|
|
|
|
{0, NULL}
|
|
|
|
};
|
|
|
|
make_button_box(box, 4, buttons);
|
|
|
|
|
|
|
|
// Show window and enter main loop
|
|
|
|
gtk_widget_show(win);
|
|
|
|
gtk_main();
|
|
|
|
return start_clicked;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* "Volumes" pane
|
|
|
|
*/
|
|
|
|
|
1999-10-19 17:41:44 +00:00
|
|
|
static GtkWidget *volume_list, *w_extfs;
|
1999-10-03 14:16:26 +00:00
|
|
|
static int selected_volume;
|
|
|
|
|
|
|
|
// Volume in list selected
|
|
|
|
static void cl_selected(GtkWidget *list, int row, int column)
|
|
|
|
{
|
|
|
|
selected_volume = row;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Volume selected for addition
|
|
|
|
static void add_volume_ok(GtkWidget *button, file_req_assoc *assoc)
|
|
|
|
{
|
2005-06-19 15:52:09 +00:00
|
|
|
gchar *file = (gchar *)gtk_file_selection_get_filename(GTK_FILE_SELECTION(assoc->req));
|
1999-10-03 14:16:26 +00:00
|
|
|
gtk_clist_append(GTK_CLIST(volume_list), &file);
|
|
|
|
gtk_widget_destroy(assoc->req);
|
|
|
|
delete assoc;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Volume selected for creation
|
|
|
|
static void create_volume_ok(GtkWidget *button, file_req_assoc *assoc)
|
|
|
|
{
|
2005-06-19 15:52:09 +00:00
|
|
|
gchar *file = (gchar *)gtk_file_selection_get_filename(GTK_FILE_SELECTION(assoc->req));
|
|
|
|
const gchar *str = gtk_entry_get_text(GTK_ENTRY(assoc->entry));
|
2022-05-02 14:40:55 +00:00
|
|
|
int disk_size = atoi(str);
|
|
|
|
if (disk_size < 1 || disk_size > 2000) {
|
|
|
|
printf("Disk size needs to be between 1 and 2000 MB.\n");
|
|
|
|
gtk_widget_destroy(GTK_WIDGET(assoc->req));
|
|
|
|
delete assoc;
|
|
|
|
return;
|
|
|
|
}
|
2022-05-03 06:08:36 +00:00
|
|
|
int fd = open(file, O_CREAT | O_WRONLY | O_EXCL, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH);
|
2022-05-02 14:40:55 +00:00
|
|
|
if (fd < 0 && errno == EEXIST) {
|
|
|
|
printf("File already exists, refusing to overwrite file.\n");
|
|
|
|
} else {
|
|
|
|
ftruncate(fd, disk_size * 1024 * 1024);
|
1999-10-03 14:16:26 +00:00
|
|
|
gtk_clist_append(GTK_CLIST(volume_list), &file);
|
2022-05-02 14:40:55 +00:00
|
|
|
}
|
|
|
|
close(fd);
|
1999-10-03 14:16:26 +00:00
|
|
|
gtk_widget_destroy(GTK_WIDGET(assoc->req));
|
|
|
|
delete assoc;
|
|
|
|
}
|
|
|
|
|
|
|
|
// "Add Volume" button clicked
|
|
|
|
static void cb_add_volume(...)
|
|
|
|
{
|
|
|
|
GtkWidget *req = gtk_file_selection_new(GetString(STR_ADD_VOLUME_TITLE));
|
|
|
|
gtk_signal_connect_object(GTK_OBJECT(req), "delete_event", GTK_SIGNAL_FUNC(gtk_widget_destroy), GTK_OBJECT(req));
|
|
|
|
gtk_signal_connect(GTK_OBJECT(GTK_FILE_SELECTION(req)->ok_button), "clicked", GTK_SIGNAL_FUNC(add_volume_ok), new file_req_assoc(req, NULL));
|
|
|
|
gtk_signal_connect_object(GTK_OBJECT(GTK_FILE_SELECTION(req)->cancel_button), "clicked", GTK_SIGNAL_FUNC(gtk_widget_destroy), GTK_OBJECT(req));
|
|
|
|
gtk_widget_show(req);
|
|
|
|
}
|
|
|
|
|
|
|
|
// "Create Hardfile" button clicked
|
|
|
|
static void cb_create_volume(...)
|
|
|
|
{
|
|
|
|
GtkWidget *req = gtk_file_selection_new(GetString(STR_CREATE_VOLUME_TITLE));
|
|
|
|
|
|
|
|
GtkWidget *box = gtk_hbox_new(FALSE, 4);
|
|
|
|
gtk_widget_show(box);
|
|
|
|
GtkWidget *label = gtk_label_new(GetString(STR_HARDFILE_SIZE_CTRL));
|
|
|
|
gtk_widget_show(label);
|
|
|
|
GtkWidget *entry = gtk_entry_new();
|
|
|
|
gtk_widget_show(entry);
|
|
|
|
char str[32];
|
|
|
|
sprintf(str, "%d", 40);
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(entry), str);
|
|
|
|
gtk_box_pack_start(GTK_BOX(box), label, FALSE, FALSE, 0);
|
|
|
|
gtk_box_pack_start(GTK_BOX(box), entry, FALSE, FALSE, 0);
|
|
|
|
gtk_box_pack_start(GTK_BOX(GTK_FILE_SELECTION(req)->main_vbox), box, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
gtk_signal_connect_object(GTK_OBJECT(req), "delete_event", GTK_SIGNAL_FUNC(gtk_widget_destroy), GTK_OBJECT(req));
|
|
|
|
gtk_signal_connect(GTK_OBJECT(GTK_FILE_SELECTION(req)->ok_button), "clicked", GTK_SIGNAL_FUNC(create_volume_ok), new file_req_assoc(req, entry));
|
|
|
|
gtk_signal_connect_object(GTK_OBJECT(GTK_FILE_SELECTION(req)->cancel_button), "clicked", GTK_SIGNAL_FUNC(gtk_widget_destroy), GTK_OBJECT(req));
|
|
|
|
gtk_widget_show(req);
|
|
|
|
}
|
|
|
|
|
|
|
|
// "Remove Volume" button clicked
|
|
|
|
static void cb_remove_volume(...)
|
|
|
|
{
|
|
|
|
gtk_clist_remove(GTK_CLIST(volume_list), selected_volume);
|
|
|
|
}
|
|
|
|
|
|
|
|
// "Boot From" selected
|
2001-01-04 19:50:23 +00:00
|
|
|
static void mn_boot_any(...) {PrefsReplaceInt32("bootdriver", 0);}
|
|
|
|
static void mn_boot_cdrom(...) {PrefsReplaceInt32("bootdriver", CDROMRefNum);}
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
// "No CD-ROM Driver" button toggled
|
|
|
|
static void tb_nocdrom(GtkWidget *widget)
|
|
|
|
{
|
|
|
|
PrefsReplaceBool("nocdrom", GTK_TOGGLE_BUTTON(widget)->active);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read settings from widgets and set preferences
|
|
|
|
static void read_volumes_settings(void)
|
|
|
|
{
|
|
|
|
while (PrefsFindString("disk"))
|
|
|
|
PrefsRemoveItem("disk");
|
|
|
|
|
|
|
|
for (int i=0; i<GTK_CLIST(volume_list)->rows; i++) {
|
|
|
|
char *str;
|
|
|
|
gtk_clist_get_text(GTK_CLIST(volume_list), i, 0, &str);
|
|
|
|
PrefsAddString("disk", str);
|
|
|
|
}
|
1999-10-19 17:41:44 +00:00
|
|
|
|
2002-01-22 17:15:10 +00:00
|
|
|
PrefsReplaceString("extfs", get_file_entry_path(w_extfs));
|
1999-10-03 14:16:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create "Volumes" pane
|
|
|
|
static void create_volumes_pane(GtkWidget *top)
|
|
|
|
{
|
2012-09-02 11:52:28 +00:00
|
|
|
GtkWidget *box, *scroll;
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
box = make_pane(top, STR_VOLUMES_PANE_TITLE);
|
|
|
|
|
|
|
|
scroll = gtk_scrolled_window_new(NULL, NULL);
|
|
|
|
gtk_widget_show(scroll);
|
|
|
|
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scroll), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
|
|
|
|
volume_list = gtk_clist_new(1);
|
|
|
|
gtk_widget_show(volume_list);
|
|
|
|
gtk_clist_set_selection_mode(GTK_CLIST(volume_list), GTK_SELECTION_SINGLE);
|
|
|
|
gtk_clist_set_shadow_type(GTK_CLIST(volume_list), GTK_SHADOW_NONE);
|
1999-10-19 13:22:11 +00:00
|
|
|
gtk_clist_set_reorderable(GTK_CLIST(volume_list), true);
|
1999-10-03 14:16:26 +00:00
|
|
|
gtk_signal_connect(GTK_OBJECT(volume_list), "select_row", GTK_SIGNAL_FUNC(cl_selected), NULL);
|
|
|
|
char *str;
|
|
|
|
int32 index = 0;
|
2002-01-22 17:15:10 +00:00
|
|
|
while ((str = const_cast<char *>(PrefsFindString("disk", index++))) != NULL)
|
1999-10-03 14:16:26 +00:00
|
|
|
gtk_clist_append(GTK_CLIST(volume_list), &str);
|
|
|
|
gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(scroll), volume_list);
|
|
|
|
gtk_box_pack_start(GTK_BOX(box), scroll, TRUE, TRUE, 0);
|
|
|
|
selected_volume = 0;
|
|
|
|
|
|
|
|
static const opt_desc buttons[] = {
|
|
|
|
{STR_ADD_VOLUME_BUTTON, GTK_SIGNAL_FUNC(cb_add_volume)},
|
|
|
|
{STR_CREATE_VOLUME_BUTTON, GTK_SIGNAL_FUNC(cb_create_volume)},
|
|
|
|
{STR_REMOVE_VOLUME_BUTTON, GTK_SIGNAL_FUNC(cb_remove_volume)},
|
|
|
|
{0, NULL},
|
|
|
|
};
|
|
|
|
make_button_box(box, 0, buttons);
|
|
|
|
make_separator(box);
|
|
|
|
|
2002-01-22 17:15:10 +00:00
|
|
|
w_extfs = make_file_entry(box, STR_EXTFS_CTRL, "extfs", true);
|
1999-10-19 17:41:44 +00:00
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
static const opt_desc options[] = {
|
|
|
|
{STR_BOOT_ANY_LAB, GTK_SIGNAL_FUNC(mn_boot_any)},
|
|
|
|
{STR_BOOT_CDROM_LAB, GTK_SIGNAL_FUNC(mn_boot_cdrom)},
|
|
|
|
{0, NULL}
|
|
|
|
};
|
2001-01-04 19:50:23 +00:00
|
|
|
int bootdriver = PrefsFindInt32("bootdriver"), active = 0;
|
1999-10-03 14:16:26 +00:00
|
|
|
switch (bootdriver) {
|
|
|
|
case 0: active = 0; break;
|
|
|
|
case CDROMRefNum: active = 1; break;
|
|
|
|
}
|
2012-09-02 11:52:28 +00:00
|
|
|
make_option_menu(box, STR_BOOTDRIVER_CTRL, options, active);
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
make_checkbox(box, STR_NOCDROM_CTRL, "nocdrom", GTK_SIGNAL_FUNC(tb_nocdrom));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-09-17 16:06:37 +00:00
|
|
|
/*
|
|
|
|
* "JIT Compiler" pane
|
|
|
|
*/
|
|
|
|
|
|
|
|
static GtkWidget *w_jit_fpu;
|
|
|
|
static GtkWidget *w_jit_cache_size;
|
|
|
|
static GtkWidget *w_jit_lazy_flush;
|
2005-06-06 20:11:50 +00:00
|
|
|
static GtkWidget *w_jit_follow_const_jumps;
|
2002-09-17 16:06:37 +00:00
|
|
|
|
2006-04-17 21:22:01 +00:00
|
|
|
// Are we running a JIT capable CPU?
|
|
|
|
static bool is_jit_capable(void)
|
|
|
|
{
|
|
|
|
#if USE_JIT && (defined __i386__ || defined __x86_64__)
|
|
|
|
return true;
|
|
|
|
#elif defined __APPLE__ && defined __MACH__
|
|
|
|
// XXX run-time detect so that we can use a PPC GUI prefs editor
|
|
|
|
static char cpu[10];
|
|
|
|
if (cpu[0] == 0) {
|
|
|
|
FILE *fp = popen("uname -p", "r");
|
|
|
|
if (fp == NULL)
|
|
|
|
return false;
|
|
|
|
fgets(cpu, sizeof(cpu) - 1, fp);
|
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
if (cpu[0] == 'i' && cpu[2] == '8' && cpu[3] == '6') // XXX assuming i?86
|
|
|
|
return true;
|
|
|
|
#endif
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2002-09-17 16:06:37 +00:00
|
|
|
// Set sensitivity of widgets
|
|
|
|
static void set_jit_sensitive(void)
|
|
|
|
{
|
|
|
|
const bool jit_enabled = PrefsFindBool("jit");
|
|
|
|
gtk_widget_set_sensitive(w_jit_fpu, jit_enabled);
|
|
|
|
gtk_widget_set_sensitive(w_jit_cache_size, jit_enabled);
|
|
|
|
gtk_widget_set_sensitive(w_jit_lazy_flush, jit_enabled);
|
2005-06-06 20:11:50 +00:00
|
|
|
gtk_widget_set_sensitive(w_jit_follow_const_jumps, jit_enabled);
|
2002-09-17 16:06:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// "Use JIT Compiler" button toggled
|
|
|
|
static void tb_jit(GtkWidget *widget)
|
|
|
|
{
|
|
|
|
PrefsReplaceBool("jit", GTK_TOGGLE_BUTTON(widget)->active);
|
|
|
|
set_jit_sensitive();
|
|
|
|
}
|
|
|
|
|
|
|
|
// "Compile FPU Instructions" button toggled
|
|
|
|
static void tb_jit_fpu(GtkWidget *widget)
|
|
|
|
{
|
|
|
|
PrefsReplaceBool("jitfpu", GTK_TOGGLE_BUTTON(widget)->active);
|
|
|
|
}
|
|
|
|
|
|
|
|
// "Lazy translation cache invalidation" button toggled
|
|
|
|
static void tb_jit_lazy_flush(GtkWidget *widget)
|
|
|
|
{
|
|
|
|
PrefsReplaceBool("jitlazyflush", GTK_TOGGLE_BUTTON(widget)->active);
|
|
|
|
}
|
|
|
|
|
2005-06-06 20:11:50 +00:00
|
|
|
// "Translate through constant jumps (inline blocks)" button toggled
|
|
|
|
static void tb_jit_follow_const_jumps(GtkWidget *widget)
|
|
|
|
{
|
|
|
|
PrefsReplaceBool("jitinline", GTK_TOGGLE_BUTTON(widget)->active);
|
|
|
|
}
|
|
|
|
|
2002-09-17 16:06:37 +00:00
|
|
|
// Read settings from widgets and set preferences
|
|
|
|
static void read_jit_settings(void)
|
|
|
|
{
|
2006-04-17 21:22:01 +00:00
|
|
|
bool jit_enabled = is_jit_capable() && PrefsFindBool("jit");
|
2002-09-17 16:06:37 +00:00
|
|
|
if (jit_enabled) {
|
|
|
|
const char *str = gtk_entry_get_text(GTK_ENTRY(GTK_COMBO(w_jit_cache_size)->entry));
|
|
|
|
PrefsReplaceInt32("jitcachesize", atoi(str));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create "JIT Compiler" pane
|
|
|
|
static void create_jit_pane(GtkWidget *top)
|
|
|
|
{
|
2006-04-17 21:22:01 +00:00
|
|
|
if (!is_jit_capable())
|
|
|
|
return;
|
|
|
|
|
2012-09-02 11:52:28 +00:00
|
|
|
GtkWidget *box;
|
2002-09-17 16:06:37 +00:00
|
|
|
|
|
|
|
box = make_pane(top, STR_JIT_PANE_TITLE);
|
|
|
|
make_checkbox(box, STR_JIT_CTRL, "jit", GTK_SIGNAL_FUNC(tb_jit));
|
|
|
|
|
|
|
|
w_jit_fpu = make_checkbox(box, STR_JIT_FPU_CTRL, "jitfpu", GTK_SIGNAL_FUNC(tb_jit_fpu));
|
|
|
|
|
|
|
|
// Translation cache size
|
|
|
|
static const combo_desc options[] = {
|
|
|
|
STR_JIT_CACHE_SIZE_2MB_LAB,
|
|
|
|
STR_JIT_CACHE_SIZE_4MB_LAB,
|
|
|
|
STR_JIT_CACHE_SIZE_8MB_LAB,
|
|
|
|
STR_JIT_CACHE_SIZE_16MB_LAB,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
w_jit_cache_size = make_combobox(box, STR_JIT_CACHE_SIZE_CTRL, "jitcachesize", options);
|
|
|
|
|
|
|
|
// Lazy translation cache invalidation
|
|
|
|
w_jit_lazy_flush = make_checkbox(box, STR_JIT_LAZY_CINV_CTRL, "jitlazyflush", GTK_SIGNAL_FUNC(tb_jit_lazy_flush));
|
2005-06-06 20:11:50 +00:00
|
|
|
|
|
|
|
// Follow constant jumps (inline basic blocks)
|
|
|
|
w_jit_follow_const_jumps = make_checkbox(box, STR_JIT_FOLLOW_CONST_JUMPS, "jitinline", GTK_SIGNAL_FUNC(tb_jit_follow_const_jumps));
|
|
|
|
|
2002-09-17 16:06:37 +00:00
|
|
|
set_jit_sensitive();
|
|
|
|
}
|
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
/*
|
|
|
|
* "SCSI" pane
|
|
|
|
*/
|
|
|
|
|
|
|
|
static GtkWidget *w_scsi[7];
|
|
|
|
|
|
|
|
// Read settings from widgets and set preferences
|
|
|
|
static void read_scsi_settings(void)
|
|
|
|
{
|
|
|
|
for (int id=0; id<7; id++) {
|
|
|
|
char prefs_name[32];
|
|
|
|
sprintf(prefs_name, "scsi%d", id);
|
2002-01-22 17:15:10 +00:00
|
|
|
const char *str = get_file_entry_path(w_scsi[id]);
|
1999-10-03 14:16:26 +00:00
|
|
|
if (str && strlen(str))
|
|
|
|
PrefsReplaceString(prefs_name, str);
|
|
|
|
else
|
|
|
|
PrefsRemoveItem(prefs_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create "SCSI" pane
|
|
|
|
static void create_scsi_pane(GtkWidget *top)
|
|
|
|
{
|
|
|
|
GtkWidget *box;
|
|
|
|
|
|
|
|
box = make_pane(top, STR_SCSI_PANE_TITLE);
|
|
|
|
|
|
|
|
for (int id=0; id<7; id++) {
|
|
|
|
char prefs_name[32];
|
|
|
|
sprintf(prefs_name, "scsi%d", id);
|
2002-01-22 17:15:10 +00:00
|
|
|
w_scsi[id] = make_file_entry(box, STR_SCSI_ID_0 + id, prefs_name);
|
1999-10-03 14:16:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* "Graphics/Sound" pane
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Display types
|
|
|
|
enum {
|
|
|
|
DISPLAY_WINDOW,
|
|
|
|
DISPLAY_SCREEN
|
|
|
|
};
|
|
|
|
|
|
|
|
static GtkWidget *w_frameskip, *w_display_x, *w_display_y;
|
|
|
|
static GtkWidget *l_frameskip, *l_display_x, *l_display_y;
|
|
|
|
static int display_type;
|
|
|
|
static int dis_width, dis_height;
|
|
|
|
|
2000-07-13 13:47:12 +00:00
|
|
|
#ifdef ENABLE_FBDEV_DGA
|
1999-10-21 16:07:36 +00:00
|
|
|
static GtkWidget *w_fbdev_name, *w_fbdevice_file;
|
|
|
|
static GtkWidget *l_fbdev_name, *l_fbdevice_file;
|
|
|
|
static char fbdev_name[256];
|
|
|
|
#endif
|
|
|
|
|
2002-10-15 16:25:04 +00:00
|
|
|
static GtkWidget *w_dspdevice_file, *w_mixerdevice_file;
|
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
// Hide/show graphics widgets
|
|
|
|
static void hide_show_graphics_widgets(void)
|
|
|
|
{
|
|
|
|
switch (display_type) {
|
|
|
|
case DISPLAY_WINDOW:
|
|
|
|
gtk_widget_show(w_frameskip); gtk_widget_show(l_frameskip);
|
2000-07-13 13:47:12 +00:00
|
|
|
#ifdef ENABLE_FBDEV_DGA
|
1999-10-21 16:07:36 +00:00
|
|
|
gtk_widget_show(w_display_x); gtk_widget_show(l_display_x);
|
|
|
|
gtk_widget_show(w_display_y); gtk_widget_show(l_display_y);
|
|
|
|
gtk_widget_hide(w_fbdev_name); gtk_widget_hide(l_fbdev_name);
|
|
|
|
#endif
|
1999-10-03 14:16:26 +00:00
|
|
|
break;
|
|
|
|
case DISPLAY_SCREEN:
|
|
|
|
gtk_widget_hide(w_frameskip); gtk_widget_hide(l_frameskip);
|
2000-07-13 13:47:12 +00:00
|
|
|
#ifdef ENABLE_FBDEV_DGA
|
1999-10-21 16:07:36 +00:00
|
|
|
gtk_widget_hide(w_display_x); gtk_widget_hide(l_display_x);
|
|
|
|
gtk_widget_hide(w_display_y); gtk_widget_hide(l_display_y);
|
|
|
|
gtk_widget_show(w_fbdev_name); gtk_widget_show(l_fbdev_name);
|
|
|
|
#endif
|
1999-10-03 14:16:26 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// "Window" video type selected
|
|
|
|
static void mn_window(...)
|
|
|
|
{
|
|
|
|
display_type = DISPLAY_WINDOW;
|
|
|
|
hide_show_graphics_widgets();
|
|
|
|
}
|
|
|
|
|
|
|
|
// "Fullscreen" video type selected
|
|
|
|
static void mn_fullscreen(...)
|
|
|
|
{
|
|
|
|
display_type = DISPLAY_SCREEN;
|
|
|
|
hide_show_graphics_widgets();
|
|
|
|
}
|
|
|
|
|
|
|
|
// "5 Hz".."60Hz" selected
|
|
|
|
static void mn_5hz(...) {PrefsReplaceInt32("frameskip", 12);}
|
|
|
|
static void mn_7hz(...) {PrefsReplaceInt32("frameskip", 8);}
|
|
|
|
static void mn_10hz(...) {PrefsReplaceInt32("frameskip", 6);}
|
|
|
|
static void mn_15hz(...) {PrefsReplaceInt32("frameskip", 4);}
|
|
|
|
static void mn_30hz(...) {PrefsReplaceInt32("frameskip", 2);}
|
|
|
|
static void mn_60hz(...) {PrefsReplaceInt32("frameskip", 1);}
|
2000-07-13 16:12:33 +00:00
|
|
|
static void mn_dynamic(...) {PrefsReplaceInt32("frameskip", 0);}
|
1999-10-03 14:16:26 +00:00
|
|
|
|
2002-10-15 16:25:04 +00:00
|
|
|
// Set sensitivity of widgets
|
|
|
|
static void set_graphics_sensitive(void)
|
|
|
|
{
|
|
|
|
const bool sound_enabled = !PrefsFindBool("nosound");
|
|
|
|
gtk_widget_set_sensitive(w_dspdevice_file, sound_enabled);
|
|
|
|
gtk_widget_set_sensitive(w_mixerdevice_file, sound_enabled);
|
|
|
|
}
|
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
// "Disable Sound Output" button toggled
|
|
|
|
static void tb_nosound(GtkWidget *widget)
|
|
|
|
{
|
|
|
|
PrefsReplaceBool("nosound", GTK_TOGGLE_BUTTON(widget)->active);
|
2002-10-15 16:25:04 +00:00
|
|
|
set_graphics_sensitive();
|
1999-10-03 14:16:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Read graphics preferences
|
|
|
|
static void parse_graphics_prefs(void)
|
|
|
|
{
|
|
|
|
display_type = DISPLAY_WINDOW;
|
|
|
|
dis_width = 512;
|
|
|
|
dis_height = 384;
|
2000-07-13 13:47:12 +00:00
|
|
|
#ifdef ENABLE_FBDEV_DGA
|
1999-10-21 16:07:36 +00:00
|
|
|
fbdev_name[0] = 0;
|
|
|
|
#endif
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
const char *str = PrefsFindString("screen");
|
|
|
|
if (str) {
|
|
|
|
if (sscanf(str, "win/%d/%d", &dis_width, &dis_height) == 2)
|
|
|
|
display_type = DISPLAY_WINDOW;
|
2000-07-13 13:47:12 +00:00
|
|
|
#ifdef ENABLE_FBDEV_DGA
|
1999-10-21 16:07:36 +00:00
|
|
|
else if (sscanf(str, "dga/%255s", fbdev_name) == 1)
|
|
|
|
#else
|
1999-10-03 19:43:28 +00:00
|
|
|
else if (sscanf(str, "dga/%d/%d", &dis_width, &dis_height) == 2)
|
1999-10-21 16:07:36 +00:00
|
|
|
#endif
|
1999-10-03 14:16:26 +00:00
|
|
|
display_type = DISPLAY_SCREEN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read settings from widgets and set preferences
|
|
|
|
static void read_graphics_settings(void)
|
|
|
|
{
|
|
|
|
const char *str;
|
|
|
|
|
|
|
|
str = gtk_entry_get_text(GTK_ENTRY(w_display_x));
|
|
|
|
dis_width = atoi(str);
|
|
|
|
|
|
|
|
str = gtk_entry_get_text(GTK_ENTRY(w_display_y));
|
|
|
|
dis_height = atoi(str);
|
|
|
|
|
|
|
|
char pref[256];
|
|
|
|
switch (display_type) {
|
|
|
|
case DISPLAY_WINDOW:
|
|
|
|
sprintf(pref, "win/%d/%d", dis_width, dis_height);
|
|
|
|
break;
|
|
|
|
case DISPLAY_SCREEN:
|
2000-07-13 13:47:12 +00:00
|
|
|
#ifdef ENABLE_FBDEV_DGA
|
1999-10-21 16:07:36 +00:00
|
|
|
str = gtk_entry_get_text(GTK_ENTRY(w_fbdev_name));
|
|
|
|
sprintf(pref, "dga/%s", str);
|
|
|
|
#else
|
1999-10-03 19:43:28 +00:00
|
|
|
sprintf(pref, "dga/%d/%d", dis_width, dis_height);
|
1999-10-21 16:07:36 +00:00
|
|
|
#endif
|
1999-10-03 14:16:26 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
PrefsRemoveItem("screen");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
PrefsReplaceString("screen", pref);
|
2002-01-22 17:15:10 +00:00
|
|
|
|
|
|
|
#ifdef ENABLE_FBDEV_DGA
|
|
|
|
str = get_file_entry_path(w_fbdevice_file);
|
|
|
|
if (str && strlen(str))
|
|
|
|
PrefsReplaceString("fbdevicefile", str);
|
|
|
|
else
|
|
|
|
PrefsRemoveItem("fbdevicefile");
|
|
|
|
#endif
|
2002-10-15 16:25:04 +00:00
|
|
|
PrefsReplaceString("dsp", get_file_entry_path(w_dspdevice_file));
|
|
|
|
PrefsReplaceString("mixer", get_file_entry_path(w_mixerdevice_file));
|
1999-10-03 14:16:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create "Graphics/Sound" pane
|
|
|
|
static void create_graphics_pane(GtkWidget *top)
|
|
|
|
{
|
1999-10-03 19:43:28 +00:00
|
|
|
GtkWidget *box, *table, *label, *opt, *menu, *combo;
|
1999-10-03 14:16:26 +00:00
|
|
|
char str[32];
|
|
|
|
|
|
|
|
parse_graphics_prefs();
|
|
|
|
|
|
|
|
box = make_pane(top, STR_GRAPHICS_SOUND_PANE_TITLE);
|
1999-10-21 16:07:36 +00:00
|
|
|
table = make_table(box, 2, 5);
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
label = gtk_label_new(GetString(STR_VIDEO_TYPE_CTRL));
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), label, 0, 1, 0, 1, (GtkAttachOptions)0, (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
|
|
|
opt = gtk_option_menu_new();
|
|
|
|
gtk_widget_show(opt);
|
|
|
|
menu = gtk_menu_new();
|
|
|
|
add_menu_item(menu, STR_WINDOW_LAB, GTK_SIGNAL_FUNC(mn_window));
|
|
|
|
add_menu_item(menu, STR_FULLSCREEN_LAB, GTK_SIGNAL_FUNC(mn_fullscreen));
|
|
|
|
switch (display_type) {
|
|
|
|
case DISPLAY_WINDOW:
|
|
|
|
gtk_menu_set_active(GTK_MENU(menu), 0);
|
|
|
|
break;
|
|
|
|
case DISPLAY_SCREEN:
|
|
|
|
gtk_menu_set_active(GTK_MENU(menu), 1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
gtk_option_menu_set_menu(GTK_OPTION_MENU(opt), menu);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), opt, 1, 2, 0, 1, (GtkAttachOptions)GTK_FILL, (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
|
|
|
l_frameskip = gtk_label_new(GetString(STR_FRAMESKIP_CTRL));
|
|
|
|
gtk_widget_show(l_frameskip);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), l_frameskip, 0, 1, 1, 2, (GtkAttachOptions)0, (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
|
|
|
w_frameskip = gtk_option_menu_new();
|
|
|
|
gtk_widget_show(w_frameskip);
|
|
|
|
menu = gtk_menu_new();
|
|
|
|
add_menu_item(menu, STR_REF_5HZ_LAB, GTK_SIGNAL_FUNC(mn_5hz));
|
|
|
|
add_menu_item(menu, STR_REF_7_5HZ_LAB, GTK_SIGNAL_FUNC(mn_7hz));
|
|
|
|
add_menu_item(menu, STR_REF_10HZ_LAB, GTK_SIGNAL_FUNC(mn_10hz));
|
|
|
|
add_menu_item(menu, STR_REF_15HZ_LAB, GTK_SIGNAL_FUNC(mn_15hz));
|
|
|
|
add_menu_item(menu, STR_REF_30HZ_LAB, GTK_SIGNAL_FUNC(mn_30hz));
|
|
|
|
add_menu_item(menu, STR_REF_60HZ_LAB, GTK_SIGNAL_FUNC(mn_60hz));
|
2000-07-13 16:12:33 +00:00
|
|
|
add_menu_item(menu, STR_REF_DYNAMIC_LAB, GTK_SIGNAL_FUNC(mn_dynamic));
|
1999-10-03 14:16:26 +00:00
|
|
|
int frameskip = PrefsFindInt32("frameskip");
|
2000-07-13 16:12:33 +00:00
|
|
|
int item = -1;
|
1999-10-03 14:16:26 +00:00
|
|
|
switch (frameskip) {
|
2000-07-13 16:12:33 +00:00
|
|
|
case 12: item = 0; break;
|
|
|
|
case 8: item = 1; break;
|
|
|
|
case 6: item = 2; break;
|
|
|
|
case 4: item = 3; break;
|
|
|
|
case 2: item = 4; break;
|
|
|
|
case 1: item = 5; break;
|
|
|
|
case 0: item = 6; break;
|
1999-10-03 14:16:26 +00:00
|
|
|
}
|
2000-07-13 16:12:33 +00:00
|
|
|
if (item >= 0)
|
|
|
|
gtk_menu_set_active(GTK_MENU(menu), item);
|
1999-10-03 14:16:26 +00:00
|
|
|
gtk_option_menu_set_menu(GTK_OPTION_MENU(w_frameskip), menu);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), w_frameskip, 1, 2, 1, 2, (GtkAttachOptions)GTK_FILL, (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
|
|
|
l_display_x = gtk_label_new(GetString(STR_DISPLAY_X_CTRL));
|
|
|
|
gtk_widget_show(l_display_x);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), l_display_x, 0, 1, 2, 3, (GtkAttachOptions)0, (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
1999-10-03 19:43:28 +00:00
|
|
|
combo = gtk_combo_new();
|
|
|
|
gtk_widget_show(combo);
|
|
|
|
GList *glist1 = NULL;
|
1999-10-14 11:37:47 +00:00
|
|
|
glist1 = g_list_append(glist1, (void *)GetString(STR_SIZE_512_LAB));
|
|
|
|
glist1 = g_list_append(glist1, (void *)GetString(STR_SIZE_640_LAB));
|
|
|
|
glist1 = g_list_append(glist1, (void *)GetString(STR_SIZE_800_LAB));
|
|
|
|
glist1 = g_list_append(glist1, (void *)GetString(STR_SIZE_1024_LAB));
|
|
|
|
glist1 = g_list_append(glist1, (void *)GetString(STR_SIZE_MAX_LAB));
|
1999-10-03 19:43:28 +00:00
|
|
|
gtk_combo_set_popdown_strings(GTK_COMBO(combo), glist1);
|
|
|
|
if (dis_width)
|
|
|
|
sprintf(str, "%d", dis_width);
|
|
|
|
else
|
1999-10-12 20:00:56 +00:00
|
|
|
strcpy(str, GetString(STR_SIZE_MAX_LAB));
|
1999-10-03 19:43:28 +00:00
|
|
|
gtk_entry_set_text(GTK_ENTRY(GTK_COMBO(combo)->entry), str);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), combo, 1, 2, 2, 3, (GtkAttachOptions)GTK_FILL, (GtkAttachOptions)0, 4, 4);
|
|
|
|
w_display_x = GTK_COMBO(combo)->entry;
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
l_display_y = gtk_label_new(GetString(STR_DISPLAY_Y_CTRL));
|
|
|
|
gtk_widget_show(l_display_y);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), l_display_y, 0, 1, 3, 4, (GtkAttachOptions)0, (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
1999-10-03 19:43:28 +00:00
|
|
|
combo = gtk_combo_new();
|
|
|
|
gtk_widget_show(combo);
|
|
|
|
GList *glist2 = NULL;
|
1999-10-14 11:37:47 +00:00
|
|
|
glist2 = g_list_append(glist2, (void *)GetString(STR_SIZE_384_LAB));
|
|
|
|
glist2 = g_list_append(glist2, (void *)GetString(STR_SIZE_480_LAB));
|
|
|
|
glist2 = g_list_append(glist2, (void *)GetString(STR_SIZE_600_LAB));
|
|
|
|
glist2 = g_list_append(glist2, (void *)GetString(STR_SIZE_768_LAB));
|
|
|
|
glist2 = g_list_append(glist2, (void *)GetString(STR_SIZE_MAX_LAB));
|
1999-10-03 19:43:28 +00:00
|
|
|
gtk_combo_set_popdown_strings(GTK_COMBO(combo), glist2);
|
|
|
|
if (dis_height)
|
|
|
|
sprintf(str, "%d", dis_height);
|
|
|
|
else
|
1999-10-12 20:00:56 +00:00
|
|
|
strcpy(str, GetString(STR_SIZE_MAX_LAB));
|
1999-10-03 19:43:28 +00:00
|
|
|
gtk_entry_set_text(GTK_ENTRY(GTK_COMBO(combo)->entry), str);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), combo, 1, 2, 3, 4, (GtkAttachOptions)GTK_FILL, (GtkAttachOptions)0, 4, 4);
|
|
|
|
w_display_y = GTK_COMBO(combo)->entry;
|
1999-10-03 14:16:26 +00:00
|
|
|
|
2000-07-13 13:47:12 +00:00
|
|
|
#ifdef ENABLE_FBDEV_DGA
|
1999-10-21 16:07:36 +00:00
|
|
|
l_fbdev_name = gtk_label_new(GetString(STR_FBDEV_NAME_CTRL));
|
|
|
|
gtk_widget_show(l_fbdev_name);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), l_fbdev_name, 0, 1, 4, 5, (GtkAttachOptions)0, (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
|
|
|
w_fbdev_name = gtk_entry_new();
|
|
|
|
gtk_widget_show(w_fbdev_name);
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(w_fbdev_name), fbdev_name);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), w_fbdev_name, 1, 2, 4, 5, (GtkAttachOptions)0, (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
2002-01-22 17:15:10 +00:00
|
|
|
w_fbdevice_file = make_file_entry(box, STR_FBDEVICE_FILE_CTRL, "fbdevicefile");
|
1999-10-21 16:07:36 +00:00
|
|
|
#endif
|
|
|
|
|
2000-07-13 17:26:17 +00:00
|
|
|
make_separator(box);
|
1999-10-03 14:16:26 +00:00
|
|
|
make_checkbox(box, STR_NOSOUND_CTRL, "nosound", GTK_SIGNAL_FUNC(tb_nosound));
|
2002-10-15 16:25:04 +00:00
|
|
|
w_dspdevice_file = make_file_entry(box, STR_DSPDEVICE_FILE_CTRL, "dsp");
|
|
|
|
w_mixerdevice_file = make_file_entry(box, STR_MIXERDEVICE_FILE_CTRL, "mixer");
|
|
|
|
|
|
|
|
set_graphics_sensitive();
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
hide_show_graphics_widgets();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-07-13 17:26:17 +00:00
|
|
|
/*
|
|
|
|
* "Input" pane
|
|
|
|
*/
|
|
|
|
|
|
|
|
static GtkWidget *w_keycode_file;
|
|
|
|
static GtkWidget *w_mouse_wheel_lines;
|
|
|
|
|
|
|
|
// Set sensitivity of widgets
|
|
|
|
static void set_input_sensitive(void)
|
|
|
|
{
|
2005-11-21 21:39:08 +00:00
|
|
|
const bool use_keycodes = PrefsFindBool("keycodes");
|
|
|
|
gtk_widget_set_sensitive(w_keycode_file, use_keycodes);
|
|
|
|
gtk_widget_set_sensitive(GTK_WIDGET(g_object_get_data(G_OBJECT(w_keycode_file), "chooser_button")), use_keycodes);
|
2001-01-04 19:50:23 +00:00
|
|
|
gtk_widget_set_sensitive(w_mouse_wheel_lines, PrefsFindInt32("mousewheelmode") == 1);
|
2000-07-13 17:26:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// "Use Raw Keycodes" button toggled
|
|
|
|
static void tb_keycodes(GtkWidget *widget)
|
|
|
|
{
|
|
|
|
PrefsReplaceBool("keycodes", GTK_TOGGLE_BUTTON(widget)->active);
|
|
|
|
set_input_sensitive();
|
|
|
|
}
|
|
|
|
|
|
|
|
// "Mouse Wheel Mode" selected
|
2001-01-04 19:50:23 +00:00
|
|
|
static void mn_wheel_page(...) {PrefsReplaceInt32("mousewheelmode", 0); set_input_sensitive();}
|
|
|
|
static void mn_wheel_cursor(...) {PrefsReplaceInt32("mousewheelmode", 1); set_input_sensitive();}
|
2000-07-13 17:26:17 +00:00
|
|
|
|
|
|
|
// Read settings from widgets and set preferences
|
|
|
|
static void read_input_settings(void)
|
|
|
|
{
|
2002-01-22 17:15:10 +00:00
|
|
|
const char *str = get_file_entry_path(w_keycode_file);
|
2000-07-13 17:26:17 +00:00
|
|
|
if (str && strlen(str))
|
|
|
|
PrefsReplaceString("keycodefile", str);
|
|
|
|
else
|
|
|
|
PrefsRemoveItem("keycodefile");
|
|
|
|
|
2001-01-04 19:50:23 +00:00
|
|
|
PrefsReplaceInt32("mousewheellines", gtk_spin_button_get_value_as_int(GTK_SPIN_BUTTON(w_mouse_wheel_lines)));
|
2000-07-13 17:26:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create "Input" pane
|
|
|
|
static void create_input_pane(GtkWidget *top)
|
|
|
|
{
|
2012-09-02 11:52:28 +00:00
|
|
|
GtkWidget *box, *hbox, *label, *button;
|
2000-07-13 17:26:17 +00:00
|
|
|
GtkObject *adj;
|
|
|
|
|
|
|
|
box = make_pane(top, STR_INPUT_PANE_TITLE);
|
|
|
|
|
|
|
|
make_checkbox(box, STR_KEYCODES_CTRL, "keycodes", GTK_SIGNAL_FUNC(tb_keycodes));
|
2005-11-21 21:39:08 +00:00
|
|
|
|
|
|
|
hbox = gtk_hbox_new(FALSE, 4);
|
|
|
|
gtk_widget_show(hbox);
|
|
|
|
gtk_box_pack_start(GTK_BOX(box), hbox, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
label = gtk_label_new(GetString(STR_KEYCODES_CTRL));
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
const char *str = PrefsFindString("keycodefile");
|
|
|
|
if (str == NULL)
|
|
|
|
str = "";
|
|
|
|
|
|
|
|
w_keycode_file = gtk_entry_new();
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(w_keycode_file), str);
|
|
|
|
gtk_widget_show(w_keycode_file);
|
|
|
|
gtk_box_pack_start(GTK_BOX(hbox), w_keycode_file, TRUE, TRUE, 0);
|
|
|
|
|
|
|
|
button = make_browse_button(w_keycode_file);
|
|
|
|
gtk_box_pack_start(GTK_BOX(hbox), button, FALSE, FALSE, 0);
|
|
|
|
g_object_set_data(G_OBJECT(w_keycode_file), "chooser_button", button);
|
2000-07-13 17:26:17 +00:00
|
|
|
|
|
|
|
make_separator(box);
|
|
|
|
|
|
|
|
static const opt_desc options[] = {
|
|
|
|
{STR_MOUSEWHEELMODE_PAGE_LAB, GTK_SIGNAL_FUNC(mn_wheel_page)},
|
|
|
|
{STR_MOUSEWHEELMODE_CURSOR_LAB, GTK_SIGNAL_FUNC(mn_wheel_cursor)},
|
|
|
|
{0, NULL}
|
|
|
|
};
|
2001-01-04 19:50:23 +00:00
|
|
|
int wheelmode = PrefsFindInt32("mousewheelmode"), active = 0;
|
2000-07-13 17:26:17 +00:00
|
|
|
switch (wheelmode) {
|
|
|
|
case 0: active = 0; break;
|
|
|
|
case 1: active = 1; break;
|
|
|
|
}
|
2012-09-02 11:52:28 +00:00
|
|
|
make_option_menu(box, STR_MOUSEWHEELMODE_CTRL, options, active);
|
2000-07-13 17:26:17 +00:00
|
|
|
|
|
|
|
hbox = gtk_hbox_new(FALSE, 4);
|
|
|
|
gtk_widget_show(hbox);
|
|
|
|
gtk_box_pack_start(GTK_BOX(box), hbox, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
label = gtk_label_new(GetString(STR_MOUSEWHEELLINES_CTRL));
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
|
|
|
|
|
2001-01-04 19:50:23 +00:00
|
|
|
adj = gtk_adjustment_new(PrefsFindInt32("mousewheellines"), 1, 1000, 1, 5, 0);
|
2000-07-13 17:26:17 +00:00
|
|
|
w_mouse_wheel_lines = gtk_spin_button_new(GTK_ADJUSTMENT(adj), 0.0, 0);
|
|
|
|
gtk_widget_show(w_mouse_wheel_lines);
|
|
|
|
gtk_box_pack_start(GTK_BOX(hbox), w_mouse_wheel_lines, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
set_input_sensitive();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
/*
|
|
|
|
* "Serial/Network" pane
|
|
|
|
*/
|
|
|
|
|
2001-07-12 19:48:28 +00:00
|
|
|
static GtkWidget *w_seriala, *w_serialb, *w_ether, *w_udp_port;
|
|
|
|
|
|
|
|
// Set sensitivity of widgets
|
|
|
|
static void set_serial_sensitive(void)
|
|
|
|
{
|
|
|
|
#if SUPPORTS_UDP_TUNNEL
|
|
|
|
gtk_widget_set_sensitive(w_ether, !PrefsFindBool("udptunnel"));
|
|
|
|
gtk_widget_set_sensitive(w_udp_port, PrefsFindBool("udptunnel"));
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
// "Tunnel AppleTalk over IP" button toggled
|
|
|
|
static void tb_udptunnel(GtkWidget *widget)
|
|
|
|
{
|
|
|
|
PrefsReplaceBool("udptunnel", GTK_TOGGLE_BUTTON(widget)->active);
|
|
|
|
set_serial_sensitive();
|
|
|
|
}
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
// Read settings from widgets and set preferences
|
|
|
|
static void read_serial_settings(void)
|
|
|
|
{
|
|
|
|
const char *str;
|
|
|
|
|
|
|
|
str = gtk_entry_get_text(GTK_ENTRY(w_seriala));
|
|
|
|
PrefsReplaceString("seriala", str);
|
|
|
|
|
|
|
|
str = gtk_entry_get_text(GTK_ENTRY(w_serialb));
|
|
|
|
PrefsReplaceString("serialb", str);
|
|
|
|
|
|
|
|
str = gtk_entry_get_text(GTK_ENTRY(w_ether));
|
|
|
|
if (str && strlen(str))
|
|
|
|
PrefsReplaceString("ether", str);
|
|
|
|
else
|
|
|
|
PrefsRemoveItem("ether");
|
2001-07-12 19:48:28 +00:00
|
|
|
|
|
|
|
#if SUPPORTS_UDP_TUNNEL
|
|
|
|
PrefsReplaceInt32("udpport", gtk_spin_button_get_value_as_int(GTK_SPIN_BUTTON(w_udp_port)));
|
|
|
|
#endif
|
1999-10-03 14:16:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add names of serial devices
|
|
|
|
static gint gl_str_cmp(gconstpointer a, gconstpointer b)
|
|
|
|
{
|
|
|
|
return strcmp((char *)a, (char *)b);
|
|
|
|
}
|
|
|
|
|
|
|
|
static GList *add_serial_names(void)
|
|
|
|
{
|
|
|
|
GList *glist = NULL;
|
|
|
|
|
|
|
|
// Search /dev for ttyS* and lp*
|
|
|
|
DIR *d = opendir("/dev");
|
|
|
|
if (d) {
|
|
|
|
struct dirent *de;
|
|
|
|
while ((de = readdir(d)) != NULL) {
|
|
|
|
#if defined(__linux__)
|
|
|
|
if (strncmp(de->d_name, "ttyS", 4) == 0 || strncmp(de->d_name, "lp", 2) == 0) {
|
|
|
|
#elif defined(__FreeBSD__)
|
2022-10-03 18:36:46 +00:00
|
|
|
if (strncmp(de->d_name, "cua", 3) == 0 || strncmp(de->d_name, "lpt", 3) == 0) {
|
1999-10-14 11:37:47 +00:00
|
|
|
#elif defined(__NetBSD__)
|
|
|
|
if (strncmp(de->d_name, "tty0", 4) == 0 || strncmp(de->d_name, "lpt", 3) == 0) {
|
1999-10-03 14:16:26 +00:00
|
|
|
#elif defined(sgi)
|
|
|
|
if (strncmp(de->d_name, "ttyf", 4) == 0 || strncmp(de->d_name, "plp", 3) == 0) {
|
|
|
|
#else
|
|
|
|
if (false) {
|
|
|
|
#endif
|
|
|
|
char *str = new char[64];
|
|
|
|
sprintf(str, "/dev/%s", de->d_name);
|
|
|
|
glist = g_list_append(glist, str);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
closedir(d);
|
|
|
|
}
|
|
|
|
if (glist)
|
2022-04-29 11:49:29 +00:00
|
|
|
glist = g_list_sort(glist, gl_str_cmp);
|
1999-10-03 14:16:26 +00:00
|
|
|
else
|
1999-10-14 11:37:47 +00:00
|
|
|
glist = g_list_append(glist, (void *)GetString(STR_NONE_LAB));
|
1999-10-03 14:16:26 +00:00
|
|
|
return glist;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add names of ethernet interfaces
|
|
|
|
static GList *add_ether_names(void)
|
|
|
|
{
|
|
|
|
GList *glist = NULL;
|
|
|
|
|
|
|
|
// Get list of all Ethernet interfaces
|
|
|
|
int s = socket(PF_INET, SOCK_DGRAM, 0);
|
|
|
|
if (s >= 0) {
|
|
|
|
char inbuf[8192];
|
|
|
|
struct ifconf ifc;
|
|
|
|
ifc.ifc_len = sizeof(inbuf);
|
|
|
|
ifc.ifc_buf = inbuf;
|
|
|
|
if (ioctl(s, SIOCGIFCONF, &ifc) == 0) {
|
|
|
|
struct ifreq req, *ifr = ifc.ifc_req;
|
|
|
|
for (int i=0; i<ifc.ifc_len; i+=sizeof(ifreq), ifr++) {
|
|
|
|
req = *ifr;
|
1999-10-14 11:37:47 +00:00
|
|
|
#if defined(__FreeBSD__) || defined(__NetBSD__) || defined(sgi)
|
1999-10-03 14:16:26 +00:00
|
|
|
if (ioctl(s, SIOCGIFADDR, &req) == 0 && (req.ifr_addr.sa_family == ARPHRD_ETHER || req.ifr_addr.sa_family == ARPHRD_ETHER+1)) {
|
1999-10-14 11:37:47 +00:00
|
|
|
#elif defined(__linux__)
|
1999-10-03 14:16:26 +00:00
|
|
|
if (ioctl(s, SIOCGIFHWADDR, &req) == 0 && req.ifr_hwaddr.sa_family == ARPHRD_ETHER) {
|
1999-10-14 11:37:47 +00:00
|
|
|
#else
|
|
|
|
if (false) {
|
1999-10-03 14:16:26 +00:00
|
|
|
#endif
|
|
|
|
char *str = new char[64];
|
|
|
|
strncpy(str, ifr->ifr_name, 63);
|
|
|
|
glist = g_list_append(glist, str);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
close(s);
|
|
|
|
}
|
2005-06-06 20:11:50 +00:00
|
|
|
#ifdef HAVE_SLIRP
|
|
|
|
static char s_slirp[] = "slirp";
|
|
|
|
glist = g_list_append(glist, s_slirp);
|
|
|
|
#endif
|
1999-10-03 14:16:26 +00:00
|
|
|
if (glist)
|
2022-04-29 11:49:29 +00:00
|
|
|
glist = g_list_sort(glist, gl_str_cmp);
|
1999-10-03 14:16:26 +00:00
|
|
|
else
|
1999-10-14 11:37:47 +00:00
|
|
|
glist = g_list_append(glist, (void *)GetString(STR_NONE_LAB));
|
1999-10-03 14:16:26 +00:00
|
|
|
return glist;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create "Serial/Network" pane
|
|
|
|
static void create_serial_pane(GtkWidget *top)
|
|
|
|
{
|
2001-07-12 19:48:28 +00:00
|
|
|
GtkWidget *box, *hbox, *table, *label, *combo, *sep;
|
|
|
|
GtkObject *adj;
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
box = make_pane(top, STR_SERIAL_NETWORK_PANE_TITLE);
|
2000-07-13 17:26:17 +00:00
|
|
|
table = make_table(box, 2, 4);
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
label = gtk_label_new(GetString(STR_SERIALA_CTRL));
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), label, 0, 1, 0, 1, (GtkAttachOptions)0, (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
2001-07-12 19:48:28 +00:00
|
|
|
GList *glist = add_serial_names();
|
1999-10-03 14:16:26 +00:00
|
|
|
combo = gtk_combo_new();
|
|
|
|
gtk_widget_show(combo);
|
|
|
|
gtk_combo_set_popdown_strings(GTK_COMBO(combo), glist);
|
|
|
|
const char *str = PrefsFindString("seriala");
|
|
|
|
if (str == NULL)
|
|
|
|
str = "";
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(GTK_COMBO(combo)->entry), str);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), combo, 1, 2, 0, 1, (GtkAttachOptions)(GTK_FILL | GTK_EXPAND), (GtkAttachOptions)0, 4, 4);
|
|
|
|
w_seriala = GTK_COMBO(combo)->entry;
|
|
|
|
|
|
|
|
label = gtk_label_new(GetString(STR_SERIALB_CTRL));
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), label, 0, 1, 1, 2, (GtkAttachOptions)0, (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
|
|
|
combo = gtk_combo_new();
|
|
|
|
gtk_widget_show(combo);
|
|
|
|
gtk_combo_set_popdown_strings(GTK_COMBO(combo), glist);
|
|
|
|
str = PrefsFindString("serialb");
|
|
|
|
if (str == NULL)
|
|
|
|
str = "";
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(GTK_COMBO(combo)->entry), str);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), combo, 1, 2, 1, 2, (GtkAttachOptions)(GTK_FILL | GTK_EXPAND), (GtkAttachOptions)0, 4, 4);
|
|
|
|
w_serialb = GTK_COMBO(combo)->entry;
|
|
|
|
|
2000-07-13 17:26:17 +00:00
|
|
|
sep = gtk_hseparator_new();
|
|
|
|
gtk_widget_show(sep);
|
|
|
|
gtk_table_attach(GTK_TABLE(table), sep, 0, 2, 2, 3, (GtkAttachOptions)0, (GtkAttachOptions)0, 4, 4);
|
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
label = gtk_label_new(GetString(STR_ETHERNET_IF_CTRL));
|
|
|
|
gtk_widget_show(label);
|
2000-07-13 17:26:17 +00:00
|
|
|
gtk_table_attach(GTK_TABLE(table), label, 0, 1, 3, 4, (GtkAttachOptions)0, (GtkAttachOptions)0, 4, 4);
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
glist = add_ether_names();
|
|
|
|
combo = gtk_combo_new();
|
|
|
|
gtk_widget_show(combo);
|
|
|
|
gtk_combo_set_popdown_strings(GTK_COMBO(combo), glist);
|
|
|
|
str = PrefsFindString("ether");
|
|
|
|
if (str == NULL)
|
|
|
|
str = "";
|
|
|
|
gtk_entry_set_text(GTK_ENTRY(GTK_COMBO(combo)->entry), str);
|
2000-07-13 17:26:17 +00:00
|
|
|
gtk_table_attach(GTK_TABLE(table), combo, 1, 2, 3, 4, (GtkAttachOptions)(GTK_FILL | GTK_EXPAND), (GtkAttachOptions)0, 4, 4);
|
1999-10-03 14:16:26 +00:00
|
|
|
w_ether = GTK_COMBO(combo)->entry;
|
2001-07-12 19:48:28 +00:00
|
|
|
|
|
|
|
#if SUPPORTS_UDP_TUNNEL
|
|
|
|
make_checkbox(box, STR_UDPTUNNEL_CTRL, "udptunnel", GTK_SIGNAL_FUNC(tb_udptunnel));
|
|
|
|
|
|
|
|
hbox = gtk_hbox_new(FALSE, 4);
|
|
|
|
gtk_widget_show(hbox);
|
|
|
|
gtk_box_pack_start(GTK_BOX(box), hbox, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
label = gtk_label_new(GetString(STR_UDPPORT_CTRL));
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
adj = gtk_adjustment_new(PrefsFindInt32("udpport"), 1, 65535, 1, 5, 0);
|
|
|
|
w_udp_port = gtk_spin_button_new(GTK_ADJUSTMENT(adj), 0.0, 0);
|
|
|
|
gtk_widget_show(w_udp_port);
|
|
|
|
gtk_box_pack_start(GTK_BOX(hbox), w_udp_port, FALSE, FALSE, 0);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
set_serial_sensitive();
|
1999-10-03 14:16:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* "Memory/Misc" pane
|
|
|
|
*/
|
|
|
|
|
2005-11-21 21:39:08 +00:00
|
|
|
static GtkWidget *w_ramsize;
|
1999-10-03 14:16:26 +00:00
|
|
|
static GtkWidget *w_rom_file;
|
|
|
|
|
2006-04-30 21:16:48 +00:00
|
|
|
// Don't use CPU when idle?
|
|
|
|
static void tb_idlewait(GtkWidget *widget)
|
|
|
|
{
|
|
|
|
PrefsReplaceBool("idlewait", GTK_TOGGLE_BUTTON(widget)->active);
|
|
|
|
}
|
|
|
|
|
2002-05-12 11:10:50 +00:00
|
|
|
// "Ignore SEGV" button toggled
|
|
|
|
#ifdef HAVE_SIGSEGV_SKIP_INSTRUCTION
|
|
|
|
static void tb_ignoresegv(GtkWidget *widget)
|
|
|
|
{
|
|
|
|
PrefsReplaceBool("ignoresegv", GTK_TOGGLE_BUTTON(widget)->active);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
// Model ID selected
|
|
|
|
static void mn_modelid_5(...) {PrefsReplaceInt32("modelid", 5);}
|
|
|
|
static void mn_modelid_14(...) {PrefsReplaceInt32("modelid", 14);}
|
|
|
|
|
1999-10-28 15:33:26 +00:00
|
|
|
// CPU/FPU type
|
|
|
|
static void mn_cpu_68020(...) {PrefsReplaceInt32("cpu", 2); PrefsReplaceBool("fpu", false);}
|
|
|
|
static void mn_cpu_68020_fpu(...) {PrefsReplaceInt32("cpu", 2); PrefsReplaceBool("fpu", true);}
|
|
|
|
static void mn_cpu_68030(...) {PrefsReplaceInt32("cpu", 3); PrefsReplaceBool("fpu", false);}
|
|
|
|
static void mn_cpu_68030_fpu(...) {PrefsReplaceInt32("cpu", 3); PrefsReplaceBool("fpu", true);}
|
|
|
|
static void mn_cpu_68040(...) {PrefsReplaceInt32("cpu", 4); PrefsReplaceBool("fpu", true);}
|
1999-10-27 16:59:54 +00:00
|
|
|
|
1999-10-03 14:16:26 +00:00
|
|
|
// Read settings from widgets and set preferences
|
|
|
|
static void read_memory_settings(void)
|
|
|
|
{
|
2005-11-21 21:39:08 +00:00
|
|
|
const char *str = gtk_entry_get_text(GTK_ENTRY(GTK_COMBO(w_ramsize)->entry));
|
|
|
|
PrefsReplaceInt32("ramsize", atoi(str) << 20);
|
1999-10-03 14:16:26 +00:00
|
|
|
|
2005-11-21 21:39:08 +00:00
|
|
|
str = get_file_entry_path(w_rom_file);
|
1999-10-03 14:16:26 +00:00
|
|
|
if (str && strlen(str))
|
|
|
|
PrefsReplaceString("rom", str);
|
|
|
|
else
|
|
|
|
PrefsRemoveItem("rom");
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create "Memory/Misc" pane
|
|
|
|
static void create_memory_pane(GtkWidget *top)
|
|
|
|
{
|
2012-09-02 11:52:28 +00:00
|
|
|
GtkWidget *box, *table;
|
1999-10-03 14:16:26 +00:00
|
|
|
|
|
|
|
box = make_pane(top, STR_MEMORY_MISC_PANE_TITLE);
|
2005-11-21 21:39:08 +00:00
|
|
|
table = make_table(box, 2, 5);
|
1999-10-03 14:16:26 +00:00
|
|
|
|
2005-11-21 21:39:08 +00:00
|
|
|
static const combo_desc options[] = {
|
|
|
|
STR_RAMSIZE_2MB_LAB,
|
|
|
|
STR_RAMSIZE_4MB_LAB,
|
|
|
|
STR_RAMSIZE_8MB_LAB,
|
|
|
|
STR_RAMSIZE_16MB_LAB,
|
|
|
|
STR_RAMSIZE_32MB_LAB,
|
|
|
|
STR_RAMSIZE_64MB_LAB,
|
|
|
|
STR_RAMSIZE_128MB_LAB,
|
|
|
|
STR_RAMSIZE_256MB_LAB,
|
|
|
|
STR_RAMSIZE_512MB_LAB,
|
|
|
|
STR_RAMSIZE_1024MB_LAB,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
char default_ramsize[10];
|
|
|
|
sprintf(default_ramsize, "%d", PrefsFindInt32("ramsize") >> 20);
|
|
|
|
w_ramsize = table_make_combobox(table, 0, STR_RAMSIZE_CTRL, default_ramsize, options);
|
1999-10-03 14:16:26 +00:00
|
|
|
|
1999-10-28 15:33:26 +00:00
|
|
|
static const opt_desc model_options[] = {
|
1999-10-03 14:16:26 +00:00
|
|
|
{STR_MODELID_5_LAB, GTK_SIGNAL_FUNC(mn_modelid_5)},
|
|
|
|
{STR_MODELID_14_LAB, GTK_SIGNAL_FUNC(mn_modelid_14)},
|
|
|
|
{0, NULL}
|
|
|
|
};
|
|
|
|
int modelid = PrefsFindInt32("modelid"), active = 0;
|
|
|
|
switch (modelid) {
|
|
|
|
case 5: active = 0; break;
|
|
|
|
case 14: active = 1; break;
|
|
|
|
}
|
2005-11-21 21:39:08 +00:00
|
|
|
table_make_option_menu(table, 2, STR_MODELID_CTRL, model_options, active);
|
1999-10-28 15:33:26 +00:00
|
|
|
|
2000-07-13 13:47:12 +00:00
|
|
|
#if EMULATED_68K
|
1999-10-28 15:33:26 +00:00
|
|
|
static const opt_desc cpu_options[] = {
|
|
|
|
{STR_CPU_68020_LAB, GTK_SIGNAL_FUNC(mn_cpu_68020)},
|
|
|
|
{STR_CPU_68020_FPU_LAB, GTK_SIGNAL_FUNC(mn_cpu_68020_fpu)},
|
|
|
|
{STR_CPU_68030_LAB, GTK_SIGNAL_FUNC(mn_cpu_68030)},
|
|
|
|
{STR_CPU_68030_FPU_LAB, GTK_SIGNAL_FUNC(mn_cpu_68030_fpu)},
|
|
|
|
{STR_CPU_68040_LAB, GTK_SIGNAL_FUNC(mn_cpu_68040)},
|
|
|
|
{0, NULL}
|
|
|
|
};
|
|
|
|
int cpu = PrefsFindInt32("cpu");
|
|
|
|
bool fpu = PrefsFindBool("fpu");
|
|
|
|
active = 0;
|
|
|
|
switch (cpu) {
|
|
|
|
case 2: active = fpu ? 1 : 0; break;
|
|
|
|
case 3: active = fpu ? 3 : 2; break;
|
|
|
|
case 4: active = 4;
|
|
|
|
}
|
2005-11-21 21:39:08 +00:00
|
|
|
table_make_option_menu(table, 3, STR_CPU_CTRL, cpu_options, active);
|
2000-07-13 13:47:12 +00:00
|
|
|
#endif
|
1999-10-03 14:16:26 +00:00
|
|
|
|
2005-11-21 21:39:08 +00:00
|
|
|
w_rom_file = table_make_file_entry(table, 4, STR_ROM_FILE_CTRL, "rom");
|
2002-05-12 11:10:50 +00:00
|
|
|
|
2006-04-30 21:16:48 +00:00
|
|
|
make_checkbox(box, STR_IDLEWAIT_CTRL, "idlewait", GTK_SIGNAL_FUNC(tb_idlewait));
|
2002-05-12 11:10:50 +00:00
|
|
|
#ifdef HAVE_SIGSEGV_SKIP_INSTRUCTION
|
|
|
|
make_checkbox(box, STR_IGNORESEGV_CTRL, "ignoresegv", GTK_SIGNAL_FUNC(tb_ignoresegv));
|
|
|
|
#endif
|
1999-10-03 14:16:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Read settings from widgets and set preferences
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void read_settings(void)
|
|
|
|
{
|
|
|
|
read_volumes_settings();
|
|
|
|
read_scsi_settings();
|
|
|
|
read_graphics_settings();
|
2000-07-13 17:26:17 +00:00
|
|
|
read_input_settings();
|
1999-10-03 14:16:26 +00:00
|
|
|
read_serial_settings();
|
|
|
|
read_memory_settings();
|
2002-09-17 16:06:37 +00:00
|
|
|
read_jit_settings();
|
1999-10-03 14:16:26 +00:00
|
|
|
}
|
2006-04-16 16:32:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
#ifdef STANDALONE_GUI
|
|
|
|
#include <errno.h>
|
2006-04-16 21:25:41 +00:00
|
|
|
#include <sys/wait.h>
|
|
|
|
#include "rpc.h"
|
2006-04-16 16:32:45 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Fake unused data and functions
|
|
|
|
*/
|
|
|
|
|
|
|
|
uint8 XPRAM[XPRAM_SIZE];
|
|
|
|
void MountVolume(void *fh) { }
|
|
|
|
void FileDiskLayout(loff_t size, uint8 *data, loff_t &start_byte, loff_t &real_size) { }
|
|
|
|
|
2006-04-17 21:22:01 +00:00
|
|
|
#if defined __APPLE__ && defined __MACH__
|
2006-05-08 15:22:56 +00:00
|
|
|
void DarwinSysInit(void) { }
|
|
|
|
void DarwinSysExit(void) { }
|
2006-04-17 21:22:01 +00:00
|
|
|
void DarwinAddFloppyPrefs(void) { }
|
|
|
|
void DarwinAddSerialPrefs(void) { }
|
|
|
|
bool DarwinCDReadTOC(char *, uint8 *) { }
|
|
|
|
#endif
|
|
|
|
|
2006-04-16 16:32:45 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Display alert
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void dl_destroyed(void)
|
|
|
|
{
|
|
|
|
gtk_main_quit();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void display_alert(int title_id, int prefix_id, int button_id, const char *text)
|
|
|
|
{
|
|
|
|
char str[256];
|
|
|
|
sprintf(str, GetString(prefix_id), text);
|
|
|
|
|
|
|
|
GtkWidget *dialog = gtk_dialog_new();
|
|
|
|
gtk_window_set_title(GTK_WINDOW(dialog), GetString(title_id));
|
|
|
|
gtk_container_border_width(GTK_CONTAINER(dialog), 5);
|
|
|
|
gtk_widget_set_uposition(GTK_WIDGET(dialog), 100, 150);
|
|
|
|
gtk_signal_connect(GTK_OBJECT(dialog), "destroy", GTK_SIGNAL_FUNC(dl_destroyed), NULL);
|
|
|
|
|
|
|
|
GtkWidget *label = gtk_label_new(str);
|
|
|
|
gtk_widget_show(label);
|
|
|
|
gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->vbox), label, TRUE, TRUE, 0);
|
|
|
|
|
|
|
|
GtkWidget *button = gtk_button_new_with_label(GetString(button_id));
|
|
|
|
gtk_widget_show(button);
|
|
|
|
gtk_signal_connect_object(GTK_OBJECT(button), "clicked", GTK_SIGNAL_FUNC(dl_quit), GTK_OBJECT(dialog));
|
|
|
|
gtk_box_pack_start(GTK_BOX(GTK_DIALOG(dialog)->action_area), button, FALSE, FALSE, 0);
|
|
|
|
GTK_WIDGET_SET_FLAGS(button, GTK_CAN_DEFAULT);
|
|
|
|
gtk_widget_grab_default(button);
|
|
|
|
gtk_widget_show(dialog);
|
|
|
|
|
|
|
|
gtk_main();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Display error alert
|
|
|
|
*/
|
|
|
|
|
2006-04-16 21:25:41 +00:00
|
|
|
void ErrorAlert(const char *text)
|
2006-04-16 16:32:45 +00:00
|
|
|
{
|
|
|
|
display_alert(STR_ERROR_ALERT_TITLE, STR_GUI_ERROR_PREFIX, STR_QUIT_BUTTON, text);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-04-16 21:25:41 +00:00
|
|
|
/*
|
|
|
|
* Display warning alert
|
|
|
|
*/
|
|
|
|
|
|
|
|
void WarningAlert(const char *text)
|
|
|
|
{
|
|
|
|
display_alert(STR_WARNING_ALERT_TITLE, STR_GUI_WARNING_PREFIX, STR_OK_BUTTON, text);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* RPC handlers
|
|
|
|
*/
|
|
|
|
|
2006-04-18 21:29:01 +00:00
|
|
|
static GMainLoop *g_gui_loop;
|
|
|
|
|
2006-04-16 21:25:41 +00:00
|
|
|
static int handle_ErrorAlert(rpc_connection_t *connection)
|
|
|
|
{
|
|
|
|
D(bug("handle_ErrorAlert\n"));
|
|
|
|
|
|
|
|
int error;
|
|
|
|
char *str;
|
|
|
|
if ((error = rpc_method_get_args(connection, RPC_TYPE_STRING, &str, RPC_TYPE_INVALID)) < 0)
|
|
|
|
return error;
|
|
|
|
|
|
|
|
ErrorAlert(str);
|
|
|
|
free(str);
|
|
|
|
return RPC_ERROR_NO_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int handle_WarningAlert(rpc_connection_t *connection)
|
|
|
|
{
|
|
|
|
D(bug("handle_WarningAlert\n"));
|
|
|
|
|
|
|
|
int error;
|
|
|
|
char *str;
|
|
|
|
if ((error = rpc_method_get_args(connection, RPC_TYPE_STRING, &str, RPC_TYPE_INVALID)) < 0)
|
|
|
|
return error;
|
|
|
|
|
|
|
|
WarningAlert(str);
|
|
|
|
free(str);
|
|
|
|
return RPC_ERROR_NO_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int handle_Exit(rpc_connection_t *connection)
|
|
|
|
{
|
|
|
|
D(bug("handle_Exit\n"));
|
|
|
|
|
2006-04-18 21:29:01 +00:00
|
|
|
g_main_quit(g_gui_loop);
|
2006-04-16 21:25:41 +00:00
|
|
|
return RPC_ERROR_NO_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-04-18 21:29:01 +00:00
|
|
|
/*
|
|
|
|
* SIGCHLD handler
|
|
|
|
*/
|
|
|
|
|
|
|
|
static char g_app_path[PATH_MAX];
|
|
|
|
static rpc_connection_t *g_gui_connection = NULL;
|
|
|
|
|
|
|
|
static void sigchld_handler(int sig, siginfo_t *sip, void *)
|
|
|
|
{
|
|
|
|
D(bug("Child %d exitted with status = %x\n", sip->si_pid, sip->si_status));
|
|
|
|
|
2006-04-18 22:17:31 +00:00
|
|
|
// XXX perform a new wait because sip->si_status is sometimes not
|
|
|
|
// the exit _value_ on MacOS X but rather the usual status field
|
|
|
|
// from waitpid() -- we could arrange this code in some other way...
|
|
|
|
int status;
|
|
|
|
if (waitpid(sip->si_pid, &status, 0) < 0)
|
|
|
|
status = sip->si_status;
|
|
|
|
if (WIFEXITED(status))
|
|
|
|
status = WEXITSTATUS(status);
|
2006-04-18 21:29:01 +00:00
|
|
|
if (status & 0x80)
|
|
|
|
status |= -1 ^0xff;
|
|
|
|
|
|
|
|
if (status < 0) { // negative -> execlp/-errno
|
|
|
|
char str[256];
|
|
|
|
sprintf(str, GetString(STR_NO_B2_EXE_FOUND), g_app_path, strerror(-status));
|
|
|
|
ErrorAlert(str);
|
|
|
|
status = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status != 0) {
|
|
|
|
if (g_gui_connection)
|
|
|
|
rpc_exit(g_gui_connection);
|
|
|
|
exit(status);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-04-16 16:32:45 +00:00
|
|
|
/*
|
|
|
|
* Start standalone GUI
|
|
|
|
*/
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
#ifdef HAVE_GNOMEUI
|
|
|
|
// Init GNOME/GTK
|
|
|
|
char version[16];
|
|
|
|
sprintf(version, "%d.%d", VERSION_MAJOR, VERSION_MINOR);
|
|
|
|
gnome_init("Basilisk II", version, argc, argv);
|
|
|
|
#else
|
|
|
|
// Init GTK
|
|
|
|
gtk_set_locale();
|
|
|
|
gtk_init(&argc, &argv);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Read preferences
|
2010-01-15 01:53:31 +00:00
|
|
|
PrefsInit(NULL, argc, argv);
|
2006-04-16 16:32:45 +00:00
|
|
|
|
|
|
|
// Show preferences editor
|
|
|
|
bool start = PrefsEditor();
|
|
|
|
|
|
|
|
// Exit preferences
|
|
|
|
PrefsExit();
|
|
|
|
|
|
|
|
// Transfer control to the executable
|
|
|
|
if (start) {
|
2006-04-16 21:25:41 +00:00
|
|
|
char gui_connection_path[64];
|
|
|
|
sprintf(gui_connection_path, "/org/BasiliskII/GUI/%d", getpid());
|
2006-04-17 21:22:01 +00:00
|
|
|
|
2006-04-18 21:29:01 +00:00
|
|
|
// Catch exits from the child process
|
|
|
|
struct sigaction sigchld_sa, old_sigchld_sa;
|
|
|
|
sigemptyset(&sigchld_sa.sa_mask);
|
|
|
|
sigchld_sa.sa_sigaction = sigchld_handler;
|
|
|
|
sigchld_sa.sa_flags = SA_NOCLDSTOP | SA_SIGINFO;
|
|
|
|
if (sigaction(SIGCHLD, &sigchld_sa, &old_sigchld_sa) < 0) {
|
|
|
|
char str[256];
|
|
|
|
sprintf(str, GetString(STR_SIG_INSTALL_ERR), SIGCHLD, strerror(errno));
|
|
|
|
ErrorAlert(str);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2006-04-17 21:22:01 +00:00
|
|
|
// Search and run the BasiliskII executable
|
2006-04-18 22:17:31 +00:00
|
|
|
char *p;
|
2006-04-18 21:29:01 +00:00
|
|
|
strcpy(g_app_path, argv[0]);
|
2006-04-18 22:17:31 +00:00
|
|
|
if ((p = strstr(g_app_path, "BasiliskIIGUI.app/Contents/MacOS")) != NULL) {
|
|
|
|
strcpy(p, "BasiliskII.app/Contents/MacOS/BasiliskII");
|
|
|
|
if (access(g_app_path, X_OK) < 0) {
|
|
|
|
char str[256];
|
|
|
|
sprintf(str, GetString(STR_NO_B2_EXE_FOUND), g_app_path, strerror(errno));
|
|
|
|
WarningAlert(str);
|
|
|
|
strcpy(g_app_path, "/Applications/BasiliskII.app/Contents/MacOS/BasiliskII");
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
p = strrchr(g_app_path, '/');
|
|
|
|
p = p ? p + 1 : g_app_path;
|
|
|
|
strcpy(p, "BasiliskII");
|
|
|
|
}
|
2006-04-17 21:22:01 +00:00
|
|
|
|
2006-04-16 21:25:41 +00:00
|
|
|
int pid = fork();
|
2006-04-17 21:22:01 +00:00
|
|
|
if (pid == 0) {
|
2006-04-18 22:17:31 +00:00
|
|
|
D(bug("Trying to execute %s\n", g_app_path));
|
2006-04-18 21:29:01 +00:00
|
|
|
execlp(g_app_path, g_app_path, "--gui-connection", gui_connection_path, (char *)NULL);
|
|
|
|
#ifdef _POSIX_PRIORITY_SCHEDULING
|
|
|
|
// XXX get a chance to run the parent process so that to not confuse/upset GTK...
|
|
|
|
sched_yield();
|
|
|
|
#endif
|
|
|
|
_exit(-errno);
|
2006-04-17 21:22:01 +00:00
|
|
|
}
|
2006-04-16 21:25:41 +00:00
|
|
|
|
2006-04-17 21:22:01 +00:00
|
|
|
// Establish a connection to Basilisk II
|
2006-04-18 21:29:01 +00:00
|
|
|
if ((g_gui_connection = rpc_init_server(gui_connection_path)) == NULL) {
|
2006-04-17 21:22:01 +00:00
|
|
|
printf("ERROR: failed to initialize GUI-side RPC server connection\n");
|
2006-04-16 21:25:41 +00:00
|
|
|
return 1;
|
|
|
|
}
|
2006-04-17 21:22:01 +00:00
|
|
|
static const rpc_method_descriptor_t vtable[] = {
|
|
|
|
{ RPC_METHOD_ERROR_ALERT, handle_ErrorAlert },
|
|
|
|
{ RPC_METHOD_WARNING_ALERT, handle_WarningAlert },
|
|
|
|
{ RPC_METHOD_EXIT, handle_Exit }
|
|
|
|
};
|
2006-04-18 21:29:01 +00:00
|
|
|
if (rpc_method_add_callbacks(g_gui_connection, vtable, sizeof(vtable) / sizeof(vtable[0])) < 0) {
|
2006-04-17 21:22:01 +00:00
|
|
|
printf("ERROR: failed to setup GUI method callbacks\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
int socket;
|
2006-04-18 21:29:01 +00:00
|
|
|
if ((socket = rpc_listen_socket(g_gui_connection)) < 0) {
|
2006-04-17 21:22:01 +00:00
|
|
|
printf("ERROR: failed to initialize RPC server thread\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2006-04-18 21:29:01 +00:00
|
|
|
g_gui_loop = g_main_new(TRUE);
|
|
|
|
while (g_main_is_running(g_gui_loop)) {
|
2006-04-16 21:25:41 +00:00
|
|
|
|
2006-04-17 21:22:01 +00:00
|
|
|
// Process a few events pending
|
|
|
|
const int N_EVENTS_DISPATCH = 10;
|
|
|
|
for (int i = 0; i < N_EVENTS_DISPATCH; i++) {
|
|
|
|
if (!g_main_iteration(FALSE))
|
|
|
|
break;
|
2006-04-16 21:25:41 +00:00
|
|
|
}
|
|
|
|
|
2006-04-18 21:29:01 +00:00
|
|
|
// Check for RPC events (100 ms timeout)
|
|
|
|
int ret = rpc_wait_dispatch(g_gui_connection, 100000);
|
2006-04-17 21:22:01 +00:00
|
|
|
if (ret == 0)
|
|
|
|
continue;
|
2006-04-18 21:29:01 +00:00
|
|
|
if (ret < 0)
|
|
|
|
break;
|
|
|
|
rpc_dispatch(g_gui_connection);
|
2006-04-16 21:25:41 +00:00
|
|
|
}
|
2006-04-17 21:22:01 +00:00
|
|
|
|
2006-04-18 21:29:01 +00:00
|
|
|
rpc_exit(g_gui_connection);
|
|
|
|
return 0;
|
2006-04-16 16:32:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|