/* gen-sframe.h - Support for generating SFrame.
  Copyright (C) 2022-2024 Free Software Foundation, Inc.

  This file is part of GAS, the GNU Assembler.

  GAS 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, or (at your option)
  any later version.

  GAS 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 GAS; see the file COPYING.  If not, write to the Free
  Software Foundation, 51 Franklin Street - Fifth Floor, Boston, MA
  02110-1301, USA.  */

#ifndef GENSFRAME_H
#define GENSFRAME_H

#define SFRAME_FRE_ELEM_LOC_REG         0
#define SFRAME_FRE_ELEM_LOC_STACK       1

/* SFrame Frame Row Entry (FRE).

  A frame row entry is a slice of the frame and can be valid for a set of
  program instructions.  It keeps all information needed to retrieve the CFA
  and the Return Address (RA) if tracked.

  A frame row entry effectively stores accumulated information gathered by
  interpreting multiple CFI instructions.  More precisely, it is a
  self-sufficient record in its own right.  Only the subset of information
  necessary for unwinding is stored: Given a PC, how to retrieve the CFA and
  the RA.
*/

struct sframe_row_entry
{
 /* A linked list.  */
 struct sframe_row_entry *next;

 /* Start and end of the frame row entry.  */
 symbolS *pc_begin;
 symbolS *pc_end;

 /* A frame row entry is a merge candidate if new information can be updated
    on it.  */
 bool merge_candidate;

 /* Whether the return address is mangled with pauth code.  */
 bool mangled_ra_p;

 /* Track CFA base (architectural) register ID.  */
 unsigned int cfa_base_reg;
 /* Offset from the CFA base register for recovering CFA.  */
 offsetT cfa_offset;

 /* Track the other register used as base register for CFA.  Specify whether
    it is in register or memory.  */
 unsigned int base_reg;
 unsigned int bp_loc;
 /* If the other register is stashed on stack, note the offset.  */
 offsetT bp_offset;

 /* Track RA location.  Specify whether it is in register or memory.  */
 unsigned int ra_loc;
 /* If RA is stashed on stack, note the offset.  */
 offsetT ra_offset;
};

/* SFrame Function Description Entry.  */

struct sframe_func_entry
{
 /* A linked list.  */
 struct sframe_func_entry *next;

 /* Reference to the FDE created from CFI in dw2gencfi.  Some information
    like the start_address and the segment is made available via this
    member.  */
 const struct fde_entry *dw_fde;

 /* Reference to the first FRE for this function.  */
 struct sframe_row_entry *sframe_fres;

 unsigned int num_fres;
};

/* SFrame Function Description Entry Translation Context.  */

struct sframe_xlate_ctx
{
 /* Reference to the FDE created from CFI in dw2gencfi.  Information
    like the FDE start_address, end_address and the cfi insns are
    made available via this member.  */
 const struct fde_entry *dw_fde;

 /* List of FREs in the current FDE translation context, bounded by first_fre
    and last_fre.  */

 /* Keep track of the first FRE for the purpose of restoring state if
    necessary (for DW_CFA_restore).  */
 struct sframe_row_entry *first_fre;
 /* The last FRE in the list.  */
 struct sframe_row_entry *last_fre;

 /* The current FRE under construction.  */
 struct sframe_row_entry *cur_fre;
 /* Remember FRE for an eventual restore.  */
 struct sframe_row_entry *remember_fre;

 unsigned num_xlate_fres;
};

/* Error codes for SFrame translation context.  */
enum sframe_xlate_err
{
 /* Success.  */
 SFRAME_XLATE_OK = 0,
 /* Error.  */
 SFRAME_XLATE_ERROR = 1,
 /* Detailed error codes.  */
 SFRAME_XLATE_ERR_INVAL = -1,
 SFRAME_XLATE_ERR_NOTREPRESENTED = -2,
};

/* Callback to create the abi/arch identifier for SFrame section.  */

unsigned char
sframe_get_abi_arch_callback (const char *target_arch,
                             int big_endian_p);

/* The list of all FDEs with data in SFrame internal representation.  */

extern struct sframe_func_entry *all_sframe_fdes;

/* SFrame version specific operations structure.  */

struct sframe_version_ops
{
 unsigned char format_version;    /* SFrame format version.  */
 /* set SFrame FRE info.  */
 unsigned char (*set_fre_info) (unsigned int, unsigned int, unsigned int,
                                bool);
 /* set SFrame Func info.  */
 unsigned char (*set_func_info) (unsigned int, unsigned int, unsigned int);
};

/* Generate SFrame stack trace info and prepare contents for the output.
  outout_sframe ()  is called at the end of file.  */

extern void output_sframe (segT sframe_seg);

#endif /* GENSFRAME_H */