2006-01-27 18:57:53 +01:00
|
|
|
// src/startgui.cc - Copyright 2005, 2006, University
|
2006-01-26 19:31:23 +01:00
|
|
|
// of Padova, dept. of Pure and Applied
|
|
|
|
// Mathematics
|
|
|
|
//
|
|
|
|
// This file is part of SGPEMv2.
|
|
|
|
//
|
|
|
|
// This 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.
|
|
|
|
//
|
|
|
|
// SGPEMv2 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 SGPEMv2; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
#include "gettext.h"
|
|
|
|
|
2006-08-31 01:37:11 +02:00
|
|
|
#include "configure_policy_dialog.hh"
|
|
|
|
#include "gui_builder.hh"
|
|
|
|
#include "graphical_preferences_editor.hh"
|
2006-08-12 17:49:37 +02:00
|
|
|
#include "schedulables_tree_widget.hh"
|
|
|
|
#include "simulation_widget.hh"
|
2006-08-21 02:12:56 +02:00
|
|
|
#include "resources_widget.hh"
|
2006-08-11 02:45:41 +02:00
|
|
|
|
2006-09-01 15:17:53 +02:00
|
|
|
#include "sequences.tcc"
|
|
|
|
|
2006-08-30 17:53:26 +02:00
|
|
|
#include "backend/cpu_policy_exception.hh"
|
|
|
|
#include "backend/malformed_policy_exception.hh"
|
|
|
|
#include "backend/null_policy_exception.hh"
|
|
|
|
#include "backend/user_interrupt_exception.hh"
|
2006-08-27 15:39:40 +02:00
|
|
|
#include "backend/cpu_policies_gatekeeper.hh"
|
|
|
|
#include "backend/cpu_policy_manager.hh"
|
2006-08-30 23:09:47 +02:00
|
|
|
#include "backend/resource_policies_gatekeeper.hh"
|
|
|
|
#include "backend/resource_policy_manager.hh"
|
2006-08-02 15:48:26 +02:00
|
|
|
#include "backend/history.hh"
|
2006-08-31 01:37:11 +02:00
|
|
|
#include "backend/policy_parameters.hh"
|
2006-08-02 15:48:26 +02:00
|
|
|
#include "backend/simulation.hh"
|
2006-08-18 09:27:00 +02:00
|
|
|
#include "backend/serializers_gatekeeper.hh"
|
|
|
|
#include "backend/serializer.hh"
|
|
|
|
|
2006-08-21 00:15:02 +02:00
|
|
|
#include <gdkmm/pixbuf.h>
|
2006-08-30 17:53:26 +02:00
|
|
|
#include <glibmm/markup.h>
|
2006-08-28 22:15:05 +02:00
|
|
|
#include <glibmm/timer.h>
|
2006-07-05 17:16:58 +02:00
|
|
|
#include <glibmm/ustring.h>
|
|
|
|
#include <gtkmm/aboutdialog.h>
|
2006-08-18 09:27:00 +02:00
|
|
|
#include <gtkmm/messagedialog.h>
|
2006-08-31 01:37:11 +02:00
|
|
|
#include <gtkmm/menutoolbutton.h>
|
2006-08-18 09:27:00 +02:00
|
|
|
#include <gtkmm/filechooserdialog.h>
|
2006-01-26 19:31:23 +01:00
|
|
|
#include <gtkmm/main.h>
|
2006-09-01 15:17:53 +02:00
|
|
|
#include <gtkmm/radiomenuitem.h>
|
2006-07-05 16:37:11 +02:00
|
|
|
#include <gtkmm/menuitem.h>
|
2006-09-01 15:17:53 +02:00
|
|
|
#include <gtkmm/radiobuttongroup.h>
|
2006-08-12 17:49:37 +02:00
|
|
|
#include <gtkmm/scrolledwindow.h>
|
2006-08-28 23:48:55 +02:00
|
|
|
#include <gtkmm/statusbar.h>
|
2006-08-18 09:27:00 +02:00
|
|
|
#include <gtkmm/stock.h>
|
2006-07-05 16:37:11 +02:00
|
|
|
|
2006-08-28 22:15:05 +02:00
|
|
|
#include <cassert>
|
2006-07-06 11:49:35 +02:00
|
|
|
#include <iostream>
|
2006-01-26 19:31:23 +01:00
|
|
|
|
2006-07-06 14:23:29 +02:00
|
|
|
using namespace sgpem;
|
|
|
|
using Gnome::Glade::Xml;
|
2006-07-05 16:37:11 +02:00
|
|
|
|
2006-08-15 19:33:47 +02:00
|
|
|
void
|
|
|
|
GuiBuilder::on_edit_preferences_activate()
|
|
|
|
{
|
|
|
|
new PreferencesEditor(); //FIXME: are we leaking this way?
|
|
|
|
}
|
|
|
|
|
2006-08-18 09:27:00 +02:00
|
|
|
void
|
|
|
|
GuiBuilder::on_file_open_activate()
|
|
|
|
{
|
|
|
|
Glib::ustring msg;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
std::vector<Serializer*> serializers =
|
|
|
|
SerializersGatekeeper::get_instance().get_registered();
|
|
|
|
|
|
|
|
// FIXME using the first serializer available, this
|
|
|
|
// will need to be changed when multiple serializers will
|
|
|
|
// be made available
|
|
|
|
Serializer& serializer = *serializers.at(0);
|
|
|
|
History& history = Simulation::get_instance().get_history();
|
|
|
|
|
|
|
|
// open file dialog...
|
|
|
|
Gtk::FileChooserDialog dialog("Please choose a file", Gtk::FILE_CHOOSER_ACTION_OPEN);
|
|
|
|
dialog.set_transient_for(get_initial_window());
|
|
|
|
|
|
|
|
//Add response buttons the the dialog:
|
|
|
|
dialog.add_button(Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
|
|
|
|
dialog.add_button(Gtk::Stock::OPEN, Gtk::RESPONSE_OK);
|
|
|
|
|
|
|
|
//Add filters, so that only certain file types can be selected:
|
|
|
|
for(std::vector<Serializer*>::const_iterator iter=serializers.begin(); iter!=serializers.end(); iter++)
|
|
|
|
{
|
|
|
|
Serializer* ser = *iter;
|
|
|
|
Gtk::FileFilter filter_sgpem;
|
|
|
|
filter_sgpem.set_name(ser->get_filename_description());
|
2006-09-02 03:01:37 +02:00
|
|
|
filter_sgpem.add_pattern(Glib::ustring("*.") + ser->get_filename_extension());
|
2006-08-18 09:27:00 +02:00
|
|
|
dialog.add_filter(filter_sgpem);
|
|
|
|
}
|
|
|
|
|
|
|
|
Gtk::FileFilter filter_any;
|
|
|
|
filter_any.set_name("Any files");
|
|
|
|
filter_any.add_pattern("*");
|
|
|
|
dialog.add_filter(filter_any);
|
|
|
|
|
|
|
|
//Show the dialog and wait for a user response:
|
|
|
|
int result = dialog.run();
|
|
|
|
if(result==Gtk::RESPONSE_OK)
|
|
|
|
{
|
2006-08-28 23:48:55 +02:00
|
|
|
Glib::ustring filename = dialog.get_filename();
|
|
|
|
serializer.restore_snapshot(filename, history);
|
|
|
|
msg = "File: " + filename + " loaded.";
|
2006-08-18 09:27:00 +02:00
|
|
|
} // end - if(result==Gtk::RESPONSE_OK)
|
|
|
|
|
|
|
|
}
|
|
|
|
catch (std::out_of_range e)
|
|
|
|
{
|
|
|
|
msg = _("ERROR: No registered serializer available");
|
|
|
|
}
|
|
|
|
catch (SerializerError e)
|
|
|
|
{
|
|
|
|
msg = _("ERROR: ") + Glib::ustring(e.what());
|
|
|
|
}
|
|
|
|
if(!msg.empty())
|
|
|
|
{
|
2006-08-28 23:48:55 +02:00
|
|
|
Gtk::Statusbar* sbar = _refXml->get_widget("MainStatusBar", sbar);
|
|
|
|
sbar->push(msg);
|
2006-08-18 09:27:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GuiBuilder::on_file_save_activate()
|
|
|
|
{
|
2006-08-18 21:54:24 +02:00
|
|
|
// _simulation_widget->change_scaling_mode();
|
2006-08-18 09:27:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GuiBuilder::on_file_saveas_activate()
|
|
|
|
{
|
|
|
|
Glib::ustring msg;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
std::vector<Serializer*> serializers =
|
|
|
|
SerializersGatekeeper::get_instance().get_registered();
|
|
|
|
|
|
|
|
// FIXME using the first serializer available, this
|
|
|
|
// will need to be changed when multiple serializers will
|
|
|
|
// be made available
|
|
|
|
Serializer& serializer = *serializers.at(0);
|
|
|
|
History& history = Simulation::get_instance().get_history();
|
|
|
|
|
|
|
|
// open file dialog...
|
|
|
|
Gtk::FileChooserDialog dialog("Please choose a file", Gtk::FILE_CHOOSER_ACTION_SAVE);
|
|
|
|
dialog.set_transient_for(get_initial_window());
|
|
|
|
|
|
|
|
//Add response buttons the the dialog:
|
|
|
|
dialog.add_button(Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
|
|
|
|
dialog.add_button(Gtk::Stock::SAVE, Gtk::RESPONSE_OK);
|
|
|
|
|
|
|
|
//Add filters, so that only certain file types can be selected:
|
|
|
|
Gtk::FileFilter filter_sgpem;
|
2006-09-02 03:01:37 +02:00
|
|
|
filter_sgpem.set_name(serializer.get_filename_description());
|
|
|
|
filter_sgpem.add_pattern(Glib::ustring("*.") + serializer.get_filename_extension());
|
2006-08-18 09:27:00 +02:00
|
|
|
dialog.add_filter(filter_sgpem);
|
|
|
|
|
|
|
|
Gtk::FileFilter filter_any;
|
|
|
|
filter_any.set_name("Any files");
|
|
|
|
filter_any.add_pattern("*");
|
|
|
|
dialog.add_filter(filter_any);
|
|
|
|
|
|
|
|
//Show the dialog and wait for a user response:
|
|
|
|
int result = dialog.run();
|
|
|
|
if(result==Gtk::RESPONSE_OK)
|
|
|
|
{
|
|
|
|
serializer.save_snapshot(dialog.get_filename(), history);
|
|
|
|
msg = "File: " + dialog.get_filename() + " saved.";
|
|
|
|
} // end - if(result==Gtk::RESPONSE_OK)
|
|
|
|
|
|
|
|
}
|
|
|
|
catch (std::out_of_range e)
|
|
|
|
{
|
|
|
|
msg = _("ERROR: No registered serializer available");
|
|
|
|
}
|
|
|
|
catch (SerializerError e)
|
|
|
|
{
|
|
|
|
msg = _("ERROR: ") + Glib::ustring(e.what());
|
|
|
|
}
|
2006-08-28 23:48:55 +02:00
|
|
|
|
2006-08-18 09:27:00 +02:00
|
|
|
if(!msg.empty())
|
|
|
|
{
|
2006-08-28 23:48:55 +02:00
|
|
|
Gtk::Statusbar* sbar = _refXml->get_widget("MainStatusBar", sbar);
|
|
|
|
sbar->push(msg);
|
2006-08-18 09:27:00 +02:00
|
|
|
}
|
|
|
|
}
|
2006-07-05 16:37:11 +02:00
|
|
|
|
2006-08-31 01:37:11 +02:00
|
|
|
|
|
|
|
void
|
|
|
|
GuiBuilder::on_configure_cpu_policy()
|
|
|
|
{
|
|
|
|
using namespace Gtk;
|
|
|
|
|
|
|
|
CPUPolicy* policy = Simulation::get_instance().get_policy();
|
|
|
|
|
|
|
|
if(policy == NULL)
|
|
|
|
{
|
2006-09-01 14:08:26 +02:00
|
|
|
MessageDialog warn(get_initial_window(),
|
|
|
|
_("<b>No CPU policy is currently selected.</b>\nPlease choose one before trying to configure it."),
|
2006-08-31 01:37:11 +02:00
|
|
|
true, MESSAGE_WARNING, BUTTONS_OK, true);
|
|
|
|
warn.run();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
PolicyParameters& params = policy->get_parameters();
|
2006-09-01 14:08:26 +02:00
|
|
|
ConfigurePolicyDialog config_dialog(_("Configuring CPU Policy ") + policy->get_name(),
|
|
|
|
get_initial_window(), policy->get_description(), params);
|
2006-08-31 01:37:11 +02:00
|
|
|
config_dialog.run();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
GuiBuilder::on_configure_resource_policy()
|
|
|
|
{
|
|
|
|
using namespace Gtk;
|
|
|
|
|
|
|
|
ResourcePolicy* policy = Simulation::get_instance().get_resource_policy();
|
|
|
|
|
|
|
|
if(policy == NULL)
|
|
|
|
{
|
2006-09-01 14:08:26 +02:00
|
|
|
MessageDialog warn(get_initial_window(),
|
|
|
|
_("<b>No CPU policy is currently selected.</b>\nPlease choose one before trying to configure it."),
|
2006-08-31 01:37:11 +02:00
|
|
|
true, MESSAGE_WARNING, BUTTONS_OK, true);
|
|
|
|
warn.run();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
PolicyParameters& params = policy->get_parameters();
|
2006-09-01 14:08:26 +02:00
|
|
|
ConfigurePolicyDialog config_dialog(_("Configuring CPU Policy ") + policy->get_name(),
|
|
|
|
get_initial_window(), policy->get_description(), params);
|
2006-08-31 01:37:11 +02:00
|
|
|
config_dialog.run();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-09-01 15:17:53 +02:00
|
|
|
void
|
|
|
|
GuiBuilder::populate_with_cpu_policies(Gtk::Menu& menu)
|
|
|
|
{
|
|
|
|
using namespace Gtk;
|
|
|
|
|
|
|
|
// NOTE: Please note that this code relies on the fact that a given
|
|
|
|
// policy never "disappears" at runtime. A *GatekeeperObserver should
|
|
|
|
// be needed to avoid dangling pointers if this behaviour changes.
|
|
|
|
|
|
|
|
typedef std::vector<CPUPolicyManager*> Managers;
|
|
|
|
typedef std::vector<CPUPolicy*> Policies;
|
|
|
|
|
|
|
|
RadioButtonGroup group;
|
|
|
|
|
|
|
|
CPUPoliciesGatekeeper& pgk = CPUPoliciesGatekeeper::get_instance();
|
|
|
|
const Managers& managers = pgk.get_registered();
|
|
|
|
|
|
|
|
for(Iseq<Managers::const_iterator> m_it = const_iseq(managers); m_it; ++m_it)
|
|
|
|
{
|
|
|
|
const Policies& policies = (*m_it)->get_avail_policies();
|
|
|
|
for(Iseq<Policies::const_iterator> p_it = const_iseq(policies); p_it; ++p_it)
|
|
|
|
{
|
|
|
|
RadioMenuItem& menuitem = *manage(new RadioMenuItem(group, (*p_it)->get_name()));
|
|
|
|
menuitem.signal_activate().connect(sigc::bind(sigc::mem_fun(*this, &GuiBuilder::on_selected_cpu_policy), *p_it));
|
|
|
|
menu.append(menuitem);
|
|
|
|
menuitem.show();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Activate the first policy available if possible
|
|
|
|
Menu::MenuList& items = menu.items();
|
|
|
|
if(!items.empty())
|
|
|
|
menu.activate_item(items.front());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
GuiBuilder::on_selected_cpu_policy(CPUPolicy* pol)
|
|
|
|
{
|
|
|
|
using namespace Gtk;
|
|
|
|
|
|
|
|
Statusbar* sbar;
|
|
|
|
_refXml->get_widget("MainStatusBar", sbar);
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
Simulation::get_instance().set_policy(pol);
|
|
|
|
if(pol != NULL) {
|
|
|
|
sbar->push(_("Selected CPU policy ") + pol->get_name());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch(CPUPolicyException& e)
|
|
|
|
{
|
|
|
|
Simulation::get_instance().set_policy(NULL);
|
|
|
|
MessageDialog error(get_initial_window(),
|
|
|
|
Glib::ustring(_("<b>Impossible to select this CPU Policy.</b>\n")) + e.what(),
|
|
|
|
true, MESSAGE_ERROR, BUTTONS_OK, true);
|
|
|
|
error.run();
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we got here, no policy is selected.
|
|
|
|
sbar->push(_("No CPU policy selected. Please select one."));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
GuiBuilder::populate_with_resource_policies(Gtk::Menu& menu)
|
|
|
|
{
|
|
|
|
using namespace Gtk;
|
|
|
|
|
|
|
|
// NOTE: Please note that this code relies on the fact that a given
|
|
|
|
// policy never "disappears" at runtime. A *GatekeeperObserver should
|
|
|
|
// be needed to avoid dangling pointers if this behaviour changes.
|
|
|
|
|
|
|
|
typedef std::vector<ResourcePolicyManager*> Managers;
|
|
|
|
typedef std::vector<ResourcePolicy*> Policies;
|
|
|
|
|
|
|
|
RadioButtonGroup group;
|
|
|
|
|
|
|
|
ResourcePoliciesGatekeeper& pgk = ResourcePoliciesGatekeeper::get_instance();
|
|
|
|
const Managers& managers = pgk.get_registered();
|
|
|
|
|
|
|
|
for(Iseq<Managers::const_iterator> m_it = const_iseq(managers); m_it; ++m_it)
|
|
|
|
{
|
|
|
|
const Policies& policies = (*m_it)->get_avail_policies();
|
|
|
|
for(Iseq<Policies::const_iterator> p_it = const_iseq(policies); p_it; ++p_it)
|
|
|
|
{
|
|
|
|
RadioMenuItem& menuitem = *manage(new RadioMenuItem(group, (*p_it)->get_name()));
|
|
|
|
menuitem.signal_activate().connect(sigc::bind(sigc::mem_fun(*this, &GuiBuilder::on_selected_resource_policy), *p_it));
|
|
|
|
menu.append(menuitem);
|
|
|
|
menuitem.show();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Activate the first policy available if possible
|
|
|
|
Menu::MenuList& items = menu.items();
|
|
|
|
if(!items.empty())
|
|
|
|
menu.activate_item(items.front());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
GuiBuilder::on_selected_resource_policy(ResourcePolicy* pol)
|
|
|
|
{
|
|
|
|
using namespace Gtk;
|
|
|
|
Simulation::get_instance().set_resource_policy(pol);
|
|
|
|
Statusbar* sbar;
|
|
|
|
_refXml->get_widget("MainStatusBar", sbar);
|
|
|
|
if(pol == NULL)
|
|
|
|
sbar->push(_("Current resource policy deselected."));
|
|
|
|
else
|
|
|
|
sbar->push(_("Selected resource policy ") + pol->get_name());
|
|
|
|
}
|
|
|
|
|
2006-08-31 01:37:11 +02:00
|
|
|
|
2006-07-06 14:23:29 +02:00
|
|
|
GuiBuilder::GuiBuilder(const std::string& gladefile)
|
2006-08-28 22:15:05 +02:00
|
|
|
: _refXml(Xml::create(gladefile)), _controller(Simulation::get_instance(), _refXml)
|
2006-07-06 14:23:29 +02:00
|
|
|
{
|
2006-08-09 16:38:45 +02:00
|
|
|
using namespace Gtk;
|
|
|
|
|
2006-08-15 19:33:47 +02:00
|
|
|
//Window& main_window = get_initial_window();
|
2006-08-09 16:38:45 +02:00
|
|
|
|
2006-08-18 09:27:00 +02:00
|
|
|
// file open dialog
|
|
|
|
MenuItem* file_open = NULL;
|
|
|
|
_refXml->get_widget("MenuItem.File.Open", file_open);
|
|
|
|
file_open->signal_activate().connect(sigc::mem_fun(*this, &GuiBuilder::on_file_open_activate));
|
|
|
|
|
|
|
|
// file save dialog
|
|
|
|
MenuItem* file_save = NULL;
|
|
|
|
_refXml->get_widget("MenuItem.File.Save", file_save);
|
|
|
|
file_save->signal_activate().connect(sigc::mem_fun(*this, &GuiBuilder::on_file_save_activate));
|
|
|
|
|
|
|
|
// file save dialog
|
|
|
|
MenuItem* file_saveas = NULL;
|
|
|
|
_refXml->get_widget("MenuItem.File.SaveAs", file_saveas);
|
|
|
|
file_saveas->signal_activate().connect(sigc::mem_fun(*this, &GuiBuilder::on_file_saveas_activate));
|
|
|
|
|
2006-08-27 15:39:40 +02:00
|
|
|
|
2006-08-09 16:38:45 +02:00
|
|
|
// Connect extra signals (decide where to do this...
|
|
|
|
// here -- ugly -- derive widgets and then use
|
|
|
|
// Glade::Xml::get_widget_derived -- better --)
|
|
|
|
MenuItem* file_quit = NULL;
|
|
|
|
_refXml->get_widget("MenuItem.File.Quit", file_quit);
|
|
|
|
file_quit->signal_activate().connect(sigc::ptr_fun(&Main::quit));
|
|
|
|
|
2006-08-15 19:33:47 +02:00
|
|
|
// preferences dialog
|
2006-08-21 00:15:02 +02:00
|
|
|
MenuItem* edit_preferences;
|
2006-08-15 19:33:47 +02:00
|
|
|
_refXml->get_widget("MenuItem.Edit.Preferences", edit_preferences);
|
|
|
|
edit_preferences->signal_activate().connect(sigc::mem_fun(*this, &GuiBuilder::on_edit_preferences_activate));
|
|
|
|
|
|
|
|
|
2006-08-31 01:37:11 +02:00
|
|
|
// ---------------- Toolbar buttons ------------------
|
|
|
|
|
|
|
|
// Note: the Play, Pause and Stop buttons are already managed by sgpem::SimulationController.
|
|
|
|
|
|
|
|
// Configure CPU Policy
|
|
|
|
MenuToolButton* cpu_policies_tb_menu;
|
|
|
|
_refXml->get_widget("ToolBar.PolicySelector", cpu_policies_tb_menu);
|
|
|
|
cpu_policies_tb_menu->signal_clicked().connect(sigc::mem_fun(*this, &GuiBuilder::on_configure_cpu_policy));
|
2006-09-01 15:17:53 +02:00
|
|
|
cpu_policies_tb_menu->set_menu(*manage(new Menu()));
|
|
|
|
populate_with_cpu_policies(*cpu_policies_tb_menu->get_menu());
|
|
|
|
|
2006-08-31 01:37:11 +02:00
|
|
|
|
|
|
|
// Configure Resource Policy
|
|
|
|
MenuToolButton* res_policies_tb_menu;
|
|
|
|
_refXml->get_widget("ToolBar.ResourceScheduling", res_policies_tb_menu);
|
|
|
|
res_policies_tb_menu->signal_clicked().connect(sigc::mem_fun(*this, &GuiBuilder::on_configure_resource_policy));
|
2006-09-01 15:17:53 +02:00
|
|
|
res_policies_tb_menu->set_menu(*manage(new Menu()));
|
|
|
|
populate_with_resource_policies(*res_policies_tb_menu->get_menu());
|
2006-08-31 01:37:11 +02:00
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
2006-08-09 16:38:45 +02:00
|
|
|
// About dialog
|
2006-08-21 00:15:02 +02:00
|
|
|
MenuItem* help_about;
|
2006-08-09 16:38:45 +02:00
|
|
|
_refXml->get_widget("MenuItem.Help.About", help_about);
|
|
|
|
AboutDialog* about_dialog = NULL;
|
|
|
|
_refXml->get_widget("AboutDialog", about_dialog);
|
2006-08-15 19:33:47 +02:00
|
|
|
help_about->signal_activate().connect(sigc::mem_fun(*about_dialog, &Window::show));
|
2006-08-21 00:15:02 +02:00
|
|
|
about_dialog->set_wrap_license(true);
|
|
|
|
about_dialog->set_logo(Gdk::Pixbuf::create_from_file(GLADEDIR "/logo.png"));
|
2006-08-15 19:33:47 +02:00
|
|
|
|
|
|
|
|
2006-08-27 15:39:40 +02:00
|
|
|
// Insert the schedulables TreeView custom widget
|
2006-08-20 23:26:14 +02:00
|
|
|
ScrolledWindow* schedulables_sw = NULL;
|
|
|
|
_refXml->get_widget("SchedulablesScrolledWindow", schedulables_sw);
|
|
|
|
SchedulablesTreeWidget* scheds_tree = manage(new SchedulablesTreeWidget());
|
|
|
|
schedulables_sw->add(*scheds_tree);
|
2006-08-09 16:38:45 +02:00
|
|
|
// we have to remember to manually show custom added widgets:
|
2006-08-20 23:26:14 +02:00
|
|
|
scheds_tree->show();
|
2006-08-12 17:49:37 +02:00
|
|
|
|
2006-08-27 15:39:40 +02:00
|
|
|
|
2006-08-31 01:37:11 +02:00
|
|
|
// Resources ListView widget
|
2006-08-21 02:12:56 +02:00
|
|
|
ResourcesWidget* resources_widget = NULL;
|
|
|
|
_refXml->get_widget_derived("Resources.Tree", resources_widget);
|
|
|
|
resources_widget->show();
|
2006-08-15 19:33:47 +02:00
|
|
|
|
2006-08-27 15:39:40 +02:00
|
|
|
|
2006-08-20 23:26:14 +02:00
|
|
|
// Main simulation widget
|
2006-08-12 17:49:37 +02:00
|
|
|
ScrolledWindow* simulation_window = NULL;
|
|
|
|
_refXml->get_widget("SimulationScrolledWindow", simulation_window);
|
2006-09-02 00:51:16 +02:00
|
|
|
_simulation_widget = manage(new SimulationWidget(Simulation::get_instance()));
|
|
|
|
simulation_window->add(*_simulation_widget);
|
|
|
|
_simulation_widget->show();
|
2006-07-06 14:23:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-08-09 16:38:45 +02:00
|
|
|
GuiBuilder::~GuiBuilder()
|
|
|
|
{}
|
2006-07-06 14:23:29 +02:00
|
|
|
|
2006-07-06 11:49:35 +02:00
|
|
|
|
2006-07-06 14:23:29 +02:00
|
|
|
Gtk::Window&
|
|
|
|
GuiBuilder::get_initial_window() const
|
|
|
|
{
|
2006-08-09 16:38:45 +02:00
|
|
|
Gtk::Window* main_window = NULL;
|
|
|
|
_refXml->get_widget("MainWindow", main_window);
|
|
|
|
return *main_window;
|
2006-01-26 19:31:23 +01:00
|
|
|
}
|
2006-02-19 23:36:24 +01:00
|
|
|
|
2006-07-06 14:23:29 +02:00
|
|
|
|
2006-08-09 16:38:45 +02:00
|
|
|
void
|
2006-07-06 14:23:29 +02:00
|
|
|
GuiBuilder::open_file(const std::string& filename)
|
|
|
|
{
|
2006-08-09 16:38:45 +02:00
|
|
|
// FIXME: to be written.
|
|
|
|
// Debug line (erase me when done):
|
|
|
|
std::cout << _("Filename to open: ") << filename << std::endl;
|
2006-07-06 14:23:29 +02:00
|
|
|
}
|
2006-08-20 23:26:14 +02:00
|
|
|
|
2006-08-28 22:15:05 +02:00
|
|
|
|
|
|
|
// ---------------------------------
|
|
|
|
|
2006-08-28 22:23:42 +02:00
|
|
|
SimulationController::SimulationController(Simulation& simulation, Glib::RefPtr<Xml> refXml)
|
|
|
|
: _sim(simulation), _break_requested(false)
|
2006-08-28 22:15:05 +02:00
|
|
|
{
|
|
|
|
using namespace Gtk;
|
|
|
|
|
|
|
|
_sim.attach(*this);
|
|
|
|
|
|
|
|
// Start, pause and stop simulation from the toolbar
|
|
|
|
// TODO: can we use action groups instead of this?
|
|
|
|
|
2006-08-28 22:23:42 +02:00
|
|
|
refXml->get_widget("ToolBar.Play", _toolbt_start);
|
|
|
|
refXml->get_widget("ToolBar.Pause", _toolbt_pause);
|
|
|
|
refXml->get_widget("ToolBar.Stop", _toolbt_stop);
|
|
|
|
|
2006-08-28 22:15:05 +02:00
|
|
|
_toolbt_start->signal_clicked().connect(sigc::mem_fun(*this, &SimulationController::on_simulation_run));
|
|
|
|
_toolbt_pause->signal_clicked().connect(sigc::mem_fun(*this, &SimulationController::on_simulation_pause));
|
|
|
|
_toolbt_stop->signal_clicked().connect(sigc::mem_fun(*this, &SimulationController::on_simulation_stop));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SimulationController::~SimulationController()
|
|
|
|
{
|
|
|
|
_sim.detach(*this);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
SimulationController::on_simulation_run()
|
|
|
|
{
|
|
|
|
// Sensitivities
|
|
|
|
_toolbt_start->set_sensitive(false);
|
|
|
|
_toolbt_pause->set_sensitive(true);
|
|
|
|
_toolbt_stop->set_sensitive(true);
|
|
|
|
|
|
|
|
_break_requested = false;
|
2006-08-30 17:53:26 +02:00
|
|
|
// Used instead of simply calling "_sim.run()" to
|
|
|
|
// have exception handling only in one place:
|
|
|
|
run_simulation_adaptor();
|
2006-08-28 22:15:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
SimulationController::on_simulation_pause()
|
|
|
|
{
|
|
|
|
// Sensitivities
|
|
|
|
_toolbt_start->set_sensitive(true);
|
|
|
|
_toolbt_pause->set_sensitive(false);
|
|
|
|
_toolbt_stop->set_sensitive(true);
|
|
|
|
|
|
|
|
_break_requested = true;
|
|
|
|
_sim.pause();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
SimulationController::on_simulation_stop()
|
|
|
|
{
|
|
|
|
// Sensitivities
|
|
|
|
_toolbt_start->set_sensitive(true);
|
|
|
|
_toolbt_pause->set_sensitive(false);
|
|
|
|
_toolbt_stop->set_sensitive(false);
|
|
|
|
|
|
|
|
_break_requested = true;
|
|
|
|
_sim.stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
SimulationController::update(const Simulation& simulation)
|
|
|
|
{
|
2006-08-28 22:23:42 +02:00
|
|
|
#ifndef NDEBUG
|
2006-08-28 22:15:05 +02:00
|
|
|
std::cerr << "SimulationController::update(), simulation state == " << std::hex << simulation.get_state() << std::endl;
|
2006-08-28 22:23:42 +02:00
|
|
|
#endif
|
2006-08-28 22:15:05 +02:00
|
|
|
|
|
|
|
if(_break_requested)
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch(simulation.get_state())
|
|
|
|
{
|
|
|
|
case Simulation::state_stopped:
|
|
|
|
on_simulation_stop();
|
|
|
|
return;
|
|
|
|
case Simulation::state_paused:
|
|
|
|
on_simulation_pause();
|
|
|
|
return;
|
|
|
|
case Simulation::state_running:
|
|
|
|
// Go on with the rest of the method...
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We should never enter here, if the code is correct:
|
|
|
|
// a step by step simulation should always be in paused
|
|
|
|
// or stopped state after performing a single step
|
|
|
|
assert(simulation.get_mode() != Simulation::mode_step_by_step);
|
|
|
|
|
|
|
|
switch(simulation.get_mode())
|
|
|
|
{
|
|
|
|
case Simulation::mode_continuous:
|
|
|
|
{
|
|
|
|
int timeout = GlobalPreferences::get_instance().get_speed();
|
|
|
|
Glib::signal_timeout().connect(sigc::mem_fun(*this, &SimulationController::run_simulation_adaptor), timeout);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// Never gets here.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
SimulationController::run_simulation_adaptor()
|
|
|
|
{
|
2006-08-30 17:53:26 +02:00
|
|
|
using Gtk::MessageDialog;
|
|
|
|
using namespace Glib;
|
|
|
|
|
2006-08-28 22:15:05 +02:00
|
|
|
if(!_break_requested)
|
2006-08-30 17:53:26 +02:00
|
|
|
try
|
|
|
|
{
|
|
|
|
_sim.run();
|
|
|
|
}
|
|
|
|
catch(const UserInterruptException& uie)
|
|
|
|
{
|
|
|
|
// Show the user a dialog
|
|
|
|
MessageDialog diag(_("<b>The selected user CPU policy stopped before returning:</b>\n") + Markup::escape_text(uie.what()),
|
|
|
|
true, Gtk::MESSAGE_ERROR, Gtk::BUTTONS_OK, true);
|
|
|
|
diag.run();
|
|
|
|
}
|
|
|
|
catch(const MalformedPolicyException& mpe)
|
|
|
|
{
|
|
|
|
// Show user a dialog
|
2006-09-01 19:03:34 +02:00
|
|
|
MessageDialog diag(_("<b>The selected user CPU policy was malformed and failed to sort the queue:</b>\n") +
|
|
|
|
Markup::escape_text(mpe.what()), true, Gtk::MESSAGE_ERROR, Gtk::BUTTONS_OK, true);
|
2006-08-30 17:53:26 +02:00
|
|
|
diag.run();
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
// Deactivate the policy
|
|
|
|
_sim.set_policy(NULL);
|
|
|
|
}
|
|
|
|
catch(const CPUPolicyException& cpe)
|
|
|
|
{
|
|
|
|
// Fatal error. We should never get here.
|
|
|
|
std::cerr << _(" [EE] Fatal error. Impossible to deactivate the policy in ") << __FILE__ << ":" << __LINE__
|
|
|
|
<< std::endl << _(" [EE] ") << cpe.what() << std::endl;
|
|
|
|
;
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch(const NullPolicyException& npe)
|
|
|
|
{
|
|
|
|
MessageDialog diag(_("<b>No active policy selected:</b>\n") + Markup::escape_text(npe.what()),
|
|
|
|
true, Gtk::MESSAGE_ERROR, Gtk::BUTTONS_OK, true);
|
|
|
|
diag.run();
|
|
|
|
|
|
|
|
}
|
|
|
|
catch(const CPUPolicyException& cpe)
|
|
|
|
{
|
2006-09-01 14:08:26 +02:00
|
|
|
MessageDialog diag(_("<b>Unexpected error</b>:\n") + Markup::escape_text(cpe.what()),
|
2006-08-30 17:53:26 +02:00
|
|
|
true, Gtk::MESSAGE_ERROR, Gtk::BUTTONS_OK, true);
|
|
|
|
diag.run();
|
|
|
|
}
|
|
|
|
|
2006-08-28 22:19:58 +02:00
|
|
|
return false;
|
2006-08-28 22:15:05 +02:00
|
|
|
}
|