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

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

?? mmc_block.c

?? mmc 記憶棒驅(qū)動(dòng) for linux ,可以看一看
?? C
字號(hào):
/* * Block driver for media (i.e., flash cards) * * Copyright 2002 Hewlett-Packard Company * * Use consistent with the GNU GPL is permitted, * provided that this copyright notice is * preserved in its entirety in all copies and derived works. * * HEWLETT-PACKARD COMPANY MAKES NO WARRANTIES, EXPRESSED OR IMPLIED, * AS TO THE USEFULNESS OR CORRECTNESS OF THIS CODE OR ITS * FITNESS FOR ANY PARTICULAR PURPOSE. * * Many thanks to Alessandro Rubini and Jonathan Corbet! * * Author:  Andrew Christian *          28 May 2002 */#include <linux/moduleparam.h>#include <linux/module.h>#include <linux/init.h>#include <linux/sched.h>#include <linux/kernel.h>#include <linux/fs.h>#include <linux/errno.h>#include <linux/hdreg.h>#include <linux/kdev_t.h>#include <linux/blkdev.h>#include <linux/devfs_fs_kernel.h>#include <linux/mutex.h>#include <linux/mmc/card.h>#include <linux/mmc/protocol.h>#include <asm/system.h>#include <asm/uaccess.h>#include "mmc_queue.h"/* * max 8 partitions per card */#define MMC_SHIFT	3static int major;/* * There is one mmc_blk_data per slot. */struct mmc_blk_data {	spinlock_t	lock;	struct gendisk	*disk;	struct mmc_queue queue;	unsigned int	usage;	unsigned int	block_bits;	unsigned int	read_only;};static DEFINE_MUTEX(open_lock);static struct mmc_blk_data *mmc_blk_get(struct gendisk *disk){	struct mmc_blk_data *md;	mutex_lock(&open_lock);	md = disk->private_data;	if (md && md->usage == 0)		md = NULL;	if (md)		md->usage++;	mutex_unlock(&open_lock);	return md;}static void mmc_blk_put(struct mmc_blk_data *md){	mutex_lock(&open_lock);	md->usage--;	if (md->usage == 0) {		put_disk(md->disk);		mmc_cleanup_queue(&md->queue);		kfree(md);	}	mutex_unlock(&open_lock);}static int mmc_blk_open(struct inode *inode, struct file *filp){	struct mmc_blk_data *md;	int ret = -ENXIO;	md = mmc_blk_get(inode->i_bdev->bd_disk);	if (md) {		if (md->usage == 2)			check_disk_change(inode->i_bdev);		ret = 0;		if ((filp->f_mode & FMODE_WRITE) && md->read_only)			ret = -EROFS;	}	return ret;}static int mmc_blk_release(struct inode *inode, struct file *filp){	struct mmc_blk_data *md = inode->i_bdev->bd_disk->private_data;	mmc_blk_put(md);	return 0;}static intmmc_blk_getgeo(struct block_device *bdev, struct hd_geometry *geo){	geo->cylinders = get_capacity(bdev->bd_disk) / (4 * 16);	geo->heads = 4;	geo->sectors = 16;	return 0;}static struct block_device_operations mmc_bdops = {	.open			= mmc_blk_open,	.release		= mmc_blk_release,	.getgeo			= mmc_blk_getgeo,	.owner			= THIS_MODULE,};struct mmc_blk_request {	struct mmc_request	mrq;	struct mmc_command	cmd;	struct mmc_command	stop;	struct mmc_data		data;};static int mmc_blk_prep_rq(struct mmc_queue *mq, struct request *req){	struct mmc_blk_data *md = mq->data;	int stat = BLKPREP_OK;	/*	 * If we have no device, we haven't finished initialising.	 */	if (!md || !mq->card) {		printk(KERN_ERR "%s: killing request - no device/host\n",		       req->rq_disk->disk_name);		stat = BLKPREP_KILL;	}	return stat;}static int mmc_blk_issue_rq(struct mmc_queue *mq, struct request *req){	struct mmc_blk_data *md = mq->data;	struct mmc_card *card = md->queue.card;	int ret;	if (mmc_card_claim_host(card))		goto cmd_err;	do {		struct mmc_blk_request brq;		struct mmc_command cmd;		memset(&brq, 0, sizeof(struct mmc_blk_request));		brq.mrq.cmd = &brq.cmd;		brq.mrq.data = &brq.data;		brq.cmd.arg = req->sector << 9;		brq.cmd.flags = MMC_RSP_R1 | MMC_CMD_ADTC;		brq.data.timeout_ns = card->csd.tacc_ns * 10;		brq.data.timeout_clks = card->csd.tacc_clks * 10;		brq.data.blksz_bits = md->block_bits;		brq.data.blocks = req->nr_sectors >> (md->block_bits - 9);		brq.stop.opcode = MMC_STOP_TRANSMISSION;		brq.stop.arg = 0;		brq.stop.flags = MMC_RSP_R1B | MMC_CMD_AC;		if (rq_data_dir(req) == READ) {			brq.cmd.opcode = brq.data.blocks > 1 ? MMC_READ_MULTIPLE_BLOCK : MMC_READ_SINGLE_BLOCK;			brq.data.flags |= MMC_DATA_READ;		} else {			brq.cmd.opcode = MMC_WRITE_BLOCK;			brq.data.flags |= MMC_DATA_WRITE;			brq.data.blocks = 1;		}		if (brq.data.blocks > 1) {			brq.data.flags |= MMC_DATA_MULTI;			brq.mrq.stop = &brq.stop;		} else {			brq.mrq.stop = NULL;		}		brq.data.sg = mq->sg;		brq.data.sg_len = blk_rq_map_sg(req->q, req, brq.data.sg);		mmc_wait_for_req(card->host, &brq.mrq);		if (brq.cmd.error) {			printk(KERN_ERR "%s: error %d sending read/write command\n",			       req->rq_disk->disk_name, brq.cmd.error);			goto cmd_err;		}		if (brq.data.error) {			printk(KERN_ERR "%s: error %d transferring data\n",			       req->rq_disk->disk_name, brq.data.error);			goto cmd_err;		}		if (brq.stop.error) {			printk(KERN_ERR "%s: error %d sending stop command\n",			       req->rq_disk->disk_name, brq.stop.error);			goto cmd_err;		}		do {			int err;			cmd.opcode = MMC_SEND_STATUS;			cmd.arg = card->rca << 16;			cmd.flags = MMC_RSP_R1 | MMC_CMD_AC;			err = mmc_wait_for_cmd(card->host, &cmd, 5);			if (err) {				printk(KERN_ERR "%s: error %d requesting status\n",				       req->rq_disk->disk_name, err);				goto cmd_err;			}		} while (!(cmd.resp[0] & R1_READY_FOR_DATA));#if 0		if (cmd.resp[0] & ~0x00000900)			printk(KERN_ERR "%s: status = %08x\n",			       req->rq_disk->disk_name, cmd.resp[0]);		if (mmc_decode_status(cmd.resp))			goto cmd_err;#endif		/*		 * A block was successfully transferred.		 */		spin_lock_irq(&md->lock);		ret = end_that_request_chunk(req, 1, brq.data.bytes_xfered);		if (!ret) {			/*			 * The whole request completed successfully.			 */			add_disk_randomness(req->rq_disk);			blkdev_dequeue_request(req);			end_that_request_last(req, 1);		}		spin_unlock_irq(&md->lock);	} while (ret);	mmc_card_release_host(card);	return 1; cmd_err:	mmc_card_release_host(card);	/*	 * This is a little draconian, but until we get proper	 * error handling sorted out here, its the best we can	 * do - especially as some hosts have no idea how much	 * data was transferred before the error occurred.	 */	spin_lock_irq(&md->lock);	do {		ret = end_that_request_chunk(req, 0,				req->current_nr_sectors << 9);	} while (ret);	add_disk_randomness(req->rq_disk);	blkdev_dequeue_request(req);	end_that_request_last(req, 0);	spin_unlock_irq(&md->lock);	return 0;}#define MMC_NUM_MINORS	(256 >> MMC_SHIFT)static unsigned long dev_use[MMC_NUM_MINORS/(8*sizeof(unsigned long))];static inline int mmc_blk_readonly(struct mmc_card *card){	return mmc_card_readonly(card) ||	       !(card->csd.cmdclass & CCC_BLOCK_WRITE);}static struct mmc_blk_data *mmc_blk_alloc(struct mmc_card *card){	struct mmc_blk_data *md;	int devidx, ret;	devidx = find_first_zero_bit(dev_use, MMC_NUM_MINORS);	if (devidx >= MMC_NUM_MINORS)		return ERR_PTR(-ENOSPC);	__set_bit(devidx, dev_use);	md = kmalloc(sizeof(struct mmc_blk_data), GFP_KERNEL);	if (!md) {		ret = -ENOMEM;		goto out;	}	memset(md, 0, sizeof(struct mmc_blk_data));	/*	 * Set the read-only status based on the supported commands	 * and the write protect switch.	 */	md->read_only = mmc_blk_readonly(card);	/*	 * Figure out a workable block size.  MMC cards have:	 *  - two block sizes, one for read and one for write.	 *  - may support partial reads and/or writes	 *    (allows block sizes smaller than specified)	 */	md->block_bits = card->csd.read_blkbits;	if (card->csd.write_blkbits != card->csd.read_blkbits) {		if (card->csd.write_blkbits < card->csd.read_blkbits &&		    card->csd.read_partial) {			/*			 * write block size is smaller than read block			 * size, but we support partial reads, so choose			 * the smaller write block size.			 */			md->block_bits = card->csd.write_blkbits;		} else if (card->csd.write_blkbits > card->csd.read_blkbits &&			   card->csd.write_partial) {			/*			 * read block size is smaller than write block			 * size, but we support partial writes.  Use read			 * block size.			 */		} else {			/*			 * We don't support this configuration for writes.			 */			printk(KERN_ERR "%s: unable to select block size for "				"writing (rb%u wb%u rp%u wp%u)\n",				md->disk->disk_name,				1 << card->csd.read_blkbits,				1 << card->csd.write_blkbits,				card->csd.read_partial,				card->csd.write_partial);			md->read_only = 1;		}	}	/*	 * Refuse to allow block sizes smaller than 512 bytes.	 */	if (md->block_bits < 9) {		printk(KERN_ERR "%s: unable to support block size %u\n",			mmc_card_id(card), 1 << md->block_bits);		ret = -EINVAL;		goto err_kfree;	}	md->disk = alloc_disk(1 << MMC_SHIFT);	if (md->disk == NULL) {		ret = -ENOMEM;		goto err_kfree;	}	spin_lock_init(&md->lock);	md->usage = 1;	ret = mmc_init_queue(&md->queue, card, &md->lock);	if (ret)		goto err_putdisk;	md->queue.prep_fn = mmc_blk_prep_rq;	md->queue.issue_fn = mmc_blk_issue_rq;	md->queue.data = md;	md->disk->major	= major;	md->disk->first_minor = devidx << MMC_SHIFT;	md->disk->fops = &mmc_bdops;	md->disk->private_data = md;	md->disk->queue = md->queue.queue;	md->disk->driverfs_dev = &card->dev;	/*	 * As discussed on lkml, GENHD_FL_REMOVABLE should:	 *	 * - be set for removable media with permanent block devices	 * - be unset for removable block devices with permanent media	 *	 * Since MMC block devices clearly fall under the second	 * case, we do not set GENHD_FL_REMOVABLE.  Userspace	 * should use the block device creation/destruction hotplug	 * messages to tell when the card is present.	 */	sprintf(md->disk->disk_name, "mmcblk%d", devidx);	sprintf(md->disk->devfs_name, "mmc/blk%d", devidx);	blk_queue_hardsect_size(md->queue.queue, 1 << md->block_bits);	/*	 * The CSD capacity field is in units of read_blkbits.	 * set_capacity takes units of 512 bytes.	 */	set_capacity(md->disk, card->csd.capacity << (card->csd.read_blkbits - 9));	return md; err_putdisk:	put_disk(md->disk); err_kfree:	kfree(md); out:	return ERR_PTR(ret);}static intmmc_blk_set_blksize(struct mmc_blk_data *md, struct mmc_card *card){	struct mmc_command cmd;	int err;	mmc_card_claim_host(card);	cmd.opcode = MMC_SET_BLOCKLEN;	cmd.arg = 1 << md->block_bits;	cmd.flags = MMC_RSP_R1 | MMC_CMD_AC;	err = mmc_wait_for_cmd(card->host, &cmd, 5);	mmc_card_release_host(card);	if (err) {		printk(KERN_ERR "%s: unable to set block size to %d: %d\n",			md->disk->disk_name, cmd.arg, err);		return -EINVAL;	}	return 0;}static int mmc_blk_probe(struct mmc_card *card){	struct mmc_blk_data *md;	int err;	/*	 * Check that the card supports the command class(es) we need.	 */	if (!(card->csd.cmdclass & CCC_BLOCK_READ))		return -ENODEV;	md = mmc_blk_alloc(card);	if (IS_ERR(md))		return PTR_ERR(md);	err = mmc_blk_set_blksize(md, card);	if (err)		goto out;	printk(KERN_INFO "%s: %s %s %lluKiB %s\n",		md->disk->disk_name, mmc_card_id(card), mmc_card_name(card),		(unsigned long long)(get_capacity(md->disk) >> 1),		md->read_only ? "(ro)" : "");	mmc_set_drvdata(card, md);	add_disk(md->disk);	return 0; out:	mmc_blk_put(md);	return err;}static void mmc_blk_remove(struct mmc_card *card){	struct mmc_blk_data *md = mmc_get_drvdata(card);	if (md) {		int devidx;		del_gendisk(md->disk);		/*		 * I think this is needed.		 */		md->disk->queue = NULL;		devidx = md->disk->first_minor >> MMC_SHIFT;		__clear_bit(devidx, dev_use);		mmc_blk_put(md);	}	mmc_set_drvdata(card, NULL);}#ifdef CONFIG_PMstatic int mmc_blk_suspend(struct mmc_card *card, pm_message_t state){	struct mmc_blk_data *md = mmc_get_drvdata(card);	if (md) {		mmc_queue_suspend(&md->queue);	}	return 0;}static int mmc_blk_resume(struct mmc_card *card){	struct mmc_blk_data *md = mmc_get_drvdata(card);	if (md) {		mmc_blk_set_blksize(md, card);		mmc_queue_resume(&md->queue);	}	return 0;}#else#define	mmc_blk_suspend	NULL#define mmc_blk_resume	NULL#endifstatic struct mmc_driver mmc_driver = {	.drv		= {		.name	= "mmcblk",	},	.probe		= mmc_blk_probe,	.remove		= mmc_blk_remove,	.suspend	= mmc_blk_suspend,	.resume		= mmc_blk_resume,};static int __init mmc_blk_init(void){	int res = -ENOMEM;	res = register_blkdev(major, "mmc");	if (res < 0) {		printk(KERN_WARNING "Unable to get major %d for MMC media: %d\n",		       major, res);		goto out;	}	if (major == 0)		major = res;	devfs_mk_dir("mmc");	return mmc_register_driver(&mmc_driver); out:	return res;}static void __exit mmc_blk_exit(void){	mmc_unregister_driver(&mmc_driver);	devfs_remove("mmc");	unregister_blkdev(major, "mmc");}module_init(mmc_blk_init);module_exit(mmc_blk_exit);MODULE_LICENSE("GPL");MODULE_DESCRIPTION("Multimedia Card (MMC) block device driver");module_param(major, int, 0444);MODULE_PARM_DESC(major, "specify the major device number for MMC block driver");

