/*
* Copyright (C) 2009, Mukesh Rathor, Oracle Corp. All rights reserved.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public
* License v2 as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public
* License along with this program; If not, see .
*/
/* This file to impelement functions that run interactively and don't
* involve remote gdb. Eg, print vcpu context and exit. */
#include
#include
#include
#include
#include
#include "gx.h"
extern vcpuid_t max_vcpuid;
extern int guest_bitness;
static void
prnt_32regs(struct xg_gdb_regs32 *r32p)
{
printf("eip:%08x esp:%08x flags:%08x\n", r32p->eip, r32p->esp,
r32p->eflags);
printf("eax:%08x ebx:%08x ecx:%08x edx:%08x\n", r32p->eax,
r32p->ebx, r32p->ecx, r32p->edx);
printf("esi:%08x edi:%08x ebp:%08x\n", r32p->esi, r32p->edi,
r32p->ebp);
printf("cs:%x ds:%x fs:%x gs:%x\n", r32p->cs, r32p->ds, r32p->fs,
r32p->gs);
printf("\n");
}
static void
prnt_64regs(struct xg_gdb_regs64 *r64p)
{
printf("rip:%016"PRIx64" rsp:%016"PRIx64" flags:%08x\n", r64p->rip, r64p->rsp,
r64p->eflags);
printf("rax:%016"PRIx64" rbx:%016"PRIx64" rcx:%016"PRIx64"\n", r64p->rax, r64p->rbx,
r64p->rcx);
printf("rdx:%016"PRIx64" rsi:%016"PRIx64" rdi:%016"PRIx64"\n", r64p->rdx, r64p->rsi,
r64p->rdi);
printf("r08:%016"PRIx64" r09:%016"PRIx64" r10:%016"PRIx64"\n", r64p->r8, r64p->r9,
r64p->r10);
printf("r11:%016"PRIx64" r12:%016"PRIx64" r13:%016"PRIx64"\n", r64p->r11, r64p->r12,
r64p->r13);
printf("r14:%016"PRIx64" r15:%016"PRIx64" rbp:%016"PRIx64"\n", r64p->r14, r64p->r15,
r64p->rbp);
printf("cs:%08x ds:%08x fs:%08x gs:%08x\n", r64p->cs,
r64p->ds, r64p->fs, r64p->gs);
printf("\n");
}
static void
prnt_call_trace32(uint32_t ip, uint32_t sp)
{
int stack_max=10; /* try to print upto 10 entries if possible */
uint32_t loopmax=0, val;
printf("Call Trace:\n");
printf(" [%08x]\n", ip);
while(stack_max > 0) {
if (xg_read_mem((uint64_t)sp, (char *)&val, sizeof(val),0) != 0)
return;
if (val > 0x0c000000) { /* kernel addr */
printf(" [%08x]\n", val);
--stack_max;
}
sp += sizeof(sp);
if (++loopmax > 10000) /* don't go forever */
break;
}
}
static void
prnt_call_trace64(uint64_t ip, uint64_t sp)
{
int stack_max=10; /* try to print upto 10 entries if possible */
uint64_t loopmax=0, val;
printf("Call Trace:\n");
printf(" [%016"PRIx64"]\n", ip);
while(stack_max > 0) {
if (xg_read_mem(sp, (char *)&val, sizeof(val),0) != 0)
return;
if (val > 0xffffffff80000000UL) { /* kernel addr */
printf(" [%016"PRIx64"]\n", val);
--stack_max;
}
sp += sizeof(sp);
if (++loopmax > 10000) /* don't go forever */
break;
}
}
static int
prnt_vcpu_context(vcpuid_t vcpuid)
{
union xg_gdb_regs gregs;
int rc;
printf("\n--> VCPU:%d\n", vcpuid);
rc = xg_regs_read(XG_GPRS, vcpuid, &gregs, guest_bitness);
if (rc) {
gxprt("ERROR: failed to read regs. errno:%d\n", errno);
return 1;
}
if (guest_bitness==32) {
prnt_32regs(&gregs.gregs_32);
prnt_call_trace32(gregs.gregs_32.eip, gregs.gregs_32.esp);
} else {
prnt_64regs(&gregs.gregs_64);
prnt_call_trace64(gregs.gregs_64.rip, gregs.gregs_64.rsp);
}
return 0;
}
/* vcpuid is already checked to be <= max_vcpuid */
int
gx_local_cmd(domid_t domid, vcpuid_t vcpuid)
{
printf("===> Context for DOMID:%d\n", domid);
if (vcpuid == -1) {
int i;
for (i=0; i <= max_vcpuid; i++)
prnt_vcpu_context(i);
} else
prnt_vcpu_context(vcpuid);
return 0;
}