Return-Path: Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1753176AbbBWQOL (ORCPT ); Mon, 23 Feb 2015 11:14:11 -0500 Received: from mailout4.samsung.com ([203.254.224.34]:26525 "EHLO mailout4.samsung.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1752351AbbBWQOI (ORCPT ); Mon, 23 Feb 2015 11:14:08 -0500 X-AuditID: cbfee61a-f79c06d000004e71-41-54eb51cc46b6 Date: Mon, 23 Feb 2015 17:13:36 +0100 From: Lukasz Majewski To: Guenter Roeck Cc: Eduardo Valentin , Kamil Debski , Jean Delvare , Kukjin Kim , lm-sensors@lm-sensors.org, Linux PM list , "linux-samsung-soc@vger.kernel.org" , devicetree@vger.kernel.org, Lukasz Majewski , Kukjin Kim , linux-kernel@vger.kernel.org, Sjoerd Simons , Abhilash Kesavan , Abhilash Kesavan Subject: Re: [PATCH v4 7/8] hwmon: pwm-fan: Read PWM FAN configuration from device tree Message-id: <20150223171336.2e09ce50@amdc2363> In-reply-to: <54E7D0B3.7040800@roeck-us.net> References: <1418897591-18332-1-git-send-email-l.majewski@samsung.com> <1424254056-5904-1-git-send-email-l.majewski@samsung.com> <1424254056-5904-8-git-send-email-l.majewski@samsung.com> <54E7D0B3.7040800@roeck-us.net> Organization: SPRC Poland X-Mailer: Claws Mail 3.8.1 (GTK+ 2.24.10; x86_64-pc-linux-gnu) MIME-version: 1.0 Content-type: text/plain; charset=US-ASCII Content-transfer-encoding: 7bit X-Brightmail-Tracker: H4sIAAAAAAAAA+NgFlrIIsWRmVeSWpSXmKPExsVy+t9jAd0zga9DDL7O17Z4vGYxk8X8I+dY LeZfucZqsfTuJzaLH68vsFms+atk0bvgKptF/+PXzBZvHnFbXN41h83ic+8RRosZ5/cxWTxZ eIbJ4s60vWwW04+/ZXXg9/j7/DqLx85Zd9k9Nq3qZPN4MHE3m8e6aW+ZPXZ+b2D36NuyitFj 8+lqj8+b5AI4o7hsUlJzMstSi/TtErgyHs08wVLQrFHxdfk5xgbGpfJdjBwcEgImEqev2HUx cgKZYhIX7q1n62Lk4hASWMQocbthLROE84tR4vP0+cwgVSwCqhIv1/8As9kE9CQ+333KBGKL CKhJNJ9qAetmFtjLIvH942E2kISwQKTEgfZOdpBtvEANyx+xgIQ5BXQkVv54xgix4AGjxNkb bWCD+AUkJdr/QSyQELCTOPdpAzuIzSsgKPFj8j2wZmYBLYnN25pYIWx5ic1r3jJPYBSchaRs FpKyWUjKFjAyr2IUTS1ILihOSs811CtOzC0uzUvXS87P3cQIjrhnUjsYVzZYHGIU4GBU4uFt yH0VIsSaWFZcmXuIUYKDWUmE943u6xAh3pTEyqrUovz4otKc1OJDjNIcLErivEr2bSFCAumJ JanZqakFqUUwWSYOTqkGxqRo9U9GU+bzVImebbo7V0uafcvheYtunNBwSD7KdXa9QkrxQYfG OKVP/g+SOFKeb+Dkigxt2L5OQmtNHM+ii3MaD7ennC+/ser71Se1D9cufvDj/+LyrWkHPpeZ /TxqdNns1X7rJpVTta4n1rSoeb/kWy27fvVTCf39R4SnqDeKtSx29JF/+luJpTgj0VCLuag4 EQCqN8YTtAIAAA== Sender: linux-kernel-owner@vger.kernel.org List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Content-Length: 5202 Lines: 171 Hi Guenter, > On 02/18/2015 02:07 AM, Lukasz Majewski wrote: > > This patch provides code for reading PWM FAN configuration data via > > device tree. The pwm-fan can work with full speed when configuration > > is not provided. However, errors are propagated when wrong DT > > bindings are found. > > Additionally the struct pwm_fan_ctx has been extended. > > > > Signed-off-by: Lukasz Majewski > > --- > > Changes for v2: > > - Rename pwm_fan_max_states to pwm_fan_cooling_levels > > - Moving pwm_fan_of_get_cooling_data() call after setting end > > enabling PWM FAN > > - pwm_fan_of_get_cooling_data() now can fail - preserving old > > behaviour > > - Remove unnecessary dev_err() call > > Changes for v3: > > - Patch's headline has been reedited > > - pwm_fan_of_get_cooling_data() return code is now being checked. > > - of_property_count_elems_of_size() is now used instead > > of_find_property() > > - More verbose patch description added > > Changes for v4: > > - dev_err() has been removed from pwm_fan_of_get_cooling_data() > > - Returning -EINVAL when "cooling-levels" are defined in DT, but > > doesn't have the value > > --- > > drivers/hwmon/pwm-fan.c | 52 > > ++++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, > > 51 insertions(+), 1 deletion(-) > > > > diff --git a/drivers/hwmon/pwm-fan.c b/drivers/hwmon/pwm-fan.c > > index bd42d39..82cd06a 100644 > > --- a/drivers/hwmon/pwm-fan.c > > +++ b/drivers/hwmon/pwm-fan.c > > @@ -30,7 +30,10 @@ > > struct pwm_fan_ctx { > > struct mutex lock; > > struct pwm_device *pwm; > > - unsigned char pwm_value; > > + unsigned int pwm_value; > > + unsigned int pwm_fan_state; > > + unsigned int pwm_fan_max_state; > > + unsigned int *pwm_fan_cooling_levels; > > }; > > > > static int __set_pwm(struct pwm_fan_ctx *ctx, unsigned long pwm) > > @@ -100,6 +103,48 @@ static struct attribute *pwm_fan_attrs[] = { > > > > ATTRIBUTE_GROUPS(pwm_fan); > > > > +int pwm_fan_of_get_cooling_data(struct device *dev, struct > > pwm_fan_ctx *ctx) +{ > > + struct device_node *np = dev->of_node; > > + int num, i, ret; > > + > > + ret = of_property_count_elems_of_size(np, "cooling-levels", > > + sizeof(u32)); > > + > > + if (ret == -EINVAL) > > + return 0; > > The function returns -EINVAL if there is no such property, > but also if prop->length % elem_size != 0. The latter _would_ > be an error. > > Overall I don't entirely understand why you do not call > of_find_property first. If that returns NULL, you would know for sure > that the property does not exist, and you would not have to second > guess the returned error from of_property_count_elems_of_size. For sake of readability I will at v5 first check of_find_property() and if it is correct, then I will call of_property_count_u32_elems(). > > On a side note, there is of_property_count_u32_elems() to count > properties of size u32. > > > + > > + if (ret <= 0) { > > + dev_err(dev, "Wrong data!\n"); > > + return ret ? ret : -EINVAL; > > + } > > If devicetree is not configured, of_property_count_elems_of_size > returns -ENOSYS, which is returned, causing the driver to fail > loading. Has of_property_count_elems_of_size() returns -ENOSYS? Maybe something has changed, but in my linux-vanila (3.19-rc4) at ./drivers/of/base.c it returns -EINVAL, -ENODATA or number of elements. Have I missed something? > > > + > > + num = ret; > > + ctx->pwm_fan_cooling_levels = devm_kzalloc(dev, num * > > sizeof(u32), > > + GFP_KERNEL); > > + if (!ctx->pwm_fan_cooling_levels) > > + return -ENOMEM; > > + > > + ret = of_property_read_u32_array(np, "cooling-levels", > > + > > ctx->pwm_fan_cooling_levels, num); > > + if (ret) { > > + dev_err(dev, "Property 'cooling-levels' cannot be > > read!\n"); > > + return ret; > > + } > > + > > + for (i = 0; i < num; i++) { > > + if (ctx->pwm_fan_cooling_levels[i] > MAX_PWM) { > > + dev_err(dev, "PWM fan state[%d]:%d > > > %d\n", i, > > + ctx->pwm_fan_cooling_levels[i], > > MAX_PWM); > > + return -EINVAL; > > + } > > + } > > + > > + ctx->pwm_fan_max_state = num - 1; > > + > > + return 0; > > +} > > + > > static int pwm_fan_probe(struct platform_device *pdev) > > { > > struct device *hwmon; > > @@ -145,6 +190,11 @@ static int pwm_fan_probe(struct > > platform_device *pdev) pwm_disable(ctx->pwm); > > return PTR_ERR(hwmon); > > } > > + > > + ret = pwm_fan_of_get_cooling_data(&pdev->dev, ctx); > > + if (ret) > > + return ret; I think that here is the confusing part. Please compare this patch with the following one. Here we configure ctx struct via DT. If of_property_count_u32_elems() returns -EINVAL, then we consider that "cooling-levels" wasn't defined in DT and return with 0. Other error codes are considered as errors and probe return error code. > > + > > return 0; > > } > > > > > -- Best regards, Lukasz Majewski Samsung R&D Institute Poland (SRPOL) | Linux Platform Group -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/