diff options
Diffstat (limited to 'fs/nfs/write.c')
| -rw-r--r-- | fs/nfs/write.c | 64 | 
1 files changed, 50 insertions, 14 deletions
diff --git a/fs/nfs/write.c b/fs/nfs/write.c index 5f4fd53e5764..e1c74d3db64d 100644 --- a/fs/nfs/write.c +++ b/fs/nfs/write.c @@ -245,8 +245,7 @@ static void nfs_mark_uptodate(struct nfs_page *req)  static int wb_priority(struct writeback_control *wbc)  {  	int ret = 0; -	if (wbc->for_reclaim) -		return FLUSH_HIGHPRI | FLUSH_COND_STABLE; +  	if (wbc->sync_mode == WB_SYNC_ALL)  		ret = FLUSH_COND_STABLE;  	return ret; @@ -737,7 +736,7 @@ static void nfs_inode_remove_request(struct nfs_page *req)  		head = req->wb_head;  		spin_lock(&inode->i_lock); -		if (likely(!PageSwapCache(head->wb_page))) { +		if (likely(head->wb_page && !PageSwapCache(head->wb_page))) {  			set_page_private(head->wb_page, 0);  			ClearPagePrivate(head->wb_page);  			smp_mb__after_atomic(); @@ -759,7 +758,8 @@ static void nfs_inode_remove_request(struct nfs_page *req)  static void  nfs_mark_request_dirty(struct nfs_page *req)  { -	__set_page_dirty_nobuffers(req->wb_page); +	if (req->wb_page) +		__set_page_dirty_nobuffers(req->wb_page);  }  /* @@ -804,7 +804,7 @@ nfs_page_search_commits_for_head_request_locked(struct nfs_inode *nfsi,   * number of outstanding requests requiring a commit as well as   * the MM page stats.   * - * The caller must hold the cinfo->lock, and the nfs_page lock. + * The caller must hold cinfo->inode->i_lock, and the nfs_page lock.   */  void  nfs_request_add_commit_list_locked(struct nfs_page *req, struct list_head *dst, @@ -832,10 +832,11 @@ EXPORT_SYMBOL_GPL(nfs_request_add_commit_list_locked);  void  nfs_request_add_commit_list(struct nfs_page *req, struct nfs_commit_info *cinfo)  { -	spin_lock(cinfo->lock); +	spin_lock(&cinfo->inode->i_lock);  	nfs_request_add_commit_list_locked(req, &cinfo->mds->list, cinfo); -	spin_unlock(cinfo->lock); -	nfs_mark_page_unstable(req->wb_page, cinfo); +	spin_unlock(&cinfo->inode->i_lock); +	if (req->wb_page) +		nfs_mark_page_unstable(req->wb_page, cinfo);  }  EXPORT_SYMBOL_GPL(nfs_request_add_commit_list); @@ -864,7 +865,7 @@ EXPORT_SYMBOL_GPL(nfs_request_remove_commit_list);  static void nfs_init_cinfo_from_inode(struct nfs_commit_info *cinfo,  				      struct inode *inode)  { -	cinfo->lock = &inode->i_lock; +	cinfo->inode = inode;  	cinfo->mds = &NFS_I(inode)->commit_info;  	cinfo->ds = pnfs_get_ds_info(inode);  	cinfo->dreq = NULL; @@ -967,7 +968,7 @@ nfs_reqs_to_commit(struct nfs_commit_info *cinfo)  	return cinfo->mds->ncommit;  } -/* cinfo->lock held by caller */ +/* cinfo->inode->i_lock held by caller */  int  nfs_scan_commit_list(struct list_head *src, struct list_head *dst,  		     struct nfs_commit_info *cinfo, int max) @@ -979,7 +980,7 @@ nfs_scan_commit_list(struct list_head *src, struct list_head *dst,  		if (!nfs_lock_request(req))  			continue;  		kref_get(&req->wb_kref); -		if (cond_resched_lock(cinfo->lock)) +		if (cond_resched_lock(&cinfo->inode->i_lock))  			list_safe_reset_next(req, tmp, wb_list);  		nfs_request_remove_commit_list(req, cinfo);  		nfs_list_add_request(req, dst); @@ -1005,7 +1006,7 @@ nfs_scan_commit(struct inode *inode, struct list_head *dst,  {  	int ret = 0; -	spin_lock(cinfo->lock); +	spin_lock(&cinfo->inode->i_lock);  	if (cinfo->mds->ncommit > 0) {  		const int max = INT_MAX; @@ -1013,7 +1014,7 @@ nfs_scan_commit(struct inode *inode, struct list_head *dst,  					   cinfo, max);  		ret += pnfs_scan_commit_lists(inode, cinfo, max - ret);  	} -	spin_unlock(cinfo->lock); +	spin_unlock(&cinfo->inode->i_lock);  	return ret;  } @@ -1709,6 +1710,10 @@ nfs_commit_list(struct inode *inode, struct list_head *head, int how,  {  	struct nfs_commit_data	*data; +	/* another commit raced with us */ +	if (list_empty(head)) +		return 0; +  	data = nfs_commitdata_alloc();  	if (!data) @@ -1724,6 +1729,36 @@ nfs_commit_list(struct inode *inode, struct list_head *head, int how,  	return -ENOMEM;  } +int nfs_commit_file(struct file *file, struct nfs_write_verifier *verf) +{ +	struct inode *inode = file_inode(file); +	struct nfs_open_context *open; +	struct nfs_commit_info cinfo; +	struct nfs_page *req; +	int ret; + +	open = get_nfs_open_context(nfs_file_open_context(file)); +	req  = nfs_create_request(open, NULL, NULL, 0, i_size_read(inode)); +	if (IS_ERR(req)) { +		ret = PTR_ERR(req); +		goto out_put; +	} + +	nfs_init_cinfo_from_inode(&cinfo, inode); + +	memcpy(&req->wb_verf, verf, sizeof(struct nfs_write_verifier)); +	nfs_request_add_commit_list(req, &cinfo); +	ret = nfs_commit_inode(inode, FLUSH_SYNC); +	if (ret > 0) +		ret = 0; + +	nfs_free_request(req); +out_put: +	put_nfs_open_context(open); +	return ret; +} +EXPORT_SYMBOL_GPL(nfs_commit_file); +  /*   * COMMIT call returned   */ @@ -1748,7 +1783,8 @@ static void nfs_commit_release_pages(struct nfs_commit_data *data)  	while (!list_empty(&data->pages)) {  		req = nfs_list_entry(data->pages.next);  		nfs_list_remove_request(req); -		nfs_clear_page_commit(req->wb_page); +		if (req->wb_page) +			nfs_clear_page_commit(req->wb_page);  		dprintk("NFS:       commit (%s/%llu %d@%lld)",  			req->wb_context->dentry->d_sb->s_id,  |