On Fri, Jun 19, 2026 at 7:44 PM Bryam Vargas via B4 Relay <[email protected]> wrote: > > From: Bryam Vargas <[email protected]> > > cache_pos_decode(), cache_key_decode() and the last-kset branches of > cache_replay(), the writeback worker and the GC worker take a cache > segment id from the cache device metadata and index cache->segments[] > with it without checking it against cache->n_segs. That metadata is only > CRC-protected with a fixed public seed, so whoever supplies the cache > device on a table load (CAP_SYS_ADMIN) controls the id; an out-of-range > value forms a wild pcache_cache_segment pointer that is dereferenced and > written through -- an out-of-bounds read and write driven by on-disk data. > > Add cache_seg_id_valid() and reject an out-of-range id at each decode > site, failing the operation with -EIO instead of indexing past the array. > Valid metadata is unaffected. > > Fixes: 1d57628ff95b ("dm-pcache: add persistent cache target in > device-mapper") > Cc: [email protected] > Signed-off-by: Bryam Vargas <[email protected]> > --- > drivers/md/dm-pcache/cache.c | 3 +++ > drivers/md/dm-pcache/cache.h | 14 ++++++++++++++ > drivers/md/dm-pcache/cache_gc.c | 16 ++++++++++++++-- > drivers/md/dm-pcache/cache_key.c | 11 +++++++++++ > drivers/md/dm-pcache/cache_writeback.c | 14 +++++++++++--- > 5 files changed, 53 insertions(+), 5 deletions(-) > > diff --git a/drivers/md/dm-pcache/cache.c b/drivers/md/dm-pcache/cache.c > index bb1ada31e483..e6a2f4460f6e 100644 > --- a/drivers/md/dm-pcache/cache.c > +++ b/drivers/md/dm-pcache/cache.c > @@ -118,6 +118,9 @@ int cache_pos_decode(struct pcache_cache *cache, > if (!latest_addr) > return -EIO; > > + if (!cache_seg_id_valid(cache, latest.cache_seg_id)) > + return -EIO; > + > pos->cache_seg = &cache->segments[latest.cache_seg_id]; > pos->seg_off = latest.seg_off; > *seq = latest.header.seq; > diff --git a/drivers/md/dm-pcache/cache.h b/drivers/md/dm-pcache/cache.h > index 27613b56be54..653ceea43131 100644 > --- a/drivers/md/dm-pcache/cache.h > +++ b/drivers/md/dm-pcache/cache.h > @@ -420,6 +420,20 @@ static inline bool cache_seg_is_ctrl_seg(u32 > cache_seg_id) > return (cache_seg_id == 0); > } > > +/** > + * cache_seg_id_valid - Validate a cache segment id read from the cache > device. > + * @cache: Pointer to the pcache_cache structure. > + * @cache_seg_id: Segment id decoded from on-media metadata. > + * > + * On-media segment ids are only protected by a CRC, which an attacker who > can > + * format the cache device computes over their chosen value. Reject any id > that > + * would index cache->segments[] out of bounds before it is dereferenced. > + */ > +static inline bool cache_seg_id_valid(struct pcache_cache *cache, u32 > cache_seg_id) > +{ > + return cache_seg_id < cache->n_segs; > +} > + > /** > * cache_key_cutfront - Cuts a specified length from the front of a cache > key. > * @key: Pointer to pcache_cache_key structure. > diff --git a/drivers/md/dm-pcache/cache_gc.c b/drivers/md/dm-pcache/cache_gc.c > index 94f8b276a021..02fa0ce03134 100644 > --- a/drivers/md/dm-pcache/cache_gc.c > +++ b/drivers/md/dm-pcache/cache_gc.c > @@ -74,11 +74,17 @@ static bool need_gc(struct pcache_cache *cache, struct > pcache_cache_pos *dirty_t > * @cache: Pointer to the pcache_cache structure. > * @kset_onmedia: Pointer to the kset_onmedia structure for the last kset. > */ > -static void last_kset_gc(struct pcache_cache *cache, struct > pcache_cache_kset_onmedia *kset_onmedia) > +static int last_kset_gc(struct pcache_cache *cache, struct > pcache_cache_kset_onmedia *kset_onmedia) > { > struct dm_pcache *pcache = CACHE_TO_PCACHE(cache); > struct pcache_cache_segment *cur_seg, *next_seg; > > + if (!cache_seg_id_valid(cache, kset_onmedia->next_cache_seg_id)) { > + pcache_dev_err(pcache, "invalid next_cache_seg_id %u in gc > (n_segs %u)\n", > + kset_onmedia->next_cache_seg_id, > cache->n_segs); > + return -EIO; > + } > + > cur_seg = cache->key_tail.cache_seg; > > next_seg = &cache->segments[kset_onmedia->next_cache_seg_id]; > @@ -94,6 +100,8 @@ static void last_kset_gc(struct pcache_cache *cache, > struct pcache_cache_kset_on > spin_lock(&cache->seg_map_lock); > __clear_bit(cur_seg->cache_seg_id, cache->seg_map); > spin_unlock(&cache->seg_map_lock); > + > + return 0; > } > > void pcache_cache_gc_fn(struct work_struct *work) > @@ -130,7 +138,11 @@ void pcache_cache_gc_fn(struct work_struct *work) > if (dirty_tail.cache_seg == key_tail.cache_seg) > break; > > - last_kset_gc(cache, kset_onmedia); > + ret = last_kset_gc(cache, kset_onmedia); > + if (ret) { > + atomic_inc(&cache->gc_errors); > + return; > + } > continue; > } > > diff --git a/drivers/md/dm-pcache/cache_key.c > b/drivers/md/dm-pcache/cache_key.c > index e068e878231b..8eec5238c5da 100644 > --- a/drivers/md/dm-pcache/cache_key.c > +++ b/drivers/md/dm-pcache/cache_key.c > @@ -94,6 +94,12 @@ int cache_key_decode(struct pcache_cache *cache, > key->off = key_onmedia->off; > key->len = key_onmedia->len; > > + if (!cache_seg_id_valid(cache, key_onmedia->cache_seg_id)) { > + pcache_dev_err(pcache, "invalid cache_seg_id %u in cache key > (n_segs %u)\n", > + key_onmedia->cache_seg_id, cache->n_segs); > + return -EIO; > + } > + > key->cache_pos.cache_seg = > &cache->segments[key_onmedia->cache_seg_id]; > key->cache_pos.seg_off = key_onmedia->cache_seg_off; > > @@ -789,6 +795,11 @@ int cache_replay(struct pcache_cache *cache) > > pcache_dev_debug(pcache, "last kset replay, next: > %u\n", kset_onmedia->next_cache_seg_id); > > + if (!cache_seg_id_valid(cache, > kset_onmedia->next_cache_seg_id)) { > + ret = -EIO; > + goto out; > + } > + > next_seg = > &cache->segments[kset_onmedia->next_cache_seg_id]; > > pos->cache_seg = next_seg; > diff --git a/drivers/md/dm-pcache/cache_writeback.c > b/drivers/md/dm-pcache/cache_writeback.c > index 87a82b3fe836..a104e6ee8aa8 100644 > --- a/drivers/md/dm-pcache/cache_writeback.c > +++ b/drivers/md/dm-pcache/cache_writeback.c > @@ -196,12 +196,18 @@ static int cache_kset_insert_tree(struct pcache_cache > *cache, struct pcache_cach > return ret; > } > > -static void last_kset_writeback(struct pcache_cache *cache, > +static int last_kset_writeback(struct pcache_cache *cache, > struct pcache_cache_kset_onmedia *last_kset_onmedia) > { > struct dm_pcache *pcache = CACHE_TO_PCACHE(cache); > struct pcache_cache_segment *next_seg; > > + if (!cache_seg_id_valid(cache, last_kset_onmedia->next_cache_seg_id)) > { > + pcache_dev_err(pcache, "invalid next_cache_seg_id %u in > writeback (n_segs %u)\n", > + last_kset_onmedia->next_cache_seg_id, > cache->n_segs); > + return -EIO; > + } > + > pcache_dev_debug(pcache, "last kset, next: %u\n", > last_kset_onmedia->next_cache_seg_id); > > next_seg = &cache->segments[last_kset_onmedia->next_cache_seg_id]; > @@ -211,6 +217,8 @@ static void last_kset_writeback(struct pcache_cache > *cache, > cache->dirty_tail.seg_off = 0; > cache_encode_dirty_tail(cache); > mutex_unlock(&cache->dirty_tail_lock); > + > + return 0; > } > > void cache_writeback_fn(struct work_struct *work) > @@ -241,8 +249,8 @@ void cache_writeback_fn(struct work_struct *work) > } > > if (kset_onmedia->flags & PCACHE_KSET_FLAGS_LAST) { > - last_kset_writeback(cache, kset_onmedia); > - delay = 0; > + ret = last_kset_writeback(cache, kset_onmedia); > + delay = ret ? PCACHE_CACHE_WRITEBACK_INTERVAL : 0; > goto queue_work; > } > > > -- > 2.43.0 > >
Reviewed-by: Zheng Gu <[email protected]>

