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

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關(guān)于我們
? 蟲蟲下載站

?? cfi_cmdset_0020.c

?? 根據(jù)fs2410移植過后的mtd驅(qū)動(dòng)源碼
?? C
?? 第 1 頁 / 共 3 頁
字號(hào):
		}		/* Latency issues. Drop the lock, wait a while and retry */		spin_unlock_bh(chip->mutex);		cfi_udelay(1);		goto retry;	default:		/* Stick ourselves on a wait queue to be woken when		   someone changes the status */		set_current_state(TASK_UNINTERRUPTIBLE);		add_wait_queue(&chip->wq, &wait);		spin_unlock_bh(chip->mutex);		schedule();		remove_wait_queue(&chip->wq, &wait);		timeo = jiffies + HZ;		goto retry;	}	ENABLE_VPP(map);	map_write(map, CMD(0xe8), cmd_adr);	chip->state = FL_WRITING_TO_BUFFER;	z = 0;	for (;;) {		status = map_read(map, cmd_adr);		if (map_word_andequal(map, status, status_OK, status_OK))			break;		spin_unlock_bh(chip->mutex);		cfi_udelay(1);		spin_lock_bh(chip->mutex);		if (++z > 100) {			/* Argh. Not ready for write to buffer */			DISABLE_VPP(map);                        map_write(map, CMD(0x70), cmd_adr);			chip->state = FL_STATUS;			spin_unlock_bh(chip->mutex);			printk(KERN_ERR "Chip not ready for buffer write. Xstatus = %lx\n", status.x[0]);			return -EIO;		}	}	/* Write length of data to come */	map_write(map, CMD(len/map_bankwidth(map)-1), cmd_adr );	/* Write data */	for (z = 0; z < len;	     z += map_bankwidth(map), buf += map_bankwidth(map)) {		map_word d;		d = map_word_load(map, buf);		map_write(map, d, adr+z);	}	/* GO GO GO */	map_write(map, CMD(0xd0), cmd_adr);	chip->state = FL_WRITING;	spin_unlock_bh(chip->mutex);	cfi_udelay(chip->buffer_write_time);	spin_lock_bh(chip->mutex);	timeo = jiffies + (HZ/2);	z = 0;	for (;;) {		if (chip->state != FL_WRITING) {			/* Someone's suspended the write. Sleep */			set_current_state(TASK_UNINTERRUPTIBLE);			add_wait_queue(&chip->wq, &wait);			spin_unlock_bh(chip->mutex);			schedule();			remove_wait_queue(&chip->wq, &wait);			timeo = jiffies + (HZ / 2); /* FIXME */			spin_lock_bh(chip->mutex);			continue;		}		status = map_read(map, cmd_adr);		if (map_word_andequal(map, status, status_OK, status_OK))			break;		/* OK Still waiting */		if (time_after(jiffies, timeo)) {                        /* clear status */                        map_write(map, CMD(0x50), cmd_adr);                        /* put back into read status register mode */                        map_write(map, CMD(0x70), adr);			chip->state = FL_STATUS;			DISABLE_VPP(map);			spin_unlock_bh(chip->mutex);			printk(KERN_ERR "waiting for chip to be ready timed out in bufwrite\n");			return -EIO;		}		/* Latency issues. Drop the lock, wait a while and retry */		spin_unlock_bh(chip->mutex);		cfi_udelay(1);		z++;		spin_lock_bh(chip->mutex);	}	if (!z) {		chip->buffer_write_time--;		if (!chip->buffer_write_time)			chip->buffer_write_time++;	}	if (z > 1)		chip->buffer_write_time++;	/* Done and happy. */	DISABLE_VPP(map);	chip->state = FL_STATUS;        /* check for errors: 'lock bit', 'VPP', 'dead cell'/'unerased cell' or 'incorrect cmd' -- saw */        if (map_word_bitsset(map, status, CMD(0x3a))) {#ifdef DEBUG_CFI_FEATURES		printk("%s: 2 status[%lx]\n", __FUNCTION__, status.x[0]);#endif		/* clear status */		map_write(map, CMD(0x50), cmd_adr);		/* put back into read status register mode */		map_write(map, CMD(0x70), adr);		wake_up(&chip->wq);		spin_unlock_bh(chip->mutex);		return map_word_bitsset(map, status, CMD(0x02)) ? -EROFS : -EIO;	}	wake_up(&chip->wq);	spin_unlock_bh(chip->mutex);        return 0;}static int cfi_staa_write_buffers (struct mtd_info *mtd, loff_t to,				       size_t len, size_t *retlen, const u_char *buf){	struct map_info *map = mtd->priv;	struct cfi_private *cfi = map->fldrv_priv;	int wbufsize = cfi_interleave(cfi) << cfi->cfiq->MaxBufWriteSize;	int ret = 0;	int chipnum;	unsigned long ofs;	*retlen = 0;	if (!len)		return 0;	chipnum = to >> cfi->chipshift;	ofs = to  - (chipnum << cfi->chipshift);#ifdef DEBUG_CFI_FEATURES        printk("%s: map_bankwidth(map)[%x]\n", __FUNCTION__, map_bankwidth(map));        printk("%s: chipnum[%x] wbufsize[%x]\n", __FUNCTION__, chipnum, wbufsize);        printk("%s: ofs[%x] len[%x]\n", __FUNCTION__, ofs, len);#endif        /* Write buffer is worth it only if more than one word to write... */        while (len > 0) {		/* We must not cross write block boundaries */		int size = wbufsize - (ofs & (wbufsize-1));                if (size > len)                    size = len;                ret = do_write_buffer(map, &cfi->chips[chipnum],				      ofs, buf, size);		if (ret)			return ret;		ofs += size;		buf += size;		(*retlen) += size;		len -= size;		if (ofs >> cfi->chipshift) {			chipnum ++;			ofs = 0;			if (chipnum == cfi->numchips)				return 0;		}	}	return 0;}/* * Writev for ECC-Flashes is a little more complicated. We need to maintain * a small buffer for this. * XXX: If the buffer size is not a multiple of 2, this will break */#define ECCBUF_SIZE (mtd->eccsize)#define ECCBUF_DIV(x) ((x) & ~(ECCBUF_SIZE - 1))#define ECCBUF_MOD(x) ((x) &  (ECCBUF_SIZE - 1))static intcfi_staa_writev(struct mtd_info *mtd, const struct kvec *vecs,		unsigned long count, loff_t to, size_t *retlen){	unsigned long i;	size_t	 totlen = 0, thislen;	int	 ret = 0;	size_t	 buflen = 0;	static char *buffer;	if (!ECCBUF_SIZE) {		/* We should fall back to a general writev implementation.		 * Until that is written, just break.		 */		return -EIO;	}	buffer = kmalloc(ECCBUF_SIZE, GFP_KERNEL);	if (!buffer)		return -ENOMEM;	for (i=0; i<count; i++) {		size_t elem_len = vecs[i].iov_len;		void *elem_base = vecs[i].iov_base;		if (!elem_len) /* FIXME: Might be unnecessary. Check that */			continue;		if (buflen) { /* cut off head */			if (buflen + elem_len < ECCBUF_SIZE) { /* just accumulate */				memcpy(buffer+buflen, elem_base, elem_len);				buflen += elem_len;				continue;			}			memcpy(buffer+buflen, elem_base, ECCBUF_SIZE-buflen);			ret = mtd->write(mtd, to, ECCBUF_SIZE, &thislen, buffer);			totlen += thislen;			if (ret || thislen != ECCBUF_SIZE)				goto write_error;			elem_len -= thislen-buflen;			elem_base += thislen-buflen;			to += ECCBUF_SIZE;		}		if (ECCBUF_DIV(elem_len)) { /* write clean aligned data */			ret = mtd->write(mtd, to, ECCBUF_DIV(elem_len), &thislen, elem_base);			totlen += thislen;			if (ret || thislen != ECCBUF_DIV(elem_len))				goto write_error;			to += thislen;		}		buflen = ECCBUF_MOD(elem_len); /* cut off tail */		if (buflen) {			memset(buffer, 0xff, ECCBUF_SIZE);			memcpy(buffer, elem_base + thislen, buflen);		}	}	if (buflen) { /* flush last page, even if not full */		/* This is sometimes intended behaviour, really */		ret = mtd->write(mtd, to, buflen, &thislen, buffer);		totlen += thislen;		if (ret || thislen != ECCBUF_SIZE)			goto write_error;	}write_error:	if (retlen)		*retlen = totlen;	kfree(buffer);	return ret;}static inline int do_erase_oneblock(struct map_info *map, struct flchip *chip, unsigned long adr){	struct cfi_private *cfi = map->fldrv_priv;	map_word status, status_OK;	unsigned long timeo;	int retries = 3;	DECLARE_WAITQUEUE(wait, current);	int ret = 0;	adr += chip->start;	/* Let's determine this according to the interleave only once */	status_OK = CMD(0x80);	timeo = jiffies + HZ;retry:	spin_lock_bh(chip->mutex);	/* Check that the chip's ready to talk to us. */	switch (chip->state) {	case FL_CFI_QUERY:	case FL_JEDEC_QUERY:	case FL_READY:		map_write(map, CMD(0x70), adr);		chip->state = FL_STATUS;	case FL_STATUS:		status = map_read(map, adr);		if (map_word_andequal(map, status, status_OK, status_OK))			break;		/* Urgh. Chip not yet ready to talk to us. */		if (time_after(jiffies, timeo)) {			spin_unlock_bh(chip->mutex);			printk(KERN_ERR "waiting for chip to be ready timed out in erase\n");			return -EIO;		}		/* Latency issues. Drop the lock, wait a while and retry */		spin_unlock_bh(chip->mutex);		cfi_udelay(1);		goto retry;	default:		/* Stick ourselves on a wait queue to be woken when		   someone changes the status */		set_current_state(TASK_UNINTERRUPTIBLE);		add_wait_queue(&chip->wq, &wait);		spin_unlock_bh(chip->mutex);		schedule();		remove_wait_queue(&chip->wq, &wait);		timeo = jiffies + HZ;		goto retry;	}	ENABLE_VPP(map);	/* Clear the status register first */	map_write(map, CMD(0x50), adr);	/* Now erase */	map_write(map, CMD(0x20), adr);	map_write(map, CMD(0xD0), adr);	chip->state = FL_ERASING;	spin_unlock_bh(chip->mutex);	msleep(1000);	spin_lock_bh(chip->mutex);	/* FIXME. Use a timer to check this, and return immediately. */	/* Once the state machine's known to be working I'll do that */	timeo = jiffies + (HZ*20);	for (;;) {		if (chip->state != FL_ERASING) {			/* Someone's suspended the erase. Sleep */			set_current_state(TASK_UNINTERRUPTIBLE);			add_wait_queue(&chip->wq, &wait);			spin_unlock_bh(chip->mutex);			schedule();			remove_wait_queue(&chip->wq, &wait);			timeo = jiffies + (HZ*20); /* FIXME */			spin_lock_bh(chip->mutex);			continue;		}		status = map_read(map, adr);		if (map_word_andequal(map, status, status_OK, status_OK))			break;		/* OK Still waiting */		if (time_after(jiffies, timeo)) {			map_write(map, CMD(0x70), adr);			chip->state = FL_STATUS;			printk(KERN_ERR "waiting for erase to complete timed out. Xstatus = %lx, status = %lx.\n", status.x[0], map_read(map, adr).x[0]);			DISABLE_VPP(map);			spin_unlock_bh(chip->mutex);			return -EIO;		}		/* Latency issues. Drop the lock, wait a while and retry */		spin_unlock_bh(chip->mutex);		cfi_udelay(1);		spin_lock_bh(chip->mutex);	}	DISABLE_VPP(map);	ret = 0;	/* We've broken this before. It doesn't hurt to be safe */	map_write(map, CMD(0x70), adr);	chip->state = FL_STATUS;	status = map_read(map, adr);	/* check for lock bit */	if (map_word_bitsset(map, status, CMD(0x3a))) {		unsigned char chipstatus = status.x[0];		if (!map_word_equal(map, status, CMD(chipstatus))) {			int i, w;			for (w=0; w<map_words(map); w++) {				for (i = 0; i<cfi_interleave(cfi); i++) {					chipstatus |= status.x[w] >> (cfi->device_type * 8);				}			}			printk(KERN_WARNING "Status is not identical for all chips: 0x%lx. Merging to give 0x%02x\n",			       status.x[0], chipstatus);		}		/* Reset the error bits */		map_write(map, CMD(0x50), adr);		map_write(map, CMD(0x70), adr);		if ((chipstatus & 0x30) == 0x30) {			printk(KERN_NOTICE "Chip reports improper command sequence: status 0x%x\n", chipstatus);			ret = -EIO;		} else if (chipstatus & 0x02) {			/* Protection bit set */			ret = -EROFS;		} else if (chipstatus & 0x8) {			/* Voltage */			printk(KERN_WARNING "Chip reports voltage low on erase: status 0x%x\n", chipstatus);			ret = -EIO;		} else if (chipstatus & 0x20) {			if (retries--) {				printk(KERN_DEBUG "Chip erase failed at 0x%08lx: status 0x%x. Retrying...\n", adr, chipstatus);				timeo = jiffies + HZ;				chip->state = FL_STATUS;				spin_unlock_bh(chip->mutex);				goto retry;			}			printk(KERN_DEBUG "Chip erase failed at 0x%08lx: status 0x%x\n", adr, chipstatus);			ret = -EIO;		}	}	wake_up(&chip->wq);	spin_unlock_bh(chip->mutex);	return ret;}int cfi_staa_erase_varsize(struct mtd_info *mtd, struct erase_info *instr){	struct map_info *map = mtd->priv;	struct cfi_private *cfi = map->fldrv_priv;	unsigned long adr, len;	int chipnum, ret = 0;	int i, first;	struct mtd_erase_region_info *regions = mtd->eraseregions;	if (instr->addr > mtd->size)		return -EINVAL;	if ((instr->len + instr->addr) > mtd->size)		return -EINVAL;	/* Check that both start and end of the requested erase are	 * aligned with the erasesize at the appropriate addresses.	 */	i = 0;	/* Skip all erase regions which are ended before the start of	   the requested erase. Actually, to save on the calculations,	   we skip to the first erase region which starts after the	   start of the requested erase, and then go back one.	*/	while (i < mtd->numeraseregions && instr->addr >= regions[i].offset)	       i++;	i--;	/* OK, now i is pointing at the erase region in which this	   erase request starts. Check the start of the requested	   erase range is aligned with the erase size which is in	   effect here.	*/	if (instr->addr & (regions[i].erasesize-1))		return -EINVAL;	/* Remember the erase region we start on */	first = i;	/* Next, check that the end of the requested erase is aligned	 * with the erase region at that address.	 */	while (i<mtd->numeraseregions && (instr->addr + instr->len) >= regions[i].offset)		i++;	/* As before, drop back one to point at the region in which	   the address actually falls	*/	i--;	if ((instr->addr + instr->len) & (regions[i].erasesize-1))		return -EINVAL;	chipnum = instr->addr >> cfi->chipshift;	adr = instr->addr - (chipnum << cfi->chipshift);	len = instr->len;

?? 快捷鍵說明

復(fù)制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號(hào) Ctrl + =
減小字號(hào) Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
粉嫩aⅴ一区二区三区四区| 久久久99久久精品欧美| 久久综合色8888| 亚洲欧洲99久久| 蜜桃传媒麻豆第一区在线观看| 九九国产精品视频| 91看片淫黄大片一级在线观看| 日韩视频在线你懂得| 最新久久zyz资源站| 狠狠色丁香久久婷婷综合丁香| 99天天综合性| 精品成人佐山爱一区二区| 亚洲综合色网站| 六月婷婷色综合| 在线看国产日韩| 日韩精品一区二区三区在线 | 亚洲久草在线视频| 另类小说综合欧美亚洲| 色哟哟在线观看一区二区三区| 精品免费一区二区三区| 亚洲综合免费观看高清完整版| 国产一区二区美女| 日本精品一级二级| 久久精品水蜜桃av综合天堂| 日本午夜精品一区二区三区电影| 国产成人av电影在线观看| 91国偷自产一区二区开放时间| 久久久青草青青国产亚洲免观| 香蕉av福利精品导航| 国产不卡免费视频| 日韩欧美国产麻豆| 日韩精品一二区| 在线观看三级视频欧美| 一区在线观看视频| 国产白丝精品91爽爽久久| 欧美一级xxx| 日韩精品亚洲专区| 色悠久久久久综合欧美99| 国产精品理伦片| 国产成人免费视频精品含羞草妖精| 精品久久久网站| 久久精品国产一区二区| 精品久久久网站| 国产精品18久久久久久久网站| 精品国产区一区| 国产一区二区免费看| 欧美激情一区不卡| av男人天堂一区| 亚洲综合激情网| 欧美精品一二三| 麻豆成人av在线| 久久久久高清精品| 成人高清伦理免费影院在线观看| 国产精品视频第一区| 一本久久a久久免费精品不卡| 亚洲欧美激情小说另类| 欧美午夜一区二区三区| 青青草视频一区| 精品乱码亚洲一区二区不卡| 国产成人自拍网| 成人免费在线视频| 欧美伊人久久久久久久久影院 | 视频精品一区二区| 91精品国产色综合久久ai换脸| 美国av一区二区| 中文字幕第一区二区| 色欧美片视频在线观看在线视频| 香蕉久久一区二区不卡无毒影院| 欧美一区二区三区在线观看视频 | 亚洲国产日韩精品| 日韩欧美你懂的| 国产成a人亚洲| 亚洲男同性视频| 欧美一区二区视频网站| 国产超碰在线一区| 一区二区在线观看视频在线观看| 欧美精品色一区二区三区| 国精产品一区一区三区mba桃花 | 日韩亚洲欧美高清| 成人一区二区三区视频 | 久久精品视频免费| 在线亚洲一区二区| 伦理电影国产精品| 亚洲色图一区二区三区| 欧美麻豆精品久久久久久| 国产最新精品免费| 亚洲另类一区二区| 精品少妇一区二区三区在线视频| 成人高清免费观看| 日本欧美一区二区三区| 国产精品色眯眯| 欧美色综合久久| 国产成人久久精品77777最新版本 国产成人鲁色资源国产91色综 | 精品一区二区日韩| 亚洲三级免费电影| 日韩精品一区在线| 91久久精品日日躁夜夜躁欧美| 琪琪久久久久日韩精品| 亚洲少妇30p| 26uuu另类欧美亚洲曰本| 欧美在线视频全部完| 国产精品一区二区三区四区| 亚洲午夜一区二区| 日本一区二区三区久久久久久久久不 | 国产精品一二三在| 五月婷婷激情综合| 国产精品久线在线观看| 欧美一二三区在线观看| 91精品国产色综合久久不卡蜜臀| 成人中文字幕电影| 成人app网站| 另类欧美日韩国产在线| 亚洲精品高清在线| 国产视频一区二区在线| 91精品国产91久久久久久一区二区| 成人丝袜18视频在线观看| 美日韩一级片在线观看| 亚洲国产一二三| 中文字幕中文字幕一区| 久久亚洲精精品中文字幕早川悠里 | 精品美女一区二区三区| 欧美日本乱大交xxxxx| 不卡高清视频专区| 国产精华液一区二区三区| 青椒成人免费视频| 天堂成人国产精品一区| 亚洲美女视频一区| 国产精品嫩草影院com| www国产成人免费观看视频 深夜成人网| 欧美唯美清纯偷拍| 91麻豆视频网站| 成人精品鲁一区一区二区| 国产真实精品久久二三区| 蜜臀av性久久久久蜜臀aⅴ | 久久一留热品黄| 欧美一级一级性生活免费录像| 在线观看日韩高清av| 一本大道av伊人久久综合| 粉嫩高潮美女一区二区三区| 狠狠色丁香婷综合久久| 捆绑变态av一区二区三区| 视频一区中文字幕| 亚洲h在线观看| 亚洲福利视频三区| 一区二区三区在线不卡| 亚洲少妇最新在线视频| 亚洲色欲色欲www在线观看| 国产精品九色蝌蚪自拍| 国产精品成人一区二区三区夜夜夜| 欧美极品xxx| 欧美国产一区二区在线观看| 久久色成人在线| 亚洲精品在线三区| 久久婷婷综合激情| 久久精品视频一区二区三区| 国产亚洲一区二区在线观看| 久久久久成人黄色影片| 日本一区二区三区国色天香 | 国产激情一区二区三区四区 | 欧美体内she精高潮| 一区二区高清在线| 欧美日韩五月天| 国产精品中文字幕欧美| 久久精品视频免费| 91麻豆精品秘密| 日韩高清不卡在线| 2021久久国产精品不只是精品| 国模少妇一区二区三区| 中文字幕亚洲一区二区va在线| 在线观看日韩高清av| 欧美日韩久久久久久| 亚洲人成影院在线观看| 中文字幕在线观看一区| 亚洲日本电影在线| 亚洲一区二区三区小说| 日日噜噜夜夜狠狠视频欧美人| 美腿丝袜在线亚洲一区| 国产河南妇女毛片精品久久久| 国产福利一区二区三区视频在线| 福利电影一区二区三区| 色香蕉成人二区免费| 欧美日韩国产大片| 日韩精品一区二区三区在线观看| 国产婷婷一区二区| 亚洲在线观看免费视频| 日韩成人一级片| 国产精品影视天天线| 99久久久免费精品国产一区二区| 欧美探花视频资源| 精品国产在天天线2019| 中文字幕一区二区三区精华液| 一区二区三区不卡视频在线观看| 日韩激情视频在线观看| 国产精品18久久久久久久网站| www.激情成人| 在线成人av网站| 久久九九久精品国产免费直播| 《视频一区视频二区| 青青草精品视频| www.色综合.com|