39101b2265
Fixes the issue caused by the fact that in C in the expression of the form -1234L only 1234L is the actual literal, the unary minus is an operation applied to the literal. Which means that to express the lower bound for the type one has to negate the upper bound and subtract 1. Original error: Expected test_data[i].expected.tv_sec == timestamp.tv_sec, but test_data[i].expected.tv_sec == -2147483648 timestamp.tv_sec == 2147483648 1901-12-13 Lower bound of 32bit < 0 timestamp, no extra bits: msb:1 lower_bound:1 extra_bits: 0 Expected test_data[i].expected.tv_sec == timestamp.tv_sec, but test_data[i].expected.tv_sec == 2147483648 timestamp.tv_sec == 6442450944 2038-01-19 Lower bound of 32bit <0 timestamp, lo extra sec bit on: msb:1 lower_bound:1 extra_bits: 1 Expected test_data[i].expected.tv_sec == timestamp.tv_sec, but test_data[i].expected.tv_sec == 6442450944 timestamp.tv_sec == 10737418240 2174-02-25 Lower bound of 32bit <0 timestamp, hi extra sec bit on: msb:1 lower_bound:1 extra_bits: 2 not ok 1 - inode_test_xtimestamp_decoding not ok 1 - ext4_inode_test Reported-by: Geert Uytterhoeven <geert@linux-m68k.org> Signed-off-by: Iurii Zaikin <yzaikin@google.com> Tested-by: Geert Uytterhoeven <geert@linux-m68k.org> Acked-by: Brendan Higgins <brendanhiggins@google.com> Signed-off-by: Shuah Khan <skhan@linuxfoundation.org>
273 lines
7.3 KiB
C
273 lines
7.3 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* KUnit test of ext4 inode that verify the seconds part of [a/c/m]
|
|
* timestamps in ext4 inode structs are decoded correctly.
|
|
*/
|
|
|
|
#include <kunit/test.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/time64.h>
|
|
|
|
#include "ext4.h"
|
|
|
|
/*
|
|
* For constructing the nonnegative timestamp lower bound value.
|
|
* binary: 00000000 00000000 00000000 00000000
|
|
*/
|
|
#define LOWER_MSB_0 0L
|
|
/*
|
|
* For constructing the nonnegative timestamp upper bound value.
|
|
* binary: 01111111 11111111 11111111 11111111
|
|
*
|
|
*/
|
|
#define UPPER_MSB_0 0x7fffffffL
|
|
/*
|
|
* For constructing the negative timestamp lower bound value.
|
|
* binary: 10000000 00000000 00000000 00000000
|
|
*/
|
|
#define LOWER_MSB_1 (-(UPPER_MSB_0) - 1L) /* avoid overflow */
|
|
/*
|
|
* For constructing the negative timestamp upper bound value.
|
|
* binary: 11111111 11111111 11111111 11111111
|
|
*/
|
|
#define UPPER_MSB_1 (-1L)
|
|
/*
|
|
* Upper bound for nanoseconds value supported by the encoding.
|
|
* binary: 00111111 11111111 11111111 11111111
|
|
*/
|
|
#define MAX_NANOSECONDS ((1L << 30) - 1)
|
|
|
|
#define CASE_NAME_FORMAT "%s: msb:%x lower_bound:%x extra_bits: %x"
|
|
|
|
#define LOWER_BOUND_NEG_NO_EXTRA_BITS_CASE\
|
|
"1901-12-13 Lower bound of 32bit < 0 timestamp, no extra bits"
|
|
#define UPPER_BOUND_NEG_NO_EXTRA_BITS_CASE\
|
|
"1969-12-31 Upper bound of 32bit < 0 timestamp, no extra bits"
|
|
#define LOWER_BOUND_NONNEG_NO_EXTRA_BITS_CASE\
|
|
"1970-01-01 Lower bound of 32bit >=0 timestamp, no extra bits"
|
|
#define UPPER_BOUND_NONNEG_NO_EXTRA_BITS_CASE\
|
|
"2038-01-19 Upper bound of 32bit >=0 timestamp, no extra bits"
|
|
#define LOWER_BOUND_NEG_LO_1_CASE\
|
|
"2038-01-19 Lower bound of 32bit <0 timestamp, lo extra sec bit on"
|
|
#define UPPER_BOUND_NEG_LO_1_CASE\
|
|
"2106-02-07 Upper bound of 32bit <0 timestamp, lo extra sec bit on"
|
|
#define LOWER_BOUND_NONNEG_LO_1_CASE\
|
|
"2106-02-07 Lower bound of 32bit >=0 timestamp, lo extra sec bit on"
|
|
#define UPPER_BOUND_NONNEG_LO_1_CASE\
|
|
"2174-02-25 Upper bound of 32bit >=0 timestamp, lo extra sec bit on"
|
|
#define LOWER_BOUND_NEG_HI_1_CASE\
|
|
"2174-02-25 Lower bound of 32bit <0 timestamp, hi extra sec bit on"
|
|
#define UPPER_BOUND_NEG_HI_1_CASE\
|
|
"2242-03-16 Upper bound of 32bit <0 timestamp, hi extra sec bit on"
|
|
#define LOWER_BOUND_NONNEG_HI_1_CASE\
|
|
"2242-03-16 Lower bound of 32bit >=0 timestamp, hi extra sec bit on"
|
|
#define UPPER_BOUND_NONNEG_HI_1_CASE\
|
|
"2310-04-04 Upper bound of 32bit >=0 timestamp, hi extra sec bit on"
|
|
#define UPPER_BOUND_NONNEG_HI_1_NS_1_CASE\
|
|
"2310-04-04 Upper bound of 32bit>=0 timestamp, hi extra sec bit 1. 1 ns"
|
|
#define LOWER_BOUND_NONNEG_HI_1_NS_MAX_CASE\
|
|
"2378-04-22 Lower bound of 32bit>= timestamp. Extra sec bits 1. Max ns"
|
|
#define LOWER_BOUND_NONNEG_EXTRA_BITS_1_CASE\
|
|
"2378-04-22 Lower bound of 32bit >=0 timestamp. All extra sec bits on"
|
|
#define UPPER_BOUND_NONNEG_EXTRA_BITS_1_CASE\
|
|
"2446-05-10 Upper bound of 32bit >=0 timestamp. All extra sec bits on"
|
|
|
|
struct timestamp_expectation {
|
|
const char *test_case_name;
|
|
struct timespec64 expected;
|
|
u32 extra_bits;
|
|
bool msb_set;
|
|
bool lower_bound;
|
|
};
|
|
|
|
static time64_t get_32bit_time(const struct timestamp_expectation * const test)
|
|
{
|
|
if (test->msb_set) {
|
|
if (test->lower_bound)
|
|
return LOWER_MSB_1;
|
|
|
|
return UPPER_MSB_1;
|
|
}
|
|
|
|
if (test->lower_bound)
|
|
return LOWER_MSB_0;
|
|
return UPPER_MSB_0;
|
|
}
|
|
|
|
|
|
/*
|
|
* Test data is derived from the table in the Inode Timestamps section of
|
|
* Documentation/filesystems/ext4/inodes.rst.
|
|
*/
|
|
static void inode_test_xtimestamp_decoding(struct kunit *test)
|
|
{
|
|
const struct timestamp_expectation test_data[] = {
|
|
{
|
|
.test_case_name = LOWER_BOUND_NEG_NO_EXTRA_BITS_CASE,
|
|
.msb_set = true,
|
|
.lower_bound = true,
|
|
.extra_bits = 0,
|
|
.expected = {.tv_sec = -0x80000000LL, .tv_nsec = 0L},
|
|
},
|
|
|
|
{
|
|
.test_case_name = UPPER_BOUND_NEG_NO_EXTRA_BITS_CASE,
|
|
.msb_set = true,
|
|
.lower_bound = false,
|
|
.extra_bits = 0,
|
|
.expected = {.tv_sec = -1LL, .tv_nsec = 0L},
|
|
},
|
|
|
|
{
|
|
.test_case_name = LOWER_BOUND_NONNEG_NO_EXTRA_BITS_CASE,
|
|
.msb_set = false,
|
|
.lower_bound = true,
|
|
.extra_bits = 0,
|
|
.expected = {0LL, 0L},
|
|
},
|
|
|
|
{
|
|
.test_case_name = UPPER_BOUND_NONNEG_NO_EXTRA_BITS_CASE,
|
|
.msb_set = false,
|
|
.lower_bound = false,
|
|
.extra_bits = 0,
|
|
.expected = {.tv_sec = 0x7fffffffLL, .tv_nsec = 0L},
|
|
},
|
|
|
|
{
|
|
.test_case_name = LOWER_BOUND_NEG_LO_1_CASE,
|
|
.msb_set = true,
|
|
.lower_bound = true,
|
|
.extra_bits = 1,
|
|
.expected = {.tv_sec = 0x80000000LL, .tv_nsec = 0L},
|
|
},
|
|
|
|
{
|
|
.test_case_name = UPPER_BOUND_NEG_LO_1_CASE,
|
|
.msb_set = true,
|
|
.lower_bound = false,
|
|
.extra_bits = 1,
|
|
.expected = {.tv_sec = 0xffffffffLL, .tv_nsec = 0L},
|
|
},
|
|
|
|
{
|
|
.test_case_name = LOWER_BOUND_NONNEG_LO_1_CASE,
|
|
.msb_set = false,
|
|
.lower_bound = true,
|
|
.extra_bits = 1,
|
|
.expected = {.tv_sec = 0x100000000LL, .tv_nsec = 0L},
|
|
},
|
|
|
|
{
|
|
.test_case_name = UPPER_BOUND_NONNEG_LO_1_CASE,
|
|
.msb_set = false,
|
|
.lower_bound = false,
|
|
.extra_bits = 1,
|
|
.expected = {.tv_sec = 0x17fffffffLL, .tv_nsec = 0L},
|
|
},
|
|
|
|
{
|
|
.test_case_name = LOWER_BOUND_NEG_HI_1_CASE,
|
|
.msb_set = true,
|
|
.lower_bound = true,
|
|
.extra_bits = 2,
|
|
.expected = {.tv_sec = 0x180000000LL, .tv_nsec = 0L},
|
|
},
|
|
|
|
{
|
|
.test_case_name = UPPER_BOUND_NEG_HI_1_CASE,
|
|
.msb_set = true,
|
|
.lower_bound = false,
|
|
.extra_bits = 2,
|
|
.expected = {.tv_sec = 0x1ffffffffLL, .tv_nsec = 0L},
|
|
},
|
|
|
|
{
|
|
.test_case_name = LOWER_BOUND_NONNEG_HI_1_CASE,
|
|
.msb_set = false,
|
|
.lower_bound = true,
|
|
.extra_bits = 2,
|
|
.expected = {.tv_sec = 0x200000000LL, .tv_nsec = 0L},
|
|
},
|
|
|
|
{
|
|
.test_case_name = UPPER_BOUND_NONNEG_HI_1_CASE,
|
|
.msb_set = false,
|
|
.lower_bound = false,
|
|
.extra_bits = 2,
|
|
.expected = {.tv_sec = 0x27fffffffLL, .tv_nsec = 0L},
|
|
},
|
|
|
|
{
|
|
.test_case_name = UPPER_BOUND_NONNEG_HI_1_NS_1_CASE,
|
|
.msb_set = false,
|
|
.lower_bound = false,
|
|
.extra_bits = 6,
|
|
.expected = {.tv_sec = 0x27fffffffLL, .tv_nsec = 1L},
|
|
},
|
|
|
|
{
|
|
.test_case_name = LOWER_BOUND_NONNEG_HI_1_NS_MAX_CASE,
|
|
.msb_set = false,
|
|
.lower_bound = true,
|
|
.extra_bits = 0xFFFFFFFF,
|
|
.expected = {.tv_sec = 0x300000000LL,
|
|
.tv_nsec = MAX_NANOSECONDS},
|
|
},
|
|
|
|
{
|
|
.test_case_name = LOWER_BOUND_NONNEG_EXTRA_BITS_1_CASE,
|
|
.msb_set = false,
|
|
.lower_bound = true,
|
|
.extra_bits = 3,
|
|
.expected = {.tv_sec = 0x300000000LL, .tv_nsec = 0L},
|
|
},
|
|
|
|
{
|
|
.test_case_name = UPPER_BOUND_NONNEG_EXTRA_BITS_1_CASE,
|
|
.msb_set = false,
|
|
.lower_bound = false,
|
|
.extra_bits = 3,
|
|
.expected = {.tv_sec = 0x37fffffffLL, .tv_nsec = 0L},
|
|
}
|
|
};
|
|
|
|
struct timespec64 timestamp;
|
|
int i;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(test_data); ++i) {
|
|
timestamp.tv_sec = get_32bit_time(&test_data[i]);
|
|
ext4_decode_extra_time(×tamp,
|
|
cpu_to_le32(test_data[i].extra_bits));
|
|
|
|
KUNIT_EXPECT_EQ_MSG(test,
|
|
test_data[i].expected.tv_sec,
|
|
timestamp.tv_sec,
|
|
CASE_NAME_FORMAT,
|
|
test_data[i].test_case_name,
|
|
test_data[i].msb_set,
|
|
test_data[i].lower_bound,
|
|
test_data[i].extra_bits);
|
|
KUNIT_EXPECT_EQ_MSG(test,
|
|
test_data[i].expected.tv_nsec,
|
|
timestamp.tv_nsec,
|
|
CASE_NAME_FORMAT,
|
|
test_data[i].test_case_name,
|
|
test_data[i].msb_set,
|
|
test_data[i].lower_bound,
|
|
test_data[i].extra_bits);
|
|
}
|
|
}
|
|
|
|
static struct kunit_case ext4_inode_test_cases[] = {
|
|
KUNIT_CASE(inode_test_xtimestamp_decoding),
|
|
{}
|
|
};
|
|
|
|
static struct kunit_suite ext4_inode_test_suite = {
|
|
.name = "ext4_inode_test",
|
|
.test_cases = ext4_inode_test_cases,
|
|
};
|
|
|
|
kunit_test_suite(ext4_inode_test_suite);
|