diff options
Diffstat (limited to 'drivers/hwmon/lm78.c')
| -rw-r--r-- | drivers/hwmon/lm78.c | 305 | 
1 files changed, 177 insertions, 128 deletions
| diff --git a/drivers/hwmon/lm78.c b/drivers/hwmon/lm78.c index 4cb24eafe318..6df0b4681710 100644 --- a/drivers/hwmon/lm78.c +++ b/drivers/hwmon/lm78.c @@ -2,7 +2,7 @@      lm78.c - Part of lm_sensors, Linux kernel modules for hardware               monitoring      Copyright (c) 1998, 1999  Frodo Looijaard <frodol@dds.nl>  -    Copyright (c) 2007        Jean Delvare <khali@linux-fr.org> +    Copyright (c) 2007, 2011  Jean Delvare <khali@linux-fr.org>      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 @@ -26,23 +26,21 @@  #include <linux/slab.h>  #include <linux/jiffies.h>  #include <linux/i2c.h> -#include <linux/platform_device.h> -#include <linux/ioport.h>  #include <linux/hwmon.h>  #include <linux/hwmon-vid.h>  #include <linux/hwmon-sysfs.h>  #include <linux/err.h>  #include <linux/mutex.h> -#include <linux/io.h> -/* ISA device, if found */ -static struct platform_device *pdev; +#ifdef CONFIG_ISA +#include <linux/platform_device.h> +#include <linux/ioport.h> +#include <linux/io.h> +#endif  /* Addresses to scan */  static const unsigned short normal_i2c[] = { 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d,  						0x2e, 0x2f, I2C_CLIENT_END }; -static unsigned short isa_address = 0x290; -  enum chips { lm78, lm79 };  /* Many LM78 constants specified below */ @@ -143,50 +141,12 @@ struct lm78_data {  }; -static int lm78_i2c_detect(struct i2c_client *client, -			   struct i2c_board_info *info); -static int lm78_i2c_probe(struct i2c_client *client, -			  const struct i2c_device_id *id); -static int lm78_i2c_remove(struct i2c_client *client); - -static int __devinit lm78_isa_probe(struct platform_device *pdev); -static int __devexit lm78_isa_remove(struct platform_device *pdev); -  static int lm78_read_value(struct lm78_data *data, u8 reg);  static int lm78_write_value(struct lm78_data *data, u8 reg, u8 value);  static struct lm78_data *lm78_update_device(struct device *dev);  static void lm78_init_device(struct lm78_data *data); -static const struct i2c_device_id lm78_i2c_id[] = { -	{ "lm78", lm78 }, -	{ "lm79", lm79 }, -	{ } -}; -MODULE_DEVICE_TABLE(i2c, lm78_i2c_id); - -static struct i2c_driver lm78_driver = { -	.class		= I2C_CLASS_HWMON, -	.driver = { -		.name	= "lm78", -	}, -	.probe		= lm78_i2c_probe, -	.remove		= lm78_i2c_remove, -	.id_table	= lm78_i2c_id, -	.detect		= lm78_i2c_detect, -	.address_list	= normal_i2c, -}; - -static struct platform_driver lm78_isa_driver = { -	.driver = { -		.owner	= THIS_MODULE, -		.name	= "lm78", -	}, -	.probe		= lm78_isa_probe, -	.remove		= __devexit_p(lm78_isa_remove), -}; - -  /* 7 Voltages */  static ssize_t show_in(struct device *dev, struct device_attribute *da,  		       char *buf) @@ -514,6 +474,16 @@ static const struct attribute_group lm78_group = {  	.attrs = lm78_attributes,  }; +/* + * ISA related code + */ +#ifdef CONFIG_ISA + +/* ISA device, if found */ +static struct platform_device *pdev; + +static unsigned short isa_address = 0x290; +  /* I2C devices get this name attribute automatically, but for ISA devices     we must create it by ourselves. */  static ssize_t show_name(struct device *dev, struct device_attribute @@ -525,6 +495,11 @@ static ssize_t show_name(struct device *dev, struct device_attribute  }  static DEVICE_ATTR(name, S_IRUGO, show_name, NULL); +static struct lm78_data *lm78_data_if_isa(void) +{ +	return pdev ? platform_get_drvdata(pdev) : NULL; +} +  /* Returns 1 if the I2C chip appears to be an alias of the ISA chip */  static int lm78_alias_detect(struct i2c_client *client, u8 chipid)  { @@ -558,12 +533,24 @@ static int lm78_alias_detect(struct i2c_client *client, u8 chipid)  	return 1;  } +#else /* !CONFIG_ISA */ + +static int lm78_alias_detect(struct i2c_client *client, u8 chipid) +{ +	return 0; +} + +static struct lm78_data *lm78_data_if_isa(void) +{ +	return NULL; +} +#endif /* CONFIG_ISA */  static int lm78_i2c_detect(struct i2c_client *client,  			   struct i2c_board_info *info)  {  	int i; -	struct lm78_data *isa = pdev ? platform_get_drvdata(pdev) : NULL; +	struct lm78_data *isa = lm78_data_if_isa();  	const char *client_name;  	struct i2c_adapter *adapter = client->adapter;  	int address = client->addr; @@ -663,76 +650,24 @@ static int lm78_i2c_remove(struct i2c_client *client)  	return 0;  } -static int __devinit lm78_isa_probe(struct platform_device *pdev) -{ -	int err; -	struct lm78_data *data; -	struct resource *res; - -	/* Reserve the ISA region */ -	res = platform_get_resource(pdev, IORESOURCE_IO, 0); -	if (!request_region(res->start + LM78_ADDR_REG_OFFSET, 2, "lm78")) { -		err = -EBUSY; -		goto exit; -	} - -	if (!(data = kzalloc(sizeof(struct lm78_data), GFP_KERNEL))) { -		err = -ENOMEM; -		goto exit_release_region; -	} -	mutex_init(&data->lock); -	data->isa_addr = res->start; -	platform_set_drvdata(pdev, data); - -	if (lm78_read_value(data, LM78_REG_CHIPID) & 0x80) { -		data->type = lm79; -		data->name = "lm79"; -	} else { -		data->type = lm78; -		data->name = "lm78"; -	} - -	/* Initialize the LM78 chip */ -	lm78_init_device(data); - -	/* Register sysfs hooks */ -	if ((err = sysfs_create_group(&pdev->dev.kobj, &lm78_group)) -	 || (err = device_create_file(&pdev->dev, &dev_attr_name))) -		goto exit_remove_files; - -	data->hwmon_dev = hwmon_device_register(&pdev->dev); -	if (IS_ERR(data->hwmon_dev)) { -		err = PTR_ERR(data->hwmon_dev); -		goto exit_remove_files; -	} - -	return 0; - - exit_remove_files: -	sysfs_remove_group(&pdev->dev.kobj, &lm78_group); -	device_remove_file(&pdev->dev, &dev_attr_name); -	kfree(data); - exit_release_region: -	release_region(res->start + LM78_ADDR_REG_OFFSET, 2); - exit: -	return err; -} - -static int __devexit lm78_isa_remove(struct platform_device *pdev) -{ -	struct lm78_data *data = platform_get_drvdata(pdev); -	struct resource *res; - -	hwmon_device_unregister(data->hwmon_dev); -	sysfs_remove_group(&pdev->dev.kobj, &lm78_group); -	device_remove_file(&pdev->dev, &dev_attr_name); -	kfree(data); - -	res = platform_get_resource(pdev, IORESOURCE_IO, 0); -	release_region(res->start + LM78_ADDR_REG_OFFSET, 2); +static const struct i2c_device_id lm78_i2c_id[] = { +	{ "lm78", lm78 }, +	{ "lm79", lm79 }, +	{ } +}; +MODULE_DEVICE_TABLE(i2c, lm78_i2c_id); -	return 0; -} +static struct i2c_driver lm78_driver = { +	.class		= I2C_CLASS_HWMON, +	.driver = { +		.name	= "lm78", +	}, +	.probe		= lm78_i2c_probe, +	.remove		= lm78_i2c_remove, +	.id_table	= lm78_i2c_id, +	.detect		= lm78_i2c_detect, +	.address_list	= normal_i2c, +};  /* The SMBus locks itself, but ISA access must be locked explicitly!      We don't want to lock the whole ISA bus, so we lock each client @@ -743,6 +678,7 @@ static int lm78_read_value(struct lm78_data *data, u8 reg)  {  	struct i2c_client *client = data->client; +#ifdef CONFIG_ISA  	if (!client) { /* ISA device */  		int res;  		mutex_lock(&data->lock); @@ -751,6 +687,7 @@ static int lm78_read_value(struct lm78_data *data, u8 reg)  		mutex_unlock(&data->lock);  		return res;  	} else +#endif  		return i2c_smbus_read_byte_data(client, reg);  } @@ -765,6 +702,7 @@ static int lm78_write_value(struct lm78_data *data, u8 reg, u8 value)  {  	struct i2c_client *client = data->client; +#ifdef CONFIG_ISA  	if (!client) { /* ISA device */  		mutex_lock(&data->lock);  		outb_p(reg, data->isa_addr + LM78_ADDR_REG_OFFSET); @@ -772,6 +710,7 @@ static int lm78_write_value(struct lm78_data *data, u8 reg, u8 value)  		mutex_unlock(&data->lock);  		return 0;  	} else +#endif  		return i2c_smbus_write_byte_data(client, reg, value);  } @@ -849,6 +788,88 @@ static struct lm78_data *lm78_update_device(struct device *dev)  	return data;  } +#ifdef CONFIG_ISA +static int __devinit lm78_isa_probe(struct platform_device *pdev) +{ +	int err; +	struct lm78_data *data; +	struct resource *res; + +	/* Reserve the ISA region */ +	res = platform_get_resource(pdev, IORESOURCE_IO, 0); +	if (!request_region(res->start + LM78_ADDR_REG_OFFSET, 2, "lm78")) { +		err = -EBUSY; +		goto exit; +	} + +	data = kzalloc(sizeof(struct lm78_data), GFP_KERNEL); +	if (!data) { +		err = -ENOMEM; +		goto exit_release_region; +	} +	mutex_init(&data->lock); +	data->isa_addr = res->start; +	platform_set_drvdata(pdev, data); + +	if (lm78_read_value(data, LM78_REG_CHIPID) & 0x80) { +		data->type = lm79; +		data->name = "lm79"; +	} else { +		data->type = lm78; +		data->name = "lm78"; +	} + +	/* Initialize the LM78 chip */ +	lm78_init_device(data); + +	/* Register sysfs hooks */ +	if ((err = sysfs_create_group(&pdev->dev.kobj, &lm78_group)) +	 || (err = device_create_file(&pdev->dev, &dev_attr_name))) +		goto exit_remove_files; + +	data->hwmon_dev = hwmon_device_register(&pdev->dev); +	if (IS_ERR(data->hwmon_dev)) { +		err = PTR_ERR(data->hwmon_dev); +		goto exit_remove_files; +	} + +	return 0; + + exit_remove_files: +	sysfs_remove_group(&pdev->dev.kobj, &lm78_group); +	device_remove_file(&pdev->dev, &dev_attr_name); +	kfree(data); + exit_release_region: +	release_region(res->start + LM78_ADDR_REG_OFFSET, 2); + exit: +	return err; +} + +static int __devexit lm78_isa_remove(struct platform_device *pdev) +{ +	struct lm78_data *data = platform_get_drvdata(pdev); +	struct resource *res; + +	hwmon_device_unregister(data->hwmon_dev); +	sysfs_remove_group(&pdev->dev.kobj, &lm78_group); +	device_remove_file(&pdev->dev, &dev_attr_name); +	kfree(data); + +	res = platform_get_resource(pdev, IORESOURCE_IO, 0); +	release_region(res->start + LM78_ADDR_REG_OFFSET, 2); + +	return 0; +} + +static struct platform_driver lm78_isa_driver = { +	.driver = { +		.owner	= THIS_MODULE, +		.name	= "lm78", +	}, +	.probe		= lm78_isa_probe, +	.remove		= __devexit_p(lm78_isa_remove), +}; +  /* return 1 if a supported chip is found, 0 otherwise */  static int __init lm78_isa_found(unsigned short address)  { @@ -969,12 +990,10 @@ static int __init lm78_isa_device_add(unsigned short address)  	return err;  } -static int __init sm_lm78_init(void) +static int __init lm78_isa_register(void)  {  	int res; -	/* We register the ISA device first, so that we can skip the -	 * registration of an I2C interface to the same device. */  	if (lm78_isa_found(isa_address)) {  		res = platform_driver_register(&lm78_isa_driver);  		if (res) @@ -986,32 +1005,62 @@ static int __init sm_lm78_init(void)  			goto exit_unreg_isa_driver;  	} -	res = i2c_add_driver(&lm78_driver); -	if (res) -		goto exit_unreg_isa_device; -  	return 0; - exit_unreg_isa_device: -	platform_device_unregister(pdev);   exit_unreg_isa_driver:  	platform_driver_unregister(&lm78_isa_driver);   exit:  	return res;  } -static void __exit sm_lm78_exit(void) +static void lm78_isa_unregister(void)  {  	if (pdev) {  		platform_device_unregister(pdev);  		platform_driver_unregister(&lm78_isa_driver);  	} -	i2c_del_driver(&lm78_driver);  } +#else /* !CONFIG_ISA */ +static int __init lm78_isa_register(void) +{ +	return 0; +} + +static void lm78_isa_unregister(void) +{ +} +#endif /* CONFIG_ISA */ +static int __init sm_lm78_init(void) +{ +	int res; + +	/* We register the ISA device first, so that we can skip the +	 * registration of an I2C interface to the same device. */ +	res = lm78_isa_register(); +	if (res) +		goto exit; + +	res = i2c_add_driver(&lm78_driver); +	if (res) +		goto exit_unreg_isa_device; + +	return 0; + + exit_unreg_isa_device: +	lm78_isa_unregister(); + exit: +	return res; +} + +static void __exit sm_lm78_exit(void) +{ +	lm78_isa_unregister(); +	i2c_del_driver(&lm78_driver); +} -MODULE_AUTHOR("Frodo Looijaard <frodol@dds.nl>"); +MODULE_AUTHOR("Frodo Looijaard, Jean Delvare <khali@linux-fr.org>");  MODULE_DESCRIPTION("LM78/LM79 driver");  MODULE_LICENSE("GPL"); | 
