summaryrefslogtreecommitdiff
path: root/sysdeps/linux-gnu/ppc/arch.h
blob: d5ad759a755cc1e9e8b67733b1b5fa2748f8af41 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
/*
 * This file is part of ltrace.
 * Copyright (C) 2012,2013,2014 Petr Machata
 * Copyright (C) 2006 Paul Gilliam
 * Copyright (C) 2002,2004 Juan Cespedes
 *
 * This program 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 2 of the
 * License, or (at your option) any later version.
 *
 * This program 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 this program; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
 * 02110-1301 USA
 */
#ifndef LTRACE_PPC_ARCH_H
#define LTRACE_PPC_ARCH_H

#include <gelf.h>
#include <stdbool.h>

#define BREAKPOINT_LENGTH 4
#define DECR_PC_AFTER_BREAK 0

#define LT_ELFCLASS	ELFCLASS32
#define LT_ELF_MACHINE	EM_PPC

#ifdef __powerpc64__ // Says 'ltrace' is 64 bits, says nothing about target.
#define LT_ELFCLASS2	ELFCLASS64
#define LT_ELF_MACHINE2	EM_PPC64

# ifdef __LITTLE_ENDIAN__
# define BREAKPOINT_VALUE { 0x08, 0x00, 0xe0, 0x7f }
# define ARCH_ENDIAN_LITTLE
# else
# define BREAKPOINT_VALUE { 0x7f, 0xe0, 0x00, 0x08 }
# define ARCH_SUPPORTS_OPD
# define ARCH_ENDIAN_BIG
# endif

# if _CALL_ELF != 2
# define ARCH_SUPPORTS_OPD
# define STACK_FRAME_OVERHEAD 112
#  ifndef EF_PPC64_ABI
#  define EF_PPC64_ABI 3
#  endif
# else /* _CALL_ELF == 2 ABIv2 */
# define STACK_FRAME_OVERHEAD 32
# endif /* CALL_ELF */

#else
#define BREAKPOINT_VALUE { 0x7f, 0xe0, 0x00, 0x08 }
#define ARCH_ENDIAN_BIG
# define STACK_FRAME_OVERHEAD 112
# ifndef EF_PPC64_ABI
# define EF_PPC64_ABI 3
# endif
#endif 	/* __powerpc64__ */

#define ARCH_HAVE_SW_SINGLESTEP
#define ARCH_HAVE_ADD_PLT_ENTRY
#define ARCH_HAVE_ADD_FUNC_ENTRY
#define ARCH_HAVE_TRANSLATE_ADDRESS
#define ARCH_HAVE_DYNLINK_DONE
#define ARCH_HAVE_FETCH_ARG
#define ARCH_HAVE_SIZEOF
#define ARCH_HAVE_ALIGNOF

struct library_symbol;

#define ARCH_HAVE_LTELF_DATA
struct arch_ltelf_data {
	GElf_Addr plt_stub_vma;
	struct library_symbol *stubs;
	Elf_Data *opd_data;
	GElf_Addr opd_base;
	GElf_Xword opd_size;
	bool secure_plt : 1;
	bool elfv2_abi  : 1;

	Elf_Data *reladyn;
	size_t reladyn_count;
};

#define ARCH_HAVE_LIBRARY_DATA
struct arch_library_data {
	GElf_Addr pltgot_addr;
	int bss_plt_prelinked;
};

enum ppc64_plt_type {
	/* Either a non-PLT symbol, or PPC32 symbol.  */
	PPC_DEFAULT = 0,

	/* PPC64 STUB, never resolved.  */
	PPC64_PLT_STUB,

	/* Unresolved PLT symbol (.plt contains PLT address).  */
	PPC_PLT_UNRESOLVED,

	/* Resolved PLT symbol.  The corresponding .plt slot contained
	 * target address, which was changed to the address of
	 * corresponding PLT entry.  The original is now saved in
	 * RESOLVED_VALUE.  */
	PPC_PLT_RESOLVED,

	/* Very similar to PPC_PLT_UNRESOLVED, but for JMP_IREL
	 * slots.  */
	PPC_PLT_IRELATIVE,

	/* Transitional state before the breakpoint is enabled.  */
	PPC_PLT_NEED_UNRESOLVE,
};

#define ARCH_HAVE_LIBRARY_SYMBOL_DATA
struct ppc_unresolve_data;
struct arch_library_symbol_data {
	enum ppc64_plt_type type;

	/* State		Contents
	 *
	 * PPC_DEFAULT		N/A
	 * PPC64_PLT_STUB	N/A
	 * PPC_PLT_UNRESOLVED	PLT entry address.
	 * PPC_PLT_IRELATIVE	Likewise.
	 * PPC_PLT_RESOLVED	The original value the slot was resolved to.
	 * PPC_PLT_NEED_UNRESOLVE	DATA.
	 */
	union {
		GElf_Addr resolved_value;
		struct ppc_unresolve_data *data;
	};

	/* Address of corresponding slot in .plt.  */
	GElf_Addr plt_slot_addr;
};

#define ARCH_HAVE_BREAKPOINT_DATA
struct arch_breakpoint_data {
	/* This is where we hide symbol for IRELATIVE breakpoint for
	 * the first time that it hits.  This is NULL for normal
	 * breakpoints.  */
	struct library_symbol *irel_libsym;
};

#define ARCH_HAVE_PROCESS_DATA
struct arch_process_data {
	/* Breakpoint that hits when the dynamic linker is about to
	 * update a .plt slot.  NULL before that address is known.  */
	struct breakpoint *dl_plt_update_bp;

	/* PLT update breakpoint looks here for the handler.  */
	struct process_stopping_handler *handler;
};

#endif /* LTRACE_PPC_ARCH_H */