summaryrefslogtreecommitdiff
path: root/src/debug/gdbrsp/gdb.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/debug/gdbrsp/gdb.c')
-rw-r--r--src/debug/gdbrsp/gdb.c1524
1 files changed, 0 insertions, 1524 deletions
diff --git a/src/debug/gdbrsp/gdb.c b/src/debug/gdbrsp/gdb.c
deleted file mode 100644
index 1d0ffe5..0000000
--- a/src/debug/gdbrsp/gdb.c
+++ /dev/null
@@ -1,1524 +0,0 @@
-
-/* Chrysalide - Outil d'analyse de fichiers binaires
- * gdb.c - débogage à l'aide de gdb.
- *
- * Copyright (C) 2009-2017 Cyrille Bagard
- *
- * This file is part of Chrysalide.
- *
- * Chrysalide 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 3 of the License, or
- * (at your option) any later version.
- *
- * Chrysalide 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 Chrysalide. If not, see <http://www.gnu.org/licenses/>.
- */
-
-
-#include "gdb.h"
-
-
-
-#include <assert.h>
-#include <malloc.h>
-#include <stdarg.h>
-#include <stdio.h>
-#include <string.h>
-
-
-#include "gdb-int.h"
-#include "helpers.h"
-#include "helpers_arm.h"
-#include "helpers_arm64.h"
-#include "tcp.h"
-#include "utils.h"
-#include "../../common/cpp.h"
-#include "../../format/format.h"
-
-
-
-
-
-
-
-
-/* Initialise la classe du débogueur utilisant gdb. */
-static void g_gdb_debugger_class_init(GGdbDebuggerClass *);
-
-/* Procède à l'initialisation du débogueur utilisant gdb. */
-static void g_gdb_debugger_init(GGdbDebugger *);
-
-/* Supprime toutes les références externes. */
-static void g_gdb_debugger_dispose(GGdbDebugger *);
-
-/* Procède à la libération totale de la mémoire. */
-static void g_gdb_debugger_finalize(GGdbDebugger *);
-
-
-/* Met en marche le débogueur utilisant un serveur GDB. */
-static bool g_gdb_debugger_run(GGdbDebugger *);
-
-/* Remet en marche le débogueur utilisant un serveur GDB. */
-//static bool g_gdb_debugger_resume(GGdbDebugger *);
-
-/* Tue le débogueur utilisant un serveur GDB. */
-static bool g_gdb_debugger_kill(GGdbDebugger *);
-
-
-
-
-
-
-/* --------------------------- ENTREES / SORTIES BASIQUES --------------------------- */
-
-
-/* Lit une valeur quelconque à une adresse arbitraire. */
-static bool g_gdb_debugger_read_memory(GGdbDebugger *, virt_t, size_t, ...);
-
-/* Ecrit une valeur quelconque à une adresse arbitraire. */
-static bool g_gdb_debugger_write_memory(GGdbDebugger *, virt_t, size_t, ...);
-
-/* Liste l'ensemble des registres appartenant à un groupe. */
-static char **g_gdb_debugger_get_register_names(const GGdbDebugger *, const char *, size_t *);
-
-/* Indique la taille associée à un registre donné. */
-static unsigned int g_gdb_debugger_get_register_size(const GGdbDebugger *, const char *);
-
-/* Effectue la lecture d'un registre donné. */
-static bool g_gdb_debugger_read_register(GGdbDebugger *, const char *, size_t, ...);
-
-/* Effectue l'écriture d'un registre donné. */
-static bool g_gdb_debugger_write_register(GGdbDebugger *, const char *, size_t, ...);
-
-
-
-/* ------------------------- MANIPULATION DE L'ETAT COURANT ------------------------- */
-
-
-/* Détermine le point d'exécution courant. */
-static bool g_gdb_debugger_get_current_pc(GGdbDebugger *, virt_t *);
-
-/* Remonte la pile d'appels jusqu'au point courant. */
-static bool g_gdb_debugger_compute_call_stack(GGdbDebugger *, virt_t **, size_t *);
-
-
-
-/* --------------------------- GESTION DES POINTS D'ARRET --------------------------- */
-
-
-/* Ajoute un point d'arrêt basique en mémoire. */
-static gdb_breakpoint *g_gdb_debugger_enable_memory_breakpoint(GGdbDebugger *, virt_t);
-
-/* Retire un point d'arrêt basique en mémoire. */
-static bool g_gdb_debugger_disable_memory_breakpoint(GGdbDebugger *, gdb_breakpoint *);
-
-
-
-/* -------------------------- CONTROLE DU FLOT D'EXECUTION -------------------------- */
-
-
-/* Redémarre le processus de débogage lié à un serveur GDB. */
-static bool g_gdb_debugger_restart(GGdbDebugger *);
-
-/* Remet en marche le débogueur utilisant un serveur GDB. */
-static bool g_gdb_debugger_resume(GGdbDebugger *);
-
-
-
-
-
-
-
-
-
-
-
-
-/* Détermine l'identifiant du thread principal courant. */
-static char *g_gdb_debugger_get_active_thread(GGdbDebugger *);
-
-
-
-
-
-
-/* ------------------------ ACCUEIL D'EVENEMENTS ASYNCHRONES ------------------------ */
-
-
-
-
-
-
-
-
-/* Indique le type défini par la GLib pour le débogueur gdb. */
-G_DEFINE_TYPE(GGdbDebugger, g_gdb_debugger, G_TYPE_BINARY_DEBUGGER);
-
-
-/******************************************************************************
-* *
-* Paramètres : klass = classe de débogueur à initialiser. *
-* *
-* Description : Initialise la classe du débogueur utilisant gdb. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void g_gdb_debugger_class_init(GGdbDebuggerClass *klass)
-{
- GObjectClass *object; /* Autre version de la classe */
- GBinaryDebuggerClass *parent; /* Version en classe parente */
-
- object = G_OBJECT_CLASS(klass);
-
- object->dispose = (GObjectFinalizeFunc/* ! */)g_gdb_debugger_dispose;
- object->finalize = (GObjectFinalizeFunc)g_gdb_debugger_finalize;
-
- parent = G_BINARY_DEBUGGER_CLASS(klass);
-
- parent->read_mem = (read_mem_any_fc)g_gdb_debugger_read_memory;
- parent->write_mem = (write_mem_any_fc)g_gdb_debugger_write_memory;
- parent->get_reg_names = (get_reg_names_fc)g_gdb_debugger_get_register_names;
- parent->get_reg_size = (get_reg_size_fc)g_gdb_debugger_get_register_size;
- parent->read_reg = (read_write_reg_any_fc)g_gdb_debugger_read_register;
- parent->write_reg = (read_write_reg_any_fc)g_gdb_debugger_write_register;
-
- parent->get_current_pc = (get_current_pc_fc)g_gdb_debugger_get_current_pc;
- parent->get_call_stack = (get_call_stack_fc)g_gdb_debugger_compute_call_stack;
-
- parent->enable_bp = (enable_mem_bp_fc)g_gdb_debugger_enable_memory_breakpoint;
- parent->disable_bp = (disable_mem_bp_fc)g_gdb_debugger_disable_memory_breakpoint;
-
- parent->restart = (restart_debugger_fc)g_gdb_debugger_restart;
- parent->resume = (resume_debugger_fc)g_gdb_debugger_resume;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = instance de débogueur à préparer. *
-* *
-* Description : Procède à l'initialisation du débogueur utilisant gdb. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void g_gdb_debugger_init(GGdbDebugger *debugger)
-{
- GBinaryDebugger *parent; /* Instance parente */
-
- parent = G_BINARY_DEBUGGER(debugger);
-
- parent->run = (basic_debugger_fc)g_gdb_debugger_run;
- //parent->resume = (resume_debugger_fc)g_gdb_debugger_resume;
- parent->kill = (basic_debugger_fc)g_gdb_debugger_kill;
-
- //parent->get_reg_values = (get_register_values_fc)get_register_values_using_gdb_debugger;
-
- //debugger->cond = g_cond_new();
- //debugger->mutex = g_mutex_new();
-
-
- // FIXME
- //debugger->compute_cstack = compute_call_stack_for_arm64;
- //debugger->fill_mem_bp = fill_memory_breakpoint_cmd_for_arm64;
-
-
-
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = instance d'objet GLib à traiter. *
-* *
-* Description : Supprime toutes les références externes. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void g_gdb_debugger_dispose(GGdbDebugger *debugger)
-{
- if (debugger->stream != NULL)
- g_object_unref(G_OBJECT(debugger->stream));
-
- if (debugger->support != NULL)
- g_object_unref(G_OBJECT(debugger->support));
-
- if (debugger->target != NULL)
- g_object_unref(G_OBJECT(debugger->target));
-
- G_OBJECT_CLASS(g_gdb_debugger_parent_class)->dispose(G_OBJECT(debugger));
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = instance d'objet GLib à traiter. *
-* *
-* Description : Procède à la libération totale de la mémoire. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static void g_gdb_debugger_finalize(GGdbDebugger *debugger)
-{
- G_OBJECT_CLASS(g_gdb_debugger_parent_class)->finalize(G_OBJECT(debugger));
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : binary = binaire représenter à déboguer. *
-* server = nom ou adresse du serveur à contacter. *
-* port = port de connexion. *
-* *
-* Description : Crée un débogueur utilisant un serveur GDB distant. *
-* *
-* Retour : Instance de débogueur mise en place ou NULL. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-GBinaryDebugger *g_gdb_debugger_new(GLoadedBinary *binary, const char *server, unsigned short port)
-{
- GGdbDebugger *result; /* Débogueur à retourner */
- GExeFormat *format; /* Format du binaire chargé */
- const char *arch; /* Architecture d'exécution */
- GArchProcessor *proc; /* Processeur lié au binaire */
- char service[sizeof(XSTR(UINT16_MAX)) + 1]; /* Conversion requise */
-
- result = g_object_new(G_TYPE_GDB_DEBUGGER, NULL);
-
- G_BINARY_DEBUGGER(result)->binary = binary;
- g_object_ref(G_OBJECT(binary));
-
- /* Propriétés de la cible */
-
- format = g_loaded_binary_get_format(binary);
-
- result->endian = g_binary_format_get_endianness(G_BIN_FORMAT(format));
-
- arch = g_exe_format_get_target_machine(format);
-
- if (strcmp(arch, "armv7") == 0)
- result->ops = get_arm_operations();
- else
- result->ops = NULL;
-
- g_object_unref(G_OBJECT(format));
-
- if (result->ops == NULL)
- goto ggdn_error;
-
- proc = g_loaded_binary_get_processor(binary);
-
- result->msize = g_arch_processor_get_memory_size(proc);
-
- g_object_unref(G_OBJECT(proc));
-
- /* Mise en place des modules auxialiaires */
-
- snprintf(service, sizeof(service), "%hu", port);
-
- result->stream = g_gdb_tcp_client_new(server, service, result);
- if (result->stream == NULL) goto ggdn_error;
-
- result->support = g_gdb_support_new(result->stream);
-
- result->target = g_gdb_target_new(result->stream);
- if (result->target == NULL) goto ggdn_error;
-
- return G_BINARY_DEBUGGER(result);
-
- ggdn_error:
-
- g_object_unref(G_OBJECT(result));
-
- return NULL;
-
-}
-
-
-
-
-
-
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à lancer. *
-* *
-* Description : Met en marche le débogueur utilisant un serveur GDB. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_gdb_debugger_run(GGdbDebugger *debugger)
-{
-
-
-
- GGdbPacket *packet;
-
- bool test;
-
- const char *data;
- size_t len;
-
-
- int sig;
- vmpa_t addr;
- pid_t thread;
-
-
- debugger->stream = g_gdb_tcp_client_new("127.0.0.1", "6666", NULL);
- if (debugger->stream == NULL) return false;
-
-
- printf("Connection done !\n");
-
-
-
- packet = g_gdb_stream_get_free_packet(debugger->stream);
-
- g_gdb_packet_start_new_command(packet);
- g_gdb_packet_append(packet, "?");
-
-
- test = g_gdb_stream_send_packet(debugger->stream, packet);
-
-
-
- printf(" >> Paquet '%s' bien envoyé ? %s\n", "?", test ? "oui" : "non");
-
-
-
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
-
- packet = g_gdb_stream_recv_packet(debugger->stream);
-
- g_gdb_packet_get_data(packet, &data, &len, NULL);
-
- printf(" << Réception de '%s'\n", data);
-
-
-
-
-
- get_stop_reply_sig_info(packet, &sig, &addr, &thread, SRE_LITTLE);
-
- g_signal_emit_by_name(debugger, "halted", sig, addr, thread);
-
-
-
- return true;
-
-}
-
-
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à relancer. *
-* *
-* Description : Tue le débogueur utilisant un serveur GDB. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_gdb_debugger_kill(GGdbDebugger *debugger)
-{
-
-
-#if 0
- int ret; /* Bilan de l'appel système */
-
- ret = kill(debugger->child, SIGKILL);
- if (ret != 0) perror("kill");
-
- debugger->child = 0;
-
- g_mutex_lock(debugger->mutex);
- debugger->run_again = TRUE;
- g_cond_signal(debugger->cond);
- g_mutex_unlock(debugger->mutex);
-#endif
- return true;
-
-}
-
-
-
-
-
-
-
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à consulter. *
-* *
-* Description : Détermine l'identifiant du thread principal courant. *
-* *
-* Retour : Identifiant du thread actif principal ou NULL en cas d'échec.*
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static char *g_gdb_debugger_get_active_thread(GGdbDebugger *debugger)
-{
- char *result; /* Identifiant à renvoyer */
- GGdbPacket *packet; /* Paquet de communication */
- bool status; /* Bilan d'une communication */
- const char *data; /* Données reçues à analyser */
- const char *start; /* Début d'identification */
- const char *end; /* Fin d'identification */
-
- result = NULL;
-
- /* Envoi de la requête */
-
- packet = g_gdb_stream_get_free_packet(debugger->stream);
-
- g_gdb_packet_start_new_command(packet);
- g_gdb_packet_append(packet, "?");
-
- status = g_gdb_stream_send_packet(debugger->stream, packet);
-
- if (!status)
- goto ggdgat_exit;
-
- /* Réception de la réponse */
-
- packet = g_gdb_stream_recv_packet(debugger->stream);
-
- g_gdb_packet_get_data(packet, &data, NULL, NULL);
-
- start = strstr(data, "thread:");
- if (start == NULL) goto ggdgat_exit;
-
- start += sizeof("thread:") - 1 /* '\0' */;
-
- end = strstr(start, ";");
- if (end == NULL) goto ggdgat_exit;
-
- result = strndup(start, end - start);
-
- ggdgat_exit:
-
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
-
- return result;
-
-}
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-/* ---------------------------------------------------------------------------------- */
-/* ENTREES / SORTIES BASIQUES */
-/* ---------------------------------------------------------------------------------- */
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à consulter. *
-* addr = emplacement en mémoire à venir consulter. *
-* size = taille des données mises en jeu. *
-* ... = emplacement de la valeur lue à conserver. [OUT] *
-* *
-* Description : Lit une valeur quelconque à une adresse arbitraire. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_gdb_debugger_read_memory(GGdbDebugger *debugger, virt_t addr, size_t size, ...)
-{
- bool result; /* Bilan d'opération à renvoyer*/
- char cmd[1 + VMPA_MAX_LEN + 3]; /* Commande à émettre */
- GGdbPacket *packet; /* Paquet de communication */
- const char *data; /* Données reçues à analyser */
- size_t len; /* Quantité de données reçues */
- va_list ap; /* Liste variable d'arguments */
- uint8_t *val8; /* Valeur sur 8 bits */
- uint16_t *val16; /* Valeur sur 16 bits */
- uint16_t conv16; /* Valeur adaptée sur 16 bits */
- uint32_t *val32; /* Valeur sur 32 bits */
- uint32_t conv32; /* Valeur adaptée sur 32 bits */
- uint64_t *val64; /* Valeur sur 64 bits */
- uint64_t conv64; /* Valeur adaptée sur 64 bits */
-
- /* Envoi de la requête */
-
- cmd[0] = 'm';
-
- result = translate_virt_to_hex(debugger, addr, &cmd[1]);
-
- switch (size)
- {
- case 8:
- strcat(cmd, ",1");
- break;
-
- case 16:
- strcat(cmd, ",2");
- break;
-
- case 32:
- strcat(cmd, ",4");
- break;
-
- case 64:
- strcat(cmd, ",8");
- break;
-
- default:
- assert(false);
- result = false;
- goto ggdrm_exit;
- break;
-
- }
-
- packet = g_gdb_stream_get_free_packet(debugger->stream);
-
- g_gdb_packet_start_new_command(packet);
- g_gdb_packet_append(packet, cmd);
-
- result = g_gdb_stream_send_packet(debugger->stream, packet);
-
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
-
- if (!result)
- goto ggdrm_exit;
-
- /* Réception de la réponse */
-
- packet = g_gdb_stream_recv_packet(debugger->stream);
-
- g_gdb_packet_get_data(packet, &data, &len, NULL);
-
- if (is_error_code(data, len))
- {
- result = false;
- goto ggdrm_error;
- }
-
- va_start(ap, size);
-
- switch (size)
- {
- case 8:
- val8 = va_arg(ap, uint8_t *);
- result = hex_to_u8(data, val8);
- break;
-
- case 16:
- val16 = va_arg(ap, uint16_t *);
- result = hex_to_u16(data, &conv16);
- *val16 = from_u16(&conv16, debugger->endian);
- break;
-
- case 32:
- val32 = va_arg(ap, uint32_t *);
- result = hex_to_u32(data, &conv32);
- *val32 = from_u32(&conv32, debugger->endian);
- break;
-
- case 64:
- val64 = va_arg(ap, uint64_t *);
- result = hex_to_u64(data, &conv64);
- *val64 = from_u64(&conv64, debugger->endian);
- break;
-
- default:
- assert(false);
- result = false;
- break;
-
- }
-
- va_end(ap);
-
- ggdrm_error:
-
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
-
- ggdrm_exit:
-
- return result;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à manipuler. *
-* addr = emplacement en mémoire à venir consulter. *
-* size = taille des données mises en jeu. *
-* ... = emplacement de la valeur lue à conserver. [OUT] *
-* *
-* Description : Lit une valeur quelconque à une adresse arbitraire. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_gdb_debugger_write_memory(GGdbDebugger *debugger, virt_t addr, size_t size, ...)
-{
- bool result; /* Bilan d'opération à renvoyer*/
- char cmd[1 + 3 * VMPA_MAX_LEN + 3]; /* Commande à émettre */
- va_list ap; /* Liste variable d'arguments */
- const uint8_t *val8; /* Valeur sur 8 bits */
- const uint16_t *val16; /* Valeur sur 16 bits */
- uint16_t conv16; /* Valeur adaptée sur 16 bits */
- const uint32_t *val32; /* Valeur sur 32 bits */
- uint32_t conv32; /* Valeur adaptée sur 32 bits */
- const uint64_t *val64; /* Valeur sur 64 bits */
- uint64_t conv64; /* Valeur adaptée sur 64 bits */
- char hexval[17]; /* Valeur sous forme hexa */
- GGdbPacket *packet; /* Paquet de communication */
- const char *data; /* Données reçues à analyser */
- size_t len; /* Quantité de données reçues */
-
- /* Envoi de la requête */
-
- cmd[0] = 'M';
-
- result = translate_virt_to_hex(debugger, addr, &cmd[1]);
-
- va_start(ap, size);
-
- switch (size)
- {
- case 8:
- val8 = va_arg(ap, uint8_t *);
- result = u8_to_hex(val8, hexval);
-
- strcat(cmd, ",1:");
- strcat(cmd, hexval);
- break;
-
- case 16:
- val16 = va_arg(ap, uint16_t *);
- conv16 = to_u16(val16, debugger->endian);
- result = u16_to_hex(&conv16, hexval);
-
- strcat(cmd, ",2:");
- strcat(cmd, hexval);
- break;
-
- case 32:
- val32 = va_arg(ap, uint32_t *);
- conv32 = to_u32(val32, debugger->endian);
- result = u32_to_hex(&conv32, hexval);
-
- strcat(cmd, ",4:");
- strcat(cmd, hexval);
- break;
-
- case 64:
- val64 = va_arg(ap, uint64_t *);
- conv64 = to_u64(val64, debugger->endian);
- result = u16_to_hex(&conv64, hexval);
-
- strcat(cmd, ",8:");
- strcat(cmd, hexval);
- break;
-
- default:
- assert(false);
- result = false;
- break;
-
- }
-
- if (!result)
- goto ggdwm_exit;
-
- packet = g_gdb_stream_get_free_packet(debugger->stream);
-
- g_gdb_packet_start_new_command(packet);
- g_gdb_packet_append(packet, cmd);
-
- result = g_gdb_stream_send_packet(debugger->stream, packet);
-
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
-
- if (!result)
- goto ggdwm_exit;
-
- /* Réception de la réponse */
-
- packet = g_gdb_stream_recv_packet(debugger->stream);
-
- g_gdb_packet_get_data(packet, &data, &len, NULL);
-
- if (len == 3 && data[0] == 'E')
- {
- result = false;
- goto ggdrm_error;
- }
-
- ggdrm_error:
-
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
-
- ggdwm_exit:
-
- va_end(ap);
-
- return result;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à consulter. *
-* group = éventuel groupe de registres ciblé ou NULL. *
-* count = nombre d'éléments dans la liste de noms. [OUT] *
-* *
-* Description : Liste l'ensemble des registres appartenant à un groupe. *
-* *
-* Retour : Liste de noms à libérer de la mémoire après utilisation. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static char **g_gdb_debugger_get_register_names(const GGdbDebugger *debugger, const char *group, size_t *count)
-{
- return g_gdb_target_get_register_names(debugger->target, group, count);
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à consulter. *
-* name = désignation du registre visé. *
-* *
-* Description : Indique la taille associée à un registre donné. *
-* *
-* Retour : Taille en bits, ou 0 si le registre n'a pas été trouvé. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static unsigned int g_gdb_debugger_get_register_size(const GGdbDebugger *debugger, const char *name)
-{
- return g_gdb_target_get_register_size(debugger->target, name);
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à consulter. *
-* reg = désignation humaine du register à consulter. *
-* size = taille des données mises en jeu. *
-* ... = emplacement de la valeur lue à conserver. [OUT] *
-* *
-* Description : Effectue la lecture d'un registre donné. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_gdb_debugger_read_register(GGdbDebugger *debugger, const char *reg, size_t size, ...)
-{
- bool result; /* Bilan d'opération à renvoyer*/
- va_list ap; /* Liste variable d'arguments */
- uint8_t *val8; /* Valeur sur 8 bits */
- uint16_t *val16; /* Valeur sur 16 bits */
- uint32_t *val32; /* Valeur sur 32 bits */
- uint64_t *val64; /* Valeur sur 64 bits */
-
- va_start(ap, size);
-
- switch (size)
- {
- case 8:
- val8 = va_arg(ap, uint8_t *);
- result = g_gdb_target_read_register(debugger->target, debugger->stream, debugger->endian,
- reg, 8, val8);
- break;
-
- case 16:
- val16 = va_arg(ap, uint16_t *);
- result = g_gdb_target_read_register(debugger->target, debugger->stream, debugger->endian,
- reg, 16, val16);
- break;
-
- case 32:
- val32 = va_arg(ap, uint32_t *);
- result = g_gdb_target_read_register(debugger->target, debugger->stream, debugger->endian,
- reg, 32, val32);
- break;
-
- case 64:
- val64 = va_arg(ap, uint64_t *);
- result = g_gdb_target_read_register(debugger->target, debugger->stream, debugger->endian,
- reg, 64, val64);
- break;
-
- default:
- assert(false);
- result = false;
- break;
-
- }
-
- va_end(ap);
-
- return result;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à manipuler. *
-* reg = désignation humaine du register à consulter. *
-* size = taille des données mises en jeu. *
-* ... = emplacement de la valeur à écrire. *
-* *
-* Description : Effectue l'écriture d'un registre donné. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_gdb_debugger_write_register(GGdbDebugger *debugger, const char *reg, size_t size, ...)
-{
- bool result; /* Bilan d'opération à renvoyer*/
- va_list ap; /* Liste variable d'arguments */
- const uint8_t *val8; /* Valeur sur 8 bits */
- const uint16_t *val16; /* Valeur sur 16 bits */
- const uint32_t *val32; /* Valeur sur 32 bits */
- const uint64_t *val64; /* Valeur sur 64 bits */
-
- va_start(ap, size);
-
- switch (size)
- {
- case 8:
- val8 = va_arg(ap, const uint8_t *);
- result = g_gdb_target_write_register(debugger->target, debugger->stream, debugger->endian,
- reg, 8, val8);
- break;
-
- case 16:
- val16 = va_arg(ap, const uint16_t *);
- result = g_gdb_target_write_register(debugger->target, debugger->stream, debugger->endian,
- reg, 16, val16);
- break;
-
- case 32:
- val32 = va_arg(ap, const uint32_t *);
- result = g_gdb_target_write_register(debugger->target, debugger->stream, debugger->endian,
- reg, 32, val32);
- break;
-
- case 64:
- val64 = va_arg(ap, const uint64_t *);
- result = g_gdb_target_write_register(debugger->target, debugger->stream, debugger->endian,
- reg, 64, val64);
- break;
-
- default:
- assert(false);
- result = false;
- break;
-
- }
-
- va_end(ap);
-
- return result;
-
-}
-
-
-
-/* ---------------------------------------------------------------------------------- */
-/* MANIPULATION DE L'ETAT COURANT */
-/* ---------------------------------------------------------------------------------- */
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à consulter. *
-* pc = adresse de l'instruction courante. [OUT] *
-* *
-* Description : Détermine le point d'exécution courant. *
-* *
-* Retour : Bilan de la récupération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_gdb_debugger_get_current_pc(GGdbDebugger *debugger, virt_t *pc)
-{
- bool result; /* Bilan à retourner */
-
- result = debugger->ops->get_pc(debugger, pc);
-
- return result;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à consulter. *
-* callstack = pile d'appels reconstituée. [OUT] *
-* size = taille de cette pile. [OUT] *
-* *
-* Description : Remonte la pile d'appels jusqu'au point courant. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_gdb_debugger_compute_call_stack(GGdbDebugger *debugger, virt_t **callstack, size_t *size)
-{
- bool result; /* Bilan global à retourner */
-
- if (debugger->ops->compute_cstack != NULL)
- result = debugger->ops->compute_cstack(debugger, callstack, size);
-
- else
- result = false;
-
- return result;
-
-}
-
-
-
-/* ---------------------------------------------------------------------------------- */
-/* GESTION DES POINTS D'ARRET */
-/* ---------------------------------------------------------------------------------- */
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à manipuler ici. *
-* addr = emplacement du point mémoire à traiter. *
-* *
-* Description : Ajoute un point d'arrêt basique en mémoire. *
-* *
-* Retour : Structure de suivi mise en place pour l'occasion, voire NULL.*
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static gdb_breakpoint *g_gdb_debugger_enable_memory_breakpoint(GGdbDebugger *debugger, virt_t addr)
-{
- gdb_breakpoint *result; /* Nouveau suivi à retourner */
- char cmd[3 + VMPA_MAX_LEN + 3]; /* Commande à émettre */
- bool status; /* Bilan d'une opération */
- const char *kind; /* Taille spécifique du point */
- GGdbPacket *packet; /* Paquet de communication */
- const char *data; /* Données reçues à analyser */
- size_t len; /* Quantité de données reçues */
- GBinaryDebugger *dbg; /* Autre version du débogueur */
- const uint8_t *bp; /* Données du point d'arrêt */
- size_t bp_len; /* Quantité de ces données */
- uint8_t memory[16]; /* Sauvegarde de la mémoire */
-
- result = NULL;
-
- /* Si l'utilisation de la commande dédiée est possible */
- if (1) //////// TODO
- {
- /* Envoi de la requête */
-
- strcpy(cmd, "Z0,");
-
- status = translate_virt_to_hex(debugger, addr, &cmd[3]);
-
- if (!status)
- goto ggdemb_exit;
-
- kind = debugger->ops->get_bp_kind(debugger, addr);
-
- if (kind == NULL)
- goto ggdemb_exit;
-
- strcat(cmd, kind);
-
- packet = g_gdb_stream_get_free_packet(debugger->stream);
-
- g_gdb_packet_start_new_command(packet);
- g_gdb_packet_append(packet, cmd);
-
- status = g_gdb_stream_send_packet(debugger->stream, packet);
-
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
-
- if (!status)
- goto ggdemb_exit;
-
- /* Réception de la réponse */
-
- packet = g_gdb_stream_recv_packet(debugger->stream);
-
- g_gdb_packet_get_data(packet, &data, &len, NULL);
-
- if (is_error_code(data, len))
- {
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
- goto ggdemb_fallback;
- }
-
- if (strcmp(data, "OK") != 0)
- {
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
- goto ggdemb_fallback;
- }
-
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
-
- /* Constitution d'un dossier de suivi */
-
- result = (gdb_breakpoint *)malloc(sizeof(gdb_breakpoint));
-
- result->is_z = true;
-
- result->kind = kind;
-
- }
-
- else
- {
-
- ggdemb_fallback:
-
- dbg = G_BINARY_DEBUGGER(debugger);
-
- /* Détermination du point d'arrêt */
-
- bp = debugger->ops->get_bp_data(debugger, addr, &bp_len);
-
- assert(bp_len <= 16);
-
- /* Sauvegarde de la mémoire courante */
-
- status = g_binary_debugger_read_memory_data(dbg, addr, memory, bp_len);
-
- if (!status) goto ggdemb_exit;
-
- /* Application du point d'arrêt */
-
- status = g_binary_debugger_write_memory_data(dbg, addr, bp, bp_len);
-
- if (!status) goto ggdemb_exit;
-
- /* Constitution d'un dossier de suivi */
-
- result = (gdb_breakpoint *)malloc(sizeof(gdb_breakpoint));
-
- result->is_z = false;
-
- memcpy(result->memory, memory, bp_len);
- result->len = bp_len;
-
- }
-
- init_raw_breakpoint((raw_breakpoint *)result, addr);
-
- ggdemb_exit:
-
- return result;
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à manipuler ici. *
-* bp = point d'arrêt à traiter. *
-* *
-* Description : Retire un point d'arrêt basique de la mémoire ciblée. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_gdb_debugger_disable_memory_breakpoint(GGdbDebugger *debugger, gdb_breakpoint *bp)
-{
- bool result; /* Bilan à retourner */
- char cmd[3 + VMPA_MAX_LEN + 3]; /* Commande à émettre */
- bool status; /* Bilan d'une opération */
- GGdbPacket *packet; /* Paquet de communication */
- const char *data; /* Données reçues à analyser */
- size_t len; /* Quantité de données reçues */
- GBinaryDebugger *dbg; /* Autre version du débogueur */
-
- result = false;
-
- /* Si l'utilisation de la commande dédiée est requise */
- if (bp->is_z)
- {
- /* Envoi de la requête */
-
- strcpy(cmd, "z0,");
-
- status = translate_virt_to_hex(debugger, bp->raw.addr, &cmd[3]);
-
- if (!status)
- goto ggddmb_exit;
-
- strcat(cmd, bp->kind);
-
- packet = g_gdb_stream_get_free_packet(debugger->stream);
-
- g_gdb_packet_start_new_command(packet);
- g_gdb_packet_append(packet, cmd);
-
- status = g_gdb_stream_send_packet(debugger->stream, packet);
-
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
-
- if (!status)
- goto ggddmb_exit;
-
- /* Réception de la réponse */
-
- packet = g_gdb_stream_recv_packet(debugger->stream);
-
- g_gdb_packet_get_data(packet, &data, &len, NULL);
-
- if (is_error_code(data, len))
- {
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
- goto ggddmb_exit;
- }
-
- if (strcmp(data, "OK") != 0)
- {
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
- goto ggddmb_exit;
- }
-
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
-
- result = true;
-
- }
-
- else
- {
- dbg = G_BINARY_DEBUGGER(debugger);
-
- result = g_binary_debugger_write_memory_data(dbg, bp->raw.addr, bp->memory, bp->len);
-
- }
-
- ggddmb_exit:
-
- return result;
-
-}
-
-
-
-/* ---------------------------------------------------------------------------------- */
-/* CONTROLE DU FLOT D'EXECUTION */
-/* ---------------------------------------------------------------------------------- */
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à redémarrer. *
-* *
-* Description : Redémarre le processus de débogage lié à un serveur GDB. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_gdb_debugger_restart(GGdbDebugger *debugger)
-{
- bool result; /* Bilan à retourner */
- GGdbPacket *packet; /* Paquet de communication */
- return true;
- /* Envoi de la requête */
-
- packet = g_gdb_stream_get_free_packet(debugger->stream);
-
- g_gdb_packet_start_new_command(packet);
- g_gdb_packet_append(packet, "R00");
-
- result = g_gdb_stream_send_packet(debugger->stream, packet);
-
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
-
- return result;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = débogueur à relancer. *
-* *
-* Description : Remet en marche le débogueur utilisant un serveur GDB. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_gdb_debugger_resume(GGdbDebugger *debugger)
-{
- bool result; /* Bilan à retourner */
- //char *id; /* Identifiant de thread */
- GGdbPacket *packet; /* Paquet de communication */
- //const char *data; /* Données reçues à analyser */
-
- static bool _twice = false;
-
-
- if (!_twice && 0)
- {
-
- packet = g_gdb_stream_get_free_packet(debugger->stream);
-
- g_gdb_packet_start_new_command(packet);
- g_gdb_packet_append(packet, "$QPassSignals:e;10;14;17;1a;1b;1c;21;24;25;2c;4c;");
-
- result = g_gdb_stream_send_packet(debugger->stream, packet);
-
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
-
- if (!result)
- goto ggdhmb_exit;
-
- }
-
-
-
-
-
-
- /* Envoi de la requête */
-
- /*
- id = g_gdb_debugger_get_active_thread(debugger);
- if (id == NULL) return false;
-
- printf("ID : %s\n", id);
- */
-
- /*
- id = g_gdb_support_get_id(debugger->support);
- if (id == NULL) return false;
-
- printf("ID : %s\n", id);
- */
-
- packet = g_gdb_stream_get_free_packet(debugger->stream);
-
- g_gdb_packet_start_new_command(packet);
- g_gdb_packet_append(packet, "vCont;c:-1");
- //g_gdb_packet_append(packet, "vCont;c:p256f.-1");
-
-
- /*
- if (_twice)
- {
- g_gdb_packet_start_new_command(packet);
- g_gdb_packet_append(packet, "vCont;c:p");
- g_gdb_packet_append(packet, id);
- g_gdb_packet_append(packet, ".");
- g_gdb_packet_append(packet, id);
- }
- else
- {
- _twice = true;
- g_gdb_packet_start_new_command(packet);
- g_gdb_packet_append(packet, "vCont;c:p");
- g_gdb_packet_append(packet, id);
- g_gdb_packet_append(packet, ".-1");
- }
- */
-
-
-
-
-
- result = g_gdb_stream_send_packet(debugger->stream, packet);
-
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
-
- if (!result)
- goto ggdhmb_exit;
-
- /* Réception de la réponse */
- /*
- packet = g_gdb_stream_recv_packet(debugger->stream);
-
- g_gdb_packet_get_data(packet, &data, NULL, NULL);
-
- printf("Ack cont...\n");
-
- //result = (strcmp(data, "OK") == 0);
-
- g_gdb_stream_mark_packet_as_free(debugger->stream, packet);
- */
- ggdhmb_exit:
-
- _twice = true;
-
- return result;
-
-}
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-/* ---------------------------------------------------------------------------------- */
-/* ACCUEIL D'EVENEMENTS ASYNCHRONES */
-/* ---------------------------------------------------------------------------------- */
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = instance liée à un débogueur GDB à manipuler. *
-* signum = indentifiant du signal concerné. *
-* *
-* Description : Réagit à la réception d'un signal par le programme étudié. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-void g_gdb_debugger_receive_signal_reply(GGdbDebugger *debugger, int signum)
-{
- virt_t pc; /* Position courante du CPU */
- bool status; /* Bilan d'une opération */
- GBinaryDebugger *base; /* Version basique du débogueur*/
-
- base = G_BINARY_DEBUGGER(debugger);
-
- status = g_binary_debugger_get_current_pc(base, &pc);
-
- if (!status)
- pc = VMPA_NO_VIRTUAL;
-
- on_binary_debugger_stopped(base, pc);
-
- g_signal_emit_by_name(debugger, "signaled", signum);
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : debugger = instance liée à un débogueur GDB à manipuler. *
-* status = indication d'état à la sortie. *
-* pid = éventuel identifiant de processus concerné ou -1. *
-* *
-* Description : Réagit à la sortie d'exécution d'un programme étudié. *
-* *
-* Retour : - *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-void g_gdb_debugger_receive_exit_reply(GGdbDebugger *debugger, int status, pid_t pid)
-{
- GBinaryDebugger *base; /* Version basique du débogueur*/
-
- base = G_BINARY_DEBUGGER(debugger);
-
- on_binary_debugger_finished(base, pid);
-
- g_signal_emit_by_name(debugger, "exited", status, pid);
-
-}