2005-07-15 21:38:08 -04:00
/*
2012-01-19 11:02:18 -08:00
* hwmon . c - part of lm_sensors , Linux kernel modules for hardware monitoring
*
* This file defines the sysfs class " hwmon " , for use by sensors drivers .
*
* Copyright ( C ) 2005 Mark M . Hoffman < mhoffman @ lightlink . 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 ; version 2 of the License .
*/
2005-07-15 21:38:08 -04:00
2010-10-20 06:51:37 +00:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2005-07-15 21:38:08 -04:00
# include <linux/module.h>
# include <linux/device.h>
# include <linux/err.h>
# include <linux/kdev_t.h>
# include <linux/idr.h>
# include <linux/hwmon.h>
2005-11-07 00:59:43 -08:00
# include <linux/gfp.h>
2006-03-05 23:13:47 +01:00
# include <linux/spinlock.h>
2009-06-15 18:39:50 +02:00
# include <linux/pci.h>
2005-07-15 21:38:08 -04:00
# define HWMON_ID_PREFIX "hwmon"
# define HWMON_ID_FORMAT HWMON_ID_PREFIX "%d"
static struct class * hwmon_class ;
2011-10-31 17:10:09 -07:00
static DEFINE_IDA ( hwmon_ida ) ;
2005-07-15 21:38:08 -04:00
/**
2007-08-20 13:46:20 -07:00
* hwmon_device_register - register w / hwmon
2005-07-15 21:38:08 -04:00
* @ dev : the device to register
*
2007-08-20 13:46:20 -07:00
* hwmon_device_unregister ( ) must be called when the device is no
2005-07-15 21:38:08 -04:00
* longer needed .
*
2007-08-20 13:46:20 -07:00
* Returns the pointer to the new device .
2005-07-15 21:38:08 -04:00
*/
2007-08-20 13:46:20 -07:00
struct device * hwmon_device_register ( struct device * dev )
2005-07-15 21:38:08 -04:00
{
2007-08-20 13:46:20 -07:00
struct device * hwdev ;
2011-10-31 17:10:09 -07:00
int id ;
2006-03-05 23:13:47 +01:00
2011-10-31 17:10:09 -07:00
id = ida_simple_get ( & hwmon_ida , 0 , 0 , GFP_KERNEL ) ;
if ( id < 0 )
return ERR_PTR ( id ) ;
2005-07-15 21:38:08 -04:00
2008-07-21 20:03:34 -07:00
hwdev = device_create ( hwmon_class , dev , MKDEV ( 0 , 0 ) , NULL ,
HWMON_ID_FORMAT , id ) ;
2005-07-15 21:38:08 -04:00
2011-10-31 17:10:09 -07:00
if ( IS_ERR ( hwdev ) )
ida_simple_remove ( & hwmon_ida , id ) ;
2005-07-15 21:38:08 -04:00
2007-08-20 13:46:20 -07:00
return hwdev ;
2005-07-15 21:38:08 -04:00
}
2012-01-08 19:34:14 +01:00
EXPORT_SYMBOL_GPL ( hwmon_device_register ) ;
2005-07-15 21:38:08 -04:00
/**
* hwmon_device_unregister - removes the previously registered class device
*
2007-08-20 13:46:20 -07:00
* @ dev : the class device to destroy
2005-07-15 21:38:08 -04:00
*/
2007-08-20 13:46:20 -07:00
void hwmon_device_unregister ( struct device * dev )
2005-07-15 21:38:08 -04:00
{
int id ;
2009-01-06 10:44:41 -08:00
if ( likely ( sscanf ( dev_name ( dev ) , HWMON_ID_FORMAT , & id ) = = 1 ) ) {
2007-08-20 13:46:20 -07:00
device_unregister ( dev ) ;
2011-10-31 17:10:09 -07:00
ida_simple_remove ( & hwmon_ida , id ) ;
2005-07-15 21:38:08 -04:00
} else
2007-08-20 13:46:20 -07:00
dev_dbg ( dev - > parent ,
2005-07-15 21:38:08 -04:00
" hwmon_device_unregister() failed: bad class ID! \n " ) ;
}
2012-01-08 19:34:14 +01:00
EXPORT_SYMBOL_GPL ( hwmon_device_unregister ) ;
2005-07-15 21:38:08 -04:00
2009-06-15 18:39:50 +02:00
static void __init hwmon_pci_quirks ( void )
{
# if defined CONFIG_X86 && defined CONFIG_PCI
struct pci_dev * sb ;
u16 base ;
u8 enable ;
/* Open access to 0x295-0x296 on MSI MS-7031 */
sb = pci_get_device ( PCI_VENDOR_ID_ATI , 0x436c , NULL ) ;
2012-12-19 22:16:59 +01:00
if ( sb ) {
if ( sb - > subsystem_vendor = = 0x1462 & & /* MSI */
sb - > subsystem_device = = 0x0031 ) { /* MS-7031 */
pci_read_config_byte ( sb , 0x48 , & enable ) ;
pci_read_config_word ( sb , 0x64 , & base ) ;
if ( base = = 0 & & ! ( enable & BIT ( 2 ) ) ) {
dev_info ( & sb - > dev ,
" Opening wide generic port at 0x295 \n " ) ;
pci_write_config_word ( sb , 0x64 , 0x295 ) ;
pci_write_config_byte ( sb , 0x48 ,
enable | BIT ( 2 ) ) ;
}
2009-06-15 18:39:50 +02:00
}
2012-12-19 22:16:59 +01:00
pci_dev_put ( sb ) ;
2009-06-15 18:39:50 +02:00
}
# endif
}
2005-07-15 21:38:08 -04:00
static int __init hwmon_init ( void )
{
2009-06-15 18:39:50 +02:00
hwmon_pci_quirks ( ) ;
2005-07-15 21:38:08 -04:00
hwmon_class = class_create ( THIS_MODULE , " hwmon " ) ;
if ( IS_ERR ( hwmon_class ) ) {
2010-10-20 06:51:37 +00:00
pr_err ( " couldn't create sysfs class \n " ) ;
2005-07-15 21:38:08 -04:00
return PTR_ERR ( hwmon_class ) ;
}
return 0 ;
}
static void __exit hwmon_exit ( void )
{
class_destroy ( hwmon_class ) ;
}
2007-02-14 21:15:04 +01:00
subsys_initcall ( hwmon_init ) ;
2005-07-15 21:38:08 -04:00
module_exit ( hwmon_exit ) ;
MODULE_AUTHOR ( " Mark M. Hoffman <mhoffman@lightlink.com> " ) ;
MODULE_DESCRIPTION ( " hardware monitoring sysfs/class support " ) ;
MODULE_LICENSE ( " GPL " ) ;