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

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

?? buffer.c

?? linux 0.11源代碼
?? C
字號:
/* *  linux/fs/buffer.c * *  (C) 1991  Linus Torvalds *//* *  'buffer.c' implements the buffer-cache functions. Race-conditions have * been avoided by NEVER letting a interrupt change a buffer (except for the * data, of course), but instead letting the caller do it. NOTE! As interrupts * can wake up a caller, some cli-sti sequences are needed to check for * sleep-on-calls. These should be extremely quick, though (I hope). *//* * NOTE! There is one discordant note here: checking floppies for * disk change. This is where it fits best, I think, as it should * invalidate changed floppy-disk-caches. */#include <stdarg.h> #include <linux/config.h>#include <linux/sched.h>#include <linux/kernel.h>#include <asm/system.h>#include <asm/io.h>extern int end;struct buffer_head * start_buffer = (struct buffer_head *) &end;struct buffer_head * hash_table[NR_HASH];static struct buffer_head * free_list;static struct task_struct * buffer_wait = NULL;int NR_BUFFERS = 0;static inline void wait_on_buffer(struct buffer_head * bh){	cli();	while (bh->b_lock)		sleep_on(&bh->b_wait);	sti();}int sys_sync(void){	int i;	struct buffer_head * bh;	sync_inodes();		/* write out inodes into buffers */	bh = start_buffer;	for (i=0 ; i<NR_BUFFERS ; i++,bh++) {		wait_on_buffer(bh);		if (bh->b_dirt)			ll_rw_block(WRITE,bh);	}	return 0;}int sync_dev(int dev){	int i;	struct buffer_head * bh;	bh = start_buffer;	for (i=0 ; i<NR_BUFFERS ; i++,bh++) {		if (bh->b_dev != dev)			continue;		wait_on_buffer(bh);		if (bh->b_dev == dev && bh->b_dirt)			ll_rw_block(WRITE,bh);	}	sync_inodes();	bh = start_buffer;	for (i=0 ; i<NR_BUFFERS ; i++,bh++) {		if (bh->b_dev != dev)			continue;		wait_on_buffer(bh);		if (bh->b_dev == dev && bh->b_dirt)			ll_rw_block(WRITE,bh);	}	return 0;}void inline invalidate_buffers(int dev){	int i;	struct buffer_head * bh;	bh = start_buffer;	for (i=0 ; i<NR_BUFFERS ; i++,bh++) {		if (bh->b_dev != dev)			continue;		wait_on_buffer(bh);		if (bh->b_dev == dev)			bh->b_uptodate = bh->b_dirt = 0;	}}/* * This routine checks whether a floppy has been changed, and * invalidates all buffer-cache-entries in that case. This * is a relatively slow routine, so we have to try to minimize using * it. Thus it is called only upon a 'mount' or 'open'. This * is the best way of combining speed and utility, I think. * People changing diskettes in the middle of an operation deserve * to loose :-) * * NOTE! Although currently this is only for floppies, the idea is * that any additional removable block-device will use this routine, * and that mount/open needn't know that floppies/whatever are * special. */void check_disk_change(int dev){	int i;	if (MAJOR(dev) != 2)		return;	if (!floppy_change(dev & 0x03))		return;	for (i=0 ; i<NR_SUPER ; i++)		if (super_block[i].s_dev == dev)			put_super(super_block[i].s_dev);	invalidate_inodes(dev);	invalidate_buffers(dev);}#define _hashfn(dev,block) (((unsigned)(dev^block))%NR_HASH)#define hash(dev,block) hash_table[_hashfn(dev,block)]static inline void remove_from_queues(struct buffer_head * bh){/* remove from hash-queue */	if (bh->b_next)		bh->b_next->b_prev = bh->b_prev;	if (bh->b_prev)		bh->b_prev->b_next = bh->b_next;	if (hash(bh->b_dev,bh->b_blocknr) == bh)		hash(bh->b_dev,bh->b_blocknr) = bh->b_next;/* remove from free list */	if (!(bh->b_prev_free) || !(bh->b_next_free))		panic("Free block list corrupted");	bh->b_prev_free->b_next_free = bh->b_next_free;	bh->b_next_free->b_prev_free = bh->b_prev_free;	if (free_list == bh)		free_list = bh->b_next_free;}static inline void insert_into_queues(struct buffer_head * bh){/* put at end of free list */	bh->b_next_free = free_list;	bh->b_prev_free = free_list->b_prev_free;	free_list->b_prev_free->b_next_free = bh;	free_list->b_prev_free = bh;/* put the buffer in new hash-queue if it has a device */	bh->b_prev = NULL;	bh->b_next = NULL;	if (!bh->b_dev)		return;	bh->b_next = hash(bh->b_dev,bh->b_blocknr);	hash(bh->b_dev,bh->b_blocknr) = bh;	bh->b_next->b_prev = bh;}static struct buffer_head * find_buffer(int dev, int block){			struct buffer_head * tmp;	for (tmp = hash(dev,block) ; tmp != NULL ; tmp = tmp->b_next)		if (tmp->b_dev==dev && tmp->b_blocknr==block)			return tmp;	return NULL;}/* * Why like this, I hear you say... The reason is race-conditions. * As we don't lock buffers (unless we are readint them, that is), * something might happen to it while we sleep (ie a read-error * will force it bad). This shouldn't really happen currently, but * the code is ready. */struct buffer_head * get_hash_table(int dev, int block){	struct buffer_head * bh;	for (;;) {		if (!(bh=find_buffer(dev,block)))			return NULL;		bh->b_count++;		wait_on_buffer(bh);		if (bh->b_dev == dev && bh->b_blocknr == block)			return bh;		bh->b_count--;	}}/* * Ok, this is getblk, and it isn't very clear, again to hinder * race-conditions. Most of the code is seldom used, (ie repeating), * so it should be much more efficient than it looks. * * The algoritm is changed: hopefully better, and an elusive bug removed. */#define BADNESS(bh) (((bh)->b_dirt<<1)+(bh)->b_lock)struct buffer_head * getblk(int dev,int block){	struct buffer_head * tmp, * bh;repeat:	if (bh = get_hash_table(dev,block))		return bh;	tmp = free_list;	do {		if (tmp->b_count)			continue;		if (!bh || BADNESS(tmp)<BADNESS(bh)) {			bh = tmp;			if (!BADNESS(tmp))				break;		}/* and repeat until we find something good */	} while ((tmp = tmp->b_next_free) != free_list);	if (!bh) {		sleep_on(&buffer_wait);		goto repeat;	}	wait_on_buffer(bh);	if (bh->b_count)		goto repeat;	while (bh->b_dirt) {		sync_dev(bh->b_dev);		wait_on_buffer(bh);		if (bh->b_count)			goto repeat;	}/* NOTE!! While we slept waiting for this block, somebody else might *//* already have added "this" block to the cache. check it */	if (find_buffer(dev,block))		goto repeat;/* OK, FINALLY we know that this buffer is the only one of it's kind, *//* and that it's unused (b_count=0), unlocked (b_lock=0), and clean */	bh->b_count=1;	bh->b_dirt=0;	bh->b_uptodate=0;	remove_from_queues(bh);	bh->b_dev=dev;	bh->b_blocknr=block;	insert_into_queues(bh);	return bh;}void brelse(struct buffer_head * buf){	if (!buf)		return;	wait_on_buffer(buf);	if (!(buf->b_count--))		panic("Trying to free free buffer");	wake_up(&buffer_wait);}/* * bread() reads a specified block and returns the buffer that contains * it. It returns NULL if the block was unreadable. */struct buffer_head * bread(int dev,int block){	struct buffer_head * bh;	if (!(bh=getblk(dev,block)))		panic("bread: getblk returned NULL\n");	if (bh->b_uptodate)		return bh;	ll_rw_block(READ,bh);	wait_on_buffer(bh);	if (bh->b_uptodate)		return bh;	brelse(bh);	return NULL;}#define COPYBLK(from,to) \__asm__("cld\n\t" \	"rep\n\t" \	"movsl\n\t" \	::"c" (BLOCK_SIZE/4),"S" (from),"D" (to) \	:"cx","di","si")/* * bread_page reads four buffers into memory at the desired address. It's * a function of its own, as there is some speed to be got by reading them * all at the same time, not waiting for one to be read, and then another * etc. */void bread_page(unsigned long address,int dev,int b[4]){	struct buffer_head * bh[4];	int i;	for (i=0 ; i<4 ; i++)		if (b[i]) {			if (bh[i] = getblk(dev,b[i]))				if (!bh[i]->b_uptodate)					ll_rw_block(READ,bh[i]);		} else			bh[i] = NULL;	for (i=0 ; i<4 ; i++,address += BLOCK_SIZE)		if (bh[i]) {			wait_on_buffer(bh[i]);			if (bh[i]->b_uptodate)				COPYBLK((unsigned long) bh[i]->b_data,address);			brelse(bh[i]);		}}/* * Ok, breada can be used as bread, but additionally to mark other * blocks for reading as well. End the argument list with a negative * number. */struct buffer_head * breada(int dev,int first, ...){	va_list args;	struct buffer_head * bh, *tmp;	va_start(args,first);	if (!(bh=getblk(dev,first)))		panic("bread: getblk returned NULL\n");	if (!bh->b_uptodate)		ll_rw_block(READ,bh);	while ((first=va_arg(args,int))>=0) {		tmp=getblk(dev,first);		if (tmp) {			if (!tmp->b_uptodate)				ll_rw_block(READA,bh);			tmp->b_count--;		}	}	va_end(args);	wait_on_buffer(bh);	if (bh->b_uptodate)		return bh;	brelse(bh);	return (NULL);}void buffer_init(long buffer_end){	struct buffer_head * h = start_buffer;	void * b;	int i;	if (buffer_end == 1<<20)		b = (void *) (640*1024);	else		b = (void *) buffer_end;	while ( (b -= BLOCK_SIZE) >= ((void *) (h+1)) ) {		h->b_dev = 0;		h->b_dirt = 0;		h->b_count = 0;		h->b_lock = 0;		h->b_uptodate = 0;		h->b_wait = NULL;		h->b_next = NULL;		h->b_prev = NULL;		h->b_data = (char *) b;		h->b_prev_free = h-1;		h->b_next_free = h+1;		h++;		NR_BUFFERS++;		if (b == (void *) 0x100000)			b = (void *) 0xA0000;	}	h--;	free_list = start_buffer;	free_list->b_prev_free = h;	h->b_next_free = free_list;	for (i=0;i<NR_HASH;i++)		hash_table[i]=NULL;}	

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
水野朝阳av一区二区三区| 国产在线看一区| 国产一区二区三区精品视频| 99麻豆久久久国产精品免费| 日韩一级大片在线观看| 欧美国产一区二区| 久久精品国产久精国产| 色婷婷亚洲婷婷| 国产欧美综合在线观看第十页| 亚洲大片在线观看| 99精品欧美一区二区三区小说| 精品精品国产高清一毛片一天堂| 亚洲乱码中文字幕综合| 懂色一区二区三区免费观看| 日韩一区二区三区视频在线观看| 亚洲一区av在线| 91一区二区在线| 国产日韩精品一区二区浪潮av| 日本亚洲天堂网| 欧美色成人综合| 亚洲欧美日韩一区二区三区在线观看| 国产精品资源网站| 日韩精品在线看片z| 首页国产欧美日韩丝袜| 欧美三级日韩在线| 亚洲午夜精品17c| 色婷婷激情综合| 亚洲欧美成aⅴ人在线观看| 国产91精品久久久久久久网曝门| 精品国产乱子伦一区| 日本亚洲免费观看| 欧美一区二区视频在线观看2022 | 精品嫩草影院久久| 9i在线看片成人免费| 日韩精品一区二区三区中文不卡 | 欧美精品一区二区精品网| 日本中文在线一区| 欧美成人aa大片| 九九精品视频在线看| 久久久久高清精品| 国产福利一区二区| 中文字幕一区三区| 色噜噜久久综合| 亚洲国产成人91porn| 欧美日韩国产不卡| 麻豆国产精品官网| 久久香蕉国产线看观看99| 国产成人av一区二区三区在线| 久久精品视频一区二区| 成人深夜福利app| 亚洲免费在线视频一区 二区| 色婷婷久久99综合精品jk白丝| 香蕉成人伊视频在线观看| 91.成人天堂一区| 国产米奇在线777精品观看| 欧美激情一区二区三区全黄| 日本伦理一区二区| 三级久久三级久久| 国产日韩欧美综合在线| 91成人免费网站| 免费黄网站欧美| 国产精品久久精品日日| 欧美色男人天堂| 韩国av一区二区| 亚洲美女视频一区| 日韩你懂的在线观看| 99久精品国产| 麻豆精品视频在线观看| 国产精品久久久久久久久果冻传媒| 一本色道久久综合亚洲aⅴ蜜桃| 热久久免费视频| 综合激情网...| 精品国产乱码久久久久久免费| 91免费国产在线观看| 日韩av电影免费观看高清完整版在线观看| 久久免费午夜影院| 91国偷自产一区二区三区观看| 蜜臀91精品一区二区三区| 中文字幕一区二区三区不卡在线| 欧美精品 日韩| 91在线一区二区| 久草这里只有精品视频| 一级女性全黄久久生活片免费| 欧美变态凌虐bdsm| 欧美色图在线观看| www.亚洲色图.com| 久久国产尿小便嘘嘘| 国产成a人亚洲精| 日本成人在线视频网站| 亚洲同性gay激情无套| 欧美草草影院在线视频| 欧美精品久久久久久久多人混战| 菠萝蜜视频在线观看一区| 麻豆91免费看| 亚洲成人免费视频| 亚洲美女电影在线| 国产精品动漫网站| 久久免费视频一区| 日韩女优av电影在线观看| 欧美亚洲国产一区在线观看网站| 丁香一区二区三区| 国产一区二区三区精品视频| 欧美aaa在线| 毛片基地黄久久久久久天堂| 午夜精品国产更新| 一区二区三区不卡在线观看| 自拍偷拍国产亚洲| 18成人在线视频| 国产精品国产自产拍高清av| 日本一区二区三区四区在线视频| 久久久噜噜噜久久中文字幕色伊伊 | 激情久久久久久久久久久久久久久久| 亚洲一区二区三区在线看| 亚洲精品免费在线播放| 成人欧美一区二区三区1314| 国产精品―色哟哟| 国产精品久久久久久妇女6080| 国产精品免费av| 中文字幕日韩一区| 亚洲视频免费观看| 亚洲美女免费在线| 亚洲成人激情自拍| 偷拍一区二区三区| 日韩成人一区二区| 久久精品99国产精品日本| 国内外成人在线| 国产成人在线视频播放| 成人午夜视频免费看| av欧美精品.com| 欧美在线观看一区| 欧美日韩精品一区二区天天拍小说| 欧美日韩三级在线| 精品三级av在线| 中文字幕av在线一区二区三区| 国产精品久久毛片a| 亚洲综合免费观看高清完整版在线 | 日韩一级免费观看| 久久婷婷色综合| 国产精品乱码一区二区三区软件| 亚洲图片欧美激情| 日本麻豆一区二区三区视频| 久久精品久久精品| 不卡的看片网站| 精品视频一区三区九区| 日韩一二在线观看| 国产精品不卡一区| 日韩中文字幕亚洲一区二区va在线| 捆绑紧缚一区二区三区视频| 粉嫩欧美一区二区三区高清影视| 91啪九色porn原创视频在线观看| 欧美福利视频导航| 久久只精品国产| 一区二区三区四区在线免费观看| 欧美aaa在线| 91在线看国产| 日韩精品一区二区三区老鸭窝 | 日韩精品每日更新| 国产精品18久久久久久久久| 色婷婷久久久综合中文字幕| 精品国产一二三| 亚洲一区在线看| 国产精品99久久久久久久女警| 欧美在线免费视屏| 国产欧美日韩综合精品一区二区| 亚洲一区自拍偷拍| 粉嫩嫩av羞羞动漫久久久| 欧美一区二区三区四区视频 | 美女网站在线免费欧美精品| 成人激情av网| 欧美xxxxx牲另类人与| 亚洲欧美日韩国产综合| 精品一区二区三区免费视频| 欧美主播一区二区三区美女| 国产亚洲欧美日韩俺去了| 日韩高清不卡在线| 色8久久人人97超碰香蕉987| 国产欧美日韩不卡| 国精产品一区一区三区mba桃花| 欧美色爱综合网| 亚洲品质自拍视频| 国产成人午夜高潮毛片| 日韩欧美一区二区在线视频| 一二三四社区欧美黄| 99精品偷自拍| 中文字幕不卡一区| 国产成人午夜高潮毛片| 久久久国际精品| 国内成人免费视频| 欧美变态tickle挠乳网站| 蜜臀久久99精品久久久画质超高清| 欧美综合亚洲图片综合区| 亚洲激情一二三区| 色呦呦国产精品| 亚洲美女免费视频| 日本丰满少妇一区二区三区| 亚洲另类春色校园小说| 91色乱码一区二区三区| 亚洲日本va在线观看| 色婷婷综合久色| 亚洲成人免费影院|