/* Chrysalide - Outil d'analyse de fichiers binaires
 * loading.c - chargements parallèles des symboles de format ELF
 *
 * Copyright (C) 2016-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 Foobar.  If not, see .
 */
#include "loading.h"
#include 
#include "elf-int.h"
#include "section.h"
#include "../../glibext/delayed-int.h"
#include "../../gui/panels/log.h"
/* Fraction de routines à limiter (instance) */
struct _GElfLoading
{
    GDelayedWork parent;                    /* A laisser en premier        */
    GElfFormat *format;                     /* Format à faire évoluer      */
    union
    {
        struct
        {
            const elf_shdr *section;        /* Section à éplucher          */
            bool use_virt;                  /* Représentatio par défaut    */
            elf_shdr strtab;                /* Section .strtab trouvée     */
            bool has_strtab;                /* Présence de cette section   */
        };
    };
    elf_loading_cb callback;                /* Routine de traitement finale*/
    phys_t first;                           /* Position du premier élément */
    phys_t begin;                           /* Point de départ du parcours */
    phys_t end;                             /* Point d'arrivée exclu       */
    activity_id_t id;                       /* Identifiant pour messages   */
};
/* Fraction de routines à limiter (classe) */
struct _GElfLoadingClass
{
    GDelayedWorkClass parent;               /* A laisser en premier        */
};
/* Initialise la classe des tâches des chargements pour ELF. */
static void g_elf_loading_class_init(GElfLoadingClass *);
/* Initialise une tâche de chargements pour ELF. */
static void g_elf_loading_init(GElfLoading *);
/* Supprime toutes les références externes. */
static void g_elf_loading_dispose(GElfLoading *);
/* Procède à la libération totale de la mémoire. */
static void g_elf_loading_finalize(GElfLoading *);
/* Assure le chargement pour un format ELF en différé. */
static void g_elf_loading_process(GElfLoading *, GtkStatusStack *);
/* Indique le type défini pour les tâches de chargements pour format ELF. */
G_DEFINE_TYPE(GElfLoading, g_elf_loading, G_TYPE_DELAYED_WORK);
/******************************************************************************
*                                                                             *
*  Paramètres  : klass = classe à initialiser.                                *
*                                                                             *
*  Description : Initialise la classe des tâches des chargements pour ELF.    *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/
static void g_elf_loading_class_init(GElfLoadingClass *klass)
{
    GObjectClass *object;                   /* Autre version de la classe  */
    GDelayedWorkClass *work;                /* Version en classe parente   */
    object = G_OBJECT_CLASS(klass);
    object->dispose = (GObjectFinalizeFunc/* ! */)g_elf_loading_dispose;
    object->finalize = (GObjectFinalizeFunc)g_elf_loading_finalize;
    work = G_DELAYED_WORK_CLASS(klass);
    work->run = (run_task_fc)g_elf_loading_process;
}
/******************************************************************************
*                                                                             *
*  Paramètres  : loading = instance à initialiser.                            *
*                                                                             *
*  Description : Initialise une tâche de chargements pour ELF.                *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/
static void g_elf_loading_init(GElfLoading *loading)
{
}
/******************************************************************************
*                                                                             *
*  Paramètres  : loading = instance d'objet GLib à traiter.                   *
*                                                                             *
*  Description : Supprime toutes les références externes.                     *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/
static void g_elf_loading_dispose(GElfLoading *loading)
{
    G_OBJECT_CLASS(g_elf_loading_parent_class)->dispose(G_OBJECT(loading));
}
/******************************************************************************
*                                                                             *
*  Paramètres  : loading = instance d'objet GLib à traiter.                   *
*                                                                             *
*  Description : Procède à la libération totale de la mémoire.                *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/
static void g_elf_loading_finalize(GElfLoading *loading)
{
    G_OBJECT_CLASS(g_elf_loading_parent_class)->finalize(G_OBJECT(loading));
}
/******************************************************************************
*                                                                             *
*  Paramètres  : format = = ensemble d'instructions désassemblées.            *
*                section  = prototypes existants à insérer.                   *
*                use_virt = quantité de ces prototypes.                       *
*                first    = position du premier élément.                      *
*                begin    = point de départ du parcours de liste.             *
*                end      = point d'arrivée exclu du parcours.                *
*                id       = identifiant du message affiché à l'utilisateur.   *
*                callback = routine de traitements particuliers.              *
*                                                                             *
*  Description : Crée une tâche de chargement pour ELF différée.              *
*                                                                             *
*  Retour      : Tâche créée.                                                 *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/
GElfLoading *g_elf_loading_new(GElfFormat *format, const elf_shdr *section, bool use_virt, phys_t first, phys_t begin, phys_t end, activity_id_t id, elf_loading_cb callback)
{
    GElfLoading *result;                    /* Tâche à retourner           */
    result = g_object_new(G_TYPE_ELF_LOADING, NULL);
    result->format = format;
    result->section = section;
    result->use_virt = use_virt;
    result->has_strtab = find_elf_section_by_index(format,
                                                   ELF_SHDR(format, *section, sh_link),
                                                   &result->strtab);
    result->callback = callback;
    result->first = first;
    result->begin = begin;
    result->end = end;
    result->id = id;
    return result;
}
/******************************************************************************
*                                                                             *
*  Paramètres  : study  = étude de routines à mener.                          *
*                status = barre de statut à tenir informée.                   *
*                                                                             *
*  Description : Assure le chargement pour un format ELF en différé.          *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/
static void g_elf_loading_process(GElfLoading *loading, GtkStatusStack *status)
{
    phys_t iter;                            /* Boucle de parcours          */
    phys_t old;                             /* Sauvegarde du point d'avant */
    bool ret;                               /* Bilan d'un appel            */
    for (iter = loading->begin; iter < loading->end; )
    {
        old = iter;
        ret = loading->callback(loading, loading->format, &iter);
        if (!ret)
        {
            log_variadic_message(LMT_ERROR, _("Error while loading ELF data @ 0x%08x!"), old);
            break;
        }
        gtk_status_stack_update_activity_value(status, loading->id, 1);
    }
}
/******************************************************************************
*                                                                             *
*  Paramètres  : loading    = chargement pour ELF à mener.                    *
*                section    = prototypes existants à insérer. [OUT]           *
*                use_virt   = quantité de ces prototypes. [OUT]               *
*                strtab     = informations quant à la table des chaînes. [OUT]*
*                has_strtab = validité du champ précédemment renseigné. [OUT] *
*                first      = position du premier élément. [OUT]              *
*                                                                             *
*  Description : Fournit les infos utiles au chargement de symbols internes.  *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/
void g_elf_loading_get_internal_info(GElfLoading *loading, const elf_shdr **section, bool *use_virt, const elf_shdr **strtab, bool *has_strtab, phys_t *first)
{
    *section = loading->section;
    *use_virt = loading->use_virt;
    *strtab = &loading->strtab;
    *has_strtab = loading->has_strtab;
    *first = loading->first;
}