168 lines
3.8 KiB
C
168 lines
3.8 KiB
C
|
/* Quanta EC driver for the Winbond Embedded Controller
|
||
|
*
|
||
|
* Copyright (C) 2009 Quanta Computer Inc.
|
||
|
*
|
||
|
* This software is licensed under the terms of the GNU General Public
|
||
|
* License version 2, as published by the Free Software Foundation, and
|
||
|
* may be copied, distributed, and modified under those terms.
|
||
|
*
|
||
|
* 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.
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#include <linux/module.h>
|
||
|
#include <linux/err.h>
|
||
|
#include <linux/i2c.h>
|
||
|
#include <linux/slab.h>
|
||
|
|
||
|
#define EC_ID_NAME "qci-i2cec"
|
||
|
#define EC_BUFFER_LEN 16
|
||
|
#define EC_CMD_POWER_OFF 0xAC
|
||
|
#define EC_CMD_RESTART 0xAB
|
||
|
|
||
|
static struct i2c_client *g_i2cec_client;
|
||
|
|
||
|
/* General structure to hold the driver data */
|
||
|
struct i2cec_drv_data {
|
||
|
struct i2c_client *i2cec_client;
|
||
|
struct work_struct work;
|
||
|
char ec_data[EC_BUFFER_LEN+1];
|
||
|
};
|
||
|
|
||
|
static int __devinit wpce_probe(struct i2c_client *client,
|
||
|
const struct i2c_device_id *id);
|
||
|
static int __devexit wpce_remove(struct i2c_client *kbd);
|
||
|
|
||
|
#ifdef CONFIG_PM
|
||
|
static int wpce_suspend(struct device *dev)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int wpce_resume(struct device *dev)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#ifdef CONFIG_PM
|
||
|
static struct dev_pm_ops wpce_pm_ops = {
|
||
|
.suspend = wpce_suspend,
|
||
|
.resume = wpce_resume,
|
||
|
};
|
||
|
#endif
|
||
|
|
||
|
static const struct i2c_device_id wpce_idtable[] = {
|
||
|
{ EC_ID_NAME, 0 },
|
||
|
{ }
|
||
|
};
|
||
|
|
||
|
static struct i2c_driver wpce_driver = {
|
||
|
.driver = {
|
||
|
.owner = THIS_MODULE,
|
||
|
.name = EC_ID_NAME,
|
||
|
#ifdef CONFIG_PM
|
||
|
.pm = &wpce_pm_ops,
|
||
|
#endif
|
||
|
},
|
||
|
.probe = wpce_probe,
|
||
|
.remove = __devexit_p(wpce_remove),
|
||
|
.id_table = wpce_idtable,
|
||
|
};
|
||
|
|
||
|
static int __devinit wpce_probe(struct i2c_client *client,
|
||
|
const struct i2c_device_id *id)
|
||
|
{
|
||
|
int err = -ENOMEM;
|
||
|
struct i2cec_drv_data *context = 0;
|
||
|
|
||
|
/* there is no need to call i2c_check_functionality() since it is the
|
||
|
client's job to use the interface (I2C vs SMBUS) appropriate for it. */
|
||
|
client->driver = &wpce_driver;
|
||
|
context = kzalloc(sizeof(struct i2cec_drv_data), GFP_KERNEL);
|
||
|
if (!context)
|
||
|
return err;
|
||
|
|
||
|
context->i2cec_client = client;
|
||
|
g_i2cec_client = client;
|
||
|
i2c_set_clientdata(context->i2cec_client, context);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int __devexit wpce_remove(struct i2c_client *dev)
|
||
|
{
|
||
|
struct i2cec_drv_data *context = i2c_get_clientdata(dev);
|
||
|
g_i2cec_client = NULL;
|
||
|
kfree(context);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int __init wpce_init(void)
|
||
|
{
|
||
|
return i2c_add_driver(&wpce_driver);
|
||
|
}
|
||
|
|
||
|
static void __exit wpce_exit(void)
|
||
|
{
|
||
|
i2c_del_driver(&wpce_driver);
|
||
|
}
|
||
|
|
||
|
struct i2c_client *wpce_get_i2c_client(void)
|
||
|
{
|
||
|
return g_i2cec_client;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(wpce_get_i2c_client);
|
||
|
|
||
|
void wpce_poweroff(void)
|
||
|
{
|
||
|
if (g_i2cec_client == NULL)
|
||
|
return;
|
||
|
i2c_smbus_write_byte(g_i2cec_client, EC_CMD_POWER_OFF);
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(wpce_poweroff);
|
||
|
|
||
|
void wpce_restart(void)
|
||
|
{
|
||
|
if (g_i2cec_client == NULL)
|
||
|
return;
|
||
|
i2c_smbus_write_byte(g_i2cec_client, EC_CMD_RESTART);
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(wpce_restart);
|
||
|
|
||
|
int wpce_i2c_transfer(struct i2c_msg *msg)
|
||
|
{
|
||
|
if (g_i2cec_client == NULL)
|
||
|
return -1;
|
||
|
msg->addr = g_i2cec_client->addr;
|
||
|
return i2c_transfer(g_i2cec_client->adapter, msg, 1);
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(wpce_i2c_transfer);
|
||
|
|
||
|
int wpce_smbus_write_word_data(u8 command, u16 value)
|
||
|
{
|
||
|
if (g_i2cec_client == NULL)
|
||
|
return -1;
|
||
|
return i2c_smbus_write_word_data(g_i2cec_client, command, value);
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(wpce_smbus_write_word_data);
|
||
|
|
||
|
int wpce_smbus_write_byte_data(u8 command, u8 value)
|
||
|
{
|
||
|
if (g_i2cec_client == NULL)
|
||
|
return -1;
|
||
|
return i2c_smbus_write_byte_data(g_i2cec_client, command, value);
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(wpce_smbus_write_byte_data);
|
||
|
|
||
|
module_init(wpce_init);
|
||
|
module_exit(wpce_exit);
|
||
|
|
||
|
MODULE_AUTHOR("Quanta Computer Inc.");
|
||
|
MODULE_DESCRIPTION("Quanta Embedded Controller I2C Bridge Driver");
|
||
|
MODULE_LICENSE("GPL v2");
|