diff options
Diffstat (limited to 'fs/cachefiles/daemon.c')
| -rw-r--r-- | fs/cachefiles/daemon.c | 189 | 
1 files changed, 90 insertions, 99 deletions
diff --git a/fs/cachefiles/daemon.c b/fs/cachefiles/daemon.c index 752c1e43416f..7ac04ee2c0a0 100644 --- a/fs/cachefiles/daemon.c +++ b/fs/cachefiles/daemon.c @@ -1,7 +1,7 @@  // SPDX-License-Identifier: GPL-2.0-or-later  /* Daemon interface   * - * Copyright (C) 2007 Red Hat, Inc. All Rights Reserved. + * Copyright (C) 2007, 2021 Red Hat, Inc. All Rights Reserved.   * Written by David Howells ([email protected])   */ @@ -41,6 +41,8 @@ static int cachefiles_daemon_dir(struct cachefiles_cache *, char *);  static int cachefiles_daemon_inuse(struct cachefiles_cache *, char *);  static int cachefiles_daemon_secctx(struct cachefiles_cache *, char *);  static int cachefiles_daemon_tag(struct cachefiles_cache *, char *); +static int cachefiles_daemon_bind(struct cachefiles_cache *, char *); +static void cachefiles_daemon_unbind(struct cachefiles_cache *);  static unsigned long cachefiles_open; @@ -78,7 +80,7 @@ static const struct cachefiles_daemon_cmd cachefiles_daemon_cmds[] = {  /* - * do various checks + * Prepare a cache for caching.   */  static int cachefiles_daemon_open(struct inode *inode, struct file *file)  { @@ -102,9 +104,10 @@ static int cachefiles_daemon_open(struct inode *inode, struct file *file)  	}  	mutex_init(&cache->daemon_mutex); -	cache->active_nodes = RB_ROOT; -	rwlock_init(&cache->active_lock);  	init_waitqueue_head(&cache->daemon_pollwq); +	INIT_LIST_HEAD(&cache->volumes); +	INIT_LIST_HEAD(&cache->object_list); +	spin_lock_init(&cache->object_list_lock);  	/* set default caching limits  	 * - limit at 1% free space and/or free files @@ -124,7 +127,7 @@ static int cachefiles_daemon_open(struct inode *inode, struct file *file)  }  /* - * release a cache + * Release a cache.   */  static int cachefiles_daemon_release(struct inode *inode, struct file *file)  { @@ -138,8 +141,6 @@ static int cachefiles_daemon_release(struct inode *inode, struct file *file)  	cachefiles_daemon_unbind(cache); -	ASSERT(!cache->active_nodes.rb_node); -  	/* clean up the control file interface */  	cache->cachefilesd = NULL;  	file->private_data = NULL; @@ -152,7 +153,7 @@ static int cachefiles_daemon_release(struct inode *inode, struct file *file)  }  /* - * read the cache state + * Read the cache state.   */  static ssize_t cachefiles_daemon_read(struct file *file, char __user *_buffer,  				      size_t buflen, loff_t *pos) @@ -169,7 +170,7 @@ static ssize_t cachefiles_daemon_read(struct file *file, char __user *_buffer,  		return 0;  	/* check how much space the cache has */ -	cachefiles_has_space(cache, 0, 0); +	cachefiles_has_space(cache, 0, 0, cachefiles_has_space_check);  	/* summarise */  	f_released = atomic_xchg(&cache->f_released, 0); @@ -206,7 +207,7 @@ static ssize_t cachefiles_daemon_read(struct file *file, char __user *_buffer,  }  /* - * command the cache + * Take a command from cachefilesd, parse it and act on it.   */  static ssize_t cachefiles_daemon_write(struct file *file,  				       const char __user *_data, @@ -225,7 +226,7 @@ static ssize_t cachefiles_daemon_write(struct file *file,  	if (test_bit(CACHEFILES_DEAD, &cache->flags))  		return -EIO; -	if (datalen < 0 || datalen > PAGE_SIZE - 1) +	if (datalen > PAGE_SIZE - 1)  		return -EOPNOTSUPP;  	/* drag the command string into the kernel so we can parse it */ @@ -284,7 +285,7 @@ found_command:  }  /* - * poll for culling state + * Poll for culling state   * - use EPOLLOUT to indicate culling state   */  static __poll_t cachefiles_daemon_poll(struct file *file, @@ -306,7 +307,7 @@ static __poll_t cachefiles_daemon_poll(struct file *file,  }  /* - * give a range error for cache space constraints + * Give a range error for cache space constraints   * - can be tail-called   */  static int cachefiles_daemon_range_error(struct cachefiles_cache *cache, @@ -318,7 +319,7 @@ static int cachefiles_daemon_range_error(struct cachefiles_cache *cache,  }  /* - * set the percentage of files at which to stop culling + * Set the percentage of files at which to stop culling   * - command: "frun <N>%"   */  static int cachefiles_daemon_frun(struct cachefiles_cache *cache, char *args) @@ -342,7 +343,7 @@ static int cachefiles_daemon_frun(struct cachefiles_cache *cache, char *args)  }  /* - * set the percentage of files at which to start culling + * Set the percentage of files at which to start culling   * - command: "fcull <N>%"   */  static int cachefiles_daemon_fcull(struct cachefiles_cache *cache, char *args) @@ -366,7 +367,7 @@ static int cachefiles_daemon_fcull(struct cachefiles_cache *cache, char *args)  }  /* - * set the percentage of files at which to stop allocating + * Set the percentage of files at which to stop allocating   * - command: "fstop <N>%"   */  static int cachefiles_daemon_fstop(struct cachefiles_cache *cache, char *args) @@ -382,7 +383,7 @@ static int cachefiles_daemon_fstop(struct cachefiles_cache *cache, char *args)  	if (args[0] != '%' || args[1] != '\0')  		return -EINVAL; -	if (fstop < 0 || fstop >= cache->fcull_percent) +	if (fstop >= cache->fcull_percent)  		return cachefiles_daemon_range_error(cache, args);  	cache->fstop_percent = fstop; @@ -390,7 +391,7 @@ static int cachefiles_daemon_fstop(struct cachefiles_cache *cache, char *args)  }  /* - * set the percentage of blocks at which to stop culling + * Set the percentage of blocks at which to stop culling   * - command: "brun <N>%"   */  static int cachefiles_daemon_brun(struct cachefiles_cache *cache, char *args) @@ -414,7 +415,7 @@ static int cachefiles_daemon_brun(struct cachefiles_cache *cache, char *args)  }  /* - * set the percentage of blocks at which to start culling + * Set the percentage of blocks at which to start culling   * - command: "bcull <N>%"   */  static int cachefiles_daemon_bcull(struct cachefiles_cache *cache, char *args) @@ -438,7 +439,7 @@ static int cachefiles_daemon_bcull(struct cachefiles_cache *cache, char *args)  }  /* - * set the percentage of blocks at which to stop allocating + * Set the percentage of blocks at which to stop allocating   * - command: "bstop <N>%"   */  static int cachefiles_daemon_bstop(struct cachefiles_cache *cache, char *args) @@ -454,7 +455,7 @@ static int cachefiles_daemon_bstop(struct cachefiles_cache *cache, char *args)  	if (args[0] != '%' || args[1] != '\0')  		return -EINVAL; -	if (bstop < 0 || bstop >= cache->bcull_percent) +	if (bstop >= cache->bcull_percent)  		return cachefiles_daemon_range_error(cache, args);  	cache->bstop_percent = bstop; @@ -462,7 +463,7 @@ static int cachefiles_daemon_bstop(struct cachefiles_cache *cache, char *args)  }  /* - * set the cache directory + * Set the cache directory   * - command: "dir <name>"   */  static int cachefiles_daemon_dir(struct cachefiles_cache *cache, char *args) @@ -490,7 +491,7 @@ static int cachefiles_daemon_dir(struct cachefiles_cache *cache, char *args)  }  /* - * set the cache security context + * Set the cache security context   * - command: "secctx <ctx>"   */  static int cachefiles_daemon_secctx(struct cachefiles_cache *cache, char *args) @@ -518,7 +519,7 @@ static int cachefiles_daemon_secctx(struct cachefiles_cache *cache, char *args)  }  /* - * set the cache tag + * Set the cache tag   * - command: "tag <name>"   */  static int cachefiles_daemon_tag(struct cachefiles_cache *cache, char *args) @@ -544,7 +545,7 @@ static int cachefiles_daemon_tag(struct cachefiles_cache *cache, char *args)  }  /* - * request a node in the cache be culled from the current working directory + * Request a node in the cache be culled from the current working directory   * - command: "cull <name>"   */  static int cachefiles_daemon_cull(struct cachefiles_cache *cache, char *args) @@ -568,7 +569,6 @@ static int cachefiles_daemon_cull(struct cachefiles_cache *cache, char *args)  		return -EIO;  	} -	/* extract the directory dentry from the cwd */  	get_fs_pwd(current->fs, &path);  	if (!d_can_lookup(path.dentry)) @@ -593,7 +593,7 @@ inval:  }  /* - * set debugging mode + * Set debugging mode   * - command: "debug <mask>"   */  static int cachefiles_daemon_debug(struct cachefiles_cache *cache, char *args) @@ -616,7 +616,7 @@ inval:  }  /* - * find out whether an object in the current working directory is in use or not + * Find out whether an object in the current working directory is in use or not   * - command: "inuse <name>"   */  static int cachefiles_daemon_inuse(struct cachefiles_cache *cache, char *args) @@ -640,7 +640,6 @@ static int cachefiles_daemon_inuse(struct cachefiles_cache *cache, char *args)  		return -EIO;  	} -	/* extract the directory dentry from the cwd */  	get_fs_pwd(current->fs, &path);  	if (!d_can_lookup(path.dentry)) @@ -665,84 +664,76 @@ inval:  }  /* - * see if we have space for a number of pages and/or a number of files in the - * cache + * Bind a directory as a cache   */ -int cachefiles_has_space(struct cachefiles_cache *cache, -			 unsigned fnr, unsigned bnr) +static int cachefiles_daemon_bind(struct cachefiles_cache *cache, char *args)  { -	struct kstatfs stats; -	struct path path = { -		.mnt	= cache->mnt, -		.dentry	= cache->mnt->mnt_root, -	}; -	int ret; +	_enter("{%u,%u,%u,%u,%u,%u},%s", +	       cache->frun_percent, +	       cache->fcull_percent, +	       cache->fstop_percent, +	       cache->brun_percent, +	       cache->bcull_percent, +	       cache->bstop_percent, +	       args); + +	if (cache->fstop_percent >= cache->fcull_percent || +	    cache->fcull_percent >= cache->frun_percent || +	    cache->frun_percent  >= 100) +		return -ERANGE; + +	if (cache->bstop_percent >= cache->bcull_percent || +	    cache->bcull_percent >= cache->brun_percent || +	    cache->brun_percent  >= 100) +		return -ERANGE; -	//_enter("{%llu,%llu,%llu,%llu,%llu,%llu},%u,%u", -	//       (unsigned long long) cache->frun, -	//       (unsigned long long) cache->fcull, -	//       (unsigned long long) cache->fstop, -	//       (unsigned long long) cache->brun, -	//       (unsigned long long) cache->bcull, -	//       (unsigned long long) cache->bstop, -	//       fnr, bnr); - -	/* find out how many pages of blockdev are available */ -	memset(&stats, 0, sizeof(stats)); - -	ret = vfs_statfs(&path, &stats); -	if (ret < 0) { -		if (ret == -EIO) -			cachefiles_io_error(cache, "statfs failed"); -		_leave(" = %d", ret); -		return ret; +	if (*args) { +		pr_err("'bind' command doesn't take an argument\n"); +		return -EINVAL;  	} -	stats.f_bavail >>= cache->bshift; - -	//_debug("avail %llu,%llu", -	//       (unsigned long long) stats.f_ffree, -	//       (unsigned long long) stats.f_bavail); - -	/* see if there is sufficient space */ -	if (stats.f_ffree > fnr) -		stats.f_ffree -= fnr; -	else -		stats.f_ffree = 0; - -	if (stats.f_bavail > bnr) -		stats.f_bavail -= bnr; -	else -		stats.f_bavail = 0; - -	ret = -ENOBUFS; -	if (stats.f_ffree < cache->fstop || -	    stats.f_bavail < cache->bstop) -		goto begin_cull; - -	ret = 0; -	if (stats.f_ffree < cache->fcull || -	    stats.f_bavail < cache->bcull) -		goto begin_cull; - -	if (test_bit(CACHEFILES_CULLING, &cache->flags) && -	    stats.f_ffree >= cache->frun && -	    stats.f_bavail >= cache->brun && -	    test_and_clear_bit(CACHEFILES_CULLING, &cache->flags) -	    ) { -		_debug("cease culling"); -		cachefiles_state_changed(cache); +	if (!cache->rootdirname) { +		pr_err("No cache directory specified\n"); +		return -EINVAL;  	} -	//_leave(" = 0"); -	return 0; +	/* Don't permit already bound caches to be re-bound */ +	if (test_bit(CACHEFILES_READY, &cache->flags)) { +		pr_err("Cache already bound\n"); +		return -EBUSY; +	} -begin_cull: -	if (!test_and_set_bit(CACHEFILES_CULLING, &cache->flags)) { -		_debug("### CULL CACHE ###"); -		cachefiles_state_changed(cache); +	/* Make sure we have copies of the tag string */ +	if (!cache->tag) { +		/* +		 * The tag string is released by the fops->release() +		 * function, so we don't release it on error here +		 */ +		cache->tag = kstrdup("CacheFiles", GFP_KERNEL); +		if (!cache->tag) +			return -ENOMEM;  	} -	_leave(" = %d", ret); -	return ret; +	return cachefiles_add_cache(cache); +} + +/* + * Unbind a cache. + */ +static void cachefiles_daemon_unbind(struct cachefiles_cache *cache) +{ +	_enter(""); + +	if (test_bit(CACHEFILES_READY, &cache->flags)) +		cachefiles_withdraw_cache(cache); + +	cachefiles_put_directory(cache->graveyard); +	cachefiles_put_directory(cache->store); +	mntput(cache->mnt); + +	kfree(cache->rootdirname); +	kfree(cache->secctx); +	kfree(cache->tag); + +	_leave("");  }  |