Commit 0365ba7f authored by Benjamin Herrenschmidt's avatar Benjamin Herrenschmidt Committed by Linus Torvalds

[PATCH] ppc64: SMU driver update & i2c support

The SMU is the "system controller" chip used by Apple recent G5 machines
including the iMac G5.  It drives things like fans, i2c busses, real time
clock, etc...

The current kernel contains a very crude driver that doesn't do much more
than reading the real time clock synchronously.  This is a completely
rewritten driver that provides interrupt based command queuing, a userland
interface, and an i2c/smbus driver for accessing the devices hanging off
the SMU i2c busses like temperature sensors.  This driver is a basic block
for upcoming work on thermal control for those machines, among others.
Signed-off-by: default avatarBenjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: Jean Delvare <khali@linux-fr.org>
Cc: Greg KH <greg@kroah.com>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@osdl.org>
parent 0f329075
...@@ -719,7 +719,8 @@ pmac_declare_of_platform_devices(void) ...@@ -719,7 +719,8 @@ pmac_declare_of_platform_devices(void)
if (np) { if (np) {
for (np = np->child; np != NULL; np = np->sibling) for (np = np->child; np != NULL; np = np->sibling)
if (strncmp(np->name, "i2c", 3) == 0) { if (strncmp(np->name, "i2c", 3) == 0) {
of_platform_device_create(np, "uni-n-i2c"); of_platform_device_create(np, "uni-n-i2c",
NULL);
break; break;
} }
} }
...@@ -727,17 +728,18 @@ pmac_declare_of_platform_devices(void) ...@@ -727,17 +728,18 @@ pmac_declare_of_platform_devices(void)
if (np) { if (np) {
for (np = np->child; np != NULL; np = np->sibling) for (np = np->child; np != NULL; np = np->sibling)
if (strncmp(np->name, "i2c", 3) == 0) { if (strncmp(np->name, "i2c", 3) == 0) {
of_platform_device_create(np, "u3-i2c"); of_platform_device_create(np, "u3-i2c",
NULL);
break; break;
} }
} }
np = find_devices("valkyrie"); np = find_devices("valkyrie");
if (np) if (np)
of_platform_device_create(np, "valkyrie"); of_platform_device_create(np, "valkyrie", NULL);
np = find_devices("platinum"); np = find_devices("platinum");
if (np) if (np)
of_platform_device_create(np, "platinum"); of_platform_device_create(np, "platinum", NULL);
return 0; return 0;
} }
......
...@@ -234,7 +234,9 @@ void of_device_unregister(struct of_device *ofdev) ...@@ -234,7 +234,9 @@ void of_device_unregister(struct of_device *ofdev)
device_unregister(&ofdev->dev); device_unregister(&ofdev->dev);
} }
struct of_device* of_platform_device_create(struct device_node *np, const char *bus_id) struct of_device* of_platform_device_create(struct device_node *np,
const char *bus_id,
struct device *parent)
{ {
struct of_device *dev; struct of_device *dev;
u32 *reg; u32 *reg;
...@@ -247,7 +249,7 @@ struct of_device* of_platform_device_create(struct device_node *np, const char * ...@@ -247,7 +249,7 @@ struct of_device* of_platform_device_create(struct device_node *np, const char *
dev->node = of_node_get(np); dev->node = of_node_get(np);
dev->dma_mask = 0xffffffffUL; dev->dma_mask = 0xffffffffUL;
dev->dev.dma_mask = &dev->dma_mask; dev->dev.dma_mask = &dev->dma_mask;
dev->dev.parent = NULL; dev->dev.parent = parent;
dev->dev.bus = &of_platform_bus_type; dev->dev.bus = &of_platform_bus_type;
dev->dev.release = of_release_dev; dev->dev.release = of_release_dev;
......
...@@ -233,7 +233,9 @@ void of_device_unregister(struct of_device *ofdev) ...@@ -233,7 +233,9 @@ void of_device_unregister(struct of_device *ofdev)
device_unregister(&ofdev->dev); device_unregister(&ofdev->dev);
} }
struct of_device* of_platform_device_create(struct device_node *np, const char *bus_id) struct of_device* of_platform_device_create(struct device_node *np,
const char *bus_id,
struct device *parent)
{ {
struct of_device *dev; struct of_device *dev;
...@@ -245,7 +247,7 @@ struct of_device* of_platform_device_create(struct device_node *np, const char * ...@@ -245,7 +247,7 @@ struct of_device* of_platform_device_create(struct device_node *np, const char *
dev->node = np; dev->node = np;
dev->dma_mask = 0xffffffffUL; dev->dma_mask = 0xffffffffUL;
dev->dev.dma_mask = &dev->dma_mask; dev->dev.dma_mask = &dev->dma_mask;
dev->dev.parent = NULL; dev->dev.parent = parent;
dev->dev.bus = &of_platform_bus_type; dev->dev.bus = &of_platform_bus_type;
dev->dev.release = of_release_dev; dev->dev.release = of_release_dev;
...@@ -259,6 +261,7 @@ struct of_device* of_platform_device_create(struct device_node *np, const char * ...@@ -259,6 +261,7 @@ struct of_device* of_platform_device_create(struct device_node *np, const char *
return dev; return dev;
} }
EXPORT_SYMBOL(of_match_device); EXPORT_SYMBOL(of_match_device);
EXPORT_SYMBOL(of_platform_bus_type); EXPORT_SYMBOL(of_platform_bus_type);
EXPORT_SYMBOL(of_register_driver); EXPORT_SYMBOL(of_register_driver);
......
...@@ -434,16 +434,24 @@ static int pmac_check_legacy_ioport(unsigned int baseport) ...@@ -434,16 +434,24 @@ static int pmac_check_legacy_ioport(unsigned int baseport)
static int __init pmac_declare_of_platform_devices(void) static int __init pmac_declare_of_platform_devices(void)
{ {
struct device_node *np; struct device_node *np, *npp;
np = find_devices("u3"); npp = of_find_node_by_name(NULL, "u3");
if (np) { if (npp) {
for (np = np->child; np != NULL; np = np->sibling) for (np = NULL; (np = of_get_next_child(npp, np)) != NULL;) {
if (strncmp(np->name, "i2c", 3) == 0) { if (strncmp(np->name, "i2c", 3) == 0) {
of_platform_device_create(np, "u3-i2c"); of_platform_device_create(np, "u3-i2c", NULL);
of_node_put(np);
break; break;
} }
} }
of_node_put(npp);
}
npp = of_find_node_by_type(NULL, "smu");
if (npp) {
of_platform_device_create(npp, "smu", NULL);
of_node_put(npp);
}
return 0; return 0;
} }
......
...@@ -84,7 +84,7 @@ void __pmac pmac_get_rtc_time(struct rtc_time *tm) ...@@ -84,7 +84,7 @@ void __pmac pmac_get_rtc_time(struct rtc_time *tm)
#ifdef CONFIG_PMAC_SMU #ifdef CONFIG_PMAC_SMU
case SYS_CTRLER_SMU: case SYS_CTRLER_SMU:
smu_get_rtc_time(tm); smu_get_rtc_time(tm, 1);
break; break;
#endif /* CONFIG_PMAC_SMU */ #endif /* CONFIG_PMAC_SMU */
default: default:
...@@ -128,7 +128,7 @@ int __pmac pmac_set_rtc_time(struct rtc_time *tm) ...@@ -128,7 +128,7 @@ int __pmac pmac_set_rtc_time(struct rtc_time *tm)
#ifdef CONFIG_PMAC_SMU #ifdef CONFIG_PMAC_SMU
case SYS_CTRLER_SMU: case SYS_CTRLER_SMU:
return smu_set_rtc_time(tm); return smu_set_rtc_time(tm, 1);
#endif /* CONFIG_PMAC_SMU */ #endif /* CONFIG_PMAC_SMU */
default: default:
return -ENODEV; return -ENODEV;
......
...@@ -245,6 +245,18 @@ config I2C_KEYWEST ...@@ -245,6 +245,18 @@ config I2C_KEYWEST
This support is also available as a module. If so, the module This support is also available as a module. If so, the module
will be called i2c-keywest. will be called i2c-keywest.
config I2C_PMAC_SMU
tristate "Powermac SMU I2C interface"
depends on I2C && PMAC_SMU
help
This supports the use of the I2C interface in the SMU
chip on recent Apple machines like the iMac G5. It is used
among others by the thermal control driver for those machines.
Say Y if you have such a machine.
This support is also available as a module. If so, the module
will be called i2c-pmac-smu.
config I2C_MPC config I2C_MPC
tristate "MPC107/824x/85xx/52xx" tristate "MPC107/824x/85xx/52xx"
depends on I2C && PPC32 depends on I2C && PPC32
......
...@@ -20,6 +20,7 @@ obj-$(CONFIG_I2C_ITE) += i2c-ite.o ...@@ -20,6 +20,7 @@ obj-$(CONFIG_I2C_ITE) += i2c-ite.o
obj-$(CONFIG_I2C_IXP2000) += i2c-ixp2000.o obj-$(CONFIG_I2C_IXP2000) += i2c-ixp2000.o
obj-$(CONFIG_I2C_IXP4XX) += i2c-ixp4xx.o obj-$(CONFIG_I2C_IXP4XX) += i2c-ixp4xx.o
obj-$(CONFIG_I2C_KEYWEST) += i2c-keywest.o obj-$(CONFIG_I2C_KEYWEST) += i2c-keywest.o
obj-$(CONFIG_I2C_PMAC_SMU) += i2c-pmac-smu.o
obj-$(CONFIG_I2C_MPC) += i2c-mpc.o obj-$(CONFIG_I2C_MPC) += i2c-mpc.o
obj-$(CONFIG_I2C_MV64XXX) += i2c-mv64xxx.o obj-$(CONFIG_I2C_MV64XXX) += i2c-mv64xxx.o
obj-$(CONFIG_I2C_NFORCE2) += i2c-nforce2.o obj-$(CONFIG_I2C_NFORCE2) += i2c-nforce2.o
......
/*
i2c Support for Apple SMU Controller
Copyright (c) 2005 Benjamin Herrenschmidt, IBM Corp.
<benh@kernel.crashing.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
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., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include <linux/config.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/types.h>
#include <linux/i2c.h>
#include <linux/init.h>
#include <linux/completion.h>
#include <linux/device.h>
#include <asm/prom.h>
#include <asm/of_device.h>
#include <asm/smu.h>
static int probe;
MODULE_AUTHOR("Benjamin Herrenschmidt <benh@kernel.crashing.org>");
MODULE_DESCRIPTION("I2C driver for Apple's SMU");
MODULE_LICENSE("GPL");
module_param(probe, bool, 0);
/* Physical interface */
struct smu_iface
{
struct i2c_adapter adapter;
struct completion complete;
u32 busid;
};
static void smu_i2c_done(struct smu_i2c_cmd *cmd, void *misc)
{
struct smu_iface *iface = misc;
complete(&iface->complete);
}
/*
* SMBUS-type transfer entrypoint
*/
static s32 smu_smbus_xfer( struct i2c_adapter* adap,
u16 addr,
unsigned short flags,
char read_write,
u8 command,
int size,
union i2c_smbus_data* data)
{
struct smu_iface *iface = i2c_get_adapdata(adap);
struct smu_i2c_cmd cmd;
int rc = 0;
int read = (read_write == I2C_SMBUS_READ);
cmd.info.bus = iface->busid;
cmd.info.devaddr = (addr << 1) | (read ? 0x01 : 0x00);
/* Prepare datas & select mode */
switch (size) {
case I2C_SMBUS_QUICK:
cmd.info.type = SMU_I2C_TRANSFER_SIMPLE;
cmd.info.datalen = 0;
break;
case I2C_SMBUS_BYTE:
cmd.info.type = SMU_I2C_TRANSFER_SIMPLE;
cmd.info.datalen = 1;
if (!read)
cmd.info.data[0] = data->byte;
break;
case I2C_SMBUS_BYTE_DATA:
cmd.info.type = SMU_I2C_TRANSFER_STDSUB;
cmd.info.datalen = 1;
cmd.info.sublen = 1;
cmd.info.subaddr[0] = command;
cmd.info.subaddr[1] = 0;
cmd.info.subaddr[2] = 0;
if (!read)
cmd.info.data[0] = data->byte;
break;
case I2C_SMBUS_WORD_DATA:
cmd.info.type = SMU_I2C_TRANSFER_STDSUB;
cmd.info.datalen = 2;
cmd.info.sublen = 1;
cmd.info.subaddr[0] = command;
cmd.info.subaddr[1] = 0;
cmd.info.subaddr[2] = 0;
if (!read) {
cmd.info.data[0] = data->byte & 0xff;
cmd.info.data[1] = (data->byte >> 8) & 0xff;
}
break;
/* Note that these are broken vs. the expected smbus API where
* on reads, the lenght is actually returned from the function,
* but I think the current API makes no sense and I don't want
* any driver that I haven't verified for correctness to go
* anywhere near a pmac i2c bus anyway ...
*/
case I2C_SMBUS_BLOCK_DATA:
cmd.info.type = SMU_I2C_TRANSFER_STDSUB;
cmd.info.datalen = data->block[0] + 1;
if (cmd.info.datalen > 6)
return -EINVAL;
if (!read)
memcpy(cmd.info.data, data->block, cmd.info.datalen);
cmd.info.sublen = 1;
cmd.info.subaddr[0] = command;
cmd.info.subaddr[1] = 0;
cmd.info.subaddr[2] = 0;
break;
case I2C_SMBUS_I2C_BLOCK_DATA:
cmd.info.type = SMU_I2C_TRANSFER_STDSUB;
cmd.info.datalen = data->block[0];
if (cmd.info.datalen > 7)
return -EINVAL;
if (!read)
memcpy(cmd.info.data, &data->block[1],
cmd.info.datalen);
cmd.info.sublen = 1;
cmd.info.subaddr[0] = command;
cmd.info.subaddr[1] = 0;
cmd.info.subaddr[2] = 0;
break;
default:
return -EINVAL;
}
/* Turn a standardsub read into a combined mode access */
if (read_write == I2C_SMBUS_READ &&
cmd.info.type == SMU_I2C_TRANSFER_STDSUB)
cmd.info.type = SMU_I2C_TRANSFER_COMBINED;
/* Finish filling command and submit it */
cmd.done = smu_i2c_done;
cmd.misc = iface;
rc = smu_queue_i2c(&cmd);
if (rc < 0)
return rc;
wait_for_completion(&iface->complete);
rc = cmd.status;
if (!read || rc < 0)
return rc;
switch (size) {
case I2C_SMBUS_BYTE:
case I2C_SMBUS_BYTE_DATA:
data->byte = cmd.info.data[0];
break;
case I2C_SMBUS_WORD_DATA:
data->word = ((u16)cmd.info.data[1]) << 8;
data->word |= cmd.info.data[0];
break;
/* Note that these are broken vs. the expected smbus API where
* on reads, the lenght is actually returned from the function,
* but I think the current API makes no sense and I don't want
* any driver that I haven't verified for correctness to go
* anywhere near a pmac i2c bus anyway ...
*/
case I2C_SMBUS_BLOCK_DATA:
case I2C_SMBUS_I2C_BLOCK_DATA:
memcpy(&data->block[0], cmd.info.data, cmd.info.datalen);
break;
}
return rc;
}
static u32
smu_smbus_func(struct i2c_adapter * adapter)
{
return I2C_FUNC_SMBUS_QUICK | I2C_FUNC_SMBUS_BYTE |
I2C_FUNC_SMBUS_BYTE_DATA | I2C_FUNC_SMBUS_WORD_DATA |
I2C_FUNC_SMBUS_BLOCK_DATA;
}
/* For now, we only handle combined mode (smbus) */
static struct i2c_algorithm smu_algorithm = {
.smbus_xfer = smu_smbus_xfer,
.functionality = smu_smbus_func,
};
static int create_iface(struct device_node *np, struct device *dev)
{
struct smu_iface* iface;
u32 *reg, busid;
int rc;
reg = (u32 *)get_property(np, "reg", NULL);
if (reg == NULL) {
printk(KERN_ERR "i2c-pmac-smu: can't find bus number !\n");
return -ENXIO;
}
busid = *reg;
iface = kmalloc(sizeof(struct smu_iface), GFP_KERNEL);
if (iface == NULL) {
printk(KERN_ERR "i2c-pmac-smu: can't allocate inteface !\n");
return -ENOMEM;
}
memset(iface, 0, sizeof(struct smu_iface));
init_completion(&iface->complete);
iface->busid = busid;
dev_set_drvdata(dev, iface);
sprintf(iface->adapter.name, "smu-i2c-%02x", busid);
iface->adapter.algo = &smu_algorithm;
iface->adapter.algo_data = NULL;
iface->adapter.client_register = NULL;
iface->adapter.client_unregister = NULL;
i2c_set_adapdata(&iface->adapter, iface);
iface->adapter.dev.parent = dev;
rc = i2c_add_adapter(&iface->adapter);
if (rc) {
printk(KERN_ERR "i2c-pamc-smu.c: Adapter %s registration "
"failed\n", iface->adapter.name);
i2c_set_adapdata(&iface->adapter, NULL);
}
if (probe) {
unsigned char addr;
printk("Probe: ");
for (addr = 0x00; addr <= 0x7f; addr++) {
if (i2c_smbus_xfer(&iface->adapter,addr,
0,0,0,I2C_SMBUS_QUICK,NULL) >= 0)
printk("%02x ", addr);
}
printk("\n");
}
printk(KERN_INFO "SMU i2c bus %x registered\n", busid);
return 0;
}
static int dispose_iface(struct device *dev)
{
struct smu_iface *iface = dev_get_drvdata(dev);
int rc;
rc = i2c_del_adapter(&iface->adapter);
i2c_set_adapdata(&iface->adapter, NULL);
/* We aren't that prepared to deal with this... */
if (rc)
printk("i2c-pmac-smu.c: Failed to remove bus %s !\n",
iface->adapter.name);
dev_set_drvdata(dev, NULL);
kfree(iface);
return 0;
}
static int create_iface_of_platform(struct of_device* dev,
const struct of_device_id *match)
{
return create_iface(dev->node, &dev->dev);
}
static int dispose_iface_of_platform(struct of_device* dev)
{
return dispose_iface(&dev->dev);
}
static struct of_device_id i2c_smu_match[] =
{
{
.compatible = "smu-i2c",
},
{},
};
static struct of_platform_driver i2c_smu_of_platform_driver =
{
.name = "i2c-smu",
.match_table = i2c_smu_match,
.probe = create_iface_of_platform,
.remove = dispose_iface_of_platform
};
static int __init i2c_pmac_smu_init(void)
{
of_register_driver(&i2c_smu_of_platform_driver);
return 0;
}
static void __exit i2c_pmac_smu_cleanup(void)
{
of_unregister_driver(&i2c_smu_of_platform_driver);
}
module_init(i2c_pmac_smu_init);
module_exit(i2c_pmac_smu_cleanup);
This diff is collapsed.
...@@ -599,7 +599,7 @@ thermostat_init(void) ...@@ -599,7 +599,7 @@ thermostat_init(void)
sensor_location[2] = "?"; sensor_location[2] = "?";
} }
of_dev = of_platform_device_create(np, "temperatures"); of_dev = of_platform_device_create(np, "temperatures", NULL);
if (of_dev == NULL) { if (of_dev == NULL) {
printk(KERN_ERR "Can't register temperatures device !\n"); printk(KERN_ERR "Can't register temperatures device !\n");
......
...@@ -2051,7 +2051,7 @@ static int __init therm_pm72_init(void) ...@@ -2051,7 +2051,7 @@ static int __init therm_pm72_init(void)
return -ENODEV; return -ENODEV;
} }
} }
of_dev = of_platform_device_create(np, "temperature"); of_dev = of_platform_device_create(np, "temperature", NULL);
if (of_dev == NULL) { if (of_dev == NULL) {
printk(KERN_ERR "Can't register FCU platform device !\n"); printk(KERN_ERR "Can't register FCU platform device !\n");
return -ENODEV; return -ENODEV;
......
...@@ -504,7 +504,7 @@ g4fan_init( void ) ...@@ -504,7 +504,7 @@ g4fan_init( void )
} }
if( !(np=of_find_node_by_name(NULL, "fan")) ) if( !(np=of_find_node_by_name(NULL, "fan")) )
return -ENODEV; return -ENODEV;
x.of_dev = of_platform_device_create( np, "temperature" ); x.of_dev = of_platform_device_create(np, "temperature", NULL);
of_node_put( np ); of_node_put( np );
if( !x.of_dev ) { if( !x.of_dev ) {
......
#ifndef __MACIO_ASIC_H__ #ifndef __MACIO_ASIC_H__
#define __MACIO_ASIC_H__ #define __MACIO_ASIC_H__
#include <linux/mod_devicetable.h>
#include <asm/of_device.h> #include <asm/of_device.h>
extern struct bus_type macio_bus_type; extern struct bus_type macio_bus_type;
......
...@@ -2,6 +2,7 @@ ...@@ -2,6 +2,7 @@
#define __OF_DEVICE_H__ #define __OF_DEVICE_H__
#include <linux/device.h> #include <linux/device.h>
#include <linux/mod_devicetable.h>
#include <asm/prom.h> #include <asm/prom.h>
/* /*
...@@ -55,7 +56,9 @@ extern int of_register_driver(struct of_platform_driver *drv); ...@@ -55,7 +56,9 @@ extern int of_register_driver(struct of_platform_driver *drv);
extern void of_unregister_driver(struct of_platform_driver *drv); extern void of_unregister_driver(struct of_platform_driver *drv);
extern int of_device_register(struct of_device *ofdev); extern int of_device_register(struct of_device *ofdev);
extern void of_device_unregister(struct of_device *ofdev); extern void of_device_unregister(struct of_device *ofdev);
extern struct of_device *of_platform_device_create(struct device_node *np, const char *bus_id); extern struct of_device *of_platform_device_create(struct device_node *np,
const char *bus_id,
struct device *parent);
extern void of_release_dev(struct device *dev); extern void of_release_dev(struct device *dev);
#endif /* __OF_DEVICE_H__ */ #endif /* __OF_DEVICE_H__ */
......
This diff is collapsed.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment