/*
  Copyright 2018-2018 David Anderson.  All Rights Reserved.

  This program is free software; you can redistribute it
  and/or modify it under the terms of version 2.1 of the
  GNU Lesser General Public License as published by the Free
  Software Foundation.

  This program is distributed in the hope that it would be
  useful, but WITHOUT ANY WARRANTY; without even the implied
  warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
  PURPOSE.

  Further, this software is distributed without any warranty
  that it is free of the rightful claim of any third person
  regarding infringement or the like.  Any license provided
  herein, whether implied or otherwise, applies only to this
  software file.  Patent licenses, if any, provided herein
  do not apply to combinations of this program with other
  software, or any other product whatsoever.

  You should have received a copy of the GNU Lesser General
  Public License along with this program; if not, write the
  Free Software Foundation, Inc., 51 Franklin Street - Fifth
  Floor, Boston MA 02110-1301, USA.

*/

#include <config.h>

#include <stddef.h> /* NULL */
#include <stdio.h>  /* printf() */
#include <string.h> /* memcpy() */

#ifdef HAVE_STDINT_H
#include <stdint.h> /* uintptr_t */
#endif

#include "dwarf.h"
#include "libdwarf.h"
#include "dwarf_base_types.h"
#include "libdwarfp.h"
#include "dwarf_pro_incl.h"
#include "dwarf_pro_opaque.h"
#include "dwarf_pro_error.h"
#include "dwarf_pro_alloc.h"
#include "dwarf_pro_arange.h"
#include "dwarf_pro_section.h"
#include "dwarf_pro_reloc.h"
#include "dwarf_pro_dnames.h"

static struct Dwarf_P_Dnames_Head_s staticdnames =
{
/* fake unit length*/58,
/* version */5,
/*offset size*/4,
/*offset*/0,
/* cu count */1,
/* tu ct */0,
/* foreigntu ct */0,
/* bucket ct */0,
/* name ct */1,
/* abbrev_table_size*/7,
/* augstringsize*/0,
/* augstring*/0
};

/* total length encoded: 6 bytes. */
unsigned char abbrv[7] =
{/* abbrev code*/ 2,
/* DW_TAG_subprogram */ 0x2e,
/* DW_IDX_compile_unit*/1,
/* DW_FORM_udata */7,
/* end abbrev */ 0,0,
/* end abbrev block */0};

unsigned char entry[3] =
{ /* abbrev code */2,
/* Offset of subprogram */0x32,
/* terminate this abbrev set */0 };

Dwarf_Unsigned stroffsets[1] = {0x42}; /* main */

/* Set to match subprog */
Dwarf_Unsigned dieoffset[1] = {12};

