From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S936229AbcCQQDu (ORCPT ); Thu, 17 Mar 2016 12:03:50 -0400 Received: from smtp02.citrix.com ([66.165.176.63]:57827 "EHLO SMTP02.CITRIX.COM" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S933961AbcCQQDq (ORCPT ); Thu, 17 Mar 2016 12:03:46 -0400 X-IronPort-AV: E=Sophos;i="5.24,350,1454976000"; d="scan'208";a="346643542" Subject: Re: [Xen-devel] [PATCH] xen/events: Mask a moving irq To: Boris Ostrovsky , , References: <1458218750-5202-1-git-send-email-boris.ostrovsky@oracle.com> CC: , , From: David Vrabel X-Enigmail-Draft-Status: N1110 Message-ID: <56EAD55A.6020801@citrix.com> Date: Thu, 17 Mar 2016 16:03:38 +0000 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:38.0) Gecko/20100101 Icedove/38.5.0 MIME-Version: 1.0 In-Reply-To: <1458218750-5202-1-git-send-email-boris.ostrovsky@oracle.com> Content-Type: text/plain; charset="utf-8" Content-Transfer-Encoding: 8bit X-DLP: MIA1 Sender: linux-kernel-owner@vger.kernel.org List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 17/03/16 12:45, Boris Ostrovsky wrote: > Moving an unmasked irq may result in irq handler being invoked on both > source and target CPUs. > > With 2-level this can happen as follows: > > On source CPU: > evtchn_2l_handle_events() -> > generic_handle_irq() -> > handle_edge_irq() -> > eoi_pirq(): > irq_move_irq(data); > > /***** WE ARE HERE *****/ > > if (VALID_EVTCHN(evtchn)) > clear_evtchn(evtchn); > > If at this moment target processor is handling an unrelated event in > evtchn_2l_handle_events()'s loop it may pick up our event since target's > cpu_evtchn_mask claims that this event belongs to it *and* the event is > unmasked and still pending. At the same time, source CPU will continue > executing its own handle_edge_irq(). > > With FIFO interrupt the scenario is similar: irq_move_irq() may result > in a EVTCHNOP_unmask hypercall which, in turn, may make the event > pending on the target CPU. > > We can avoid this situation by moving and clearing the event while > keeping event masked. Can you do: if (unlikely(irqd_is_setaffinity_pending(data))) { masked = test_and_set_mask() clear_evtchn() irq_move_masked_irq() unmask(masked); } else clear_evtchn() David