Blame s390x.c

Packit Service 501009
/* s390.c - core analysis suite
Packit Service 501009
 *
Packit Service 501009
 * Copyright (C) 2001, 2002 Mission Critical Linux, Inc.
Packit Service 501009
 * Copyright (C) 2002-2006, 2009-2014 David Anderson
Packit Service 501009
 * Copyright (C) 2002-2006, 2009-2014 Red Hat, Inc. All rights reserved.
Packit Service 501009
 * Copyright (C) 2005, 2006, 2010-2013 Michael Holzheu, IBM Corporation
Packit Service 501009
 *
Packit Service 501009
 * This program is free software; you can redistribute it and/or modify
Packit Service 501009
 * it under the terms of the GNU General Public License as published by
Packit Service 501009
 * the Free Software Foundation; either version 2 of the License, or
Packit Service 501009
 * (at your option) any later version.
Packit Service 501009
 *
Packit Service 501009
 * This program is distributed in the hope that it will be useful,
Packit Service 501009
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 501009
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service 501009
 * GNU General Public License for more details.
Packit Service 501009
 */
Packit Service 501009
#ifdef S390X
Packit Service 501009
#include <elf.h>
Packit Service 501009
#include "defs.h"
Packit Service 501009
#include "netdump.h"
Packit Service 501009
Packit Service 501009
#define S390X_WORD_SIZE   8
Packit Service 501009
Packit Service 501009
#define S390X_PAGE_BASE_MASK     (~((1ULL<<12)-1))
Packit Service 501009
Packit Service 501009
/* Flags used in entries of page dirs and page tables.
Packit Service 501009
 */
Packit Service 501009
#define S390X_PTE_FLAG_BITS     0xfffULL /* Page table entry flag bits */
Packit Service 501009
#define S390X_PAGE_PRESENT      0x001ULL /* set: loaded in physical memory
Packit Service 501009
                                          * clear: not loaded in physical mem */
Packit Service 501009
#define S390X_PAGE_RO           0x200ULL /* HW read-only */
Packit Service 501009
#define S390X_PAGE_INVALID      0x400ULL /* HW invalid */
Packit Service 501009
#define S390X_PAGE_INVALID_MASK 0x601ULL /* for linux 2.6 */
Packit Service 501009
#define S390X_PAGE_INVALID_NONE 0x401ULL /* for linux 2.6 */
Packit Service 501009
Packit Service 501009
/* bits 52, 55 must contain zeroes in a pte */
Packit Service 501009
#define S390X_PTE_INVALID_MASK  0x900ULL
Packit Service 501009
#define S390X_PTE_INVALID(x) ((x) & S390X_PTE_INVALID_MASK)
Packit Service 501009
Packit Service 501009
#define INT_STACK_SIZE    STACKSIZE() // can be 8192 or 16384
Packit Service 501009
#define KERNEL_STACK_SIZE STACKSIZE() // can be 8192 or 16384
Packit Service 501009
Packit Service 501009
#define LOWCORE_SIZE 8192
Packit Service 501009
#define VX_SA_SIZE (32 * 16)
Packit Service 501009
Packit Service 501009
#define S390X_PSW_MASK_PSTATE	0x0001000000000000UL
Packit Service 501009
Packit Service 501009
#define S390X_LC_VMCORE_INFO	0xe0c
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Flags for Region and Segment table entries.
Packit Service 501009
 */
Packit Service 501009
#define S390X_RTE_FLAG_BITS_FC0    0xfffULL
Packit Service 501009
#define S390X_RTE_FLAG_BITS_FC1    0x7fffffffULL
Packit Service 501009
#define S390X_RTE_TL           0x3ULL
Packit Service 501009
#define S390X_RTE_TL_10        0x2ULL
Packit Service 501009
#define S390X_RTE_TL_01        0x1ULL
Packit Service 501009
#define S390X_RTE_TT           0xcULL
Packit Service 501009
#define S390X_RTE_TT_10        0x8ULL
Packit Service 501009
#define S390X_RTE_TT_01        0x4ULL
Packit Service 501009
#define S390X_RTE_CR           0x10ULL
Packit Service 501009
#define S390X_RTE_I            0x20ULL
Packit Service 501009
#define S390X_RTE_TF           0xc0ULL
Packit Service 501009
#define S390X_RTE_TF_10        0x80ULL
Packit Service 501009
#define S390X_RTE_TF_01        0x40ULL
Packit Service 501009
#define S390X_RTE_P            0x200ULL
Packit Service 501009
#define S390X_RTE_FC           0x400ULL
Packit Service 501009
#define S390X_RTE_F            0x800ULL
Packit Service 501009
#define S390X_RTE_ACC          0xf000ULL
Packit Service 501009
#define S390X_RTE_ACC_1000     0x8000ULL
Packit Service 501009
#define S390X_RTE_ACC_0100     0x4000ULL
Packit Service 501009
#define S390X_RTE_ACC_0010     0x2000ULL
Packit Service 501009
#define S390X_RTE_ACC_0001     0x1000ULL
Packit Service 501009
#define S390X_RTE_AV           0x10000ULL
Packit Service 501009
Packit Service 501009
#define S390X_STE_FLAG_BITS_FC0    0x7ffULL
Packit Service 501009
#define S390X_STE_FLAG_BITS_FC1    0xfffffULL
Packit Service 501009
#define S390X_STE_TT           0xcULL
Packit Service 501009
#define S390X_STE_TT_10        0x8ULL
Packit Service 501009
#define S390X_STE_TT_01        0x4ULL
Packit Service 501009
#define S390X_STE_CS           0x10ULL
Packit Service 501009
#define S390X_STE_I            0x20ULL
Packit Service 501009
#define S390X_STE_P            0x200ULL
Packit Service 501009
#define S390X_STE_FC           0x400ULL
Packit Service 501009
#define S390X_STE_F            0x800ULL
Packit Service 501009
#define S390X_STE_ACC          0xf000ULL
Packit Service 501009
#define S390X_STE_ACC_1000     0x8000ULL
Packit Service 501009
#define S390X_STE_ACC_0100     0x4000ULL
Packit Service 501009
#define S390X_STE_ACC_0010     0x2000ULL
Packit Service 501009
#define S390X_STE_ACC_0001     0x1000ULL
Packit Service 501009
#define S390X_STE_AV           0x10000ULL
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * S390x prstatus ELF Note
Packit Service 501009
 */
Packit Service 501009
struct s390x_nt_prstatus {
Packit Service 501009
	uint8_t		pad1[32];
Packit Service 501009
	uint32_t	pr_pid;
Packit Service 501009
	uint8_t		pad2[76];
Packit Service 501009
	uint64_t	psw[2];
Packit Service 501009
	uint64_t	gprs[16];
Packit Service 501009
	uint32_t	acrs[16];
Packit Service 501009
	uint64_t	orig_gpr2;
Packit Service 501009
	uint32_t	pr_fpvalid;
Packit Service 501009
	uint8_t		pad3[4];
Packit Service 501009
} __attribute__ ((packed));
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * S390x floating point register ELF Note
Packit Service 501009
 */
Packit Service 501009
#ifndef NT_FPREGSET
Packit Service 501009
#define NT_FPREGSET 0x2
Packit Service 501009
#endif
Packit Service 501009
Packit Service 501009
struct s390x_nt_fpregset {
Packit Service 501009
	uint32_t	fpc;
Packit Service 501009
	uint32_t	pad;
Packit Service 501009
	uint64_t	fprs[16];
Packit Service 501009
} __attribute__ ((packed));
Packit Service 501009
Packit Service 501009
struct s390x_vxrs {
Packit Service 501009
	uint64_t	low;
Packit Service 501009
	uint64_t	high;
Packit Service 501009
} __attribute__ ((packed));
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * s390x CPU info
Packit Service 501009
 */
Packit Service 501009
struct s390x_cpu
Packit Service 501009
{
Packit Service 501009
	uint64_t	gprs[16];
Packit Service 501009
	uint64_t	ctrs[16];
Packit Service 501009
	uint32_t	acrs[16];
Packit Service 501009
	uint64_t	fprs[16];
Packit Service 501009
	uint32_t	fpc;
Packit Service 501009
	uint64_t	psw[2];
Packit Service 501009
	uint32_t	prefix;
Packit Service 501009
	uint64_t	timer;
Packit Service 501009
	uint64_t	todcmp;
Packit Service 501009
	uint32_t	todpreg;
Packit Service 501009
	uint64_t		vxrs_low[16];
Packit Service 501009
	struct s390x_vxrs	vxrs_high[16];
Packit Service 501009
};
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * declarations of static functions
Packit Service 501009
 */
Packit Service 501009
static void s390x_print_lowcore(char*, struct bt_info*,int);
Packit Service 501009
static int s390x_kvtop(struct task_context *, ulong, physaddr_t *, int);
Packit Service 501009
static int s390x_uvtop(struct task_context *, ulong, physaddr_t *, int);
Packit Service 501009
static int s390x_vtop(unsigned long, ulong, physaddr_t*, int);
Packit Service 501009
static ulong s390x_vmalloc_start(void);
Packit Service 501009
static int s390x_is_task_addr(ulong);
Packit Service 501009
static int s390x_verify_symbol(const char *, ulong, char type);
Packit Service 501009
static ulong s390x_get_task_pgd(ulong);
Packit Service 501009
static int s390x_translate_pte(ulong, void *, ulonglong);
Packit Service 501009
static ulong s390x_processor_speed(void);
Packit Service 501009
static int s390x_eframe_search(struct bt_info *);
Packit Service 501009
static void s390x_back_trace_cmd(struct bt_info *);
Packit Service 501009
static void s390x_get_stack_frame(struct bt_info *, ulong *, ulong *);
Packit Service 501009
static int s390x_dis_filter(ulong, char *, unsigned int);
Packit Service 501009
static void s390x_cmd_mach(void);
Packit Service 501009
static int s390x_get_smp_cpus(void);
Packit Service 501009
static void s390x_display_machine_stats(void);
Packit Service 501009
static void s390x_dump_line_number(ulong);
Packit Service 501009
static struct line_number_hook s390x_line_number_hooks[];
Packit Service 501009
static int s390x_is_uvaddr(ulong, struct task_context *);
Packit Service 501009
static int s390x_get_kvaddr_ranges(struct vaddr_range *);
Packit Service 501009
static int set_s390x_max_physmem_bits(void);
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * struct lowcore name (old: "_lowcore", new: "lowcore")
Packit Service 501009
 */
Packit Service 501009
static char *lc_struct;
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Read a unsigned long value from address
Packit Service 501009
 */
