On 6/15/21 7:49 AM, Coly Li wrote: > This patch implements two methods to store bcache journal to, > 1) __journal_write_unlocked() for block interface device > The latency method to compose bio and issue the jset bio to cache > device (e.g. SSD). c->journal.key.ptr[0] indicates the LBA on cache > device to store the journal jset. > 2) __journal_nvdimm_write_unlocked() for memory interface NVDIMM > Use memory interface to access NVDIMM pages and store the jset by > memcpy_flushcache(). c->journal.key.ptr[0] indicates the linear > address from the NVDIMM pages to store the journal jset. > > For lagency configuration without NVDIMM meta device, journal I/O is legacy? > handled by __journal_write_unlocked() with existing code logic. If the > NVDIMM meta device is used (by bcache-tools), the journal I/O will > be handled by __journal_nvdimm_write_unlocked() and go into the NVDIMM > pages. > > And when NVDIMM meta device is used, sb.d[] stores the linear addresses > from NVDIMM pages (no more bucket index), in journal_reclaim() the > journaling location in c->journal.key.ptr[0] should also be updated by > linear address from NVDIMM pages (no more LBA combined by sectors offset > and bucket index). > > Signed-off-by: Coly Li <colyli@xxxxxxx> > Cc: Jianpeng Ma <jianpeng.ma@xxxxxxxxx> > Cc: Qiaowei Ren <qiaowei.ren@xxxxxxxxx> > --- > drivers/md/bcache/journal.c | 119 ++++++++++++++++++++++++---------- > drivers/md/bcache/nvm-pages.h | 1 + > drivers/md/bcache/super.c | 28 +++++++- > 3 files changed, 110 insertions(+), 38 deletions(-) > > diff --git a/drivers/md/bcache/journal.c b/drivers/md/bcache/journal.c > index 32599d2ff5d2..03ecedf813b0 100644 > --- a/drivers/md/bcache/journal.c > +++ b/drivers/md/bcache/journal.c > @@ -596,6 +596,8 @@ static void do_journal_discard(struct cache *ca) > return; > } > > + BUG_ON(bch_has_feature_nvdimm_meta(&ca->sb)); > + > switch (atomic_read(&ja->discard_in_flight)) { > case DISCARD_IN_FLIGHT: > return; > @@ -661,9 +663,13 @@ static void journal_reclaim(struct cache_set *c) > goto out; > > ja->cur_idx = next; > - k->ptr[0] = MAKE_PTR(0, > - bucket_to_sector(c, ca->sb.d[ja->cur_idx]), > - ca->sb.nr_this_dev); > + if (!bch_has_feature_nvdimm_meta(&ca->sb)) > + k->ptr[0] = MAKE_PTR(0, > + bucket_to_sector(c, ca->sb.d[ja->cur_idx]), > + ca->sb.nr_this_dev); > + else > + k->ptr[0] = ca->sb.d[ja->cur_idx]; > + > atomic_long_inc(&c->reclaimed_journal_buckets); > > bkey_init(k); > @@ -729,46 +735,21 @@ static void journal_write_unlock(struct closure *cl) > spin_unlock(&c->journal.lock); > } > > -static void journal_write_unlocked(struct closure *cl) > + > +static void __journal_write_unlocked(struct cache_set *c) > __releases(c->journal.lock) > { > - struct cache_set *c = container_of(cl, struct cache_set, journal.io); > - struct cache *ca = c->cache; > - struct journal_write *w = c->journal.cur; > struct bkey *k = &c->journal.key; > - unsigned int i, sectors = set_blocks(w->data, block_bytes(ca)) * > - ca->sb.block_size; > - > + struct journal_write *w = c->journal.cur; > + struct closure *cl = &c->journal.io; > + struct cache *ca = c->cache; > struct bio *bio; > struct bio_list list; > + unsigned int i, sectors = set_blocks(w->data, block_bytes(ca)) * > + ca->sb.block_size; > > bio_list_init(&list); > > - if (!w->need_write) { > - closure_return_with_destructor(cl, journal_write_unlock); > - return; > - } else if (journal_full(&c->journal)) { > - journal_reclaim(c); > - spin_unlock(&c->journal.lock); > - > - btree_flush_write(c); > - continue_at(cl, journal_write, bch_journal_wq); > - return; > - } > - > - c->journal.blocks_free -= set_blocks(w->data, block_bytes(ca)); > - > - w->data->btree_level = c->root->level; > - > - bkey_copy(&w->data->btree_root, &c->root->key); > - bkey_copy(&w->data->uuid_bucket, &c->uuid_bucket); > - > - w->data->prio_bucket[ca->sb.nr_this_dev] = ca->prio_buckets[0]; > - w->data->magic = jset_magic(&ca->sb); > - w->data->version = BCACHE_JSET_VERSION; > - w->data->last_seq = last_seq(&c->journal); > - w->data->csum = csum_set(w->data); > - > for (i = 0; i < KEY_PTRS(k); i++) { > ca = c->cache; > bio = &ca->journal.bio; > @@ -793,7 +774,6 @@ static void journal_write_unlocked(struct closure *cl) > > ca->journal.seq[ca->journal.cur_idx] = w->data->seq; > } > - > /* If KEY_PTRS(k) == 0, this jset gets lost in air */ > BUG_ON(i == 0); > > @@ -805,6 +785,73 @@ static void journal_write_unlocked(struct closure *cl) > > while ((bio = bio_list_pop(&list))) > closure_bio_submit(c, bio, cl); > +} > + > +#if defined(CONFIG_BCACHE_NVM_PAGES) > + > +static void __journal_nvdimm_write_unlocked(struct cache_set *c) > + __releases(c->journal.lock) > +{ > + struct journal_write *w = c->journal.cur; > + struct cache *ca = c->cache; > + unsigned int sectors; > + > + sectors = set_blocks(w->data, block_bytes(ca)) * ca->sb.block_size; > + atomic_long_add(sectors, &ca->meta_sectors_written); > + > + memcpy_flushcache((void *)c->journal.key.ptr[0], w->data, sectors << 9); > + > + c->journal.key.ptr[0] += sectors << 9; > + ca->journal.seq[ca->journal.cur_idx] = w->data->seq; > + > + atomic_dec_bug(&fifo_back(&c->journal.pin)); > + bch_journal_next(&c->journal); > + journal_reclaim(c); > + > + spin_unlock(&c->journal.lock); > +} > + > +#else /* CONFIG_BCACHE_NVM_PAGES */ > + > +static void __journal_nvdimm_write_unlocked(struct cache_set *c) { } > + > +#endif /* CONFIG_BCACHE_NVM_PAGES */ > + > +static void journal_write_unlocked(struct closure *cl) > +{ > + struct cache_set *c = container_of(cl, struct cache_set, journal.io); > + struct cache *ca = c->cache; > + struct journal_write *w = c->journal.cur; > + > + if (!w->need_write) { > + closure_return_with_destructor(cl, journal_write_unlock); > + return; > + } else if (journal_full(&c->journal)) { > + journal_reclaim(c); > + spin_unlock(&c->journal.lock); > + > + btree_flush_write(c); > + continue_at(cl, journal_write, bch_journal_wq); > + return; > + } > + > + c->journal.blocks_free -= set_blocks(w->data, block_bytes(ca)); > + > + w->data->btree_level = c->root->level; > + > + bkey_copy(&w->data->btree_root, &c->root->key); > + bkey_copy(&w->data->uuid_bucket, &c->uuid_bucket); > + > + w->data->prio_bucket[ca->sb.nr_this_dev] = ca->prio_buckets[0]; > + w->data->magic = jset_magic(&ca->sb); > + w->data->version = BCACHE_JSET_VERSION; > + w->data->last_seq = last_seq(&c->journal); > + w->data->csum = csum_set(w->data); > + > + if (!bch_has_feature_nvdimm_meta(&ca->sb)) > + __journal_write_unlocked(c); > + else > + __journal_nvdimm_write_unlocked(c); > > continue_at(cl, journal_write_done, NULL); > } > diff --git a/drivers/md/bcache/nvm-pages.h b/drivers/md/bcache/nvm-pages.h > index c763bf2e2721..736a661777b7 100644 > --- a/drivers/md/bcache/nvm-pages.h > +++ b/drivers/md/bcache/nvm-pages.h > @@ -5,6 +5,7 @@ > > #if defined(CONFIG_BCACHE_NVM_PAGES) > #include <linux/bcache-nvm.h> > +#include <linux/libnvdimm.h> > #endif /* CONFIG_BCACHE_NVM_PAGES */ > > /* > diff --git a/drivers/md/bcache/super.c b/drivers/md/bcache/super.c > index cce0f6bf0944..4d6666d03aa7 100644 > --- a/drivers/md/bcache/super.c > +++ b/drivers/md/bcache/super.c > @@ -1686,7 +1686,32 @@ void bch_cache_set_release(struct kobject *kobj) > static void cache_set_free(struct closure *cl) > { > struct cache_set *c = container_of(cl, struct cache_set, cl); > - struct cache *ca; > + struct cache *ca = c->cache; > + > +#if defined(CONFIG_BCACHE_NVM_PAGES) > + /* Flush cache if journal stored in NVDIMM */ > + if (ca && bch_has_feature_nvdimm_meta(&ca->sb)) { > + unsigned long bucket_size = ca->sb.bucket_size; > + int i; > + > + for (i = 0; i < ca->sb.keys; i++) { > + unsigned long offset = 0; > + unsigned int len = round_down(UINT_MAX, 2); > + > + if ((void *)ca->sb.d[i] == NULL) > + continue; > + > + while (bucket_size > 0) { > + if (len > bucket_size) > + len = bucket_size; > + arch_invalidate_pmem( > + (void *)(ca->sb.d[i] + offset), len); > + offset += len; > + bucket_size -= len; > + } > + } > + } > +#endif /* CONFIG_BCACHE_NVM_PAGES */ > > debugfs_remove(c->debug); > > @@ -1698,7 +1723,6 @@ static void cache_set_free(struct closure *cl) > bch_bset_sort_state_free(&c->sort); > free_pages((unsigned long) c->uuids, ilog2(meta_bucket_pages(&c->cache->sb))); > > - ca = c->cache; > if (ca) { > ca->set = NULL; > c->cache = NULL; > Reviewed-by: Hannes Reinecke <hare@xxxxxxx> Cheers, Hannes -- Dr. Hannes Reinecke Kernel Storage Architect hare@xxxxxxx +49 911 74053 688 SUSE Software Solutions Germany GmbH, Maxfeldstr. 5, 90409 Nürnberg HRB 36809 (AG Nürnberg), GF: Felix Imendörffer