亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關于我們
? 蟲蟲下載站

?? iicdriver.c

?? 在網上搜索到的2410的IIC driver. 還沒有用過
?? C
?? 第 1 頁 / 共 2 頁
字號:
 /* S3C2410 I2C Controller
 *
 * 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., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
*/

#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/interrupt.h>
#include <linux/sched.h>
#include <linux/delay.h>
#include <linux/errno.h>
#include <linux/err.h>
#include <linux/device.h>

#include <asm/hardware.h>
#include <asm/irq.h>

#include <asm/hardware/clock.h>
#include <asm/arch/regs-gpio.h>
#include <asm/arch/regs-iic.h>
#include <asm/arch/iic.h>

static int debug_info;
static int debug_norm = 1;

enum s3c24xx_i2c_state {
	STATE_IDLE,
	STATE_START,
	STATE_READ,
	STATE_WRITE,
	STATE_STOP
};

struct s3c24xx_i2c {
	spinlock_t		lock;
	wait_queue_head_t	wait;

	struct i2c_msg		*msg;
	unsigned int		msg_num;
	unsigned int		msg_idx;
	unsigned int		msg_ptr;

	enum s3c24xx_i2c_state	state;

	void __iomem		*regs;
	struct clk		*clk;
	struct device		*dev;
	struct resource		*irq;
	struct resource		*ioarea;
	struct i2c_adapter	adap;
};

/* default platform data to use if the user does not supply
 * any 
*/

static struct s3c2410_platform_i2c s3c24xx_i2c_default_platform = {
	.flags		= 0,
	.slave_addr	= 0x10,
	.bus_freq	= 100*1000,
	.max_freq	= 400*1000,
};

/* Debug */

#define DBG(lev, msg...) do {
	if (lev < debug_info) 	dev_info(i2c->dev, msg);
	if (lev < debug_norm ) dev_dbg(i2c->dev, msg); 
	} while(0)


/* s3c24xx_i2c_get_platformdata
 *
 * get the platform data associated with the given device, or return
 * the default if there is none
*/

static inline struct s3c2410_platform_i2c *s3c24xx_i2c_get_platformdata(struct device *dev)
{
	if (dev->platform_data != NULL)
		return (struct s3c2410_platform_i2c *)dev->platform_data;

	return &s3c24xx_i2c_default_platform;
}

/* s3c24xx_i2c_master_complete
 *
 * complete the message and wake up the caller, using the given return code,
 * or zero to mean ok.
*/

static inline void s3c24xx_i2c_master_complete(struct s3c24xx_i2c *i2c, int ret)
{
	DBG(2, "s3c24xx_i2c_master_complete, code %d\n", ret);

	i2c->msg_ptr = 0;
	i2c->msg = NULL;
	i2c->msg_idx ++;
	i2c->msg_num = 0;
	if (ret)
		i2c->msg_idx = ret;

	wake_up(&i2c->wait);
}

static inline void s3c24xx_i2c_disable_ack(struct s3c24xx_i2c *i2c)
{
	unsigned long tmp;
	
	DBG(3, "s3c24xx_i2c_disable_ack:\n");

	tmp = readl(i2c->regs + S3C2410_IICCON);
	writel(tmp & ~S3C2410_IICCON_ACKEN, i2c->regs + S3C2410_IICCON);

}

static inline void s3c24xx_i2c_enable_ack(struct s3c24xx_i2c *i2c)
{
	unsigned long tmp;
	
	tmp = readl(i2c->regs + S3C2410_IICCON);
	writel(tmp | S3C2410_IICCON_ACKEN, i2c->regs + S3C2410_IICCON);

}

/* irq enable/disable functions */

static inline void s3c24xx_i2c_disable_irq(struct s3c24xx_i2c *i2c)
{
	unsigned long tmp;
	
	tmp = readl(i2c->regs + S3C2410_IICCON);
	writel(tmp & ~S3C2410_IICCON_IRQEN, i2c->regs + S3C2410_IICCON);
}

