I2C bus glue for Cirrus EP93xx

转自 http://arm.cirrus.com/forum/viewtopic.php?t=51&highlight=i2c

 

The attached file should be placed in linux-2.4.21\drivers\i2c\ . 

In Config.in I have added: 

dep_tristate ' I2C bus glue for Cirrus EP93xx processors' CONFIG_I2C_EP93XX $CONFIG_I2C_ALGOBIT 

And in Makefile: 

obj-$(CONFIG_I2C_EP93XX) += i2c-ep93xx.o 

 

 

/* ------------------------------------------------------------------------ *
 * i2c-ep933xx.c I2C bus glue for Cirrus EP93xx                             *
 * ------------------------------------------------------------------------ *

   Copyright (C) 2004 Michael Burian
   
   Based on i2c-parport-light.c
   Copyright (C) 2003-2004 Jean Delvare <[email protected]>
  
   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/kernel.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/ioport.h>
#include <linux/delay.h>
#include <linux/i2c.h>
#include <linux/i2c-algo-bit.h>
#include <asm/io.h>

//1/(2*clockfrequency)
#define EE_DELAY_USEC       50
#define GPIOG_EECLK 1
#define GPIOG_EEDAT 2

/* ----- I2C algorithm call-back functions and structures ----------------- */

// TODO: optimize
static void ep93xx_setscl(void *data, int state)
{
	unsigned int val, dir;

	val = inl(GPIO_PGDR);
	dir = inl(GPIO_PGDDR);

	/* Configure the clock line as output. */
	dir |= GPIOG_EECLK;
	outl(dir, GPIO_PGDDR);

	/* Set clock line to state */
	if(state)
		val |= GPIOG_EECLK;
	else
		val &= ~GPIOG_EECLK;
	
	outl(val, GPIO_PGDR);
}

static void ep93xx_setsda(void *data, int state)
{
	unsigned int val, dir;
	val = inl(GPIO_PGDR);
	dir = inl(GPIO_PGDDR);

	/* Configure the data line as output. */
	dir |= GPIOG_EEDAT;
	outl(dir, GPIO_PGDDR);

	/* Set data line to state */
	if(state)
		val |= GPIOG_EEDAT;
	else
		val &= ~GPIOG_EEDAT;
	
	outl(val, GPIO_PGDR);
}

static int ep93xx_getscl(void *data)
{
	unsigned int val, dir;
	val = inl(GPIO_PGDR);
	dir = inl(GPIO_PGDDR);

	/* Configure the clock line as input */
	dir &= ~GPIOG_EECLK;
	outl(dir, GPIO_PGDDR);
	
	/* Return state of the clock line */
	return (inl(GPIO_PGDR) & GPIOG_EECLK) ? 1 : 0;
}

static int ep93xx_getsda(void *data)
{
	unsigned int val, dir;
	val = inl(GPIO_PGDR);
	dir = inl(GPIO_PGDDR);

	/* Configure the data line as input */
	dir &= ~GPIOG_EEDAT;
	outl(dir, GPIO_PGDDR);

	/* Return state of the data line */
	return (inl(GPIO_PGDR) & GPIOG_EEDAT) ? 1 : 0;
}

static int ep93xx_reg(struct i2c_client *client)
{
	return 0;
}

static int ep93xx_unreg(struct i2c_client *client)
{
	return 0;
}

static void ep93xx_inc_use(struct i2c_adapter *adap)
{
#ifdef MODULE
	MOD_INC_USE_COUNT;
#endif
}

static void ep93xx_dec_use(struct i2c_adapter *adap)
{
#ifdef MODULE
	MOD_DEC_USE_COUNT;
#endif
}
/* ------------------------------------------------------------------------
 * Encapsulate the above functions in the correct operations structure.
 * This is only done when more than one hardware adapter is supported.
 */

/* last line (us, ms, timout)
 * us dominates the bit rate: 10us  means: 100Kbit/sec(25 means 40kbps)
 *                            10ms  not known
 *                            100ms timeout
 */
static struct i2c_algo_bit_data ep93xx_data = {
	NULL,
	ep93xx_setsda,
	ep93xx_setscl,
	ep93xx_getsda,
	ep93xx_getscl,
//	EE_DELAY_USEC, EE_DELAY_USEC, 100,	/* orginal (non-guide) value 10, 10, 100  */
	10, 10, 100,	/* orginal (non-guide) value 10, 10, 100  */
};

/* ----- I2c structure ---------------------------------------------------- */
static struct i2c_adapter ep93xx_adapter = {
	"EP93XX I2C interface",
	I2C_HW_B_LP, //I2C_HW_B_GUIDE
	NULL,
	&ep93xx_data,
	ep93xx_inc_use,
	ep93xx_dec_use,
	ep93xx_reg,
	ep93xx_unreg,
};

/* ----- Module loading, unloading and information ------------------------ */

static int __init i2c_ep93xx_init(void)
{
	unsigned long uiVal, uiDDR;
	
	/* Read the current value of the GPIO data and data direction registers. */
	uiVal = inl(GPIO_PGDR);
	uiDDR = inl(GPIO_PGDDR);

	/* If the GPIO pins have not been configured since reset, the data 
	 * and clock lines will be set as inputs and with data value of 0.
	 * External pullup resisters are pulling them high.
	 * Set them both high before configuring them as outputs. */
	uiVal |= (GPIOG_EEDAT | GPIOG_EECLK);
	outl(uiVal, GPIO_PGDR);

	/* Delay to meet the EE Interface timing specification. */
	udelay(EE_DELAY_USEC);

	
	/* Configure the EE data and clock lines as outputs. */
	uiDDR |= (GPIOG_EEDAT | GPIOG_EECLK);
	outl(uiDDR, GPIO_PGDDR);

	/* Delay to meet the EE Interface timing specification. */
	udelay(EE_DELAY_USEC);

	/* Reset hardware to a sane state (SCL and SDA high) */
	ep93xx_setsda(NULL, 1);
	ep93xx_setscl(NULL, 1);

	if (i2c_bit_add_bus(&ep93xx_adapter) < 0) {
		printk(KERN_ERR "i2c-ep93xx: Unable to register with I2C\n");
		return -ENODEV;
	}
	
	return 0;
}

static void __exit i2c_ep93xx_exit(void)
{
	i2c_bit_del_bus(&ep93xx_adapter);
}

EXPORT_NO_SYMBOLS;

MODULE_AUTHOR("Michael Burian");
MODULE_DESCRIPTION("I2C bus glue for Cirrus EP93xx processors");
MODULE_LICENSE("GPL");

module_init(i2c_ep93xx_init);
module_exit(i2c_ep93xx_exit);

你可能感兴趣的:(for)