514 lines
15 KiB
C
514 lines
15 KiB
C
/* $Id$ */
|
|
/* Copyright (c) 2011 Pierre Pronchery <khorben@defora.org> */
|
|
/* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. Neither the name of the authors nor the names of the contributors
|
|
* may be used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
* THIS SOFTWARE IS PROVIDED BY ITS AUTHORS AND CONTRIBUTORS ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE. */
|
|
/* TODO:
|
|
* - add a preferences structure
|
|
* - complete the function list
|
|
* - add a window automatically displaying integers in base 2, 8, 10 and 16 */
|
|
|
|
|
|
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <gtk/gtk.h>
|
|
#include <Devel/Asm.h>
|
|
|
|
|
|
/* gdeasm */
|
|
/* private */
|
|
/* types */
|
|
typedef struct _GDeasm
|
|
{
|
|
char * arch;
|
|
char * format;
|
|
|
|
/* widgets */
|
|
GtkListStore * func_store;
|
|
GtkListStore * str_store;
|
|
GtkTreeStore * asm_store;
|
|
GtkWidget * asm_view;
|
|
} GDeasm;
|
|
|
|
|
|
/* prototypes */
|
|
static GDeasm * _gdeasm_new(char const * arch, char const * format);
|
|
static void _gdeasm_delete(GDeasm * gdeasm);
|
|
|
|
static int _gdeasm_open(GDeasm * gdeasm, char const * filename, int raw);
|
|
|
|
|
|
/* callbacks */
|
|
static gboolean _gdeasm_on_closex(void);
|
|
static void _gdeasm_on_comment_edited(GtkCellRendererText * renderer,
|
|
gchar * arg1, gchar * arg2, gpointer data);
|
|
static void _gdeasm_on_function_activated(GtkTreeView * view,
|
|
GtkTreePath * path, GtkTreeViewColumn * column, gpointer data);
|
|
static void _gdeasm_on_open(gpointer data);
|
|
|
|
static int _usage(void);
|
|
|
|
|
|
/* functions */
|
|
/* gdeasm_new */
|
|
static GDeasm * _gdeasm_new(char const * arch, char const * format)
|
|
{
|
|
GDeasm * gdeasm;
|
|
GtkWidget * window;
|
|
GtkWidget * vbox;
|
|
GtkWidget * toolbar;
|
|
GtkToolItem * toolitem;
|
|
GtkWidget * hpaned;
|
|
GtkWidget * vpaned;
|
|
GtkWidget * scrolled;
|
|
GtkWidget * treeview;
|
|
GtkCellRenderer * renderer;
|
|
GtkTreeViewColumn * column;
|
|
char const * headers1[] = { "Functions", "Offset" };
|
|
char const * headers2[] = { "Strings" };
|
|
char const * headers3[] = { "Address", "Instruction", "Operand",
|
|
"Operand", "Operand", "Operand", "Operand", "Comment" };
|
|
size_t i;
|
|
|
|
if((gdeasm = malloc(sizeof(*gdeasm))) == NULL)
|
|
return NULL;
|
|
gdeasm->arch = (arch != NULL) ? strdup(arch) : NULL;
|
|
gdeasm->format = (format != NULL) ? strdup(format) : NULL;
|
|
gdeasm->func_store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_STRING,
|
|
G_TYPE_UINT);
|
|
gdeasm->str_store = gtk_list_store_new(1, G_TYPE_STRING);
|
|
gdeasm->asm_store = gtk_tree_store_new(9, G_TYPE_STRING, G_TYPE_STRING,
|
|
G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING,
|
|
G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING,
|
|
G_TYPE_UINT);
|
|
window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
|
|
gtk_window_set_default_size(GTK_WINDOW(window), 640, 480);
|
|
g_signal_connect_swapped(window, "delete-event", G_CALLBACK(
|
|
_gdeasm_on_closex), NULL);
|
|
vbox = gtk_vbox_new(FALSE, 0);
|
|
/* toolbar */
|
|
toolbar = gtk_toolbar_new();
|
|
toolitem = gtk_tool_button_new_from_stock(GTK_STOCK_OPEN);
|
|
g_signal_connect_swapped(toolitem, "clicked", G_CALLBACK(
|
|
_gdeasm_on_open), gdeasm);
|
|
gtk_toolbar_insert(GTK_TOOLBAR(toolbar), toolitem, -1);
|
|
gtk_box_pack_start(GTK_BOX(vbox), toolbar, FALSE, TRUE, 0);
|
|
/* view */
|
|
hpaned = gtk_hpaned_new();
|
|
vpaned = gtk_vpaned_new();
|
|
/* functions */
|
|
scrolled = gtk_scrolled_window_new(NULL, NULL);
|
|
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
|
|
GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
|
|
treeview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(
|
|
gdeasm->func_store));
|
|
gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview), TRUE);
|
|
gtk_tree_view_set_reorderable(GTK_TREE_VIEW(treeview), FALSE);
|
|
g_signal_connect(treeview, "row-activated", G_CALLBACK(
|
|
_gdeasm_on_function_activated), gdeasm);
|
|
for(i = 0; i < sizeof(headers1) / sizeof(*headers1); i++)
|
|
{
|
|
renderer = gtk_cell_renderer_text_new();
|
|
column = gtk_tree_view_column_new_with_attributes(headers1[i],
|
|
renderer, "text", i, NULL);
|
|
if(i == 1)
|
|
g_object_set(renderer, "family", "Monospace", NULL);
|
|
gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column);
|
|
}
|
|
gtk_container_add(GTK_CONTAINER(scrolled), treeview);
|
|
gtk_paned_pack1(GTK_PANED(vpaned), scrolled, FALSE, TRUE);
|
|
/* strings */
|
|
scrolled = gtk_scrolled_window_new(NULL, NULL);
|
|
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
|
|
GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
|
|
treeview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(
|
|
gdeasm->str_store));
|
|
gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview), TRUE);
|
|
gtk_tree_view_set_reorderable(GTK_TREE_VIEW(treeview), FALSE);
|
|
for(i = 0; i < sizeof(headers2) / sizeof(*headers2); i++)
|
|
{
|
|
renderer = gtk_cell_renderer_text_new();
|
|
column = gtk_tree_view_column_new_with_attributes(headers2[i],
|
|
renderer, "text", i, NULL);
|
|
gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column);
|
|
}
|
|
gtk_container_add(GTK_CONTAINER(scrolled), treeview);
|
|
gtk_paned_pack2(GTK_PANED(vpaned), scrolled, FALSE, FALSE);
|
|
gtk_paned_add1(GTK_PANED(hpaned), vpaned);
|
|
/* assembly */
|
|
scrolled = gtk_scrolled_window_new(NULL, NULL);
|
|
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
|
|
GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
|
|
treeview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(
|
|
gdeasm->asm_store));
|
|
gdeasm->asm_view = treeview;
|
|
gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview), FALSE);
|
|
gtk_tree_view_set_reorderable(GTK_TREE_VIEW(treeview), FALSE);
|
|
gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(treeview), TRUE);
|
|
for(i = 0; i < sizeof(headers3) / sizeof(*headers3); i++)
|
|
{
|
|
renderer = gtk_cell_renderer_text_new();
|
|
column = gtk_tree_view_column_new_with_attributes(headers3[i],
|
|
renderer, "text", i, NULL);
|
|
if(i == 0)
|
|
g_object_set(renderer, "family", "Monospace", NULL);
|
|
else if(i == 7) /* the last column is editable */
|
|
{
|
|
g_object_set(renderer, "editable", TRUE, "style",
|
|
PANGO_STYLE_ITALIC, NULL);
|
|
g_signal_connect(renderer, "edited", G_CALLBACK(
|
|
_gdeasm_on_comment_edited),
|
|
gdeasm);
|
|
}
|
|
gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column);
|
|
}
|
|
gtk_container_add(GTK_CONTAINER(scrolled), treeview);
|
|
gtk_paned_add2(GTK_PANED(hpaned), scrolled);
|
|
gtk_box_pack_start(GTK_BOX(vbox), hpaned, TRUE, TRUE, 0);
|
|
gtk_container_add(GTK_CONTAINER(window), vbox);
|
|
gtk_widget_show_all(window);
|
|
return gdeasm;
|
|
}
|
|
|
|
|
|
/* gdeasm_delete */
|
|
static void _gdeasm_delete(GDeasm * gdeasm)
|
|
{
|
|
free(gdeasm->arch);
|
|
free(gdeasm->format);
|
|
free(gdeasm);
|
|
}
|
|
|
|
|
|
/* gdeasm_open */
|
|
static int _open_code(GDeasm * gdeasm, AsmCode * af);
|
|
static int _open_code_section(GDeasm * gdeasm, AsmCode * code,
|
|
AsmSection * section);
|
|
static void _open_functions(GDeasm * gdeasm, AsmFunction * af, size_t af_cnt);
|
|
static void _open_instruction(GDeasm * gdeasm, GtkTreeIter * parent,
|
|
ArchInstructionCall * call);
|
|
static void _open_parse_dregister(char * buf, size_t size, ArchOperand * ao);
|
|
static void _open_parse_dregister2(char * buf, size_t size, ArchOperand * ao);
|
|
static void _open_parse_immediate(char * buf, size_t size, ArchOperand * ao);
|
|
static void _open_strings(GDeasm * gdeasm, AsmString * as, size_t as_cnt);
|
|
|
|
static int _gdeasm_open(GDeasm * gdeasm, char const * filename, int raw)
|
|
{
|
|
int ret = -1;
|
|
Asm * a;
|
|
AsmCode * code;
|
|
AsmFunction * af;
|
|
size_t af_cnt;
|
|
AsmString * as;
|
|
size_t as_cnt;
|
|
|
|
if((a = asm_new(gdeasm->arch, gdeasm->format)) == NULL)
|
|
return -1;
|
|
if((code = asm_open_deassemble(a, filename, raw)) != NULL)
|
|
ret = _open_code(gdeasm, code);
|
|
asmcode_get_functions(code, &af, &af_cnt);
|
|
_open_functions(gdeasm, af, af_cnt);
|
|
asmcode_get_strings(code, &as, &as_cnt);
|
|
_open_strings(gdeasm, as, as_cnt);
|
|
asm_close(a);
|
|
asm_delete(a);
|
|
return ret;
|
|
}
|
|
|
|
static int _open_code(GDeasm * gdeasm, AsmCode * code)
|
|
{
|
|
int ret;
|
|
AsmSection * sections;
|
|
size_t sections_cnt;
|
|
size_t i;
|
|
|
|
asmcode_get_sections(code, §ions, §ions_cnt);
|
|
for(i = 0; i < sections_cnt; i++)
|
|
if((ret = _open_code_section(gdeasm, code, §ions[i])) != 0)
|
|
break;
|
|
return ret;
|
|
}
|
|
|
|
static int _open_code_section(GDeasm * gdeasm, AsmCode * code,
|
|
AsmSection * section)
|
|
{
|
|
GtkTreeIter iter;
|
|
ArchInstructionCall * calls = NULL;
|
|
size_t calls_cnt = 0;
|
|
size_t i;
|
|
|
|
gtk_tree_store_append(gdeasm->asm_store, &iter, NULL);
|
|
gtk_tree_store_set(gdeasm->asm_store, &iter, 1, section->name, -1);
|
|
if(asmcode_decode_section(code, section, &calls, &calls_cnt) != 0)
|
|
return -1;
|
|
for(i = 0; i < calls_cnt; i++)
|
|
_open_instruction(gdeasm, &iter, &calls[i]);
|
|
free(calls);
|
|
return 0;
|
|
}
|
|
|
|
static void _open_functions(GDeasm * gdeasm, AsmFunction * af, size_t af_cnt)
|
|
{
|
|
size_t i;
|
|
GtkTreeIter iter;
|
|
char buf[16];
|
|
|
|
for(i = 0; i < af_cnt; i++)
|
|
{
|
|
snprintf(buf, sizeof(buf), "%08lx", af[i].offset);
|
|
gtk_list_store_append(gdeasm->func_store, &iter);
|
|
gtk_list_store_set(gdeasm->func_store, &iter, 0, af[i].name,
|
|
1, buf, 2, af[i].offset, -1);
|
|
}
|
|
}
|
|
|
|
static void _open_instruction(GDeasm * gdeasm, GtkTreeIter * parent,
|
|
ArchInstructionCall * call)
|
|
{
|
|
GtkTreeIter iter;
|
|
char buf[32];
|
|
size_t i;
|
|
ArchOperand * ao;
|
|
char const * name;
|
|
|
|
gtk_tree_store_append(gdeasm->asm_store, &iter, parent);
|
|
snprintf(buf, sizeof(buf), "%08lx", call->base);
|
|
gtk_tree_store_set(gdeasm->asm_store, &iter, 0, buf, 1, call->name,
|
|
8, call->offset, -1);
|
|
for(i = 0; i < call->operands_cnt; i++)
|
|
{
|
|
ao = &call->operands[i];
|
|
switch(AO_GET_TYPE(ao->definition))
|
|
{
|
|
case AOT_DREGISTER:
|
|
_open_parse_dregister(buf, sizeof(buf), ao);
|
|
break;
|
|
case AOT_DREGISTER2:
|
|
_open_parse_dregister2(buf, sizeof(buf), ao);
|
|
break;
|
|
case AOT_IMMEDIATE:
|
|
_open_parse_immediate(buf, sizeof(buf), ao);
|
|
if(AO_GET_VALUE(ao->definition)
|
|
== AOI_REFERS_STRING
|
|
|| AO_GET_VALUE(ao->definition)
|
|
== AOI_REFERS_FUNCTION)
|
|
gtk_tree_store_set(gdeasm->asm_store,
|
|
&iter, 7,
|
|
ao->value.immediate.name,
|
|
-1);
|
|
break;
|
|
case AOT_REGISTER:
|
|
name = call->operands[i].value._register.name;
|
|
snprintf(buf, sizeof(buf), "%%%s", name);
|
|
break;
|
|
default:
|
|
buf[0] = '\0';
|
|
break;
|
|
}
|
|
gtk_tree_store_set(gdeasm->asm_store, &iter, i + 2, buf, -1);
|
|
}
|
|
}
|
|
|
|
static void _open_parse_dregister(char * buf, size_t size, ArchOperand * ao)
|
|
{
|
|
char const * name;
|
|
|
|
name = ao->value.dregister.name;
|
|
if(ao->value.dregister.offset == 0)
|
|
snprintf(buf, size, "[%%%s]", name);
|
|
else
|
|
snprintf(buf, size, "[%%%s + $0x%lx]", name,
|
|
(unsigned long)ao->value.dregister.offset);
|
|
}
|
|
|
|
static void _open_parse_dregister2(char * buf, size_t size, ArchOperand * ao)
|
|
{
|
|
snprintf(buf, size, "[%%%s + %%%s]", ao->value.dregister2.name,
|
|
ao->value.dregister2.name2);
|
|
}
|
|
|
|
static void _open_parse_immediate(char * buf, size_t size, ArchOperand * ao)
|
|
{
|
|
snprintf(buf, size, "%s$0x%lx", ao->value.immediate.negative
|
|
? "-" : "", (unsigned long)ao->value.immediate.value);
|
|
}
|
|
|
|
static void _open_strings(GDeasm * gdeasm, AsmString * as, size_t as_cnt)
|
|
{
|
|
size_t i;
|
|
GtkTreeIter iter;
|
|
|
|
for(i = 0; i < as_cnt; i++)
|
|
{
|
|
gtk_list_store_append(gdeasm->str_store, &iter);
|
|
gtk_list_store_set(gdeasm->str_store, &iter, 0, as[i].name, -1);
|
|
}
|
|
}
|
|
|
|
|
|
/* callbacks */
|
|
/* gdeasm_on_closex */
|
|
static gboolean _gdeasm_on_closex(void)
|
|
{
|
|
gtk_main_quit();
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/* gdeasm_on_comment_edited */
|
|
static void _gdeasm_on_comment_edited(GtkCellRendererText * renderer,
|
|
gchar * arg1, gchar * arg2, gpointer data)
|
|
{
|
|
GDeasm * gdeasm = data;
|
|
GtkTreeModel * model = GTK_TREE_MODEL(gdeasm->asm_store);
|
|
GtkTreeIter iter;
|
|
|
|
if(gtk_tree_model_get_iter_from_string(model, &iter, arg1) == TRUE)
|
|
gtk_tree_store_set(gdeasm->asm_store, &iter, 7, arg2, -1);
|
|
}
|
|
|
|
|
|
/* gdeasm_on_function_activated */
|
|
static void _gdeasm_on_function_activated(GtkTreeView * view,
|
|
GtkTreePath * path, GtkTreeViewColumn * column, gpointer data)
|
|
{
|
|
GDeasm * gdeasm = data;
|
|
GtkTreeModel * model = GTK_TREE_MODEL(gdeasm->func_store);
|
|
GtkTreeIter iter;
|
|
GtkTreeIter parent;
|
|
guint offset;
|
|
gboolean valid;
|
|
guint u;
|
|
GtkTreeSelection * treesel;
|
|
|
|
if(gtk_tree_model_get_iter(model, &iter, path) != TRUE)
|
|
return;
|
|
gtk_tree_model_get(model, &iter, 2, &offset, -1);
|
|
model = GTK_TREE_MODEL(gdeasm->asm_store);
|
|
for(valid = gtk_tree_model_get_iter_first(model, &parent); valid;
|
|
valid = gtk_tree_model_iter_next(model, &parent))
|
|
for(valid = gtk_tree_model_iter_children(model, &iter, &parent);
|
|
valid;
|
|
valid = gtk_tree_model_iter_next(model, &iter))
|
|
{
|
|
gtk_tree_model_get(model, &iter, 8, &u, -1);
|
|
if(offset != u)
|
|
continue;
|
|
view = GTK_TREE_VIEW(gdeasm->asm_view);
|
|
treesel = gtk_tree_view_get_selection(view);
|
|
gtk_tree_selection_select_iter(treesel, &iter);
|
|
path = gtk_tree_model_get_path(model, &iter);
|
|
gtk_tree_view_expand_to_path(view, path);
|
|
gtk_tree_view_scroll_to_cell(view, path, NULL, FALSE,
|
|
0.0, 0.0);
|
|
gtk_tree_path_free(path);
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
/* gdeasm_on_open */
|
|
static void _gdeasm_on_open(gpointer data)
|
|
{
|
|
GDeasm * gdeasm = data;
|
|
GtkWidget * dialog;
|
|
GtkWidget * vbox;
|
|
GtkWidget * widget;
|
|
char * filename = NULL;
|
|
int raw;
|
|
|
|
dialog = gtk_file_chooser_dialog_new("Open file...", NULL,
|
|
GTK_FILE_CHOOSER_ACTION_OPEN,
|
|
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
|
|
GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT, NULL);
|
|
#if GTK_CHECK_VERSION(2, 14, 0)
|
|
vbox = gtk_dialog_get_content_area(GTK_DIALOG(dialog));
|
|
#else
|
|
vbox = GTK_DIALOG(dialog)->vbox;
|
|
#endif
|
|
widget = gtk_check_button_new_with_mnemonic("Open file in _raw mode");
|
|
gtk_box_pack_start(GTK_BOX(vbox), widget, FALSE, TRUE, 0);
|
|
gtk_widget_show_all(vbox);
|
|
if(gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT)
|
|
filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(
|
|
dialog));
|
|
gtk_widget_destroy(dialog);
|
|
if(filename == NULL)
|
|
return;
|
|
raw = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
|
|
gtk_list_store_clear(gdeasm->func_store);
|
|
gtk_list_store_clear(gdeasm->str_store);
|
|
gtk_tree_store_clear(gdeasm->asm_store);
|
|
_gdeasm_open(gdeasm, filename, raw);
|
|
g_free(filename);
|
|
}
|
|
|
|
|
|
/* usage */
|
|
static int _usage(void)
|
|
{
|
|
fputs("Usage: gdeasm [-D][-a arch][-f format] filename\n", stderr);
|
|
return 1;
|
|
}
|
|
|
|
|
|
/* public */
|
|
/* functions */
|
|
/* main */
|
|
int main(int argc, char * argv[])
|
|
{
|
|
int o;
|
|
GDeasm * gdeasm;
|
|
int raw = 0;
|
|
char const * arch = NULL;
|
|
char const * format = NULL;
|
|
|
|
gtk_init(&argc, &argv);
|
|
while((o = getopt(argc, argv, "Da:f:")) != -1)
|
|
switch(o)
|
|
{
|
|
case 'D':
|
|
raw = 1;
|
|
break;
|
|
case 'a':
|
|
arch = optarg;
|
|
break;
|
|
case 'f':
|
|
format = optarg;
|
|
break;
|
|
default:
|
|
return _usage();
|
|
}
|
|
if(optind != argc && optind + 1 != argc)
|
|
return _usage();
|
|
if((gdeasm = _gdeasm_new(arch, format)) == NULL)
|
|
return 2;
|
|
if(optind + 1 == argc)
|
|
_gdeasm_open(gdeasm, argv[optind], raw);
|
|
gtk_main();
|
|
_gdeasm_delete(gdeasm);
|
|
return 0;
|
|
}
|