Packit Service 501009
static unsigned long readmem_ul(unsigned long addr)
Packit Service 501009
{
Packit Service 501009
	unsigned long rc;
Packit Service 501009
Packit Service 501009
	readmem(addr, KVADDR, &rc, sizeof(rc), "readmem_ul", FAULT_ON_ERROR);
Packit Service 501009
	return rc;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Print hex data
Packit Service 501009
 */
Packit Service 501009
static void print_hex_buf(void *buf, int len, int cols, char *tag)
Packit Service 501009
{
Packit Service 501009
	int j, first = 1;
Packit Service 501009
Packit Service 501009
	for (j = 0; j < len; j += 8) {
Packit Service 501009
		if (j % (cols * 8) == 0) {
Packit Service 501009
			if (first)
Packit Service 501009
				first = 0;
Packit Service 501009
			else
Packit Service 501009
				fprintf(fp, "\n");
Packit Service 501009
			fprintf(fp, "%s", tag);
Packit Service 501009
		}
Packit Service 501009
		fprintf(fp, "%#018lx ", *((unsigned long *)(buf + j)));
Packit Service 501009
	}
Packit Service 501009
	if (len)
Packit Service 501009
		fprintf(fp, "\n");
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Initialize member offsets
Packit Service 501009
 */
Packit Service 501009
static void s390x_offsets_init(void)
Packit Service 501009
{
Packit Service 501009
	if (STRUCT_EXISTS("lowcore"))
Packit Service 501009
		lc_struct = "lowcore";
Packit Service 501009
	else
Packit Service 501009
		lc_struct = "_lowcore";
Packit Service 501009
Packit Service 501009
	if (MEMBER_EXISTS(lc_struct, "st_status_fixed_logout"))
Packit Service 501009
		MEMBER_OFFSET_INIT(s390_lowcore_psw_save_area, lc_struct,
Packit Service 501009
				   "st_status_fixed_logout");
Packit Service 501009
	else
Packit Service 501009
		MEMBER_OFFSET_INIT(s390_lowcore_psw_save_area, lc_struct,
Packit Service 501009
				   "psw_save_area");
Packit Service 501009
	if (!STRUCT_EXISTS("stack_frame")) {
Packit Service 501009
		ASSIGN_OFFSET(s390_stack_frame_back_chain) = 0;
Packit Service 501009
		ASSIGN_OFFSET(s390_stack_frame_r14) = 112;
Packit Service 501009
		ASSIGN_SIZE(s390_stack_frame) = 160;
Packit Service 501009
	} else {
Packit Service 501009
		ASSIGN_OFFSET(s390_stack_frame_back_chain) =
Packit Service 501009
			MEMBER_OFFSET("stack_frame", "back_chain");
Packit Service 501009
		ASSIGN_OFFSET(s390_stack_frame_r14) =
Packit Service 501009
			MEMBER_OFFSET("stack_frame", "gprs") + 8 * 8;
Packit Service 501009
		ASSIGN_SIZE(s390_stack_frame) = STRUCT_SIZE("stack_frame");
Packit Service 501009
	}
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 *  MAX_PHYSMEM_BITS is 42 on older kernels, and 46 on newer kernels.
Packit Service 501009
 */
Packit Service 501009
static int
Packit Service 501009
set_s390x_max_physmem_bits(void)
Packit Service 501009
{
Packit Service 501009
	int array_len, dimension;
Packit Service 501009
	char *string;
Packit Service 501009
Packit Service 501009
	if ((string = pc->read_vmcoreinfo("NUMBER(MAX_PHYSMEM_BITS)"))) {
Packit Service 501009
		machdep->max_physmem_bits = atol(string);
Packit Service 501009
		free(string);
Packit Service 501009
		return TRUE;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	machdep->max_physmem_bits = _MAX_PHYSMEM_BITS_OLD;
Packit Service 501009
Packit Service 501009
	if (!kernel_symbol_exists("mem_section"))
Packit Service 501009
		return TRUE;
Packit Service 501009
Packit Service 501009
	/*
Packit Service 501009
	 *  The mem_section was changed to be a pointer in 4.15, so it's 
Packit Service 501009
	 *  guaranteed to be a newer kernel.
Packit Service 501009
	 */
Packit Service 501009
	if (get_symbol_type("mem_section", NULL, NULL) == TYPE_CODE_PTR) {
Packit Service 501009
		machdep->max_physmem_bits = _MAX_PHYSMEM_BITS_NEW;
Packit Service 501009
		return TRUE;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	if (!(array_len = get_array_length("mem_section", &dimension, 0)))
Packit Service 501009
		return FALSE;
Packit Service 501009
Packit Service 501009
	/*
Packit Service 501009
	 * !CONFIG_SPARSEMEM_EXTREME
Packit Service 501009
	 */
Packit Service 501009
	if (dimension) {
Packit Service 501009
		machdep->max_physmem_bits = _MAX_PHYSMEM_BITS_OLD;
Packit Service 501009
		if (array_len == (NR_MEM_SECTIONS() / _SECTIONS_PER_ROOT()))
Packit Service 501009
			return TRUE;
Packit Service 501009
Packit Service 501009
		machdep->max_physmem_bits = _MAX_PHYSMEM_BITS_NEW;
Packit Service 501009
		if (array_len == (NR_MEM_SECTIONS() / _SECTIONS_PER_ROOT()))
Packit Service 501009
			return TRUE;
Packit Service 501009
Packit Service 501009
		return FALSE;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	/*
Packit Service 501009
	 * CONFIG_SPARSEMEM_EXTREME
Packit Service 501009
	 */
Packit Service 501009
	machdep->max_physmem_bits = _MAX_PHYSMEM_BITS_OLD;
Packit Service 501009
	if (array_len == (NR_MEM_SECTIONS() / _SECTIONS_PER_ROOT_EXTREME()))
Packit Service 501009
		return TRUE;
Packit Service 501009
Packit Service 501009
	machdep->max_physmem_bits = _MAX_PHYSMEM_BITS_NEW;
Packit Service 501009
	if (array_len == (NR_MEM_SECTIONS() / _SECTIONS_PER_ROOT_EXTREME()))
Packit Service 501009
		return TRUE;
Packit Service 501009
Packit Service 501009
	return FALSE;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static struct s390x_cpu *s390x_cpu_vec;
Packit Service 501009
static int s390x_cpu_cnt;
Packit Service 501009
/*
Packit Service 501009
 * Return s390x CPU data for backtrace
Packit Service 501009
 */
Packit Service 501009
static struct s390x_cpu *s390x_cpu_get(struct bt_info *bt)
Packit Service 501009
{
Packit Service 501009
	unsigned int cpu = bt->tc->processor;
Packit Service 501009
	unsigned long lowcore_ptr, prefix;
Packit Service 501009
	unsigned int i;
Packit Service 501009
Packit Service 501009
	lowcore_ptr = symbol_value("lowcore_ptr");
Packit Service 501009
	readmem(lowcore_ptr + cpu * sizeof(long), KVADDR,
Packit Service 501009
		&prefix, sizeof(long), "lowcore_ptr", FAULT_ON_ERROR);
Packit Service 501009
	for (i = 0; i < s390x_cpu_cnt; i++) {
Packit Service 501009
		if (s390x_cpu_vec[i].prefix == prefix)
Packit Service 501009
			return &s390x_cpu_vec[i];
Packit Service 501009
	}
Packit Service 501009
	error(FATAL, "cannot determine CPU for task: %lx\n", bt->task);
Packit Service 501009
	return NULL;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * ELF core dump fuctions for storing CPU data
Packit Service 501009
 */
Packit Service 501009
static void s390x_elf_nt_prstatus_add(struct s390x_cpu *cpu,
Packit Service 501009
				      struct s390x_nt_prstatus *prstatus)
Packit Service 501009
{
Packit Service 501009
	memcpy(&cpu->psw, &prstatus->psw, sizeof(cpu->psw));
Packit Service 501009
	memcpy(&cpu->gprs, &prstatus->gprs, sizeof(cpu->gprs));
Packit Service 501009
	memcpy(&cpu->acrs, &prstatus->acrs, sizeof(cpu->acrs));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static void s390x_elf_nt_fpregset_add(struct s390x_cpu *cpu,
Packit Service 501009
				      struct s390x_nt_fpregset *fpregset)
Packit Service 501009
{
Packit Service 501009
	memcpy(&cpu->fpc, &fpregset->fpc, sizeof(cpu->fpc));
Packit Service 501009
	memcpy(&cpu->fprs, &fpregset->fprs, sizeof(cpu->fprs));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static void s390x_elf_nt_timer_add(struct s390x_cpu *cpu, void *desc)
Packit Service 501009
{
Packit Service 501009
	memcpy(&cpu->timer, desc, sizeof(cpu->timer));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static void s390x_elf_nt_todcmp_add(struct s390x_cpu *cpu, void *desc)
Packit Service 501009
{
Packit Service 501009
	memcpy(&cpu->todcmp, desc, sizeof(cpu->todcmp));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static void s390x_elf_nt_todpreg_add(struct s390x_cpu *cpu, void *desc)
Packit Service 501009
{
Packit Service 501009
	memcpy(&cpu->todpreg, desc, sizeof(cpu->todpreg));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static void s390x_elf_nt_ctrs_add(struct s390x_cpu *cpu, void *desc)
Packit Service 501009
{
Packit Service 501009
	memcpy(&cpu->ctrs, desc, sizeof(cpu->ctrs));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static void s390x_elf_nt_prefix_add(struct s390x_cpu *cpu, void *desc)
Packit Service 501009
{
Packit Service 501009
	memcpy(&cpu->prefix, desc, sizeof(cpu->prefix));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static void s390x_elf_nt_vxrs_low_add(struct s390x_cpu *cpu, void *desc)
Packit Service 501009
{
Packit Service 501009
	memcpy(&cpu->vxrs_low, desc, sizeof(cpu->vxrs_low));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static void s390x_elf_nt_vxrs_high_add(struct s390x_cpu *cpu, void *desc)
Packit Service 501009
{
Packit Service 501009
	memcpy(&cpu->vxrs_high, desc, sizeof(cpu->vxrs_high));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static void *get_elf_note_desc(Elf64_Nhdr *note)
Packit Service 501009
{
Packit Service 501009
	void *ptr = note;
Packit Service 501009
Packit Service 501009
	return ptr + roundup(sizeof(*note) + note->n_namesz, 4);
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static void s390x_elf_note_add(int elf_cpu_nr, void *note_ptr)
Packit Service 501009
{
Packit Service 501009
	Elf64_Nhdr *note = note_ptr;
Packit Service 501009
	struct s390x_cpu *cpu;
Packit Service 501009
	void *desc;
Packit Service 501009
Packit Service 501009
	desc = get_elf_note_desc(note);
Packit Service 501009
	if (elf_cpu_nr != s390x_cpu_cnt) {
Packit Service 501009
		s390x_cpu_cnt++;
Packit Service 501009
		s390x_cpu_vec = realloc(s390x_cpu_vec,
Packit Service 501009
					s390x_cpu_cnt * sizeof(*s390x_cpu_vec));
Packit Service 501009
		if (!s390x_cpu_vec)
Packit Service 501009
			error(FATAL, "cannot malloc cpu space.");
Packit Service 501009
	}
Packit Service 501009
	cpu = &s390x_cpu_vec[s390x_cpu_cnt - 1];
Packit Service 501009
	switch (note->n_type) {
Packit Service 501009
	case NT_PRSTATUS:
Packit Service 501009
		s390x_elf_nt_prstatus_add(cpu, desc);
Packit Service 501009
		break;
Packit Service 501009
	case NT_FPREGSET:
Packit Service 501009
		s390x_elf_nt_fpregset_add(cpu, desc);
Packit Service 501009
		break;
Packit Service 501009
	case NT_S390_TIMER:
Packit Service 501009
		s390x_elf_nt_timer_add(cpu, desc);
Packit Service 501009
		break;
Packit Service 501009
	case NT_S390_TODCMP:
Packit Service 501009
		s390x_elf_nt_todcmp_add(cpu, desc);
Packit Service 501009
		break;
Packit Service 501009
	case NT_S390_TODPREG:
Packit Service 501009
		s390x_elf_nt_todpreg_add(cpu, desc);
Packit Service 501009
		break;
Packit Service 501009
	case NT_S390_CTRS:
Packit Service 501009
		s390x_elf_nt_ctrs_add(cpu, desc);
Packit Service 501009
		break;
Packit Service 501009
	case NT_S390_PREFIX:
Packit Service 501009
		s390x_elf_nt_prefix_add(cpu, desc);
Packit Service 501009
		break;
Packit Service 501009
	case NT_S390_VXRS_LOW:
Packit Service 501009
		s390x_elf_nt_vxrs_low_add(cpu, desc);
Packit Service 501009
		break;
Packit Service 501009
	case NT_S390_VXRS_HIGH:
Packit Service 501009
		s390x_elf_nt_vxrs_high_add(cpu, desc);
Packit Service 501009
		break;
Packit Service 501009
	}
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static void s390x_process_elf_notes(void *note_ptr, unsigned long size_note)
Packit Service 501009
{
Packit Service 501009
	Elf64_Nhdr *note = NULL;
Packit Service 501009
	size_t tot, len;
Packit Service 501009
	static int num_prstatus_notes = 0;
Packit Service 501009
Packit Service 501009
	for (tot = 0; tot < size_note; tot += len) {
Packit Service 501009
		note = note_ptr + tot;
Packit Service 501009
Packit Service 501009
		if (note->n_type == NT_PRSTATUS)
Packit Service 501009
			num_prstatus_notes++;
Packit Service 501009
Packit Service 501009
		machdep->dumpfile_init(num_prstatus_notes, note);
Packit Service 501009
Packit Service 501009
		len = sizeof(Elf64_Nhdr);
Packit Service 501009
		len = roundup(len + note->n_namesz, 4);
Packit Service 501009
		len = roundup(len + note->n_descsz, 4);
Packit Service 501009
	}
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static void s390x_check_live(void)
Packit Service 501009
{
Packit Service 501009
	unsigned long long live_magic;
Packit Service 501009
Packit Service 501009
	readmem(0, KVADDR, &live_magic, sizeof(live_magic), "live_magic",
Packit Service 501009
		RETURN_ON_ERROR | QUIET);
Packit Service 501009
Packit Service 501009
	if (live_magic == 0x4c49564544554d50ULL)
Packit Service 501009
		pc->flags2 |= LIVE_DUMP;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static char *
Packit Service 501009
vmcoreinfo_read_string_s390x(const char *vmcoreinfo, const char *key)
Packit Service 501009
{
Packit Service 501009
	char *value_string = NULL;
Packit Service 501009
	size_t value_length;
Packit Service 501009
	char keybuf[128];
Packit Service 501009
	char *p1, *p2;
Packit Service 501009
Packit Service 501009
	sprintf(keybuf, "%s=", key);
Packit Service 501009
Packit Service 501009
	if ((p1 = strstr(vmcoreinfo, keybuf))) {
Packit Service 501009
		p2 = p1 + strlen(keybuf);
Packit Service 501009
		p1 = strstr(p2, "\n");
Packit Service 501009
		value_length = p1-p2;
Packit Service 501009
		value_string = calloc(value_length + 1, sizeof(char));
Packit Service 501009
		strncpy(value_string, p2, value_length);
Packit Service 501009
		value_string[value_length] = NULLCHAR;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	return value_string;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Check the value in well-known lowcore location and process it as either
Packit Service 501009
 * an explicit KASLR offset (early dump case) or as vmcoreinfo pointer to
Packit Service 501009
 * read the relocated _stext symbol value (important for s390 and lkcd dump
Packit Service 501009
 * formats).
Packit Service 501009
 */
Packit Service 501009
static void s390x_check_kaslr(void)
Packit Service 501009
{
Packit Service 501009
	char *_stext_string, *vmcoreinfo;
Packit Service 501009
	Elf64_Nhdr note;
Packit Service 501009
	char str[128];
Packit Service 501009
	ulong addr;
Packit Service 501009
Packit Service 501009
	/* Read the value from well-known lowcore location*/
Packit Service 501009
	if (!readmem(S390X_LC_VMCORE_INFO, PHYSADDR, &addr,
Packit Service 501009
		    sizeof(addr), "s390x vmcoreinfo ptr",
Packit Service 501009
		    QUIET|RETURN_ON_ERROR))
Packit Service 501009
		return;
Packit Service 501009
	if (addr == 0)
Packit Service 501009
		return;
Packit Service 501009
	/* Check for explicit kaslr offset flag */
Packit Service 501009
	if (addr & 0x1UL) {
Packit Service 501009
		/* Drop the last bit to get an offset value */
Packit Service 501009
		addr &= ~(0x1UL);
Packit Service 501009
		/* Make sure the offset is aligned by 0x1000 */
Packit Service 501009
		if (addr && !(addr & 0xfff)) {
Packit Service 501009
					kt->relocate = addr * (-1);
Packit Service 501009
					kt->flags |= RELOC_SET;
Packit Service 501009
					kt->flags2 |= KASLR;
Packit Service 501009
		}
Packit Service 501009
		return;
Packit Service 501009
	}
Packit Service 501009
	/* Use the addr value as vmcoreinfo pointer */
Packit Service 501009
	if (!readmem(addr, PHYSADDR, &note,
Packit Service 501009
		     sizeof(note), "Elf64_Nhdr vmcoreinfo",
Packit Service 501009
		     QUIET|RETURN_ON_ERROR))
Packit Service 501009
		return;
Packit Service 501009
	memset(str, 0, sizeof(str));
Packit Service 501009
	if (!readmem(addr + sizeof(note), PHYSADDR, str,
Packit Service 501009
		     note.n_namesz, "VMCOREINFO",
Packit Service 501009
		     QUIET|RETURN_ON_ERROR))
Packit Service 501009
		return;
Packit Service 501009
	if (memcmp(str, "VMCOREINFO", sizeof("VMCOREINFO")) != 0)
Packit Service 501009
		return;
Packit Service 501009
	if ((vmcoreinfo = malloc(note.n_descsz + 1)) == NULL) {
Packit Service 501009
		error(INFO, "s390x_check_kaslr: cannot malloc vmcoreinfo buffer\n");
Packit Service 501009
		return;
Packit Service 501009
	}
Packit Service 501009
	addr = addr + sizeof(note) + note.n_namesz + 1;
Packit Service 501009
	if (!readmem(addr, PHYSADDR, vmcoreinfo,
Packit Service 501009
		     note.n_descsz, "s390x vmcoreinfo",
Packit Service 501009
		     QUIET|RETURN_ON_ERROR)) {
Packit Service 501009
		free(vmcoreinfo);
Packit Service 501009
		return;
Packit Service 501009
	}
Packit Service 501009
	vmcoreinfo[note.n_descsz] = NULLCHAR;
Packit Service 501009
	/*
Packit Service 501009
	 * Read relocated _stext symbol value and store it in the kernel_table
Packit Service 501009
	 * for further processing within derive_kaslr_offset().
Packit Service 501009
	 */
Packit Service 501009
	if ((_stext_string = vmcoreinfo_read_string_s390x(vmcoreinfo,
Packit Service 501009
							  "SYMBOL(_stext)"))) {
Packit Service 501009
		kt->vmcoreinfo._stext_SYMBOL = htol(_stext_string,
Packit Service 501009
						    RETURN_ON_ERROR, NULL);
Packit Service 501009
		free(_stext_string);
Packit Service 501009
	}
Packit Service 501009
	free(vmcoreinfo);
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 *  Do all necessary machine-specific setup here.  This is called several
Packit Service 501009
 *  times during initialization.
Packit Service 501009
 */
Packit Service 501009
void
Packit Service 501009
s390x_init(int when)
Packit Service 501009
{
Packit Service 501009
	switch (when)
Packit Service 501009
	{
Packit Service 501009
	case SETUP_ENV:
Packit Service 501009
		machdep->dumpfile_init = s390x_elf_note_add;
Packit Service 501009
		machdep->process_elf_notes = s390x_process_elf_notes;
Packit Service 501009
		break;
Packit Service 501009
	case PRE_SYMTAB:
Packit Service 501009
		machdep->verify_symbol = s390x_verify_symbol;
Packit Service 501009
		if (pc->flags & KERNEL_DEBUG_QUERY)
Packit Service 501009
			return;
Packit Service 501009
		machdep->pagesize = memory_page_size();
Packit Service 501009
		machdep->pageshift = ffs(machdep->pagesize) - 1;
Packit Service 501009
		machdep->pageoffset = machdep->pagesize - 1;
Packit Service 501009
		machdep->pagemask = ~((ulonglong)machdep->pageoffset);
Packit Service 501009
		// machdep->stacksize = KERNEL_STACK_SIZE;
Packit Service 501009
		if ((machdep->pgd = (char *)malloc(SEGMENT_TABLE_SIZE)) == NULL)
Packit Service 501009
			error(FATAL, "cannot malloc pgd space.");
Packit Service 501009
		machdep->pmd = machdep->pgd;
Packit Service 501009
		if ((machdep->ptbl = (char *)malloc(PAGESIZE())) == NULL)
Packit Service 501009
			error(FATAL, "cannot malloc ptbl space.");
Packit Service 501009
		machdep->last_pgd_read = 0;
Packit Service 501009
		machdep->last_pmd_read = 0;
Packit Service 501009
		machdep->last_ptbl_read = 0;
Packit Service 501009
		machdep->verify_paddr = generic_verify_paddr;
Packit Service 501009
		machdep->get_kvaddr_ranges = s390x_get_kvaddr_ranges;
Packit Service 501009
		machdep->ptrs_per_pgd = PTRS_PER_PGD;
Packit Service 501009
		if (DUMPFILE() && !(kt->flags & RELOC_SET))
Packit Service 501009
			s390x_check_kaslr();
Packit Service 501009
		break;
Packit Service 501009
Packit Service 501009
	case PRE_GDB:
Packit Service 501009
		machdep->kvbase = 0;
Packit Service 501009
		machdep->identity_map_base = 0;
Packit Service 501009
		machdep->is_kvaddr =  generic_is_kvaddr;
Packit Service 501009
		machdep->is_uvaddr =  s390x_is_uvaddr;
Packit Service 501009
		machdep->eframe_search = s390x_eframe_search;
Packit Service 501009
		machdep->back_trace = s390x_back_trace_cmd;
Packit Service 501009
		machdep->processor_speed = s390x_processor_speed;
Packit Service 501009
		machdep->uvtop = s390x_uvtop;
Packit Service 501009
		machdep->kvtop = s390x_kvtop;
Packit Service 501009
		machdep->get_task_pgd = s390x_get_task_pgd;
Packit Service 501009
		machdep->get_stack_frame = s390x_get_stack_frame;
Packit Service 501009
		machdep->get_stackbase = generic_get_stackbase;
Packit Service 501009
		machdep->get_stacktop = generic_get_stacktop;
Packit Service 501009
		machdep->translate_pte = s390x_translate_pte;
Packit Service 501009
		machdep->memory_size = generic_memory_size;
Packit Service 501009
		machdep->is_task_addr = s390x_is_task_addr;
Packit Service 501009
		machdep->dis_filter = s390x_dis_filter;
Packit Service 501009
		machdep->cmd_mach = s390x_cmd_mach;
Packit Service 501009
		machdep->get_smp_cpus = s390x_get_smp_cpus;
Packit Service 501009
		machdep->line_number_hooks = s390x_line_number_hooks;
Packit Service 501009
		machdep->value_to_symbol = generic_machdep_value_to_symbol;
Packit Service 501009
		machdep->init_kernel_pgd = NULL;
Packit Service 501009
		vt->flags |= COMMON_VADDR;
Packit Service 501009
		s390x_check_live();
Packit Service 501009
		break;
Packit Service 501009
Packit Service 501009
	case POST_GDB:
Packit Service 501009
		if (symbol_exists("irq_desc"))
Packit Service 501009
			ARRAY_LENGTH_INIT(machdep->nr_irqs, irq_desc,
Packit Service 501009
				"irq_desc", NULL, 0);
Packit Service 501009
		else if (kernel_symbol_exists("nr_irqs"))
Packit Service 501009
			get_symbol_data("nr_irqs", sizeof(unsigned int),
Packit Service 501009
				&machdep->nr_irqs);
Packit Service 501009
		else
Packit Service 501009
			machdep->nr_irqs = 0;
Packit Service 501009
Packit Service 501009
		machdep->vmalloc_start = s390x_vmalloc_start;
Packit Service 501009
		machdep->dump_irq = generic_dump_irq;
Packit Service 501009
		if (!machdep->hz)
Packit Service 501009
			machdep->hz = HZ;
Packit Service 501009
		machdep->section_size_bits = _SECTION_SIZE_BITS;
Packit Service 501009
		if (!set_s390x_max_physmem_bits())
Packit Service 501009
			error(WARNING, "cannot determine MAX_PHYSMEM_BITS\n");
Packit Service 501009
		s390x_offsets_init();
Packit Service 501009
		break;
Packit Service 501009
Packit Service 501009
	case POST_INIT:
Packit Service 501009
		break;
Packit Service 501009
	}
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Dump machine dependent information
Packit Service 501009
 */
Packit Service 501009
void
Packit Service 501009
s390x_dump_machdep_table(ulong arg)
Packit Service 501009
{
Packit Service 501009
	int others; 
Packit Service 501009
 
Packit Service 501009
	others = 0;
Packit Service 501009
	fprintf(fp, "              flags: %lx (", machdep->flags);
Packit Service 501009
	if (machdep->flags & KSYMS_START)
Packit Service 501009
		fprintf(fp, "%sKSYMS_START", others++ ? "|" : "");
Packit Service 501009
	fprintf(fp, ")\n");
Packit Service 501009
Packit Service 501009
	fprintf(fp, "             kvbase: %lx\n", machdep->kvbase);
Packit Service 501009
	fprintf(fp, "  identity_map_base: %lx\n", machdep->kvbase);
Packit Service 501009
	fprintf(fp, "           pagesize: %d\n", machdep->pagesize);
Packit Service 501009
	fprintf(fp, "          pageshift: %d\n", machdep->pageshift);
Packit Service 501009
	fprintf(fp, "           pagemask: %llx\n", machdep->pagemask);
Packit Service 501009
	fprintf(fp, "         pageoffset: %lx\n", machdep->pageoffset);
Packit Service 501009
	fprintf(fp, "          stacksize: %ld\n", machdep->stacksize);
Packit Service 501009
	fprintf(fp, "                 hz: %d\n", machdep->hz);
Packit Service 501009
	fprintf(fp, "                mhz: %ld\n", machdep->mhz);
Packit Service 501009
	fprintf(fp, "            memsize: %lld (0x%llx)\n", 
Packit Service 501009
		(unsigned long long)machdep->memsize,
Packit Service 501009
		(unsigned long long)machdep->memsize);
Packit Service 501009
	fprintf(fp, "               bits: %d\n", machdep->bits);
Packit Service 501009
	fprintf(fp, "            nr_irqs: %d\n", machdep->nr_irqs);
Packit Service 501009
	fprintf(fp, "      eframe_search: s390x_eframe_search()\n");
Packit Service 501009
	fprintf(fp, "         back_trace: s390x_back_trace_cmd()\n");
Packit Service 501009
	fprintf(fp, "    processor_speed: s390x_processor_speed()\n");
Packit Service 501009
	fprintf(fp, "              uvtop: s390x_uvtop()\n");
Packit Service 501009
	fprintf(fp, "              kvtop: s390x_kvtop()\n");
Packit Service 501009
	fprintf(fp, "       get_task_pgd: s390x_get_task_pgd()\n");
Packit Service 501009
	fprintf(fp, "           dump_irq: generic_dump_irq()\n");
Packit Service 501009
	fprintf(fp, "    get_stack_frame: s390x_get_stack_frame()\n");
Packit Service 501009
	fprintf(fp, "      get_stackbase: generic_get_stackbase()\n");
Packit Service 501009
	fprintf(fp, "       get_stacktop: generic_get_stacktop()\n");
Packit Service 501009
	fprintf(fp, "      translate_pte: s390x_translate_pte()\n");
Packit Service 501009
	fprintf(fp, "        memory_size: generic_memory_size()\n");
Packit Service 501009
	fprintf(fp, "      vmalloc_start: s390x_vmalloc_start()\n");
Packit Service 501009
	fprintf(fp, "       is_task_addr: s390x_is_task_addr()\n");
Packit Service 501009
	fprintf(fp, "      verify_symbol: s390x_verify_symbol()\n");
Packit Service 501009
	fprintf(fp, "         dis_filter: s390x_dis_filter()\n");
Packit Service 501009
	fprintf(fp, "           cmd_mach: s390x_cmd_mach()\n");
Packit Service 501009
	fprintf(fp, "       get_smp_cpus: s390x_get_smp_cpus()\n");
Packit Service 501009
	fprintf(fp, "          is_kvaddr: generic_is_kvaddr()\n");
Packit Service 501009
	fprintf(fp, "          is_uvaddr: s390x_is_uvaddr()\n");
Packit Service 501009
	fprintf(fp, "       verify_paddr: generic_verify_paddr()\n");
Packit Service 501009
	fprintf(fp, "  get_kvaddr_ranges: s390x_get_kvaddr_ranges()\n");
Packit Service 501009
	fprintf(fp, "    init_kernel_pgd: NULL\n");
Packit Service 501009
	fprintf(fp, "    value_to_symbol: generic_machdep_value_to_symbol()\n");
Packit Service 501009
	fprintf(fp, "      dumpfile_init: s390x_elf_note_add()\n");
Packit Service 501009
	fprintf(fp, "  process_elf_notes: s390x_process_elf_notes()\n");
Packit Service 501009
	fprintf(fp, "  line_number_hooks: s390x_line_number_hooks\n");
Packit Service 501009
	fprintf(fp, "      last_pgd_read: %lx\n", machdep->last_pgd_read);
Packit Service 501009
	fprintf(fp, "      last_pmd_read: %lx\n", machdep->last_pmd_read);
Packit Service 501009
	fprintf(fp, "     last_ptbl_read: %lx\n", machdep->last_ptbl_read);
Packit Service 501009
	fprintf(fp, "                pgd: %lx\n", (ulong)machdep->pgd);
Packit Service 501009
	fprintf(fp, "                pmd: %lx\n", (ulong)machdep->pmd);
Packit Service 501009
	fprintf(fp, "               ptbl: %lx\n", (ulong)machdep->ptbl);
Packit Service 501009
	fprintf(fp, "       ptrs_per_pgd: %d\n", machdep->ptrs_per_pgd);
Packit Service 501009
	fprintf(fp, "   max_physmem_bits: %ld\n", machdep->max_physmem_bits);
Packit Service 501009
	fprintf(fp, "  section_size_bits: %ld\n", machdep->section_size_bits);
Packit Service 501009
	fprintf(fp, "           machspec: %lx\n", (ulong)machdep->machspec);
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Check if address is in context's address space
Packit Service 501009
 */
Packit Service 501009
static int 
Packit Service 501009
s390x_is_uvaddr(ulong vaddr, struct task_context *tc)
Packit Service 501009
{
Packit Service 501009
	return IN_TASK_VMA(tc->task, vaddr);
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 *  Translates a user virtual address to its physical address
Packit Service 501009
 */
Packit Service 501009
static int
Packit Service 501009
s390x_uvtop(struct task_context *tc, ulong vaddr, physaddr_t *paddr, int verbose)
Packit Service 501009
{
Packit Service 501009
	unsigned long pgd_base;
Packit Service 501009
	readmem(tc->mm_struct + OFFSET(mm_struct_pgd), KVADDR,
Packit Service 501009
		&pgd_base,sizeof(long), "pgd_base",FAULT_ON_ERROR);
Packit Service 501009
	return s390x_vtop(pgd_base, vaddr, paddr, verbose);	
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 *  Translates a kernel virtual address to its physical address
Packit Service 501009
 */
Packit Service 501009
static int
Packit Service 501009
s390x_kvtop(struct task_context *tc, ulong vaddr, physaddr_t *paddr, int verbose)
Packit Service 501009
{
Packit Service 501009
	unsigned long pgd_base;
Packit Service 501009
Packit Service 501009
	if (!IS_KVADDR(vaddr)){
Packit Service 501009
		*paddr = 0;
Packit Service 501009
		return FALSE;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	if (!vt->vmalloc_start) {
Packit Service 501009
	       *paddr = VTOP(vaddr);
Packit Service 501009
	       return TRUE;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	if (!IS_VMALLOC_ADDR(vaddr)) {
Packit Service 501009
	       *paddr = VTOP(vaddr);
Packit Service 501009
	       return TRUE;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	pgd_base = (unsigned long)vt->kernel_pgd[0];
Packit Service 501009
	return s390x_vtop(pgd_base, vaddr, paddr, verbose);	
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Check if page is mapped
Packit Service 501009
 */
Packit Service 501009
static inline int s390x_pte_present(unsigned long x){
Packit Service 501009
	if(THIS_KERNEL_VERSION >= LINUX(2,6,0)){
Packit Service 501009
		return !((x) & S390X_PAGE_INVALID) ||
Packit Service 501009
			((x) & S390X_PAGE_INVALID_MASK) == S390X_PAGE_INVALID_NONE;
Packit Service 501009
	} else {
Packit Service 501009
		return ((x) & S390X_PAGE_PRESENT);
Packit Service 501009
	}
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * page table traversal functions 
Packit Service 501009
 */
Packit Service 501009
Packit Service 501009
/* Print flags of Segment-Table entry with format control = 1 */
Packit Service 501009
static void print_segment_entry_fc1(ulong val)
Packit Service 501009
{
Packit Service 501009
	fprintf(fp, "AV=%u; ACC=%u%u%u%u; F=%u; FC=%u; P=%u; I=%u; CS=%u; TT=%u%u\n",
Packit Service 501009
		!!(val & S390X_STE_AV),
Packit Service 501009
		!!(val & S390X_STE_ACC_1000),
Packit Service 501009
		!!(val & S390X_STE_ACC_0100),
Packit Service 501009
		!!(val & S390X_STE_ACC_0010),
Packit Service 501009
		!!(val & S390X_STE_ACC_0001),
Packit Service 501009
		!!(val & S390X_STE_F),
Packit Service 501009
		!!(val & S390X_STE_FC),
Packit Service 501009
		!!(val & S390X_STE_P),
Packit Service 501009
		!!(val & S390X_STE_I),
Packit Service 501009
		!!(val & S390X_STE_CS),
Packit Service 501009
		!!(val & S390X_STE_TT_10),
Packit Service 501009
		!!(val & S390X_STE_TT_01));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/* Print flags of Segment-Table entry with format control = 0 */
Packit Service 501009
static void print_segment_entry_fc0(ulong val)
Packit Service 501009
{
Packit Service 501009
	fprintf(fp, "FC=%u; P=%u; I=%u; CS=%u; TT=%u%u\n",
Packit Service 501009
		!!(val & S390X_STE_FC),
Packit Service 501009
		!!(val & S390X_STE_P),
Packit Service 501009
		!!(val & S390X_STE_I),
Packit Service 501009
		!!(val & S390X_STE_CS),
Packit Service 501009
		!!(val & S390X_STE_TT_10),
Packit Service 501009
		!!(val & S390X_STE_TT_01));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/* Print flags of Region-Third-Table entry with format control = 1 */
Packit Service 501009
static void print_region_third_entry_fc1(ulong val)
Packit Service 501009
{
Packit Service 501009
	fprintf(fp, "AV=%u; ACC=%u%u%u%u; F=%u; FC=%u; P=%u; I=%u; CR=%u; TT=%u%u\n",
Packit Service 501009
		!!(val & S390X_RTE_AV),
Packit Service 501009
		!!(val & S390X_RTE_ACC_1000),
Packit Service 501009
		!!(val & S390X_RTE_ACC_0100),
Packit Service 501009
		!!(val & S390X_RTE_ACC_0010),
Packit Service 501009
		!!(val & S390X_RTE_ACC_0001),
Packit Service 501009
		!!(val & S390X_RTE_F),
Packit Service 501009
		!!(val & S390X_RTE_FC),
Packit Service 501009
		!!(val & S390X_RTE_P),
Packit Service 501009
		!!(val & S390X_RTE_I),
Packit Service 501009
		!!(val & S390X_RTE_CR),
Packit Service 501009
		!!(val & S390X_RTE_TT_10),
Packit Service 501009
		!!(val & S390X_RTE_TT_01));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/* Print flags of Region-Third-Table entry with format control = 0 */
Packit Service 501009
static void print_region_third_entry_fc0(ulong val)
Packit Service 501009
{
Packit Service 501009
	fprintf(fp, "FC=%u; P=%u; TF=%u%u; I=%u; CR=%u; TT=%u%u; TL=%u%u\n",
Packit Service 501009
		!!(val & S390X_RTE_FC),
Packit Service 501009
		!!(val & S390X_RTE_P),
Packit Service 501009
		!!(val & S390X_RTE_TF_10),
Packit Service 501009
		!!(val & S390X_RTE_TF_01),
Packit Service 501009
		!!(val & S390X_RTE_I),
Packit Service 501009
		!!(val & S390X_RTE_CR),
Packit Service 501009
		!!(val & S390X_RTE_TT_10),
Packit Service 501009
		!!(val & S390X_RTE_TT_01),
Packit Service 501009
		!!(val & S390X_RTE_TL_10),
Packit Service 501009
		!!(val & S390X_RTE_TL_01));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/* Print flags of Region-First/Second-Table entry */
Packit Service 501009
static void print_region_first_second_entry(ulong val)
Packit Service 501009
{
Packit Service 501009
	fprintf(fp, "P=%u; TF=%u%u; I=%u; TT=%u%u; TL=%u%u\n",
Packit Service 501009
		!!(val & S390X_RTE_P),
Packit Service 501009
		!!(val & S390X_RTE_TF_10),
Packit Service 501009
		!!(val & S390X_RTE_TF_01),
Packit Service 501009
		!!(val & S390X_RTE_I),
Packit Service 501009
		!!(val & S390X_RTE_TT_10),
Packit Service 501009
		!!(val & S390X_RTE_TT_01),
Packit Service 501009
		!!(val & S390X_RTE_TL_10),
Packit Service 501009
		!!(val & S390X_RTE_TL_01));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/* Print the binary flags for Region or Segment table entry */
Packit Service 501009
static void s390x_print_te_binary_flags(ulong val, int level)
Packit Service 501009
{
Packit Service 501009
	fprintf(fp, "       flags in binary : ");
Packit Service 501009
	switch (level) {
Packit Service 501009
	case 0:
Packit Service 501009
		if (val & S390X_STE_FC)
Packit Service 501009
			print_segment_entry_fc1(val);
Packit Service 501009
		else
Packit Service 501009
			print_segment_entry_fc0(val);
Packit Service 501009
		break;
Packit Service 501009
	case 1:
Packit Service 501009
		if (val & S390X_RTE_FC)
Packit Service 501009
			print_region_third_entry_fc1(val);
Packit Service 501009
		else
Packit Service 501009
			print_region_third_entry_fc0(val);
Packit Service 501009
		break;
Packit Service 501009
	case 2:
Packit Service 501009
	case 3:
Packit Service 501009
		print_region_first_second_entry(val);
Packit Service 501009
		break;
Packit Service 501009
	}
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/* Region or segment table traversal function */
Packit Service 501009
static ulong _kl_rsg_table_deref_s390x(ulong vaddr, ulong table,
Packit Service 501009
				       int len, int level, int verbose)
Packit Service 501009
{
Packit Service 501009
	const char *name_vec[] = {"STE", "RTTE", "RSTE", "RFTE"};
Packit Service 501009
	ulong offset, entry, flags, addr;
Packit Service 501009
	int flags_prt_len;
Packit Service 501009
Packit Service 501009
	offset = ((vaddr >> (11*level + 20)) & 0x7ffULL) * 8;
Packit Service 501009
	if (offset >= (len + 1)*4096)
Packit Service 501009
		/* Offset is over the table limit. */
Packit Service 501009
		return 0;
Packit Service 501009
	addr = table + offset;
Packit Service 501009
	readmem(addr, KVADDR, &entry, sizeof(entry), "entry", FAULT_ON_ERROR);
Packit Service 501009
	if (verbose) {
Packit Service 501009
		flags_prt_len = 3;
Packit Service 501009
		if (entry & S390X_RTE_FC)
Packit Service 501009
			if (level) {
Packit Service 501009
				flags = entry & S390X_RTE_FLAG_BITS_FC1;
Packit Service 501009
				flags_prt_len = 8;
Packit Service 501009
			} else {
Packit Service 501009
				flags = entry & S390X_STE_FLAG_BITS_FC1;
Packit Service 501009
				flags_prt_len = 5;
Packit Service 501009
			}
Packit Service 501009
		else
Packit Service 501009
			if (level)
Packit Service 501009
				flags = entry & S390X_RTE_FLAG_BITS_FC0;
Packit Service 501009
			else
Packit Service 501009
				flags = entry & S390X_STE_FLAG_BITS_FC0;
Packit Service 501009
		fprintf(fp, "%5s: %016lx => %016lx (flags = %0*lx)\n",
Packit Service 501009
			name_vec[level], addr, entry, flags_prt_len, flags);
Packit Service 501009
		s390x_print_te_binary_flags(entry, level);
Packit Service 501009
	}
Packit Service 501009
	/*
Packit Service 501009
	 * Check if the segment table entry could be read and doesn't have
Packit Service 501009
	 * any of the reserved bits set.
Packit Service 501009
	 */
Packit Service 501009
	if ((entry & S390X_RTE_TT) != (level << 2))
Packit Service 501009
		return 0;
Packit Service 501009
	/* Check if the region table entry has the invalid bit set. */
Packit Service 501009
	if (entry & S390X_RTE_I)
Packit Service 501009
		return 0;
Packit Service 501009
	/* Region table entry is valid and well formed. */
Packit Service 501009
	return entry;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/* Check for swap entry */
Packit Service 501009
static int swap_entry(ulong entry)
Packit Service 501009
{
Packit Service 501009
	if (THIS_KERNEL_VERSION < LINUX(2,6,19)) {
Packit Service 501009
		if ((entry & 0x601ULL) == 0x600ULL)
Packit Service 501009
			return 1;
Packit Service 501009
	} if (THIS_KERNEL_VERSION < LINUX(3,12,0)) {
Packit Service 501009
		if ((entry & 0x403ULL) == 0x403ULL)
Packit Service 501009
			return 1;
Packit Service 501009
	} else {
Packit Service 501009
		if ((entry & 0x603ULL) == 0x402ULL)
Packit Service 501009
			return 1;
Packit Service 501009
	}
Packit Service 501009
	return 0;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/* Page table traversal function */
Packit Service 501009
static ulong _kl_pg_table_deref_s390x(ulong vaddr, ulong table, int verbose)
Packit Service 501009
{
Packit Service 501009
	ulong offset, entry, addr;
Packit Service 501009
Packit Service 501009
	offset = ((vaddr >> 12) & 0xffULL) * 8;
Packit Service 501009
	addr = table + offset;
Packit Service 501009
	readmem(addr, KVADDR, &entry, sizeof(entry), "entry", FAULT_ON_ERROR);
Packit Service 501009
	if (verbose) {
Packit Service 501009
		fprintf(fp, "%5s: %016lx => %016lx (flags = %03llx)\n",
Packit Service 501009
			"PTE", addr, entry, entry & S390X_PTE_FLAG_BITS);
Packit Service 501009
		fprintf(fp, "       flags in binary : I=%u; P=%u\n",
Packit Service 501009
			!!(entry & S390X_PAGE_INVALID), !!(entry & S390X_PAGE_RO));
Packit Service 501009
		fprintf(fp, "%5s: %016llx\n", "PAGE", entry & ~S390X_PTE_FLAG_BITS);
Packit Service 501009
	}
Packit Service 501009
	/*
Packit Service 501009
	 * Return zero if the page table entry has the reserved (0x800) or
Packit Service 501009
	 * the invalid (0x400) bit set and it is not a swap entry.
Packit Service 501009
	 */
Packit Service 501009
	if ((entry & 0xc00ULL) && !swap_entry(entry))
Packit Service 501009
		return 0;
Packit Service 501009
	/* Page table entry is valid and well formed. */
Packit Service 501009
	return entry;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/* lookup virtual address in page tables */
Packit Service 501009
int s390x_vtop(ulong table, ulong vaddr, physaddr_t *phys_addr, int verbose)
Packit Service 501009
{
Packit Service 501009
	ulong entry, paddr;
Packit Service 501009
	int level, len;
Packit Service 501009
Packit Service 501009
	if (verbose)
Packit Service 501009
		fprintf(fp, "PAGE DIRECTORY: %016lx\n", table);
Packit Service 501009
Packit Service 501009
	*phys_addr = 0;
Packit Service 501009
	/*
Packit Service 501009
	 * Walk the region and segment tables.
Packit Service 501009
	 * We assume that the table length field in the asce is set to the
Packit Service 501009
	 * maximum value of 3 (which translates to a region first, region
Packit Service 501009
	 * second, region third or segment table with 2048 entries) and that
Packit Service 501009
	 * the addressing mode is 64 bit.
Packit Service 501009
	 */
Packit Service 501009
	len = 3;
Packit Service 501009
	/* Read the first entry to find the number of page table levels. */
Packit Service 501009
	readmem(table, KVADDR, &entry, sizeof(entry), "entry", FAULT_ON_ERROR);
Packit Service 501009
	level = (entry & 0xcULL) >> 2;
Packit Service 501009
	if ((level < 3) && (vaddr >> (31 + 11*level)) != 0ULL) {
Packit Service 501009
		/* Address too big for the number of page table levels. */
Packit Service 501009
		return FALSE;
Packit Service 501009
	}
Packit Service 501009
	while (level >= 0) {
Packit Service 501009
		entry = _kl_rsg_table_deref_s390x(vaddr, table, len, level,
Packit Service 501009
						  verbose);
Packit Service 501009
		if (!entry)
Packit Service 501009
			return FALSE;
Packit Service 501009
		table = entry & ~0xfffULL;
Packit Service 501009
		/* Check if this a 2GB page */
Packit Service 501009
		if ((entry & 0x400ULL) && (level == 1)) {
Packit Service 501009
			/* Add the 2GB frame offset & return the final value. */
Packit Service 501009
			table &= ~0x7fffffffULL;
Packit Service 501009
			*phys_addr = table + (vaddr & 0x7fffffffULL);
Packit Service 501009
			return TRUE;
Packit Service 501009
		}
Packit Service 501009
		len = entry & 0x3ULL;
Packit Service 501009
		level--;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	/* Check if this is a large page. */
Packit Service 501009
	if (entry & 0x400ULL) {
Packit Service 501009
		/* Add the 1MB page offset and return the final value. */
Packit Service 501009
		table &= ~0xfffffULL;
Packit Service 501009
		*phys_addr = table + (vaddr & 0xfffffULL);
Packit Service 501009
		return TRUE;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	/* Get the page table entry */
Packit Service 501009
	entry = _kl_pg_table_deref_s390x(vaddr, entry & ~0x7ffULL, verbose);
Packit Service 501009
	if (!entry)
Packit Service 501009
		return FALSE;
Packit Service 501009
Packit Service 501009
	/* For swap entries we have to return FALSE and phys_addr = PTE */
Packit Service 501009
	if (swap_entry(entry)) {
Packit Service 501009
		*phys_addr = entry;
Packit Service 501009
		return FALSE;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	/* Isolate the page origin from the page table entry. */
Packit Service 501009
	paddr = entry & ~0xfffULL;
Packit Service 501009
Packit Service 501009
	/* Add the page offset and return the final value. */
Packit Service 501009
	*phys_addr = paddr + (vaddr & 0xfffULL);
Packit Service 501009
Packit Service 501009
	return TRUE;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 *  Determine where vmalloc'd memory starts.
Packit Service 501009
 */
Packit Service 501009
static ulong
Packit Service 501009
s390x_vmalloc_start(void)
Packit Service 501009
{
Packit Service 501009
	unsigned long highmem_addr,high_memory;
Packit Service 501009
	highmem_addr=symbol_value("high_memory");
Packit Service 501009
       	readmem(highmem_addr, PHYSADDR, &high_memory,sizeof(long),
Packit Service 501009
		"highmem",FAULT_ON_ERROR);
Packit Service 501009
	return high_memory;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Check if address can be a valid task_struct
Packit Service 501009
 */
Packit Service 501009
static int
Packit Service 501009
s390x_is_task_addr(ulong task)
Packit Service 501009
{
Packit Service 501009
	if (tt->flags & THREAD_INFO)
Packit Service 501009
		return IS_KVADDR(task);
Packit Service 501009
	else
Packit Service 501009
		return (IS_KVADDR(task) && (ALIGNED_STACK_OFFSET(task) == 0));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * return MHz - unfortunately it is not possible to get this on linux 
Packit Service 501009
 *              for zSeries
Packit Service 501009
 */
Packit Service 501009
static ulong
Packit Service 501009
s390x_processor_speed(void)
Packit Service 501009
{
Packit Service 501009
	return 0;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 *  Accept or reject a symbol from the kernel namelist.
Packit Service 501009
 */
Packit Service 501009
static int
Packit Service 501009
s390x_verify_symbol(const char *name, ulong value, char type)
Packit Service 501009
{
Packit Service 501009
	int i;
Packit Service 501009
Packit Service 501009
	if (CRASHDEBUG(8) && name && strlen(name))
Packit Service 501009
		fprintf(fp, "%08lx %s\n", value, name);
Packit Service 501009
Packit Service 501009
	if (STREQ(name, "startup") || STREQ(name, "_stext"))
Packit Service 501009
		machdep->flags |= KSYMS_START;
Packit Service 501009
Packit Service 501009
	if (!name || !strlen(name) || !(machdep->flags & KSYMS_START))
Packit Service 501009
		return FALSE;
Packit Service 501009
Packit Service 501009
	if ((type == 'A') && STRNEQ(name, "__crc_"))
Packit Service 501009
		return FALSE;
Packit Service 501009
Packit Service 501009
	if (STREQ(name, "Letext") || STREQ(name, "gcc2_compiled."))
Packit Service 501009
		return FALSE;
Packit Service 501009
Packit Service 501009
        /* reject L2^B symbols */
Packit Service 501009
	if (strstr(name, "L2\002") == name)
Packit Service 501009
	    	return FALSE;
Packit Service 501009
Packit Service 501009
	/* throw away all symbols containing a '.' */
Packit Service 501009
	for(i = 0; i < strlen(name);i++){
Packit Service 501009
		if(name[i] == '.')
Packit Service 501009
			return FALSE;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	return TRUE;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 *  Get the relevant page directory pointer from a task structure.
Packit Service 501009
 */
Packit Service 501009
static ulong
Packit Service 501009
s390x_get_task_pgd(ulong task)
Packit Service 501009
{
Packit Service 501009
	return (error(FATAL, "s390x_get_task_pgd: TBD\n"));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 *  Translate a PTE, returning TRUE if the page is present.
Packit Service 501009
 *  If a physaddr pointer is passed in, don't print anything.
Packit Service 501009
 */
Packit Service 501009
static int
Packit Service 501009
s390x_translate_pte(ulong pte, void *physaddr, ulonglong unused)
Packit Service 501009
{
Packit Service 501009
	char *arglist[MAXARGS];
Packit Service 501009
	char buf[BUFSIZE];
Packit Service 501009
	char buf2[BUFSIZE];
Packit Service 501009
	char buf3[BUFSIZE];
Packit Service 501009
	char ptebuf[BUFSIZE];
Packit Service 501009
	int c,len1,len2,len3;
Packit Service 501009
Packit Service 501009
	if(S390X_PTE_INVALID(pte)){
Packit Service 501009
		fprintf(fp,"PTE is invalid\n");
Packit Service 501009
		return FALSE;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	if(physaddr)
Packit Service 501009
		*((ulong *)physaddr) = pte & S390X_PAGE_BASE_MASK;
Packit Service 501009
Packit Service 501009
	if(!s390x_pte_present(pte)){
Packit Service 501009
		swap_location(pte, buf);
Packit Service 501009
		if ((c = parse_line(buf, arglist)) != 3)
Packit Service 501009
			error(FATAL, "cannot determine swap location\n");
Packit Service 501009
Packit Service 501009
		sprintf(ptebuf, "%lx", pte);
Packit Service 501009
		len1 = MAX(strlen(ptebuf), strlen("PTE"));
Packit Service 501009
		len2 = MAX(strlen(arglist[0]), strlen("SWAP"));
Packit Service 501009
		len3 = MAX(strlen(arglist[2]), strlen("OFFSET"));
Packit Service 501009
Packit Service 501009
		fprintf(fp, "%s  %s  %s\n",
Packit Service 501009
			mkstring(ptebuf, len1, CENTER|LJUST, "PTE"),
Packit Service 501009
			mkstring(buf2, len2, CENTER|LJUST, "SWAP"),
Packit Service 501009
			mkstring(buf3, len3, CENTER|LJUST, "OFFSET"));
Packit Service 501009
Packit Service 501009
		sprintf(ptebuf, "%lx", pte);
Packit Service 501009
		strcpy(buf2, arglist[0]);
Packit Service 501009
		strcpy(buf3, arglist[2]);
Packit Service 501009
		fprintf(fp, "%s  %s  %s\n",
Packit Service 501009
			mkstring(ptebuf, len1, CENTER|RJUST, NULL),
Packit Service 501009
			mkstring(buf2, len2, CENTER|RJUST, NULL),
Packit Service 501009
			mkstring(buf3, len3, CENTER|RJUST, NULL));
Packit Service 501009
		return FALSE;
Packit Service 501009
	}
Packit Service 501009
	fprintf(fp,"PTE      PHYSICAL  FLAGS\n");
Packit Service 501009
	fprintf(fp,"%08lx %08llx",pte, pte & S390X_PAGE_BASE_MASK);
Packit Service 501009
	fprintf(fp,"  (");
Packit Service 501009
	if(pte & S390X_PAGE_INVALID)
Packit Service 501009
		fprintf(fp,"INVALID ");
Packit Service 501009
	if(pte & S390X_PAGE_RO)
Packit Service 501009
		fprintf(fp,"PROTECTION");
Packit Service 501009
	fprintf(fp,")");
Packit Service 501009
	return TRUE;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 *  Look for likely exception frames in a stack.
Packit Service 501009
 */
Packit Service 501009
static int 
Packit Service 501009
s390x_eframe_search(struct bt_info *bt)
Packit Service 501009
{
Packit Service 501009
	if(bt->flags & BT_EFRAME_SEARCH2)
Packit Service 501009
		return (error(FATAL, 
Packit Service 501009
		    "Option '-E' is not implemented for this architecture\n"));
Packit Service 501009
	else
Packit Service 501009
		return (error(FATAL, 
Packit Service 501009
		    "Option '-e' is not implemented for this architecture\n"));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
#ifdef DEPRECATED
Packit Service 501009
/*
Packit Service 501009
 * returns cpu number of task
Packit Service 501009
 */ 
Packit Service 501009
static int 
Packit Service 501009
s390x_cpu_of_task(unsigned long task)
Packit Service 501009
{
Packit Service 501009
	unsigned int cpu;
Packit Service 501009
Packit Service 501009
	if(VALID_MEMBER(task_struct_processor)){
Packit Service 501009
		/* linux 2.4 */
Packit Service 501009
		readmem(task + OFFSET(task_struct_processor),KVADDR,
Packit Service 501009
                        &cpu, sizeof(cpu), "task_struct_processor", 
Packit Service 501009
			FAULT_ON_ERROR);		
Packit Service 501009
	} else {
Packit Service 501009
		/* linux 2.6 */
Packit Service 501009
		char thread_info[8192];
Packit Service 501009
		unsigned long thread_info_addr;
Packit Service 501009
		readmem(task + OFFSET(task_struct_thread_info),KVADDR,
Packit Service 501009
                        &thread_info_addr, sizeof(thread_info_addr),
Packit Service 501009
                        "thread info addr", FAULT_ON_ERROR);
Packit Service 501009
		readmem(thread_info_addr,KVADDR,thread_info,sizeof(thread_info),
Packit Service 501009
			"thread info", FAULT_ON_ERROR);
Packit Service 501009
		cpu = *((int*) &thread_info[OFFSET(thread_info_cpu)]);
Packit Service 501009
	}
Packit Service 501009
	return cpu;
Packit Service 501009
}
Packit Service 501009
#endif
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * returns true, if task of bt currently is executed by a cpu
Packit Service 501009
 */ 
Packit Service 501009
static int 
Packit Service 501009
s390x_has_cpu(struct bt_info *bt)
Packit Service 501009
{
Packit Service 501009
	int cpu = bt->tc->processor;
Packit Service 501009
Packit Service 501009
	if (is_task_active(bt->task) && (kt->cpu_flags[cpu] & ONLINE_MAP))
Packit Service 501009
		return TRUE;
Packit Service 501009
	else
Packit Service 501009
		return FALSE;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * read lowcore for cpu
Packit Service 501009
 */
Packit Service 501009
static void
Packit Service 501009
s390x_get_lowcore(struct bt_info *bt, char* lowcore)
Packit Service 501009
{
Packit Service 501009
	unsigned long lowcore_array,lowcore_ptr;
Packit Service 501009
	struct s390x_cpu *s390x_cpu;
Packit Service 501009
	int cpu = bt->tc->processor;
Packit Service 501009
Packit Service 501009
	lowcore_array = symbol_value("lowcore_ptr");
Packit Service 501009
	readmem(lowcore_array + cpu * S390X_WORD_SIZE,KVADDR,
Packit Service 501009
		&lowcore_ptr, sizeof(long), "lowcore_ptr", FAULT_ON_ERROR);
Packit Service 501009
	readmem(lowcore_ptr, KVADDR, lowcore, LOWCORE_SIZE, "lowcore", 
Packit Service 501009
		FAULT_ON_ERROR);
Packit Service 501009
Packit Service 501009
	if (!s390x_cpu_vec)
Packit Service 501009
		return;
Packit Service 501009
Packit Service 501009
	/* Copy register information to defined places in lowcore */
Packit Service 501009
	s390x_cpu = s390x_cpu_get(bt);
Packit Service 501009
Packit Service 501009
	memcpy(lowcore + 4864, &s390x_cpu->psw, sizeof(s390x_cpu->psw));
Packit Service 501009
	memcpy(lowcore + 4736, &s390x_cpu->gprs, sizeof(s390x_cpu->gprs));
Packit Service 501009
	memcpy(lowcore + 4928, &s390x_cpu->acrs, sizeof(s390x_cpu->acrs));
Packit Service 501009
Packit Service 501009
	memcpy(lowcore + 4892, &s390x_cpu->fpc, sizeof(s390x_cpu->fpc));
Packit Service 501009
	memcpy(lowcore + 4608, &s390x_cpu->fprs, sizeof(s390x_cpu->fprs));
Packit Service 501009
Packit Service 501009
	memcpy(lowcore + 4888, &s390x_cpu->prefix, sizeof(s390x_cpu->prefix));
Packit Service 501009
	memcpy(lowcore + 4992, &s390x_cpu->ctrs, sizeof(s390x_cpu->ctrs));
Packit Service 501009
Packit Service 501009
	memcpy(lowcore + 4900, &s390x_cpu->todpreg, sizeof(s390x_cpu->todpreg));
Packit Service 501009
	memcpy(lowcore + 4904, &s390x_cpu->timer, sizeof(s390x_cpu->timer));
Packit Service 501009
	memcpy(lowcore + 4912, &s390x_cpu->todcmp, sizeof(s390x_cpu->todcmp));
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Copy VX registers out of s390x cpu
Packit Service 501009
 */
Packit Service 501009
static void vx_copy(void *buf, struct s390x_cpu *s390x_cpu)
Packit Service 501009
{
Packit Service 501009
	char *_buf = buf;
Packit Service 501009
	int i;
Packit Service 501009
Packit Service 501009
	for (i = 0; i < 16; i++) {
Packit Service 501009
		memcpy(&_buf[i * 16], &s390x_cpu->fprs[i], 8);
Packit Service 501009
		memcpy(&_buf[i * 16 + 8], &s390x_cpu->vxrs_low[i], 8);
Packit Service 501009
	}
Packit Service 501009
	memcpy(&_buf[16 * 16], &s390x_cpu->vxrs_high[0], 16 * 16);
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Check if VX registers are available
Packit Service 501009
 */
Packit Service 501009
static int has_vx_regs(char *lowcore)
Packit Service 501009
{
Packit Service 501009
	unsigned long addr = *((uint64_t *)(lowcore + 0x11b0));
Packit Service 501009
Packit Service 501009
	if (addr == 0 || addr % 1024)
Packit Service 501009
		return 0;
Packit Service 501009
	return 1;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Print vector registers for cpu
Packit Service 501009
 */
Packit Service 501009
static void
Packit Service 501009
s390x_print_vx_sa(struct bt_info *bt, char *lc)
Packit Service 501009
{
Packit Service 501009
	char vx_sa[VX_SA_SIZE];
Packit Service 501009
	uint64_t addr;
Packit Service 501009
Packit Service 501009
	if (!(bt->flags & BT_SHOW_ALL_REGS))
Packit Service 501009
		return;
Packit Service 501009
	if (!has_vx_regs(lc))
Packit Service 501009
		return;
Packit Service 501009
	if (!s390x_cpu_vec) {
Packit Service 501009
		/* Pointer to save area */
Packit Service 501009
		addr = *((uint64_t *)(lc + 0x11b0));
Packit Service 501009
		readmem(addr, KVADDR, vx_sa, sizeof(vx_sa), "vx_sa",
Packit Service 501009
			FAULT_ON_ERROR);
Packit Service 501009
	} else {
Packit Service 501009
		/* Get data from s390x cpu */
Packit Service 501009
		vx_copy(vx_sa, s390x_cpu_get(bt));
Packit Service 501009
	}
Packit Service 501009
	fprintf(fp, "  -vector registers:\n");
Packit Service 501009
	print_hex_buf(vx_sa, sizeof(vx_sa), 2, "     ");
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Get stack address for interrupt stack using the pcpu array
Packit Service 501009
 */
Packit Service 501009
static unsigned long get_int_stack_pcpu(char *stack_name, int cpu)
Packit Service 501009
{
Packit Service 501009
	unsigned long addr;
Packit Service 501009
Packit Service 501009
	addr = symbol_value("pcpu_devices") +
Packit Service 501009
		cpu * STRUCT_SIZE("pcpu") + MEMBER_OFFSET("pcpu", stack_name);
Packit Service 501009
	return readmem_ul(addr) + INT_STACK_SIZE;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Get stack address for interrupt stack using the lowcore
Packit Service 501009
 */
Packit Service 501009
static unsigned long get_int_stack_lc(char *stack_name, char *lc)
Packit Service 501009
{
Packit Service 501009
	if (!MEMBER_EXISTS(lc_struct, stack_name))
Packit Service 501009
		return 0;
Packit Service 501009
	return roundup(ULONG(lc + MEMBER_OFFSET(lc_struct, stack_name)),
Packit Service 501009
			     PAGESIZE());
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Read interrupt stack (either "async_stack" or "panic_stack");
Packit Service 501009
 */
Packit Service 501009
static void get_int_stack(char *stack_name, int cpu, char *lc,
Packit Service 501009
			  unsigned long *start, unsigned long *end)
Packit Service 501009
{
Packit Service 501009
	unsigned long stack_addr;
Packit Service 501009
Packit Service 501009
	*start = *end = 0;
Packit Service 501009
	if (strcmp(stack_name, "restart_stack") == 0) {
Packit Service 501009
		stack_addr = symbol_value("restart_stack");
Packit Service 501009
		stack_addr = readmem_ul(stack_addr);
Packit Service 501009
	} else {
Packit Service 501009
		if (symbol_exists("pcpu_devices") && MEMBER_EXISTS("pcpu", stack_name))
Packit Service 501009
			stack_addr = get_int_stack_pcpu(stack_name, cpu);
Packit Service 501009
		else
Packit Service 501009
			stack_addr = get_int_stack_lc(stack_name, lc);
Packit Service 501009
	}
Packit Service 501009
	if (stack_addr == 0)
Packit Service 501009
		return;
Packit Service 501009
	*start = stack_addr - INT_STACK_SIZE;
Packit Service 501009
	*end = stack_addr;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Print hex data
Packit Service 501009
 */
Packit Service 501009
static void print_hex(unsigned long addr, int len, int cols)
Packit Service 501009
{
Packit Service 501009
	int j, first = 1;
Packit Service 501009
Packit Service 501009
	for (j = 0; j < len; j += 8) {
Packit Service 501009
		if (j % (cols * 8) == 0) {
Packit Service 501009
			if (!first)
Packit Service 501009
				fprintf(fp, "\n");
Packit Service 501009
			else
Packit Service 501009
				first = 0;
Packit Service 501009
			fprintf(fp, "    %016lx: ", addr + j);
Packit Service 501009
		}
Packit Service 501009
		fprintf(fp, " %016lx", readmem_ul(addr + j));
Packit Service 501009
	}
Packit Service 501009
	if (len)
Packit Service 501009
		fprintf(fp, "\n");
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Print hexdump of stack frame data
Packit Service 501009
 */
Packit Service 501009
static void print_frame_data(unsigned long sp, unsigned long high)
Packit Service 501009
{
Packit Service 501009
	unsigned long next_sp, len = high - sp;
Packit Service 501009
Packit Service 501009
	next_sp = readmem_ul(sp + MEMBER_OFFSET("stack_frame", "back_chain"));
Packit Service 501009
	if (next_sp == 0)
Packit Service 501009
		len = MIN(len, SIZE(s390_stack_frame) + STRUCT_SIZE("pt_regs"));
Packit Service 501009
	else
Packit Service 501009
		len = MIN(len, next_sp - sp);
Packit Service 501009
	print_hex(sp, len, 2);
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Do reference check and set flags
Packit Service 501009
 */
Packit Service 501009
static int bt_reference_check(struct bt_info *bt, unsigned long addr)
Packit Service 501009
{
Packit Service 501009
	if (!BT_REFERENCE_CHECK(bt))
Packit Service 501009
		return 0;
Packit Service 501009
Packit Service 501009
	if (bt->ref->cmdflags & BT_REF_HEXVAL) {
Packit Service 501009
		if (addr == bt->ref->hexval)
Packit Service 501009
			bt->ref->cmdflags |= BT_REF_FOUND;
Packit Service 501009
	} else {
Packit Service 501009
		if (STREQ(closest_symbol(addr), bt->ref->str))
Packit Service 501009
			bt->ref->cmdflags |= BT_REF_FOUND;
Packit Service 501009
	}
Packit Service 501009
	return 1;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Print stack frame
Packit Service 501009
 */
Packit Service 501009
static void print_frame(struct bt_info *bt, int cnt, unsigned long sp,
Packit Service 501009
			unsigned long r14)
Packit Service 501009
{
Packit Service 501009
	struct load_module *lm;
Packit Service 501009
	char *sym;
Packit Service 501009
	ulong offset;
Packit Service 501009
	struct syment *symp;
Packit Service 501009
	char *name_plus_offset;
Packit Service 501009
	char buf[BUFSIZE];
Packit Service 501009
Packit Service 501009
	if (bt_reference_check(bt, r14))
Packit Service 501009
		return;
Packit Service 501009
	fprintf(fp, "%s#%d [%08lx] ", cnt < 10 ? " " : "", cnt, sp);
Packit Service 501009
	sym = closest_symbol(r14);
Packit Service 501009
	name_plus_offset = NULL;
Packit Service 501009
	if (bt->flags & BT_SYMBOL_OFFSET) {
Packit Service 501009
		symp = value_search(r14, &offset);
Packit Service 501009
		if (symp && offset)
Packit Service 501009
			name_plus_offset = value_to_symstr(r14, buf, bt->radix);
Packit Service 501009
	}
Packit Service 501009
	fprintf(fp, "%s at %lx", name_plus_offset ? name_plus_offset : sym, r14);
Packit Service 501009
	if (module_symbol(r14, NULL, &lm, NULL, 0))
Packit Service 501009
		fprintf(fp, " [%s]", lm->mod_name);
Packit Service 501009
	fprintf(fp, "\n");
Packit Service 501009
	if (bt->flags & BT_LINE_NUMBERS)
Packit Service 501009
		s390x_dump_line_number(r14);
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Print pt_regs structure
Packit Service 501009
 */
Packit Service 501009
static void print_ptregs(struct bt_info *bt, unsigned long sp)
Packit Service 501009
{
Packit Service 501009
	unsigned long addr, psw_flags, psw_addr, offs;
Packit Service 501009
	struct load_module *lm;
Packit Service 501009
	char *sym;
Packit Service 501009
	int i;
Packit Service 501009
Packit Service 501009
	addr = sp + MEMBER_OFFSET("pt_regs", "psw");
Packit Service 501009
	psw_flags = readmem_ul(addr);
Packit Service 501009
	psw_addr = readmem_ul(addr + sizeof(long));
Packit Service 501009
	if (bt_reference_check(bt, psw_addr))
Packit Service 501009
		return;
Packit Service 501009
Packit Service 501009
	fprintf(fp, " PSW:  %016lx %016lx ", psw_flags, psw_addr);
Packit Service 501009
	if (psw_flags & S390X_PSW_MASK_PSTATE) {
Packit Service 501009
		fprintf(fp, "(user space)\n");
Packit Service 501009
	} else {
Packit Service 501009
		sym = closest_symbol(psw_addr);
Packit Service 501009
		offs = psw_addr - closest_symbol_value(psw_addr);
Packit Service 501009
		if (module_symbol(psw_addr, NULL, &lm, NULL, 0))
Packit Service 501009
			fprintf(fp, "(%s+%ld [%s])\n", sym, offs, lm->mod_name);
Packit Service 501009
		else
Packit Service 501009
			fprintf(fp, "(%s+%ld)\n", sym, offs);
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	addr = sp + MEMBER_OFFSET("pt_regs", "gprs");
Packit Service 501009
	for (i = 0; i < 16; i++) {
Packit Service 501009
		if (i != 0 && i % 4 == 0)
Packit Service 501009
			fprintf(fp, "\n");
Packit Service 501009
		if (i % 4 == 0) {
Packit Service 501009
			if (i == 0)
Packit Service 501009
				fprintf(fp, " GPRS: ");
Packit Service 501009
			else
Packit Service 501009
				fprintf(fp, "       ");
Packit Service 501009
		}
Packit Service 501009
		fprintf(fp, "%016lx ", readmem_ul(addr + i * sizeof(long)));
Packit Service 501009
	}
Packit Service 501009
	fprintf(fp, "\n");
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Print back trace for one stack
Packit Service 501009
 */
Packit Service 501009
static unsigned long show_trace(struct bt_info *bt, int cnt, unsigned long sp,
Packit Service 501009
				unsigned long low, unsigned long high)
Packit Service 501009
{
Packit Service 501009
	unsigned long reg; 
Packit Service 501009
	unsigned long psw_addr ATTRIBUTE_UNUSED;
Packit Service 501009
Packit Service 501009
	while (1) {
Packit Service 501009
		if (sp < low || sp > high - SIZE(s390_stack_frame))
Packit Service 501009
			return sp;
Packit Service 501009
		reg = readmem_ul(sp + OFFSET(s390_stack_frame_r14));
Packit Service 501009
		if (!s390x_has_cpu(bt))
Packit Service 501009
			print_frame(bt, cnt++, sp, reg);
Packit Service 501009
		if (bt->flags & BT_FULL)
Packit Service 501009
			print_frame_data(sp, high);
Packit Service 501009
		/* Follow the backchain. */
Packit Service 501009
		while (1) {
Packit Service 501009
			low = sp;
Packit Service 501009
			sp = readmem_ul(sp +
Packit Service 501009
					OFFSET(s390_stack_frame_back_chain));
Packit Service 501009
			if (!sp) {
Packit Service 501009
				sp = low;
Packit Service 501009
				break;
Packit Service 501009
			}
Packit Service 501009
			if (sp <= low || sp > high - SIZE(s390_stack_frame))
Packit Service 501009
				return sp;
Packit Service 501009
			reg = readmem_ul(sp + OFFSET(s390_stack_frame_r14));
Packit Service 501009
			print_frame(bt, cnt++, sp, reg);
Packit Service 501009
			if (bt->flags & BT_FULL)
Packit Service 501009
				print_frame_data(sp, high);
Packit Service 501009
		}
Packit Service 501009
		/* Zero backchain detected, check for interrupt frame. */
Packit Service 501009
		sp += SIZE(s390_stack_frame);
Packit Service 501009
		if (sp <= low || sp > high - STRUCT_SIZE("pt_regs"))
Packit Service 501009
			return sp;
Packit Service 501009
		/* Check for user PSW */
Packit Service 501009
		reg = readmem_ul(sp + MEMBER_OFFSET("pt_regs", "psw"));
Packit Service 501009
		if (reg & S390X_PSW_MASK_PSTATE) {
Packit Service 501009
			print_ptregs(bt, sp);
Packit Service 501009
			return sp;
Packit Service 501009
		}
Packit Service 501009
		/* Get new backchain from r15 */
Packit Service 501009
		reg = readmem_ul(sp + MEMBER_OFFSET("pt_regs", "gprs") +
Packit Service 501009
				 15 * sizeof(long));
Packit Service 501009
		/* Get address of interrupted function */
Packit Service 501009
		psw_addr = readmem_ul(sp + MEMBER_OFFSET("pt_regs", "psw") +
Packit Service 501009
				      sizeof(long));
Packit Service 501009
		/* Check for loop (kernel_thread_starter) of second zero bc */
Packit Service 501009
		if (low == reg || reg == 0)
Packit Service 501009
			return reg;
Packit Service 501009
		print_ptregs(bt, sp);
Packit Service 501009
		low = sp;
Packit Service 501009
		sp = reg;
Packit Service 501009
		cnt = 0;
Packit Service 501009
	}
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * Unroll a kernel stack
Packit Service 501009
 */
Packit Service 501009
static void s390x_back_trace_cmd(struct bt_info *bt)
Packit Service 501009
{
Packit Service 501009
	unsigned long low, high, sp = bt->stkptr;
Packit Service 501009
	int cpu = bt->tc->processor, cnt = 0;
Packit Service 501009
	char lowcore[LOWCORE_SIZE];
Packit Service 501009
	unsigned long psw_flags;
Packit Service 501009
Packit Service 501009
	if (bt->hp && bt->hp->eip) {
Packit Service 501009
		error(WARNING,
Packit Service 501009
	        "instruction pointer argument ignored on this architecture!\n");
Packit Service 501009
	}
Packit Service 501009
	if (is_task_active(bt->task) && !(kt->cpu_flags[cpu] & ONLINE_MAP)) {
Packit Service 501009
		fprintf(fp, " CPU offline\n");
Packit Service 501009
		return;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	/*
Packit Service 501009
	 * Print lowcore and print interrupt stacks when task has cpu
Packit Service 501009
	 */
Packit Service 501009
	if (s390x_has_cpu(bt)) {
Packit Service 501009
		s390x_get_lowcore(bt, lowcore);
Packit Service 501009
		psw_flags = ULONG(lowcore + OFFSET(s390_lowcore_psw_save_area));
Packit Service 501009
Packit Service 501009
		if (psw_flags & S390X_PSW_MASK_PSTATE) {
Packit Service 501009
			fprintf(fp,"Task runs in userspace\n");
Packit Service 501009
			s390x_print_lowcore(lowcore,bt,0);
Packit Service 501009
			s390x_print_vx_sa(bt, lowcore);
Packit Service 501009
			return;
Packit Service 501009
		}
Packit Service 501009
		s390x_print_lowcore(lowcore,bt,1);
Packit Service 501009
		s390x_print_vx_sa(bt, lowcore);
Packit Service 501009
		fprintf(fp,"\n");
Packit Service 501009
		if (symbol_exists("restart_stack")) {
Packit Service 501009
			get_int_stack("restart_stack",
Packit Service 501009
				      cpu, lowcore, &low, &high);
Packit Service 501009
			sp = show_trace(bt, cnt, sp, low, high);
Packit Service 501009
		}
Packit Service 501009
		get_int_stack("panic_stack", cpu, lowcore, &low, &high);
Packit Service 501009
		sp = show_trace(bt, cnt, sp, low, high);
Packit Service 501009
		get_int_stack("async_stack", cpu, lowcore, &low, &high);
Packit Service 501009
		sp = show_trace(bt, cnt, sp, low, high);
Packit Service 501009
	}
Packit Service 501009
	/*
Packit Service 501009
	 * Print task stack
Packit Service 501009
	 */
Packit Service 501009
	if (THIS_KERNEL_VERSION >= LINUX(2, 6, 0)) {
Packit Service 501009
		low = task_to_stackbase(bt->task);
Packit Service 501009
	} else {
Packit Service 501009
		low = bt->task;
Packit Service 501009
	}
Packit Service 501009
	high = low + KERNEL_STACK_SIZE;
Packit Service 501009
	sp = show_trace(bt, cnt, sp, low, high);
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 * print lowcore info (psw and all registers)
Packit Service 501009
 */
Packit Service 501009
static void
Packit Service 501009
s390x_print_lowcore(char* lc, struct bt_info *bt,int show_symbols)
Packit Service 501009
{
Packit Service 501009
	char* ptr;
Packit Service 501009
	unsigned long tmp[4];
Packit Service 501009
Packit Service 501009
	ptr = lc + OFFSET(s390_lowcore_psw_save_area);
Packit Service 501009
	tmp[0]=ULONG(ptr);
Packit Service 501009
	tmp[1]=ULONG(ptr + S390X_WORD_SIZE);
Packit Service 501009
Packit Service 501009
	if(BT_REFERENCE_CHECK(bt)){
Packit Service 501009
		if(bt->ref->cmdflags & BT_REF_HEXVAL){
Packit Service 501009
			if(tmp[1] == bt->ref->hexval)
Packit Service 501009
				bt->ref->cmdflags |= BT_REF_FOUND;
Packit Service 501009
		} else {
Packit Service 501009
			if(STREQ(closest_symbol(tmp[1]),bt->ref->str))
Packit Service 501009
				bt->ref->cmdflags |= BT_REF_FOUND;
Packit Service 501009
		}
Packit Service 501009
		return;
Packit Service 501009
	}
Packit Service 501009
	fprintf(fp," LOWCORE INFO:\n");
Packit Service 501009
	fprintf(fp,"  -psw      : %#018lx %#018lx\n", tmp[0], tmp[1]);
Packit Service 501009
	if(show_symbols){
Packit Service 501009
		fprintf(fp,"  -function : %s at %lx\n", 
Packit Service 501009
			closest_symbol(tmp[1]), tmp[1]);
Packit Service 501009
		if (bt->flags & BT_LINE_NUMBERS)
Packit Service 501009
			s390x_dump_line_number(tmp[1]);
Packit Service 501009
	}
Packit Service 501009
	ptr = lc + MEMBER_OFFSET(lc_struct, "prefixreg_save_area");
Packit Service 501009
	tmp[0] = UINT(ptr);
Packit Service 501009
	fprintf(fp,"  -prefix   : %#010lx\n", tmp[0]);
Packit Service 501009
	
Packit Service 501009
	ptr = lc + MEMBER_OFFSET(lc_struct, "cpu_timer_save_area");
Packit Service 501009
	tmp[0]=ULONG(ptr);
Packit Service 501009
	fprintf(fp,"  -cpu timer: %#018lx\n", tmp[0]);
Packit Service 501009
Packit Service 501009
	ptr = lc + MEMBER_OFFSET(lc_struct, "clock_comp_save_area");
Packit Service 501009
	/*
Packit Service 501009
	 * Shift clock comparator by 8 because we got bit positions 0-55
Packit Service 501009
	 * in byte 1 to 8. The first byte is always zero.
Packit Service 501009
	 */
Packit Service 501009
	tmp[0]=ULONG(ptr) << 8;
Packit Service 501009
	fprintf(fp,"  -clock cmp: %#018lx\n", tmp[0]);
Packit Service 501009
Packit Service 501009
	fprintf(fp,"  -general registers:\n");
Packit Service 501009
	ptr = lc + MEMBER_OFFSET(lc_struct, "gpregs_save_area");
Packit Service 501009
	tmp[0]=ULONG(ptr);
Packit Service 501009
	tmp[1]=ULONG(ptr + S390X_WORD_SIZE);
Packit Service 501009
	tmp[2]=ULONG(ptr + 2 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[3]=ULONG(ptr + 3 * S390X_WORD_SIZE);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[0],tmp[1]);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[2],tmp[3]);
Packit Service 501009
	tmp[0]=ULONG(ptr + 4 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[1]=ULONG(ptr + 5 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[2]=ULONG(ptr + 6 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[3]=ULONG(ptr + 7 * S390X_WORD_SIZE);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[0],tmp[1]);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[2],tmp[3]);
Packit Service 501009
	tmp[0]=ULONG(ptr + 8 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[1]=ULONG(ptr + 9 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[2]=ULONG(ptr + 10* S390X_WORD_SIZE);
Packit Service 501009
	tmp[3]=ULONG(ptr + 11* S390X_WORD_SIZE);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[0],tmp[1]);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[2],tmp[3]);
Packit Service 501009
	tmp[0]=ULONG(ptr + 12* S390X_WORD_SIZE);
Packit Service 501009
	tmp[1]=ULONG(ptr + 13* S390X_WORD_SIZE);
Packit Service 501009
	tmp[2]=ULONG(ptr + 14* S390X_WORD_SIZE);
Packit Service 501009
	tmp[3]=ULONG(ptr + 15* S390X_WORD_SIZE);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[0],tmp[1]);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[2],tmp[3]);
Packit Service 501009
Packit Service 501009
	fprintf(fp,"  -access registers:\n");
Packit Service 501009
	ptr = lc + MEMBER_OFFSET(lc_struct, "access_regs_save_area");
Packit Service 501009
	tmp[0]=UINT(ptr);
Packit Service 501009
	tmp[1]=UINT(ptr + 4);
Packit Service 501009
	tmp[2]=UINT(ptr + 2 * 4);
Packit Service 501009
	tmp[3]=UINT(ptr + 3 * 4);
Packit Service 501009
	fprintf(fp,"     %#010lx %#010lx %#010lx %#010lx\n", 
Packit Service 501009
		tmp[0], tmp[1], tmp[2], tmp[3]);
Packit Service 501009
	tmp[0]=UINT(ptr + 4 * 4);
Packit Service 501009
	tmp[1]=UINT(ptr + 5 * 4);
Packit Service 501009
	tmp[2]=UINT(ptr + 6 * 4);
Packit Service 501009
	tmp[3]=UINT(ptr + 7 * 4);
Packit Service 501009
	fprintf(fp,"     %#010lx %#010lx %#010lx %#010lx\n", 
Packit Service 501009
		tmp[0], tmp[1], tmp[2], tmp[3]);
Packit Service 501009
	tmp[0]=UINT(ptr + 8 * 4);
Packit Service 501009
	tmp[1]=UINT(ptr + 9 * 4);
Packit Service 501009
	tmp[2]=UINT(ptr + 10 * 4);
Packit Service 501009
	tmp[3]=UINT(ptr + 11 * 4);
Packit Service 501009
	fprintf(fp,"     %#010lx %#010lx %#010lx %#010lx\n", 
Packit Service 501009
		tmp[0], tmp[1], tmp[2], tmp[3]);
Packit Service 501009
	tmp[0]=UINT(ptr + 12 * 4);
Packit Service 501009
	tmp[1]=UINT(ptr + 13 * 4);
Packit Service 501009
	tmp[2]=UINT(ptr + 14 * 4);
Packit Service 501009
	tmp[3]=UINT(ptr + 15 * 4);
Packit Service 501009
	fprintf(fp,"     %#010lx %#010lx %#010lx %#010lx\n", 
Packit Service 501009
		tmp[0], tmp[1], tmp[2], tmp[3]);
Packit Service 501009
Packit Service 501009
	fprintf(fp,"  -control registers:\n");
Packit Service 501009
	ptr = lc + MEMBER_OFFSET(lc_struct, "cregs_save_area");
Packit Service 501009
	tmp[0]=ULONG(ptr);
Packit Service 501009
	tmp[1]=ULONG(ptr + S390X_WORD_SIZE);
Packit Service 501009
	tmp[2]=ULONG(ptr + 2 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[3]=ULONG(ptr + 3 * S390X_WORD_SIZE);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[0],tmp[1]);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[2],tmp[3]);
Packit Service 501009
	tmp[0]=ULONG(ptr + 4 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[1]=ULONG(ptr + 5 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[2]=ULONG(ptr + 6 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[3]=ULONG(ptr + 7 * S390X_WORD_SIZE);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[0],tmp[1]);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[2],tmp[3]);
Packit Service 501009
	tmp[0]=ULONG(ptr + 8 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[1]=ULONG(ptr + 9 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[2]=ULONG(ptr + 10 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[3]=ULONG(ptr + 11 * S390X_WORD_SIZE);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[0],tmp[1]);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[2],tmp[3]);
Packit Service 501009
	tmp[0]=ULONG(ptr + 12 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[1]=ULONG(ptr + 13 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[2]=ULONG(ptr + 14 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[3]=ULONG(ptr + 15 * S390X_WORD_SIZE);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[0],tmp[1]);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[2],tmp[3]);
Packit Service 501009
Packit Service 501009
	ptr = lc + MEMBER_OFFSET(lc_struct, "floating_pt_save_area");
Packit Service 501009
	fprintf(fp,"  -floating point registers:\n");
Packit Service 501009
	tmp[0]=ULONG(ptr);
Packit Service 501009
	tmp[1]=ULONG(ptr +  S390X_WORD_SIZE);
Packit Service 501009
	tmp[2]=ULONG(ptr + 2 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[3]=ULONG(ptr + 3 * S390X_WORD_SIZE);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[0],tmp[1]);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[2],tmp[3]);
Packit Service 501009
	tmp[0]=ULONG(ptr + 4 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[1]=ULONG(ptr + 5 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[2]=ULONG(ptr + 6 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[3]=ULONG(ptr + 7 * S390X_WORD_SIZE);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[0],tmp[1]);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[2],tmp[3]);
Packit Service 501009
	tmp[0]=ULONG(ptr + 8 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[1]=ULONG(ptr + 9 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[2]=ULONG(ptr + 10 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[3]=ULONG(ptr + 11 * S390X_WORD_SIZE);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[0],tmp[1]);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[2],tmp[3]);
Packit Service 501009
	tmp[0]=ULONG(ptr + 12 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[1]=ULONG(ptr + 13 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[2]=ULONG(ptr + 14 * S390X_WORD_SIZE);
Packit Service 501009
	tmp[3]=ULONG(ptr + 15 * S390X_WORD_SIZE);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[0],tmp[1]);
Packit Service 501009
	fprintf(fp,"     %#018lx %#018lx\n", tmp[2],tmp[3]);
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 *  Get a stack frame combination of pc and ra from the most relevent spot.
Packit Service 501009
 */
Packit Service 501009
static void
Packit Service 501009
s390x_get_stack_frame(struct bt_info *bt, ulong *eip, ulong *esp)
Packit Service 501009
{
Packit Service 501009
	unsigned long ksp, r14;
Packit Service 501009
	int r14_offset;
Packit Service 501009
	char lowcore[LOWCORE_SIZE];
Packit Service 501009
Packit Service 501009
	if(s390x_has_cpu(bt))
Packit Service 501009
		s390x_get_lowcore(bt, lowcore);
Packit Service 501009
Packit Service 501009
	/* get the stack pointer */
Packit Service 501009
	if(esp){
Packit Service 501009
		if (!LIVE() && s390x_has_cpu(bt)) {
Packit Service 501009
			ksp = ULONG(lowcore + MEMBER_OFFSET(lc_struct,
Packit Service 501009
				"gpregs_save_area") + (15 * S390X_WORD_SIZE));
Packit Service 501009
		} else {
Packit Service 501009
			readmem(bt->task + OFFSET(task_struct_thread_ksp), 
Packit Service 501009
				KVADDR, &ksp, sizeof(void *),
Packit Service 501009
				"thread_struct ksp", FAULT_ON_ERROR);
Packit Service 501009
		}
Packit Service 501009
		*esp = ksp;
Packit Service 501009
	} else {
Packit Service 501009
		/* for 'bt -S' */
Packit Service 501009
		ksp=bt->hp->esp;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	/* get the instruction address */
Packit Service 501009
	if(!eip)
Packit Service 501009
		return;
Packit Service 501009
Packit Service 501009
	if(s390x_has_cpu(bt) && esp){
Packit Service 501009
		*eip = ULONG(lowcore + OFFSET(s390_lowcore_psw_save_area) +
Packit Service 501009
			S390X_WORD_SIZE);
Packit Service 501009
	} else {
Packit Service 501009
		if(!STRUCT_EXISTS("stack_frame")){
Packit Service 501009
			r14_offset = 112;
Packit Service 501009
		} else {
Packit Service 501009
			r14_offset = MEMBER_OFFSET("stack_frame","gprs") + 
Packit Service 501009
						   8 * S390X_WORD_SIZE;
Packit Service 501009
		}
Packit Service 501009
		readmem(ksp + r14_offset,KVADDR,&r14,sizeof(void*),"eip",
Packit Service 501009
			FAULT_ON_ERROR);
Packit Service 501009
		*eip=r14; 
Packit Service 501009
	}
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 *  Filter disassembly output if the output radix is not gdb's default 10
Packit Service 501009
 */
Packit Service 501009
static int 
Packit Service 501009
s390x_dis_filter(ulong vaddr, char *inbuf, unsigned int output_radix)
Packit Service 501009
{
Packit Service 501009
	char buf1[BUFSIZE];
Packit Service 501009
	char buf2[BUFSIZE];
Packit Service 501009
	char *colon, *p1;
Packit Service 501009
	int argc;
Packit Service 501009
	char *argv[MAXARGS];
Packit Service 501009
	ulong value;
Packit Service 501009
Packit Service 501009
	if (!inbuf) 
Packit Service 501009
		return TRUE;
Packit Service 501009
/*
Packit Service 501009
 *  For some reason gdb can go off into the weeds translating text addresses,
Packit Service 501009
 *  so this routine both fixes the references as well as imposing the current 
Packit Service 501009
 *  output radix on the translations.
Packit Service 501009
 */
Packit Service 501009
	console("IN: %s", inbuf);
Packit Service 501009
Packit Service 501009
	colon = strstr(inbuf, ":");
Packit Service 501009
Packit Service 501009
	if (colon) {
Packit Service 501009
		sprintf(buf1, "0x%lx <%s>", vaddr,
Packit Service 501009
			value_to_symstr(vaddr, buf2, output_radix));
Packit Service 501009
		sprintf(buf2, "%s%s", buf1, colon);
Packit Service 501009
		strcpy(inbuf, buf2);
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	strcpy(buf1, inbuf);
Packit Service 501009
	argc = parse_line(buf1, argv);
Packit Service 501009
Packit Service 501009
	if ((FIRSTCHAR(argv[argc-1]) == '<') && 
Packit Service 501009
	    (LASTCHAR(argv[argc-1]) == '>')) {
Packit Service 501009
		p1 = rindex(inbuf, '<');
Packit Service 501009
		while ((p1 > inbuf) && 
Packit Service 501009
		    !(STRNEQ(p1, " 0x") || STRNEQ(p1, "\t0x") || STRNEQ(p1, ",0x"))) 
Packit Service 501009
			p1--;
Packit Service 501009
Packit Service 501009
		if (!(STRNEQ(p1, " 0x") || STRNEQ(p1, "\t0x") || STRNEQ(p1, ",0x"))) 
Packit Service 501009
			return FALSE;
Packit Service 501009
		p1++;
Packit Service 501009
Packit Service 501009
		if (!extract_hex(p1, &value, NULLCHAR, TRUE))
Packit Service 501009
			return FALSE;
Packit Service 501009
Packit Service 501009
		sprintf(buf1, "0x%lx <%s>\n", value,
Packit Service 501009
			value_to_symstr(value, buf2, output_radix));
Packit Service 501009
Packit Service 501009
		sprintf(p1, "%s", buf1);
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	console("    %s", inbuf);
Packit Service 501009
Packit Service 501009
	return TRUE;
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 *   Override smp_num_cpus if possible and necessary.
Packit Service 501009
 */
Packit Service 501009
int
Packit Service 501009
s390x_get_smp_cpus(void)
Packit Service 501009
{
Packit Service 501009
	return MAX(get_cpus_online(), get_highest_cpu_online()+1);
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 *  Machine dependent command.
Packit Service 501009
 */
Packit Service 501009
void
Packit Service 501009
s390x_cmd_mach(void)
Packit Service 501009
{
Packit Service 501009
	int c;
Packit Service 501009
Packit Service 501009
	while ((c = getopt(argcnt, args, "cm")) != EOF) {
Packit Service 501009
		switch(c)
Packit Service 501009
		{
Packit Service 501009
		case 'c':
Packit Service 501009
			fprintf(fp,"'-c' option is not implemented on this architecture\n");
Packit Service 501009
			return;
Packit Service 501009
		case 'm':
Packit Service 501009
			fprintf(fp,"'-m' option is not implemented on this architecture\n");
Packit Service 501009
			return;
Packit Service 501009
		default:
Packit Service 501009
			argerrs++;
Packit Service 501009
			break;
Packit Service 501009
		}
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	if (argerrs)
Packit Service 501009
		cmd_usage(pc->curcmd, SYNOPSIS);
Packit Service 501009
Packit Service 501009
	s390x_display_machine_stats();
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
/*
Packit Service 501009
 *  "mach" command output.
Packit Service 501009
 */
Packit Service 501009
static void
Packit Service 501009
s390x_display_machine_stats(void)
Packit Service 501009
{
Packit Service 501009
	struct new_utsname *uts;
Packit Service 501009
	char buf[BUFSIZE];
Packit Service 501009
	ulong mhz;
Packit Service 501009
Packit Service 501009
	uts = &kt->utsname;
Packit Service 501009
Packit Service 501009
	fprintf(fp, "       MACHINE TYPE: %s\n", uts->machine);
Packit Service 501009
	fprintf(fp, "        MEMORY SIZE: %s\n", get_memory_size(buf));
Packit Service 501009
	fprintf(fp, "               CPUS: %d\n", kt->cpus);
Packit Service 501009
	fprintf(fp, "    PROCESSOR SPEED: ");
Packit Service 501009
	if ((mhz = machdep->processor_speed()))
Packit Service 501009
		fprintf(fp, "%ld Mhz\n", mhz);
Packit Service 501009
	else
Packit Service 501009
		fprintf(fp, "(unknown)\n");
Packit Service 501009
	fprintf(fp, "                 HZ: %d\n", machdep->hz);
Packit Service 501009
	fprintf(fp, "          PAGE SIZE: %d\n", PAGESIZE());
Packit Service 501009
	// fprintf(fp, "      L1 CACHE SIZE: %d\n", l1_cache_size());
Packit Service 501009
	fprintf(fp, "KERNEL VIRTUAL BASE: %lx\n", machdep->kvbase);
Packit Service 501009
	fprintf(fp, "KERNEL VMALLOC BASE: %lx\n", vt->vmalloc_start);
Packit Service 501009
	fprintf(fp, "  KERNEL STACK SIZE: %ld\n", STACKSIZE());
Packit Service 501009
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static const char *hook_files[] = {
Packit Service 501009
	"arch/s390x/kernel/entry.S",
Packit Service 501009
	"arch/s390x/kernel/head.S"
Packit Service 501009
};
Packit Service 501009
Packit Service 501009
#define ENTRY_S      ((char **)&hook_files[0])
Packit Service 501009
#define HEAD_S       ((char **)&hook_files[1])
Packit Service 501009
Packit Service 501009
static struct line_number_hook s390x_line_number_hooks[] = {
Packit Service 501009
       {"startup",HEAD_S},
Packit Service 501009
       {"_stext",HEAD_S},
Packit Service 501009
       {"_pstart",HEAD_S},
Packit Service 501009
       {"system_call",ENTRY_S},
Packit Service 501009
       {"sysc_do_svc",ENTRY_S},
Packit Service 501009
       {"sysc_do_restart",ENTRY_S},
Packit Service 501009
       {"sysc_return",ENTRY_S},
Packit Service 501009
       {"sysc_sigpending",ENTRY_S},
Packit Service 501009
       {"sysc_restart",ENTRY_S},
Packit Service 501009
       {"sysc_singlestep",ENTRY_S},
Packit Service 501009
       {"sysc_tracesys",ENTRY_S},
Packit Service 501009
       {"ret_from_fork",ENTRY_S},
Packit Service 501009
       {"pgm_check_handler",ENTRY_S},
Packit Service 501009
       {"io_int_handler",ENTRY_S},
Packit Service 501009
       {"io_return",ENTRY_S},
Packit Service 501009
       {"ext_int_handler",ENTRY_S},
Packit Service 501009
       {"mcck_int_handler",ENTRY_S},
Packit Service 501009
       {"mcck_return",ENTRY_S},
Packit Service 501009
       {"restart_int_handler",ENTRY_S},
Packit Service 501009
       {NULL, NULL}    /* list must be NULL-terminated */
Packit Service 501009
};
Packit Service 501009
Packit Service 501009
static void
Packit Service 501009
s390x_dump_line_number(ulong callpc)
Packit Service 501009
{
Packit Service 501009
	int retries;
Packit Service 501009
	char buf[BUFSIZE], *p;
Packit Service 501009
Packit Service 501009
	retries = 0;
Packit Service 501009
try_closest:
Packit Service 501009
	get_line_number(callpc, buf, FALSE);
Packit Service 501009
Packit Service 501009
	if (strlen(buf)) {
Packit Service 501009
		if (retries) {
Packit Service 501009
			p = strstr(buf, ": ");
Packit Service 501009
			if (p)
Packit Service 501009
				*p = NULLCHAR;
Packit Service 501009
		}
Packit Service 501009
		fprintf(fp, "    %s\n", buf);
Packit Service 501009
	} else {
Packit Service 501009
		if (retries) {
Packit Service 501009
			fprintf(fp, GDB_PATCHED() ?
Packit Service 501009
			  "" : "    (cannot determine file and line number)\n");
Packit Service 501009
		} else {
Packit Service 501009
			retries++;
Packit Service 501009
			callpc = closest_symbol_value(callpc);
Packit Service 501009
			goto try_closest;
Packit Service 501009
		}
Packit Service 501009
	}
Packit Service 501009
}
Packit Service 501009
Packit Service 501009
static int 
Packit Service 501009
s390x_get_kvaddr_ranges(struct vaddr_range *vrp)
Packit Service 501009
{
Packit Service 501009
	int cnt;
Packit Service 501009
	physaddr_t phys1, phys2;
Packit Service 501009
	ulong pp1, pp2;
Packit Service 501009
Packit Service 501009
	cnt = 0;
Packit Service 501009
Packit Service 501009
	vrp[cnt].type = KVADDR_UNITY_MAP;
Packit Service 501009
	vrp[cnt].start = machdep->kvbase;
Packit Service 501009
	vrp[cnt++].end = vt->high_memory;
Packit Service 501009
Packit Service 501009
	vrp[cnt].type = KVADDR_VMALLOC;
Packit Service 501009
	vrp[cnt].start = first_vmalloc_address();
Packit Service 501009
	vrp[cnt++].end = last_vmalloc_address();
Packit Service 501009
Packit Service 501009
	phys1 = (physaddr_t)(0);
Packit Service 501009
	phys2 = (physaddr_t)VTOP(vt->high_memory - PAGESIZE());
Packit Service 501009
	if (phys_to_page(phys1, &pp1) && 
Packit Service 501009
	    phys_to_page(phys2, &pp2) &&
Packit Service 501009
	    (pp1 >= vrp[cnt-1].end)) {
Packit Service 501009
		vrp[cnt].type = KVADDR_VMEMMAP;
Packit Service 501009
		vrp[cnt].start = pp1;
Packit Service 501009
		vrp[cnt++].end = pp2;
Packit Service 501009
	}
Packit Service 501009
Packit Service 501009
	return cnt;
Packit Service 501009
}
Packit Service 501009
#endif  /* S390X */