int
dwarf_force_dnames(Dwarf_P_Debug dbg,
    int elfsectno,
    Dwarf_Error * error)
{
    Dwarf_P_Dnames dn;
    Dwarf_Unsigned datalen = 0;
    Dwarf_Unsigned withunitlength = 0;
    unsigned char *data = 0;
    unsigned char *startdata = 0;
    unsigned int zero = 0;
    struct Dwarf_P_Dnames_Head_s *dh = &staticdnames;
    uintptr_t bytes = 0;

    if (dbg == NULL) {
        _dwarf_p_error(NULL, error, DW_DLE_DBG_NULL);
        return DW_DLV_ERROR;
    }
    if (!elfsectno) {
        dbg->de_force_dnames = TRUE;
        dn = (Dwarf_P_Dnames)
            _dwarf_p_get_alloc(dbg, sizeof(struct Dwarf_P_Dnames_s));
        if (dn == NULL) {
            _dwarf_p_error(dbg, error, DW_DLE_ALLOC_FAIL);
            return DW_DLV_ERROR;
        }
        dbg->de_dnames = dn;
        return DW_DLV_OK;
    }
    dn = dbg->de_dnames;
    if (!dbg->de_dnames) {
        return DW_DLV_NO_ENTRY;
    }
    dn->dn_dbg = dbg;
    dn->dn_create_section = TRUE;
    datalen = 8*4 +
        /* offset CU */
        dh->dh_offset_size  +
        /* str offsets, entry offsets */
        dh->dh_offset_size *2 +
        sizeof(abbrv) +
        sizeof(entry);
    withunitlength = datalen + 4;/* 4 byte length, 32 bit offset */

    GET_CHUNK(dbg, dbg->de_elf_sects[DEBUG_NAMES],
        data, (unsigned long)withunitlength, error);
    startdata = data;

    /*WRITE_UNALIGNED(dbg,dest,source, srclength,len_out)*/
    /* 1 */
    WRITE_UNALIGNED(dbg, (void *)data,
        (const void *)&datalen,
        sizeof(datalen) , SIZEOFT32);
    data += SIZEOFT32;
    /* 2 */
    WRITE_UNALIGNED(dbg, (void *)data,
        (const void *)&dh->dh_version,
        sizeof(dh->dh_version),
        SIZEOFT16);
    data += SIZEOFT16;

    /* 3 */
    WRITE_UNALIGNED(dbg, (void *)data,
        (const void *)&zero,sizeof(zero),
        SIZEOFT16);
    data += SIZEOFT16;

    /* 4 */
    WRITE_UNALIGNED(dbg, (void *)data,
        (const void *)&dh->dh_comp_unit_count,
        sizeof(dh->dh_comp_unit_count),
        SIZEOFT32);
    data += SIZEOFT32;
    /* 5 */
    WRITE_UNALIGNED(dbg, (void *)data,
        (const void *)&dh->dh_local_type_unit_count,
        sizeof(dh->dh_local_type_unit_count),
        SIZEOFT32);
    data += SIZEOFT32;
    /* 6 */

    WRITE_UNALIGNED(dbg, (void *)data,
        (const void *)&dh->dh_foreign_type_unit_count,
        sizeof(dh->dh_foreign_type_unit_count),
        SIZEOFT32);
    data += SIZEOFT32;
    /* 7 */
    WRITE_UNALIGNED(dbg, (void *)data,
        (const void *)&dh->dh_bucket_count,
        sizeof(dh->dh_bucket_count),
        SIZEOFT32);
    data += SIZEOFT32;
    /* 8 */
    WRITE_UNALIGNED(dbg, (void *)data,
        (const void *)&dh->dh_name_count,
        sizeof(dh->dh_name_count),
        SIZEOFT32);
    data += SIZEOFT32;

    /* 9 */
    WRITE_UNALIGNED(dbg, (void *)data,
        (const void *)&dh->dh_abbrev_table_size,
        sizeof(dh->dh_abbrev_table_size),
        SIZEOFT32);
    data += SIZEOFT32;

    /* 10 */
    WRITE_UNALIGNED(dbg, (void *)data,
        (const void *)&dh->dh_augmentation_string_size,
        sizeof(dh->dh_augmentation_string_size),
        SIZEOFT32);
    data += SIZEOFT32;
    if (dh->dh_augmentation_string_size) {
        /* 11 */
        memcpy((void *)data,dh->dh_augmentation_string,
            dh->dh_augmentation_string_size);
        data += dh->dh_augmentation_string_size;
        bytes = data - startdata;
    }
    bytes = data - startdata;

    /*  The CU offset table. A single entry */
    WRITE_UNALIGNED(dbg, (void *)data,
        (const void *)&zero,
        sizeof(stroffsets[0]),
        dh->dh_offset_size);
    data +=  dh->dh_offset_size;
    bytes = data - startdata;

    /* Now the string offsets table */
    WRITE_UNALIGNED(dbg, (void *)data,
        (const void *)&stroffsets[0],
        sizeof(stroffsets[0]),
        dh->dh_offset_size);
    data +=  dh->dh_offset_size;
    bytes = data - startdata;
    /* Now the Entry Offsets (DIE offsets)  array */
    WRITE_UNALIGNED(dbg, (void *)data,
        (const void *)&dieoffset[0],
        sizeof(dieoffset[0]),
        dh->dh_offset_size);
    data +=  dh->dh_offset_size;
    bytes = data - startdata;

    memcpy((void *)data,abbrv,sizeof(abbrv));
    data += sizeof(abbrv);
    bytes = data - startdata;

    memcpy((void *)data,entry,sizeof(entry));
    data += sizeof(entry);
    bytes = data - startdata;
    if (bytes != withunitlength) {
        printf("FAIL writing debug_names "
            "bytes written: %lu "
            "bytes allocated: %lu\n",
            (unsigned long)bytes,
            (unsigned long)withunitlength);
    }
#if 0
    dbg->de_dnames_blob = startdata;
    dbg->de_dnames_bloblength = withunitlength;
#endif
    return DW_DLV_OK;
}