Blame SOURCES/ltrace-0.7.91-ppc64le-support.patch

93f2b0
From eea4ad2cce289753aaa35b4e0258a76d8f8f367c Mon Sep 17 00:00:00 2001
93f2b0
From: Thierry Fauck <thierry@linux.vnet.ibm.com>
93f2b0
Date: Tue, 13 May 2014 07:48:24 -0400
93f2b0
Subject: [PATCH] Support for powerpc64 arch ppc64el
93f2b0
93f2b0
Signed-off-by: Thierry Fauck <thierry@linux.vnet.ibm.com>
93f2b0
93f2b0
	Add support for ppc64le proc and ELF ABIv2.
93f2b0
	Provides support for irelative and wchar
93f2b0
---
93f2b0
 ltrace-elf.c                           |    2 +-
93f2b0
 ltrace-elf.h                           |    1 +
93f2b0
 sysdeps/linux-gnu/ppc/arch.h           |   35 ++++-
93f2b0
 sysdeps/linux-gnu/ppc/fetch.c          |  244 +++++++++++++++++++++++++++++---
93f2b0
 sysdeps/linux-gnu/ppc/plt.c            |   98 ++++++++++++--
93f2b0
 sysdeps/linux-gnu/ppc/trace.c          |   10 ++
93f2b0
 testsuite/ltrace.main/system_calls.exp |    2 +-
93f2b0
 7 files changed, 356 insertions(+), 36 deletions(-)
93f2b0
93f2b0
diff --git a/ltrace-elf.c b/ltrace-elf.c
93f2b0
index 8997518..f638342 100644
93f2b0
--- a/ltrace-elf.c
93f2b0
+++ b/ltrace-elf.c
93f2b0
@@ -859,7 +859,7 @@ populate_plt(struct process *proc, const char *filename,
93f2b0
 	return 0;
93f2b0
 }
93f2b0
 
93f2b0
-static void
93f2b0
+void
93f2b0
 delete_symbol_chain(struct library_symbol *libsym)
