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

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

?? svclock.c

?? Linux中關于遠程文件鎖定的支持的源代碼
?? C
?? 第 1 頁 / 共 2 頁
字號:
/* * linux/fs/lockd/svclock.c * * Handling of server-side locks, mostly of the blocked variety. * This is the ugliest part of lockd because we tread on very thin ice. * GRANT and CANCEL calls may get stuck, meet in mid-flight, etc. * IMNSHO introducing the grant callback into the NLM protocol was one * of the worst ideas Sun ever had. Except maybe for the idea of doing * NFS file locking at all. * * I'm trying hard to avoid race conditions by protecting most accesses * to a file's list of blocked locks through a semaphore. The global * list of blocked locks is not protected in this fashion however. * Therefore, some functions (such as the RPC callback for the async grant * call) move blocked locks towards the head of the list *while some other * process might be traversing it*. This should not be a problem in * practice, because this will only cause functions traversing the list * to visit some blocks twice. * * Copyright (C) 1996, Olaf Kirch <okir@monad.swb.de> */#include <linux/types.h>#include <linux/errno.h>#include <linux/kernel.h>#include <linux/sched.h>#include <linux/smp_lock.h>#include <linux/sunrpc/clnt.h>#include <linux/sunrpc/svc.h>#include <linux/lockd/nlm.h>#include <linux/lockd/lockd.h>#include <linux/kthread.h>#define NLMDBG_FACILITY		NLMDBG_SVCLOCK#ifdef CONFIG_LOCKD_V4#define nlm_deadlock	nlm4_deadlock#else#define nlm_deadlock	nlm_lck_denied#endifstatic void nlmsvc_release_block(struct nlm_block *block);static void	nlmsvc_insert_block(struct nlm_block *block, unsigned long);static void	nlmsvc_remove_block(struct nlm_block *block);static int nlmsvc_setgrantargs(struct nlm_rqst *call, struct nlm_lock *lock);static void nlmsvc_freegrantargs(struct nlm_rqst *call);static const struct rpc_call_ops nlmsvc_grant_ops;/* * The list of blocked locks to retry */static LIST_HEAD(nlm_blocked);/* * Insert a blocked lock into the global list */static voidnlmsvc_insert_block(struct nlm_block *block, unsigned long when){	struct nlm_block *b;	struct list_head *pos;	dprintk("lockd: nlmsvc_insert_block(%p, %ld)\n", block, when);	if (list_empty(&block->b_list)) {		kref_get(&block->b_count);	} else {		list_del_init(&block->b_list);	}	pos = &nlm_blocked;	if (when != NLM_NEVER) {		if ((when += jiffies) == NLM_NEVER)			when ++;		list_for_each(pos, &nlm_blocked) {			b = list_entry(pos, struct nlm_block, b_list);			if (time_after(b->b_when,when) || b->b_when == NLM_NEVER)				break;		}		/* On normal exit from the loop, pos == &nlm_blocked,		 * so we will be adding to the end of the list - good		 */	}	list_add_tail(&block->b_list, pos);	block->b_when = when;}/* * Remove a block from the global list */static inline voidnlmsvc_remove_block(struct nlm_block *block){	if (!list_empty(&block->b_list)) {		list_del_init(&block->b_list);		nlmsvc_release_block(block);	}}/* * Find a block for a given lock */static struct nlm_block *nlmsvc_lookup_block(struct nlm_file *file, struct nlm_lock *lock){	struct nlm_block	*block;	struct file_lock	*fl;	dprintk("lockd: nlmsvc_lookup_block f=%p pd=%d %Ld-%Ld ty=%d\n",				file, lock->fl.fl_pid,				(long long)lock->fl.fl_start,				(long long)lock->fl.fl_end, lock->fl.fl_type);	list_for_each_entry(block, &nlm_blocked, b_list) {		fl = &block->b_call->a_args.lock.fl;		dprintk("lockd: check f=%p pd=%d %Ld-%Ld ty=%d cookie=%s\n",				block->b_file, fl->fl_pid,				(long long)fl->fl_start,				(long long)fl->fl_end, fl->fl_type,				nlmdbg_cookie2a(&block->b_call->a_args.cookie));		if (block->b_file == file && nlm_compare_locks(fl, &lock->fl)) {			kref_get(&block->b_count);			return block;		}	}	return NULL;}static inline int nlm_cookie_match(struct nlm_cookie *a, struct nlm_cookie *b){	if (a->len != b->len)		return 0;	if (memcmp(a->data, b->data, a->len))		return 0;	return 1;}/* * Find a block with a given NLM cookie. */static inline struct nlm_block *nlmsvc_find_block(struct nlm_cookie *cookie){	struct nlm_block *block;	list_for_each_entry(block, &nlm_blocked, b_list) {		if (nlm_cookie_match(&block->b_call->a_args.cookie,cookie))			goto found;	}	return NULL;found:	dprintk("nlmsvc_find_block(%s): block=%p\n", nlmdbg_cookie2a(cookie), block);	kref_get(&block->b_count);	return block;}/* * Create a block and initialize it. * * Note: we explicitly set the cookie of the grant reply to that of * the blocked lock request. The spec explicitly mentions that the client * should _not_ rely on the callback containing the same cookie as the * request, but (as I found out later) that's because some implementations * do just this. Never mind the standards comittees, they support our * logging industries. * * 10 years later: I hope we can safely ignore these old and broken * clients by now. Let's fix this so we can uniquely identify an incoming * GRANTED_RES message by cookie, without having to rely on the client's IP * address. --okir */static struct nlm_block *nlmsvc_create_block(struct svc_rqst *rqstp, struct nlm_host *host,		    struct nlm_file *file, struct nlm_lock *lock,		    struct nlm_cookie *cookie){	struct nlm_block	*block;	struct nlm_rqst		*call = NULL;	nlm_get_host(host);	call = nlm_alloc_call(host);	if (call == NULL)		return NULL;	/* Allocate memory for block, and initialize arguments */	block = kzalloc(sizeof(*block), GFP_KERNEL);	if (block == NULL)		goto failed;	kref_init(&block->b_count);	INIT_LIST_HEAD(&block->b_list);	INIT_LIST_HEAD(&block->b_flist);	if (!nlmsvc_setgrantargs(call, lock))		goto failed_free;	/* Set notifier function for VFS, and init args */	call->a_args.lock.fl.fl_flags |= FL_SLEEP;	call->a_args.lock.fl.fl_lmops = &nlmsvc_lock_operations;	nlmclnt_next_cookie(&call->a_args.cookie);	dprintk("lockd: created block %p...\n", block);	/* Create and initialize the block */	block->b_daemon = rqstp->rq_server;	block->b_host   = host;	block->b_file   = file;	block->b_fl = NULL;	file->f_count++;	/* Add to file's list of blocks */	list_add(&block->b_flist, &file->f_blocks);	/* Set up RPC arguments for callback */	block->b_call = call;	call->a_flags   = RPC_TASK_ASYNC;	call->a_block = block;	return block;failed_free:	kfree(block);failed:	nlm_release_call(call);	return NULL;}/* * Delete a block. * It is the caller's responsibility to check whether the file * can be closed hereafter. */static int nlmsvc_unlink_block(struct nlm_block *block){	int status;	dprintk("lockd: unlinking block %p...\n", block);	/* Remove block from list */	status = posix_unblock_lock(block->b_file->f_file, &block->b_call->a_args.lock.fl);	nlmsvc_remove_block(block);	return status;}static void nlmsvc_free_block(struct kref *kref){	struct nlm_block *block = container_of(kref, struct nlm_block, b_count);	struct nlm_file		*file = block->b_file;	dprintk("lockd: freeing block %p...\n", block);	/* Remove block from file's list of blocks */	mutex_lock(&file->f_mutex);	list_del_init(&block->b_flist);	mutex_unlock(&file->f_mutex);	nlmsvc_freegrantargs(block->b_call);	nlm_release_call(block->b_call);	nlm_release_file(block->b_file);	kfree(block->b_fl);	kfree(block);}static void nlmsvc_release_block(struct nlm_block *block){	if (block != NULL)		kref_put(&block->b_count, nlmsvc_free_block);}/* * Loop over all blocks and delete blocks held by * a matching host. */void nlmsvc_traverse_blocks(struct nlm_host *host,			struct nlm_file *file,			nlm_host_match_fn_t match){	struct nlm_block *block, *next;restart:	mutex_lock(&file->f_mutex);	list_for_each_entry_safe(block, next, &file->f_blocks, b_flist) {		if (!match(block->b_host, host))			continue;		/* Do not destroy blocks that are not on		 * the global retry list - why? */		if (list_empty(&block->b_list))			continue;		kref_get(&block->b_count);		mutex_unlock(&file->f_mutex);		nlmsvc_unlink_block(block);		nlmsvc_release_block(block);		goto restart;	}	mutex_unlock(&file->f_mutex);}/* * Initialize arguments for GRANTED call. The nlm_rqst structure * has been cleared already. */static int nlmsvc_setgrantargs(struct nlm_rqst *call, struct nlm_lock *lock){	locks_copy_lock(&call->a_args.lock.fl, &lock->fl);	memcpy(&call->a_args.lock.fh, &lock->fh, sizeof(call->a_args.lock.fh));	call->a_args.lock.caller = utsname()->nodename;	call->a_args.lock.oh.len = lock->oh.len;	/* set default data area */	call->a_args.lock.oh.data = call->a_owner;	call->a_args.lock.svid = lock->fl.fl_pid;	if (lock->oh.len > NLMCLNT_OHSIZE) {		void *data = kmalloc(lock->oh.len, GFP_KERNEL);		if (!data)			return 0;		call->a_args.lock.oh.data = (u8 *) data;	}	memcpy(call->a_args.lock.oh.data, lock->oh.data, lock->oh.len);	return 1;}static void nlmsvc_freegrantargs(struct nlm_rqst *call){	if (call->a_args.lock.oh.data != call->a_owner)		kfree(call->a_args.lock.oh.data);}/* * Deferred lock request handling for non-blocking lock */static __be32nlmsvc_defer_lock_rqst(struct svc_rqst *rqstp, struct nlm_block *block){	__be32 status = nlm_lck_denied_nolocks;	block->b_flags |= B_QUEUED;	nlmsvc_insert_block(block, NLM_TIMEOUT);	block->b_cache_req = &rqstp->rq_chandle;	if (rqstp->rq_chandle.defer) {		block->b_deferred_req =			rqstp->rq_chandle.defer(block->b_cache_req);		if (block->b_deferred_req != NULL)			status = nlm_drop_reply;	}	dprintk("lockd: nlmsvc_defer_lock_rqst block %p flags %d status %d\n",		block, block->b_flags, ntohl(status));	return status;}/* * Attempt to establish a lock, and if it can't be granted, block it * if required. */__be32nlmsvc_lock(struct svc_rqst *rqstp, struct nlm_file *file,	    struct nlm_host *host, struct nlm_lock *lock, int wait,	    struct nlm_cookie *cookie, int reclaim){	struct nlm_block	*block = NULL;	int			error;	__be32			ret;	dprintk("lockd: nlmsvc_lock(%s/%ld, ty=%d, pi=%d, %Ld-%Ld, bl=%d)\n",				file->f_file->f_path.dentry->d_inode->i_sb->s_id,				file->f_file->f_path.dentry->d_inode->i_ino,				lock->fl.fl_type, lock->fl.fl_pid,				(long long)lock->fl.fl_start,				(long long)lock->fl.fl_end,				wait);	/* Lock file against concurrent access */	mutex_lock(&file->f_mutex);	/* Get existing block (in case client is busy-waiting)	 * or create new block	 */	block = nlmsvc_lookup_block(file, lock);	if (block == NULL) {		block = nlmsvc_create_block(rqstp, host, file, lock, cookie);		ret = nlm_lck_denied_nolocks;		if (block == NULL)			goto out;		lock = &block->b_call->a_args.lock;	} else		lock->fl.fl_flags &= ~FL_SLEEP;	if (block->b_flags & B_QUEUED) {		dprintk("lockd: nlmsvc_lock deferred block %p flags %d\n",							block, block->b_flags);		if (block->b_granted) {			nlmsvc_unlink_block(block);			ret = nlm_granted;			goto out;		}		if (block->b_flags & B_TIMED_OUT) {			nlmsvc_unlink_block(block);			ret = nlm_lck_denied;			goto out;		}		ret = nlm_drop_reply;		goto out;	}	if (locks_in_grace() && !reclaim) {		ret = nlm_lck_denied_grace_period;		goto out;	}	if (reclaim && !locks_in_grace()) {		ret = nlm_lck_denied_grace_period;		goto out;	}	if (!wait)		lock->fl.fl_flags &= ~FL_SLEEP;	error = vfs_lock_file(file->f_file, F_SETLK, &lock->fl, NULL);	lock->fl.fl_flags &= ~FL_SLEEP;	dprintk("lockd: vfs_lock_file returned %d\n", error);	switch (error) {		case 0:			ret = nlm_granted;			goto out;		case -EAGAIN:			ret = nlm_lck_denied;			break;		case FILE_LOCK_DEFERRED:			if (wait)				break;			/* Filesystem lock operation is in progress			   Add it to the queue waiting for callback */			ret = nlmsvc_defer_lock_rqst(rqstp, block);			goto out;		case -EDEADLK:			ret = nlm_deadlock;			goto out;		default:			/* includes ENOLCK */			ret = nlm_lck_denied_nolocks;			goto out;	}	ret = nlm_lck_denied;	if (!wait)		goto out;	ret = nlm_lck_blocked;	/* Append to list of blocked */	nlmsvc_insert_block(block, NLM_NEVER);out:	mutex_unlock(&file->f_mutex);	nlmsvc_release_block(block);	dprintk("lockd: nlmsvc_lock returned %u\n", ret);	return ret;}

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
国产午夜三级一区二区三| 日韩国产欧美在线视频| 亚洲www啪成人一区二区麻豆| 石原莉奈一区二区三区在线观看| 国产成人精品影视| 欧美情侣在线播放| 亚洲人成小说网站色在线 | 亚洲视频免费看| 久久99精品久久久| 欧美精品第1页| 日韩理论电影院| 成人午夜精品一区二区三区| 日韩欧美第一区| 午夜精品久久久| 色婷婷亚洲一区二区三区| 国产日本欧美一区二区| 久久er99精品| 欧美一二三四在线| 香蕉成人伊视频在线观看| 色女孩综合影院| 亚洲三级理论片| 成人免费看视频| 国产亚洲精品免费| 国产伦精一区二区三区| 日韩免费电影一区| 日本一不卡视频| 欧美日韩精品高清| 性久久久久久久久久久久| 色综合天天综合给合国产| 国产精品国产三级国产专播品爱网| 久久aⅴ国产欧美74aaa| 精品国内二区三区| 久久成人麻豆午夜电影| 久久久久99精品一区| 国产精品一区二区黑丝| 久久久久久久网| 国产高清成人在线| 欧美国产欧美综合| 97久久超碰国产精品| 亚洲美女区一区| 欧美视频在线一区二区三区| 亚洲第一激情av| 欧美猛男超大videosgay| 视频在线在亚洲| 欧美成人艳星乳罩| 国产成人精品网址| 中文字幕一区二区三区在线不卡| 91美女精品福利| 一区二区三区四区激情| 欧美日韩三级视频| 久久成人综合网| 最近日韩中文字幕| 欧美日韩视频在线观看一区二区三区| 偷偷要91色婷婷| 久久夜色精品国产噜噜av| aaa亚洲精品一二三区| 亚洲国产日日夜夜| 亚洲精品一区二区三区福利| 99国产精品久久久久| 亚洲123区在线观看| 欧美mv日韩mv| av不卡在线观看| 奇米888四色在线精品| 国产午夜精品久久久久久久| 91精品1区2区| 国产一区二区三区四区五区美女 | 一本色道a无线码一区v| 午夜精品福利一区二区三区av | 亚洲欧美一区二区三区久本道91 | 欧美中文字幕一二三区视频| 免费欧美日韩国产三级电影| 欧美激情一区二区在线| 欧美日韩三级在线| 成人一区二区在线观看| 日韩在线一二三区| 欧美激情一区二区三区不卡| 欧美日韩一区不卡| 国产福利精品一区| 日本三级亚洲精品| 日本一区二区免费在线| 欧美一区二区免费观在线| 99久久久久免费精品国产| 美女视频黄 久久| 亚洲综合视频在线观看| 欧美精彩视频一区二区三区| 91精品国产综合久久精品图片| 国产成人精品免费| 秋霞电影网一区二区| 亚洲另类春色国产| 国产婷婷色一区二区三区在线| 欧美精品一级二级| av日韩在线网站| 国产成人综合亚洲网站| 看国产成人h片视频| 亚洲精品国产a| 中文字幕一区二区不卡| 国产午夜精品在线观看| 欧美精品一区二区三区在线| 欧美美女直播网站| 欧美性高清videossexo| 91免费看`日韩一区二区| 国产高清在线观看免费不卡| 久久99在线观看| 久久精品国产一区二区| 日本不卡的三区四区五区| 亚洲午夜在线观看视频在线| 亚洲精品国久久99热| 亚洲欧美一区二区三区国产精品| 欧美国产激情二区三区| 国产日产欧美一区| 国产精品日日摸夜夜摸av| 国产亚洲婷婷免费| 国产欧美日韩卡一| 国产精品乱人伦一区二区| 亚洲国产成人午夜在线一区| 国产婷婷精品av在线| 久久久国产一区二区三区四区小说| 精品久久国产97色综合| 久久久久久久av麻豆果冻| 国产肉丝袜一区二区| 国产欧美精品在线观看| 国产精品不卡视频| 亚洲欧美色综合| 亚洲综合在线观看视频| 亚洲成a人片在线观看中文| 亚洲va欧美va人人爽| 蜜臀国产一区二区三区在线播放| 另类小说一区二区三区| 国产九色精品成人porny| 盗摄精品av一区二区三区| 成人18精品视频| 欧美三级日韩三级国产三级| 日韩一区二区三区免费看| 精品久久久久久久久久久院品网| 国产欧美日韩麻豆91| 伊人婷婷欧美激情| 日韩电影在线看| 国产一区二区导航在线播放| 成人午夜激情在线| 欧美在线999| 日韩西西人体444www| 日本一区二区三区dvd视频在线| 亚洲码国产岛国毛片在线| 午夜欧美大尺度福利影院在线看| 九九精品视频在线看| 97超碰欧美中文字幕| 在线不卡中文字幕播放| 中文字幕欧美日本乱码一线二线| 亚洲猫色日本管| 精品一区二区三区久久久| 99re这里都是精品| 日韩欧美亚洲国产另类| 亚洲视频一区二区免费在线观看| 日韩av不卡一区二区| a亚洲天堂av| 欧美精品国产精品| 亚洲欧洲日本在线| 另类小说色综合网站| 日本福利一区二区| 久久久久久久久久久久电影| 午夜欧美视频在线观看| 成人性视频网站| 欧美一区二区成人| 亚洲欧美日韩中文字幕一区二区三区 | 国产一区二区在线观看免费| 在线观看亚洲一区| 国产精品欧美久久久久无广告 | 99久久99久久精品免费看蜜桃| 91精品在线麻豆| 亚洲欧美在线高清| 精品亚洲国产成人av制服丝袜| 在线视频你懂得一区二区三区| 国产亚洲欧美一区在线观看| 亚洲h在线观看| 色综合久久精品| 国产精品污www在线观看| 精品影院一区二区久久久| 欧美日韩亚洲另类| 亚洲精品中文在线影院| 成人午夜视频网站| 久久精品欧美日韩精品| 精品一二三四区| 欧美精品日韩一区| 亚洲一区二区在线观看视频| av中文字幕一区| 国产精品情趣视频| 国产经典欧美精品| 欧美电视剧在线看免费| 全部av―极品视觉盛宴亚洲| 9191久久久久久久久久久| 亚洲综合精品自拍| 欧美中文字幕一二三区视频| 一级女性全黄久久生活片免费| 99久久精品国产一区| 中文av字幕一区| 粗大黑人巨茎大战欧美成人| 国产欧美日韩久久| 成人激情电影免费在线观看| 国产精品美女久久久久aⅴ | 精品中文av资源站在线观看|