From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-5.3 required=3.0 tests=BAYES_00, HEADER_FROM_DIFFERENT_DOMAINS,MAILING_LIST_MULTI,NICE_REPLY_A,SPF_HELO_NONE, SPF_PASS,URIBL_BLOCKED,USER_AGENT_SANE_1 autolearn=no autolearn_force=no version=3.4.0 Received: from mail.kernel.org (mail.kernel.org [198.145.29.99]) by smtp.lore.kernel.org (Postfix) with ESMTP id 21637C63777 for ; Tue, 1 Dec 2020 00:14:33 +0000 (UTC) Received: from lists.gnu.org (lists.gnu.org [209.51.188.17]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by mail.kernel.org (Postfix) with ESMTPS id 41A4320706 for ; Tue, 1 Dec 2020 00:14:32 +0000 (UTC) DMARC-Filter: OpenDMARC Filter v1.3.2 mail.kernel.org 41A4320706 Authentication-Results: mail.kernel.org; dmarc=none (p=none dis=none) header.from=csgraf.de Authentication-Results: mail.kernel.org; spf=pass smtp.mailfrom=qemu-devel-bounces+qemu-devel=archiver.kernel.org@nongnu.org Received: from localhost ([::1]:57520 helo=lists1p.gnu.org) by lists.gnu.org with esmtp (Exim 4.90_1) (envelope-from ) id 1kjtJO-00039O-Oa for qemu-devel@archiver.kernel.org; Mon, 30 Nov 2020 19:14:30 -0500 Received: from eggs.gnu.org ([2001:470:142:3::10]:60268) by lists.gnu.org with esmtps (TLS1.2:ECDHE_RSA_AES_256_GCM_SHA384:256) (Exim 4.90_1) (envelope-from ) id 1kjtIL-0002Xz-ST; Mon, 30 Nov 2020 19:13:25 -0500 Received: from mail.csgraf.de ([188.138.100.120]:41640 helo=zulu616.server4you.de) by eggs.gnu.org with esmtp (Exim 4.90_1) (envelope-from ) id 1kjtII-0003Ct-VI; Mon, 30 Nov 2020 19:13:25 -0500 Received: from edge-cache-175.e-lhr50.amazon.com (ec2-3-122-114-9.eu-central-1.compute.amazonaws.com [3.122.114.9]) by csgraf.de (Postfix) with UTF8SMTPSA id 81CC63900111; Tue, 1 Dec 2020 01:13:19 +0100 (CET) Subject: Re: [PATCH 2/8] hvf: Move common code out To: Peter Collingbourne References: <20201126215017.41156-1-agraf@csgraf.de> <20201126215017.41156-3-agraf@csgraf.de> <20201127200054.GC56950@SPB-NB-133.local> <392c2465-157e-e15a-0a2c-2e3faa166d22@csgraf.de> <6975b4a3-1568-df40-8594-bfcf488ac425@csgraf.de> <4e1d93a4-9dcc-c6b6-e060-6eea39ae2f16@csgraf.de> From: Alexander Graf Message-ID: Date: Tue, 1 Dec 2020 01:13:19 +0100 User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10.16; rv:84.0) Gecko/20100101 Thunderbird/84.0 MIME-Version: 1.0 In-Reply-To: Content-Type: text/plain; charset=UTF-8; format=flowed Content-Transfer-Encoding: 7bit Content-Language: en-US Received-SPF: pass client-ip=188.138.100.120; envelope-from=agraf@csgraf.de; helo=zulu616.server4you.de X-Spam_score_int: -18 X-Spam_score: -1.9 X-Spam_bar: - X-Spam_report: (-1.9 / 5.0 requ) BAYES_00=-1.9, NICE_REPLY_A=-0.001, SPF_HELO_NONE=0.001, SPF_PASS=-0.001 autolearn=ham autolearn_force=no X-Spam_action: no action X-BeenThere: qemu-devel@nongnu.org X-Mailman-Version: 2.1.23 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Cc: Peter Maydell , Eduardo Habkost , Richard Henderson , qemu-devel , Cameron Esfahani , Roman Bolshakov , qemu-arm@nongnu.org, Claudio Fontana , Frank Yang , Paolo Bonzini Errors-To: qemu-devel-bounces+qemu-devel=archiver.kernel.org@nongnu.org Sender: "Qemu-devel" On 01.12.20 01:00, Peter Collingbourne wrote: > On Mon, Nov 30, 2020 at 3:18 PM Alexander Graf wrote: >> >> On 01.12.20 00:01, Peter Collingbourne wrote: >>> On Mon, Nov 30, 2020 at 1:40 PM Alexander Graf wrote: >>>> Hi Peter, >>>> >>>> On 30.11.20 22:08, Peter Collingbourne wrote: >>>>> On Mon, Nov 30, 2020 at 12:56 PM Frank Yang wrote: >>>>>> On Mon, Nov 30, 2020 at 12:34 PM Alexander Graf wrote: >>>>>>> Hi Frank, >>>>>>> >>>>>>> Thanks for the update :). Your previous email nudged me into the right direction. I previously had implemented WFI through the internal timer framework which performed way worse. >>>>>> Cool, glad it's helping. Also, Peter found out that the main thing keeping us from just using cntpct_el0 on the host directly and compare with cval is that if we sleep, cval is going to be much < cntpct_el0 by the sleep time. If we can get either the architecture or macos to read out the sleep time then we might be able to not have to use a poll interval either! >>>>>>> Along the way, I stumbled over a few issues though. For starters, the signal mask for SIG_IPI was not set correctly, so while pselect() would exit, the signal would never get delivered to the thread! For a fix, check out >>>>>>> >>>>>>> https://patchew.org/QEMU/20201130030723.78326-1-agraf@csgraf.de/20201130030723.78326-4-agraf@csgraf.de/ >>>>>>> >>>>>> Thanks, we'll take a look :) >>>>>> >>>>>>> Please also have a look at my latest stab at WFI emulation. It doesn't handle WFE (that's only relevant in overcommitted scenarios). But it does handle WFI and even does something similar to hlt polling, albeit not with an adaptive threshold. >>>>> Sorry I'm not subscribed to qemu-devel (I'll subscribe in a bit) so >>>>> I'll reply to your patch here. You have: >>>>> >>>>> + /* Set cpu->hvf->sleeping so that we get a >>>>> SIG_IPI signal. */ >>>>> + cpu->hvf->sleeping = true; >>>>> + smp_mb(); >>>>> + >>>>> + /* Bail out if we received an IRQ meanwhile */ >>>>> + if (cpu->thread_kicked || (cpu->interrupt_request & >>>>> + (CPU_INTERRUPT_HARD | CPU_INTERRUPT_FIQ))) { >>>>> + cpu->hvf->sleeping = false; >>>>> + break; >>>>> + } >>>>> + >>>>> + /* nanosleep returns on signal, so we wake up on kick. */ >>>>> + nanosleep(ts, NULL); >>>>> >>>>> and then send the signal conditional on whether sleeping is true, but >>>>> I think this is racy. If the signal is sent after sleeping is set to >>>>> true but before entering nanosleep then I think it will be ignored and >>>>> we will miss the wakeup. That's why in my implementation I block IPI >>>>> on the CPU thread at startup and then use pselect to atomically >>>>> unblock and begin sleeping. The signal is sent unconditionally so >>>>> there's no need to worry about races between actually sleeping and the >>>>> "we think we're sleeping" state. It may lead to an extra wakeup but >>>>> that's better than missing it entirely. >>>> Thanks a bunch for the comment! So the trick I was using here is to >>>> modify the timespec from the kick function before sending the IPI >>>> signal. That way, we know that either we are inside the sleep (where the >>>> signal wakes it up) or we are outside the sleep (where timespec={} will >>>> make it return immediately). >>>> >>>> The only race I can think of is if nanosleep does calculations based on >>>> the timespec and we happen to send the signal right there and then. >>> Yes that's the race I was thinking of. Admittedly it's a small window >>> but it's theoretically possible and part of the reason why pselect was >>> created. >>> >>>> The problem with blocking IPIs is basically what Frank was describing >>>> earlier: How do you unset the IPI signal pending status? If the signal >>>> is never delivered, how can pselect differentiate "signal from last time >>>> is still pending" from "new signal because I got an IPI"? >>> In this case we would take the additional wakeup which should be >>> harmless since we will take the WFx exit again and put us in the >>> correct state. But that's a lot better than busy looping. >> >> I'm not sure I follow. I'm thinking of the following scenario: >> >> - trap into WFI handler >> - go to sleep with blocked SIG_IPI >> - SIG_IPI arrives, pselect() exits >> - signal is still pending because it's blocked >> - enter guest >> - trap into WFI handler >> - run pselect(), but it immediate exits because SIG_IPI is still pending >> >> This was the loop I was seeing when running with SIG_IPI blocked. That's >> part of the reason why I switched to a different model. > What I observe is that when returning from a pending signal pselect > consumes the signal (which is also consistent with my understanding of > what pselect does). That means that it doesn't matter if we take a > second WFx exit because once we reach the pselect in the second WFx > exit the signal will have been consumed by the pselect in the first > exit and we will just wait for the next one. > > I don't know why things may have been going wrong in your > implementation but it may be related to the issue with > mach_absolute_time() which I posted about separately and was also > causing busy loops for us in some cases. Once that issue was fixed in > our implementation we started seeing sleep until VTIMER due work > properly. > >> >>> I reckon that you could improve things a little by unblocking the >>> signal and then reblocking it before unlocking iothread (e.g. with a >>> pselect with zero time interval), which would flush any pending >>> signals. Since any such signal would correspond to a signal from last >>> time (because we still have the iothread lock) we know that any future >>> signals should correspond to new IPIs. >> >> Yeah, I think you actually *have* to do exactly that, because otherwise >> pselect() will always return after 0ns because the signal is still pending. >> >> And yes, I agree that that starts to sound a bit less racy now. But it >> means we can probably also just do >> >> - WFI handler >> - block SIG_IPI >> - set hvf->sleeping = true >> - check for pending interrupts >> - pselect() >> - unblock SIG_IPI >> >> which means we run with SIG_IPI unmasked by default. I don't think the >> number of signal mask changes is any different with that compared to >> running with SIG_IPI always masked, right? > And unlock/lock iothread around the pselect? I suppose that could work > but as I mentioned it would just be an optimization. > > Maybe I can try to make my approach work on top of your series, or if > you already have a patch I can try to debug it. Let me know. I would love to take a patch from you here :). I'll still be stuck for a while with the sysreg sync rework that Peter asked for before I can look at WFI again. Alex