2011-06-20 16:48:19 +09:00
/*
* ntc_thermistor . c - NTC Thermistors
*
* Copyright ( C ) 2010 Samsung Electronics
* MyungJoo Ham < myungjoo . ham @ samsung . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*
*/
# include <linux/slab.h>
# include <linux/module.h>
# include <linux/pm_runtime.h>
# include <linux/math64.h>
# include <linux/platform_device.h>
# include <linux/err.h>
2013-03-13 09:38:20 +05:30
# include <linux/of.h>
# include <linux/of_device.h>
2011-06-20 16:48:19 +09:00
# include <linux/platform_data/ntc_thermistor.h>
2013-03-13 09:38:20 +05:30
# include <linux/iio/iio.h>
# include <linux/iio/machine.h>
# include <linux/iio/driver.h>
# include <linux/iio/consumer.h>
2011-06-20 16:48:19 +09:00
# include <linux/hwmon.h>
# include <linux/hwmon-sysfs.h>
struct ntc_compensation {
2013-03-13 16:40:39 -07:00
int temp_c ;
2011-06-20 16:48:19 +09:00
unsigned int ohm ;
} ;
2013-03-13 09:38:20 +05:30
static const struct platform_device_id ntc_thermistor_id [ ] = {
{ " ncp15wb473 " , TYPE_NCPXXWB473 } ,
{ " ncp18wb473 " , TYPE_NCPXXWB473 } ,
{ " ncp21wb473 " , TYPE_NCPXXWB473 } ,
{ " ncp03wb473 " , TYPE_NCPXXWB473 } ,
{ " ncp15wl333 " , TYPE_NCPXXWL333 } ,
{ } ,
} ;
2011-06-20 16:48:19 +09:00
/*
* A compensation table should be sorted by the values of . ohm
* in descending order .
* The following compensation tables are from the specification of Murata NTC
* Thermistors Datasheet
*/
2013-02-19 15:07:34 +05:30
static const struct ntc_compensation ncpXXwb473 [ ] = {
2013-03-13 16:40:39 -07:00
{ . temp_c = - 40 , . ohm = 1747920 } ,
{ . temp_c = - 35 , . ohm = 1245428 } ,
{ . temp_c = - 30 , . ohm = 898485 } ,
{ . temp_c = - 25 , . ohm = 655802 } ,
{ . temp_c = - 20 , . ohm = 483954 } ,
{ . temp_c = - 15 , . ohm = 360850 } ,
{ . temp_c = - 10 , . ohm = 271697 } ,
{ . temp_c = - 5 , . ohm = 206463 } ,
{ . temp_c = 0 , . ohm = 158214 } ,
{ . temp_c = 5 , . ohm = 122259 } ,
{ . temp_c = 10 , . ohm = 95227 } ,
{ . temp_c = 15 , . ohm = 74730 } ,
{ . temp_c = 20 , . ohm = 59065 } ,
{ . temp_c = 25 , . ohm = 47000 } ,
{ . temp_c = 30 , . ohm = 37643 } ,
{ . temp_c = 35 , . ohm = 30334 } ,
{ . temp_c = 40 , . ohm = 24591 } ,
{ . temp_c = 45 , . ohm = 20048 } ,
{ . temp_c = 50 , . ohm = 16433 } ,
{ . temp_c = 55 , . ohm = 13539 } ,
{ . temp_c = 60 , . ohm = 11209 } ,
{ . temp_c = 65 , . ohm = 9328 } ,
{ . temp_c = 70 , . ohm = 7798 } ,
{ . temp_c = 75 , . ohm = 6544 } ,
{ . temp_c = 80 , . ohm = 5518 } ,
{ . temp_c = 85 , . ohm = 4674 } ,
{ . temp_c = 90 , . ohm = 3972 } ,
{ . temp_c = 95 , . ohm = 3388 } ,
{ . temp_c = 100 , . ohm = 2902 } ,
{ . temp_c = 105 , . ohm = 2494 } ,
{ . temp_c = 110 , . ohm = 2150 } ,
{ . temp_c = 115 , . ohm = 1860 } ,
{ . temp_c = 120 , . ohm = 1615 } ,
{ . temp_c = 125 , . ohm = 1406 } ,
2011-06-20 16:48:19 +09:00
} ;
2013-02-19 15:07:34 +05:30
static const struct ntc_compensation ncpXXwl333 [ ] = {
2013-03-13 16:40:39 -07:00
{ . temp_c = - 40 , . ohm = 1610154 } ,
{ . temp_c = - 35 , . ohm = 1130850 } ,
{ . temp_c = - 30 , . ohm = 802609 } ,
{ . temp_c = - 25 , . ohm = 575385 } ,
{ . temp_c = - 20 , . ohm = 416464 } ,
{ . temp_c = - 15 , . ohm = 304219 } ,
{ . temp_c = - 10 , . ohm = 224193 } ,
{ . temp_c = - 5 , . ohm = 166623 } ,
{ . temp_c = 0 , . ohm = 124850 } ,
{ . temp_c = 5 , . ohm = 94287 } ,
{ . temp_c = 10 , . ohm = 71747 } ,
{ . temp_c = 15 , . ohm = 54996 } ,
{ . temp_c = 20 , . ohm = 42455 } ,
{ . temp_c = 25 , . ohm = 33000 } ,
{ . temp_c = 30 , . ohm = 25822 } ,
{ . temp_c = 35 , . ohm = 20335 } ,
{ . temp_c = 40 , . ohm = 16115 } ,
{ . temp_c = 45 , . ohm = 12849 } ,
{ . temp_c = 50 , . ohm = 10306 } ,
{ . temp_c = 55 , . ohm = 8314 } ,
{ . temp_c = 60 , . ohm = 6746 } ,
{ . temp_c = 65 , . ohm = 5503 } ,
{ . temp_c = 70 , . ohm = 4513 } ,
{ . temp_c = 75 , . ohm = 3721 } ,
{ . temp_c = 80 , . ohm = 3084 } ,
{ . temp_c = 85 , . ohm = 2569 } ,
{ . temp_c = 90 , . ohm = 2151 } ,
{ . temp_c = 95 , . ohm = 1809 } ,
{ . temp_c = 100 , . ohm = 1529 } ,
{ . temp_c = 105 , . ohm = 1299 } ,
{ . temp_c = 110 , . ohm = 1108 } ,
{ . temp_c = 115 , . ohm = 949 } ,
{ . temp_c = 120 , . ohm = 817 } ,
{ . temp_c = 125 , . ohm = 707 } ,
2011-06-20 16:48:19 +09:00
} ;
struct ntc_data {
struct device * hwmon_dev ;
struct ntc_thermistor_platform_data * pdata ;
const struct ntc_compensation * comp ;
struct device * dev ;
int n_comp ;
char name [ PLATFORM_NAME_SIZE ] ;
} ;
2013-03-13 09:38:20 +05:30
# ifdef CONFIG_OF
static int ntc_adc_iio_read ( struct ntc_thermistor_platform_data * pdata )
{
struct iio_channel * channel = pdata - > chan ;
unsigned int result ;
int val , ret ;
ret = iio_read_channel_raw ( channel , & val ) ;
if ( ret < 0 ) {
pr_err ( " read channel() error: %d \n " , ret ) ;
return ret ;
}
/* unit: mV */
2013-03-13 16:40:39 -07:00
result = pdata - > pullup_uv * val ;
2013-03-13 09:38:20 +05:30
result > > = 12 ;
return result ;
}
static const struct of_device_id ntc_match [ ] = {
{ . compatible = " ntc,ncp15wb473 " ,
. data = & ntc_thermistor_id [ TYPE_NCPXXWB473 ] } ,
{ . compatible = " ntc,ncp18wb473 " ,
. data = & ntc_thermistor_id [ TYPE_NCPXXWB473 ] } ,
{ . compatible = " ntc,ncp21wb473 " ,
. data = & ntc_thermistor_id [ TYPE_NCPXXWB473 ] } ,
{ . compatible = " ntc,ncp03wb473 " ,
. data = & ntc_thermistor_id [ TYPE_NCPXXWB473 ] } ,
{ . compatible = " ntc,ncp15wl333 " ,
. data = & ntc_thermistor_id [ TYPE_NCPXXWL333 ] } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , ntc_match ) ;
static struct ntc_thermistor_platform_data *
ntc_thermistor_parse_dt ( struct platform_device * pdev )
{
struct iio_channel * chan ;
struct device_node * np = pdev - > dev . of_node ;
struct ntc_thermistor_platform_data * pdata ;
if ( ! np )
return NULL ;
pdata = devm_kzalloc ( & pdev - > dev , sizeof ( * pdata ) , GFP_KERNEL ) ;
if ( ! pdata )
return ERR_PTR ( - ENOMEM ) ;
chan = iio_channel_get ( & pdev - > dev , NULL ) ;
if ( IS_ERR ( chan ) )
return ERR_CAST ( chan ) ;
2013-03-13 16:40:39 -07:00
if ( of_property_read_u32 ( np , " pullup-uv " , & pdata - > pullup_uv ) )
2013-03-13 09:38:20 +05:30
return ERR_PTR ( - ENODEV ) ;
if ( of_property_read_u32 ( np , " pullup-ohm " , & pdata - > pullup_ohm ) )
return ERR_PTR ( - ENODEV ) ;
if ( of_property_read_u32 ( np , " pulldown-ohm " , & pdata - > pulldown_ohm ) )
return ERR_PTR ( - ENODEV ) ;
if ( of_find_property ( np , " connected-positive " , NULL ) )
pdata - > connect = NTC_CONNECTED_POSITIVE ;
else /* status change should be possible if not always on. */
pdata - > connect = NTC_CONNECTED_GROUND ;
pdata - > chan = chan ;
2013-03-13 16:40:39 -07:00
pdata - > read_uv = ntc_adc_iio_read ;
2013-03-13 09:38:20 +05:30
return pdata ;
}
static void ntc_iio_channel_release ( struct ntc_thermistor_platform_data * pdata )
{
if ( pdata - > chan )
iio_channel_release ( pdata - > chan ) ;
}
# else
static struct ntc_thermistor_platform_data *
ntc_thermistor_parse_dt ( struct platform_device * pdev )
{
return NULL ;
}
static void ntc_iio_channel_release ( struct ntc_thermistor_platform_data * pdata )
{ }
# endif
2011-06-20 16:48:19 +09:00
static inline u64 div64_u64_safe ( u64 dividend , u64 divisor )
{
if ( divisor = = 0 & & dividend = = 0 )
return 0 ;
if ( divisor = = 0 )
return UINT_MAX ;
return div64_u64 ( dividend , divisor ) ;
}
2013-03-13 16:40:39 -07:00
static int get_ohm_of_thermistor ( struct ntc_data * data , unsigned int uv )
2011-06-20 16:48:19 +09:00
{
struct ntc_thermistor_platform_data * pdata = data - > pdata ;
2013-03-13 16:40:39 -07:00
u64 mv = uv / 1000 ;
u64 pmv = pdata - > pullup_uv / 1000 ;
u64 n , puo , pdo ;
puo = pdata - > pullup_ohm ;
pdo = pdata - > pulldown_ohm ;
2011-06-20 16:48:19 +09:00
2013-03-13 16:40:39 -07:00
if ( mv = = 0 ) {
2011-06-20 16:48:19 +09:00
if ( pdata - > connect = = NTC_CONNECTED_POSITIVE )
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
return INT_MAX ;
2011-06-20 16:48:19 +09:00
return 0 ;
}
2013-03-13 16:40:39 -07:00
if ( mv > = pmv )
2011-06-20 16:48:19 +09:00
return ( pdata - > connect = = NTC_CONNECTED_POSITIVE ) ?
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
0 : INT_MAX ;
2011-06-20 16:48:19 +09:00
2013-03-13 16:40:39 -07:00
if ( pdata - > connect = = NTC_CONNECTED_POSITIVE & & puo = = 0 )
n = div64_u64_safe ( pdo * ( pmv - mv ) , mv ) ;
else if ( pdata - > connect = = NTC_CONNECTED_GROUND & & pdo = = 0 )
n = div64_u64_safe ( puo * mv , pmv - mv ) ;
2011-06-20 16:48:19 +09:00
else if ( pdata - > connect = = NTC_CONNECTED_POSITIVE )
2013-03-13 16:40:39 -07:00
n = div64_u64_safe ( pdo * puo * ( pmv - mv ) ,
puo * mv - pdo * ( pmv - mv ) ) ;
2011-06-20 16:48:19 +09:00
else
2013-03-13 16:40:39 -07:00
n = div64_u64_safe ( pdo * puo * mv , pdo * ( pmv - mv ) - puo * mv ) ;
2011-06-20 16:48:19 +09:00
2013-03-13 16:40:39 -07:00
if ( n > INT_MAX )
n = INT_MAX ;
return n ;
2011-06-20 16:48:19 +09:00
}
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
static void lookup_comp ( struct ntc_data * data , unsigned int ohm ,
int * i_low , int * i_high )
2011-06-20 16:48:19 +09:00
{
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
int start , end , mid ;
/*
* Handle special cases : Resistance is higher than or equal to
* resistance in first table entry , or resistance is lower or equal
* to resistance in last table entry .
* In these cases , return i_low = = i_high , either pointing to the
* beginning or to the end of the table depending on the condition .
*/
if ( ohm > = data - > comp [ 0 ] . ohm ) {
* i_low = 0 ;
* i_high = 0 ;
return ;
}
if ( ohm < = data - > comp [ data - > n_comp - 1 ] . ohm ) {
* i_low = data - > n_comp - 1 ;
* i_high = data - > n_comp - 1 ;
return ;
}
2011-06-20 16:48:19 +09:00
/* Do a binary search on compensation table */
start = 0 ;
end = data - > n_comp ;
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
while ( start < end ) {
2011-06-20 16:48:19 +09:00
mid = start + ( end - start ) / 2 ;
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
/*
* start < = mid < end
* data - > comp [ start ] . ohm > ohm > = data - > comp [ end ] . ohm
*
* We could check for " ohm == data->comp[mid].ohm " here , but
* that is a quite unlikely condition , and we would have to
* check again after updating start . Check it at the end instead
* for simplicity .
*/
if ( ohm > = data - > comp [ mid ] . ohm ) {
2011-06-20 16:48:19 +09:00
end = mid ;
} else {
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
start = mid + 1 ;
/*
* ohm > = data - > comp [ start ] . ohm might be true here ,
* since we set start to mid + 1. In that case , we are
* done . We could keep going , but the condition is quite
* likely to occur , so it is worth checking for it .
*/
if ( ohm > = data - > comp [ start ] . ohm )
end = start ;
2011-06-20 16:48:19 +09:00
}
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
/*
* start < = end
* data - > comp [ start ] . ohm > = ohm > = data - > comp [ end ] . ohm
*/
2011-06-20 16:48:19 +09:00
}
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
/*
* start = = end
* ohm > = data - > comp [ end ] . ohm
*/
* i_low = end ;
if ( ohm = = data - > comp [ end ] . ohm )
* i_high = end ;
else
* i_high = end - 1 ;
2011-06-20 16:48:19 +09:00
}
2013-03-13 16:40:39 -07:00
static int get_temp_mc ( struct ntc_data * data , unsigned int ohm )
2011-06-20 16:48:19 +09:00
{
int low , high ;
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
int temp ;
2011-06-20 16:48:19 +09:00
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
lookup_comp ( data , ohm , & low , & high ) ;
if ( low = = high ) {
2011-06-20 16:48:19 +09:00
/* Unable to use linear approximation */
2013-03-13 16:40:39 -07:00
temp = data - > comp [ low ] . temp_c * 1000 ;
2011-06-20 16:48:19 +09:00
} else {
2013-03-13 16:40:39 -07:00
temp = data - > comp [ low ] . temp_c * 1000 +
( ( data - > comp [ high ] . temp_c - data - > comp [ low ] . temp_c ) *
2011-06-20 16:48:19 +09:00
1000 * ( ( int ) ohm - ( int ) data - > comp [ low ] . ohm ) ) /
( ( int ) data - > comp [ high ] . ohm - ( int ) data - > comp [ low ] . ohm ) ;
}
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
return temp ;
2011-06-20 16:48:19 +09:00
}
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
static int ntc_thermistor_get_ohm ( struct ntc_data * data )
2011-06-20 16:48:19 +09:00
{
2013-03-13 16:40:39 -07:00
int read_uv ;
2011-06-20 16:48:19 +09:00
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
if ( data - > pdata - > read_ohm )
return data - > pdata - > read_ohm ( ) ;
2011-06-20 16:48:19 +09:00
2013-03-13 16:40:39 -07:00
if ( data - > pdata - > read_uv ) {
read_uv = data - > pdata - > read_uv ( data - > pdata ) ;
if ( read_uv < 0 )
return read_uv ;
return get_ohm_of_thermistor ( data , read_uv ) ;
2011-06-20 16:48:19 +09:00
}
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
return - EINVAL ;
2011-06-20 16:48:19 +09:00
}
static ssize_t ntc_show_name ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct ntc_data * data = dev_get_drvdata ( dev ) ;
return sprintf ( buf , " %s \n " , data - > name ) ;
}
static ssize_t ntc_show_type ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
return sprintf ( buf , " 4 \n " ) ;
}
static ssize_t ntc_show_temp ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
struct ntc_data * data = dev_get_drvdata ( dev ) ;
hwmon: (ntc_thermistor) Optimize and fix build warning
The following build warning is seen in some configurations:
drivers/hwmon/ntc_thermistor.c: In function 'ntc_show_temp':
drivers/hwmon/ntc_thermistor.c:293: warning: 'temp' may be used uninitialized in this function
Fix the problem by re-arranging the code to overload return values with error
codes, and by avoiding error returns whenever possible.
Specifically,
Simplify lookup_comp() to not return an error. Instead, return i_low == i_high
if there is an exact match, or if the ohm value is outside the lookup table
range.
Modify get_temp_mC() to not return an error. Since it only returns an error
after lookup_comp() returned an error, this is quite straightforward after above
change.
Separate ntc_thermistor_read() into a function to read the resistor value (which
can return an error), and the call to get_temp_mC() which doesn't. Call the
functions directly from ntc_show_temp().
Code was tested using a test program, comparing the result of the old and new
versions of get_temp_mC() for resistor values between 0 and 2,000,000 ohm.
As a side effect, this patch reduces code size by approximately 400 bytes on
x86_64.
Signed-off-by: Guenter Roeck <guenter.roeck@ericsson.com>
Cc: Donggeun Kim <dg77.kim@samsung.com>
Reviewed-by: Robert Coulson <robert.coulson@ericsson.com>
2012-04-22 20:58:51 -07:00
int ohm ;
ohm = ntc_thermistor_get_ohm ( data ) ;
if ( ohm < 0 )
return ohm ;
2011-06-20 16:48:19 +09:00
2013-03-13 16:40:39 -07:00
return sprintf ( buf , " %d \n " , get_temp_mc ( data , ohm ) ) ;
2011-06-20 16:48:19 +09:00
}
static SENSOR_DEVICE_ATTR ( temp1_type , S_IRUGO , ntc_show_type , NULL , 0 ) ;
static SENSOR_DEVICE_ATTR ( temp1_input , S_IRUGO , ntc_show_temp , NULL , 0 ) ;
static DEVICE_ATTR ( name , S_IRUGO , ntc_show_name , NULL ) ;
static struct attribute * ntc_attributes [ ] = {
& dev_attr_name . attr ,
& sensor_dev_attr_temp1_type . dev_attr . attr ,
& sensor_dev_attr_temp1_input . dev_attr . attr ,
NULL ,
} ;
static const struct attribute_group ntc_attr_group = {
. attrs = ntc_attributes ,
} ;
2012-11-19 13:22:35 -05:00
static int ntc_thermistor_probe ( struct platform_device * pdev )
2011-06-20 16:48:19 +09:00
{
2013-03-13 09:38:20 +05:30
const struct of_device_id * of_id =
of_match_device ( of_match_ptr ( ntc_match ) , & pdev - > dev ) ;
const struct platform_device_id * pdev_id ;
struct ntc_thermistor_platform_data * pdata ;
2011-06-20 16:48:19 +09:00
struct ntc_data * data ;
2013-03-13 09:38:20 +05:30
int ret ;
pdata = ntc_thermistor_parse_dt ( pdev ) ;
if ( IS_ERR ( pdata ) )
return PTR_ERR ( pdata ) ;
else if ( pdata = = NULL )
2013-07-30 17:13:06 +09:00
pdata = dev_get_platdata ( & pdev - > dev ) ;
2011-06-20 16:48:19 +09:00
if ( ! pdata ) {
dev_err ( & pdev - > dev , " No platform init data supplied. \n " ) ;
return - ENODEV ;
}
/* Either one of the two is required. */
2013-03-13 16:40:39 -07:00
if ( ! pdata - > read_uv & & ! pdata - > read_ohm ) {
2012-04-23 10:33:45 -07:00
dev_err ( & pdev - > dev ,
2013-03-13 16:40:39 -07:00
" Both read_uv and read_ohm missing. Need either one of the two. \n " ) ;
2011-06-20 16:48:19 +09:00
return - EINVAL ;
}
2013-03-13 16:40:39 -07:00
if ( pdata - > read_uv & & pdata - > read_ohm ) {
2012-04-23 10:33:45 -07:00
dev_warn ( & pdev - > dev ,
2013-03-13 16:40:39 -07:00
" Only one of read_uv and read_ohm is needed; ignoring read_uv. \n " ) ;
pdata - > read_uv = NULL ;
2011-06-20 16:48:19 +09:00
}
2013-03-13 16:40:39 -07:00
if ( pdata - > read_uv & & ( pdata - > pullup_uv = = 0 | |
2011-06-20 16:48:19 +09:00
( pdata - > pullup_ohm = = 0 & & pdata - > connect = =
NTC_CONNECTED_GROUND ) | |
( pdata - > pulldown_ohm = = 0 & & pdata - > connect = =
NTC_CONNECTED_POSITIVE ) | |
( pdata - > connect ! = NTC_CONNECTED_POSITIVE & &
pdata - > connect ! = NTC_CONNECTED_GROUND ) ) ) {
2012-04-23 10:33:45 -07:00
dev_err ( & pdev - > dev ,
2013-03-13 16:40:39 -07:00
" Required data to use read_uv not supplied. \n " ) ;
2011-06-20 16:48:19 +09:00
return - EINVAL ;
}
2012-04-23 10:39:00 -07:00
data = devm_kzalloc ( & pdev - > dev , sizeof ( struct ntc_data ) , GFP_KERNEL ) ;
2011-06-20 16:48:19 +09:00
if ( ! data )
return - ENOMEM ;
2013-03-13 09:38:20 +05:30
pdev_id = of_id ? of_id - > data : platform_get_device_id ( pdev ) ;
2011-06-20 16:48:19 +09:00
data - > dev = & pdev - > dev ;
data - > pdata = pdata ;
2013-03-13 09:38:20 +05:30
strlcpy ( data - > name , pdev_id - > name , sizeof ( data - > name ) ) ;
2011-06-20 16:48:19 +09:00
2013-03-13 09:38:20 +05:30
switch ( pdev_id - > driver_data ) {
2011-06-20 16:48:19 +09:00
case TYPE_NCPXXWB473 :
data - > comp = ncpXXwb473 ;
data - > n_comp = ARRAY_SIZE ( ncpXXwb473 ) ;
break ;
case TYPE_NCPXXWL333 :
data - > comp = ncpXXwl333 ;
data - > n_comp = ARRAY_SIZE ( ncpXXwl333 ) ;
break ;
default :
dev_err ( & pdev - > dev , " Unknown device type: %lu(%s) \n " ,
2013-03-13 09:38:20 +05:30
pdev_id - > driver_data , pdev_id - > name ) ;
2012-04-23 10:39:00 -07:00
return - EINVAL ;
2011-06-20 16:48:19 +09:00
}
platform_set_drvdata ( pdev , data ) ;
ret = sysfs_create_group ( & data - > dev - > kobj , & ntc_attr_group ) ;
if ( ret ) {
dev_err ( data - > dev , " unable to create sysfs files \n " ) ;
2012-04-23 10:39:00 -07:00
return ret ;
2011-06-20 16:48:19 +09:00
}
data - > hwmon_dev = hwmon_device_register ( data - > dev ) ;
2012-04-23 10:41:03 -07:00
if ( IS_ERR ( data - > hwmon_dev ) ) {
2011-06-20 16:48:19 +09:00
dev_err ( data - > dev , " unable to register as hwmon device. \n " ) ;
2012-04-23 10:41:03 -07:00
ret = PTR_ERR ( data - > hwmon_dev ) ;
2011-06-20 16:48:19 +09:00
goto err_after_sysfs ;
}
2013-03-13 09:38:20 +05:30
dev_info ( & pdev - > dev , " Thermistor type: %s successfully probed. \n " ,
pdev - > name ) ;
2011-06-20 16:48:19 +09:00
return 0 ;
err_after_sysfs :
sysfs_remove_group ( & data - > dev - > kobj , & ntc_attr_group ) ;
2013-03-13 09:38:20 +05:30
ntc_iio_channel_release ( pdata ) ;
2011-06-20 16:48:19 +09:00
return ret ;
}
2012-11-19 13:25:51 -05:00
static int ntc_thermistor_remove ( struct platform_device * pdev )
2011-06-20 16:48:19 +09:00
{
struct ntc_data * data = platform_get_drvdata ( pdev ) ;
2013-03-13 09:38:20 +05:30
struct ntc_thermistor_platform_data * pdata = data - > pdata ;
2011-06-20 16:48:19 +09:00
hwmon_device_unregister ( data - > hwmon_dev ) ;
sysfs_remove_group ( & data - > dev - > kobj , & ntc_attr_group ) ;
2013-03-13 09:38:20 +05:30
ntc_iio_channel_release ( pdata ) ;
2011-06-20 16:48:19 +09:00
return 0 ;
}
static struct platform_driver ntc_thermistor_driver = {
. driver = {
. name = " ntc-thermistor " ,
. owner = THIS_MODULE ,
2013-03-13 09:38:20 +05:30
. of_match_table = of_match_ptr ( ntc_match ) ,
2011-06-20 16:48:19 +09:00
} ,
. probe = ntc_thermistor_probe ,
2012-11-19 13:21:20 -05:00
. remove = ntc_thermistor_remove ,
2011-06-20 16:48:19 +09:00
. id_table = ntc_thermistor_id ,
} ;
2011-11-25 02:31:00 -05:00
module_platform_driver ( ntc_thermistor_driver ) ;
2011-06-20 16:48:19 +09:00
MODULE_DESCRIPTION ( " NTC Thermistor Driver " ) ;
MODULE_AUTHOR ( " MyungJoo Ham <myungjoo.ham@samsung.com> " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS ( " platform:ntc-thermistor " ) ;