summaryrefslogtreecommitdiffstats
path: root/sediff/open_policies_dialog.c
diff options
context:
space:
mode:
Diffstat (limited to 'sediff/open_policies_dialog.c')
-rw-r--r--sediff/open_policies_dialog.c527
1 files changed, 527 insertions, 0 deletions
diff --git a/sediff/open_policies_dialog.c b/sediff/open_policies_dialog.c
new file mode 100644
index 0000000..482b95f
--- /dev/null
+++ b/sediff/open_policies_dialog.c
@@ -0,0 +1,527 @@
+/**
+ * @file
+ * Run the dialog to allow the user to open a policy.
+ *
+ * @author Jeremy A. Mowery jmowery@tresys.com
+ * @author Jason Tang jtang@tresys.com
+ *
+ * Copyright (C) 2007 Tresys Technology, LLC
+ *
+ * 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
+ */
+
+#include <config.h>
+
+#include "open_policies_dialog.h"
+#include "utilgui.h"
+
+#include <assert.h>
+#include <errno.h>
+#include <string.h>
+#include <apol/util.h>
+#include <glade/glade.h>
+
+struct open_policy_pane
+{
+ struct open_policy *op;
+ GtkRadioButton *monolithic_radio, *modular_radio;
+ GtkLabel *main_label;
+
+ GtkHBox *bottom_hbox;
+ GtkListStore *module_store;
+
+ GtkEntry *base_entry;
+ GtkButton *base_browse_button;
+
+ GtkTreeView *module_view;
+ GtkButton *add_button, *remove_button, *import_button, *export_button;
+ char *last_module_path;
+};
+
+struct open_policy
+{
+ GladeXML *xml;
+ toplevel_t *top;
+ GtkDialog *dialog;
+ GtkButton *ok_button, *rundiff_button;
+ struct open_policy_pane pane[SEDIFFX_POLICY_NUM];
+};
+
+enum module_columns
+{
+ PATH_COLUMN = 0, NAME_COLUMN, VERSION_COLUMN, NUM_COLUMNS
+};
+
+/**
+ * Sort columns in alphabetical order.
+ */
+static gint open_policy_sort(GtkTreeModel * model, GtkTreeIter * a, GtkTreeIter * b, gpointer user_data)
+{
+ GValue value_a = { 0 }, value_b = {
+ 0};
+ const char *name_a, *name_b;
+ int retval, column_id = GPOINTER_TO_INT(user_data);
+
+ gtk_tree_model_get_value(model, a, column_id, &value_a);
+ gtk_tree_model_get_value(model, b, column_id, &value_b);
+ name_a = g_value_get_string(&value_a);
+ name_b = g_value_get_string(&value_b);
+ retval = strcmp(name_a, name_b);
+ g_value_unset(&value_a);
+ g_value_unset(&value_b);
+ return retval;
+}
+
+static void open_policy_init_pane(struct open_policy *op, sediffx_policy_e which, const char *suffix)
+{
+ struct open_policy_pane *pane = op->pane + which;
+ GString *s = g_string_new(NULL);
+ GtkCellRenderer *renderer = gtk_cell_renderer_text_new();
+ GtkTreeViewColumn *column;
+ GtkTreeSelection *selection;
+
+ g_string_printf(s, "monolithic radio%s", suffix);
+ pane->monolithic_radio = GTK_RADIO_BUTTON(glade_xml_get_widget(op->xml, s->str));
+ g_string_printf(s, "modular radio%s", suffix);
+ pane->modular_radio = GTK_RADIO_BUTTON(glade_xml_get_widget(op->xml, s->str));
+ g_string_printf(s, "main filename label%s", suffix);
+ pane->main_label = GTK_LABEL(glade_xml_get_widget(op->xml, s->str));
+ assert(pane->monolithic_radio != NULL && pane->modular_radio != NULL && pane->main_label != NULL);
+
+ g_string_printf(s, "base entry%s", suffix);
+ pane->base_entry = GTK_ENTRY(glade_xml_get_widget(op->xml, s->str));
+ g_string_printf(s, "base browse%s", suffix);
+ pane->base_browse_button = GTK_BUTTON(glade_xml_get_widget(op->xml, s->str));
+ assert(pane->base_entry != NULL && pane->base_browse_button != NULL);
+
+ g_string_printf(s, "hbox.3%s", suffix);
+ pane->bottom_hbox = GTK_HBOX(glade_xml_get_widget(op->xml, s->str));
+ assert(pane->bottom_hbox != NULL);
+
+ g_string_printf(s, "module view%s", suffix);
+ pane->module_view = GTK_TREE_VIEW(glade_xml_get_widget(op->xml, s->str));
+ assert(pane->module_view != NULL);
+ pane->module_store = gtk_list_store_new(NUM_COLUMNS, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING);
+ gtk_tree_view_set_model(pane->module_view, GTK_TREE_MODEL(pane->module_store));
+
+ g_string_printf(s, "module add button%s", suffix);
+ pane->add_button = GTK_BUTTON(glade_xml_get_widget(op->xml, s->str));
+ g_string_printf(s, "module remove button%s", suffix);
+ pane->remove_button = GTK_BUTTON(glade_xml_get_widget(op->xml, s->str));
+ g_string_printf(s, "module list import button%s", suffix);
+ pane->import_button = GTK_BUTTON(glade_xml_get_widget(op->xml, s->str));
+ g_string_printf(s, "module list export button%s", suffix);
+ pane->export_button = GTK_BUTTON(glade_xml_get_widget(op->xml, s->str));
+ assert(pane->add_button != NULL && pane->remove_button != NULL &&
+ pane->import_button != NULL && pane->export_button != NULL);
+
+ g_string_free(s, TRUE);
+
+ selection = gtk_tree_view_get_selection(pane->module_view);
+ gtk_tree_selection_set_mode(selection, GTK_SELECTION_BROWSE);
+
+ column = gtk_tree_view_column_new_with_attributes("Module", renderer, "text", NAME_COLUMN, NULL);
+ gtk_tree_view_column_set_sort_column_id(column, NAME_COLUMN);
+ gtk_tree_view_column_set_resizable(column, TRUE);
+ gtk_tree_view_append_column(pane->module_view, column);
+ gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(pane->module_store), NAME_COLUMN, open_policy_sort,
+ (gpointer) NAME_COLUMN, NULL);
+
+ column = gtk_tree_view_column_new_with_attributes("Version", renderer, "text", VERSION_COLUMN, NULL);
+ gtk_tree_view_column_set_sort_column_id(column, VERSION_COLUMN);
+ gtk_tree_view_column_set_resizable(column, TRUE);
+ gtk_tree_view_append_column(pane->module_view, column);
+ gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(pane->module_store), VERSION_COLUMN, open_policy_sort,
+ (gpointer) VERSION_COLUMN, NULL);
+
+ column = gtk_tree_view_column_new_with_attributes("Path", renderer, "text", PATH_COLUMN, NULL);
+ gtk_tree_view_column_set_sort_column_id(column, PATH_COLUMN);
+ gtk_tree_view_column_set_resizable(column, TRUE);
+ gtk_tree_view_append_column(pane->module_view, column);
+ gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(pane->module_store), PATH_COLUMN, open_policy_sort,
+ (gpointer) PATH_COLUMN, NULL);
+
+ gtk_tree_sortable_set_sort_column_id(GTK_TREE_SORTABLE(pane->module_store), NAME_COLUMN, GTK_SORT_ASCENDING);
+}
+
+static void open_policy_init_widgets(struct open_policy *op)
+{
+ op->dialog = GTK_DIALOG(glade_xml_get_widget(op->xml, "PoliciesOpenWindow"));
+ assert(op->dialog != NULL);
+ gtk_window_set_transient_for(GTK_WINDOW(op->dialog), toplevel_get_window(op->top));
+ op->ok_button = GTK_BUTTON(glade_xml_get_widget(op->xml, "ok button"));
+ op->rundiff_button = GTK_BUTTON(glade_xml_get_widget(op->xml, "rundiff button"));
+ assert(op->ok_button != NULL && op->rundiff_button != NULL);
+
+ open_policy_init_pane(op, SEDIFFX_POLICY_ORIG, "");
+ open_policy_init_pane(op, SEDIFFX_POLICY_MOD, " 1");
+}
+
+static void open_policy_on_policy_type_toggle(GtkToggleButton * widget, gpointer user_data)
+{
+ struct open_policy_pane *pane = (struct open_policy_pane *)user_data;
+ /* clicking on the radio buttons emit two toggle signals, one for
+ * the original button and one for the new one. thus only need to
+ * handle half of all signals */
+ if (!gtk_toggle_button_get_active(widget)) {
+ return;
+ }
+ char *prefix;
+ GString *s = g_string_new(NULL);
+ if (pane == &(pane->op->pane[SEDIFFX_POLICY_ORIG])) {
+ prefix = "Original";
+ } else {
+ prefix = "Modified";
+ }
+ if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(pane->monolithic_radio))) {
+ gtk_widget_set_sensitive(GTK_WIDGET(pane->bottom_hbox), FALSE);
+ g_string_printf(s, "<b>%s Policy Filename:</b>", prefix);
+ gtk_label_set_markup(pane->main_label, s->str);
+ } else {
+ gtk_widget_set_sensitive(GTK_WIDGET(pane->bottom_hbox), TRUE);
+ g_string_printf(s, "<b>%s Base Filename:</b>", prefix);
+ gtk_label_set_markup(pane->main_label, s->str);
+ }
+ g_string_free(s, TRUE);
+}
+
+static void open_policy_on_entry_event_after(GtkWidget * widget __attribute__ ((unused)), GdkEvent * event
+ __attribute__ ((unused)), gpointer user_data)
+{
+ struct open_policy *op = (struct open_policy *)user_data;
+ gboolean sens_orig = FALSE, sens_mod = FALSE;
+ if (strcmp(gtk_entry_get_text(op->pane[SEDIFFX_POLICY_ORIG].base_entry), "") != 0) {
+ sens_orig = TRUE;
+ }
+ if (strcmp(gtk_entry_get_text(op->pane[SEDIFFX_POLICY_MOD].base_entry), "") != 0) {
+ sens_mod = TRUE;
+ }
+ gboolean sens = sens_orig && sens_mod;
+ gtk_widget_set_sensitive(GTK_WIDGET(op->pane[SEDIFFX_POLICY_ORIG].export_button), sens_orig);
+ gtk_widget_set_sensitive(GTK_WIDGET(op->pane[SEDIFFX_POLICY_MOD].export_button), sens_mod);
+ gtk_widget_set_sensitive(GTK_WIDGET(op->ok_button), sens);
+ gtk_widget_set_sensitive(GTK_WIDGET(op->rundiff_button), sens);
+}
+
+static void open_policy_on_base_browse_click(GtkButton * button __attribute__ ((unused)), gpointer user_data)
+{
+ struct open_policy_pane *pane = (struct open_policy_pane *)user_data;
+ const char *current_path = gtk_entry_get_text(pane->base_entry);
+ char *title;
+ apol_vector_t *paths;
+ if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(pane->monolithic_radio))) {
+ title = "Open Monolithic Policy";
+ } else {
+ title = "Open Modular Policy";
+ }
+ if (strcmp(current_path, "") == 0) {
+ current_path = NULL;
+ }
+ if ((paths = util_open_file(GTK_WINDOW(pane->op->dialog), title, current_path, 0)) == NULL) {
+ return;
+ }
+ gtk_entry_set_text(pane->base_entry, apol_vector_get_element(paths, 0));
+ apol_vector_destroy(&paths);
+}
+
+/**
+ * Attempt to load a module and retrieve its name and version. Upon
+ * success add an entry to the list store.
+ *
+ * @return 0 on success, < 0 on error.
+ */
+static int open_policy_load_module(struct open_policy *op, struct open_policy_pane *pane, const char *path)
+{
+ const char *module_name, *version_string;
+ int module_type;
+ qpol_module_t *module = NULL;
+ GtkTreeIter iter;
+
+ /* check if modulue was already loaded */
+ gboolean iter_valid = gtk_tree_model_get_iter_first(GTK_TREE_MODEL(pane->module_store), &iter);
+ while (iter_valid) {
+ char *s;
+ gtk_tree_model_get(GTK_TREE_MODEL(pane->module_store), &iter, PATH_COLUMN, &s, -1);
+ if (strcmp(s, path) == 0) {
+ toplevel_ERR(op->top, "Module %s was already added.", path);
+ return -1;
+ }
+ iter_valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(pane->module_store), &iter);
+ }
+ if ((qpol_module_create_from_file(path, &module)) < 0) {
+ toplevel_ERR(op->top, "Error opening module %s: %s", path, strerror(errno));
+ return -1;
+ }
+ if (qpol_module_get_name(module, &module_name) < 0 ||
+ qpol_module_get_version(module, &version_string) < 0 || qpol_module_get_type(module, &module_type) < 0) {
+ toplevel_ERR(op->top, "Error reading module %s: %s", path, strerror(errno));
+ qpol_module_destroy(&module);
+ return -1;
+ }
+ if (module_type != QPOL_MODULE_OTHER) {
+ toplevel_ERR(op->top, "%s is not a loadable module.", path);
+ qpol_module_destroy(&module);
+ return -1;
+ }
+ gtk_list_store_append(pane->module_store, &iter);
+ gtk_list_store_set(pane->module_store, &iter, PATH_COLUMN, path, NAME_COLUMN, module_name, VERSION_COLUMN, version_string,
+ -1);
+ qpol_module_destroy(&module);
+ return 0;
+}
+
+static void open_policy_on_add_click(GtkButton * button __attribute__ ((unused)), gpointer user_data)
+{
+ struct open_policy_pane *pane = (struct open_policy_pane *)user_data;
+ apol_vector_t *paths;
+ const char *path = NULL, *prev_path;
+ size_t i;
+ if ((prev_path = pane->last_module_path) == NULL) {
+ prev_path = gtk_entry_get_text(pane->base_entry);
+ if (strcmp(prev_path, "") == 0) {
+ prev_path = NULL;
+ }
+ }
+ paths = util_open_file(GTK_WINDOW(pane->op->dialog), "Open Module", prev_path, 1);
+ if (paths == NULL) {
+ return;
+ }
+ int all_succeed = 1;
+ for (i = 0; i < apol_vector_get_size(paths); i++) {
+ path = apol_vector_get_element(paths, i);
+ if (open_policy_load_module(pane->op, pane, path) < 0) {
+ all_succeed = 0;
+ }
+ }
+ if (all_succeed) {
+ assert(path != NULL);
+ free(pane->last_module_path);
+ pane->last_module_path = strdup(path);
+ }
+ apol_vector_destroy(&paths);
+}
+
+static void open_policy_on_remove_click(GtkButton * button __attribute__ ((unused)), gpointer user_data)
+{
+ struct open_policy_pane *pane = (struct open_policy_pane *)user_data;
+ GtkTreeSelection *selection = gtk_tree_view_get_selection(pane->module_view);
+ GtkTreeIter iter;
+ char *path;
+ if (!gtk_tree_selection_get_selected(selection, NULL, &iter)) {
+ return;
+ }
+ gtk_tree_model_get(GTK_TREE_MODEL(pane->module_store), &iter, 0, &path, -1);
+ gtk_list_store_remove(pane->module_store, &iter);
+}
+
+static void open_policy_on_import_click(GtkButton * button __attribute__ ((unused)), gpointer user_data);
+static void open_policy_on_export_click(GtkButton * button __attribute__ ((unused)), gpointer user_data);
+
+static void open_policy_on_selection_change(GtkTreeSelection * selection, gpointer user_data)
+{
+ struct open_policy_pane *pane = (struct open_policy_pane *)user_data;
+ gboolean sens = gtk_tree_selection_get_selected(selection, NULL, NULL);
+ gtk_widget_set_sensitive(GTK_WIDGET(pane->remove_button), sens);
+}
+
+static void open_policy_init_signals(struct open_policy *op)
+{
+ sediffx_policy_e i;
+ for (i = SEDIFFX_POLICY_ORIG; i < SEDIFFX_POLICY_NUM; i++) {
+ struct open_policy_pane *pane = op->pane + i;
+ GtkTreeSelection *selection = gtk_tree_view_get_selection(pane->module_view);
+ g_signal_connect(pane->monolithic_radio, "toggled", G_CALLBACK(open_policy_on_policy_type_toggle), pane);
+ g_signal_connect(pane->modular_radio, "toggled", G_CALLBACK(open_policy_on_policy_type_toggle), pane);
+ g_signal_connect(pane->base_entry, "event-after", G_CALLBACK(open_policy_on_entry_event_after), op);
+ g_signal_connect(pane->base_browse_button, "clicked", G_CALLBACK(open_policy_on_base_browse_click), pane);
+ g_signal_connect(selection, "changed", G_CALLBACK(open_policy_on_selection_change), pane);
+ g_signal_connect(pane->add_button, "clicked", G_CALLBACK(open_policy_on_add_click), pane);
+ g_signal_connect(pane->remove_button, "clicked", G_CALLBACK(open_policy_on_remove_click), pane);
+ g_signal_connect(pane->import_button, "clicked", G_CALLBACK(open_policy_on_import_click), pane);
+ g_signal_connect(pane->export_button, "clicked", G_CALLBACK(open_policy_on_export_click), pane);
+ }
+}
+
+static void open_policy_init_value(struct open_policy *op, const apol_policy_path_t * path, struct open_policy_pane *pane)
+{
+ apol_policy_path_type_e path_type = apol_policy_path_get_type(path);
+ const char *primary_path = apol_policy_path_get_primary(path);
+ gtk_entry_set_text(pane->base_entry, primary_path);
+ gtk_list_store_clear(pane->module_store);
+ if (path_type == APOL_POLICY_PATH_TYPE_MONOLITHIC) {
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(pane->monolithic_radio), TRUE);
+ } else if (path_type == APOL_POLICY_PATH_TYPE_MODULAR) {
+ const apol_vector_t *modules = apol_policy_path_get_modules(path);
+ size_t i;
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(pane->modular_radio), TRUE);
+ for (i = 0; i < apol_vector_get_size(modules); i++) {
+ char *module_path = apol_vector_get_element(modules, i);
+ if (open_policy_load_module(op, pane, module_path) < 0) {
+ break;
+ }
+ }
+ } else {
+ /* should never get here */
+ toplevel_ERR(pane->op->top, "Unknown policy path type %d.", path_type);
+ }
+}
+
+static void open_policy_init_values(struct open_policy *op, const apol_policy_path_t * orig_path,
+ const apol_policy_path_t * mod_path)
+{
+ if (orig_path != NULL) {
+ open_policy_init_value(op, orig_path, op->pane + SEDIFFX_POLICY_ORIG);
+ }
+ if (mod_path != NULL) {
+ open_policy_init_value(op, mod_path, op->pane + SEDIFFX_POLICY_MOD);
+ }
+}
+
+/**
+ * Build the policy path corresponding to the user's inputs on this
+ * dialog.
+ *
+ * @return path for the dialog, or NULL upon error. The caller must
+ * call apol_policy_path_destroy() afterwards.
+ */
+static apol_policy_path_t *open_policy_build_path(struct open_policy_pane *pane)
+{
+ const char *primary_path = gtk_entry_get_text(pane->base_entry);
+ apol_policy_path_type_e path_type = APOL_POLICY_PATH_TYPE_MONOLITHIC;
+ apol_vector_t *modules = NULL;
+ apol_policy_path_t *path = NULL;
+ if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(pane->modular_radio))) {
+ path_type = APOL_POLICY_PATH_TYPE_MODULAR;
+ GtkTreeIter iter;
+ if ((modules = apol_vector_create(free)) == NULL) {
+ toplevel_ERR(pane->op->top, "%s", strerror(errno));
+ return NULL;
+ }
+ if (gtk_tree_model_get_iter_first(GTK_TREE_MODEL(pane->module_store), &iter)) {
+ do {
+ GValue value = { 0 };
+ char *module_path;
+ gtk_tree_model_get_value(GTK_TREE_MODEL(pane->module_store), &iter, PATH_COLUMN, &value);
+ module_path = g_value_dup_string(&value);
+ g_value_unset(&value);
+ if (apol_vector_append(modules, module_path) < 0) {
+ toplevel_ERR(pane->op->top, "%s", strerror(errno));
+ free(module_path);
+ apol_vector_destroy(&modules);
+ return NULL;
+ }
+ } while (gtk_tree_model_iter_next(GTK_TREE_MODEL(pane->module_store), &iter));
+ }
+ }
+ path = apol_policy_path_create(path_type, primary_path, modules);
+ apol_vector_destroy(&modules);
+ if (path == NULL) {
+ toplevel_ERR(pane->op->top, "%s", strerror(errno));
+ return NULL;
+ }
+ return path;
+}
+
+static void open_policy_on_import_click(GtkButton * button __attribute__ ((unused)), gpointer user_data)
+{
+ struct open_policy_pane *pane = (struct open_policy_pane *)user_data;
+ apol_vector_t *paths = NULL;
+ apol_policy_path_t *ppath = NULL;
+ const char *path = NULL, *prev_path;
+ if ((prev_path = pane->last_module_path) == NULL) {
+ prev_path = gtk_entry_get_text(pane->base_entry);
+ if (strcmp(prev_path, "") == 0) {
+ prev_path = NULL;
+ }
+ }
+ paths = util_open_file(GTK_WINDOW(pane->op->dialog), "Import Policy List", prev_path, 1);
+ if (paths == NULL) {
+ return;
+ }
+ path = apol_vector_get_element(paths, 0);
+ ppath = apol_policy_path_create_from_file(path);
+ if (ppath == NULL) {
+ toplevel_ERR(pane->op->top, "Error importing policy list %s: %s", path, strerror(errno));
+ goto cleanup;
+ }
+ open_policy_init_value(pane->op, ppath, pane);
+ free(pane->last_module_path);
+ pane->last_module_path = strdup(path);
+ cleanup:
+ apol_vector_destroy(&paths);
+ apol_policy_path_destroy(&ppath);
+}
+
+static void open_policy_on_export_click(GtkButton * button __attribute__ ((unused)), gpointer user_data)
+{
+ struct open_policy_pane *pane = (struct open_policy_pane *)user_data;
+ char *path = util_save_file(GTK_WINDOW(pane->op->dialog), "Export Policy List", NULL);
+ apol_policy_path_t *ppath = NULL;
+ if (path == NULL) {
+ return;
+ }
+ ppath = open_policy_build_path(pane);
+ if (ppath == NULL) {
+ goto cleanup;
+ }
+ if (apol_policy_path_to_file(ppath, path) < 0) {
+ toplevel_ERR(pane->op->top, "Error exporting policy list %s: %s", path, strerror(errno));
+ }
+ cleanup:
+ g_free(path);
+ apol_policy_path_destroy(&ppath);
+}
+
+void open_policies_dialog_run(toplevel_t * top, const apol_policy_path_t * orig_path, const apol_policy_path_t * mod_path)
+{
+ struct open_policy op;
+ gint response;
+ apol_policy_path_t *input[SEDIFFX_POLICY_NUM];
+
+ memset(&op, 0, sizeof(op));
+ op.top = top;
+ op.xml = glade_xml_new(toplevel_get_glade_xml(top), "PoliciesOpenWindow", NULL);
+ op.pane[SEDIFFX_POLICY_ORIG].op = &op;
+ op.pane[SEDIFFX_POLICY_MOD].op = &op;
+
+ open_policy_init_widgets(&op);
+ open_policy_init_signals(&op);
+ open_policy_init_values(&op, orig_path, mod_path);
+ open_policy_on_entry_event_after(NULL, NULL, &op);
+
+ while (1) {
+ response = gtk_dialog_run(op.dialog);
+ if (response == GTK_RESPONSE_CANCEL || response == GTK_RESPONSE_DELETE_EVENT) {
+ break;
+ }
+ if ((input[SEDIFFX_POLICY_ORIG] = open_policy_build_path(&op.pane[SEDIFFX_POLICY_ORIG])) == NULL ||
+ (input[SEDIFFX_POLICY_MOD] = open_policy_build_path(&op.pane[SEDIFFX_POLICY_MOD])) == NULL) {
+ continue;
+ }
+ if (toplevel_open_policies(op.top, input[SEDIFFX_POLICY_ORIG], input[SEDIFFX_POLICY_MOD]) == 0) {
+ break;
+ }
+ }
+ gtk_widget_destroy(GTK_WIDGET(op.dialog));
+ free(op.pane[SEDIFFX_POLICY_ORIG].last_module_path);
+ free(op.pane[SEDIFFX_POLICY_MOD].last_module_path);
+ g_object_unref(op.pane[SEDIFFX_POLICY_ORIG].module_store);
+ g_object_unref(op.pane[SEDIFFX_POLICY_MOD].module_store);
+
+ if (response == 0) {
+ /* Run Diff button was clicked */
+ toplevel_run_diff(op.top);
+ }
+}