/*
* 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 .
*/
#include
#include
#include
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);