/* * GRUB -- GRand Unified Bootloader * Copyright (C) 2015 Free Software Foundation, Inc. * * GRUB is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * GRUB is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with GRUB. If not, see <http://www.gnu.org/licenses/>. */ #include <grub/test.h> #include <grub/dl.h> #include <grub/misc.h> GRUB_MOD_LICENSE ("GPLv3+"); static grub_uint64_t vectors[] = { 0xffffffffffffffffULL, 1, 2, 0, 0x0102030405060708ULL }; /* We're testing shifts, don't replace access to this with a shift. */ static const grub_uint8_t bitmask[] = { 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80 }; typedef union { grub_uint64_t v64; grub_uint8_t v8[8]; } grub_raw_u64_t; static int get_bit64 (grub_uint64_t v, int b) { grub_raw_u64_t vr = { .v64 = v }; grub_uint8_t *p = vr.v8; if (b >= 64) return 0; #ifdef GRUB_CPU_WORDS_BIGENDIAN p += 7 - b / 8; #else p += b / 8; #endif return !!(*p & bitmask[b % 8]); } static grub_uint64_t set_bit64 (grub_uint64_t v, int b) { grub_raw_u64_t vr = { .v64 = v }; grub_uint8_t *p = vr.v8; if (b >= 64) return v; #ifdef GRUB_CPU_WORDS_BIGENDIAN p += 7 - b / 8; #else p += b / 8; #endif *p |= bitmask[b % 8]; return vr.v64; } static grub_uint64_t left_shift64 (grub_uint64_t v, int s) { grub_uint64_t r = 0; int i; for (i = 0; i + s < 64; i++) if (get_bit64 (v, i)) r = set_bit64 (r, i + s); return r; } static grub_uint64_t right_shift64 (grub_uint64_t v, int s) { grub_uint64_t r = 0; int i; for (i = s; i < 64; i++) if (get_bit64 (v, i)) r = set_bit64 (r, i - s); return r; } static grub_uint64_t arithmetic_right_shift64 (grub_uint64_t v, int s) { grub_uint64_t r = 0; int i; for (i = s; i < 64; i++) if (get_bit64 (v, i)) r = set_bit64 (r, i - s); if (get_bit64 (v, 63)) for (i -= s; i < 64; i++) r = set_bit64 (r, i); return r; } static void test64 (grub_uint64_t v) { int i; for (i = 0; i < 64; i++) { grub_test_assert ((v << i) == left_shift64 (v, i), "lshift wrong: 0x%llx << %d: 0x%llx, 0x%llx", (long long) v, i, (long long) (v << i), (long long) left_shift64 (v, i)); grub_test_assert ((v >> i) == right_shift64 (v, i), "rshift wrong: 0x%llx >> %d: 0x%llx, 0x%llx", (long long) v, i, (long long) (v >> i), (long long) right_shift64 (v, i)); grub_test_assert ((((grub_int64_t) v) >> i) == (grub_int64_t) arithmetic_right_shift64 (v, i), "arithmetic rshift wrong: ((grub_int64_t) 0x%llx) >> %d: 0x%llx, 0x%llx", (long long) v, i, (long long) (((grub_int64_t) v) >> i), (long long) arithmetic_right_shift64 (v, i)); } } static void test_all(grub_uint64_t a) { test64 (a); } static void shift_test (void) { grub_uint64_t a = 404, b = 7; grub_size_t i; for (i = 0; i < ARRAY_SIZE (vectors); i++) { test_all (vectors[i]); } for (i = 0; i < 4000; i++) { a = 17 * a + 13 * b; b = 23 * a + 29 * b; if (b == 0) b = 1; if (a == 0) a = 1; test_all (a); test_all (b); } } /* Register example_test method as a functional test. */ GRUB_FUNCTIONAL_TEST (shift_test, shift_test);