static inline void s3c24xx_i2c_enable_irq(struct s3c24xx_i2c *i2c)
{
	unsigned long tmp;
	
	tmp = readl(i2c->regs + S3C2410_IICCON);
	writel(tmp | S3C2410_IICCON_IRQEN, i2c->regs + S3C2410_IICCON);
}


/* s3c24xx_i2c_message_start
 *
 * put the start of a message onto the bus 
*/

static void s3c24xx_i2c_message_start(struct s3c24xx_i2c *i2c, 
				      struct i2c_msg *msg)
{
	unsigned int addr = (msg->addr & 0x7f) << 1;
	unsigned long stat;

	stat = readl(i2c->regs + S3C2410_IICSTAT);
	stat &= ~S3C2410_IICSTAT_MODEMASK;
	stat |=  S3C2410_IICSTAT_START;
	stat |=  S3C2410_IICSTAT_TXRXEN;

	if (msg->flags & I2C_M_RD) {
		stat |= S3C2410_IICSTAT_MASTER_RX;
		addr |= 1;
	} else
		stat |= S3C2410_IICSTAT_MASTER_TX;

	// todo - check for wether ack wanted or not
	s3c24xx_i2c_enable_ack(i2c);

	DBG(3, "START Written %08lx to IICSTAT, %02x to DS\n", stat, addr);
	writeb(addr, i2c->regs + S3C2410_IICDS);
	writel(stat, i2c->regs + S3C2410_IICSTAT);
}

static inline void s3c24xx_i2c_stop(struct s3c24xx_i2c *i2c, int ret)
{
	unsigned long iicstat = readl(i2c->regs + S3C2410_IICSTAT);

	DBG(2, "s3c24xx_i2c_msg_stop:\n");

	/* stop the transfer */
	iicstat &= ~ S3C2410_IICSTAT_START;
	writel(iicstat, i2c->regs + S3C2410_IICSTAT);
	
	i2c->state = STATE_STOP;
	
	s3c24xx_i2c_master_complete(i2c, ret);
	s3c24xx_i2c_disable_irq(i2c);
}

/* helper functions to determine the current state in the set of
 * messages we are sending */

/* is_lastmsg()
 *
 * returns TRUE if the current message is the last in the set 
*/

static inline int is_lastmsg(struct s3c24xx_i2c *i2c)
{
	return i2c->msg_idx >= (i2c->msg_num - 1);
}

/* is_msglast
 *
 * returns TRUE if we this is the last byte in the current message
*/

static inline int is_msglast(struct s3c24xx_i2c *i2c)
{
	return i2c->msg_ptr == i2c->msg->len-1;
}

/* is_msgend
 *
 * returns TRUE if we reached the end of the current message
*/

static inline int is_msgend(struct s3c24xx_i2c *i2c)
{
	return i2c->msg_ptr >= i2c->msg->len;
}

