diff options
author | Kent Overstreet <kmo@daterainc.com> | 2013-09-24 02:17:29 -0400 |
---|---|---|
committer | Greg Kroah-Hartman <gregkh@linuxfoundation.org> | 2013-10-05 10:13:09 -0400 |
commit | 808eea9d2912e4a3fb8cd45e3e3da94d114757fb (patch) | |
tree | ee108cbf8b8e1b2cbf910268df03d71e2df9348b | |
parent | bb343115253500dcee63ea78d765e4c2f2a57fc4 (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.c | 30 |
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; | ||
179 | bsearch: | 182 | bsearch: |
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 |