Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2013 Fusion IO. All rights reserved. |
| 3 | * |
| 4 | * This program is free software; you can redistribute it and/or |
| 5 | * modify it under the terms of the GNU General Public |
| 6 | * License v2 as published by the Free Software Foundation. |
| 7 | * |
| 8 | * This program is distributed in the hope that it will be useful, |
| 9 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 10 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 11 | * General Public License for more details. |
| 12 | * |
| 13 | * You should have received a copy of the GNU General Public |
| 14 | * License along with this program; if not, write to the |
| 15 | * Free Software Foundation, Inc., 59 Temple Place - Suite 330, |
| 16 | * Boston, MA 021110-1307, USA. |
| 17 | */ |
| 18 | |
| 19 | #include <linux/pagemap.h> |
| 20 | #include <linux/sched.h> |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 21 | #include <linux/slab.h> |
Byongho Lee | ee22184 | 2015-12-15 01:42:10 +0900 | [diff] [blame] | 22 | #include <linux/sizes.h> |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 23 | #include "btrfs-tests.h" |
Feifei Xu | ed9e4af | 2016-06-01 19:18:26 +0800 | [diff] [blame] | 24 | #include "../ctree.h" |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 25 | #include "../extent_io.h" |
| 26 | |
| 27 | #define PROCESS_UNLOCK (1 << 0) |
| 28 | #define PROCESS_RELEASE (1 << 1) |
| 29 | #define PROCESS_TEST_LOCKED (1 << 2) |
| 30 | |
| 31 | static noinline int process_page_range(struct inode *inode, u64 start, u64 end, |
| 32 | unsigned long flags) |
| 33 | { |
| 34 | int ret; |
| 35 | struct page *pages[16]; |
Kirill A. Shutemov | 09cbfea | 2016-04-01 15:29:47 +0300 | [diff] [blame] | 36 | unsigned long index = start >> PAGE_SHIFT; |
| 37 | unsigned long end_index = end >> PAGE_SHIFT; |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 38 | unsigned long nr_pages = end_index - index + 1; |
| 39 | int i; |
| 40 | int count = 0; |
| 41 | int loops = 0; |
| 42 | |
| 43 | while (nr_pages > 0) { |
| 44 | ret = find_get_pages_contig(inode->i_mapping, index, |
| 45 | min_t(unsigned long, nr_pages, |
| 46 | ARRAY_SIZE(pages)), pages); |
| 47 | for (i = 0; i < ret; i++) { |
| 48 | if (flags & PROCESS_TEST_LOCKED && |
| 49 | !PageLocked(pages[i])) |
| 50 | count++; |
| 51 | if (flags & PROCESS_UNLOCK && PageLocked(pages[i])) |
| 52 | unlock_page(pages[i]); |
Kirill A. Shutemov | 09cbfea | 2016-04-01 15:29:47 +0300 | [diff] [blame] | 53 | put_page(pages[i]); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 54 | if (flags & PROCESS_RELEASE) |
Kirill A. Shutemov | 09cbfea | 2016-04-01 15:29:47 +0300 | [diff] [blame] | 55 | put_page(pages[i]); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 56 | } |
| 57 | nr_pages -= ret; |
| 58 | index += ret; |
| 59 | cond_resched(); |
| 60 | loops++; |
| 61 | if (loops > 100000) { |
| 62 | printk(KERN_ERR "stuck in a loop, start %Lu, end %Lu, nr_pages %lu, ret %d\n", start, end, nr_pages, ret); |
| 63 | break; |
| 64 | } |
| 65 | } |
| 66 | return count; |
| 67 | } |
| 68 | |
Feifei Xu | b9ef22d | 2016-06-01 19:18:25 +0800 | [diff] [blame] | 69 | static int test_find_delalloc(u32 sectorsize) |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 70 | { |
| 71 | struct inode *inode; |
| 72 | struct extent_io_tree tmp; |
| 73 | struct page *page; |
| 74 | struct page *locked_page = NULL; |
| 75 | unsigned long index = 0; |
Byongho Lee | ee22184 | 2015-12-15 01:42:10 +0900 | [diff] [blame] | 76 | u64 total_dirty = SZ_256M; |
| 77 | u64 max_bytes = SZ_128M; |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 78 | u64 start, end, test_start; |
| 79 | u64 found; |
| 80 | int ret = -EINVAL; |
| 81 | |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 82 | test_msg("Running find delalloc tests\n"); |
| 83 | |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 84 | inode = btrfs_new_test_inode(); |
| 85 | if (!inode) { |
| 86 | test_msg("Failed to allocate test inode\n"); |
| 87 | return -ENOMEM; |
| 88 | } |
| 89 | |
Josef Bacik | c6100a4 | 2017-05-05 11:57:13 -0400 | [diff] [blame] | 90 | extent_io_tree_init(&tmp, inode); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 91 | |
| 92 | /* |
| 93 | * First go through and create and mark all of our pages dirty, we pin |
| 94 | * everything to make sure our pages don't get evicted and screw up our |
| 95 | * test. |
| 96 | */ |
Kirill A. Shutemov | 09cbfea | 2016-04-01 15:29:47 +0300 | [diff] [blame] | 97 | for (index = 0; index < (total_dirty >> PAGE_SHIFT); index++) { |
David Sterba | 8cce83b | 2016-01-22 10:28:24 +0100 | [diff] [blame] | 98 | page = find_or_create_page(inode->i_mapping, index, GFP_KERNEL); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 99 | if (!page) { |
| 100 | test_msg("Failed to allocate test page\n"); |
| 101 | ret = -ENOMEM; |
| 102 | goto out; |
| 103 | } |
| 104 | SetPageDirty(page); |
| 105 | if (index) { |
| 106 | unlock_page(page); |
| 107 | } else { |
Kirill A. Shutemov | 09cbfea | 2016-04-01 15:29:47 +0300 | [diff] [blame] | 108 | get_page(page); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 109 | locked_page = page; |
| 110 | } |
| 111 | } |
| 112 | |
| 113 | /* Test this scenario |
| 114 | * |--- delalloc ---| |
| 115 | * |--- search ---| |
| 116 | */ |
Filipe Manana | e3b8a48 | 2017-11-04 00:16:59 +0000 | [diff] [blame] | 117 | set_extent_delalloc(&tmp, 0, sectorsize - 1, 0, NULL); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 118 | start = 0; |
| 119 | end = 0; |
| 120 | found = find_lock_delalloc_range(inode, &tmp, locked_page, &start, |
| 121 | &end, max_bytes); |
| 122 | if (!found) { |
| 123 | test_msg("Should have found at least one delalloc\n"); |
| 124 | goto out_bits; |
| 125 | } |
Feifei Xu | b9ef22d | 2016-06-01 19:18:25 +0800 | [diff] [blame] | 126 | if (start != 0 || end != (sectorsize - 1)) { |
| 127 | test_msg("Expected start 0 end %u, got start %llu end %llu\n", |
| 128 | sectorsize - 1, start, end); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 129 | goto out_bits; |
| 130 | } |
| 131 | unlock_extent(&tmp, start, end); |
| 132 | unlock_page(locked_page); |
Kirill A. Shutemov | 09cbfea | 2016-04-01 15:29:47 +0300 | [diff] [blame] | 133 | put_page(locked_page); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 134 | |
| 135 | /* |
| 136 | * Test this scenario |
| 137 | * |
| 138 | * |--- delalloc ---| |
| 139 | * |--- search ---| |
| 140 | */ |
Byongho Lee | ee22184 | 2015-12-15 01:42:10 +0900 | [diff] [blame] | 141 | test_start = SZ_64M; |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 142 | locked_page = find_lock_page(inode->i_mapping, |
Kirill A. Shutemov | 09cbfea | 2016-04-01 15:29:47 +0300 | [diff] [blame] | 143 | test_start >> PAGE_SHIFT); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 144 | if (!locked_page) { |
| 145 | test_msg("Couldn't find the locked page\n"); |
| 146 | goto out_bits; |
| 147 | } |
Filipe Manana | e3b8a48 | 2017-11-04 00:16:59 +0000 | [diff] [blame] | 148 | set_extent_delalloc(&tmp, sectorsize, max_bytes - 1, 0, NULL); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 149 | start = test_start; |
| 150 | end = 0; |
| 151 | found = find_lock_delalloc_range(inode, &tmp, locked_page, &start, |
| 152 | &end, max_bytes); |
| 153 | if (!found) { |
| 154 | test_msg("Couldn't find delalloc in our range\n"); |
| 155 | goto out_bits; |
| 156 | } |
| 157 | if (start != test_start || end != max_bytes - 1) { |
| 158 | test_msg("Expected start %Lu end %Lu, got start %Lu, end " |
| 159 | "%Lu\n", test_start, max_bytes - 1, start, end); |
| 160 | goto out_bits; |
| 161 | } |
| 162 | if (process_page_range(inode, start, end, |
| 163 | PROCESS_TEST_LOCKED | PROCESS_UNLOCK)) { |
| 164 | test_msg("There were unlocked pages in the range\n"); |
| 165 | goto out_bits; |
| 166 | } |
| 167 | unlock_extent(&tmp, start, end); |
| 168 | /* locked_page was unlocked above */ |
Kirill A. Shutemov | 09cbfea | 2016-04-01 15:29:47 +0300 | [diff] [blame] | 169 | put_page(locked_page); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 170 | |
| 171 | /* |
| 172 | * Test this scenario |
| 173 | * |--- delalloc ---| |
| 174 | * |--- search ---| |
| 175 | */ |
Feifei Xu | b9ef22d | 2016-06-01 19:18:25 +0800 | [diff] [blame] | 176 | test_start = max_bytes + sectorsize; |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 177 | locked_page = find_lock_page(inode->i_mapping, test_start >> |
Kirill A. Shutemov | 09cbfea | 2016-04-01 15:29:47 +0300 | [diff] [blame] | 178 | PAGE_SHIFT); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 179 | if (!locked_page) { |
Nicholas D Steeves | 0132761 | 2016-05-19 21:18:45 -0400 | [diff] [blame] | 180 | test_msg("Couldn't find the locked page\n"); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 181 | goto out_bits; |
| 182 | } |
| 183 | start = test_start; |
| 184 | end = 0; |
| 185 | found = find_lock_delalloc_range(inode, &tmp, locked_page, &start, |
| 186 | &end, max_bytes); |
| 187 | if (found) { |
| 188 | test_msg("Found range when we shouldn't have\n"); |
| 189 | goto out_bits; |
| 190 | } |
| 191 | if (end != (u64)-1) { |
| 192 | test_msg("Did not return the proper end offset\n"); |
| 193 | goto out_bits; |
| 194 | } |
| 195 | |
| 196 | /* |
| 197 | * Test this scenario |
| 198 | * [------- delalloc -------| |
| 199 | * [max_bytes]|-- search--| |
| 200 | * |
| 201 | * We are re-using our test_start from above since it works out well. |
| 202 | */ |
Filipe Manana | e3b8a48 | 2017-11-04 00:16:59 +0000 | [diff] [blame] | 203 | set_extent_delalloc(&tmp, max_bytes, total_dirty - 1, 0, NULL); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 204 | start = test_start; |
| 205 | end = 0; |
| 206 | found = find_lock_delalloc_range(inode, &tmp, locked_page, &start, |
| 207 | &end, max_bytes); |
| 208 | if (!found) { |
| 209 | test_msg("Didn't find our range\n"); |
| 210 | goto out_bits; |
| 211 | } |
| 212 | if (start != test_start || end != total_dirty - 1) { |
| 213 | test_msg("Expected start %Lu end %Lu, got start %Lu end %Lu\n", |
| 214 | test_start, total_dirty - 1, start, end); |
| 215 | goto out_bits; |
| 216 | } |
| 217 | if (process_page_range(inode, start, end, |
| 218 | PROCESS_TEST_LOCKED | PROCESS_UNLOCK)) { |
| 219 | test_msg("Pages in range were not all locked\n"); |
| 220 | goto out_bits; |
| 221 | } |
| 222 | unlock_extent(&tmp, start, end); |
| 223 | |
| 224 | /* |
| 225 | * Now to test where we run into a page that is no longer dirty in the |
| 226 | * range we want to find. |
| 227 | */ |
Byongho Lee | ee22184 | 2015-12-15 01:42:10 +0900 | [diff] [blame] | 228 | page = find_get_page(inode->i_mapping, |
Kirill A. Shutemov | 09cbfea | 2016-04-01 15:29:47 +0300 | [diff] [blame] | 229 | (max_bytes + SZ_1M) >> PAGE_SHIFT); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 230 | if (!page) { |
| 231 | test_msg("Couldn't find our page\n"); |
| 232 | goto out_bits; |
| 233 | } |
| 234 | ClearPageDirty(page); |
Kirill A. Shutemov | 09cbfea | 2016-04-01 15:29:47 +0300 | [diff] [blame] | 235 | put_page(page); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 236 | |
| 237 | /* We unlocked it in the previous test */ |
| 238 | lock_page(locked_page); |
| 239 | start = test_start; |
| 240 | end = 0; |
| 241 | /* |
| 242 | * Currently if we fail to find dirty pages in the delalloc range we |
Kirill A. Shutemov | ea1754a | 2016-04-01 15:29:48 +0300 | [diff] [blame] | 243 | * will adjust max_bytes down to PAGE_SIZE and then re-search. If |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 244 | * this changes at any point in the future we will need to fix this |
| 245 | * tests expected behavior. |
| 246 | */ |
| 247 | found = find_lock_delalloc_range(inode, &tmp, locked_page, &start, |
| 248 | &end, max_bytes); |
| 249 | if (!found) { |
| 250 | test_msg("Didn't find our range\n"); |
| 251 | goto out_bits; |
| 252 | } |
Kirill A. Shutemov | 09cbfea | 2016-04-01 15:29:47 +0300 | [diff] [blame] | 253 | if (start != test_start && end != test_start + PAGE_SIZE - 1) { |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 254 | test_msg("Expected start %Lu end %Lu, got start %Lu end %Lu\n", |
Kirill A. Shutemov | 09cbfea | 2016-04-01 15:29:47 +0300 | [diff] [blame] | 255 | test_start, test_start + PAGE_SIZE - 1, start, |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 256 | end); |
| 257 | goto out_bits; |
| 258 | } |
| 259 | if (process_page_range(inode, start, end, PROCESS_TEST_LOCKED | |
| 260 | PROCESS_UNLOCK)) { |
| 261 | test_msg("Pages in range were not all locked\n"); |
| 262 | goto out_bits; |
| 263 | } |
| 264 | ret = 0; |
| 265 | out_bits: |
David Sterba | 9116621 | 2016-04-26 23:54:39 +0200 | [diff] [blame] | 266 | clear_extent_bits(&tmp, 0, total_dirty - 1, (unsigned)-1); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 267 | out: |
| 268 | if (locked_page) |
Kirill A. Shutemov | 09cbfea | 2016-04-01 15:29:47 +0300 | [diff] [blame] | 269 | put_page(locked_page); |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 270 | process_page_range(inode, 0, total_dirty - 1, |
| 271 | PROCESS_UNLOCK | PROCESS_RELEASE); |
| 272 | iput(inode); |
| 273 | return ret; |
| 274 | } |
| 275 | |
Omar Sandoval | 9426ce7 | 2016-09-22 17:24:23 -0700 | [diff] [blame] | 276 | static int check_eb_bitmap(unsigned long *bitmap, struct extent_buffer *eb, |
| 277 | unsigned long len) |
Feifei Xu | 34b3e6c | 2016-06-01 19:18:30 +0800 | [diff] [blame] | 278 | { |
Omar Sandoval | 9426ce7 | 2016-09-22 17:24:23 -0700 | [diff] [blame] | 279 | unsigned long i; |
| 280 | |
| 281 | for (i = 0; i < len * BITS_PER_BYTE; i++) { |
| 282 | int bit, bit1; |
| 283 | |
| 284 | bit = !!test_bit(i, bitmap); |
| 285 | bit1 = !!extent_buffer_test_bit(eb, 0, i); |
| 286 | if (bit1 != bit) { |
| 287 | test_msg("Bits do not match\n"); |
| 288 | return -EINVAL; |
| 289 | } |
| 290 | |
| 291 | bit1 = !!extent_buffer_test_bit(eb, i / BITS_PER_BYTE, |
| 292 | i % BITS_PER_BYTE); |
| 293 | if (bit1 != bit) { |
| 294 | test_msg("Offset bits do not match\n"); |
| 295 | return -EINVAL; |
| 296 | } |
| 297 | } |
| 298 | return 0; |
Feifei Xu | 34b3e6c | 2016-06-01 19:18:30 +0800 | [diff] [blame] | 299 | } |
| 300 | |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 301 | static int __test_eb_bitmaps(unsigned long *bitmap, struct extent_buffer *eb, |
| 302 | unsigned long len) |
| 303 | { |
Omar Sandoval | 9426ce7 | 2016-09-22 17:24:23 -0700 | [diff] [blame] | 304 | unsigned long i, j; |
| 305 | u32 x; |
| 306 | int ret; |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 307 | |
| 308 | memset(bitmap, 0, len); |
David Sterba | b159fa2 | 2016-11-08 18:09:03 +0100 | [diff] [blame] | 309 | memzero_extent_buffer(eb, 0, len); |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 310 | if (memcmp_extent_buffer(eb, bitmap, 0, len) != 0) { |
| 311 | test_msg("Bitmap was not zeroed\n"); |
| 312 | return -EINVAL; |
| 313 | } |
| 314 | |
| 315 | bitmap_set(bitmap, 0, len * BITS_PER_BYTE); |
| 316 | extent_buffer_bitmap_set(eb, 0, 0, len * BITS_PER_BYTE); |
Omar Sandoval | 9426ce7 | 2016-09-22 17:24:23 -0700 | [diff] [blame] | 317 | ret = check_eb_bitmap(bitmap, eb, len); |
| 318 | if (ret) { |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 319 | test_msg("Setting all bits failed\n"); |
Omar Sandoval | 9426ce7 | 2016-09-22 17:24:23 -0700 | [diff] [blame] | 320 | return ret; |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 321 | } |
| 322 | |
| 323 | bitmap_clear(bitmap, 0, len * BITS_PER_BYTE); |
| 324 | extent_buffer_bitmap_clear(eb, 0, 0, len * BITS_PER_BYTE); |
Omar Sandoval | 9426ce7 | 2016-09-22 17:24:23 -0700 | [diff] [blame] | 325 | ret = check_eb_bitmap(bitmap, eb, len); |
| 326 | if (ret) { |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 327 | test_msg("Clearing all bits failed\n"); |
Omar Sandoval | 9426ce7 | 2016-09-22 17:24:23 -0700 | [diff] [blame] | 328 | return ret; |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 329 | } |
| 330 | |
Feifei Xu | ed9e4af | 2016-06-01 19:18:26 +0800 | [diff] [blame] | 331 | /* Straddling pages test */ |
| 332 | if (len > PAGE_SIZE) { |
| 333 | bitmap_set(bitmap, |
| 334 | (PAGE_SIZE - sizeof(long) / 2) * BITS_PER_BYTE, |
| 335 | sizeof(long) * BITS_PER_BYTE); |
| 336 | extent_buffer_bitmap_set(eb, PAGE_SIZE - sizeof(long) / 2, 0, |
| 337 | sizeof(long) * BITS_PER_BYTE); |
Omar Sandoval | 9426ce7 | 2016-09-22 17:24:23 -0700 | [diff] [blame] | 338 | ret = check_eb_bitmap(bitmap, eb, len); |
| 339 | if (ret) { |
Feifei Xu | ed9e4af | 2016-06-01 19:18:26 +0800 | [diff] [blame] | 340 | test_msg("Setting straddling pages failed\n"); |
Omar Sandoval | 9426ce7 | 2016-09-22 17:24:23 -0700 | [diff] [blame] | 341 | return ret; |
Feifei Xu | ed9e4af | 2016-06-01 19:18:26 +0800 | [diff] [blame] | 342 | } |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 343 | |
Feifei Xu | ed9e4af | 2016-06-01 19:18:26 +0800 | [diff] [blame] | 344 | bitmap_set(bitmap, 0, len * BITS_PER_BYTE); |
| 345 | bitmap_clear(bitmap, |
| 346 | (PAGE_SIZE - sizeof(long) / 2) * BITS_PER_BYTE, |
| 347 | sizeof(long) * BITS_PER_BYTE); |
| 348 | extent_buffer_bitmap_set(eb, 0, 0, len * BITS_PER_BYTE); |
| 349 | extent_buffer_bitmap_clear(eb, PAGE_SIZE - sizeof(long) / 2, 0, |
| 350 | sizeof(long) * BITS_PER_BYTE); |
Omar Sandoval | 9426ce7 | 2016-09-22 17:24:23 -0700 | [diff] [blame] | 351 | ret = check_eb_bitmap(bitmap, eb, len); |
| 352 | if (ret) { |
Feifei Xu | ed9e4af | 2016-06-01 19:18:26 +0800 | [diff] [blame] | 353 | test_msg("Clearing straddling pages failed\n"); |
Omar Sandoval | 9426ce7 | 2016-09-22 17:24:23 -0700 | [diff] [blame] | 354 | return ret; |
Feifei Xu | ed9e4af | 2016-06-01 19:18:26 +0800 | [diff] [blame] | 355 | } |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 356 | } |
| 357 | |
| 358 | /* |
| 359 | * Generate a wonky pseudo-random bit pattern for the sake of not using |
| 360 | * something repetitive that could miss some hypothetical off-by-n bug. |
| 361 | */ |
| 362 | x = 0; |
Omar Sandoval | 9426ce7 | 2016-09-22 17:24:23 -0700 | [diff] [blame] | 363 | bitmap_clear(bitmap, 0, len * BITS_PER_BYTE); |
| 364 | extent_buffer_bitmap_clear(eb, 0, 0, len * BITS_PER_BYTE); |
| 365 | for (i = 0; i < len * BITS_PER_BYTE / 32; i++) { |
| 366 | x = (0x19660dULL * (u64)x + 0x3c6ef35fULL) & 0xffffffffU; |
| 367 | for (j = 0; j < 32; j++) { |
| 368 | if (x & (1U << j)) { |
| 369 | bitmap_set(bitmap, i * 32 + j, 1); |
| 370 | extent_buffer_bitmap_set(eb, 0, i * 32 + j, 1); |
| 371 | } |
| 372 | } |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 373 | } |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 374 | |
Omar Sandoval | 9426ce7 | 2016-09-22 17:24:23 -0700 | [diff] [blame] | 375 | ret = check_eb_bitmap(bitmap, eb, len); |
| 376 | if (ret) { |
| 377 | test_msg("Random bit pattern failed\n"); |
| 378 | return ret; |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 379 | } |
| 380 | |
| 381 | return 0; |
| 382 | } |
| 383 | |
Feifei Xu | b9ef22d | 2016-06-01 19:18:25 +0800 | [diff] [blame] | 384 | static int test_eb_bitmaps(u32 sectorsize, u32 nodesize) |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 385 | { |
Jeff Mahoney | da17066 | 2016-06-15 09:22:56 -0400 | [diff] [blame] | 386 | struct btrfs_fs_info *fs_info; |
Feifei Xu | b9ef22d | 2016-06-01 19:18:25 +0800 | [diff] [blame] | 387 | unsigned long len; |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 388 | unsigned long *bitmap; |
| 389 | struct extent_buffer *eb; |
| 390 | int ret; |
| 391 | |
| 392 | test_msg("Running extent buffer bitmap tests\n"); |
Feifei Xu | ed9e4af | 2016-06-01 19:18:26 +0800 | [diff] [blame] | 393 | |
| 394 | /* |
| 395 | * In ppc64, sectorsize can be 64K, thus 4 * 64K will be larger than |
| 396 | * BTRFS_MAX_METADATA_BLOCKSIZE. |
| 397 | */ |
| 398 | len = (sectorsize < BTRFS_MAX_METADATA_BLOCKSIZE) |
| 399 | ? sectorsize * 4 : sectorsize; |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 400 | |
Jeff Mahoney | da17066 | 2016-06-15 09:22:56 -0400 | [diff] [blame] | 401 | fs_info = btrfs_alloc_dummy_fs_info(len, len); |
| 402 | |
David Sterba | 8cce83b | 2016-01-22 10:28:24 +0100 | [diff] [blame] | 403 | bitmap = kmalloc(len, GFP_KERNEL); |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 404 | if (!bitmap) { |
| 405 | test_msg("Couldn't allocate test bitmap\n"); |
| 406 | return -ENOMEM; |
| 407 | } |
| 408 | |
Jeff Mahoney | da17066 | 2016-06-15 09:22:56 -0400 | [diff] [blame] | 409 | eb = __alloc_dummy_extent_buffer(fs_info, 0, len); |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 410 | if (!eb) { |
| 411 | test_msg("Couldn't allocate test extent buffer\n"); |
| 412 | kfree(bitmap); |
| 413 | return -ENOMEM; |
| 414 | } |
| 415 | |
| 416 | ret = __test_eb_bitmaps(bitmap, eb, len); |
| 417 | if (ret) |
| 418 | goto out; |
| 419 | |
| 420 | /* Do it over again with an extent buffer which isn't page-aligned. */ |
| 421 | free_extent_buffer(eb); |
Feifei Xu | b9ef22d | 2016-06-01 19:18:25 +0800 | [diff] [blame] | 422 | eb = __alloc_dummy_extent_buffer(NULL, nodesize / 2, len); |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 423 | if (!eb) { |
| 424 | test_msg("Couldn't allocate test extent buffer\n"); |
| 425 | kfree(bitmap); |
| 426 | return -ENOMEM; |
| 427 | } |
| 428 | |
| 429 | ret = __test_eb_bitmaps(bitmap, eb, len); |
| 430 | out: |
| 431 | free_extent_buffer(eb); |
| 432 | kfree(bitmap); |
| 433 | return ret; |
| 434 | } |
| 435 | |
Feifei Xu | b9ef22d | 2016-06-01 19:18:25 +0800 | [diff] [blame] | 436 | int btrfs_test_extent_io(u32 sectorsize, u32 nodesize) |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 437 | { |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 438 | int ret; |
| 439 | |
| 440 | test_msg("Running extent I/O tests\n"); |
| 441 | |
Feifei Xu | b9ef22d | 2016-06-01 19:18:25 +0800 | [diff] [blame] | 442 | ret = test_find_delalloc(sectorsize); |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 443 | if (ret) |
| 444 | goto out; |
| 445 | |
Feifei Xu | b9ef22d | 2016-06-01 19:18:25 +0800 | [diff] [blame] | 446 | ret = test_eb_bitmaps(sectorsize, nodesize); |
Omar Sandoval | 0f33122 | 2015-09-29 20:50:31 -0700 | [diff] [blame] | 447 | out: |
| 448 | test_msg("Extent I/O tests finished\n"); |
| 449 | return ret; |
Josef Bacik | 294e30f | 2013-10-09 12:00:56 -0400 | [diff] [blame] | 450 | } |