Received: by 2002:a05:7412:b10a:b0:f3:1519:9f41 with SMTP id az10csp2634819rdb; Mon, 4 Dec 2023 03:21:01 -0800 (PST) X-Google-Smtp-Source: AGHT+IHdmo3M/fouZlPJcaGlqA3hzSFkS1/a8cKYQRMApeO283b/NfkUyWeNe7j9eDXHlg23HK6Z X-Received: by 2002:aa7:9edc:0:b0:6ce:4552:659a with SMTP id r28-20020aa79edc000000b006ce4552659amr663825pfq.66.1701688860736; Mon, 04 Dec 2023 03:21:00 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1701688860; cv=none; d=google.com; s=arc-20160816; b=bVBKDYIkDVro9SYAsW18cJhdWtwev1HG61RYsZqss/ucpVXORuwDMIwd5qFhvYL9bM ZHARZRRn1Xzfqw43SuQBEKE1x1+UPo0apTW8yvAi8zkv4plGcJxRCMdsUxEg8JOaq9lN 0C5Lyv2rK/3epD4fu1NqKaabf7yG6MG9JJgYaMOJwjkRsZoeY8AdQTJWkcF97BGurvjp W3sNHZ1f03cwrAuXrFno/uZIvZpaeS0FaGQFRuKFy1quOiodLixfKZ/2g/H0Lm8lYS0X 9yaDLkxem6l57O3gH1Z6oQvsxLJ8JTWVi8sNfXRgElllKwt55CoBAcwsJsACa1ugcFFT prSg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:in-reply-to:from :references:cc:to:content-language:subject:user-agent:mime-version :date:message-id:dkim-signature; bh=9b889mv5myf1mCmXwJt/3JkndxdGSM3c+7rCItOP4tY=; fh=hU76Ixc44NDFo0hzv1clQGJ0lDxwjIfhHVHdoat7LUY=; b=RBhHRkpsN4F/0Xw35vdIzpv/t5U5gcrhMcaFPxN4Man+y1Vb+YUTV3MUocmgTOvaU7 rEA2hpCpMu0UQ10V70f+oC2Q1F0OzxmRxVH5rBzEMwfAdsCFprIY2jcm0NZgj9zSPwJ4 4x+wjyp4gP3eOoviahMlXziti+/GTdUvQ16kod85a6e0Am7fUIGopmc7y6491W0DRGLr r3vgzx6lDiIFV/JVFR8fuupAIceak7qkQ2yoqcgSsvGB8cvSLavyBDhv35RzoDip85Rg VD5/yc+mpl+34W+7SLHN54j+UJjABec9oBAK8Ns6omJ9781fhjTCO8itg6zWgiC1TRlb ypFQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@collabora.com header.s=mail header.b=buSFgO4r; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.38 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=QUARANTINE sp=QUARANTINE dis=NONE) header.from=collabora.com Return-Path: Received: from fry.vger.email (fry.vger.email. [23.128.96.38]) by mx.google.com with ESMTPS id e187-20020a6369c4000000b005be109a793csi7389642pgc.444.2023.12.04.03.21.00 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Mon, 04 Dec 2023 03:21:00 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.38 as permitted sender) client-ip=23.128.96.38; Authentication-Results: mx.google.com; dkim=pass header.i=@collabora.com header.s=mail header.b=buSFgO4r; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.38 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=QUARANTINE sp=QUARANTINE dis=NONE) header.from=collabora.com Received: from out1.vger.email (depot.vger.email [IPv6:2620:137:e000::3:0]) by fry.vger.email (Postfix) with ESMTP id 441CE80942DD; Mon, 4 Dec 2023 03:20:57 -0800 (PST) X-Virus-Status: Clean X-Virus-Scanned: clamav-milter 0.103.11 at fry.vger.email Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S231767AbjLDLU1 (ORCPT + 99 others); Mon, 4 Dec 2023 06:20:27 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:56742 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S231571AbjLDLUZ (ORCPT ); Mon, 4 Dec 2023 06:20:25 -0500 Received: from madras.collabora.co.uk (madras.collabora.co.uk [46.235.227.172]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 67B58E6 for ; Mon, 4 Dec 2023 03:20:30 -0800 (PST) Received: from [100.113.186.2] (cola.collaboradmins.com [195.201.22.229]) (using TLSv1.3 with cipher TLS_AES_128_GCM_SHA256 (128/128 bits) key-exchange X25519 server-signature RSA-PSS (4096 bits) server-digest SHA256) (No client certificate requested) (Authenticated sender: kholk11) by madras.collabora.co.uk (Postfix) with ESMTPSA id D953D6602030; Mon, 4 Dec 2023 11:20:27 +0000 (GMT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=collabora.com; s=mail; t=1701688828; bh=cZeI5VKu1LE+0g50TIvM40Dlj8CQw2yZgDTALs1PHU0=; h=Date:Subject:To:Cc:References:From:In-Reply-To:From; b=buSFgO4rAzYsi+vvoqP2Clk3yGypsraT+LVLmh2mmv5LcqDgb6ihDfku0yaCby9Q9 u4KCCtQqFynTk9mMnYK+dXZuInrI1VBgNNjQnQXcUDfgYw+/871X/Whbh1QMptTDml n8T3PPC+WNuSiE5Tn2g+0poOhnJT7kGz+r7ERyr+B6q5AGeklD0aW408fWMxi9Wj0w t5una1nqiNjsuvZtItmVOwsaxI+8O2od7Fq0cAi30yZ8byofY/kwkRSvhifQkKL9mL CoWAXOXhJwJ/WVh60CJQvs1NtHTkIWGmaC3S0BcmCJI5aweoLfUzUJZju3n0mRG0d9 BFZN2B0hDE1Xg== Message-ID: <8b58d97d-7643-46bd-b702-b315addf32a1@collabora.com> Date: Mon, 4 Dec 2023 12:20:25 +0100 MIME-Version: 1.0 User-Agent: Mozilla Thunderbird Subject: Re: [PATCH v3 3/3] drm/panfrost: Synchronize and disable interrupts before powering off Content-Language: en-US To: Boris Brezillon Cc: robh@kernel.org, steven.price@arm.com, maarten.lankhorst@linux.intel.com, mripard@kernel.org, tzimmermann@suse.de, airlied@gmail.com, daniel@ffwll.ch, dri-devel@lists.freedesktop.org, linux-kernel@vger.kernel.org, kernel@collabora.com, m.szyprowski@samsung.com, krzysztof.kozlowski@linaro.org References: <20231201104027.35273-1-angelogioacchino.delregno@collabora.com> <20231201104027.35273-4-angelogioacchino.delregno@collabora.com> <20231201121437.7d5cdefb@collabora.com> From: AngeloGioacchino Del Regno In-Reply-To: <20231201121437.7d5cdefb@collabora.com> Content-Type: text/plain; charset=UTF-8; format=flowed Content-Transfer-Encoding: 7bit X-Spam-Status: No, score=-0.9 required=5.0 tests=DKIM_SIGNED,DKIM_VALID, DKIM_VALID_AU,HEADER_FROM_DIFFERENT_DOMAINS,MAILING_LIST_MULTI, SPF_HELO_NONE,SPF_PASS,T_SCC_BODY_TEXT_LINE autolearn=unavailable autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on fry.vger.email Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org X-Greylist: Sender passed SPF test, not delayed by milter-greylist-4.6.4 (fry.vger.email [0.0.0.0]); Mon, 04 Dec 2023 03:20:57 -0800 (PST) Il 01/12/23 12:14, Boris Brezillon ha scritto: > On Fri, 1 Dec 2023 11:40:27 +0100 > AngeloGioacchino Del Regno > wrote: > >> To make sure that we don't unintentionally perform any unclocked and/or >> unpowered R/W operation on GPU registers, before turning off clocks and >> regulators we must make sure that no GPU, JOB or MMU ISR execution is >> pending: doing that required to add a mechanism to synchronize the > > ^ requires the addition of a mechanism... > >> interrupts on suspend. >> >> Add functions panfrost_{gpu,job,mmu}_suspend_irq() which will perform >> interrupts masking and ISR execution synchronization, and then call >> those in the panfrost_device_runtime_suspend() handler in the exact >> sequence of job (may require mmu!) -> mmu -> gpu. >> >> As a side note, JOB and MMU suspend_irq functions needed some special >> treatment: as their interrupt handlers will unmask interrupts, it was >> necessary to add a bitmap for `is_suspended` which is used to address > > to add an `is_suspended` bitmap which is used... > >> the possible corner case of unintentional IRQ unmasking because of ISR >> execution after a call to synchronize_irq(). > > Also fixes the case where the interrupt handler is called when the > device is suspended because the IRQ line is shared with another device. > No need to update the commit message for that though. > >> >> At resume, clear each is_suspended bit in the reset path of JOB/MMU >> to allow unmasking the interrupts. >> >> Signed-off-by: AngeloGioacchino Del Regno >> --- >> drivers/gpu/drm/panfrost/panfrost_device.c | 3 +++ >> drivers/gpu/drm/panfrost/panfrost_device.h | 7 +++++++ >> drivers/gpu/drm/panfrost/panfrost_gpu.c | 6 ++++++ >> drivers/gpu/drm/panfrost/panfrost_gpu.h | 1 + >> drivers/gpu/drm/panfrost/panfrost_job.c | 20 +++++++++++++++++--- >> drivers/gpu/drm/panfrost/panfrost_job.h | 1 + >> drivers/gpu/drm/panfrost/panfrost_mmu.c | 19 ++++++++++++++++--- >> drivers/gpu/drm/panfrost/panfrost_mmu.h | 1 + >> 8 files changed, 52 insertions(+), 6 deletions(-) >> >> diff --git a/drivers/gpu/drm/panfrost/panfrost_device.c b/drivers/gpu/drm/panfrost/panfrost_device.c >> index c90ad5ee34e7..a45e4addcc19 100644 >> --- a/drivers/gpu/drm/panfrost/panfrost_device.c >> +++ b/drivers/gpu/drm/panfrost/panfrost_device.c >> @@ -421,6 +421,9 @@ static int panfrost_device_runtime_suspend(struct device *dev) >> return -EBUSY; >> >> panfrost_devfreq_suspend(pfdev); >> + panfrost_job_suspend_irq(pfdev); >> + panfrost_mmu_suspend_irq(pfdev); >> + panfrost_gpu_suspend_irq(pfdev); >> panfrost_gpu_power_off(pfdev); >> >> return 0; >> diff --git a/drivers/gpu/drm/panfrost/panfrost_device.h b/drivers/gpu/drm/panfrost/panfrost_device.h >> index 54a8aad54259..5c24f01f8904 100644 >> --- a/drivers/gpu/drm/panfrost/panfrost_device.h >> +++ b/drivers/gpu/drm/panfrost/panfrost_device.h >> @@ -25,6 +25,12 @@ struct panfrost_perfcnt; >> #define NUM_JOB_SLOTS 3 >> #define MAX_PM_DOMAINS 5 >> >> +enum panfrost_drv_comp_bits { >> + PANFROST_COMP_BIT_MMU, >> + PANFROST_COMP_BIT_JOB, > > I think we need one for the GPU interrupt too, for the > irq-line-is-shared-with-another-device thing I was mentioning. > Yes, I've also reordered the entries by name for v4. >> + PANFROST_COMP_BIT_MAX >> +}; >> + >> /** >> * enum panfrost_gpu_pm - Supported kernel power management features >> * @GPU_PM_CLK_DIS: Allow disabling clocks during system suspend >> @@ -109,6 +115,7 @@ struct panfrost_device { >> >> struct panfrost_features features; >> const struct panfrost_compatible *comp; >> + DECLARE_BITMAP(is_suspended, PANFROST_COMP_BIT_MAX); >> >> spinlock_t as_lock; >> unsigned long as_in_use_mask; >> diff --git a/drivers/gpu/drm/panfrost/panfrost_gpu.c b/drivers/gpu/drm/panfrost/panfrost_gpu.c >> index 7adc4441fa14..3a6a4fe7aca1 100644 >> --- a/drivers/gpu/drm/panfrost/panfrost_gpu.c >> +++ b/drivers/gpu/drm/panfrost/panfrost_gpu.c >> @@ -452,6 +452,12 @@ void panfrost_gpu_power_off(struct panfrost_device *pfdev) >> dev_err(pfdev->dev, "l2 power transition timeout"); >> } >> >> +void panfrost_gpu_suspend_irq(struct panfrost_device *pfdev) >> +{ > > set_bit(PANFROST_COMP_BIT_GPU, pfdev->is_suspended); > > here, and an extra check in panfrost_gpu_irq_handler() to bail out > before the register accesses if PANFROST_COMP_BIT_GPU is set. > Right. >> + gpu_write(pfdev, GPU_INT_MASK, 0); >> + synchronize_irq(pfdev->gpu_irq); >> +} >> + >> int panfrost_gpu_init(struct panfrost_device *pfdev) >> { >> int err; >> diff --git a/drivers/gpu/drm/panfrost/panfrost_gpu.h b/drivers/gpu/drm/panfrost/panfrost_gpu.h >> index 876fdad9f721..d841b86504ea 100644 >> --- a/drivers/gpu/drm/panfrost/panfrost_gpu.h >> +++ b/drivers/gpu/drm/panfrost/panfrost_gpu.h >> @@ -15,6 +15,7 @@ u32 panfrost_gpu_get_latest_flush_id(struct panfrost_device *pfdev); >> int panfrost_gpu_soft_reset(struct panfrost_device *pfdev); >> void panfrost_gpu_power_on(struct panfrost_device *pfdev); >> void panfrost_gpu_power_off(struct panfrost_device *pfdev); >> +void panfrost_gpu_suspend_irq(struct panfrost_device *pfdev); >> >> void panfrost_cycle_counter_get(struct panfrost_device *pfdev); >> void panfrost_cycle_counter_put(struct panfrost_device *pfdev); >> diff --git a/drivers/gpu/drm/panfrost/panfrost_job.c b/drivers/gpu/drm/panfrost/panfrost_job.c >> index f9446e197428..7600e7741211 100644 >> --- a/drivers/gpu/drm/panfrost/panfrost_job.c >> +++ b/drivers/gpu/drm/panfrost/panfrost_job.c >> @@ -405,6 +405,8 @@ void panfrost_job_enable_interrupts(struct panfrost_device *pfdev) >> int j; >> u32 irq_mask = 0; >> >> + clear_bit(PANFROST_COMP_BIT_JOB, pfdev->is_suspended); >> + >> for (j = 0; j < NUM_JOB_SLOTS; j++) { >> irq_mask |= MK_JS_MASK(j); >> } >> @@ -413,6 +415,14 @@ void panfrost_job_enable_interrupts(struct panfrost_device *pfdev) >> job_write(pfdev, JOB_INT_MASK, irq_mask); >> } >> >> +void panfrost_job_suspend_irq(struct panfrost_device *pfdev) >> +{ >> + set_bit(PANFROST_COMP_BIT_JOB, pfdev->is_suspended); >> + >> + job_write(pfdev, JOB_INT_MASK, 0); >> + synchronize_irq(pfdev->js->irq); >> +} >> + >> static void panfrost_job_handle_err(struct panfrost_device *pfdev, >> struct panfrost_job *job, >> unsigned int js) >> @@ -792,9 +802,13 @@ static irqreturn_t panfrost_job_irq_handler_thread(int irq, void *data) >> struct panfrost_device *pfdev = data; >> >> panfrost_job_handle_irqs(pfdev); >> - job_write(pfdev, JOB_INT_MASK, >> - GENMASK(16 + NUM_JOB_SLOTS - 1, 16) | >> - GENMASK(NUM_JOB_SLOTS - 1, 0)); >> + >> + /* Enable interrupts only if we're not about to get suspended */ >> + if (!test_bit(PANFROST_COMP_BIT_JOB, pfdev->is_suspended)) >> + job_write(pfdev, JOB_INT_MASK, >> + GENMASK(16 + NUM_JOB_SLOTS - 1, 16) | >> + GENMASK(NUM_JOB_SLOTS - 1, 0)); >> + > > Missing if (test_bit(PANFROST_COMP_BIT_JOB, pfdev->is_suspended)) in > panfrost_job_irq_handler(), to make sure you don't access the registers > if the GPU is suspended. > Ok. Cheers, Angelo