Received: by 2002:a05:6902:102b:0:0:0:0 with SMTP id x11csp3031005ybt; Mon, 22 Jun 2020 13:08:13 -0700 (PDT) X-Google-Smtp-Source: ABdhPJxjjGjBHo0Qg9rxO9/D4dmfQw2UPesBtnv4b4YrcMwHAQ9SDg+ZD09ThyE14M4k1Eai2Ex8 X-Received: by 2002:a17:906:5e08:: with SMTP id n8mr17203601eju.132.1592856493835; Mon, 22 Jun 2020 13:08:13 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1592856493; cv=none; d=google.com; s=arc-20160816; b=qj2y/92SzZxAF5SOc9awQLA0Aqy/D+pvYEkeYv8RfHYBsPxR082lNKRO0b/dG1wRXd VRu1fOPZhAnePjtJ0e4EmeYcuJJQdqrZwemgsOGZF1mUIWI48rfTXYFMHM+Ha5DLrhMP 9BCqPvsWG/D2zrsMpJxQHRr+KOTUhd9TioA5KAaMxNAjO9q8tUnY3WzwHNn6N7QByU5u bbkeLzQ0goyuuUtoKLfIRXqsX3hrcU7GkuNEYDOik9qHgritpJKwfM4qsJRTKQiMpKwR wxHPPj2iBsPZ2Y0tJD+ODc+ojQOsis1u0PSzENv5sT1YPXpsKZjzgTR008mULQya/8MX ZlMQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding :content-language:in-reply-to:mime-version:user-agent:date :message-id:from:references:cc:to:subject:dkim-signature; bh=ODOSMs7hlDlueY6tO0Y70BJvb4BVdVy4itgUewuyPXQ=; b=ADrE/smT76fKOUj9rnTLDJ9uyt3cgMPBtMNi0bJdIQSaB6nwwN693EmfzZixBjuFuG ye+mVp4CLZE52uD50SWoR+R+whicHWN/kIeTiIJHEFDVpXEXZIKXM9jIaNriWB2z7lHP 7cK42H4bBJ+rzhd9sFqMTZktnJfll95unvVFhCnA40juEoky2h+274NLRXCFS9pzByuT SfFMnrZjEERXY4yFh10y0jq8qv1JgKFUv5XAHgpSeEdodf4WFMxr7csPxMv7muIYCN4x eQwSkCwJO7xIv7f2vNtluuuVtlqy1f3lE+mGI3srVFxTM+bK9aZEncOLb/CDD7uiDV1t mJgg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@linaro.org header.s=google header.b=uFaTK3d5; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=linaro.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id c11si3705613edr.502.2020.06.22.13.07.51; Mon, 22 Jun 2020 13:08:13 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; dkim=pass header.i=@linaro.org header.s=google header.b=uFaTK3d5; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=linaro.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1728646AbgFVUCh (ORCPT + 99 others); Mon, 22 Jun 2020 16:02:37 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:52488 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1728050AbgFVUCP (ORCPT ); Mon, 22 Jun 2020 16:02:15 -0400 Received: from mail-wm1-x344.google.com (mail-wm1-x344.google.com [IPv6:2a00:1450:4864:20::344]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 4535BC06179B for ; Mon, 22 Jun 2020 12:47:00 -0700 (PDT) Received: by mail-wm1-x344.google.com with SMTP id g10so698114wmh.4 for ; Mon, 22 Jun 2020 12:47:00 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=linaro.org; s=google; h=subject:to:cc:references:from:message-id:date:user-agent :mime-version:in-reply-to:content-language:content-transfer-encoding; bh=ODOSMs7hlDlueY6tO0Y70BJvb4BVdVy4itgUewuyPXQ=; b=uFaTK3d56s7T4bkZ+sGL815DQdbAIf5cC8yFeAHB7iaBtf7T1St4qfhuFmFMJ7f6/r t3xdDkBr2I+T39H09CDtHgv6oo/c7ibkh9AIfNq+39DuGI2KibjJMbijs4S4rv6k95+4 6HUte0FXf/ZEXY12mInPLRA8OUYmrEzkOssBKxRitUovbFgfMVJH8hUDZts6iIO67xB+ hRzNmZTbsoYDo6NjbpLWAxZnBVHBs6QniFGG65hb6aBZHA3jbq6CcBN3mIE/o93azWMy O/MOwa8kjvX6+gA7jakig8mo/V8Q2EqgdgjEmdhoRH0dQD+FvrNICMjEFKXgNifncJAk 1dCA== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:subject:to:cc:references:from:message-id:date :user-agent:mime-version:in-reply-to:content-language :content-transfer-encoding; bh=ODOSMs7hlDlueY6tO0Y70BJvb4BVdVy4itgUewuyPXQ=; b=YraAWAe60G8/4I6ZyTw8qkZzuBMTZTbRNQl7i4cALUmnU6qz324LCM1Ixu/N07dP6K 99u2OBmNMIJ1jqMMPWqJcKzwmnvmHCJCyKxlwaXtDNUHh/2csqLVwQ2SkRuEoqXteBNP 5yCLPdYwOOxmjKPz+z40j2dMdphpzhCjzNWMu8YcJJyskIdnVEDNxkiTTgngYqorlWrV D0oJOLaUtkffEdoSU7x8FueACDCGhx+Oh9LACN1d/X8wECS57mg5HNHyRRnjyfEH73Kx m9EzSMXiRggtiNPx5cejsbMsyCp2t/1whJI6Qs1MkZP4W6ZJQ4LQOgVYTQ7mPsz+QIdb Au8w== X-Gm-Message-State: AOAM5303fu0pQak915NwQLJSveSh02QCd2c3XC25MvK/Ba4rPtV1HawY 9sB8kPdGTqHINJp/LxXBG3c+zQ== X-Received: by 2002:a1c:4444:: with SMTP id r65mr1767496wma.129.1592855218748; Mon, 22 Jun 2020 12:46:58 -0700 (PDT) Received: from [192.168.0.41] (lns-bzn-59-82-252-131-168.adsl.proxad.net. [82.252.131.168]) by smtp.googlemail.com with ESMTPSA id i17sm12945242wrc.34.2020.06.22.12.46.57 (version=TLS1_3 cipher=TLS_AES_128_GCM_SHA256 bits=128/128); Mon, 22 Jun 2020 12:46:58 -0700 (PDT) Subject: Re: [PATCH v4 1/2] thermal: add support for the MCU controlled FAN on Khadas boards To: Neil Armstrong , lee.jones@linaro.org Cc: khilman@baylibre.com, linux-amlogic@lists.infradead.org, linux-pm@vger.kernel.org, linux-arm-kernel@lists.infradead.org, linux-kernel@vger.kernel.org, rui.zhang@intel.com, amit.kucheria@verdurent.com, Amit Kucheria References: <20200618133818.15857-1-narmstrong@baylibre.com> <20200618133818.15857-2-narmstrong@baylibre.com> From: Daniel Lezcano Message-ID: <53aa62a3-1d8e-bc91-1a2b-88c766276beb@linaro.org> Date: Mon, 22 Jun 2020 21:46:56 +0200 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:68.0) Gecko/20100101 Thunderbird/68.8.0 MIME-Version: 1.0 In-Reply-To: <20200618133818.15857-2-narmstrong@baylibre.com> Content-Type: text/plain; charset=utf-8 Content-Language: en-US Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 18/06/2020 15:38, Neil Armstrong wrote: > The new Khadas VIM2 and VIM3 boards controls the cooling fan via the > on-board microcontroller. > > This implements the FAN control as thermal devices and as cell of the Khadas > MCU MFD driver. > > Signed-off-by: Neil Armstrong > Reviewed-by: Amit Kucheria > --- > Hi Lee, > > Could you apply this patch via the MFD tree since it depends on > the linux/mfd/khadas-mcu.h header ? > > This patch is unchanged from the v3 serie. > > Thanks, > Neil > > drivers/thermal/Kconfig | 11 ++ > drivers/thermal/Makefile | 1 + > drivers/thermal/khadas_mcu_fan.c | 174 +++++++++++++++++++++++++++++++ > 3 files changed, 186 insertions(+) > create mode 100644 drivers/thermal/khadas_mcu_fan.c > > diff --git a/drivers/thermal/Kconfig b/drivers/thermal/Kconfig > index 3eb2348e5242..0125561488c9 100644 > --- a/drivers/thermal/Kconfig > +++ b/drivers/thermal/Kconfig > @@ -500,4 +500,15 @@ config SPRD_THERMAL > help > Support for the Spreadtrum thermal sensor driver in the Linux thermal > framework. > + > +config KHADAS_MCU_FAN_THERMAL > + tristate "Khadas MCU controller FAN cooling support" > + depends on OF || COMPILE_TEST > + depends on MFD_KHADAS_MCU > + select MFD_CORE > + select REGMAP > + help > + If you say yes here you get support for the FAN controlled > + by the Microcontroller found on the Khadas VIM boards. > + > endif > diff --git a/drivers/thermal/Makefile b/drivers/thermal/Makefile > index 0c8b84a09b9a..4b6aabaa7e31 100644 > --- a/drivers/thermal/Makefile > +++ b/drivers/thermal/Makefile > @@ -61,3 +61,4 @@ obj-$(CONFIG_ZX2967_THERMAL) += zx2967_thermal.o > obj-$(CONFIG_UNIPHIER_THERMAL) += uniphier_thermal.o > obj-$(CONFIG_AMLOGIC_THERMAL) += amlogic_thermal.o > obj-$(CONFIG_SPRD_THERMAL) += sprd_thermal.o > +obj-$(CONFIG_KHADAS_MCU_FAN_THERMAL) += khadas_mcu_fan.o > diff --git a/drivers/thermal/khadas_mcu_fan.c b/drivers/thermal/khadas_mcu_fan.c > new file mode 100644 > index 000000000000..6995b443cad4 > --- /dev/null > +++ b/drivers/thermal/khadas_mcu_fan.c > @@ -0,0 +1,174 @@ > +// SPDX-License-Identifier: GPL-2.0-or-later > +/* > + * Khadas MCU Controlled FAN driver > + * > + * Copyright (C) 2020 BayLibre SAS > + * Author(s): Neil Armstrong > + */ > + > +#include > +#include > +#include > +#include > +#include > +#include > +#include > + > +#define MAX_LEVEL 3 > + > +struct khadas_mcu_fan_ctx { > + struct khadas_mcu *mcu; > + unsigned int level; > + struct thermal_cooling_device *cdev; > +}; > + > +static int khadas_mcu_fan_set_level(struct khadas_mcu_fan_ctx *ctx, > + unsigned int level) > +{ > + int ret; > + > + ret = regmap_write(ctx->mcu->regmap, KHADAS_MCU_CMD_FAN_STATUS_CTRL_REG, > + level); > + if (ret) > + return ret; > + > + ctx->level = level; > + > + return 0; > +} > + > +static int khadas_mcu_fan_get_max_state(struct thermal_cooling_device *cdev, > + unsigned long *state) > +{ > + struct khadas_mcu_fan_ctx *ctx = cdev->devdata; > + > + if (!ctx) > + return -EINVAL; It is pointless to check 'ctx' is NULL, that was done at probe time. > + *state = MAX_LEVEL; > + > + return 0; > +} > + > +static int khadas_mcu_fan_get_cur_state(struct thermal_cooling_device *cdev, > + unsigned long *state) > +{ > + struct khadas_mcu_fan_ctx *ctx = cdev->devdata; > + > + if (!ctx) > + return -EINVAL; Ditto > + *state = ctx->level; > + > + return 0; > +} > + > +static int > +khadas_mcu_fan_set_cur_state(struct thermal_cooling_device *cdev, > + unsigned long state) > +{ > + struct khadas_mcu_fan_ctx *ctx = cdev->devdata; > + > + if (!ctx || (state > MAX_LEVEL)) > + return -EINVAL; Ditto > + if (state == ctx->level) > + return 0; > + > + return khadas_mcu_fan_set_level(ctx, state); > +} > + > +static const struct thermal_cooling_device_ops khadas_mcu_fan_cooling_ops = { > + .get_max_state = khadas_mcu_fan_get_max_state, > + .get_cur_state = khadas_mcu_fan_get_cur_state, > + .set_cur_state = khadas_mcu_fan_set_cur_state, > +}; > + > +static int khadas_mcu_fan_probe(struct platform_device *pdev) > +{ > + struct khadas_mcu *mcu = dev_get_drvdata(pdev->dev.parent); > + struct thermal_cooling_device *cdev; > + struct device *dev = &pdev->dev; > + struct khadas_mcu_fan_ctx *ctx; > + int ret; > + > + ctx = devm_kzalloc(dev, sizeof(*ctx), GFP_KERNEL); > + if (!ctx) > + return -ENOMEM; > + ctx->mcu = mcu; > + platform_set_drvdata(pdev, ctx); > + > + cdev = devm_thermal_of_cooling_device_register(dev->parent, > + dev->parent->of_node, "khadas-mcu-fan", ctx, > + &khadas_mcu_fan_cooling_ops); > + if (IS_ERR(cdev)) { > + ret = PTR_ERR(cdev); > + dev_err(dev, > + "Failed to register khadas-mcu-fan as cooling device: %d\n", > + ret); > + return ret; > + } > + ctx->cdev = cdev; > + thermal_cdev_update(cdev); > + > + return 0; > +} > + > +static int khadas_mcu_fan_disable(struct device *dev) > +{ > + struct khadas_mcu_fan_ctx *ctx = dev_get_drvdata(dev); > + unsigned int level_save = ctx->level; > + int ret; > + > + ret = khadas_mcu_fan_set_level(ctx, 0); > + if (ret) > + return ret; > + > + ctx->level = level_save; Nitpicking : move the save section to suspend. > + return 0; > +} > + > +static void khadas_mcu_fan_shutdown(struct platform_device *pdev) > +{ > + khadas_mcu_fan_disable(&pdev->dev); > +} > + > +#ifdef CONFIG_PM_SLEEP > +static int khadas_mcu_fan_suspend(struct device *dev) > +{ > + return khadas_mcu_fan_disable(dev); > +} > + > +static int khadas_mcu_fan_resume(struct device *dev) > +{ > + struct khadas_mcu_fan_ctx *ctx = dev_get_drvdata(dev); > + > + return khadas_mcu_fan_set_level(ctx, ctx->level); Out of curiosity, did you check the fan is not continuously spinning after a resume when the suspend happened during a mitigation phase? > +} > +#endif > + > +static SIMPLE_DEV_PM_OPS(khadas_mcu_fan_pm, khadas_mcu_fan_suspend, > + khadas_mcu_fan_resume); > + > +static const struct platform_device_id khadas_mcu_fan_id_table[] = { > + { .name = "khadas-mcu-fan-ctrl", }, > + {}, > +}; > +MODULE_DEVICE_TABLE(platform, khadas_mcu_fan_id_table); > + > +static struct platform_driver khadas_mcu_fan_driver = { > + .probe = khadas_mcu_fan_probe, > + .shutdown = khadas_mcu_fan_shutdown, > + .driver = { > + .name = "khadas-mcu-fan-ctrl", > + .pm = &khadas_mcu_fan_pm, > + }, > + .id_table = khadas_mcu_fan_id_table, > +}; > + > +module_platform_driver(khadas_mcu_fan_driver); > + > +MODULE_AUTHOR("Neil Armstrong "); > +MODULE_DESCRIPTION("Khadas MCU FAN driver"); > +MODULE_LICENSE("GPL"); > -- Linaro.org │ Open source software for ARM SoCs Follow Linaro: Facebook | Twitter | Blog