This patch switches both libkvm as well as the qemu pieces over to the
new guest debug interface. It comes with full support for software-based
breakpoints (via guest code modification), hardware-assisted breakpoints
and watchpoints (x86-only so far).

Breakpoint management is done inside qemu-kvm, transparent to gdbstub
and also avoiding that the gdb frontend takes over. This allows for
running debuggers inside the guest while guest debugging it active,
because the host can cleanly tell apart host- and guest-originated
breakpoint events.

Yet improvable are x86 corner cases when using single-step ("forgotten"
debug flags on the guest's stack). And, of course, the yet empty non-x86
helper functions have to be populated.

Signed-off-by: Jan Kiszka <[EMAIL PROTECTED]>
---
 libkvm/libkvm.c         |   14 ++-
 libkvm/libkvm.h         |    9 +-
 qemu/exec.c             |   10 +-
 qemu/gdbstub.c          |   18 ++--
 qemu/gdbstub.h          |    7 +
 qemu/qemu-kvm-ia64.c    |   35 ++++++++
 qemu/qemu-kvm-powerpc.c |   35 ++++++++
 qemu/qemu-kvm-x86.c     |  172 +++++++++++++++++++++++++++++++++++++++++++
 qemu/qemu-kvm.c         |  192 ++++++++++++++++++++++++++++++++++++++++++------
 qemu/qemu-kvm.h         |   29 +++++++
 user/main.c             |    7 +
 11 files changed, 489 insertions(+), 39 deletions(-)

Index: b/libkvm/libkvm.c
===================================================================
--- a/libkvm/libkvm.c
+++ b/libkvm/libkvm.c
@@ -674,7 +674,13 @@ static int handle_io(kvm_context_t kvm,
 
 int handle_debug(kvm_context_t kvm, int vcpu)
 {
-       return kvm->callbacks->debug(kvm->opaque, vcpu);
+#ifdef KVM_CAP_SET_GUEST_DEBUG
+       struct kvm_run *run = kvm->run[vcpu];
+
+       return kvm->callbacks->debug(kvm->opaque, vcpu, &run->debug.arch);
+#else
+       return 0;
+#endif
 }
 
 int kvm_get_regs(kvm_context_t kvm, int vcpu, struct kvm_regs *regs)
@@ -930,10 +936,12 @@ int kvm_inject_irq(kvm_context_t kvm, in
        return ioctl(kvm->vcpu_fd[vcpu], KVM_INTERRUPT, &intr);
 }
 
-int kvm_guest_debug(kvm_context_t kvm, int vcpu, struct kvm_debug_guest *dbg)
+#ifdef KVM_CAP_SET_GUEST_DEBUG
+int kvm_set_guest_debug(kvm_context_t kvm, int vcpu, struct kvm_guest_debug 
*dbg)
 {
-       return ioctl(kvm->vcpu_fd[vcpu], KVM_DEBUG_GUEST, dbg);
+       return ioctl(kvm->vcpu_fd[vcpu], KVM_SET_GUEST_DEBUG, dbg);
 }
+#endif
 
 int kvm_set_signal_mask(kvm_context_t kvm, int vcpu, const sigset_t *sigset)
 {
Index: b/qemu/exec.c
===================================================================
--- a/qemu/exec.c
+++ b/qemu/exec.c
@@ -1488,9 +1488,13 @@ void cpu_single_step(CPUState *env, int
 #if defined(TARGET_HAS_ICE)
     if (env->singlestep_enabled != enabled) {
         env->singlestep_enabled = enabled;
-        /* must flush all the translated code to avoid inconsistancies */
-        /* XXX: only flush what is necessary */
-        tb_flush(env);
+        if (kvm_enabled())
+            kvm_update_guest_debug(env, 0);
+        else {
+            /* must flush all the translated code to avoid inconsistancies */
+            /* XXX: only flush what is necessary */
+            tb_flush(env);
+        }
     }
 #endif
 }
Index: b/qemu/qemu-kvm.c
===================================================================
--- a/qemu/qemu-kvm.c
+++ b/qemu/qemu-kvm.c
@@ -20,6 +20,8 @@ int kvm_pit = 1;
 #include "console.h"
 #include "block.h"
 #include "compatfd.h"
+#include "gdbstub.h"
+#include "monitor.h"
 
 #include "qemu-kvm.h"
 #include <libkvm.h>
@@ -72,7 +74,7 @@ pthread_t io_thread;
 static int io_thread_fd = -1;
 static int io_thread_sigfd = -1;
 
-static int kvm_debug_stop_requested;
+static CPUState *kvm_debug_cpu_requested;
 
 static inline unsigned long kvm_get_thread_id(void)
 {
@@ -599,9 +601,10 @@ int kvm_main_loop(void)
             qemu_system_powerdown();
         else if (qemu_reset_requested())
            qemu_kvm_system_reset();
-       else if (kvm_debug_stop_requested) {
+       else if (kvm_debug_cpu_requested) {
+           mon_set_cpu(kvm_debug_cpu_requested);
            vm_stop(EXCP_DEBUG);
-           kvm_debug_stop_requested = 0;
+           kvm_debug_cpu_requested = NULL;
        }
     }
 
@@ -611,12 +614,18 @@ int kvm_main_loop(void)
     return 0;
 }
 
-static int kvm_debug(void *opaque, int vcpu)
+#ifdef KVM_CAP_SET_GUEST_DEBUG
+int kvm_debug(void *opaque, int vcpu, struct kvm_debug_exit_arch *arch_info)
 {
-    kvm_debug_stop_requested = 1;
-    vcpu_info[vcpu].stopped = 1;
-    return 1;
+    int handle = kvm_arch_debug(arch_info);
+
+    if (handle) {
+       kvm_debug_cpu_requested = cpu_single_env;
+       vcpu_info[vcpu].stopped = 1;
+    }
+    return handle;
 }
+#endif
 
 static int kvm_inb(void *opaque, uint16_t addr, uint8_t *data)
 {
@@ -718,7 +727,9 @@ static int kvm_shutdown(void *opaque, in
 }
  
 static struct kvm_callbacks qemu_kvm_ops = {
+#ifdef KVM_CAP_SET_GUEST_DEBUG
     .debug = kvm_debug,
+#endif
     .inb   = kvm_inb,
     .inw   = kvm_inw,
     .inl   = kvm_inl,
@@ -835,34 +846,171 @@ int kvm_qemu_init_env(CPUState *cenv)
     return kvm_arch_qemu_init_env(cenv);
 }
 
-struct kvm_guest_debug_data {
-    struct kvm_debug_guest dbg;
+#ifdef KVM_CAP_SET_GUEST_DEBUG
+struct kvm_sw_breakpoint *first_sw_breakpoint;
+
+struct kvm_sw_breakpoint *kvm_find_sw_breakpoint(target_ulong pc)
+{
+    struct kvm_sw_breakpoint *bp = first_sw_breakpoint;
+
+    while (bp) {
+       if (bp->pc == pc)
+           break;
+       bp = bp->next;
+    }
+    return bp;
+}
+
+struct kvm_set_guest_debug_data {
+    struct kvm_guest_debug dbg;
     int err;
 };
 
-void kvm_invoke_guest_debug(void *data)
+void kvm_invoke_set_guest_debug(void *data)
 {
-    struct kvm_guest_debug_data *dbg_data = data;
+    struct kvm_set_guest_debug_data *dbg_data = data;
 
-    dbg_data->err = kvm_guest_debug(kvm_context, cpu_single_env->cpu_index,
-                                    &dbg_data->dbg);
+    dbg_data->err = kvm_set_guest_debug(kvm_context, cpu_single_env->cpu_index,
+                                        &dbg_data->dbg);
 }
 
-int kvm_update_debugger(CPUState *env)
+int kvm_update_guest_debug(CPUState *env, unsigned long reinject_trap)
 {
-    struct kvm_guest_debug_data data;
+    struct kvm_set_guest_debug_data data;
 
-    memset(data.dbg.breakpoints, 0, sizeof(data.dbg.breakpoints));
+    data.dbg.control = 0;
+    if (env->singlestep_enabled)
+       data.dbg.control = KVM_GUESTDBG_ENABLE | KVM_GUESTDBG_SINGLESTEP;
 
-    data.dbg.enabled = 0;
-    if (env->singlestep_enabled) {
-       data.dbg.enabled = 1;
-       data.dbg.singlestep = env->singlestep_enabled;
-    }
-    on_vcpu(env, kvm_invoke_guest_debug, &data);
+    kvm_arch_update_guest_debug(env, &data.dbg);
+    data.dbg.control |= reinject_trap;
+
+    on_vcpu(env, kvm_invoke_set_guest_debug, &data);
     return data.err;
 }
 
+int kvm_insert_breakpoint(target_ulong addr, target_ulong len, int type)
+{
+    struct kvm_sw_breakpoint *bp;
+    CPUState *env;
+    int err;
+
+    if (type == GDB_BREAKPOINT_SW) {
+       bp = kvm_find_sw_breakpoint(addr);
+       if (bp) {
+           bp->usecount++;
+           return 0;
+       }
+
+       bp = qemu_malloc(sizeof(struct kvm_sw_breakpoint));
+       if (!bp)
+           return -ENOMEM;
+
+       bp->pc = addr;
+       bp->usecount = 1;
+       bp->prev = NULL;
+       bp->next = first_sw_breakpoint;
+       err = kvm_arch_insert_sw_breakpoint(bp);
+       if (err) {
+           free(bp);
+           return err;
+       }
+
+       first_sw_breakpoint = bp;
+       if (bp->next)
+           bp->next->prev = bp;
+    } else {
+       err = kvm_arch_insert_hw_breakpoint(addr, len, type);
+       if (err)
+           return err;
+    }
+
+    for (env = first_cpu; env != NULL; env = env->next_cpu) {
+       err = kvm_update_guest_debug(env, 0);
+       if (err)
+           return err;
+    }
+    return 0;
+}
+
+int kvm_remove_breakpoint(target_ulong addr, target_ulong len, int type)
+{
+    struct kvm_sw_breakpoint *bp;
+    CPUState *env;
+    int err;
+
+    if (type == GDB_BREAKPOINT_SW) {
+       bp = kvm_find_sw_breakpoint(addr);
+       if (!bp)
+           return -ENOENT;
+
+       if (bp->usecount > 1) {
+           bp->usecount--;
+           return 0;
+       }
+
+       err = kvm_arch_remove_sw_breakpoint(bp);
+       if (err)
+           return err;
+
+       if (bp->prev)
+           bp->prev->next = bp->next;
+       else
+           first_sw_breakpoint = bp->next;
+       if (bp->next)
+           bp->next->prev = bp->prev;
+
+       qemu_free(bp);
+    } else {
+       err = kvm_arch_remove_hw_breakpoint(addr, len, type);
+       if (err)
+           return err;
+    }
+
+    for (env = first_cpu; env != NULL; env = env->next_cpu) {
+       err = kvm_update_guest_debug(env, 0);
+       if (err)
+           return err;
+    }
+    return 0;
+}
+
+void kvm_remove_all_breakpoints(void)
+{
+    struct kvm_sw_breakpoint *bp = first_sw_breakpoint;
+    CPUState *env;
+
+    while (bp) {
+       kvm_arch_remove_sw_breakpoint(bp);
+       bp = bp->next;
+    }
+    kvm_arch_remove_all_hw_breakpoints();
+
+    for (env = first_cpu; env != NULL; env = env->next_cpu)
+       kvm_update_guest_debug(env, 0);
+}
+
+#else /* !KVM_CAP_SET_GUEST_DEBUG */
+
+int kvm_update_guest_debug(CPUState *env, unsigned long reinject_trap)
+{
+    return -EINVAL;
+}
+
+int kvm_insert_breakpoint(target_ulong addr, target_ulong len, int type)
+{
+    return -EINVAL;
+}
+
+int kvm_remove_breakpoint(target_ulong addr, target_ulong len, int type)
+{
+    return -EINVAL;
+}
+
+void kvm_remove_all_breakpoints(void)
+{
+}
+#endif /* !KVM_CAP_SET_GUEST_DEBUG */
 
 /*
  * dirty pages logging
Index: b/libkvm/libkvm.h
===================================================================
--- a/libkvm/libkvm.h
+++ b/libkvm/libkvm.h
@@ -55,7 +55,10 @@ struct kvm_callbacks {
        /// generic memory writes to unmapped memory (For MMIO devices)
     int (*mmio_write)(void *opaque, uint64_t addr, uint8_t *data,
                                        int len);
-    int (*debug)(void *opaque, int vcpu);
+#ifdef KVM_CAP_SET_GUEST_DEBUG
+    int (*debug)(void *opaque, int vcpu,
+                struct kvm_debug_exit_arch *arch_info);
+#endif
        /*!
         * \brief Called when the VCPU issues an 'hlt' instruction.
         *
@@ -359,7 +362,9 @@ static inline int kvm_reset_mpstate(kvm_
  */
 int kvm_inject_irq(kvm_context_t kvm, int vcpu, unsigned irq);
 
-int kvm_guest_debug(kvm_context_t, int vcpu, struct kvm_debug_guest *dbg);
+#ifdef KVM_CAP_SET_GUEST_DEBUG
+int kvm_set_guest_debug(kvm_context_t, int vcpu, struct kvm_guest_debug *dbg);
+#endif
 
 #if defined(__i386__) || defined(__x86_64__)
 /*!
Index: b/qemu/qemu-kvm.h
===================================================================
--- a/qemu/qemu-kvm.h
+++ b/qemu/qemu-kvm.h
@@ -23,7 +23,10 @@ void kvm_save_registers(CPUState *env);
 void kvm_load_mpstate(CPUState *env);
 void kvm_save_mpstate(CPUState *env);
 int kvm_cpu_exec(CPUState *env);
-int kvm_update_debugger(CPUState *env);
+int kvm_insert_breakpoint(target_ulong addr, target_ulong len, int type);
+int kvm_remove_breakpoint(target_ulong addr, target_ulong len, int type);
+void kvm_remove_all_breakpoints(void);
+int kvm_update_guest_debug(CPUState *env, unsigned long reinject_trap);
 int kvm_qemu_init_env(CPUState *env);
 int kvm_qemu_check_extension(int ext);
 void kvm_apic_init(CPUState *env);
@@ -66,6 +69,30 @@ int kvm_arch_try_push_nmi(void *opaque);
 void kvm_arch_update_regs_for_sipi(CPUState *env);
 void kvm_arch_cpu_reset(CPUState *env);
 
+struct kvm_guest_debug;
+struct kvm_debug_exit_arch;
+
+struct kvm_sw_breakpoint {
+       target_ulong pc;
+       target_ulong saved_insn;
+       int usecount;
+       struct kvm_sw_breakpoint *prev;
+       struct kvm_sw_breakpoint *next;
+};
+
+extern struct kvm_sw_breakpoint *first_sw_breakpoint;
+
+int kvm_arch_debug(struct kvm_debug_exit_arch *arch_info);
+struct kvm_sw_breakpoint *kvm_find_sw_breakpoint(target_ulong pc);
+int kvm_arch_insert_sw_breakpoint(struct kvm_sw_breakpoint *bp);
+int kvm_arch_remove_sw_breakpoint(struct kvm_sw_breakpoint *bp);
+int kvm_arch_insert_hw_breakpoint(target_ulong addr,
+                                 target_ulong len, int type);
+int kvm_arch_remove_hw_breakpoint(target_ulong addr,
+                                 target_ulong len, int type);
+void kvm_arch_remove_all_hw_breakpoints(void);
+void kvm_arch_update_guest_debug(CPUState *env, struct kvm_guest_debug *dbg);
+
 CPUState *qemu_kvm_cpu_env(int index);
 
 void qemu_kvm_aio_wait_start(void);
Index: b/qemu/qemu-kvm-ia64.c
===================================================================
--- a/qemu/qemu-kvm-ia64.c
+++ b/qemu/qemu-kvm-ia64.c
@@ -66,6 +66,41 @@ void kvm_arch_update_regs_for_sipi(CPUSt
 {
 }
 
+int kvm_arch_insert_sw_breakpoint(struct kvm_sw_breakpoint *bp)
+{
+    return -EINVAL;
+}
+
+int kvm_arch_remove_sw_breakpoint(struct kvm_sw_breakpoint *bp)
+{
+    return -EINVAL;
+}
+
+int kvm_arch_insert_hw_breakpoint(target_ulong addr,
+                                 target_ulong len, int type)
+{
+    return -ENOSYS;
+}
+
+int kvm_arch_remove_hw_breakpoint(target_ulong addr,
+                                 target_ulong len, int type)
+{
+    return -ENOSYS;
+}
+
+void kvm_arch_remove_all_hw_breakpoints(void)
+{
+}
+
+int kvm_arch_debug(struct kvm_debug_exit_arch *arch_info)
+{
+    return 0;
+}
+
+void kvm_arch_update_guest_debug(CPUState *env, struct kvm_guest_debug *dbg)
+{
+}
+
 void kvm_save_mpstate(CPUState *env)
 {
 #ifdef KVM_CAP_MP_STATE
Index: b/qemu/qemu-kvm-powerpc.c
===================================================================
--- a/qemu/qemu-kvm-powerpc.c
+++ b/qemu/qemu-kvm-powerpc.c
@@ -216,3 +216,38 @@ int handle_powerpc_dcr_write(int vcpu, u
 void kvm_arch_cpu_reset(CPUState *env)
 {
 }
+
+int kvm_arch_insert_sw_breakpoint(struct kvm_sw_breakpoint *bp)
+{
+    return -EINVAL;
+}
+
+int kvm_arch_remove_sw_breakpoint(struct kvm_sw_breakpoint *bp)
+{
+    return -EINVAL;
+}
+
+int kvm_arch_insert_hw_breakpoint(target_ulong addr,
+                                 target_ulong len, int type)
+{
+    return -ENOSYS;
+}
+
+int kvm_arch_remove_hw_breakpoint(target_ulong addr,
+                                 target_ulong len, int type)
+{
+    return -ENOSYS;
+}
+
+void kvm_arch_remove_all_hw_breakpoints(void)
+{
+}
+
+int kvm_arch_debug(struct kvm_debug_exit_arch *arch_info)
+{
+    return 0;
+}
+
+void kvm_arch_update_guest_debug(CPUState *env, struct kvm_guest_debug *dbg)
+{
+}
Index: b/qemu/qemu-kvm-x86.c
===================================================================
--- a/qemu/qemu-kvm-x86.c
+++ b/qemu/qemu-kvm-x86.c
@@ -11,6 +11,8 @@
 
 #include <string.h>
 #include "hw/hw.h"
+#include "monitor.h"
+#include "gdbstub.h"
 
 #include "qemu-kvm.h"
 #include <libkvm.h>
@@ -717,3 +719,173 @@ void kvm_arch_cpu_reset(CPUState *env)
        }
     }
 }
+
+int kvm_arch_insert_sw_breakpoint(struct kvm_sw_breakpoint *bp)
+{
+    uint8_t int3 = 0xcc;
+
+    if (cpu_memory_rw_debug(mon_get_cpu(), bp->pc, (uint8_t *)&bp->saved_insn,
+                           1, 0) ||
+       cpu_memory_rw_debug(mon_get_cpu(), bp->pc, &int3, 1, 1))
+       return -EINVAL;
+    return 0;
+}
+
+int kvm_arch_remove_sw_breakpoint(struct kvm_sw_breakpoint *bp)
+{
+    if (cpu_memory_rw_debug(mon_get_cpu(), bp->pc, (uint8_t *)&bp->saved_insn,
+                           1, 1))
+       return -EINVAL;
+    return 0;
+}
+
+#ifdef KVM_CAP_SET_GUEST_DEBUG
+static struct {
+    target_ulong addr;
+    int len;
+    int type;
+} hw_breakpoint[4];
+
+static int nb_hw_breakpoint;
+
+static int find_hw_breakpoint(target_ulong addr, int len, int type)
+{
+    int n;
+
+    for (n = 0; n < nb_hw_breakpoint; n++)
+       if (hw_breakpoint[n].addr == addr && hw_breakpoint[n].type == type &&
+           (hw_breakpoint[n].len == len || len == -1))
+           return n;
+    return -1;
+}
+
+int kvm_arch_insert_hw_breakpoint(target_ulong addr,
+                                  target_ulong len, int type)
+{
+    switch (type) {
+    case GDB_BREAKPOINT_HW:
+       len = 1;
+       break;
+    case GDB_WATCHPOINT_WRITE:
+    case GDB_WATCHPOINT_ACCESS:
+       switch (len) {
+       case 1:
+           break;
+       case 2:
+       case 4:
+       case 8:
+           if (addr & (len - 1))
+               return -EINVAL;
+           break;
+       default:
+           return -EINVAL;
+       }
+       break;
+    default:
+       return -ENOSYS;
+    }
+
+    if (nb_hw_breakpoint == 4)
+        return -ENOBUFS;
+
+    if (find_hw_breakpoint(addr, len, type) >= 0)
+        return -EEXIST;
+
+    hw_breakpoint[nb_hw_breakpoint].addr = addr;
+    hw_breakpoint[nb_hw_breakpoint].len = len;
+    hw_breakpoint[nb_hw_breakpoint].type = type;
+    nb_hw_breakpoint++;
+
+    return 0;
+}
+
+int kvm_arch_remove_hw_breakpoint(target_ulong addr,
+                                  target_ulong len, int type)
+{
+    int n;
+
+    n = find_hw_breakpoint(addr, (type == GDB_BREAKPOINT_HW) ? 1 : len, type);
+    if (n < 0)
+        return -ENOENT;
+
+    nb_hw_breakpoint--;
+    hw_breakpoint[n] = hw_breakpoint[nb_hw_breakpoint];
+
+    return 0;
+}
+
+void kvm_arch_remove_all_hw_breakpoints(void)
+{
+    nb_hw_breakpoint = 0;
+}
+
+static CPUWatchpoint hw_watchpoint;
+
+int kvm_arch_debug(struct kvm_debug_exit_arch *arch_info)
+{
+    int handle = 0;
+    int n;
+
+    if (arch_info->exception == 1) {
+       if (arch_info->dr6 & (1 << 14)) {
+           if (cpu_single_env->singlestep_enabled)
+               handle = 1;
+       } else {
+           for (n = 0; n < 4; n++)
+               if (arch_info->dr6 & (1 << n))
+                   switch ((arch_info->dr7 >> (16 + n*4)) & 0x3) {
+                   case 0x0:
+                       handle = 1;
+                       break;
+                   case 0x1:
+                       handle = 1;
+                       cpu_single_env->watchpoint_hit = &hw_watchpoint;
+                       hw_watchpoint.vaddr = hw_breakpoint[n].addr;
+                       hw_watchpoint.flags = BP_MEM_WRITE;
+                       break;
+                   case 0x3:
+                       handle = 1;
+                       cpu_single_env->watchpoint_hit = &hw_watchpoint;
+                       hw_watchpoint.vaddr = hw_breakpoint[n].addr;
+                       hw_watchpoint.flags = BP_MEM_ACCESS;
+                       break;
+                   }
+       }
+    } else if (kvm_find_sw_breakpoint(arch_info->pc))
+       handle = 1;
+
+    if (!handle)
+       kvm_update_guest_debug(cpu_single_env,
+                       (arch_info->exception == 1) ?
+                       KVM_GUESTDBG_INJECT_DB : KVM_GUESTDBG_INJECT_BP);
+
+    return handle;
+}
+
+void kvm_arch_update_guest_debug(CPUState *env, struct kvm_guest_debug *dbg)
+{
+    const uint8_t type_code[] = {
+       [GDB_BREAKPOINT_HW] = 0x0,
+       [GDB_WATCHPOINT_WRITE] = 0x1,
+       [GDB_WATCHPOINT_ACCESS] = 0x3
+    };
+    const uint8_t len_code[] = {
+       [1] = 0x0, [2] = 0x1, [4] = 0x3, [8] = 0x2
+    };
+    int n;
+
+    if (first_sw_breakpoint)
+       dbg->control |= KVM_GUESTDBG_ENABLE | KVM_GUESTDBG_USE_SW_BP;
+
+    if (nb_hw_breakpoint > 0) {
+       dbg->control |= KVM_GUESTDBG_ENABLE | KVM_GUESTDBG_USE_HW_BP;
+       dbg->arch.debugreg[7] = 0x0600;
+       for (n = 0; n < nb_hw_breakpoint; n++) {
+           dbg->arch.debugreg[n] = hw_breakpoint[n].addr;
+           dbg->arch.debugreg[7] |= (2 << (n * 2)) |
+               (type_code[hw_breakpoint[n].type] << (16 + n*4)) |
+               (len_code[hw_breakpoint[n].len] << (18 + n*4));
+       }
+    }
+}
+#endif
Index: b/user/main.c
===================================================================
--- a/user/main.c
+++ b/user/main.c
@@ -284,11 +284,14 @@ static int test_outl(void *opaque, uint1
        return 0;
 }
 
-static int test_debug(void *opaque, int vcpu)
+#ifdef KVM_CAP_SET_GUEST_DEBUG
+static int test_debug(void *opaque, int vcpu,
+                     struct kvm_debug_exit_arch *arch_info)
 {
        printf("test_debug\n");
        return 0;
 }
+#endif
 
 static int test_halt(void *opaque, int vcpu)
 {
@@ -343,7 +346,9 @@ static struct kvm_callbacks test_callbac
        .outl        = test_outl,
        .mmio_read   = test_mem_read,
        .mmio_write  = test_mem_write,
+#ifdef KVM_CAP_SET_GUEST_DEBUG
        .debug       = test_debug,
+#endif
        .halt        = test_halt,
        .io_window = test_io_window,
        .try_push_interrupts = test_try_push_interrupts,
Index: b/qemu/gdbstub.c
===================================================================
--- a/qemu/gdbstub.c
+++ b/qemu/gdbstub.c
@@ -963,13 +963,6 @@ static void cpu_gdb_write_registers(CPUS
 
 #endif
 
-/* GDB breakpoint/watchpoint types */
-#define GDB_BREAKPOINT_SW        0
-#define GDB_BREAKPOINT_HW        1
-#define GDB_WATCHPOINT_WRITE     2
-#define GDB_WATCHPOINT_READ      3
-#define GDB_WATCHPOINT_ACCESS    4
-
 #ifndef CONFIG_USER_ONLY
 static const int xlat_gdb_type[] = {
     [GDB_WATCHPOINT_WRITE]  = BP_GDB | BP_MEM_WRITE,
@@ -983,6 +976,9 @@ static int gdb_breakpoint_insert(target_
     CPUState *env;
     int err = 0;
 
+    if (kvm_enabled())
+        return kvm_insert_breakpoint(addr, len, type);
+
     switch (type) {
     case GDB_BREAKPOINT_SW ... GDB_BREAKPOINT_HW:
         for (env = first_cpu; env != NULL; env = env->next_cpu) {
@@ -1011,6 +1007,9 @@ static int gdb_breakpoint_remove(target_
     CPUState *env;
     int err = 0;
 
+    if (kvm_enabled())
+        return kvm_remove_breakpoint(addr, len, type);
+
     switch (type) {
     case GDB_BREAKPOINT_SW ... GDB_BREAKPOINT_HW:
         for (env = first_cpu; env != NULL; env = env->next_cpu) {
@@ -1037,6 +1036,11 @@ static void gdb_breakpoint_remove_all(vo
 {
     CPUState *env;
 
+    if (kvm_enabled()) {
+        kvm_remove_all_breakpoints();
+        return;
+    }
+
     for (env = first_cpu; env != NULL; env = env->next_cpu) {
         cpu_breakpoint_remove_all(env, BP_GDB);
 #ifndef CONFIG_USER_ONLY
Index: b/qemu/gdbstub.h
===================================================================
--- a/qemu/gdbstub.h
+++ b/qemu/gdbstub.h
@@ -3,6 +3,13 @@
 
 #define DEFAULT_GDBSTUB_PORT "1234"
 
+/* GDB breakpoint/watchpoint types */
+#define GDB_BREAKPOINT_SW        0
+#define GDB_BREAKPOINT_HW        1
+#define GDB_WATCHPOINT_WRITE     2
+#define GDB_WATCHPOINT_READ      3
+#define GDB_WATCHPOINT_ACCESS    4
+
 typedef void (*gdb_syscall_complete_cb)(CPUState *env,
                                         target_ulong ret, target_ulong err);
 

--
To unsubscribe from this list: send the line "unsubscribe kvm" in
the body of a message to [EMAIL PROTECTED]
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to