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

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

?? buffer.c,v

?? linux-0.10 對于想了解linux內核,而又不想花太多精力的人,最好就是這種低版本,而具有核心價值的程序
?? C,V
字號:
head     1.2;branch   ;access   ;symbols  ;locks    ; strict;comment  @ * @;1.2date     91.11.20.00.10.40;  author tytso;  state Exp;branches ;next     1.1;1.1date     91.11.12.15.49.42;  author tytso;  state Exp;branches ;next     ;desc@@1.2log@Used new version of buffer.c supplied from Linus.@text@/* *  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;}/* * 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;	struct buffer_head * bh;	if (MAJOR(dev) != 2)		return;	dev=MINOR(dev) & 0x03;	/* which floppy is it? */	if (!floppy_change(dev))		return;	dev |= 0x200;	for (i=0 ; i<NR_SUPER ; i++)		if ((super_block[i].s_dev & 0xff03)==dev)			put_super(super_block[i].s_dev);	bh = start_buffer;	for (i=0 ; i<NR_BUFFERS ; i++,bh++) {		if ((bh->b_dev & 0xff03) != dev)			continue;		wait_on_buffer(bh);		if ((bh->b_dev & 0xff03) == dev)			bh->b_uptodate = bh->b_dirt = 0;	}}#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: better, and an elusive bug removed. *		LBT 11.11.91 */#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;		}	} 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;}/* * 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;}	@1.1log@Initial revision@text@d72 9d196 3d200 1d203 1a203 1	struct buffer_head * tmp;d206 2a207 2	if (tmp=get_hash_table(dev,block))		return tmp;d209 6a214 4	for (;;) {		if (!tmp->b_count) {			wait_on_buffer(tmp);	/* we still have to wait */			if (!tmp->b_count)	/* on it, it might be dirty */d217 12a228 3		tmp = tmp->b_next_free;		if (tmp == free_list) {			sleep_on(&buffer_wait);a229 1		}d231 3a233 23	tmp->b_count++;	remove_from_queues(tmp);/* * Now, when we know nobody can get to this node (as it's removed from the * free list), we write it out. We can sleep here without fear of race- * conditions. */ 	while (tmp->b_dirt) {		sync_dev(tmp->b_dev);		wait_on_buffer(tmp);	}/* update buffer contents */	tmp->b_dev=dev;	tmp->b_blocknr=block;	tmp->b_dirt=0;	tmp->b_uptodate=0;/* NOTE!! While we possibly slept in sync_dev(), somebody else might have *//* added "this" block already, so check for that. Thank God for goto's. */	if (find_buffer(dev,block)) {		tmp->b_dev=0;		/* ok, someone else has beaten us */		tmp->b_blocknr=0;	/* to it - free this block and */		tmp->b_count=0;		/* try again */		insert_into_queues(tmp);d235 10a244 4	}/* and then insert into correct position */	insert_into_queues(tmp);	return tmp;@

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
九九国产精品视频| 欧美午夜不卡视频| 欧美在线三级电影| 国产精品亚洲第一区在线暖暖韩国| 日韩精品乱码免费| 成人av资源站| 色丁香久综合在线久综合在线观看| 色婷婷一区二区| 欧美区在线观看| 国产精品中文字幕一区二区三区| 色综合视频在线观看| 国产精品视频线看| 亚洲一区中文日韩| 97se狠狠狠综合亚洲狠狠| 久久久精品欧美丰满| 久久国产精品一区二区| 欧美美女喷水视频| 国产无一区二区| 久久福利资源站| 91一区二区三区在线观看| 欧美日韩高清在线播放| 精品成人a区在线观看| 亚洲欧美一区二区在线观看| 亚洲成在人线在线播放| 91丝袜呻吟高潮美腿白嫩在线观看| 久久久亚洲综合| 一区二区三区蜜桃网| 毛片av一区二区| 欧美一区二区三区四区在线观看 | 日日噜噜夜夜狠狠视频欧美人| 99国产精品99久久久久久| 国产精品毛片a∨一区二区三区| 狠狠狠色丁香婷婷综合久久五月| 99久久久免费精品国产一区二区| 国产欧美精品一区二区三区四区 | 99国产精品久| 亚洲欧美视频在线观看| 91网站最新网址| 精品国产乱码久久久久久久| 亚洲欧美影音先锋| 在线观看国产一区二区| 欧美国产日韩在线观看| av午夜一区麻豆| 亚洲丝袜自拍清纯另类| 欧美性生活久久| 国产精品短视频| 欧美在线视频日韩| 麻豆国产精品一区二区三区 | 最新久久zyz资源站| 一本久久a久久精品亚洲| 久久精品免费在线观看| 日韩精彩视频在线观看| 99精品桃花视频在线观看| 日韩精品自拍偷拍| 国产成人免费视频| 一区二区三区久久| 成人午夜激情视频| 亚洲大型综合色站| 精品成a人在线观看| 99re66热这里只有精品3直播| 亚洲国产婷婷综合在线精品| 日韩欧美一二区| av男人天堂一区| 日韩国产精品久久| 综合亚洲深深色噜噜狠狠网站| 欧美日韩中文字幕一区| 国产在线视频一区二区| 一区二区三区在线视频免费| 成人av动漫网站| 日韩精品每日更新| 欧美一区二区在线视频| 婷婷综合五月天| 久久久久久黄色| 国产91精品露脸国语对白| 亚洲午夜免费电影| 久久丝袜美腿综合| 日韩国产欧美在线观看| 成人黄色一级视频| 在线播放91灌醉迷j高跟美女| 免费观看日韩av| 午夜一区二区三区在线观看| 久久午夜国产精品| 久久亚洲欧美国产精品乐播| 欧美精品一区二区三区在线| 亚洲精品在线一区二区| 久久精品亚洲麻豆av一区二区 | www.爱久久.com| 视频在线观看一区二区三区| 久久精品人人做| 亚洲欧洲成人精品av97| 欧美一级高清大全免费观看| 日韩免费成人网| 欧美优质美女网站| 成人少妇影院yyyy| 欧美亚洲禁片免费| 日韩欧美综合一区| 国产欧美日韩三区| 日韩女优av电影| 国产日产欧美一区| 亚洲欧美一区二区三区国产精品| 久久麻豆一区二区| 综合网在线视频| 日韩精品亚洲一区| 国产99久久久国产精品免费看| 日本大胆欧美人术艺术动态 | 成人永久免费视频| 激情国产一区二区| 精品国产髙清在线看国产毛片| 久久人人爽爽爽人久久久| 日韩欧美在线123| 欧美丰满美乳xxx高潮www| 91国产免费看| 成人免费黄色大片| 成人禁用看黄a在线| 成人综合在线观看| 懂色av一区二区在线播放| 91日韩在线专区| 日韩午夜激情电影| 日韩理论片网站| 亚洲免费av高清| 一区二区在线看| 韩国理伦片一区二区三区在线播放 | 成人黄色一级视频| 成人av高清在线| 日韩亚洲欧美一区| 成人免费在线播放视频| 青青草91视频| 久久精品999| 国产一区高清在线| 国产成都精品91一区二区三| 欧美日韩大陆一区二区| 91精品国产综合久久久久久久 | 亚洲综合在线电影| 国产成人综合亚洲91猫咪| 国产91高潮流白浆在线麻豆| 欧美日韩亚洲另类| 国产精品第五页| 国产伦精品一区二区三区免费| 国产成人日日夜夜| 精品美女在线观看| 欧美国产日韩在线观看| 麻豆成人91精品二区三区| 欧美在线不卡视频| 亚洲视频在线观看一区| 亚洲成a人v欧美综合天堂| 蜜桃视频一区二区| 欧美日韩国产一级二级| 国产精品自拍av| www国产精品av| 免费在线看一区| 欧美猛男超大videosgay| 亚洲精品高清视频在线观看| 成人在线视频首页| 欧美电影一区二区三区| 日韩美女主播在线视频一区二区三区| 久久久精品人体av艺术| 精品一区二区三区免费视频| 7878成人国产在线观看| 亚洲成a人v欧美综合天堂| 99久久99久久久精品齐齐| 日韩西西人体444www| 亚洲国产精品t66y| 国产电影精品久久禁18| 2021久久国产精品不只是精品 | 精品在线免费观看| 成人激情开心网| 国产精品短视频| 91视频国产资源| 亚洲综合一区在线| 欧美三级电影精品| 免费欧美高清视频| 亚洲精品一区在线观看| 激情综合一区二区三区| 一本色道久久综合精品竹菊| 中文字幕日韩av资源站| 色综合久久66| 午夜激情一区二区| 精品区一区二区| 亚洲va欧美va人人爽午夜| 欧美乱妇15p| 精品在线播放免费| 国产精品欧美久久久久无广告| 麻豆成人久久精品二区三区小说| 91蝌蚪porny九色| 亚洲第一综合色| 日韩欧美一区二区免费| 亚洲一区二区美女| 欧美一区在线视频| 国产一区二区女| 尤物视频一区二区| 成人午夜在线免费| 一区二区三区小说| 日韩一级片在线观看| 婷婷成人综合网| 欧美三级中文字幕在线观看| 免费成人在线观看视频| 国产精品国产三级国产专播品爱网 | 日韩欧美中文一区二区| 同产精品九九九| 欧美日韩一区二区在线观看 |