X-Git-Url: http://v3vee.org/palacios/gitweb/gitweb.cgi?a=blobdiff_plain;f=palacios%2Fsrc%2Fpalacios%2Fvmm_debug.c;h=8e1819e9b7521ac84c5e0b9147847c9f941afdb6;hb=6d3926e16dc1ba6ca0144e65b737e432c48aaa91;hp=2a5bd948b49eea76f456afe88b6d5b6cb79613a4;hpb=4869d835f8fdbc29fbcaae1d888c2561e9165b9c;p=palacios.git diff --git a/palacios/src/palacios/vmm_debug.c b/palacios/src/palacios/vmm_debug.c index 2a5bd94..8e1819e 100644 --- a/palacios/src/palacios/vmm_debug.c +++ b/palacios/src/palacios/vmm_debug.c @@ -90,6 +90,20 @@ static int core_handler(struct guest_info * core, uint32_t cmd) { v3_lower_barrier(core->vm_info); break; + case PRINT_ALL: + v3_raise_barrier(core->vm_info, NULL); + +#ifdef V3_CONFIG_TELEMETRY + v3_print_core_telemetry(core); +#endif + v3_print_guest_state(core); + v3_print_arch_state(core); + v3_print_stack(core); + v3_print_backtrace(core); + + v3_lower_barrier(core->vm_info); + break; + } return 0; @@ -98,7 +112,7 @@ static int core_handler(struct guest_info * core, uint32_t cmd) { static int evt_handler(struct v3_vm_info * vm, struct v3_debug_event * evt, void * priv_data) { - V3_Print("Debug Event Handler for core %d\n", evt->core_id); + V3_Print(vm, VCORE_NONE,"Debug Event Handler for core %d cmd=%x\n", evt->core_id, evt->cmd); if (evt->core_id == -1) { int i = 0; @@ -134,14 +148,24 @@ void v3_print_segments(struct v3_segments * segs) { seg_ptr=(struct v3_segment *)segs; char *seg_names[] = {"CS", "DS" , "ES", "FS", "GS", "SS" , "LDTR", "GDTR", "IDTR", "TR", NULL}; - V3_Print("Segments\n"); + V3_Print(VM_NONE, VCORE_NONE, "Segments\n"); for (i = 0; seg_names[i] != NULL; i++) { - V3_Print("\t%s: Sel=%x, base=%p, limit=%x (long_mode=%d, db=%d)\n", seg_names[i], seg_ptr[i].selector, - (void *)(addr_t)seg_ptr[i].base, seg_ptr[i].limit, - seg_ptr[i].long_mode, seg_ptr[i].db); - + V3_Print(VM_NONE, VCORE_NONE, "\t%s: selector=0x%x, base=%p, limit=0x%x type=0x%x system=0x%x dpl=0x%x present=0x%x avail=0x%x long_mode=0x%x db=0x%x granularity=0x%x unusable=0x%x\n", + seg_names[i], + seg_ptr[i].selector, + (void *)(addr_t)seg_ptr[i].base, + seg_ptr[i].limit, + seg_ptr[i].type, + seg_ptr[i].system, + seg_ptr[i].dpl, + seg_ptr[i].present, + seg_ptr[i].avail, + seg_ptr[i].long_mode, + seg_ptr[i].db, + seg_ptr[i].granularity, + seg_ptr[i].unusable); } } @@ -156,10 +180,10 @@ void v3_print_ctrl_regs(struct guest_info * core) { reg_ptr = (v3_reg_t *)regs; - V3_Print("Ctrl Regs:\n"); + V3_Print(core->vm_info, core,"Ctrl Regs:\n"); for (i = 0; reg_names[i] != NULL; i++) { - V3_Print("\t%s=0x%p (at %p)\n", reg_names[i], (void *)(addr_t)reg_ptr[i], &(reg_ptr[i])); + V3_Print(core->vm_info, core, "\t%s=0x%p (at %p)\n", reg_names[i], (void *)(addr_t)reg_ptr[i], &(reg_ptr[i])); } @@ -188,12 +212,12 @@ static int v3_print_disassembly(struct guest_info * core) { /* start disassembly 64 bytes before current RIP, continue 32 bytes after */ rip = (addr_t) core->rip - 64; while ((int) (rip - core->rip) < 32) { - V3_Print("disassembly step\n"); + V3_Print(info->vm_info, info, "disassembly step\n"); /* always print RIP, even if the instructions before were bad */ if (!passed_rip && rip >= core->rip) { if (rip != core->rip) { - V3_Print("***** bad disassembly up to this point *****\n"); + V3_Print(info->vm_info, info, "***** bad disassembly up to this point *****\n"); rip = core->rip; } passed_rip = 1; @@ -222,16 +246,16 @@ static int v3_print_disassembly(struct guest_info * core) { void v3_print_guest_state(struct guest_info * core) { addr_t linear_addr = 0; - V3_Print("RIP: %p\n", (void *)(addr_t)(core->rip)); + V3_Print(core->vm_info, core, "RIP: %p\n", (void *)(addr_t)(core->rip)); linear_addr = get_addr_linear(core, core->rip, &(core->segments.cs)); - V3_Print("RIP Linear: %p\n", (void *)linear_addr); + V3_Print(core->vm_info, core, "RIP Linear: %p\n", (void *)linear_addr); - V3_Print("NumExits: %u\n", (uint32_t)core->num_exits); + V3_Print(core->vm_info, core, "NumExits: %u\n", (uint32_t)core->num_exits); - V3_Print("IRQ STATE: started=%d, pending=%d\n", + V3_Print(core->vm_info, core, "IRQ STATE: started=%d, pending=%d\n", core->intr_core_state.irq_started, core->intr_core_state.irq_pending); - V3_Print("EXCP STATE: err_code_valid=%d, err_code=%x\n", + V3_Print(core->vm_info, core, "EXCP STATE: err_code_valid=%d, err_code=%x\n", core->excp_state.excp_error_code_valid, core->excp_state.excp_error_code); @@ -240,14 +264,19 @@ void v3_print_guest_state(struct guest_info * core) { v3_print_ctrl_regs(core); if (core->shdw_pg_mode == SHADOW_PAGING) { - V3_Print("Shadow Paging Guest Registers:\n"); - V3_Print("\tGuest CR0=%p\n", (void *)(addr_t)(core->shdw_pg_state.guest_cr0)); - V3_Print("\tGuest CR3=%p\n", (void *)(addr_t)(core->shdw_pg_state.guest_cr3)); - V3_Print("\tGuest EFER=%p\n", (void *)(addr_t)(core->shdw_pg_state.guest_efer.value)); + V3_Print(core->vm_info, core, "Shadow Paging Guest Registers:\n"); + V3_Print(core->vm_info, core, "\tGuest CR0=%p\n", (void *)(addr_t)(core->shdw_pg_state.guest_cr0)); + V3_Print(core->vm_info, core, "\tGuest CR3=%p\n", (void *)(addr_t)(core->shdw_pg_state.guest_cr3)); + V3_Print(core->vm_info, core, "\tGuest EFER=%p\n", (void *)(addr_t)(core->shdw_pg_state.guest_efer.value)); // CR4 } v3_print_GPRs(core); + v3_print_idt(core,core->segments.idtr.base); + v3_print_gdt(core,core->segments.gdtr.base); + v3_print_ldt(core,core->segments.ldtr.base); + v3_print_tss(core,core->segments.tr.base); + v3_print_mem_map(core->vm_info); v3_print_stack(core); @@ -265,10 +294,10 @@ void v3_print_arch_state(struct guest_info * core) { void v3_print_guest_state_all(struct v3_vm_info * vm) { int i = 0; - V3_Print("VM Core states for %s\n", vm->name); + V3_Print(vm, VCORE_NONE,"VM Core states for %s\n", vm->name); for (i = 0; i < 80; i++) { - V3_Print("-"); + V3_Print(vm, VCORE_NONE, "-"); } for (i = 0; i < vm->num_cores; i++) { @@ -276,10 +305,10 @@ void v3_print_guest_state_all(struct v3_vm_info * vm) { } for (i = 0; i < 80; i++) { - V3_Print("-"); + V3_Print(vm, VCORE_NONE, "-"); } - V3_Print("\n"); + V3_Print(vm, VCORE_NONE, "\n"); } @@ -292,32 +321,32 @@ void v3_print_stack(struct guest_info * core) { linear_addr = get_addr_linear(core, core->vm_regs.rsp, &(core->segments.ss)); - V3_Print("Stack at %p:\n", (void *)linear_addr); + V3_Print(core->vm_info, core, "Stack at %p:\n", (void *)linear_addr); if (core->mem_mode == PHYSICAL_MEM) { if (v3_gpa_to_hva(core, linear_addr, &host_addr) == -1) { - PrintError("Could not translate Stack address\n"); + PrintError(core->vm_info, core, "Could not translate Stack address\n"); return; } } else if (core->mem_mode == VIRTUAL_MEM) { if (v3_gva_to_hva(core, linear_addr, &host_addr) == -1) { - PrintError("Could not translate Virtual Stack address\n"); + PrintError(core->vm_info, core, "Could not translate Virtual Stack address\n"); return; } } - V3_Print("Host Address of rsp = 0x%p\n", (void *)host_addr); + V3_Print(core->vm_info, core, "Host Address of rsp = 0x%p\n", (void *)host_addr); // We start i at one because the current stack pointer points to an unused stack element for (i = 0; i <= 24; i++) { if (cpu_mode == REAL) { - V3_Print("\t0x%.4x\n", *((uint16_t *)host_addr + (i * 2))); + V3_Print(core->vm_info, core, "\t0x%.4x\n", *((uint16_t *)host_addr + (i * 2))); } else if (cpu_mode == LONG) { - V3_Print("\t%p\n", (void *)*(addr_t *)(host_addr + (i * 8))); + V3_Print(core->vm_info, core, "\t%p\n", (void *)*(addr_t *)(host_addr + (i * 8))); } else { // 32 bit stacks... - V3_Print("\t0x%.8x\n", *(uint32_t *)(host_addr + (i * 4))); + V3_Print(core->vm_info, core, "\t0x%.8x\n", *(uint32_t *)(host_addr + (i * 4))); } } @@ -330,8 +359,8 @@ void v3_print_backtrace(struct guest_info * core) { v3_cpu_mode_t cpu_mode = v3_get_vm_cpu_mode(core); struct v3_cfg_file * system_map = v3_cfg_get_file(core->vm_info, "System.map"); - V3_Print("Performing Backtrace for Core %d\n", core->vcpu_id); - V3_Print("\tRSP=%p, RBP=%p\n", (void *)core->vm_regs.rsp, (void *)core->vm_regs.rbp); + V3_Print(core->vm_info, core, "Performing Backtrace for Core %d\n", core->vcpu_id); + V3_Print(core->vm_info, core, "\tRSP=%p, RBP=%p\n", (void *)core->vm_regs.rsp, (void *)core->vm_regs.rbp); gla_rbp = get_addr_linear(core, core->vm_regs.rbp, &(core->segments.ss)); @@ -344,12 +373,12 @@ void v3_print_backtrace(struct guest_info * core) { if (core->mem_mode == PHYSICAL_MEM) { if (v3_gpa_to_hva(core, gla_rbp, &hva_rbp) == -1) { - PrintError("Could not translate Stack address\n"); + PrintError(core->vm_info, core, "Could not translate Stack address\n"); return; } } else if (core->mem_mode == VIRTUAL_MEM) { if (v3_gva_to_hva(core, gla_rbp, &hva_rbp) == -1) { - PrintError("Could not translate Virtual Stack address\n"); + PrintError(core->vm_info, core, "Could not translate Virtual Stack address\n"); return; } } @@ -377,17 +406,21 @@ void v3_print_backtrace(struct guest_info * core) { tmp_ptr += 3; // pass over symbol type if (sym_offset > rip_val) { - char * end_ptr = strchr(sym_ptr, '\n'); - - if (end_ptr) { - *end_ptr = 0; // null terminate symbol... + if (sym_ptr) { + char * end_ptr = strchr(sym_ptr, '\n'); + + if (end_ptr) { + *end_ptr = 0; // null terminate symbol... + } + sym_name = sym_ptr; + } else { + sym_name = NULL; } - - sym_name = sym_ptr; break; } sym_ptr = tmp_ptr; + { char * end_ptr2 = strchr(tmp_ptr, '\n'); @@ -405,18 +438,18 @@ void v3_print_backtrace(struct guest_info * core) { } if (cpu_mode == REAL) { - V3_Print("Next RBP=0x%.4x, RIP=0x%.4x (%s)\n", + V3_Print(core->vm_info, core, "Next RBP=0x%.4x, RIP=0x%.4x (%s)\n", *(uint16_t *)hva_rbp,*(uint16_t *)hva_rip, sym_name); gla_rbp = *(uint16_t *)hva_rbp; } else if (cpu_mode == LONG) { - V3_Print("Next RBP=%p, RIP=%p (%s)\n", + V3_Print(core->vm_info, core, "Next RBP=%p, RIP=%p (%s)\n", (void *)*(uint64_t *)hva_rbp, (void *)*(uint64_t *)hva_rip, sym_name); gla_rbp = *(uint64_t *)hva_rbp; } else { - V3_Print("Next RBP=0x%.8x, RIP=0x%.8x (%s)\n", + V3_Print(core->vm_info, core, "Next RBP=0x%.8x, RIP=0x%.8x (%s)\n", *(uint32_t *)hva_rbp, *(uint32_t *)hva_rip, sym_name); gla_rbp = *(uint32_t *)hva_rbp; @@ -436,13 +469,131 @@ void v3_print_GPRs(struct guest_info * core) { reg_ptr = (v3_reg_t *)regs; - V3_Print("32 bit GPRs:\n"); + V3_Print(info->vm_info, info, "32 bit GPRs:\n"); for (i = 0; reg_names[i] != NULL; i++) { - V3_Print("\t%s=0x%p (at %p)\n", reg_names[i], (void *)(addr_t)reg_ptr[i], &(reg_ptr[i])); + V3_Print(info->vm_info, info, "\t%s=0x%p (at %p)\n", reg_names[i], (void *)(addr_t)reg_ptr[i], &(reg_ptr[i])); } } +void v3_print_idt(struct guest_info * core, addr_t idtr_base) { + addr_t base_hva; + + if (v3_get_vm_cpu_mode(core)!=LONG) { + V3_Print(core->vm_info, core, "= IDT ========\n"); + V3_Print(core->vm_info, core, "(currently only supported in long mode)\n"); + return; + } + + + if (core->mem_mode == PHYSICAL_MEM) { + if (v3_gpa_to_hva(core, + get_addr_linear(core, idtr_base, &(core->segments.cs)), + &base_hva)) { + PrintError(core->vm_info, core, "Cannot translate address\n"); + return; + } + PrintError(core->vm_info, core, "Kind of weird that we got here.... physical mem?\n"); + } else if (core->mem_mode == VIRTUAL_MEM) { + if (v3_gva_to_hva(core, + get_addr_linear(core, idtr_base, &(core->segments.cs)), + &base_hva)) { + PrintError(core->vm_info, core, "Cannot translate address\n"); + return; + } + } + + // SANITY CHECK + if (idtr_base != get_addr_linear(core, idtr_base, &(core->segments.cs))) { + PrintError(core->vm_info, core, "idtr base address != linear translation, might be something funky with cs\n"); + } + + if (!base_hva) { + PrintError(core->vm_info, core "idtr address does not translate! skipping.\n"); + return ; + } + + int i; + char *types[16] = {" ILGL","aTSS16"," LDT","bTSS16","call16"," task","intr16","trap16", + " ILGL","aTSS32"," ILGL","bTSS32","call32"," ILGL","intr32","trap32"}; + + struct int_trap_gate_lgcy * entry; + entry = (struct int_trap_gate_lgcy *)base_hva; + V3_Print(core->vm_info, core, "= IDT ========\n"); + V3_Print(core->vm_info, core, " # | hex | selector | si:ti:rpl | offset | type | dpl | s | p\n"); + for (i = 0; i < NUM_IDT_ENTRIES; i++) { + uint32_t tmp = entry->selector; + struct segment_selector * seg = (struct segment_selector *)(&tmp); + V3_Print(core->vm_info, core, "%3d | %3x | %04x | %03x:%x:%x | %04x%04x | %s | %x | %x | %x | %x\n", i, i, + entry->selector, + seg->index, seg->ti, seg->rpl, + entry->offset_hi, entry->offset_lo, + types[entry->type], entry->dpl, entry->s, entry->p); + entry++; + } +} + +void v3_print_gdt(struct guest_info * core, addr_t gdtr_base) { + addr_t base_hva; + + if (v3_get_vm_cpu_mode(core)!=LONG) { + V3_Print(core->vm_info, core, "= GDT ========\n"); + V3_Print(core->vm_info, core, "(currently only supported in long mode)\n"); + return; + } + + if (core->mem_mode == PHYSICAL_MEM) { + if (v3_gpa_to_hva(core, + get_addr_linear(core, gdtr_base, &(core->segments.cs)), + &base_hva)) { + PrintError(core->vm_info, core, "Cannot translate address\n"); + return; + } + PrintError(core->vm_info, core, "Kind of weird that we got here.... physical mem?\n"); + } else if (core->mem_mode == VIRTUAL_MEM) { + if (v3_gva_to_hva(core, + get_addr_linear(core, gdtr_base, &(core->segments.cs)), + &base_hva)) { + PrintError(core->vm_info, core, "Cannot translate address\n"); + return; + } + } + + // SANITY CHECK + if (gdtr_base != get_addr_linear(core, gdtr_base, &(core->segments.cs))) { + PrintError(core->vm_info, core, "gdtr base address != linear translation, might be something funky with cs\n"); + } + + if (!base_hva) { + PrintError(core->vm_info, core "gdtr address does not translate! skipping.\n"); + return ; + } + + int i; + char* cd[2] = {"data","code"}; + // TODO: handle possibility of gate/segment descriptor + + struct code_desc_lgcy * entry; + entry = (struct code_desc_long *)base_hva; + V3_Print(core->vm_info, core, "= GDT ========\n"); + V3_Print(core->vm_info, core, " # | hex | limit | base | c/d | dpl | p\n"); + for (i = 0; i < NUM_GDT_ENTRIES; i++) { + V3_Print(core->vm_info, core, "%3d | %3x | %x%04x | %02x%02x%04x | %s | %x | %x\n", i, i, + entry->limit_hi, entry->limit_lo, + entry->base_hi, entry->base_mid, entry->base_lo, + cd[entry->one1], entry->dpl, entry->p); + entry++; + } +} + +void v3_print_gp_error(struct guest_info * core, addr_t exit_info1) { + struct selector_error_code * error = (struct selector_error_code *)(&exit_info1); + + V3_Print(core->vm_info, core, " selector index: %x, TI: %x, IDT: %x, EXT: %x (error=%llx)\n", + error->index, error->ti, error->idt, error->ext, + (unsigned long long)exit_info1); +} + #elif __V3_64BIT__ void v3_print_GPRs(struct guest_info * core) { @@ -454,11 +605,255 @@ void v3_print_GPRs(struct guest_info * core) { reg_ptr = (v3_reg_t *)regs; - V3_Print("64 bit GPRs:\n"); + V3_Print(core->vm_info, core, "64 bit GPRs:\n"); for (i = 0; reg_names[i] != NULL; i++) { - V3_Print("\t%s=0x%p (at %p)\n", reg_names[i], (void *)(addr_t)reg_ptr[i], &(reg_ptr[i])); + V3_Print(core->vm_info, core, "\t%s=0x%p (at %p)\n", reg_names[i], (void *)(addr_t)reg_ptr[i], &(reg_ptr[i])); + } +} + +void v3_print_idt(struct guest_info * core, addr_t idtr_base) { + addr_t base_hva; + + if (v3_get_vm_cpu_mode(core)!=LONG) { + V3_Print(core->vm_info, core, "= IDT ========\n"); + V3_Print(core->vm_info, core, "(currently only supported in long mode)\n"); + return; + } + + if (core->mem_mode == PHYSICAL_MEM) { + if (v3_gpa_to_hva(core, + get_addr_linear(core, idtr_base, &(core->segments.cs)), + &base_hva)) { + PrintError(core->vm_info, core, "Cannot translate address\n"); + return; + } + } else if (core->mem_mode == VIRTUAL_MEM) { + if (v3_gva_to_hva(core, + get_addr_linear(core, idtr_base, &(core->segments.cs)), + &base_hva)) { + PrintError(core->vm_info, core, "Cannot translate address\n"); + return; + } } + + // SANITY CHECK + if (idtr_base != get_addr_linear(core, idtr_base, &(core->segments.cs))) { + PrintError(core->vm_info, core, "idtr base address != linear translation, might be something funky with cs\n"); + } + + if (!base_hva) { + PrintError(core->vm_info, core, "idtr address does not translate! skipping.\n"); + return ; + } + + int i; + char *types[16] = {"ILGL","ILGL"," LDT","ILGL","ILGL","ILGL","ILGL","ILGL","ILGL", + "aTSS","ILGL","bTSS","call","ILGL","intr","trap"}; + + struct int_trap_gate_long * entry; + entry = (struct int_trap_gate_long *)base_hva; + V3_Print(core->vm_info, core, "= IDT ========\n"); + V3_Print(core->vm_info, core, " # | hex | selector | si:ti:rpl | offset | type | dpl | s | r | p\n"); + for (i = 0; i < NUM_IDT_ENTRIES; i++) { + uint32_t tmp = entry->selector; + struct segment_selector * seg = (struct segment_selector *)(&tmp); + V3_Print(core->vm_info, core, "%3d | %3x | %04x | %03x:%x:%x | %08x%04x%04x | %s | %x | %x | %x | %x\n", i, i, + entry->selector, + seg->index, seg->ti, seg->rpl, + entry->offset_hi, entry->offset_mid, entry->offset_lo, + types[entry->type], entry->dpl, entry->s, + entry->s, entry->p); + entry++; + } +} + +void v3_print_gdt(struct guest_info * core, addr_t gdtr_base) { + addr_t base_hva; + + if (v3_get_vm_cpu_mode(core)!=LONG) { + V3_Print(core->vm_info, core, "= GDT ========\n"); + V3_Print(core->vm_info, core, "(currently only supported in long mode)\n"); + return; + } + + if (core->mem_mode == PHYSICAL_MEM) { + if (v3_gpa_to_hva(core, + get_addr_linear(core, gdtr_base, &(core->segments.cs)), + &base_hva)) { + PrintError(core->vm_info, core, "Cannot translate address\n"); + return; + } + } else if (core->mem_mode == VIRTUAL_MEM) { + if (v3_gva_to_hva(core, + get_addr_linear(core, gdtr_base, &(core->segments.cs)), + &base_hva)) { + PrintError(core->vm_info, core, "Cannot translate address\n"); + return; + } + } + + // SANITY CHECK + if (gdtr_base != get_addr_linear(core, gdtr_base, &(core->segments.cs))) { + PrintError(core->vm_info, core, "gdtr base address != linear translation, might be something funky with cs\n"); + } + + if (!base_hva) { + PrintError(core->vm_info, core, "gdtr address does not translate! skipping.\n"); + return ; + } + + int i; + char* cd[2] = {"data","code"}; + // TODO: handle possibility of gate/segment descriptor + + struct code_desc_long * entry; + entry = (struct code_desc_long *)base_hva; + V3_Print(core->vm_info, core, "= GDT ========\n"); + V3_Print(core->vm_info, core, " # | hex | limit | base | c/d | dpl | p\n"); + for (i = 0; i < NUM_GDT_ENTRIES; i++) { + V3_Print(core->vm_info, core, "%3d | %3x | %x%04x | %02x%02x%04x | %s | %x | %x\n", i, i, + entry->limit_hi, entry->limit_lo, + entry->base_hi, entry->base_mid, entry->base_lo, + cd[entry->one1], entry->dpl, entry->p); + entry++; + } +} + +void v3_print_ldt(struct guest_info * core, addr_t ldtr_base) { + addr_t base_hva; + + if (v3_get_vm_cpu_mode(core)!=LONG) { + V3_Print(core->vm_info, core, "= LDT ========\n"); + V3_Print(core->vm_info, core, "(currently only supported in long mode)\n"); + return; + } + + V3_Print(core->vm_info, core, "= LDT ========\n"); + + if (ldtr_base == 0) { + V3_Print(core->vm_info, core, " (no LDT is installed)\n"); + return; + } + + if (core->mem_mode == PHYSICAL_MEM) { + if (v3_gpa_to_hva(core, + get_addr_linear(core, ldtr_base, &(core->segments.cs)), + &base_hva)) { + PrintError(core->vm_info, core, "Cannot translate address\n"); + return; + } + } else if (core->mem_mode == VIRTUAL_MEM) { + if (v3_gva_to_hva(core, + get_addr_linear(core, ldtr_base, &(core->segments.cs)), + &base_hva)) { + PrintError(core->vm_info, core, "Cannot translate address\n"); + return; + } + } + + // SANITY CHECK + if (ldtr_base != get_addr_linear(core, ldtr_base, &(core->segments.cs))) { + PrintError(core->vm_info, core, "ldtr base address != linear translation, might be something funky with cs\n"); + } + + if (!base_hva) { + PrintError(core->vm_info, core, "ldtr address does not translate! skipping.\n"); + return ; + } + + int i; + char* cd[2] = {"data","code"}; + // TODO: handle possibility of gate/segment descriptor + + struct code_desc_long * entry; + entry = (struct code_desc_long *)base_hva; + V3_Print(core->vm_info, core, " # | hex | limit | base | c/d | dpl | p\n"); + for (i = 0; i < NUM_LDT_ENTRIES; i++) { + V3_Print(core->vm_info, core, "%3d | %3x | %x%04x | %02x%02x%04x | %s | %x | %x\n", i, i, + entry->limit_hi, entry->limit_lo, + entry->base_hi, entry->base_mid, entry->base_lo, + cd[entry->one1], entry->dpl, entry->p); + entry++; + } +} + +void v3_print_tss(struct guest_info * core, addr_t tr_base) { + addr_t base_hva; + struct tss_long *t; + + if (v3_get_vm_cpu_mode(core)!=LONG) { + V3_Print(core->vm_info, core, "= TSS ========\n"); + V3_Print(core->vm_info, core, "(currently only supported in long mode)\n"); + return; + } + + V3_Print(core->vm_info, core, "= TSS ========\n"); + + if (tr_base == 0) { + V3_Print(core->vm_info, core, " (no TSS is installed)\n"); + return; + } + + if (core->mem_mode == PHYSICAL_MEM) { + if (v3_gpa_to_hva(core, + get_addr_linear(core, tr_base, &(core->segments.cs)), + &base_hva)) { + PrintError(core->vm_info, core, "Cannot translate address\n"); + return; + } + } else if (core->mem_mode == VIRTUAL_MEM) { + if (v3_gva_to_hva(core, + get_addr_linear(core, tr_base, &(core->segments.cs)), + &base_hva)) { + PrintError(core->vm_info, core, "Cannot translate address\n"); + return; + } + } + + // SANITY CHECK + if (tr_base != get_addr_linear(core, tr_base, &(core->segments.cs))) { + PrintError(core->vm_info, core, "tr base address != linear translation, might be something funky with cs\n"); + } + + if (!base_hva) { + PrintError(core->vm_info, core, "tr address does not translate! skipping.\n"); + return ; + } + + t=(struct tss_long*)base_hva; + + V3_Print(core->vm_info, core," res1 : 0x%llx\n", (uint64_t) t->res1); + V3_Print(core->vm_info, core," rsp0 : 0x%llx\n", t->rsp0); + V3_Print(core->vm_info, core," rsp1 : 0x%llx\n", t->rsp1); + V3_Print(core->vm_info, core," rsp2 : 0x%llx\n", t->rsp2); + V3_Print(core->vm_info, core," res2 : 0x%llx\n", t->res2); + V3_Print(core->vm_info, core," ist1 : 0x%llx\n", t->ist1); + V3_Print(core->vm_info, core," ist2 : 0x%llx\n", t->ist2); + V3_Print(core->vm_info, core," ist3 : 0x%llx\n", t->ist3); + V3_Print(core->vm_info, core," ist4 : 0x%llx\n", t->ist4); + V3_Print(core->vm_info, core," ist5 : 0x%llx\n", t->ist5); + V3_Print(core->vm_info, core," ist6 : 0x%llx\n", t->ist6); + V3_Print(core->vm_info, core," ist7 : 0x%llx\n", t->ist7); + V3_Print(core->vm_info, core," res3 : 0x%llx\n", t->res3); + V3_Print(core->vm_info, core," res4 : 0x%llx\n", (uint64_t) t->res4); + V3_Print(core->vm_info, core," iomap_base : 0x%llx\n", (uint64_t) t->iomap_base); + V3_Print(core->vm_info, core," (following io permission bitmap not currently printed)\n"); + +} + +void v3_print_gp_error(struct guest_info * core, addr_t exit_info1) { + struct selector_error_code * error = (struct selector_error_code *)(&exit_info1); + + if (v3_get_vm_cpu_mode(core)!=LONG) { + V3_Print(core->vm_info, core, "= IDT ========\n"); + V3_Print(core->vm_info, core, "(currently only supported in long mode)\n"); + return; + } + + V3_Print(core->vm_info, core, " selector index: %x, TI: %x, IDT: %x, EXT: %x (error=%llx)\n", + error->index, error->ti, error->idt, error->ext, + (unsigned long long)exit_info1); } #endif