/* Chrysalide - Outil d'analyse de fichiers binaires
* utils.h - prototypes pour les fonctions d'aisance vis à vis du format DWARF
*
* Copyright (C) 2008-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 "utils.h"
#include
#include "dwarf-int.h"
/******************************************************************************
* *
* Paramètres : format = informations de débogage à consulter. *
* pos = tête de lecture à mettre à jour. [OUT] *
* value = valeur au format LEB128 lue. [OUT] *
* update = indique si la position est à mettre à jour. *
* *
* Description : Lit une valeur Little Endian Base 128 signée. *
* *
* Retour : Bilan de l'opération. *
* *
* Remarques : En cas d'échec, la tête de lecture est indéterminée. *
* *
******************************************************************************/
bool read_leb128(dwarf_format *format, off_t *pos, int64_t *value, bool update)
{
off_t curpos; /* Tête de lecture effective */
int shift; /* Décalage à appliquer */
off_t i; /* Boucle de parcours */
curpos = *pos;
shift = 0;
*value = 0;
for (i = 0; i < 8; i++)
{
/* On évite les débordements... */
if (curpos >= DBG_FORMAT(format)->length) return false;
*value |= (DBG_FORMAT(format)->content[curpos] & 0x7f) << shift;
shift += 7;
curpos++;
if ((DBG_FORMAT(format)->content[*pos + i] & 0x80) == 0x00) break;
}
if ((shift < 64) && (DBG_FORMAT(format)->content[curpos - 1] & 0x40) == 0x40)
*value |= - (1 << shift);
if (update) *pos = curpos;
return (i < 8);
}
/******************************************************************************
* *
* Paramètres : format = informations de débogage à consulter. *
* pos = tête de lecture à mettre à jour. [OUT] *
* value = valeur au format LEB128 lue. [OUT] *
* update = indique si la position est à mettre à jour. *
* *
* Description : Lit une valeur Little Endian Base 128 non signée. *
* *
* Retour : Bilan de l'opération. *
* *
* Remarques : En cas d'échec, la tête de lecture est indéterminée. *
* *
******************************************************************************/
bool read_uleb128(dwarf_format *format, off_t *pos, uint64_t *value, bool update)
{
off_t curpos; /* Tête de lecture effective */
int shift; /* Décalage à appliquer */
off_t i; /* Boucle de parcours */
curpos = *pos;
shift = 0;
*value = 0;
for (i = 0; i < 8; i++)
{
/* On évite les débordements... */
if (curpos >= DBG_FORMAT(format)->length) return false;
*value |= (DBG_FORMAT(format)->content[curpos] & 0x7f) << shift;
shift += 7;
curpos++;
if ((DBG_FORMAT(format)->content[*pos + i] & 0x80) == 0x00) break;
}
if (update) *pos = curpos;
return (i < 8);
}
/******************************************************************************
* *
* Paramètres : format = informations de débogage à consulter. *
* pos = tête de lecture à mettre à jour. [OUT] *
* value = valeur entière lue. [OUT] *
* *
* Description : Lit une valeur représentant une longueur d'unité. *
* *
* Retour : Bilan de l'opération. *
* *
* Remarques : Un peu sale : la sortie est signée et dépend du système, *
* alors que la valeur est non signée et dépend de la cible. *
* *
******************************************************************************/
bool read_unit_length(dwarf_format *format, off_t *pos, off_t *value)
{
bool result; /* Bilan à retourner */
uint32_t val32; /* Entier sur 4 octets */
uint64_t val64; /* Entier sur 8 octets */
/* FIXME : Endian... */
if (format->format == DWF_32_BITS)
{
result = ((*pos + 4) <= DBG_FORMAT(format)->length);
if (result)
{
memcpy(&val32, &DBG_FORMAT(format)->content[*pos], 4);
(*pos) += 4;
*value = val32;
}
}
else
{
result = ((*pos + 4 + 8) <= DBG_FORMAT(format)->length);
if (result)
{
memcpy(&val64, &DBG_FORMAT(format)->content[*pos + 4], 8);
(*pos) += 4 + 8;
*value = val64;
}
}
return result;
}
/******************************************************************************
* *
* Paramètres : format = informations de débogage à consulter. *
* pos = tête de lecture à mettre à jour. [OUT] *
* value = valeur entière non signée lue. [OUT] *
* *
* Description : Lit une valeur non signée sur deux octets. *
* *
* Retour : Bilan de l'opération. *
* *
* Remarques : - *
* *
******************************************************************************/
bool read_uhalf(dwarf_format *format, off_t *pos, uint16_t *value)
{
bool result; /* Bilan à retourner */
/* FIXME : Endian... */
result = ((*pos + 2) <= DBG_FORMAT(format)->length);
if (result)
{
memcpy(value, &DBG_FORMAT(format)->content[*pos], 2);
(*pos) += 2;
}
return result;
}
/******************************************************************************
* *
* Paramètres : format = informations de débogage à consulter. *
* pos = tête de lecture à mettre à jour. [OUT] *
* value = valeur entière lue. [OUT] *
* *
* Description : Lit une valeur indiquant une position dans les abréviations. *
* *
* Retour : Bilan de l'opération. *
* *
* Remarques : Un peu sale : la sortie est signée et dépend du système, *
* alors que la valeur est non signée et dépend de la cible. *
* *
******************************************************************************/
bool read_abbrev_offset(dwarf_format *format, off_t *pos, off_t *value)
{
bool result; /* Bilan à retourner */
uint32_t val32; /* Entier sur 4 octets */
uint64_t val64; /* Entier sur 8 octets */
/* FIXME : Endian... */
if (format->format == DWF_32_BITS)
{
result = ((*pos + 4) <= DBG_FORMAT(format)->length);
if (result)
{
memcpy(&val32, &DBG_FORMAT(format)->content[*pos], 4);
(*pos) += 4;
*value = val32;
}
}
else
{
result = ((*pos + 8) <= DBG_FORMAT(format)->length);
if (result)
{
memcpy(&val64, &DBG_FORMAT(format)->content[*pos], 8);
(*pos) += 8;
*value = val64;
}
}
return result;
}
/******************************************************************************
* *
* Paramètres : format = informations de débogage à consulter. *
* pos = tête de lecture à mettre à jour. [OUT] *
* value = valeur entière non signée lue. [OUT] *
* *
* Description : Lit une valeur indiquant la taille des adresses mémoire. *
* *
* Retour : Bilan de l'opération. *
* *
* Remarques : - *
* *
******************************************************************************/
bool read_address_size(dwarf_format *format, off_t *pos, uint8_t *value)
{
bool result; /* Bilan à retourner */
result = ((*pos + 1) <= DBG_FORMAT(format)->length);
if (result)
{
*value = DBG_FORMAT(format)->content[*pos];
(*pos)++;
}
return result;
}