aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorKent Overstreet <kmo@daterainc.com>2013-09-24 02:17:29 -0400
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>2013-10-05 10:13:09 -0400
commit808eea9d2912e4a3fb8cd45e3e3da94d114757fb (patch)
treeee108cbf8b8e1b2cbf910268df03d71e2df9348b
parentbb343115253500dcee63ea78d765e4c2f2a57fc4 (diff)
bcache: Fix for when no journal entries are found
commit c426c4fd46f709ade2bddd51c5738729c7ae1db5 upstream. The journal replay code didn't handle this case, causing it to go into an infinite loop... Signed-off-by: Kent Overstreet <kmo@daterainc.com> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
-rw-r--r--drivers/md/bcache/journal.c30
1 files changed, 18 insertions, 12 deletions
diff --git a/drivers/md/bcache/journal.c b/drivers/md/bcache/journal.c
index fb1bfafa21e4..228f2c37b245 100644
--- a/drivers/md/bcache/journal.c
+++ b/drivers/md/bcache/journal.c
@@ -151,7 +151,8 @@ int bch_journal_read(struct cache_set *c, struct list_head *list,
151 bitmap_zero(bitmap, SB_JOURNAL_BUCKETS); 151 bitmap_zero(bitmap, SB_JOURNAL_BUCKETS);
152 pr_debug("%u journal buckets", ca->sb.njournal_buckets); 152 pr_debug("%u journal buckets", ca->sb.njournal_buckets);
153 153
154 /* Read journal buckets ordered by golden ratio hash to quickly 154 /*
155 * Read journal buckets ordered by golden ratio hash to quickly
155 * find a sequence of buckets with valid journal entries 156 * find a sequence of buckets with valid journal entries
156 */ 157 */
157 for (i = 0; i < ca->sb.njournal_buckets; i++) { 158 for (i = 0; i < ca->sb.njournal_buckets; i++) {
@@ -164,18 +165,20 @@ int bch_journal_read(struct cache_set *c, struct list_head *list,
164 goto bsearch; 165 goto bsearch;
165 } 166 }
166 167
167 /* If that fails, check all the buckets we haven't checked 168 /*
169 * If that fails, check all the buckets we haven't checked
168 * already 170 * already
169 */ 171 */
170 pr_debug("falling back to linear search"); 172 pr_debug("falling back to linear search");
171 173
172 for (l = 0; l < ca->sb.njournal_buckets; l++) { 174 for (l = find_first_zero_bit(bitmap, ca->sb.njournal_buckets);
173 if (test_bit(l, bitmap)) 175 l < ca->sb.njournal_buckets;
174 continue; 176 l = find_next_zero_bit(bitmap, ca->sb.njournal_buckets, l + 1))
175
176 if (read_bucket(l)) 177 if (read_bucket(l))
177 goto bsearch; 178 goto bsearch;
178 } 179
180 if (list_empty(list))
181 continue;
179bsearch: 182bsearch:
180 /* Binary search */ 183 /* Binary search */
181 m = r = find_next_bit(bitmap, ca->sb.njournal_buckets, l + 1); 184 m = r = find_next_bit(bitmap, ca->sb.njournal_buckets, l + 1);
@@ -195,10 +198,12 @@ bsearch:
195 r = m; 198 r = m;
196 } 199 }
197 200
198 /* Read buckets in reverse order until we stop finding more 201 /*
202 * Read buckets in reverse order until we stop finding more
199 * journal entries 203 * journal entries
200 */ 204 */
201 pr_debug("finishing up"); 205 pr_debug("finishing up: m %u njournal_buckets %u",
206 m, ca->sb.njournal_buckets);
202 l = m; 207 l = m;
203 208
204 while (1) { 209 while (1) {
@@ -226,9 +231,10 @@ bsearch:
226 } 231 }
227 } 232 }
228 233
229 c->journal.seq = list_entry(list->prev, 234 if (!list_empty(list))
230 struct journal_replay, 235 c->journal.seq = list_entry(list->prev,
231 list)->j.seq; 236 struct journal_replay,
237 list)->j.seq;
232 238
233 return 0; 239 return 0;
234#undef read_bucket 240#undef read_bucket