------------------------------------------------------------------------------ ------------------------------------------------------------------------------ -- Cheddar is a GNU GPL real-time scheduling analysis tool. -- This program provides services to automatically check schedulability and -- other performance criteria of real-time architecture models. -- -- Copyright (C) 2002-2023, Frank Singhoff, Alain Plantec, Jerome Legrand, -- Hai Nam Tran, Stephane Rubini -- -- The Cheddar project was started in 2002 by -- Frank Singhoff, Lab-STICC UMR 6285, Université de Bretagne Occidentale -- -- Cheddar has been published in the "Agence de Protection des Programmes/France" in 2008. -- Since 2008, Ellidiss technologies also contributes to the development of -- Cheddar and provides industrial support. -- -- The full list of contributors and sponsors can be found in README.md -- -- 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 -- -- -- Contact : cheddar@listes.univ-brest.fr -- ------------------------------------------------------------------------------ -- Last update : -- $Rev: 3657 $ -- $Date: 2020-12-13 13:25:49 +0100 (dim., 13 déc. 2020) $ -- $Author: singhoff $ ------------------------------------------------------------------------------ ------------------------------------------------------------------------------ with Ada.Exceptions; use Ada.Exceptions; with Ada.Text_IO; use Ada.Text_IO; with Ada.Strings.Unbounded; use Ada.Strings.Unbounded; with GNAT.Current_Exception; use GNAT.Current_Exception; with Gtk.Menu_Item; use Gtk.Menu_Item; with Gtk; use Gtk; with Gtk.Main; use Gtk.Main; with Glib.Error; use Glib.Error; with Gtk.Widget; use Gtk.Widget; with Gtk.Dialog; use Gtk.Dialog; with Gtk.About_Dialog; use Gtk.About_Dialog; with Gtk.Label; use Gtk.Label; with Glib.Object; use Glib.Object; with Gtk.Text_Buffer; use Gtk.Text_Buffer; with Gtk.Combo_Box; use Gtk.Combo_Box; with Gtk.Combo_Box_Text; use Gtk.Combo_Box_Text; with Gtk.Text_Iter; use Gtk.Text_Iter; with Gtk.List_Store; use Gtk.List_Store; with Gtk.Tree_Model; use Gtk.Tree_Model; with Gtk.Tree_View; use Gtk.Tree_View; with Gtk.Tree_View_Column; use Gtk.Tree_View_Column; with Glib; use Glib; with Glib.Object; use Glib.Object; with Glib.Values; use Glib.Values; with unbounded_strings; use unbounded_strings; with processors; use processors; with processor_set; use processor_set; with unbounded_strings; use unbounded_strings; with core_units; use core_units; use core_units.core_units_table_package; with framework_config; use framework_config; with buffer_set; use buffer_set; use buffer_set.generic_buffer_set; with resource_set; use resource_set; use resource_set.generic_resource_set; with translate; use translate; with networks; use networks; with task_set; use task_set; use task_set.generic_task_set; with scheduler; use scheduler; with scheduler.user_defined.interpreted.pipeline; use scheduler.user_defined.interpreted.pipeline; with scheduler.user_defined.interpreted.automata; use scheduler.user_defined.interpreted.automata; with scheduler_interface; use scheduler_interface; with network_set; use network_set; with graphical_editor.convert_text; use graphical_editor.convert_text; with graphical_editor.user_message; use graphical_editor.user_message; package body graphical_editor.processors is package es_entiers is new Ada.Text_IO.Integer_IO (Integer); use es_entiers; processor_table_length : constant Natural := 9; procedure initialize_processor (my_tab : in out table_widget_id; object : access gtkada_builder_record'class) is ob1 : gobject := Get_Object (object, "textbuffer1"); ob2 : gobject := Get_Object (object, "textbuffer3"); ob3 : gobject := Get_Object (object, "combobox1"); ob4 : gobject := Get_Object (object, "combobox2"); ob5 : gobject := Get_Object (object, "liststore4"); ob6 : gobject := Get_Object (object, "treeview2"); ob7 : gobject := Get_Object (object, "liststore3"); ob8 : gobject := Get_Object (object, "treeview1"); wid1 : t_widget := (ptype => buffer_string, buffer_value => gtk_text_buffer (ob1)); wid2 : t_widget := (ptype => buffer_string, buffer_value => gtk_text_buffer (ob2)); wid3 : t_widget := (ptype => combo, combo_value => gtk_combo_box_text (ob3)); wid4 : t_widget := (ptype => combo, combo_value => gtk_combo_box_text (ob4)); wid5 : t_widget := (ptype => lists, list_value => gtk_list_store (ob5)); wid6 : t_widget := (ptype => view, view_value => gtk_tree_view (ob6)); wid7 : t_widget := (ptype => lists, list_value => gtk_list_store (ob7)); wid8 : t_widget := (ptype => view, view_value => gtk_tree_view (ob8)); wid9 : t_widget := (ptype => my_window, window_value => processor_window); begin my_tab (1 .. processor_table_length) := (new t_widget'(wid1), new t_widget'(wid2), new t_widget'(wid3), new t_widget'(wid4), new t_widget'(wid5), new t_widget'(wid6), new t_widget'(wid7), new t_widget'(wid8), new t_widget'(wid9)); end initialize_processor; -- Data retrieved from the widgets -- processor_name : Unbounded_String; processor_type : processors_type; processor_migration : migrations_type; processor_multiprocessors_type : Unbounded_String; processor_core : core_unit_ptr; processor_cores : core_units_table; processor_number_cores : Integer; procedure check_processor (my_tab : in out table_widget_id; object : access gtkada_builder_record'class) is core_name_value : gvalue; migration_string, processor_string : Unbounded_String; a_core : core_unit_ptr; tree_iter_inter : gtk_tree_iter; begin processor_name := get_value_buffer_string (gtk_text_buffer (my_tab (1).buffer_value)); --processor_multiprocessors_type := get_value_buffer_string(Gtk_Text_Buffer(my_tab(2).buffer_value)); processor_type := to_type (To_Unbounded_String (Get_Active_Text (gtk_combo_box_text (my_tab (3).combo_value)))) .processors_value; processor_migration := to_type (To_Unbounded_String (Get_Active_Text (gtk_combo_box_text (my_tab (4).combo_value)))) .migrations_value; processor_number_cores := get_number_elements (gtk_tree_model (Gtk.Tree_View.Get_Model (my_tab (6).view_value))); if processor_number_cores = 0 then check_processor (sys.processors, processor_name, processor_core); else if processor_number_cores = 1 then tree_iter_inter := Get_Iter_First (gtk_tree_model (Gtk.Tree_View.Get_Model (my_tab (6).view_value))); Gtk.Tree_Model.Get_Value (gtk_tree_model (Gtk.Tree_View.Get_Model (my_tab (6).view_value)), tree_iter_inter, 0, core_name_value); processor_core := search_core_unit (sys.core_units, To_Unbounded_String (Get_String (core_name_value))); if to_label (new t_all_enumeration_type' (processors_enum, processor_type)) /= To_Unbounded_String ("Monocore Type") then Raise_Exception (processor_set.invalid_parameter'identity, To_String (lb_create_only_one_core_unit (current_language))); end if; if to_label (new t_all_enumeration_type' (migrations, processor_migration)) /= To_Unbounded_String ("No Migration Type") then Raise_Exception (processor_set.invalid_parameter'identity, To_String (lb_create_only_one_core_unit (current_language))); end if; check_processor (sys.processors, processor_name, processor_core); else tree_iter_inter := Get_Iter_First (gtk_tree_model (Gtk.Tree_View.Get_Model (my_tab (6).view_value))); initialize (processor_cores); for i in 1 .. processor_number_cores loop Gtk.Tree_Model.Get_Value (gtk_tree_model (Gtk.Tree_View.Get_Model (my_tab (6).view_value)), tree_iter_inter, 0, core_name_value); a_core := search_core_unit (sys.core_units, To_Unbounded_String (Get_String (core_name_value))); core_units_table_package.add (processor_cores, a_core); Unset (core_name_value); Next (gtk_tree_model (Gtk.Tree_View.Get_Model (my_tab (6).view_value)), tree_iter_inter); end loop; if to_label (new t_all_enumeration_type' (processors_enum, processor_type)) = To_Unbounded_String ("Monocore Type") then Raise_Exception (processor_set.invalid_parameter'identity, To_String (lb_processor_type_not_allow_with_several_cores (current_language))); end if; check_processor (sys.processors, processor_name, processor_cores, processor_migration, processor_type); end if; end if; end check_processor; procedure add_in_sys_processor (my_tab : in out table_widget_id; object : access gtkada_builder_record'class) is begin if processor_number_cores = 1 then add_processor (sys.processors, processor_name, processor_core); else add_processor (sys.processors, processor_name, processor_cores, processor_migration, processor_type); end if; end add_in_sys_processor; procedure deleted_in_sys_processor (my_tab : in out table_widget_id; object : access gtkada_builder_record'class) is a_processor : generic_processor_ptr; name : Unbounded_String; begin name := get_value_buffer_string (gtk_text_buffer (my_tab (1).buffer_value)); a_processor := search_processor (sys.processors, name); check_entity_referencing_a_processor (sys, a_processor); delete (sys.processors, a_processor); end deleted_in_sys_processor; procedure show_in_sys_processor (my_tab : in out table_widget_id; object : access gtkada_builder_record'class) is a_processor : generic_processor_ptr; name : Unbounded_String; the_cores : core_units_table; a_core : core_unit_ptr; tree_iter : gtk_tree_iter; liststore : gobject; begin name := get_value_buffer_string (gtk_text_buffer (my_tab (1).buffer_value)); a_processor := search_processor (sys.processors, name); the_cores := build_core_table (a_processor); liststore := Get_Object (object, "liststore4"); Gtk.List_Store.Clear (gtk_list_store (liststore)); for i in 0 .. the_cores.nb_entries - 1 loop a_core := the_cores.entries (i); Append (gtk_list_store (liststore), tree_iter); Gtk.List_Store.Set (gtk_list_store (liststore), tree_iter, 0, To_String (a_core.name)); end loop; end show_in_sys_processor; procedure modify_in_sys_processor (my_tab : in out table_widget_id; object : access gtkada_builder_record'class) is a_processor : generic_processor_ptr; a_processor_buffer : generic_processor_ptr; id : Unbounded_String; begin a_processor := search_processor (sys.processors, processor_name); id := a_processor.cheddar_private_id; delete (sys.processors, a_processor); if processor_number_cores = 1 then add_processor (sys.processors, a_processor_buffer, processor_name, processor_core); else add_processor (sys.processors, a_processor_buffer, processor_name, processor_cores, processor_migration, processor_type); end if; a_processor_buffer.cheddar_private_id := id; end modify_in_sys_processor; package processor_callbacks is new generic_callbacks (number_of_widget => processor_table_length, initialize => initialize_processor, check => check_processor, add_in_sys => add_in_sys_processor, deleted_in_sys => deleted_in_sys_processor, modify_in_sys => modify_in_sys_processor, show_in_sys => show_in_sys_processor); use processor_callbacks; procedure add_view_core (object : access gtkada_builder_record'class) is tree_iter : gtk_tree_iter; core_name : Unbounded_String; core_name_value : gvalue; number_rows : Integer; ok : Boolean := False; i : Integer := 0; tree_iter_inter : gtk_tree_iter; liststore : gobject; begin liststore := Get_Object (object, "liststore4"); core_name := To_Unbounded_String (Get_Active_Text (gtk_combo_box_text (Get_Object (object, "combobox3")))); if (core_name = "") then show_message_box (lb_define_core_units_before (current_language)); return; end if; number_rows := get_number_elements (gtk_tree_model (Gtk.Tree_View.Get_Model (gtk_tree_view (Get_Object (object, "treeview2"))))); if number_rows = 0 then Append (gtk_list_store (liststore), tree_iter); Gtk.List_Store.Set (gtk_list_store (liststore), tree_iter, 0, Get_Active_Text (gtk_combo_box_text (Get_Object (object, "combobox3")))); else tree_iter_inter := Get_Iter_First (gtk_tree_model (Gtk.Tree_View.Get_Model (gtk_tree_view (Get_Object (object, "treeview2"))))); while i <= number_rows and ok = False and tree_iter_inter /= Null_Iter loop Gtk.Tree_Model.Get_Value (gtk_tree_model (Gtk.Tree_View.Get_Model (gtk_tree_view (Get_Object (object, "treeview2")))), tree_iter_inter, 0, core_name_value); if core_name = To_Unbounded_String (Get_String (core_name_value)) then ok := True; exit; end if; Glib.Values.Unset (core_name_value); Next (gtk_tree_model (Gtk.Tree_View.Get_Model (gtk_tree_view (Get_Object (object, "treeview2")))), tree_iter_inter); i := i + 1; end loop; if ok = False then Append (gtk_list_store (liststore), tree_iter); Gtk.List_Store.Set (gtk_list_store (liststore), tree_iter, 0, Get_Active_Text (gtk_combo_box_text (Get_Object (object, "combobox3")))); end if; end if; exception when others => show_message_box (Exception_Name & " : " & Exception_Message); return; end add_view_core; procedure delete_view_core (object : access gtkada_builder_record'class) is tree_iter : gtk_tree_iter; path : gtk_tree_path; focus : gtk_tree_view_column; begin Get_Cursor (gtk_tree_view (Get_Object (object, "treeview2")), path, focus); if focus /= null then tree_iter := Get_Iter (gtk_list_store (Get_Object (object, "liststore4")), path); Gtk.List_Store.Remove (gtk_list_store (Get_Object (object, "liststore4")), tree_iter); end if; exception when others => show_message_box (Exception_Name & " : " & Exception_Message); return; end delete_view_core; procedure show_lign_core (object : access gtkada_builder_record'class) is tree_iter : gtk_tree_iter; path1 : gtk_tree_path; focus1 : gtk_tree_view_column; combo_text : Unbounded_String; begin Get_Cursor (gtk_tree_view (Get_Object (object, "treeview2")), path1, focus1); if focus1 /= null then tree_iter := Get_Iter (gtk_list_store (Get_Object (object, "liststore4")), path1); combo_text := To_Unbounded_String (Get_String (gtk_list_store (Get_Object (object, "liststore4")), tree_iter, 0)); Set_Active (gtk_combo_box_text (Get_Object (object, "combobox3")), gint (get_int_combo (gtk_combo_box_text (Get_Object (object, "combobox3")), combo_text))); end if; exception when others => show_message_box (Exception_Name & " : " & Exception_Message); return; end show_lign_core; procedure show_processor_widget is builder : gtkada_builder; error : aliased Glib.Error.gerror; diag : guint; tree_iter : gtk_tree_iter; liststore : gobject; a_processor : generic_processor_ptr; my_iterator : processors_iterator; a_core : core_unit_ptr; my_core_iterator : core_units_iterator; my_combo : gobject; my_network_iterator : networks_iterator; begin Gtk.Main.Init; Gtk_New (builder); diag := Add_From_File (builder, To_String (glade_path) & "processor.glade",error'access); if diag = 0 then Ada.Text_IO.Put_Line ("Error : " & Get_Message (error)); Error_Free (error); return; end if; initialize (sys_buffer); duplicate (sys, sys_buffer); my_combo := Get_Object (builder, "combobox3"); reset_iterator (sys.core_units, my_core_iterator); if not is_empty (sys.core_units) then loop current_element (sys.core_units, a_core, my_core_iterator); Append_Text (gtk_combo_box_text (my_combo), To_String (a_core.name)); exit when is_last_element (sys.core_units, my_core_iterator); next_element (sys.core_units, my_core_iterator); end loop; else show_message_box (lb_define_core_units_before (current_language)); end if; Set_Active (gtk_combo_box_text (Get_Object (builder, "combobox1")), 0); Set_Active (gtk_combo_box_text (Get_Object (builder, "combobox2")), 0); Set_Active (gtk_combo_box_text (Get_Object (builder, "combobox3")), 0); liststore := Get_Object (builder, "liststore3"); reset_iterator (sys.processors, my_iterator); if not is_empty (sys.processors) then loop current_element (sys.processors, a_processor, my_iterator); Append (gtk_list_store (liststore), tree_iter); Gtk.List_Store.Set (gtk_list_store (liststore), tree_iter, 0, To_String (a_processor.name)); Gtk.List_Store.Set (gtk_list_store (liststore), tree_iter, 2, To_String (to_label (new t_all_enumeration_type' (processors_enum, a_processor.processor_type)))); Gtk.List_Store.Set (gtk_list_store (liststore), tree_iter, 3, To_String (to_label (new t_all_enumeration_type' (migrations, a_processor.migration_type)))); exit when is_last_element (sys.processors, my_iterator); next_element (sys.processors, my_iterator); end loop; end if; Gtkada.Builder.Register_Handler (Builder => builder, Handler_Name => "on_button6_clicked", Handler => processor_callbacks.add_view'access); Gtkada.Builder.Register_Handler (Builder => builder, Handler_Name => "on_button4_clicked", Handler => processor_callbacks.delected_lign_selected'access); Gtkada.Builder.Register_Handler (Builder => builder, Handler_Name => "on_button2_clicked", Handler => processor_callbacks.cancel_view'access); Gtkada.Builder.Register_Handler (Builder => builder, Handler_Name => "on_treeview1_cursor_changed", Handler => processor_callbacks.show_lign_selected'access); Gtkada.Builder.Register_Handler (Builder => builder, Handler_Name => "on_treeview2_cursor_changed", Handler => show_lign_core'access); Gtkada.Builder.Register_Handler (Builder => builder, Handler_Name => "on_button5_clicked", Handler => processor_callbacks.modify_lign'access); Gtkada.Builder.Register_Handler (Builder => builder, Handler_Name => "on_button1_clicked", Handler => processor_callbacks.close_view'access); Gtkada.Builder.Register_Handler (Builder => builder, Handler_Name => "on_button8_clicked", Handler => add_view_core'access); Gtkada.Builder.Register_Handler (Builder => builder, Handler_Name => "on_button7_clicked", Handler => delete_view_core'access); Do_Connect (builder); declare Custom_Widget : constant Gtk.Widget.Gtk_Widget := Gtk.Widget.Gtk_Widget (Get_Object (Builder, "window1")); begin Gtk.Widget.Show_All (Custom_Widget); end; Gtk.Main.Main; Unref (builder); end show_processor_widget; end graphical_editor.processors;