93f2b0
 {
93f2b0
 	while (libsym != NULL) {
93f2b0
diff --git a/ltrace-elf.h b/ltrace-elf.h
93f2b0
index db4ffe9..4a824c4 100644
93f2b0
--- a/ltrace-elf.h
93f2b0
+++ b/ltrace-elf.h
93f2b0
@@ -166,6 +166,7 @@ int elf_read_next_uleb128(Elf_Data *data, GElf_Xword *offset, uint64_t *retp);
93f2b0
 /* Return whether there's AMOUNT more bytes after OFFSET in DATA.  */
93f2b0
 int elf_can_read_next(Elf_Data *data, GElf_Xword offset, GElf_Xword amount);
93f2b0
 
93f2b0
+void delete_symbol_chain(struct library_symbol *);
93f2b0
 #if __WORDSIZE == 32
93f2b0
 #define PRI_ELF_ADDR		PRIx32
93f2b0
 #define GELF_ADDR_CAST(x)	(void *)(uint32_t)(x)
93f2b0
diff --git a/sysdeps/linux-gnu/ppc/arch.h b/sysdeps/linux-gnu/ppc/arch.h
93f2b0
index bf9b5dc..7918a13 100644
93f2b0
--- a/sysdeps/linux-gnu/ppc/arch.h
93f2b0
+++ b/sysdeps/linux-gnu/ppc/arch.h
93f2b0
@@ -23,8 +23,8 @@
93f2b0
 #define LTRACE_PPC_ARCH_H
93f2b0
 
93f2b0
 #include <gelf.h>
93f2b0
+#include <stdbool.h>
93f2b0
 
93f2b0
-#define BREAKPOINT_VALUE { 0x7f, 0xe0, 0x00, 0x08 }
93f2b0
 #define BREAKPOINT_LENGTH 4
93f2b0
 #define DECR_PC_AFTER_BREAK 0
93f2b0
 
93f2b0
@@ -34,8 +34,33 @@
93f2b0
 #ifdef __powerpc64__ // Says 'ltrace' is 64 bits, says nothing about target.
93f2b0
 #define LT_ELFCLASS2	ELFCLASS64
93f2b0
 #define LT_ELF_MACHINE2	EM_PPC64
93f2b0
-#define ARCH_SUPPORTS_OPD
93f2b0
-#endif
93f2b0
+
93f2b0
+# ifdef __LITTLE_ENDIAN__
93f2b0
+# define BREAKPOINT_VALUE { 0x08, 0x00, 0xe0, 0x7f }
93f2b0
+# define ARCH_ENDIAN_LITTLE
93f2b0
+# else
93f2b0
+# define BREAKPOINT_VALUE { 0x7f, 0xe0, 0x00, 0x08 }
93f2b0
+# define ARCH_SUPPORTS_OPD
93f2b0
+# define ARCH_ENDIAN_BIG
93f2b0
+# endif
93f2b0
+
93f2b0
+# if _CALL_ELF != 2
93f2b0
+# define ARCH_SUPPORTS_OPD
93f2b0
+# define STACK_FRAME_OVERHEAD 112
93f2b0
+#  ifndef EF_PPC64_ABI
93f2b0
+#  define EF_PPC64_ABI 3
93f2b0
+#  endif
93f2b0
+# else /* _CALL_ELF == 2 ABIv2 */
93f2b0
+# define STACK_FRAME_OVERHEAD 32
93f2b0
+# endif /* CALL_ELF */
93f2b0
+
93f2b0
+#else
93f2b0
+#define BREAKPOINT_VALUE { 0x7f, 0xe0, 0x00, 0x08 }
93f2b0
+#define ARCH_ENDIAN_BIG
93f2b0
+# ifndef EF_PPC64_ABI
93f2b0
+# define EF_PPC64_ABI 3
93f2b0
+# endif
93f2b0
+#endif 	/* __powerpc64__ */
93f2b0
 
93f2b0
 #define ARCH_HAVE_SW_SINGLESTEP
93f2b0
 #define ARCH_HAVE_ADD_PLT_ENTRY
93f2b0
@@ -43,7 +68,6 @@
93f2b0
 #define ARCH_HAVE_TRANSLATE_ADDRESS
93f2b0
 #define ARCH_HAVE_DYNLINK_DONE
93f2b0
 #define ARCH_HAVE_FETCH_ARG
93f2b0
-#define ARCH_ENDIAN_BIG
93f2b0
 #define ARCH_HAVE_SIZEOF
93f2b0
 #define ARCH_HAVE_ALIGNOF
93f2b0
 
93f2b0
@@ -56,7 +80,8 @@ struct arch_ltelf_data {
93f2b0
 	Elf_Data *opd_data;
93f2b0
 	GElf_Addr opd_base;
93f2b0
 	GElf_Xword opd_size;
93f2b0
-	int secure_plt;
93f2b0
+	bool secure_plt : 1;
93f2b0
+	bool elfv2_abi  : 1;
93f2b0
 
93f2b0
 	Elf_Data *reladyn;
93f2b0
 	size_t reladyn_count;
93f2b0
diff --git a/sysdeps/linux-gnu/ppc/fetch.c b/sysdeps/linux-gnu/ppc/fetch.c
93f2b0
index ed38336..c9381c3 100644
93f2b0
--- a/sysdeps/linux-gnu/ppc/fetch.c
93f2b0
+++ b/sysdeps/linux-gnu/ppc/fetch.c
93f2b0
@@ -30,9 +30,11 @@
93f2b0
 #include "ptrace.h"
93f2b0
 #include "proc.h"
93f2b0
 #include "value.h"
93f2b0
+#include "ltrace-elf.h"
93f2b0
 
93f2b0
 static int allocate_gpr(struct fetch_context *ctx, struct process *proc,
93f2b0
-			struct arg_type_info *info, struct value *valuep);
93f2b0
+			struct arg_type_info *info, struct value *valuep,
93f2b0
+			size_t off, bool is_hfa_type);
93f2b0
 
93f2b0
 /* Floating point registers have the same width on 32-bit as well as
93f2b0
  * 64-bit PPC, but <ucontext.h> presents a different API depending on
93f2b0
@@ -62,7 +64,10 @@ struct fetch_context {
93f2b0
 		gregs64_t r64;
93f2b0
 	} regs;
93f2b0
 	struct fpregs_t fpregs;
93f2b0
-
93f2b0
+	int vgreg;
93f2b0
+	int struct_size;
93f2b0
+	int struct_hfa_size;
93f2b0
+	int struct_hfa_count;
93f2b0
 };
93f2b0
 
93f2b0
 static int
93f2b0
@@ -74,7 +79,8 @@ fetch_context_init(struct process *proc, struct fetch_context *context)
93f2b0
 	if (proc->e_machine == EM_PPC)
93f2b0
 		context->stack_pointer = proc->stack_pointer + 8;
93f2b0
 	else
93f2b0
-		context->stack_pointer = proc->stack_pointer + 112;
93f2b0
+		context->stack_pointer = proc->stack_pointer
93f2b0
+			+ STACK_FRAME_OVERHEAD;
93f2b0
 
93f2b0
 	/* When ltrace is 64-bit, we might use PTRACE_GETREGS to
93f2b0
 	 * obtain 64-bit as well as 32-bit registers.  But if we do it
93f2b0
@@ -118,6 +124,11 @@ arch_fetch_arg_init(enum tof type, struct process *proc,
93f2b0
 		return NULL;
93f2b0
 	}
93f2b0
 
93f2b0
+	context->vgreg = context->greg;
93f2b0
+	context->struct_size = 0;
93f2b0
+	context->struct_hfa_size = 0;
93f2b0
+	context->struct_hfa_count = 0;
93f2b0
+
93f2b0
 	/* Aggregates or unions of any length, and character strings
93f2b0
 	 * of length longer than 8 bytes, will be returned in a
93f2b0
 	 * storage buffer allocated by the caller. The caller will
93f2b0
@@ -125,8 +136,20 @@ arch_fetch_arg_init(enum tof type, struct process *proc,
93f2b0
 	 * in r3, causing the first explicit argument to be passed in
93f2b0
 	 * r4.  */
93f2b0
 	context->ret_struct = ret_info->type == ARGTYPE_STRUCT;
93f2b0
-	if (context->ret_struct)
93f2b0
+	if (context->ret_struct) {
93f2b0
+#if _CALL_ELF == 2
93f2b0
+		/* if R3 points to stack, parameters will be in R4.  */
93f2b0
+		uint64_t pstack_end = ptrace(PTRACE_PEEKTEXT, proc->pid,
93f2b0
+					proc->stack_pointer, 0);
93f2b0
+		if (((arch_addr_t)context->regs.r64[3] > proc->stack_pointer)
93f2b0
+		    && (context->regs.r64[3] < pstack_end)) {
93f2b0
+			context->greg++;
93f2b0
+			context->stack_pointer += 8;
93f2b0
+		}
93f2b0
+#else
93f2b0
 		context->greg++;
93f2b0
+#endif
93f2b0
+	}
93f2b0
 
93f2b0
 	return context;
93f2b0
 }
93f2b0
@@ -144,7 +167,8 @@ arch_fetch_arg_clone(struct process *proc,
93f2b0
 
93f2b0
 static int
93f2b0
 allocate_stack_slot(struct fetch_context *ctx, struct process *proc,
93f2b0
-		    struct arg_type_info *info, struct value *valuep)
93f2b0
+		    struct arg_type_info *info, struct value *valuep,
93f2b0
+		    bool is_hfa_type)
93f2b0
 {
93f2b0
 	size_t sz = type_sizeof(proc, info);
93f2b0
 	if (sz == (size_t)-1)
93f2b0
@@ -154,7 +178,14 @@ allocate_stack_slot(struct fetch_context *ctx, struct process *proc,
93f2b0
 	size_t off = 0;
93f2b0
 	if (proc->e_machine == EM_PPC && a < 4)
93f2b0
 		a = 4;
93f2b0
+#if _CALL_ELF == 2
93f2b0
+	else if (proc->e_machine == EM_PPC64 && sz == 4 && is_hfa_type)
93f2b0
+		a = 4;
93f2b0
+	else
93f2b0
+		a = 8;
93f2b0
+#else
93f2b0
 	else if (proc->e_machine == EM_PPC64 && a < 8)
93f2b0
+#endif
93f2b0
 		a = 8;
93f2b0
 
93f2b0
 	/* XXX Remove the two double casts when arch_addr_t
93f2b0
@@ -164,7 +195,7 @@ allocate_stack_slot(struct fetch_context *ctx, struct process *proc,
93f2b0
 
93f2b0
 	if (valuep != NULL)
93f2b0
 		value_in_inferior(valuep, ctx->stack_pointer + off);
93f2b0
-	ctx->stack_pointer += sz;
93f2b0
+	ctx->stack_pointer += a;
93f2b0
 
93f2b0
 	return 0;
93f2b0
 }
93f2b0
@@ -216,19 +247,34 @@ align_small_int(unsigned char *buf, size_t w, size_t sz)
93f2b0
 
93f2b0
 static int
93f2b0
 allocate_gpr(struct fetch_context *ctx, struct process *proc,
93f2b0
-	     struct arg_type_info *info, struct value *valuep)
93f2b0
+	     struct arg_type_info *info, struct value *valuep,
93f2b0
+	     size_t off, bool is_hfa_type)
93f2b0
 {
93f2b0
 	if (ctx->greg > 10)
93f2b0
-		return allocate_stack_slot(ctx, proc, info, valuep);
93f2b0
+		return allocate_stack_slot(ctx, proc, info, valuep, is_hfa_type);
93f2b0
 
93f2b0
-	int reg_num = ctx->greg++;
93f2b0
-	if (valuep == NULL)
93f2b0
-		return 0;
93f2b0
+	int reg_num = ctx->greg;
93f2b0
 
93f2b0
 	size_t sz = type_sizeof(proc, info);
93f2b0
 	if (sz == (size_t)-1)
93f2b0
 		return -1;
93f2b0
 	assert(sz == 1 || sz == 2 || sz == 4 || sz == 8);
93f2b0
+#if _CALL_ELF == 2
93f2b0
+	/* Consume the stack slot corresponding to this arg.  */
93f2b0
+	if ((sz + off) >= 8)
93f2b0
+		ctx->greg++;
93f2b0
+
93f2b0
+	if (is_hfa_type)
93f2b0
+		ctx->stack_pointer += sz;
93f2b0
+	else
93f2b0
+		ctx->stack_pointer += 8;
93f2b0
+#else
93f2b0
+	ctx->greg++;
93f2b0
+#endif
93f2b0
+
93f2b0
+	if (valuep == NULL)
93f2b0
+		return 0;
93f2b0
+
93f2b0
 	if (value_reserve(valuep, sz) == NULL)
93f2b0
 		return -1;
93f2b0
 
93f2b0
@@ -240,13 +286,14 @@ allocate_gpr(struct fetch_context *ctx, struct process *proc,
93f2b0
 	u.i64 = read_gpr(ctx, proc, reg_num);
93f2b0
 	if (proc->e_machine == EM_PPC)
93f2b0
 		align_small_int(u.buf, 8, sz);
93f2b0
-	memcpy(value_get_raw_data(valuep), u.buf, sz);
93f2b0
+	memcpy(value_get_raw_data(valuep), u.buf + off, sz);
93f2b0
 	return 0;
93f2b0
 }
93f2b0
 
93f2b0
 static int
93f2b0
 allocate_float(struct fetch_context *ctx, struct process *proc,
93f2b0
-	       struct arg_type_info *info, struct value *valuep)
93f2b0
+	       struct arg_type_info *info, struct value *valuep,
93f2b0
+	       size_t off, bool is_hfa_type)
93f2b0
 {
93f2b0
 	int pool = proc->e_machine == EM_PPC64 ? 13 : 8;
93f2b0
 	if (ctx->freg <= pool) {
93f2b0
@@ -257,8 +304,12 @@ allocate_float(struct fetch_context *ctx, struct process *proc,
93f2b0
 		} u = { .d = ctx->fpregs.fpregs[ctx->freg] };
93f2b0
 
93f2b0
 		ctx->freg++;
93f2b0
+
93f2b0
+		if (!is_hfa_type)
93f2b0
+			ctx->vgreg++;
93f2b0
+
93f2b0
 		if (proc->e_machine == EM_PPC64)
93f2b0
-			allocate_gpr(ctx, proc, info, NULL);
93f2b0
+			allocate_gpr(ctx, proc, info, NULL, off, is_hfa_type);
93f2b0
 
93f2b0
 		size_t sz = sizeof(double);
93f2b0
 		if (info->type == ARGTYPE_FLOAT) {
93f2b0
@@ -272,8 +323,128 @@ allocate_float(struct fetch_context *ctx, struct process *proc,
93f2b0
 		memcpy(value_get_raw_data(valuep), u.buf, sz);
93f2b0
 		return 0;
93f2b0
 	}
93f2b0
-	return allocate_stack_slot(ctx, proc, info, valuep);
93f2b0
+	return allocate_stack_slot(ctx, proc, info, valuep, is_hfa_type);
93f2b0
+}
93f2b0
+
93f2b0
+#if _CALL_ELF == 2
93f2b0
+static int
93f2b0
+allocate_hfa(struct fetch_context *ctx, struct process *proc,
93f2b0
+	     struct arg_type_info *info, struct value *valuep,
93f2b0
+	     enum arg_type hfa_type, size_t hfa_count)
93f2b0
+{
93f2b0
+	size_t sz = type_sizeof(proc, info);
93f2b0
+	if (sz == (size_t)-1)
93f2b0
+		return -1;
93f2b0
+
93f2b0
+	ctx->struct_hfa_size += sz;
93f2b0
+
93f2b0
+	/* There are two changes regarding structure return types:
93f2b0
+	 * * heterogeneous float/vector structs are returned
93f2b0
+	 *   in (multiple) FP/vector registers,
93f2b0
+	 *   instead of via implicit reference.
93f2b0
+	 * * small structs (up to 16 bytes) are return
93f2b0
+	 *   in one or two GPRs, instead of via implicit reference.
93f2b0
+	 *
93f2b0
+	 * Other structures (larger than 16 bytes, not heterogeneous)
93f2b0
+	 * are still returned via implicit reference (i.e. a pointer
93f2b0
+	 * to memory where to return the struct being passed in r3).
93f2b0
+	 * Of course, whether or not an implicit reference pointer
93f2b0
+	 * is present will shift the remaining arguments,
93f2b0
+	 * so you need to get this right for ELFv2 in order
93f2b0
+	 * to get the arguments correct.
93f2b0
+	 * If an actual parameter is known to correspond to an HFA
93f2b0
+	 * formal parameter, each element is passed in the next
93f2b0
+	 * available floating-point argument register starting at fp1
93f2b0
+	 * until the fp13. The remaining elements of the aggregate are
93f2b0
+	 * passed on the stack.  */
93f2b0
+	size_t slot_off = 0;
93f2b0
+
93f2b0
+	unsigned char *buf = value_reserve(valuep, sz);
93f2b0
+	if (buf == NULL)
93f2b0
+		return -1;
93f2b0
+
93f2b0
+	struct arg_type_info *hfa_info = type_get_simple(hfa_type);
93f2b0
+	size_t hfa_sz = type_sizeof(proc, hfa_info);
93f2b0
+
93f2b0
+	if (hfa_count > 8)
93f2b0
+		ctx->struct_hfa_count += hfa_count;
93f2b0
+
93f2b0
+	while (hfa_count > 0 && ctx->freg <= 13) {
93f2b0
+		int rc;
93f2b0
+		struct value tmp;
93f2b0
+
93f2b0
+		value_init(&tmp, proc, NULL, hfa_info, 0);
93f2b0
+
93f2b0
+		/* Hetereogeneous struct - get value on GPR or stack.  */
93f2b0
+		if (((hfa_type == ARGTYPE_FLOAT
93f2b0
+		    || hfa_type == ARGTYPE_DOUBLE)
93f2b0
+		      && hfa_count <= 8))
93f2b0
+			rc = allocate_float(ctx, proc, hfa_info, &tmp,
93f2b0
+						slot_off, true);
93f2b0
+		else
93f2b0
+			rc = allocate_gpr(ctx, proc, hfa_info, &tmp,
93f2b0
+						slot_off, true);
93f2b0
+
93f2b0
+		memcpy(buf, value_get_data(&tmp, NULL), hfa_sz);
93f2b0
+
93f2b0
+		slot_off += hfa_sz;
93f2b0
+		buf += hfa_sz;
93f2b0
+		hfa_count--;
93f2b0
+		if (slot_off == 8) {
93f2b0
+			slot_off = 0;
93f2b0
+			ctx->vgreg++;
93f2b0
+		}
93f2b0
+
93f2b0
+		value_destroy(&tmp);
93f2b0
+		if (rc < 0)
93f2b0
+			return -1;
93f2b0
+	}
93f2b0
+	if (hfa_count == 0)
93f2b0
+		return 0;
93f2b0
+
93f2b0
+	/* if no remaining FP, GPR corresponding to slot is used
93f2b0
+	* Mostly it is in part of r10.  */
93f2b0
+	if (ctx->struct_hfa_size <= 64 && ctx->vgreg == 10) {
93f2b0
+		while (ctx->vgreg <= 10) {
93f2b0
+			struct value tmp;
93f2b0
+			value_init(&tmp, proc, NULL, hfa_info, 0);
93f2b0
+			union {
93f2b0
+				uint64_t i64;
93f2b0
+				unsigned char buf[0];
93f2b0
+			} u;
93f2b0
+
93f2b0
+			u.i64 = read_gpr(ctx, proc, ctx->vgreg);
93f2b0
+
93f2b0
+			memcpy(buf, u.buf + slot_off, hfa_sz);
93f2b0
+			slot_off += hfa_sz;
93f2b0
+			buf += hfa_sz;
93f2b0
+			hfa_count--;
93f2b0
+			ctx->stack_pointer += hfa_sz;
93f2b0
+			if (slot_off >= 8 ) {
93f2b0
+				slot_off = 0;
93f2b0
+				ctx->vgreg++;
93f2b0
+			}
93f2b0
+			value_destroy(&tmp);
93f2b0
+		}
93f2b0
+	}
93f2b0
+
93f2b0
+	if (hfa_count == 0)
93f2b0
+		return 0;
93f2b0
+
93f2b0
+	/* Remaining values are on stack */
93f2b0
+	while (hfa_count) {
93f2b0
+		struct value tmp;
93f2b0
+		value_init(&tmp, proc, NULL, hfa_info, 0);
93f2b0
+
93f2b0
+		value_in_inferior(&tmp, ctx->stack_pointer);
93f2b0
+		memcpy(buf, value_get_data(&tmp, NULL), hfa_sz);
93f2b0
+		ctx->stack_pointer += hfa_sz;
93f2b0
+		buf += hfa_sz;
93f2b0
+		hfa_count--;
93f2b0
+	}
93f2b0
+	return 0;
93f2b0
 }
93f2b0
+#endif
93f2b0
 
93f2b0
 static int
93f2b0
 allocate_argument(struct fetch_context *ctx, struct process *proc,
93f2b0
@@ -287,13 +458,25 @@ allocate_argument(struct fetch_context *ctx, struct process *proc,
93f2b0
 
93f2b0
 	case ARGTYPE_FLOAT:
93f2b0
 	case ARGTYPE_DOUBLE:
93f2b0
-		return allocate_float(ctx, proc, info, valuep);
93f2b0
+		return allocate_float(ctx, proc, info, valuep,
93f2b0
+					8 - type_sizeof(proc,info), false);
93f2b0
 
93f2b0
 	case ARGTYPE_STRUCT:
93f2b0
 		if (proc->e_machine == EM_PPC) {
93f2b0
 			if (value_pass_by_reference(valuep) < 0)
93f2b0
 				return -1;
93f2b0
 		} else {
93f2b0
+#if _CALL_ELF == 2
93f2b0
+			struct arg_type_info *hfa_info;
93f2b0
+			size_t hfa_size;
93f2b0
+			hfa_info = type_get_hfa_type(info, &hfa_size);
93f2b0
+			if (hfa_info != NULL ) {
93f2b0
+				size_t sz = type_sizeof(proc, info);
93f2b0
+				ctx->struct_size += sz;
93f2b0
+				return allocate_hfa(ctx, proc, info, valuep,
93f2b0
+						hfa_info->type, hfa_size);
93f2b0
+			}
93f2b0
+#endif
93f2b0
 			/* PPC64: Fixed size aggregates and unions passed by
93f2b0
 			 * value are mapped to as many doublewords of the
93f2b0
 			 * parameter save area as the value uses in memory.
93f2b0
@@ -326,6 +509,10 @@ allocate_argument(struct fetch_context *ctx, struct process *proc,
93f2b0
 	size_t sz = type_sizeof(proc, valuep->type);
93f2b0
 	if (sz == (size_t)-1)
93f2b0
 		return -1;
93f2b0
+
93f2b0
+	if (ctx->ret_struct)
93f2b0
+		ctx->struct_size += sz;
93f2b0
+
93f2b0
 	size_t slots = (sz + width - 1) / width;  /* Round up.  */
93f2b0
 	unsigned char *buf = value_reserve(valuep, slots * width);
93f2b0
 	if (buf == NULL)
93f2b0
@@ -346,9 +533,11 @@ allocate_argument(struct fetch_context *ctx, struct process *proc,
93f2b0
 		struct arg_type_info *fp_info
93f2b0
 			= type_get_fp_equivalent(valuep->type);
93f2b0
 		if (fp_info != NULL)
93f2b0
-			rc = allocate_float(ctx, proc, fp_info, &val;;
93f2b0
+			rc = allocate_float(ctx, proc, fp_info, &val,
93f2b0
+					8-type_sizeof(proc,info), false);
93f2b0
 		else
93f2b0
-			rc = allocate_gpr(ctx, proc, long_info, &val;;
93f2b0
+			rc = allocate_gpr(ctx, proc, long_info, &val,
93f2b0
+					0, false);
93f2b0
 
93f2b0
 		if (rc >= 0) {
93f2b0
 			memcpy(ptr, value_get_data(&val, NULL), width);
93f2b0
@@ -363,6 +552,7 @@ allocate_argument(struct fetch_context *ctx, struct process *proc,
93f2b0
 			return rc;
93f2b0
 	}
93f2b0
 
93f2b0
+#ifndef __LITTLE_ENDIAN__
93f2b0
 	/* Small values need post-processing.  */
93f2b0
 	if (sz < width) {
93f2b0
 		switch (info->type) {
93f2b0
@@ -394,6 +584,7 @@ allocate_argument(struct fetch_context *ctx, struct process *proc,
93f2b0
 			break;
93f2b0
 		}
93f2b0
 	}
93f2b0
+#endif
93f2b0
 
93f2b0
 	return 0;
93f2b0
 }
93f2b0
@@ -411,7 +602,22 @@ arch_fetch_retval(struct fetch_context *ctx, enum tof type,
93f2b0
 		  struct process *proc, struct arg_type_info *info,
93f2b0
 		  struct value *valuep)
93f2b0
 {
93f2b0
+	if (fetch_context_init(proc, ctx) < 0)
93f2b0
+		return -1;
93f2b0
+
93f2b0
+#if _CALL_ELF == 2
93f2b0
+	void *ptr = (void *)(ctx->regs.r64[1]+32);
93f2b0
+	uint64_t val = ptrace(PTRACE_PEEKTEXT, proc->pid, ptr, 0);
93f2b0
+
93f2b0
+	if (ctx->ret_struct
93f2b0
+	   && ((ctx->struct_size > 64
93f2b0
+	      || ctx->struct_hfa_count > 8
93f2b0
+	      || (ctx->struct_hfa_size == 0 && ctx->struct_size > 56)
93f2b0
+	      || (ctx->regs.r64[3] == ctx->regs.r64[1]+32)
93f2b0
+	      || (ctx->regs.r64[3] == val )))) {
93f2b0
+#else
93f2b0
 	if (ctx->ret_struct) {
93f2b0
+#endif
93f2b0
 		assert(info->type == ARGTYPE_STRUCT);
93f2b0
 
93f2b0
 		uint64_t addr = read_gpr(ctx, proc, 3);
93f2b0
@@ -424,8 +630,6 @@ arch_fetch_retval(struct fetch_context *ctx, enum tof type,
93f2b0
 		return 0;
93f2b0
 	}
93f2b0
 
93f2b0
-	if (fetch_context_init(proc, ctx) < 0)
93f2b0
-		return -1;
93f2b0
 	return allocate_argument(ctx, proc, info, valuep);
93f2b0
 }
93f2b0
 
93f2b0
diff --git a/sysdeps/linux-gnu/ppc/plt.c b/sysdeps/linux-gnu/ppc/plt.c
93f2b0
index 332daa8..45ed7fb 100644
93f2b0
--- a/sysdeps/linux-gnu/ppc/plt.c
93f2b0
+++ b/sysdeps/linux-gnu/ppc/plt.c
93f2b0
@@ -136,7 +136,11 @@
93f2b0
  */
93f2b0
 
93f2b0
 #define PPC_PLT_STUB_SIZE 16
93f2b0
-#define PPC64_PLT_STUB_SIZE 8 //xxx
93f2b0
+#if _CALL_ELF != 2
93f2b0
+#define PPC64_PLT_STUB_SIZE 8
93f2b0
+#else
93f2b0
+#define PPC64_PLT_STUB_SIZE 4
93f2b0
+#endif
93f2b0
 
93f2b0
 static inline int
93f2b0
 host_powerpc64()
93f2b0
@@ -186,8 +190,13 @@ ppc32_delayed_symbol(struct library_symbol *libsym)
93f2b0
 	if ((insn1 & BRANCH_MASK) == B_INSN
93f2b0
 	    || ((insn2 & BRANCH_MASK) == B_INSN
93f2b0
 		/* XXX double cast  */
93f2b0
+#ifdef __LITTLE_ENDIAN__
93f2b0
+		&& (ppc_branch_dest(libsym->enter_addr + 4, insn1)
93f2b0
+		    == (arch_addr_t) (long) libsym->lib->arch.pltgot_addr)))
93f2b0
+#else
93f2b0
 		&& (ppc_branch_dest(libsym->enter_addr + 4, insn2)
93f2b0
 		    == (arch_addr_t) (long) libsym->lib->arch.pltgot_addr)))
93f2b0
+#endif
93f2b0
 	{
93f2b0
 		mark_as_resolved(libsym, libsym->arch.resolved_value);
93f2b0
 	}
93f2b0
@@ -206,7 +215,7 @@ arch_dynlink_done(struct process *proc)
93f2b0
 				"couldn't read PLT value for %s(%p): %s\n",
93f2b0
 				libsym->name, libsym->enter_addr,
93f2b0
 				strerror(errno));
93f2b0
-			return;
93f2b0
+				return;
93f2b0
 		}
93f2b0
 
93f2b0
 		if (proc->e_machine == EM_PPC)
93f2b0
@@ -227,8 +236,14 @@ reloc_is_irelative(int machine, GElf_Rela *rela)
93f2b0
 {
93f2b0
 	bool irelative = false;
93f2b0
 	if (machine == EM_PPC64) {
93f2b0
-#ifdef R_PPC64_JMP_IREL
93f2b0
+#ifdef __LITTLE_ENDIAN__
93f2b0
+# ifdef R_PPC64_IRELATIVE
93f2b0
+		irelative = GELF_R_TYPE(rela->r_info) == R_PPC64_IRELATIVE;
93f2b0
+# endif
93f2b0
+#else
93f2b0
+# ifdef R_PPC64_JMP_IREL
93f2b0
 		irelative = GELF_R_TYPE(rela->r_info) == R_PPC64_JMP_IREL;
93f2b0
+# endif
93f2b0
 #endif
93f2b0
 	} else {
93f2b0
 		assert(machine == EM_PPC);
93f2b0
@@ -285,6 +300,7 @@ arch_translate_address_dyn(struct process *proc,
93f2b0
 			   arch_addr_t addr, arch_addr_t *ret)
93f2b0
 {
93f2b0
 	if (proc->e_machine == EM_PPC64) {
93f2b0
+#if _CALL_ELF != 2
93f2b0
 		uint64_t value;
93f2b0
 		if (proc_read_64(proc, addr, &value) < 0) {
93f2b0
 			fprintf(stderr,
93f2b0
@@ -296,6 +312,7 @@ arch_translate_address_dyn(struct process *proc,
93f2b0
 		 * arch_addr_t becomes integral type.  */
93f2b0
 		*ret = (arch_addr_t)(uintptr_t)value;
93f2b0
 		return 0;
93f2b0
+#endif
93f2b0
 	}
93f2b0
 
93f2b0
 	*ret = addr;
93f2b0
@@ -306,7 +323,8 @@ int
93f2b0
 arch_translate_address(struct ltelf *lte,
93f2b0
 		       arch_addr_t addr, arch_addr_t *ret)
93f2b0
 {
93f2b0
-	if (lte->ehdr.e_machine == EM_PPC64) {
93f2b0
+	if (lte->ehdr.e_machine == EM_PPC64
93f2b0
+	    && !lte->arch.elfv2_abi) {
93f2b0
 		/* XXX The double cast should be removed when
93f2b0
 		 * arch_addr_t becomes integral type.  */
93f2b0
 		GElf_Xword offset
93f2b0
@@ -430,7 +448,16 @@ reloc_copy_if_irelative(GElf_Rela *rela, void *data)
93f2b0
 int
93f2b0
 arch_elf_init(struct ltelf *lte, struct library *lib)
93f2b0
 {
93f2b0
+
93f2b0
+	/* Check for ABIv2 in ELF header processor specific flag.  */
93f2b0
+#ifndef EF_PPC64_ABI
93f2b0
+	assert (! (lte->ehdr.e_flags & 3 ) == 2)
93f2b0
+#else
93f2b0
+	lte->arch.elfv2_abi=((lte->ehdr.e_flags & EF_PPC64_ABI) == 2) ;
93f2b0
+#endif
93f2b0
+
93f2b0
 	if (lte->ehdr.e_machine == EM_PPC64
93f2b0
+	    && !lte->arch.elfv2_abi
93f2b0
 	    && load_opd_data(lte, lib) < 0)
93f2b0
 		return -1;
93f2b0
 
93f2b0
@@ -599,7 +626,7 @@ read_plt_slot_value(struct process *proc, GElf_Addr addr, GElf_Addr *valp)
93f2b0
 	uint64_t l;
93f2b0
 	/* XXX double cast.  */
93f2b0
 	if (proc_read_64(proc, (arch_addr_t)(uintptr_t)addr, &l) < 0) {
93f2b0
-		fprintf(stderr, "ptrace .plt slot value @%#" PRIx64": %s\n",
93f2b0
+		debug(DEBUG_EVENT, "ptrace .plt slot value @%#" PRIx64": %s",
93f2b0
 			addr, strerror(errno));
93f2b0
 		return -1;
93f2b0
 	}
93f2b0
@@ -616,7 +643,7 @@ unresolve_plt_slot(struct process *proc, GElf_Addr addr, GElf_Addr value)
93f2b0
 	 * pointers intact.  Hence the only adjustment that we need to
93f2b0
 	 * do is to IP.  */
93f2b0
 	if (ptrace(PTRACE_POKETEXT, proc->pid, addr, value) < 0) {
93f2b0
-		fprintf(stderr, "failed to unresolve .plt slot: %s\n",
93f2b0
+		debug(DEBUG_EVENT, "failed to unresolve .plt slot: %s",
93f2b0
 			strerror(errno));
93f2b0
 		return -1;
93f2b0
 	}
93f2b0
@@ -629,9 +656,48 @@ arch_elf_add_func_entry(struct process *proc, struct ltelf *lte,
93f2b0
 			arch_addr_t addr, const char *name,
93f2b0
 			struct library_symbol **ret)
93f2b0
 {
93f2b0
-	if (lte->ehdr.e_machine != EM_PPC || lte->ehdr.e_type == ET_DYN)
93f2b0
+#ifndef PPC64_LOCAL_ENTRY_OFFSET
93f2b0
+	assert(! lte->arch.elfv2_abi);
93f2b0
+#else
93f2b0
+	/* With ABIv2 st_other field contains an offset.  */
93f2b0
+	 if (lte->arch.elfv2_abi)
93f2b0
+		addr += PPC64_LOCAL_ENTRY_OFFSET(sym->st_other);
93f2b0
+#endif
93f2b0
+
93f2b0
+	int st_info = GELF_ST_TYPE(sym->st_info);
93f2b0
+
93f2b0
+	if ((lte->ehdr.e_machine != EM_PPC && sym->st_other == 0)
93f2b0
+	    || lte->ehdr.e_type == ET_DYN
93f2b0
+	    || (st_info == STT_FUNC && ! sym->st_other))
93f2b0
 		return PLT_DEFAULT;
93f2b0
 
93f2b0
+	if (st_info == STT_FUNC) {
93f2b0
+		/* Put the default symbol to the chain.
93f2b0
+		 * The addr has already been updated with
93f2b0
+		 * symbol offset  */
93f2b0
+		char *full_name = strdup(name);
93f2b0
+		if (full_name == NULL) {
93f2b0
+			fprintf(stderr, "couldn't copy name of %s: %s\n",
93f2b0
+			name, strerror(errno));
93f2b0
+			free(full_name);
93f2b0
+			return PLT_FAIL;
93f2b0
+		}
93f2b0
+		struct library_symbol *libsym = malloc(sizeof *libsym);
93f2b0
+		if (libsym == NULL
93f2b0
+		    || library_symbol_init(libsym, addr, full_name, 1,
93f2b0
+					   LS_TOPLT_NONE) < 0) {
93f2b0
+			free(libsym);
93f2b0
+			delete_symbol_chain(libsym);
93f2b0
+			libsym = NULL;
93f2b0
+			fprintf(stderr, "Couldn't add symbol %s"
93f2b0
+				"for tracing.\n", name);
93f2b0
+		}
93f2b0
+		full_name = NULL;
93f2b0
+		libsym->next = *ret;
93f2b0
+		*ret = libsym;
93f2b0
+		return PLT_OK;
93f2b0
+	}
93f2b0
+
93f2b0
 	bool ifunc = false;
93f2b0
 #ifdef STT_GNU_IFUNC
93f2b0
 	ifunc = GELF_ST_TYPE(sym->st_info) == STT_GNU_IFUNC;
93f2b0
@@ -761,9 +827,15 @@ arch_elf_add_plt_entry(struct process *proc, struct ltelf *lte,
93f2b0
 	assert(plt_slot_addr >= lte->plt_addr
93f2b0
 	       || plt_slot_addr < lte->plt_addr + lte->plt_size);
93f2b0
 
93f2b0
+	/* Should avoid to do read if dynamic linker hasn't run yet
93f2b0
+	 * or allow -1 a valid return code.  */
93f2b0
 	GElf_Addr plt_slot_value;
93f2b0
-	if (read_plt_slot_value(proc, plt_slot_addr, &plt_slot_value) < 0)
93f2b0
-		goto fail;
93f2b0
+	if (read_plt_slot_value(proc, plt_slot_addr, &plt_slot_value) < 0) {
93f2b0
+		if (!lte->arch.elfv2_abi)
93f2b0
+			goto fail;
93f2b0
+		else
93f2b0
+			return PPC_PLT_UNRESOLVED;
93f2b0
+	}
93f2b0
 
93f2b0
 	struct library_symbol *libsym = malloc(sizeof(*libsym));
93f2b0
 	if (libsym == NULL) {
93f2b0
@@ -997,8 +1069,12 @@ ppc_plt_bp_continue(struct breakpoint *bp, struct process *proc)
93f2b0
 			return;
93f2b0
 		}
93f2b0
 
93f2b0
+#if _CALL_ELF == 2
93f2b0
+		continue_after_breakpoint(proc, bp);
93f2b0
+#else
93f2b0
 		jump_to_entry_point(proc, bp);
93f2b0
 		continue_process(proc->pid);
93f2b0
+#endif
93f2b0
 		return;
93f2b0
 
93f2b0
 	case PPC64_PLT_STUB:
93f2b0
@@ -1123,7 +1199,11 @@ arch_library_symbol_init(struct library_symbol *libsym)
93f2b0
 	/* We set type explicitly in the code above, where we have the
93f2b0
 	 * necessary context.  This is for calls from ltrace-elf.c and
93f2b0
 	 * such.  */
93f2b0
+#if _CALL_ELF == 2
93f2b0
+	libsym->arch.type = PPC_PLT_UNRESOLVED;
93f2b0
+#else
93f2b0
 	libsym->arch.type = PPC_DEFAULT;
93f2b0
+#endif
93f2b0
 	return 0;
93f2b0
 }
93f2b0
 
93f2b0
diff --git a/sysdeps/linux-gnu/ppc/trace.c b/sysdeps/linux-gnu/ppc/trace.c
93f2b0
index ee9a6b5..5aab538 100644
93f2b0
--- a/sysdeps/linux-gnu/ppc/trace.c
93f2b0
+++ b/sysdeps/linux-gnu/ppc/trace.c
93f2b0
@@ -65,9 +65,15 @@ syscall_p(struct process *proc, int status, int *sysnum)
93f2b0
 	if (WIFSTOPPED(status)
93f2b0
 	    && WSTOPSIG(status) == (SIGTRAP | proc->tracesysgood)) {
93f2b0
 		long pc = (long)get_instruction_pointer(proc);
93f2b0
+#ifndef __LITTLE_ENDIAN__
93f2b0
 		int insn =
93f2b0
 		    (int)ptrace(PTRACE_PEEKTEXT, proc->pid, pc - sizeof(long),
93f2b0
 				0);
93f2b0
+#else
93f2b0
+		int insn =
93f2b0
+		    (int)ptrace(PTRACE_PEEKTEXT, proc->pid, pc - sizeof(int),
93f2b0
+				0);
93f2b0
+#endif
93f2b0
 
93f2b0
 		if (insn == SYSCALL_INSN) {
93f2b0
 			*sysnum =
93f2b0
diff -up ltrace-0.7.91/sysdeps/linux-gnu/ppc/trace.c\~ ltrace-0.7.91/sysdeps/linux-gnu/ppc/trace.c
93f2b0
--- ltrace-0.7.91/sysdeps/linux-gnu/ppc/trace.c~	2014-08-08 14:05:58.000000000 +0200
93f2b0
+++ ltrace-0.7.91/sysdeps/linux-gnu/ppc/trace.c	2014-08-08 14:07:55.000000000 +0200
93f2b0
@@ -133,7 +133,11 @@ arch_sw_singlestep(struct process *proc,
93f2b0
 			return SWS_FAIL;
93f2b0
 		uint32_t insn;
93f2b0
 #ifdef __powerpc64__
93f2b0
+# ifdef __LITTLE_ENDIAN__
93f2b0
+		insn = (uint32_t) l;
93f2b0
+# else
93f2b0
 		insn = l >> 32;
93f2b0
+# endif
93f2b0
 #else
93f2b0
 		insn = l;
93f2b0
 #endif
93f2b0
diff -up ltrace-0.7.91/configure\~ ltrace-0.7.91/configure
93f2b0
--- ltrace-0.7.91/configure~	2014-08-08 14:09:12.000000000 +0200
93f2b0
+++ ltrace-0.7.91/configure	2014-08-08 14:18:30.000000000 +0200
93f2b0
@@ -2555,7 +2555,7 @@ case "${host_cpu}" in
93f2b0
     arm*|sa110)		HOST_CPU="arm" ;;
93f2b0
     cris*)		HOST_CPU="cris" ;;
93f2b0
     mips*)		HOST_CPU="mips" ;;
93f2b0
-    powerpc|powerpc64)	HOST_CPU="ppc" ;;
93f2b0
+    powerpc|powerpc64|powerpc64le)	HOST_CPU="ppc" ;;
93f2b0
     sun4u|sparc64)	HOST_CPU="sparc" ;;
93f2b0
     s390x)		HOST_CPU="s390" ;;
93f2b0
     i?86|x86_64)	HOST_CPU="x86" ;;
93f2b0
@@ -12094,7 +12094,7 @@ if test x"$enable_libunwind" = xyes; the
93f2b0
       arm*|sa110)         UNWIND_ARCH="arm" ;;
93f2b0
       i?86)               UNWIND_ARCH="x86" ;;
93f2b0
       powerpc)            UNWIND_ARCH="ppc32" ;;
93f2b0
-      powerpc64)          UNWIND_ARCH="ppc64" ;;
93f2b0
+      powerpc64|powerpc64le)          UNWIND_ARCH="ppc64" ;;
93f2b0
       mips*)              UNWIND_ARCH="mips" ;;
93f2b0
       *)                  UNWIND_ARCH="${host_cpu}" ;;
93f2b0
   esac