static int i2s_s3c_irq_nextbyte(struct s3c24xx_i2c *i2c, unsigned long iicstat)
{
	unsigned long tmp;
	unsigned char byte;
	int ret = 0;

	DBG(3, "nextbyte: msg %p, num=%d, idx=%d, ptr=%d, state=%d\n",
	    i2c->msg, i2c->msg_num, i2c->msg_idx, i2c->msg_ptr, i2c->state);

	switch (i2c->state) {

	case STATE_IDLE:
		dev_err(i2c->dev, "%s: called in STATE_IDLE\n", __FUNCTION__);
		goto out;
		break;

	case STATE_STOP:
		dev_err(i2c->dev, "%s: called in STATE_STOP\n", __FUNCTION__);
		s3c24xx_i2c_disable_irq(i2c);		
		goto out_ack;

	case STATE_START:
		/* last thing we did was send a start condition on the
		 * bus, or started a new i2c message
		 */
		
		DBG(3, "STATE_START: iicstat=%lx\n", iicstat);

		if (iicstat  & S3C2410_IICSTAT_LASTBIT &&
		    !(i2c->msg->flags & I2C_M_IGNORE_NAK)) {
			/* ack was not received... */
			DBG(2, "ACK not received from slave\n");

			s3c24xx_i2c_stop(i2c, -EREMOTEIO);
			goto out_ack;
		}

		if (i2c->msg->flags & I2C_M_RD)
			i2c->state = STATE_READ;
		else
			i2c->state = STATE_WRITE;

		/* terminate the transfer if there is nothing to do
		 * (used by the i2c probe to find devices */

		if (is_lastmsg(i2c) && i2c->msg->len == 0) {
			DBG(3, "START: Looks like probe\n");
			s3c24xx_i2c_stop(i2c, 0);
			goto out_ack;
		}

		if (i2c->state == STATE_READ)
			goto prepare_read;

		/* fall through to the write state, as we will need to 
		 * send a byte as well */

	case STATE_WRITE:
		/* we are writing data to the device... check for the
		 * end of the message, and if so, work out what to do
		 */

		if (!is_msgend(i2c)) {
			byte = i2c->msg->buf[i2c->msg_ptr++];

			DBG(3, "WRITE byte %02x\n", byte);
			writeb(byte, i2c->regs + S3C2410_IICDS);
			
		} else if (!is_lastmsg(i2c)) {
			/* we need to go to the next i2c message */

			DBG(3, "WRITE: Next Message\n");

			i2c->msg_ptr = 0;
			i2c->msg_idx ++;
			i2c->msg++;
			
			/* check to see if we need to do another message */
			if (i2c->msg->flags & I2C_M_NOSTART) {
				// todo //
				DBG(1, "WRITE: Error I2C_M_NOSTART\n");
			}
			
			/* send the new start */
			s3c24xx_i2c_message_start(i2c, i2c->msg);
			i2c->state = STATE_START;
		} else {
			/* send stop */
			DBG(3, "WRITE: Send Stop\n");

			s3c24xx_i2c_stop(i2c, 0);
		}
		break;

	case STATE_READ:
		/* we have a byte of data in the data register, do 
		 * something with it, and then work out wether we are
		 * going to do any more read/write
		 */

		if (!(i2c->msg->flags & I2C_M_IGNORE_NAK) &&
		    !(is_msglast(i2c) && is_lastmsg(i2c))) {

			if (iicstat & S3C2410_IICSTAT_LASTBIT) {
				DBG(2, "READ: No Ack\n");

				s3c24xx_i2c_stop(i2c, -ECONNREFUSED);
				goto out_ack;
			}
		}

		i2c->msg->buf[i2c->msg_ptr++] = readb(i2c->regs + S3C2410_IICDS);

		DBG(3, "READ: Read %02x\n", i2c->msg->buf[i2c->msg_ptr-1]);

	prepare_read:
		if (is_msglast(i2c)) {
			/* last byte of buffer */

			if (is_lastmsg(i2c))
				s3c24xx_i2c_disable_ack(i2c);
			
		} else if (is_msgend(i2c)) {
			/* ok, we've read the entire buffer, see if there
			 * is anything else we need to do */

			if (is_lastmsg(i2c)) {
				/* last message, send stop and complete */
				DBG(3, "READ: Send Stop\n");

				s3c24xx_i2c_stop(i2c, 0);
			} else {
				/* go to the next transfer */
				DBG(3, "READ: Next Transfer\n");

				i2c->msg_ptr = 0;
				i2c->msg_idx ++;
				i2c->msg++;
			}
		}

		break;
	}

	/* acknowlegde the IRQ and get back on with the work */

 out_ack:
	DBG(3, "out_ack:\n");
	tmp = readl(i2c->regs + S3C2410_IICCON);	
	tmp &= ~S3C2410_IICCON_IRQPEND;
	writel(tmp, i2c->regs + S3C2410_IICCON);
 out:
	return ret;
}
/* s3c24xx_i2c_irq
 *
 * top level IRQ servicing routine
*/

