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

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

?? iicdriver.c

?? 在網(wǎng)上搜索到的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一区二区三区免费野_久草精品视频
欧美三级中文字幕| 色av成人天堂桃色av| 精品国产一区二区三区忘忧草| 美日韩黄色大片| 久久一二三国产| 国产成人夜色高潮福利影视| 中文字幕一区在线观看视频| 日本精品一级二级| 天天影视涩香欲综合网| 欧美电影免费提供在线观看| 国产精品一区二区91| 1024成人网| 日韩一区二区三区视频在线| 国产精品一区二区三区四区| 亚洲精品美腿丝袜| 91精品国产高清一区二区三区 | 一本一道综合狠狠老| 亚洲人亚洲人成电影网站色| 欧美日韩激情一区二区三区| 国产在线精品一区二区不卡了| 日本一区二区免费在线观看视频| 色综合久久久久综合体桃花网| 日本欧美在线观看| 国产精品免费视频一区| 欧美日韩国产欧美日美国产精品| 激情欧美日韩一区二区| 夜夜嗨av一区二区三区| 亚洲精品一区二区三区精华液 | 另类专区欧美蜜桃臀第一页| 国产精品麻豆视频| 欧美一级夜夜爽| 成人国产在线观看| 裸体健美xxxx欧美裸体表演| 亚洲日本一区二区三区| 欧美大白屁股肥臀xxxxxx| 91农村精品一区二区在线| 奇米影视7777精品一区二区| 亚洲视频免费观看| 国产女同互慰高潮91漫画| 欧美丰满少妇xxxbbb| 成人av免费在线播放| 久久激情综合网| 亚洲电影在线免费观看| 国产精品青草综合久久久久99| 91精品国产入口在线| 在线免费观看一区| av激情综合网| 国产成人自拍高清视频在线免费播放 | 99久久婷婷国产综合精品电影| 美日韩一区二区| 午夜久久久久久久久| 一区二区三区四区中文字幕| 国产精品免费视频一区| 26uuu成人网一区二区三区| 在线不卡免费欧美| 色婷婷久久一区二区三区麻豆| 国产伦精品一区二区三区免费迷| 天堂蜜桃91精品| 亚洲制服丝袜av| 亚洲男人的天堂av| 国产精品三级久久久久三级| 久久久久97国产精华液好用吗| 日韩欧美卡一卡二| 91精品国产综合久久精品| 在线一区二区三区四区| 91同城在线观看| 成人精品一区二区三区四区 | 国内精品久久久久影院一蜜桃| 五月婷婷综合网| 婷婷开心激情综合| 午夜视频一区二区三区| 亚洲大片一区二区三区| 亚洲mv大片欧洲mv大片精品| 亚洲欧美激情一区二区| 亚洲日本va午夜在线影院| 亚洲欧美日韩在线播放| 亚洲你懂的在线视频| 一区二区三区精品| 亚洲高清在线视频| 日韩中文欧美在线| 蜜臀a∨国产成人精品| 久久99久久久久久久久久久| 激情伊人五月天久久综合| 国产一区啦啦啦在线观看| 国产精品99久久久久久久女警| 国产91在线看| 9i看片成人免费高清| 日本久久一区二区三区| 欧美蜜桃一区二区三区| 欧美www视频| 国产精品色哟哟| 一区二区三区四区亚洲| 亚洲成人综合在线| 久久91精品久久久久久秒播| 国产精品88av| 一本久道中文字幕精品亚洲嫩| 欧美日韩一级片网站| 欧美成人猛片aaaaaaa| 久久久久久久精| 亚洲精品亚洲人成人网 | 亚洲欧洲国产日韩| 一区二区三区不卡在线观看 | 一区二区三区在线影院| 日本欧美一区二区三区乱码| 激情小说亚洲一区| 99re8在线精品视频免费播放| 欧美日韩精品欧美日韩精品一| 日韩精品一区二区三区四区| 中文一区在线播放| 亚洲影院在线观看| 极品少妇一区二区三区精品视频| 成人免费视频免费观看| 欧美体内she精高潮| 亚洲精品一区二区三区在线观看| 国产精品国产精品国产专区不蜜| 亚洲一卡二卡三卡四卡五卡| 精品一区二区综合| 色婷婷狠狠综合| 国产亚洲美州欧州综合国| 亚洲国产综合色| 成人激情av网| 日韩欧美一级二级| 一区二区三区四区不卡视频| 国内精品第一页| 欧美精品一二三| 日韩毛片在线免费观看| 麻豆久久一区二区| 91国偷自产一区二区三区观看| 久久久久青草大香线综合精品| 亚洲电影在线播放| 91在线丨porny丨国产| 欧美videos大乳护士334| 一区二区三区国产精华| 国产91丝袜在线播放0| 日韩欧美一级二级| 午夜伊人狠狠久久| 在线免费视频一区二区| 国产精品久久久久三级| 国产精品99久久久久| 日韩欧美国产麻豆| 石原莉奈在线亚洲三区| 色综合婷婷久久| 国产精品麻豆久久久| 国产一区二区伦理| 日韩精品一区二区三区在线 | 五月天一区二区| 日本二三区不卡| 国产精品久99| 成人禁用看黄a在线| 国产日韩欧美在线一区| 久久99这里只有精品| 日韩无一区二区| 丝袜美腿成人在线| 精品视频免费看| 亚洲一区av在线| 欧美亚洲自拍偷拍| 亚洲午夜三级在线| 欧美综合在线视频| 亚洲国产cao| 欧美精品xxxxbbbb| 日韩在线一区二区三区| 欧美日韩免费视频| 婷婷综合另类小说色区| 欧美精品99久久久**| 日韩国产欧美三级| 欧美一区二区三区免费观看视频| 视频在线观看91| 欧美大胆一级视频| 国产在线精品免费av| 久久综合久久综合九色| 国产麻豆91精品| 国产精品久久久99| 日本精品视频一区二区| 亚洲电影一区二区| 日韩视频一区二区三区| 狠狠色丁香久久婷婷综| 久久亚洲精华国产精华液| 成人一区二区三区视频| 1000精品久久久久久久久| 欧美伊人久久久久久久久影院| 亚洲bdsm女犯bdsm网站| 日韩视频免费观看高清完整版| 日本在线观看不卡视频| www国产成人免费观看视频 深夜成人网 | 精品av久久707| 大白屁股一区二区视频| 亚洲精品视频免费观看| 欧美日本一区二区在线观看| 久热成人在线视频| 国产拍欧美日韩视频二区 | 欧美午夜电影一区| 日本欧美韩国一区三区| 国产亚洲精品免费| 日本久久一区二区三区| 久久疯狂做爰流白浆xx| 中文字幕在线不卡一区二区三区 | 一本色道久久加勒比精品| 午夜激情一区二区| 欧美国产日本韩| 欧美区视频在线观看|