Received: by 2002:a05:6358:d09b:b0:dc:cd0c:909e with SMTP id jc27csp740352rwb; Tue, 29 Nov 2022 04:57:00 -0800 (PST) X-Google-Smtp-Source: AA0mqf5yJ3oMJfLpGpimORAxe+gbiHUrCLdGQgLEKZ7z6taHEM1FRgzh2vegLYFh+rV/GQsFClsV X-Received: by 2002:a17:902:c40a:b0:189:884b:96db with SMTP id k10-20020a170902c40a00b00189884b96dbmr10674678plk.9.1669726620443; Tue, 29 Nov 2022 04:57:00 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1669726620; cv=none; d=google.com; s=arc-20160816; b=Fmv5/FKi3SKx2ncc+xC+TVBsgF6n0s7hBBWKL0dRw/Ut7NgQavxz4QhBrHvaOYqxAQ KYmO+Rq+PR7qnSTz6lXZsVsw8xse8JerimZV9M1KOfXK0zpEnY2JXeqyxk5I216d+/As qssNF4f493zOhFtwRsr53amMU05FBJpFBeym45LynSxboBC8ZPYZoJPCU7gdAJ1cMvvZ 02YJRUseKZoAur0ClgA/HWxdVh/CI4AUTFvMb4xd5M4wrfJgPx8hBsUi2jUufCSm07Dr OghkQof8EgGI1eEiPmm0HaoeycLAPRSyVmoJ/qrM78RqgE3Eo+DXMJKK1zxWKmtO7tVE goTA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:mime-version :references:in-reply-to:message-id:cc:to:subject:from:date :dkim-signature; bh=3OAstYW1TcYgu5eEwQmk/GWGExhOJXuywR9MICWyv4w=; b=jasjfJob9MpejqJ6+Tlcb8cRsq/yI9QHaR/qFZV2b9Au7bZY4XC2WZb85MHVxP4S0N 9ctZVYH3ctmKw330SL677O7zsl2GdABtljMy/vLWcA+hFpT7PvYm5NkzyE2waJmlCc1+ jL/ZNJDX5t10FwCJ5aor46J/QT+K2hHMPK8or1yIifnwzqHE0mHv8GZfv5vtfJIxq2qh EVi0pm2nCGI3eDPGD3R4X14cNrPpPG75Pt23WpPs2AdnXSCKILCeaXAUxlF7Mq4WA/QH xTIl6e0YjU9bnGR8xDe0Urje9ATPHKwtFl/XKdNvAaZ3G5C8jDYgCknZkIG1DwxaDGFl Jzmw== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@crapouillou.net header.s=mail header.b=hmsSQxCh; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=crapouillou.net Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id u38-20020a631426000000b0046fb2a57348si14336967pgl.84.2022.11.29.04.56.48; Tue, 29 Nov 2022 04:57:00 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; dkim=pass header.i=@crapouillou.net header.s=mail header.b=hmsSQxCh; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=crapouillou.net Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S234093AbiK2M0L (ORCPT + 83 others); Tue, 29 Nov 2022 07:26:11 -0500 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:50692 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S234112AbiK2M0K (ORCPT ); Tue, 29 Nov 2022 07:26:10 -0500 Received: from aposti.net (aposti.net [89.234.176.197]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id BCA16532CC; Tue, 29 Nov 2022 04:26:07 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=crapouillou.net; s=mail; t=1669724765; h=from:from:sender:reply-to:subject:subject:date:date: message-id:message-id:to:to:cc:cc:mime-version:mime-version: content-type:content-type: content-transfer-encoding:content-transfer-encoding: in-reply-to:in-reply-to:references:references; bh=3OAstYW1TcYgu5eEwQmk/GWGExhOJXuywR9MICWyv4w=; b=hmsSQxChCPqRM/sZ2awpWC87pw2oT4CtTMfd+XtrPYMnsc4Su06uR4UJLAOdSQYa+hF6IW 2PDidp+hZXCyuMInjbrQeKbIamHWxUiJmuyA2wFQsUzxDCEEMzdxKkK88dhAuChA4JdD3V iqb1OqnIUtyIWwvRLmI91BscDQGaDPw= Date: Tue, 29 Nov 2022 12:25:56 +0000 From: Paul Cercueil Subject: Re: [PATCH 2/5] pwm: jz4740: Fix pin level of disabled TCU2 channels, part 2 To: Uwe =?iso-8859-1?q?Kleine-K=F6nig?= Cc: Thierry Reding , od@opendingux.net, linux-pwm@vger.kernel.org, linux-kernel@vger.kernel.org, linux-mips@vger.kernel.org, stable@vger.kernel.org Message-Id: <8VZ3MR.B9R316RWSFMQ@crapouillou.net> In-Reply-To: <20221128143911.n3woy6mjom5n4sad@pengutronix.de> References: <20221024205213.327001-1-paul@crapouillou.net> <20221024205213.327001-3-paul@crapouillou.net> <20221025064410.brrx5faa4jtwo67b@pengutronix.de> <20221128143911.n3woy6mjom5n4sad@pengutronix.de> MIME-Version: 1.0 Content-Type: text/plain; charset=iso-8859-1; format=flowed Content-Transfer-Encoding: quoted-printable X-Spam-Status: No, score=-2.1 required=5.0 tests=BAYES_00,DKIM_SIGNED, DKIM_VALID,DKIM_VALID_AU,DKIM_VALID_EF,SPF_HELO_PASS,SPF_PASS autolearn=ham autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Hi Uwe, Le lun. 28 nov. 2022 =E0 15:39:11 +0100, Uwe Kleine-K=F6nig=20 a =E9crit : > Hello, >=20 > On Tue, Oct 25, 2022 at 11:10:46AM +0100, Paul Cercueil wrote: >> Le mar. 25 oct. 2022 =E0 08:44:10 +0200, Uwe Kleine-K=F6nig >> a =E9crit : >> > On Mon, Oct 24, 2022 at 09:52:10PM +0100, Paul Cercueil wrote: >> > > After commit a020f22a4ff5 ("pwm: jz4740: Make PWM start with=20 >> the >> > > active part"), >> > > the trick to set duty > period to properly shut down TCU2=20 >> channels >> > > did >> > > not work anymore, because of the polarity inversion. >> > > >> > > Address this issue by restoring the proper polarity before >> > > disabling the >> > > channels. >> > > >> > > Fixes: a020f22a4ff5 ("pwm: jz4740: Make PWM start with the=20 >> active >> > > part") >> > > Signed-off-by: Paul Cercueil >> > > Cc: stable@vger.kernel.org >> > > --- >> > > drivers/pwm/pwm-jz4740.c | 62 >> > > ++++++++++++++++++++++++++-------------- >> > > 1 file changed, 40 insertions(+), 22 deletions(-) >> > > >> > > diff --git a/drivers/pwm/pwm-jz4740.c=20 >> b/drivers/pwm/pwm-jz4740.c >> > > index 228eb104bf1e..65462a0052af 100644 >> > > --- a/drivers/pwm/pwm-jz4740.c >> > > +++ b/drivers/pwm/pwm-jz4740.c >> > > @@ -97,6 +97,19 @@ static int jz4740_pwm_enable(struct pwm_chip >> > > *chip, struct pwm_device *pwm) >> > > return 0; >> > > } >> > > >> > > +static void jz4740_pwm_set_polarity(struct jz4740_pwm_chip=20 >> *jz, >> > > + unsigned int hwpwm, >> > > + enum pwm_polarity polarity) >> > > +{ >> > > + unsigned int value =3D 0; >> > > + >> > > + if (polarity =3D=3D PWM_POLARITY_INVERSED) >> > > + value =3D TCU_TCSR_PWM_INITL_HIGH; >> > > + >> > > + regmap_update_bits(jz->map, TCU_REG_TCSRc(hwpwm), >> > > + TCU_TCSR_PWM_INITL_HIGH, value); >> > > +} >> > > + >> > > static void jz4740_pwm_disable(struct pwm_chip *chip, struct >> > > pwm_device *pwm) >> > > { >> > > struct jz4740_pwm_chip *jz =3D to_jz4740(chip); >> > > @@ -130,6 +143,7 @@ static int jz4740_pwm_apply(struct pwm_chip >> > > *chip, struct pwm_device *pwm, >> > > unsigned long long tmp =3D 0xffffull * NSEC_PER_SEC; >> > > struct clk *clk =3D pwm_get_chip_data(pwm); >> > > unsigned long period, duty; >> > > + enum pwm_polarity polarity; >> > > long rate; >> > > int err; >> > > >> > > @@ -169,6 +183,9 @@ static int jz4740_pwm_apply(struct pwm_chip >> > > *chip, struct pwm_device *pwm, >> > > if (duty >=3D period) >> > > duty =3D period - 1; >> > > >> > > + /* Restore regular polarity before disabling the channel. */ >> > > + jz4740_pwm_set_polarity(jz4740, pwm->hwpwm, state->polarity); >> > > + >> > >> > Does this introduce a glitch? >>=20 >> Maybe. But the PWM is shut down before finishing its period anyway,=20 >> so there >> was already a glitch. >>=20 >> > > jz4740_pwm_disable(chip, pwm); >> > > >> > > err =3D clk_set_rate(clk, rate); >> > > @@ -190,29 +207,30 @@ static int jz4740_pwm_apply(struct=20 >> pwm_chip >> > > *chip, struct pwm_device *pwm, >> > > regmap_update_bits(jz4740->map, TCU_REG_TCSRc(pwm->hwpwm), >> > > TCU_TCSR_PWM_SD, TCU_TCSR_PWM_SD); >> > > >> > > - /* >> > > - * Set polarity. >> > > - * >> > > - * The PWM starts in inactive state until the internal timer >> > > reaches the >> > > - * duty value, then becomes active until the timer reaches=20 >> the >> > > period >> > > - * value. In theory, we should then use (period - duty) as=20 >> the >> > > real duty >> > > - * value, as a high duty value would otherwise result in the=20 >> PWM >> > > pin >> > > - * being inactive most of the time. >> > > - * >> > > - * Here, we don't do that, and instead invert the polarity=20 >> of the >> > > PWM >> > > - * when it is active. This trick makes the PWM start with its >> > > active >> > > - * state instead of its inactive state. >> > > - */ >> > > - if ((state->polarity =3D=3D PWM_POLARITY_NORMAL) ^=20 >> state->enabled) >> > > - regmap_update_bits(jz4740->map, TCU_REG_TCSRc(pwm->hwpwm), >> > > - TCU_TCSR_PWM_INITL_HIGH, 0); >> > > - else >> > > - regmap_update_bits(jz4740->map, TCU_REG_TCSRc(pwm->hwpwm), >> > > - TCU_TCSR_PWM_INITL_HIGH, >> > > - TCU_TCSR_PWM_INITL_HIGH); >> > > - >> > > - if (state->enabled) >> > > + if (state->enabled) { >> > > + /* >> > > + * Set polarity. >> > > + * >> > > + * The PWM starts in inactive state until the internal timer >> > > + * reaches the duty value, then becomes active until the=20 >> timer >> > > + * reaches the period value. In theory, we should then use >> > > + * (period - duty) as the real duty value, as a high duty=20 >> value >> > > + * would otherwise result in the PWM pin being inactive=20 >> most of >> > > + * the time. >> > > + * >> > > + * Here, we don't do that, and instead invert the polarity=20 >> of >> > > + * the PWM when it is active. This trick makes the PWM start >> > > + * with its active state instead of its inactive state. >> > > + */ >> > > + if (state->polarity =3D=3D PWM_POLARITY_NORMAL) >> > > + polarity =3D PWM_POLARITY_INVERSED; >> > > + else >> > > + polarity =3D PWM_POLARITY_NORMAL; >> > > + >> > > + jz4740_pwm_set_polarity(jz4740, pwm->hwpwm, polarity); >> > > + >> > > jz4740_pwm_enable(chip, pwm); >> > > + } >> > >> > Note that for disabled PWMs there is no official guaranty about=20 >> the pin >> > state. So it would be ok (but admittedly not great) to simplify=20 >> the >> > driver and accept that the pinstate is active while the PWM is=20 >> off. >> > IMHO this is also better than a glitch. >> > >> > If a consumer wants the PWM to be in its inactive state, they=20 >> should >> > not disable it. >>=20 >> Completely disagree. I absolutely do not want the backlight to go=20 >> full >> bright mode when the PWM pin is disabled. And disabling the=20 >> backlight is a >> thing (for screen blanking and during mode changes). >=20 > For some hardwares there is no pretty choice. So the gist is: If the > backlight driver wants to ensure that the PWM pin is driven to its > inactive level, it should use: >=20 > pwm_apply(pwm, { .period =3D ..., .duty_cycle =3D 0, .enabled =3D true }= ); >=20 > and better not >=20 > pwm_apply(pwm, { ..., .enabled =3D false }); Well that sounds pretty stupid to me; why doesn't the PWM subsystem=20 enforce that the pins must be driven to their inactive level when the=20 PWM function is disabled? Then for such hardware you describe, the=20 corresponding PWM driver could itself apply a duty_cycle =3D 0 if that's=20 what it takes to get an inactive state. Cheers, -Paul