static irqreturn_t s3c24xx_i2c_irq(int irqno, void *dev_id,
			   struct pt_regs *regs)
{
	struct s3c24xx_i2c *i2c = dev_id;
	unsigned long status;
	unsigned long tmp;

	status = readl(i2c->regs + S3C2410_IICSTAT);

	DBG(3, "s3c24xx_i2c_irq: status %08lx\n", status);

	if (status & S3C2410_IICSTAT_ARBITR) {
		// deal with arbitration loss
	}

	if (i2c->state == STATE_IDLE) {
		DBG(3, "IRQ: error i2c->state == IDLE\n");

		tmp = readl(i2c->regs + S3C2410_IICCON);	
		tmp &= ~S3C2410_IICCON_IRQPEND;
		writel(tmp, i2c->regs +  S3C2410_IICCON);
		goto out;
	}
	
	// pretty much this leaves us with the fact that we've
	// transmitted or received whatever byte we last sent

	i2s_s3c_irq_nextbyte(i2c, status);

 out:
	return IRQ_HANDLED;
}


/* s3c24xx_i2c_set_master
 *

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
午夜私人影院久久久久| 91久久精品一区二区二区| 成人av片在线观看| 日本乱人伦aⅴ精品| 国产亚洲成aⅴ人片在线观看 | 在线影院国内精品| www精品美女久久久tv| 亚洲午夜在线观看视频在线| 国产91在线观看丝袜| 日韩三级精品电影久久久 | 国产精品妹子av| 日本系列欧美系列| 在线观看免费一区| 成人欧美一区二区三区黑人麻豆| 免费看欧美女人艹b| 欧美日韩国产免费| 亚洲精品美国一| 成人av网站在线观看| 久久女同性恋中文字幕| 老司机午夜精品99久久| 欧美高清dvd| 五月综合激情网| 欧美视频在线一区二区三区| 亚洲欧美偷拍卡通变态| 成人黄色片在线观看| 国产午夜精品久久久久久免费视| 麻豆国产精品一区二区三区| 欧美一级电影网站| 蜜桃视频在线观看一区| 日韩视频免费观看高清完整版| 天天av天天翘天天综合网色鬼国产 | 亚洲精品乱码久久久久久日本蜜臀| 国产精品亚洲а∨天堂免在线| 日韩欧美国产1| 久久激情五月激情| 欧美v亚洲v综合ⅴ国产v| 蜜桃av一区二区| 欧美一级电影网站| 国产一区二区三区免费在线观看| 久久免费精品国产久精品久久久久 | 成人激情免费电影网址| 国产精品乱码人人做人人爱| 91在线观看一区二区| 综合久久久久综合| 色综合久久88色综合天天6 | 久草在线在线精品观看| 2021国产精品久久精品| 国产成人精品免费视频网站| 亚洲婷婷国产精品电影人久久| 99免费精品在线| 亚洲午夜日本在线观看| 91精品国产91久久综合桃花| 精品亚洲免费视频| 国产精品国产精品国产专区不片| 色国产精品一区在线观看| 天天影视色香欲综合网老头| 久久伊99综合婷婷久久伊| 丁香一区二区三区| 一区av在线播放| 欧美岛国在线观看| 91视频国产观看| 日韩av成人高清| 国产日产欧产精品推荐色 | 日韩美女主播在线视频一区二区三区| 国产精品一区一区三区| 亚洲欧美成人一区二区三区| 欧美日韩国产综合一区二区| 国产一区二区三区在线观看免费视频| 亚洲欧美一区二区三区久本道91| 欧美日韩在线亚洲一区蜜芽| 国产在线国偷精品产拍免费yy| 18成人在线观看| 欧美成人女星排行榜| 99综合电影在线视频| 奇米一区二区三区| 日韩毛片视频在线看| 精品久久久久一区二区国产| 色综合天天综合给合国产| 喷白浆一区二区| 亚洲美女精品一区| 国产欧美精品一区| 欧美一级高清大全免费观看| 色伊人久久综合中文字幕| 久久成人羞羞网站| 亚洲五月六月丁香激情| 国产精品乱码一区二区三区软件| 欧美一区二区三区电影| 欧美制服丝袜第一页| a亚洲天堂av| 国产精品77777竹菊影视小说| 日韩激情av在线| 亚洲成av人影院| 亚洲精品国产品国语在线app| 久久久精品国产免大香伊| 日韩免费一区二区三区在线播放| 欧美影视一区在线| 99riav一区二区三区| 成人美女视频在线看| 国产一区视频网站| 国产揄拍国内精品对白| 美国十次综合导航| 日本大胆欧美人术艺术动态| 亚洲国产日韩a在线播放性色| 亚洲视频在线一区二区| 日韩理论片网站| 国产精品久久久久久久岛一牛影视| 久久久欧美精品sm网站| 久久精品亚洲精品国产欧美| 欧美精品一区二区三区蜜桃| 久久综合色8888| 久久久夜色精品亚洲| 久久免费视频色| 国产喷白浆一区二区三区| 亚洲国产电影在线观看| 国产欧美日韩精品一区| 国产精品无圣光一区二区| 中文字幕第一区综合| 国产精品乱人伦| 亚洲欧美日韩国产综合| 亚洲精品乱码久久久久久久久 | 国产一区二区三区在线观看精品| 国产在线精品一区在线观看麻豆| 激情综合色丁香一区二区| 国内成人免费视频| 国产91精品久久久久久久网曝门| 粉嫩一区二区三区在线看| 成人av网址在线观看| 色综合久久久久综合99| 欧美日韩高清影院| 日韩欧美成人一区二区| 久久久久国产精品麻豆ai换脸 | 久久精品国产77777蜜臀| 国产精品亚洲第一区在线暖暖韩国| 国产精选一区二区三区| 色哟哟亚洲精品| 91精品国产综合久久久久久| 久久综合色婷婷| 亚洲欧美色一区| 美国毛片一区二区| 风间由美性色一区二区三区| 91视视频在线直接观看在线看网页在线看| 91黄色免费版| 日韩亚洲欧美高清| 国产精品久久久久一区 | 亚洲精品你懂的| 免费精品视频在线| av亚洲精华国产精华| 欧美三级电影在线观看| 26uuu精品一区二区三区四区在线| 国产精品丝袜在线| 日产精品久久久久久久性色 | 麻豆国产精品777777在线| 成人黄色综合网站| 欧美精品在线一区二区三区| 国产视频一区不卡| 午夜国产不卡在线观看视频| 国产91精品在线观看| 6080日韩午夜伦伦午夜伦| 中文字幕av资源一区| 日韩国产欧美在线播放| 成人黄色av网站在线| 日韩久久精品一区| 亚洲综合色区另类av| 成人免费视频播放| 日韩欧美一区二区久久婷婷| 亚洲欧美精品午睡沙发| 国产成人免费视| 91精品国产美女浴室洗澡无遮挡| 成人免费在线视频| 国产精品白丝jk黑袜喷水| 7777女厕盗摄久久久| 亚洲黄色小说网站| 波多野结衣亚洲| 2017欧美狠狠色| 免费看精品久久片| 欧美影院一区二区三区| 最新不卡av在线| 风间由美性色一区二区三区| 精品免费视频一区二区| 天天av天天翘天天综合网色鬼国产| 波多野结衣精品在线| 国产日韩精品一区二区三区 | 久久综合久久综合亚洲| 免费观看在线色综合| 91精品国产综合久久香蕉麻豆| 亚洲精品日日夜夜| 99久久精品一区二区| 国产精品毛片大码女人 | 国产毛片精品视频| 日韩免费一区二区| 精品一区二区三区久久| 日韩女优视频免费观看| 麻豆视频一区二区| 日韩一级免费观看| 久久国产三级精品| 精品国产乱码久久久久久久| 久久精品国产77777蜜臀| 精品久久久久久久久久久久包黑料 | 视频一区在线播放| 欧美剧在线免费观看网站 |