?? 快捷鍵說明

復(fù)制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號(hào) Ctrl + =
減小字號(hào) Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
久久精品夜色噜噜亚洲aⅴ| 亚洲婷婷综合久久一本伊一区| 久久久久久久久久美女| 亚洲特黄一级片| 国内精品不卡在线| 欧美人xxxx| 国产精品你懂的| 国产美女在线精品| 日韩欧美一区二区视频| 久久久国产精品麻豆| 蜜乳av一区二区| 欧美日韩亚洲综合一区二区三区| 国产精品久久午夜夜伦鲁鲁| 国产麻豆91精品| 精品少妇一区二区三区日产乱码 | 成人一区二区视频| 日韩欧美国产综合在线一区二区三区| 亚洲人成精品久久久久久| 高清日韩电视剧大全免费| 欧美成人三级在线| 久久99国产精品久久99| 91精品国产综合久久福利软件 | 欧美一区二视频| 午夜精品久久久久久| 91黄视频在线| 亚洲自拍另类综合| 欧美自拍偷拍一区| 亚洲国产综合人成综合网站| 色噜噜偷拍精品综合在线| 亚洲欧美日本韩国| 日本国产一区二区| 夜夜精品浪潮av一区二区三区 | 顶级嫩模精品视频在线看| 久久网站热最新地址| 国产精品123| 国产精品欧美久久久久无广告 | 久久婷婷国产综合精品青草| 韩国精品一区二区| 精品国产电影一区二区| 国产精品综合在线视频| 亚洲国产精品高清| 91视频在线看| 午夜欧美在线一二页| 日韩精品一区国产麻豆| 精品一区二区三区在线播放视频| 久久久亚洲国产美女国产盗摄| 国产福利一区二区三区视频| 中文幕一区二区三区久久蜜桃| 不卡视频一二三四| 亚洲一区二区视频在线| 欧美一级高清片| 国产黄色精品视频| 亚洲视频一二区| 51精品秘密在线观看| 国产中文字幕一区| 亚洲色欲色欲www在线观看| 欧美日韩在线三级| 国产真实乱子伦精品视频| 日本一区二区三区在线不卡| 91一区二区三区在线观看| 视频一区在线视频| 欧美经典一区二区三区| 日本精品一级二级| 激情偷乱视频一区二区三区| 国产精品久久久久精k8| 欧美顶级少妇做爰| 国产精品66部| 亚洲国产日产av| 久久久夜色精品亚洲| 色婷婷亚洲婷婷| 国产精品一区一区三区| 亚洲国产日韩a在线播放| 91精品国产综合久久蜜臀| 黄色日韩三级电影| 一区二区在线观看视频| 精品美女在线播放| 欧美视频一区二区三区在线观看 | 国产精品成人免费在线| av一区二区三区黑人| 日韩高清不卡一区二区| 亚洲欧美日韩精品久久久久| 欧美成人性福生活免费看| 95精品视频在线| 国产激情精品久久久第一区二区 | 91丝袜高跟美女视频| 精品一区二区三区免费| 一区二区三区精品视频| 久久久久久久一区| 日韩女优制服丝袜电影| 555www色欧美视频| 欧美一a一片一级一片| 国产白丝精品91爽爽久久 | 日韩午夜在线影院| 欧洲精品一区二区三区在线观看| 国产成人超碰人人澡人人澡| 麻豆一区二区三| 亚洲成年人网站在线观看| 亚洲三级在线播放| 国产精品入口麻豆九色| 国产日产亚洲精品系列| 欧美精品一区二区高清在线观看| 91精品国产综合久久久蜜臀图片 | 欧洲精品在线观看| 91麻豆免费看| 91免费看`日韩一区二区| 成人18视频在线播放| 懂色av一区二区三区免费观看| 国产美女一区二区三区| 经典一区二区三区| 精品亚洲成a人| 国产在线国偷精品产拍免费yy| 久久66热re国产| 韩国成人在线视频| 国产一区亚洲一区| 国产jizzjizz一区二区| 福利视频网站一区二区三区| 国产激情一区二区三区| 成人h动漫精品一区二区| 成人av在线资源| 91色婷婷久久久久合中文| 欧美又粗又大又爽| 欧美另类变人与禽xxxxx| 欧美日韩aaa| 精品少妇一区二区三区免费观看| 欧美精品一区二区三区在线 | 色国产综合视频| 精品污污网站免费看| 777精品伊人久久久久大香线蕉| 91麻豆精品91久久久久久清纯| 日韩精品自拍偷拍| 久久毛片高清国产| 亚洲欧洲国产专区| 亚洲va欧美va天堂v国产综合| 热久久国产精品| 成人午夜电影久久影院| 91国在线观看| 日韩午夜三级在线| 国产人成一区二区三区影院| 亚洲男同1069视频| 日本欧美在线看| 成人一区在线观看| 欧美福利电影网| 国产婷婷色一区二区三区四区| 国产精品国产三级国产aⅴ中文| 亚洲一区二区免费视频| 久久er99热精品一区二区| jlzzjlzz亚洲日本少妇| 91精品国产入口| 国产精品视频一二三区| 天堂va蜜桃一区二区三区漫画版| 国产精品香蕉一区二区三区| 欧洲激情一区二区| 久久久久久久国产精品影院| 一区二区三区四区在线播放| 精品一区二区三区不卡| 色综合色综合色综合| 精品久久一区二区三区| 亚洲美女区一区| 国产精品一区二区视频| 欧美视频在线观看一区| 精品粉嫩aⅴ一区二区三区四区| 久久精品一区四区| 午夜精品视频在线观看| 91在线一区二区| 久久久久国产精品厨房| 日韩精品电影一区亚洲| 色综合天天做天天爱| 久久精品在这里| 麻豆免费精品视频| 欧美日韩一区二区三区免费看| 国产精品免费视频一区| 久久av资源网| 欧美一区二区在线视频| 亚洲综合无码一区二区| 99久久婷婷国产综合精品| 久久久99久久| 狠狠色丁香久久婷婷综合丁香| 欧美电影一区二区三区| 亚洲伊人伊色伊影伊综合网| 国产成都精品91一区二区三| 日韩欧美成人一区| 日本成人中文字幕在线视频| 欧美日韩在线观看一区二区 | 波多野结衣中文字幕一区| 精品国产麻豆免费人成网站| 日日摸夜夜添夜夜添国产精品| 日本韩国欧美三级| 亚洲精品免费看| 日本韩国精品在线| 亚洲在线观看免费视频| 色综合久久中文综合久久97| 成人免费在线观看入口| av一二三不卡影片| 最新不卡av在线| 99re8在线精品视频免费播放| 国产精品久久久久aaaa樱花| www.在线成人| 亚洲欧美乱综合| 欧美视频一区二区三区在线观看| 亚洲国产成人tv|