vfs: Add cache reset for --vfs-cache-max-size handling at cache poll interval
The vfs-cache-max-size parameter is probably confusing to many users. The cache cleaner checks cache size periodically at the --vfs-cache-poll-interval (default 60 seconds) interval and remove cache items in the following order. (1) cache items that are not in use and with age > vfs-cache-max-age (2) if the cache space used at this time still is larger than vfs-cache-max-size, the cleaner continues to remove cache items that are not in use. The cache cleaning process does not remove cache items that are currently in use. If the total space consumed by in-use cache items exceeds vfs-cache-max-size, the periodical cache cleaner thread does not do anything further and leaves the in-use cache items alone with a total space larger than vfs-cache-max-size. A cache reset feature was introduced in 1.53 which resets in-use (but not dirty, i.e., not being updated) cache items when additional cache data incurs an ENOSPC error. But this code was not activated in the periodical cache cleaning thread. This patch adds the cache reset step in the cache cleaner thread during cache poll to reset cache items until the total size of the remaining cache items is below vfs-cache-max-size.
This commit is contained in:
parent
964088affa
commit
8f23cae1c0
2 changed files with 12 additions and 12 deletions
1
go.sum
1
go.sum
|
@ -600,6 +600,7 @@ github.com/smartystreets/goconvey v1.6.4 h1:fv0U8FUIMPNf1L9lnHLvLhgicrIVChEkdzIK
|
||||||
github.com/smartystreets/goconvey v1.6.4/go.mod h1:syvi0/a8iFYH4r/RixwvyeAJjdLS9QV7WQ/tjFTllLA=
|
github.com/smartystreets/goconvey v1.6.4/go.mod h1:syvi0/a8iFYH4r/RixwvyeAJjdLS9QV7WQ/tjFTllLA=
|
||||||
github.com/soheilhy/cmux v0.1.4/go.mod h1:IM3LyeVVIOuxMH7sFAkER9+bJ4dT7Ms6E4xg4kGIyLM=
|
github.com/soheilhy/cmux v0.1.4/go.mod h1:IM3LyeVVIOuxMH7sFAkER9+bJ4dT7Ms6E4xg4kGIyLM=
|
||||||
github.com/sony/gobreaker v0.4.1/go.mod h1:ZKptC7FHNvhBz7dN2LGjPVBz2sZJmc0/PkyDJOjmxWY=
|
github.com/sony/gobreaker v0.4.1/go.mod h1:ZKptC7FHNvhBz7dN2LGjPVBz2sZJmc0/PkyDJOjmxWY=
|
||||||
|
github.com/spacemonkeygo/monkit v0.0.0-20190623001553-09813957f0a8 h1:sO833wsQOMWK0ZW68mWYVhRwlcUaL125bUh1LeR7vxI=
|
||||||
github.com/spacemonkeygo/monkit/v3 v3.0.4/go.mod h1:JcK1pCbReQsOsMKF/POFSZCq7drXFybgGmbc27tuwes=
|
github.com/spacemonkeygo/monkit/v3 v3.0.4/go.mod h1:JcK1pCbReQsOsMKF/POFSZCq7drXFybgGmbc27tuwes=
|
||||||
github.com/spacemonkeygo/monkit/v3 v3.0.7/go.mod h1:kj1ViJhlyADa7DiA4xVnTuPA46lFKbM7mxQTrXCuJP4=
|
github.com/spacemonkeygo/monkit/v3 v3.0.7/go.mod h1:kj1ViJhlyADa7DiA4xVnTuPA46lFKbM7mxQTrXCuJP4=
|
||||||
github.com/spacemonkeygo/monkit/v3 v3.0.10/go.mod h1:kj1ViJhlyADa7DiA4xVnTuPA46lFKbM7mxQTrXCuJP4=
|
github.com/spacemonkeygo/monkit/v3 v3.0.10/go.mod h1:kj1ViJhlyADa7DiA4xVnTuPA46lFKbM7mxQTrXCuJP4=
|
||||||
|
|
|
@ -658,7 +658,7 @@ func (c *Cache) purgeOverQuota(quota int64) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// clean empties the cache of stuff if it can
|
// clean empties the cache of stuff if it can
|
||||||
func (c *Cache) clean(removeCleanFiles bool) {
|
func (c *Cache) clean(kicked bool) {
|
||||||
// Cache may be empty so end
|
// Cache may be empty so end
|
||||||
_, err := os.Stat(c.root)
|
_, err := os.Stat(c.root)
|
||||||
if os.IsNotExist(err) {
|
if os.IsNotExist(err) {
|
||||||
|
@ -674,18 +674,17 @@ func (c *Cache) clean(removeCleanFiles bool) {
|
||||||
// Remove any files that are over age
|
// Remove any files that are over age
|
||||||
c.purgeOld(c.opt.CacheMaxAge)
|
c.purgeOld(c.opt.CacheMaxAge)
|
||||||
|
|
||||||
|
if int64(c.opt.CacheMaxSize) <= 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
// Now remove files not in use until cache size is below quota starting from the
|
// Now remove files not in use until cache size is below quota starting from the
|
||||||
// oldest first
|
// oldest first
|
||||||
c.purgeOverQuota(int64(c.opt.CacheMaxSize))
|
c.purgeOverQuota(int64(c.opt.CacheMaxSize))
|
||||||
|
|
||||||
// removeCleanFiles indicates that we got ENOSPC error
|
// Remove cache files that are not dirty if we are still above the max cache size
|
||||||
// We remove cache files that are not dirty if we are still above the max cache size
|
|
||||||
if removeCleanFiles {
|
|
||||||
c.purgeClean(int64(c.opt.CacheMaxSize))
|
c.purgeClean(int64(c.opt.CacheMaxSize))
|
||||||
c.retryFailedResets()
|
c.retryFailedResets()
|
||||||
} else {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
|
|
||||||
used := c.updateUsed()
|
used := c.updateUsed()
|
||||||
if used <= int64(c.opt.CacheMaxSize) && len(c.errItems) == 0 {
|
if used <= int64(c.opt.CacheMaxSize) && len(c.errItems) == 0 {
|
||||||
|
@ -694,7 +693,7 @@ func (c *Cache) clean(removeCleanFiles bool) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Was kicked?
|
// Was kicked?
|
||||||
if removeCleanFiles {
|
if kicked {
|
||||||
c.kickerMu.Lock() // Make sure this is called with cache mutex unlocked
|
c.kickerMu.Lock() // Make sure this is called with cache mutex unlocked
|
||||||
// Reenable io threads to kick me
|
// Reenable io threads to kick me
|
||||||
c.cleanerKicked = false
|
c.cleanerKicked = false
|
||||||
|
@ -737,9 +736,9 @@ func (c *Cache) cleaner(ctx context.Context) {
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
case <-c.kick: // a thread encountering ENOSPC kicked me
|
case <-c.kick: // a thread encountering ENOSPC kicked me
|
||||||
c.clean(true) // remove inUse files that are clean (!item.info.Dirty)
|
c.clean(true) // kicked is true
|
||||||
case <-timer.C:
|
case <-timer.C:
|
||||||
c.clean(false) // do not remove inUse files
|
c.clean(false) // timer driven cache poll, kicked is false
|
||||||
case <-ctx.Done():
|
case <-ctx.Done():
|
||||||
fs.Debugf(nil, "vfs cache: cleaner exiting")
|
fs.Debugf(nil, "vfs cache: cleaner exiting")
|
||||||
return
|
return
|
||||||
|
|
Loading…
Reference in a new issue