1 /* 2 * Copyright (C) 2013 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #include <stdlib.h> 18 #include <string.h> 19 #include <sys/mman.h> 20 21 #include <gtest/gtest.h> 22 #include "buffer_tests.h" 23 24 // For the comparison buffer tests, the maximum length to test for the 25 // miscompare checks. 26 #define MISCMP_MAX_LENGTH 512 27 28 #define FENCEPOST_LENGTH 8 29 30 static int g_single_aligns[][2] = { 31 // Both buffers at same alignment. 32 { 1, 0 }, 33 { 2, 0 }, 34 { 4, 0 }, 35 { 8, 0 }, 36 { 16, 0 }, 37 { 32, 0 }, 38 { 64, 0 }, 39 { 128, 0 }, 40 41 // General unaligned cases. 42 { 4, 1 }, 43 { 4, 2 }, 44 { 4, 3 }, 45 46 { 8, 1 }, 47 { 8, 2 }, 48 { 8, 3 }, 49 { 8, 4 }, 50 { 8, 5 }, 51 { 8, 6 }, 52 { 8, 7 }, 53 54 { 128, 1 }, 55 { 128, 4 }, 56 { 128, 8 }, 57 { 128, 12 }, 58 { 128, 16 }, 59 }; 60 61 static const size_t g_single_aligns_len = sizeof(g_single_aligns)/sizeof(int[2]); 62 63 // Set of multiple buffer alignment combinations to be used for string/memory 64 // testing routines. 65 static int g_double_aligns[][4] = { 66 // Both buffers at same alignment. 67 { 1, 0, 1, 0 }, 68 { 2, 0, 2, 0 }, 69 { 4, 0, 4, 0 }, 70 { 8, 0, 8, 0 }, 71 { 16, 0, 16, 0 }, 72 { 32, 0, 32, 0 }, 73 { 64, 0, 64, 0 }, 74 { 128, 0, 128, 0 }, 75 76 // Different word alignments between buffers. 77 { 8, 0, 4, 0 }, 78 { 4, 0, 8, 0 }, 79 { 16, 0, 4, 0 }, 80 { 4, 0, 16, 0 }, 81 82 // General unaligned cases. 83 { 4, 0, 4, 1 }, 84 { 4, 0, 4, 2 }, 85 { 4, 0, 4, 3 }, 86 87 { 4, 1, 4, 0 }, 88 { 4, 1, 4, 1 }, 89 { 4, 1, 4, 2 }, 90 { 4, 1, 4, 3 }, 91 92 { 4, 2, 4, 0 }, 93 { 4, 2, 4, 1 }, 94 { 4, 2, 4, 2 }, 95 { 4, 2, 4, 3 }, 96 97 { 4, 3, 4, 0 }, 98 { 4, 3, 4, 1 }, 99 { 4, 3, 4, 2 }, 100 { 4, 3, 4, 3 }, 101 102 { 8, 0, 8, 1 }, 103 { 8, 0, 8, 2 }, 104 { 8, 0, 8, 3 }, 105 { 8, 0, 8, 4 }, 106 { 8, 0, 8, 5 }, 107 { 8, 0, 8, 6 }, 108 { 8, 0, 8, 7 }, 109 110 { 8, 1, 8, 0 }, 111 { 8, 1, 8, 1 }, 112 { 8, 1, 8, 2 }, 113 { 8, 1, 8, 3 }, 114 { 8, 1, 8, 4 }, 115 { 8, 1, 8, 5 }, 116 { 8, 1, 8, 6 }, 117 { 8, 1, 8, 7 }, 118 119 { 8, 2, 8, 0 }, 120 { 8, 2, 8, 1 }, 121 { 8, 2, 8, 2 }, 122 { 8, 2, 8, 3 }, 123 { 8, 2, 8, 4 }, 124 { 8, 2, 8, 5 }, 125 { 8, 2, 8, 6 }, 126 { 8, 2, 8, 7 }, 127 128 { 8, 3, 8, 0 }, 129 { 8, 3, 8, 1 }, 130 { 8, 3, 8, 2 }, 131 { 8, 3, 8, 3 }, 132 { 8, 3, 8, 4 }, 133 { 8, 3, 8, 5 }, 134 { 8, 3, 8, 6 }, 135 { 8, 3, 8, 7 }, 136 137 { 8, 4, 8, 0 }, 138 { 8, 4, 8, 1 }, 139 { 8, 4, 8, 2 }, 140 { 8, 4, 8, 3 }, 141 { 8, 4, 8, 4 }, 142 { 8, 4, 8, 5 }, 143 { 8, 4, 8, 6 }, 144 { 8, 4, 8, 7 }, 145 146 { 8, 5, 8, 0 }, 147 { 8, 5, 8, 1 }, 148 { 8, 5, 8, 2 }, 149 { 8, 5, 8, 3 }, 150 { 8, 5, 8, 4 }, 151 { 8, 5, 8, 5 }, 152 { 8, 5, 8, 6 }, 153 { 8, 5, 8, 7 }, 154 155 { 8, 6, 8, 0 }, 156 { 8, 6, 8, 1 }, 157 { 8, 6, 8, 2 }, 158 { 8, 6, 8, 3 }, 159 { 8, 6, 8, 4 }, 160 { 8, 6, 8, 5 }, 161 { 8, 6, 8, 6 }, 162 { 8, 6, 8, 7 }, 163 164 { 8, 7, 8, 0 }, 165 { 8, 7, 8, 1 }, 166 { 8, 7, 8, 2 }, 167 { 8, 7, 8, 3 }, 168 { 8, 7, 8, 4 }, 169 { 8, 7, 8, 5 }, 170 { 8, 7, 8, 6 }, 171 { 8, 7, 8, 7 }, 172 173 { 128, 1, 128, 4 }, 174 { 128, 1, 128, 8 }, 175 { 128, 1, 128, 12 }, 176 { 128, 1, 128, 16 }, 177 { 128, 4, 128, 1 }, 178 { 128, 8, 128, 1 }, 179 { 128, 12, 128, 1 }, 180 { 128, 16, 128, 1 }, 181 }; 182 183 static const size_t g_double_aligns_len = sizeof(g_double_aligns)/sizeof(int[4]); 184 SetIncrement(size_t len)185 static size_t SetIncrement(size_t len) { 186 if (len >= 4096) { 187 return 1024; 188 } else if (len >= 1024) { 189 return 256; 190 } 191 return 1; 192 } 193 194 // Return a pointer into the current buffer with the specified alignment. GetAlignedPtr(void * orig_ptr,int alignment,int or_mask)195 static void *GetAlignedPtr(void *orig_ptr, int alignment, int or_mask) { 196 uint64_t ptr = reinterpret_cast<uint64_t>(orig_ptr); 197 if (alignment > 0) { 198 // When setting the alignment, set it to exactly the alignment chosen. 199 // The pointer returned will be guaranteed not to be aligned to anything 200 // more than that. 201 ptr += alignment - (ptr & (alignment - 1)); 202 ptr |= alignment | or_mask; 203 } 204 205 return reinterpret_cast<void*>(ptr); 206 } 207 SetFencepost(uint8_t * buffer)208 static void SetFencepost(uint8_t *buffer) { 209 for (int i = 0; i < FENCEPOST_LENGTH; i += 2) { 210 buffer[i] = 0xde; 211 buffer[i+1] = 0xad; 212 } 213 } 214 VerifyFencepost(uint8_t * buffer)215 static void VerifyFencepost(uint8_t *buffer) { 216 for (int i = 0; i < FENCEPOST_LENGTH; i += 2) { 217 if (buffer[i] != 0xde || buffer[i+1] != 0xad) { 218 uint8_t expected_value; 219 if (buffer[i] == 0xde) { 220 i++; 221 expected_value = 0xad; 222 } else { 223 expected_value = 0xde; 224 } 225 ASSERT_EQ(expected_value, buffer[i]); 226 } 227 } 228 } 229 RunSingleBufferAlignTest(size_t max_test_size,void (* test_func)(uint8_t *,size_t),size_t (* set_incr)(size_t))230 void RunSingleBufferAlignTest( 231 size_t max_test_size, void (*test_func)(uint8_t*, size_t), 232 size_t (*set_incr)(size_t)) { 233 if (!set_incr) { 234 set_incr = SetIncrement; 235 } 236 237 // Allocate one large buffer with lots of extra space so that we can 238 // guarantee that the all possible alignments will fit. 239 uint8_t *buf = new uint8_t[3*max_test_size]; 240 241 uint8_t *buf_align; 242 for (size_t i = 0; i < g_single_aligns_len; i++) { 243 size_t incr = 1; 244 for (size_t len = 0; len <= max_test_size; len += incr) { 245 incr = set_incr(len); 246 247 buf_align = reinterpret_cast<uint8_t*>(GetAlignedPtr( 248 buf+FENCEPOST_LENGTH, g_single_aligns[i][0], g_single_aligns[i][1])); 249 250 SetFencepost(&buf_align[-FENCEPOST_LENGTH]); 251 SetFencepost(&buf_align[len]); 252 253 test_func(buf_align, len); 254 255 VerifyFencepost(&buf_align[-FENCEPOST_LENGTH]); 256 VerifyFencepost(&buf_align[len]); 257 } 258 } 259 delete[] buf; 260 } 261 RunSrcDstBufferAlignTest(size_t max_test_size,void (* test_func)(uint8_t *,uint8_t *,size_t),size_t (* set_incr)(size_t))262 void RunSrcDstBufferAlignTest( 263 size_t max_test_size, void (*test_func)(uint8_t*, uint8_t*, size_t), 264 size_t (*set_incr)(size_t)) { 265 if (!set_incr) { 266 set_incr = SetIncrement; 267 } 268 269 // Allocate two large buffers for all of the testing. 270 uint8_t* src = new uint8_t[3*max_test_size]; 271 uint8_t* dst = new uint8_t[3*max_test_size]; 272 273 uint8_t* src_align; 274 uint8_t* dst_align; 275 for (size_t i = 0; i < g_double_aligns_len; i++) { 276 size_t incr = 1; 277 for (size_t len = 0; len <= max_test_size; len += incr) { 278 incr = set_incr(len); 279 280 src_align = 281 reinterpret_cast<uint8_t*>(GetAlignedPtr( 282 src+FENCEPOST_LENGTH, g_double_aligns[i][0], g_double_aligns[i][1])); 283 dst_align = 284 reinterpret_cast<uint8_t*>(GetAlignedPtr( 285 dst+FENCEPOST_LENGTH, g_double_aligns[i][2], g_double_aligns[i][3])); 286 SetFencepost(&dst_align[-FENCEPOST_LENGTH]); 287 SetFencepost(&dst_align[len]); 288 289 test_func(src_align, dst_align, len); 290 291 VerifyFencepost(&dst_align[-FENCEPOST_LENGTH]); 292 VerifyFencepost(&dst_align[len]); 293 } 294 } 295 delete[] src; 296 delete[] dst; 297 } 298 RunCmpBufferAlignTest(size_t max_test_size,void (* test_cmp_func)(uint8_t *,uint8_t *,size_t),void (* test_miscmp_func)(uint8_t *,uint8_t *,size_t,size_t),size_t (* set_incr)(size_t))299 void RunCmpBufferAlignTest( 300 size_t max_test_size, void (*test_cmp_func)(uint8_t*, uint8_t*, size_t), 301 void (*test_miscmp_func)(uint8_t*, uint8_t*, size_t, size_t), 302 size_t (*set_incr)(size_t)) { 303 if (!set_incr) { 304 set_incr = SetIncrement; 305 } 306 307 // Allocate two large buffers for all of the testing. 308 uint8_t* buf1 = new uint8_t[3*max_test_size]; 309 uint8_t* buf2 = new uint8_t[3*max_test_size]; 310 311 uint8_t* buf1_align; 312 uint8_t* buf2_align; 313 for (size_t i = 0; i < g_double_aligns_len; i++) { 314 size_t incr = 1; 315 for (size_t len = 0; len <= max_test_size; len += incr) { 316 incr = set_incr(len); 317 318 buf1_align = 319 reinterpret_cast<uint8_t*>(GetAlignedPtr( 320 buf1, g_double_aligns[i][0], g_double_aligns[i][1])); 321 buf2_align = 322 reinterpret_cast<uint8_t*>(GetAlignedPtr( 323 buf2, g_double_aligns[i][2], g_double_aligns[i][3])); 324 325 // Check by putting all zeroes after both buffers. 326 memset(buf1_align+len, 0, 32); 327 memset(buf2_align+len, 0, 32); 328 test_cmp_func(buf1_align, buf2_align, len); 329 330 // Check by putting different values after both buffers. 331 for (size_t j = 0; j < 32; j++) { 332 buf1_align[len+j] = j; 333 buf2_align[len+j] = j+1; 334 } 335 test_cmp_func(buf1_align, buf2_align, len); 336 337 if (len > 0) { 338 // Change the lengths of the buffers and verify that there are 339 // miscompares. 340 for (size_t len2 = len+1; len2 < len+32; len2++) { 341 test_miscmp_func(buf1_align, buf2_align, len, len2); 342 test_miscmp_func(buf1_align, buf2_align, len2, len); 343 } 344 } 345 } 346 } 347 delete[] buf1; 348 delete[] buf2; 349 } 350 RunSingleBufferOverreadTest(void (* test_func)(uint8_t *,size_t))351 void RunSingleBufferOverreadTest(void (*test_func)(uint8_t*, size_t)) { 352 // In order to verify that functions are not reading past the end of the 353 // src, create data that ends exactly at an unreadable memory boundary. 354 size_t pagesize = static_cast<size_t>(sysconf(_SC_PAGE_SIZE)); 355 uint8_t* memory; 356 ASSERT_TRUE(posix_memalign(reinterpret_cast<void**>(&memory), pagesize, 357 2*pagesize) == 0); 358 memset(memory, 0x23, 2*pagesize); 359 360 // Make the second page unreadable and unwritable. 361 ASSERT_TRUE(mprotect(&memory[pagesize], pagesize, PROT_NONE) == 0); 362 363 for (size_t i = 0; i < pagesize; i++) { 364 uint8_t* buf = &memory[pagesize-i]; 365 366 test_func(buf, i); 367 } 368 ASSERT_TRUE(mprotect(&memory[pagesize], pagesize, PROT_READ | PROT_WRITE) == 0); 369 free(memory); 370 } 371 RunSrcDstBufferOverreadTest(void (* test_func)(uint8_t *,uint8_t *,size_t))372 void RunSrcDstBufferOverreadTest(void (*test_func)(uint8_t*, uint8_t*, size_t)) { 373 // In order to verify that functions are not reading past the end of the 374 // src, create data that ends exactly at an unreadable memory boundary. 375 size_t pagesize = static_cast<size_t>(sysconf(_SC_PAGE_SIZE)); 376 uint8_t* memory; 377 ASSERT_TRUE(posix_memalign(reinterpret_cast<void**>(&memory), pagesize, 378 2*pagesize) == 0); 379 memset(memory, 0x23, 2*pagesize); 380 381 // Make the second page unreadable and unwritable. 382 ASSERT_TRUE(mprotect(&memory[pagesize], pagesize, PROT_NONE) == 0); 383 384 uint8_t* dst_buffer = new uint8_t[2*pagesize]; 385 // Change the dst alignment as we change the source. 386 for (size_t i = 0; i < 16; i++) { 387 uint8_t* dst = &dst_buffer[i]; 388 for (size_t j = 0; j < pagesize; j++) { 389 uint8_t* src = &memory[pagesize-j]; 390 391 test_func(src, dst, j); 392 } 393 } 394 ASSERT_TRUE(mprotect(&memory[pagesize], pagesize, PROT_READ | PROT_WRITE) == 0); 395 free(memory); 396 delete[] dst_buffer; 397 } 398 RunCmpBufferOverreadTest(void (* test_cmp_func)(uint8_t *,uint8_t *,size_t),void (* test_miscmp_func)(uint8_t *,uint8_t *,size_t,size_t))399 void RunCmpBufferOverreadTest( 400 void (*test_cmp_func)(uint8_t*, uint8_t*, size_t), 401 void (*test_miscmp_func)(uint8_t*, uint8_t*, size_t, size_t)) { 402 // In order to verify that functions are not reading past the end of either 403 // of the bufs, create both buffers that end exactly at an unreadable memory 404 // boundary. 405 size_t pagesize = static_cast<size_t>(sysconf(_SC_PAGE_SIZE)); 406 uint8_t* memory1; 407 ASSERT_TRUE(posix_memalign(reinterpret_cast<void**>(&memory1), pagesize, 408 2*pagesize) == 0); 409 memset(memory1, 0x23, 2*pagesize); 410 411 // Make the second page unreadable and unwritable. 412 ASSERT_TRUE(mprotect(&memory1[pagesize], pagesize, PROT_NONE) == 0); 413 414 uint8_t* memory2; 415 ASSERT_TRUE(posix_memalign(reinterpret_cast<void**>(&memory2), pagesize, 416 2*pagesize) == 0); 417 memset(memory2, 0x23, 2*pagesize); 418 419 // Make the second page unreadable and unwritable. 420 ASSERT_TRUE(mprotect(&memory2[pagesize], pagesize, PROT_NONE) == 0); 421 422 for (size_t i = 0; i < pagesize; i++) { 423 uint8_t* buf1 = &memory1[pagesize-i]; 424 uint8_t* buf2 = &memory2[pagesize-i]; 425 426 test_cmp_func(buf1, buf2, i); 427 } 428 429 // Don't cycle through pagesize, MISCMP_MAX_LENGTH bytes should be good. 430 size_t miscmp_len; 431 if (pagesize > MISCMP_MAX_LENGTH) { 432 miscmp_len = MISCMP_MAX_LENGTH; 433 } else { 434 miscmp_len = pagesize; 435 } 436 for (size_t i = 1; i < miscmp_len; i++) { 437 uint8_t* buf1 = &memory1[pagesize-i]; 438 for (size_t j = 1; j < miscmp_len; j++) { 439 if (j == i) 440 continue; 441 442 uint8_t* buf2 = &memory2[pagesize-j]; 443 444 test_miscmp_func(buf1, buf2, i, j); 445 } 446 } 447 448 ASSERT_TRUE(mprotect(&memory1[pagesize], pagesize, PROT_READ | PROT_WRITE) == 0); 449 ASSERT_TRUE(mprotect(&memory2[pagesize], pagesize, PROT_READ | PROT_WRITE) == 0); 450 free(memory1); 451 free(memory2); 452 } 453