/* 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");