mirror of
git://projects.qi-hardware.com/openwrt-xburst.git
synced 2025-01-01 07:32:23 +02:00
dc3d3f1c49
it's basically also provided by ingenic and nativly based on 2.6.27, adjusted to fit into the OpenWrt-environment
331 lines
6.8 KiB
C
331 lines
6.8 KiB
C
/*
|
|
* i2c_jz47xx.c
|
|
* I2C adapter for the INGENIC I2C bus access.
|
|
*
|
|
* Copyright (C) 2006 - 2008 Ingenic Semiconductor Inc.
|
|
* Author: <cwjia@ingenic.cn>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
*/
|
|
#include <linux/kernel.h>
|
|
#include <linux/module.h>
|
|
#include <linux/i2c.h>
|
|
#include <linux/i2c-id.h>
|
|
#include <linux/init.h>
|
|
#include <linux/time.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/errno.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/delay.h>
|
|
|
|
#include <asm/irq.h>
|
|
#include <asm/io.h>
|
|
#include <linux/module.h>
|
|
#include <asm/addrspace.h>
|
|
|
|
#include <asm/jzsoc.h>
|
|
#include "i2c-jz47xx.h"
|
|
|
|
/* I2C protocol */
|
|
#define I2C_READ 1
|
|
#define I2C_WRITE 0
|
|
|
|
#define TIMEOUT 1000
|
|
unsigned short sub_addr = 0;
|
|
int addr_val = 0;
|
|
struct jz_i2c {
|
|
spinlock_t lock;
|
|
wait_queue_head_t wait;
|
|
struct i2c_msg *msg;
|
|
unsigned int msg_num;
|
|
unsigned int slave_addr;
|
|
struct i2c_adapter adap;
|
|
struct clk *clk;
|
|
};
|
|
|
|
/*
|
|
* I2C bus protocol basic routines
|
|
*/
|
|
static int i2c_put_data(unsigned char data)
|
|
{
|
|
unsigned int timeout = TIMEOUT*10;
|
|
|
|
__i2c_write(data);
|
|
__i2c_set_drf();
|
|
while (__i2c_check_drf() != 0);
|
|
while (!__i2c_transmit_ended());
|
|
while (!__i2c_received_ack() && timeout)
|
|
timeout--;
|
|
|
|
if (timeout)
|
|
return 0;
|
|
else
|
|
return -ETIMEDOUT;
|
|
}
|
|
|
|
static int i2c_get_data(unsigned char *data, int ack)
|
|
{
|
|
int timeout = TIMEOUT*10;
|
|
|
|
if (!ack)
|
|
__i2c_send_nack();
|
|
else
|
|
__i2c_send_ack();
|
|
|
|
while (__i2c_check_drf() == 0 && timeout)
|
|
timeout--;
|
|
|
|
if (timeout) {
|
|
if (!ack)
|
|
__i2c_send_stop();
|
|
*data = __i2c_read();
|
|
__i2c_clear_drf();
|
|
return 0;
|
|
} else
|
|
return -ETIMEDOUT;
|
|
}
|
|
|
|
/*
|
|
* I2C interface
|
|
*/
|
|
void i2c_jz_setclk(unsigned int i2cclk)
|
|
{
|
|
__i2c_set_clk(jz_clocks.extalclk, i2cclk);
|
|
}
|
|
|
|
static int xfer_read(unsigned char device, struct i2c_adapter *adap, unsigned char *buf, int length)
|
|
{
|
|
int cnt = length;
|
|
int timeout = 5;
|
|
|
|
device = (0xa << 3) | ((sub_addr & 0x0700) >> 8);
|
|
sub_addr = sub_addr & 0xff;
|
|
|
|
L_try_again:
|
|
|
|
if (timeout < 0)
|
|
goto L_timeout;
|
|
|
|
__i2c_send_nack(); /* Master does not send ACK, slave sends it */
|
|
|
|
if (addr_val) {
|
|
__i2c_send_start();
|
|
if (i2c_put_data( (device << 1) | I2C_WRITE ) < 0)
|
|
goto device_werr;
|
|
if (i2c_put_data(sub_addr) < 0)
|
|
goto address_err;
|
|
}
|
|
__i2c_send_start();
|
|
|
|
if (i2c_put_data((device << 1) | I2C_READ ) < 0)
|
|
goto device_rerr;
|
|
|
|
__i2c_send_ack(); /* Master sends ACK for continue reading */
|
|
__i2c_send_start();
|
|
|
|
while (cnt) {
|
|
if (cnt == 1) {
|
|
if (i2c_get_data(buf, 0) < 0)
|
|
break;
|
|
} else {
|
|
if (i2c_get_data(buf, 1) < 0)
|
|
break;
|
|
}
|
|
cnt--;
|
|
buf++;
|
|
}
|
|
addr_val = 0;
|
|
return length - cnt;
|
|
device_rerr:
|
|
device_werr:
|
|
address_err:
|
|
timeout --;
|
|
__i2c_send_stop();
|
|
goto L_try_again;
|
|
|
|
L_timeout:
|
|
__i2c_send_stop();
|
|
printk("Read I2C device 0x%2x failed.\n", device);
|
|
return -ENODEV;
|
|
}
|
|
|
|
static int xfer_write(unsigned char device, struct i2c_adapter *adap, unsigned char *buf, int length)
|
|
{
|
|
int cnt = length;
|
|
int cnt_in_pg;
|
|
int timeout = 5;
|
|
unsigned char *tmpbuf;
|
|
unsigned char tmpaddr;
|
|
|
|
device = (0xa << 3) | ((sub_addr & 0x0700) >> 8);
|
|
sub_addr = sub_addr & 0xff;
|
|
|
|
__i2c_send_nack(); /* Master does not send ACK, slave sends it */
|
|
|
|
W_try_again:
|
|
if (timeout < 0)
|
|
goto W_timeout;
|
|
|
|
cnt = length;
|
|
tmpbuf = (unsigned char *)buf;
|
|
tmpaddr = device;
|
|
start_write_page:
|
|
cnt_in_pg = 0;
|
|
__i2c_send_start();
|
|
if (i2c_put_data( (device << 1) | I2C_WRITE ) < 0)
|
|
goto device_err;
|
|
if (addr_val) {
|
|
if (i2c_put_data(sub_addr) < 0)
|
|
goto address_err;
|
|
}
|
|
while (cnt) {
|
|
if (++cnt_in_pg > 8) {
|
|
__i2c_send_stop();
|
|
mdelay(1);
|
|
sub_addr += 8;
|
|
goto start_write_page;
|
|
}
|
|
if (i2c_put_data(*tmpbuf) < 0)
|
|
break;
|
|
cnt--;
|
|
tmpbuf++;
|
|
}
|
|
__i2c_send_stop();
|
|
addr_val = 0;
|
|
return length - cnt;
|
|
device_err:
|
|
address_err:
|
|
timeout--;
|
|
__i2c_send_stop();
|
|
goto W_try_again;
|
|
|
|
W_timeout:
|
|
printk(KERN_DEBUG "Write I2C device 0x%2x failed.\n", device);
|
|
__i2c_send_stop();
|
|
return -ENODEV;
|
|
}
|
|
|
|
static int i2c_jz_xfer(struct i2c_adapter *adap, struct i2c_msg *pmsg, int num)
|
|
{
|
|
int ret, i;
|
|
|
|
dev_dbg(&adap->dev, "jz47xx_xfer: processing %d messages:\n", num);
|
|
for (i = 0; i < num; i++) {
|
|
dev_dbg(&adap->dev, " #%d: %sing %d byte%s %s 0x%02x\n", i,
|
|
pmsg->flags & I2C_M_RD ? "read" : "writ",
|
|
pmsg->len, pmsg->len > 1 ? "s" : "",
|
|
pmsg->flags & I2C_M_RD ? "from" : "to", pmsg->addr);
|
|
if (pmsg->len && pmsg->buf) { /* sanity check */
|
|
if (pmsg->flags & I2C_M_RD)
|
|
ret = xfer_read(pmsg->addr, adap, pmsg->buf, pmsg->len);
|
|
else
|
|
ret = xfer_write(pmsg->addr, adap, pmsg->buf, pmsg->len);
|
|
|
|
if (ret)
|
|
return ret;
|
|
/* Wait until transfer is finished */
|
|
}
|
|
dev_dbg(&adap->dev, "transfer complete\n");
|
|
pmsg++; /* next message */
|
|
}
|
|
return i;
|
|
}
|
|
|
|
static u32 i2c_jz_functionality(struct i2c_adapter *adap)
|
|
{
|
|
return I2C_FUNC_I2C | I2C_FUNC_SMBUS_EMUL;
|
|
}
|
|
|
|
static const struct i2c_algorithm i2c_jz_algorithm = {
|
|
.master_xfer = i2c_jz_xfer,
|
|
.functionality = i2c_jz_functionality,
|
|
};
|
|
|
|
static int i2c_jz_probe(struct platform_device *dev)
|
|
{
|
|
struct jz_i2c *i2c;
|
|
struct i2c_jz_platform_data *plat = dev->dev.platform_data;
|
|
int ret;
|
|
|
|
__i2c_set_clk(jz_clocks.extalclk, 10000); /* default 10 KHz */
|
|
__i2c_enable();
|
|
|
|
i2c = kzalloc(sizeof(struct jz_i2c), GFP_KERNEL);
|
|
if (!i2c) {
|
|
printk("There is no enough memory\n");
|
|
ret = -ENOMEM;
|
|
goto emalloc;
|
|
}
|
|
|
|
i2c->adap.owner = THIS_MODULE;
|
|
i2c->adap.algo = &i2c_jz_algorithm;
|
|
i2c->adap.retries = 5;
|
|
spin_lock_init(&i2c->lock);
|
|
init_waitqueue_head(&i2c->wait);
|
|
sprintf(i2c->adap.name, "jz_i2c-i2c.%u", dev->id);
|
|
i2c->adap.algo_data = i2c;
|
|
i2c->adap.dev.parent = &dev->dev;
|
|
|
|
if (plat) {
|
|
i2c->adap.class = plat->class;
|
|
}
|
|
|
|
/*
|
|
* If "dev->id" is negative we consider it as zero.
|
|
* The reason to do so is to avoid sysfs names that only make
|
|
* sense when there are multiple adapters.
|
|
*/
|
|
i2c->adap.nr = dev->id != -1 ? dev->id : 0;
|
|
/* ret = i2c_add_adapter(&i2c->adap); */
|
|
ret = i2c_add_numbered_adapter(&i2c->adap);
|
|
if (ret < 0) {
|
|
printk(KERN_INFO "I2C: Failed to add bus\n");
|
|
goto eadapt;
|
|
}
|
|
|
|
platform_set_drvdata(dev, i2c);
|
|
dev_info(&dev->dev, "JZ47xx i2c bus driver.\n");
|
|
return 0;
|
|
eadapt:
|
|
__i2c_disable();
|
|
emalloc:
|
|
return ret;
|
|
}
|
|
|
|
static int i2c_jz_remove(struct platform_device *dev)
|
|
{
|
|
struct i2c_adapter *adapter = platform_get_drvdata(dev);
|
|
int rc;
|
|
|
|
rc = i2c_del_adapter(adapter);
|
|
platform_set_drvdata(dev, NULL);
|
|
return rc;
|
|
}
|
|
|
|
static struct platform_driver i2c_jz_driver = {
|
|
.probe = i2c_jz_probe,
|
|
.remove = i2c_jz_remove,
|
|
.driver = {
|
|
.name = "jz_i2c",
|
|
},
|
|
};
|
|
|
|
static int __init i2c_adap_jz_init(void)
|
|
{
|
|
return platform_driver_register(&i2c_jz_driver);
|
|
}
|
|
|
|
static void __exit i2c_adap_jz_exit(void)
|
|
{
|
|
return platform_driver_unregister(&i2c_jz_driver);
|
|
}
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
module_init(i2c_adap_jz_init);
|
|
module_exit(i2c_adap_jz_exit);
|