From patchwork Fri Jun 17 16:33:45 2016 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 8bit X-Patchwork-Submitter: =?utf-8?q?Alex_Benn=C3=A9e?= X-Patchwork-Id: 9184753 Return-Path: Received: from mail.wl.linuxfoundation.org (pdx-wl-mail.web.codeaurora.org [172.30.200.125]) by pdx-korg-patchwork.web.codeaurora.org (Postfix) with ESMTP id 10735601C0 for ; Fri, 17 Jun 2016 17:58:34 +0000 (UTC) Received: from mail.wl.linuxfoundation.org (localhost [127.0.0.1]) by mail.wl.linuxfoundation.org (Postfix) with ESMTP id E73DF28380 for ; Fri, 17 Jun 2016 17:58:33 +0000 (UTC) Received: by mail.wl.linuxfoundation.org (Postfix, from userid 486) id DA613283A9; Fri, 17 Jun 2016 17:58:33 +0000 (UTC) X-Spam-Checker-Version: SpamAssassin 3.3.1 (2010-03-16) on pdx-wl-mail.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-6.8 required=2.0 tests=BAYES_00,DKIM_SIGNED, RCVD_IN_DNSWL_HI,T_DKIM_INVALID autolearn=ham version=3.3.1 Received: from lists.gnu.org (lists.gnu.org [208.118.235.17]) (using TLSv1 with cipher AES256-SHA (256/256 bits)) (No client certificate requested) by mail.wl.linuxfoundation.org (Postfix) with ESMTPS id D4DDE28380 for ; Fri, 17 Jun 2016 17:58:32 +0000 (UTC) Received: from localhost ([::1]:59498 helo=lists.gnu.org) by lists.gnu.org with esmtp (Exim 4.71) (envelope-from ) id 1bDy2J-0004ZO-UP for patchwork-qemu-devel@patchwork.kernel.org; Fri, 17 Jun 2016 13:58:32 -0400 Received: from eggs.gnu.org ([2001:4830:134:3::10]:36306) by lists.gnu.org with esmtp (Exim 4.71) (envelope-from ) id 1bDwia-0006FK-EU for qemu-devel@nongnu.org; Fri, 17 Jun 2016 12:34:06 -0400 Received: from Debian-exim by eggs.gnu.org with spam-scanned (Exim 4.71) (envelope-from ) id 1bDwiY-0002uM-LT for qemu-devel@nongnu.org; Fri, 17 Jun 2016 12:34:04 -0400 Received: from mail-wm0-x229.google.com ([2a00:1450:400c:c09::229]:38573) by eggs.gnu.org with esmtp (Exim 4.71) (envelope-from ) id 1bDwiY-0002u4-BM for qemu-devel@nongnu.org; Fri, 17 Jun 2016 12:34:02 -0400 Received: by mail-wm0-x229.google.com with SMTP id m124so6769852wme.1 for ; Fri, 17 Jun 2016 09:34:02 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=linaro.org; s=google; h=from:to:cc:subject:date:message-id:in-reply-to:references :mime-version:content-transfer-encoding; bh=X12bAyZsYi08oUp1Wwg6I1AP0omFU3Dje9qD7CKseLI=; b=SED2r+S9zH7c5XoEbc5/5V1Gr1SR3v9GdYt1aEJE4vdUvf6IdsdJgL2Cnb87lnTL7O lSOspT8GCqn/buPBO7+q4FKsjzN7LxtpBNQLCrFbnuCi5nvcs65i2rrB9RnNZedR1i9C 6wgO7/+6033cFnbFIEzT1sv2CHbF+dv26/5tY= X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20130820; h=x-gm-message-state:from:to:cc:subject:date:message-id:in-reply-to :references:mime-version:content-transfer-encoding; bh=X12bAyZsYi08oUp1Wwg6I1AP0omFU3Dje9qD7CKseLI=; b=MUub/GFaEyrZOUruabtyrdk1AYiiaRUYtdZKfw6bEzF+41qoEawuLQvZmDtnVXcsSF oSwA3QRI04X5hSKYJwlCQ2Mco0+noSgQvvSw0x24V9yfypQEubBx1gofQdXe+cgLxWNU cWNPyE0E7FZltUcnnQ8RVwOuDhCAv9Iq2VMRRFiIQVxUb4YXWdCNpoPBUsKpItLhHL7Z ZOnUDQ6GbBLXmlmnt4ym6U1bKWS1YZyi5hCScHCbZrU5EEvqv5xcvGnow/0rdacDzz8E jsMdykGLIJ5BB8G1wSEewwINwXUPrff/MerK6BvIc/Lj4uQOr+QEx++98hFJubJEDGXL bRUQ== X-Gm-Message-State: ALyK8tLtTK36fbhiOVWpyo4XmlDjo3rlBNU3ynsQnHH8n7ulHxTb5mIwqjn2mVS+3NXZUEza X-Received: by 10.194.114.72 with SMTP id je8mr3022349wjb.88.1466181241572; Fri, 17 Jun 2016 09:34:01 -0700 (PDT) Received: from zen.linaro.local ([81.128.185.34]) by smtp.gmail.com with ESMTPSA id ej9sm15801277wjd.7.2016.06.17.09.33.55 (version=TLS1_2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); Fri, 17 Jun 2016 09:33:56 -0700 (PDT) Received: from zen.linaroharston (localhost [127.0.0.1]) by zen.linaro.local (Postfix) with ESMTP id 11A633E0A7D; Fri, 17 Jun 2016 17:33:59 +0100 (BST) From: =?UTF-8?q?Alex=20Benn=C3=A9e?= To: mttcg@listserver.greensocs.com, qemu-devel@nongnu.org, fred.konrad@greensocs.com, a.rigo@virtualopensystems.com, serge.fdrv@gmail.com, cota@braap.org, bobby.prani@gmail.com Date: Fri, 17 Jun 2016 17:33:45 +0100 Message-Id: <1466181227-14934-6-git-send-email-alex.bennee@linaro.org> X-Mailer: git-send-email 2.7.4 In-Reply-To: <1466181227-14934-1-git-send-email-alex.bennee@linaro.org> References: <1466181227-14934-1-git-send-email-alex.bennee@linaro.org> MIME-Version: 1.0 X-detected-operating-system: by eggs.gnu.org: GNU/Linux 2.2.x-3.x [generic] X-Received-From: 2a00:1450:400c:c09::229 Subject: [Qemu-devel] [RFC 5/7] breakpoints: put breakpoints under RCU control X-BeenThere: qemu-devel@nongnu.org X-Mailman-Version: 2.1.21 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Cc: peter.maydell@linaro.org, Peter Crosthwaite , claudio.fontana@huawei.com, Riku Voipio , mark.burton@greensocs.com, jan.kiszka@siemens.com, pbonzini@redhat.com, =?UTF-8?q?Alex=20Benn=C3=A9e?= , rth@twiddle.net Errors-To: qemu-devel-bounces+patchwork-qemu-devel=patchwork.kernel.org@nongnu.org Sender: "Qemu-devel" X-Virus-Scanned: ClamAV using ClamSMTP Each time breakpoints are added/removed from the array it's done using an read-copy-update cycle. Simultaneous writes are protected by the debug_update_lock. Signed-off-by: Alex Bennée --- cpus.c | 3 + exec.c | 167 ++++++++++++++++++++++++++++++++++++++++++++---------- include/qom/cpu.h | 42 ++++++++++---- linux-user/main.c | 11 +--- 4 files changed, 175 insertions(+), 48 deletions(-) diff --git a/cpus.c b/cpus.c index 84c3520..b76300b 100644 --- a/cpus.c +++ b/cpus.c @@ -1441,6 +1441,9 @@ void qemu_init_vcpu(CPUState *cpu) cpu_address_space_init(cpu, as, 0); } + /* protects updates to debug info */ + qemu_mutex_init(&cpu->update_debug_lock); + if (kvm_enabled()) { qemu_kvm_start_vcpu(cpu); } else if (tcg_enabled()) { diff --git a/exec.c b/exec.c index c8e8823..d1d14c1 100644 --- a/exec.c +++ b/exec.c @@ -919,31 +919,94 @@ static inline bool cpu_watchpoint_address_matches(CPUWatchpoint *wp, } #endif -/* Find watchpoint with external reference */ -CPUBreakpoint *cpu_breakpoint_get_by_ref(CPUState *cpu, int ref) + +/* Create a working copy of the breakpoints. + * + * The rcu_read_lock() may already be held depending on where this + * update has been triggered from. However it is safe to nest + * rcu_read_locks() so we do the copy under the lock here. + */ + +static GArray *rcu_copy_breakpoints(CPUState *cpu) { - CPUBreakpoint *bp = NULL; + GArray *old, *new; + + rcu_read_lock(); + old = atomic_rcu_read(&cpu->breakpoints); + if (old) { + new = g_array_sized_new(false, false, sizeof(CPUBreakpoint), old->len); + memcpy(new->data, old->data, old->len * sizeof(CPUBreakpoint)); + new->len = old->len; + } else { + new = g_array_new(false, false, sizeof(CPUBreakpoint)); + } + rcu_read_unlock(); + + return new; +} + +struct BreakRCU { + struct rcu_head rcu; + GArray *bkpts; +}; + +/* RCU reclaim step */ +static void rcu_free_breakpoints(struct BreakRCU *rcu_free) +{ + g_array_free(rcu_free->bkpts, false); + g_free(rcu_free); +} + +/* Called with update lock held */ +static void rcu_update_breakpoints(CPUState *cpu, GArray *new_bkpts) +{ + GArray *bpts = atomic_rcu_read(&cpu->breakpoints); + atomic_rcu_set(&cpu->breakpoints, new_bkpts); + + if (bpts) { + struct BreakRCU *rcu_free = g_malloc(sizeof(*rcu_free)); + rcu_free->bkpts = bpts; + call_rcu(rcu_free, rcu_free_breakpoints, rcu); + } +} + +/* Find watchpoint with external reference, only valid for duration of + * rcu_read_lock */ +static CPUBreakpoint *find_bkpt_with_ref(GArray *bkpts, int ref) +{ + CPUBreakpoint *bp; int i = 0; + do { - bp = &g_array_index(cpu->breakpoints, CPUBreakpoint, i++); - } while (i < cpu->breakpoints->len && bp && bp->ref != ref); + bp = &g_array_index(bkpts, CPUBreakpoint, i); + if (bp->ref == ref) { + return bp; + } + } while (i++ < bkpts->len); - return bp; + return NULL; +} + +CPUBreakpoint *cpu_breakpoint_get_by_ref(CPUState *cpu, int ref) +{ + GArray *bkpts = atomic_rcu_read(&cpu->breakpoints); + return find_bkpt_with_ref(bkpts, ref); } /* Add a breakpoint. */ int cpu_breakpoint_insert_with_ref(CPUState *cpu, vaddr pc, int flags, int ref) { + GArray *bkpts; CPUBreakpoint *bp = NULL; - /* Allocate if no previous breakpoints */ - if (!cpu->breakpoints) { - cpu->breakpoints = g_array_new(false, true, sizeof(CPUBreakpoint)); - } + qemu_mutex_lock(&cpu->update_debug_lock); + + /* This will allocate if no previous breakpoints */ + bkpts = rcu_copy_breakpoints(cpu); /* Find old watchpoint */ if (ref != BPWP_NOREF) { - bp = cpu_breakpoint_get_by_ref(cpu, ref); + bp = find_bkpt_with_ref(bkpts, ref); } if (bp) { @@ -958,14 +1021,17 @@ int cpu_breakpoint_insert_with_ref(CPUState *cpu, vaddr pc, int flags, int ref) /* keep all GDB-injected breakpoints in front */ if (flags & BP_GDB) { - g_array_prepend_val(cpu->breakpoints, brk); + g_array_prepend_val(bkpts, brk); } else { - g_array_append_val(cpu->breakpoints, brk); + g_array_append_val(bkpts, brk); } } breakpoint_invalidate(cpu, pc); + rcu_update_breakpoints(cpu, bkpts); + qemu_mutex_unlock(&cpu->update_debug_lock); + return 0; } @@ -974,67 +1040,110 @@ int cpu_breakpoint_insert(CPUState *cpu, vaddr pc, int flags) return cpu_breakpoint_insert_with_ref(cpu, pc, flags, BPWP_NOREF); } -static void cpu_breakpoint_delete(CPUState *cpu, int index) +/* Called with update_debug_lock held */ +static void cpu_breakpoint_delete(CPUState *cpu, GArray *bkpts, int index) { CPUBreakpoint *bp; - bp = &g_array_index(cpu->breakpoints, CPUBreakpoint, index); + bp = &g_array_index(bkpts, CPUBreakpoint, index); breakpoint_invalidate(cpu, bp->pc); - g_array_remove_index(cpu->breakpoints, index); + g_array_remove_index(bkpts, index); } /* Remove a specific breakpoint. */ int cpu_breakpoint_remove(CPUState *cpu, vaddr pc, int flags) { + GArray *bkpts = atomic_rcu_read(&cpu->breakpoints); CPUBreakpoint *bp; + int retval = -ENOENT; - if (unlikely(cpu->breakpoints) && unlikely(cpu->breakpoints->len)) { + if (unlikely(bkpts) && unlikely(bkpts->len)) { int i = 0; + + qemu_mutex_lock(&cpu->update_debug_lock); + bkpts = rcu_copy_breakpoints(cpu); + do { - bp = &g_array_index(cpu->breakpoints, CPUBreakpoint, i); + bp = &g_array_index(bkpts, CPUBreakpoint, i); if (bp && bp->pc == pc && bp->flags == flags) { - cpu_breakpoint_delete(cpu, i); + cpu_breakpoint_delete(cpu, bkpts, i); + retval = 0; } else { i++; } - } while (i < cpu->breakpoints->len); + } while (i < bkpts->len); + + rcu_update_breakpoints(cpu, bkpts); + qemu_mutex_unlock(&cpu->update_debug_lock); + } - return -ENOENT; + return retval; } +#ifdef CONFIG_USER_ONLY +void cpu_breakpoints_clone(CPUState *old_cpu, CPUState *new_cpu) +{ + GArray *bkpts = atomic_rcu_read(&old_cpu->breakpoints); + + if (unlikely(bkpts) && unlikely(bkpts->len)) { + qemu_mutex_lock(&new_cpu->update_debug_lock); + bkpts = rcu_copy_breakpoints(old_cpu); + rcu_update_breakpoints(new_cpu, bkpts); + qemu_mutex_unlock(&new_cpu->update_debug_lock); + } +} +#endif + + /* Remove a specific breakpoint by reference. */ void cpu_breakpoint_remove_by_ref(CPUState *cpu, int ref) { + GArray *bkpts = atomic_rcu_read(&cpu->breakpoints); CPUBreakpoint *bp; - if (unlikely(cpu->breakpoints) && unlikely(cpu->breakpoints->len)) { + if (unlikely(bkpts) && unlikely(bkpts->len)) { int i = 0; + + qemu_mutex_lock(&cpu->update_debug_lock); + bkpts = rcu_copy_breakpoints(cpu); + do { - bp = &g_array_index(cpu->breakpoints, CPUBreakpoint, i); + bp = &g_array_index(bkpts, CPUBreakpoint, i); if (bp && bp->ref == ref) { - cpu_breakpoint_delete(cpu, i); + cpu_breakpoint_delete(cpu, bkpts, i); } else { i++; } - } while (i < cpu->breakpoints->len); + } while (i < bkpts->len); + + rcu_update_breakpoints(cpu, bkpts); + qemu_mutex_unlock(&cpu->update_debug_lock); } } /* Remove all matching breakpoints. */ void cpu_breakpoint_remove_all(CPUState *cpu, int mask) { + GArray *bkpts = atomic_rcu_read(&cpu->breakpoints); CPUBreakpoint *bp; - if (unlikely(cpu->breakpoints) && unlikely(cpu->breakpoints->len)) { + if (unlikely(bkpts) && unlikely(bkpts->len)) { int i = 0; + + qemu_mutex_lock(&cpu->update_debug_lock); + bkpts = rcu_copy_breakpoints(cpu); + do { - bp = &g_array_index(cpu->breakpoints, CPUBreakpoint, i); + bp = &g_array_index(bkpts, CPUBreakpoint, i); if (bp->flags & mask) { - cpu_breakpoint_delete(cpu, i); + cpu_breakpoint_delete(cpu, bkpts, i); } else { i++; } - } while (i < cpu->breakpoints->len); + } while (i < bkpts->len); + + rcu_update_breakpoints(cpu, bkpts); + qemu_mutex_unlock(&cpu->update_debug_lock); } } diff --git a/include/qom/cpu.h b/include/qom/cpu.h index f695afb..51ca28c 100644 --- a/include/qom/cpu.h +++ b/include/qom/cpu.h @@ -326,8 +326,11 @@ struct CPUState { /* Debugging support: * * Both the gdbstub and architectural debug support will add - * references to these arrays. + * references to these arrays. The list of breakpoints and + * watchpoints are updated with RCU. The update_debug_lock is only + * required for updating, not just reading. */ + QemuMutex update_debug_lock; GArray *breakpoints; GArray *watchpoints; CPUWatchpoint *watchpoint_hit; @@ -849,12 +852,13 @@ int cpu_breakpoint_insert_with_ref(CPUState *cpu, vaddr pc, int flags, int ref); * @cpu: CPU to monitor * @ref: unique reference * - * @return: a pointer to working copy of the breakpoint. + * @return: a pointer to working copy of the breakpoint or NULL. * - * Return a working copy of the current referenced breakpoint. This - * obviously only works for breakpoints inserted with a reference. The - * lifetime of this objected will be limited and should not be kept - * beyond its immediate use. Otherwise return NULL. + * Return short term working copy of the a breakpoint marked with an + * external reference. This obviously only works for breakpoints + * inserted with a reference. The lifetime of this object is the + * duration of the rcu_read_lock() and it may be released when + * rcu_synchronize is called. */ CPUBreakpoint *cpu_breakpoint_get_by_ref(CPUState *cpu, int ref); @@ -871,14 +875,32 @@ void cpu_breakpoint_remove_by_ref(CPUState *cpu, int ref); void cpu_breakpoint_remove_all(CPUState *cpu, int mask); -/* Return true if PC matches an installed breakpoint. */ +#ifdef CONFIG_USER_ONLY +/** + * cpu_breakpoints_clone: + * + * @old_cpu: source of breakpoints + * @new_cpu: destination for breakpoints + * + * When a new user-mode thread is created the CPU structure behind it + * needs a copy of the exiting breakpoint conditions. This helper does + * the copying. + */ +void cpu_breakpoints_clone(CPUState *old_cpu, CPUState *new_cpu); +#endif + +/* Return true if PC matches an installed breakpoint. + * Called while the RCU read lock is held. + */ static inline bool cpu_breakpoint_test(CPUState *cpu, vaddr pc, int mask) { - if (unlikely(cpu->breakpoints) && unlikely(cpu->breakpoints->len)) { + GArray *bkpts = atomic_rcu_read(&cpu->breakpoints); + + if (unlikely(bkpts) && unlikely(bkpts->len)) { CPUBreakpoint *bp; int i; - for (i = 0; i < cpu->breakpoints->len; i++) { - bp = &g_array_index(cpu->breakpoints, CPUBreakpoint, i); + for (i = 0; i < bkpts->len; i++) { + bp = &g_array_index(bkpts, CPUBreakpoint, i); if (bp->pc == pc && (bp->flags & mask)) { return true; } diff --git a/linux-user/main.c b/linux-user/main.c index 179f2f2..2901626 100644 --- a/linux-user/main.c +++ b/linux-user/main.c @@ -3808,17 +3808,10 @@ CPUArchState *cpu_copy(CPUArchState *env) memcpy(new_env, env, sizeof(CPUArchState)); - /* Clone all break/watchpoints. + /* Clone all breakpoints. Note: Once we support ptrace with hw-debug register access, make sure BP_CPU break/watchpoints are handled correctly on clone. */ - if (unlikely(cpu->breakpoints) && unlikely(cpu->breakpoints->len)) { - CPUBreakpoint *bp; - int i; - for (i = 0; i < cpu->breakpoints->len; i++) { - bp = &g_array_index(cpu->breakpoints, CPUBreakpoint, i); - cpu_breakpoint_insert(new_cpu, bp->pc, bp->flags); - } - } + cpu_breakpoints_clone(cpu, new_cpu); if (unlikely(cpu->watchpoints) && unlikely(cpu->watchpoints->len)) { CPUWatchpoint *wp; int i;