KVM: selftests: Add atoi_paranoid() to catch errors missed by atoi()
authorVipin Sharma <vipinsh@google.com>
Thu, 3 Nov 2022 19:17:15 +0000 (12:17 -0700)
committerSean Christopherson <seanjc@google.com>
Wed, 16 Nov 2022 18:03:24 +0000 (10:03 -0800)
atoi() doesn't detect errors. There is no way to know that a 0 return
is correct conversion or due to an error.

Introduce atoi_paranoid() to detect errors and provide correct
conversion. Replace all atoi() calls with atoi_paranoid().

Signed-off-by: Vipin Sharma <vipinsh@google.com>
Suggested-by: David Matlack <dmatlack@google.com>
Suggested-by: Sean Christopherson <seanjc@google.com>
Reviewed-by: Sean Christopherson <seanjc@google.com>
Link: https://lore.kernel.org/r/20221103191719.1559407-4-vipinsh@google.com
Signed-off-by: Sean Christopherson <seanjc@google.com>
14 files changed:
tools/testing/selftests/kvm/aarch64/arch_timer.c
tools/testing/selftests/kvm/aarch64/debug-exceptions.c
tools/testing/selftests/kvm/aarch64/vgic_irq.c
tools/testing/selftests/kvm/access_tracking_perf_test.c
tools/testing/selftests/kvm/demand_paging_test.c
tools/testing/selftests/kvm/dirty_log_perf_test.c
tools/testing/selftests/kvm/include/test_util.h
tools/testing/selftests/kvm/kvm_page_table_test.c
tools/testing/selftests/kvm/lib/test_util.c
tools/testing/selftests/kvm/max_guest_memory_test.c
tools/testing/selftests/kvm/memslot_modification_stress_test.c
tools/testing/selftests/kvm/memslot_perf_test.c
tools/testing/selftests/kvm/set_memory_region_test.c
tools/testing/selftests/kvm/x86_64/nx_huge_pages_test.c

index 574eb73f0e904e80b23229d7fe3cab3a1ce4d7c6..251e7ff0488317f55239eb1c263570697d1f8c0a 100644 (file)
@@ -414,7 +414,7 @@ static bool parse_args(int argc, char *argv[])
        while ((opt = getopt(argc, argv, "hn:i:p:m:")) != -1) {
                switch (opt) {
                case 'n':
-                       test_args.nr_vcpus = atoi(optarg);
+                       test_args.nr_vcpus = atoi_paranoid(optarg);
                        if (test_args.nr_vcpus <= 0) {
                                pr_info("Positive value needed for -n\n");
                                goto err;
@@ -425,21 +425,21 @@ static bool parse_args(int argc, char *argv[])
                        }
                        break;
                case 'i':
-                       test_args.nr_iter = atoi(optarg);
+                       test_args.nr_iter = atoi_paranoid(optarg);
                        if (test_args.nr_iter <= 0) {
                                pr_info("Positive value needed for -i\n");
                                goto err;
                        }
                        break;
                case 'p':
-                       test_args.timer_period_ms = atoi(optarg);
+                       test_args.timer_period_ms = atoi_paranoid(optarg);
                        if (test_args.timer_period_ms <= 0) {
                                pr_info("Positive value needed for -p\n");
                                goto err;
                        }
                        break;
                case 'm':
-                       test_args.migration_freq_ms = atoi(optarg);
+                       test_args.migration_freq_ms = atoi_paranoid(optarg);
                        if (test_args.migration_freq_ms < 0) {
                                pr_info("0 or positive value needed for -m\n");
                                goto err;
index 947bd201435ce27adedbd8216b54524ba82dac11..19fffdf19c9f9455a00168cfbfaeece33f1130ff 100644 (file)
@@ -423,7 +423,7 @@ int main(int argc, char *argv[])
        while ((opt = getopt(argc, argv, "i:")) != -1) {
                switch (opt) {
                case 'i':
-                       ss_iteration = atoi(optarg);
+                       ss_iteration = atoi_paranoid(optarg);
                        break;
                case 'h':
                default:
index 17417220a083f65a889d934b013c6ca557134948..ae90b718070aab1cbb03d5814eed74b37eb88630 100644 (file)
@@ -824,16 +824,16 @@ int main(int argc, char **argv)
        while ((opt = getopt(argc, argv, "hn:e:l:")) != -1) {
                switch (opt) {
                case 'n':
-                       nr_irqs = atoi(optarg);
+                       nr_irqs = atoi_paranoid(optarg);
                        if (nr_irqs > 1024 || nr_irqs % 32)
                                help(argv[0]);
                        break;
                case 'e':
-                       eoi_split = (bool)atoi(optarg);
+                       eoi_split = (bool)atoi_paranoid(optarg);
                        default_args = false;
                        break;
                case 'l':
-                       level_sensitive = (bool)atoi(optarg);
+                       level_sensitive = (bool)atoi_paranoid(optarg);
                        default_args = false;
                        break;
                case 'h':
index 76c583a07ea2211e9a2e66bd08c986a040d6e7cf..c6bcc5301e2c2becc68115c3c47944b65d6646b0 100644 (file)
@@ -368,7 +368,7 @@ int main(int argc, char *argv[])
                        params.vcpu_memory_bytes = parse_size(optarg);
                        break;
                case 'v':
-                       params.nr_vcpus = atoi(optarg);
+                       params.nr_vcpus = atoi_paranoid(optarg);
                        break;
                case 'o':
                        overlap_memory_access = true;
index 779ae54f89c40f878dccd52b846bfe88d3235250..82597fb04146f49bcdc88fe65138093d455d66f9 100644 (file)
@@ -427,7 +427,7 @@ int main(int argc, char *argv[])
                        p.src_type = parse_backing_src_type(optarg);
                        break;
                case 'v':
-                       nr_vcpus = atoi(optarg);
+                       nr_vcpus = atoi_paranoid(optarg);
                        TEST_ASSERT(nr_vcpus > 0 && nr_vcpus <= max_vcpus,
                                    "Invalid number of vcpus, must be between 1 and %d", max_vcpus);
                        break;
index 5bb6954b2358769c98a8db536526fe68d75f2c26..ecda802b78ffae57f1c2ad2146e4c2bec1061eb1 100644 (file)
@@ -416,7 +416,7 @@ int main(int argc, char *argv[])
                        run_vcpus_while_disabling_dirty_logging = true;
                        break;
                case 'f':
-                       p.wr_fract = atoi(optarg);
+                       p.wr_fract = atoi_paranoid(optarg);
                        TEST_ASSERT(p.wr_fract >= 1,
                                    "Write fraction cannot be less than one");
                        break;
@@ -427,7 +427,7 @@ int main(int argc, char *argv[])
                        help(argv[0]);
                        break;
                case 'i':
-                       p.iterations = atoi(optarg);
+                       p.iterations = atoi_paranoid(optarg);
                        break;
                case 'm':
                        guest_modes_cmdline(optarg);
@@ -445,12 +445,12 @@ int main(int argc, char *argv[])
                        p.backing_src = parse_backing_src_type(optarg);
                        break;
                case 'v':
-                       nr_vcpus = atoi(optarg);
+                       nr_vcpus = atoi_paranoid(optarg);
                        TEST_ASSERT(nr_vcpus > 0 && nr_vcpus <= max_vcpus,
                                    "Invalid number of vcpus, must be between 1 and %d", max_vcpus);
                        break;
                case 'x':
-                       p.slots = atoi(optarg);
+                       p.slots = atoi_paranoid(optarg);
                        break;
                default:
                        help(argv[0]);
index befc754ce9b3b712d5dde489bc7bf358f6809e64..feae428637599df49bbca2aa3c9db96f8ac00ec1 100644 (file)
@@ -152,4 +152,6 @@ static inline void *align_ptr_up(void *x, size_t size)
        return (void *)align_up((unsigned long)x, size);
 }
 
+int atoi_paranoid(const char *num_str);
+
 #endif /* SELFTEST_KVM_TEST_UTIL_H */
index f42c6ac6d71dbfa81809185692095317b84dce80..ea7feb69bb88aafd03382a71ba6e92bda1bd9349 100644 (file)
@@ -461,7 +461,7 @@ int main(int argc, char *argv[])
                        p.test_mem_size = parse_size(optarg);
                        break;
                case 'v':
-                       nr_vcpus = atoi(optarg);
+                       nr_vcpus = atoi_paranoid(optarg);
                        TEST_ASSERT(nr_vcpus > 0 && nr_vcpus <= max_vcpus,
                                    "Invalid number of vcpus, must be between 1 and %d", max_vcpus);
                        break;
index 6d23878bbfe1a644e8a1c7a2f30d05190edd5f05..c2d9c68277793cb2197457b3d6b2379a5101435b 100644 (file)
@@ -334,3 +334,22 @@ long get_run_delay(void)
 
        return val[1];
 }
+
+int atoi_paranoid(const char *num_str)
+{
+       char *end_ptr;
+       long num;
+
+       errno = 0;
+       num = strtol(num_str, &end_ptr, 0);
+       TEST_ASSERT(!errno, "strtol(\"%s\") failed", num_str);
+       TEST_ASSERT(num_str != end_ptr,
+                   "strtol(\"%s\") didn't find a valid integer.", num_str);
+       TEST_ASSERT(*end_ptr == '\0',
+                   "strtol(\"%s\") failed to parse trailing characters \"%s\".",
+                   num_str, end_ptr);
+       TEST_ASSERT(num >= INT_MIN && num <= INT_MAX,
+                   "%ld not in range of [%d, %d]", num, INT_MIN, INT_MAX);
+
+       return num;
+}
index 9a6e4f3ad6b57ba2fa867d6975025ae5cbd8d8e9..1595b73dc09a8db59b9b8d97b26e6fc307a10a33 100644 (file)
@@ -193,15 +193,15 @@ int main(int argc, char *argv[])
        while ((opt = getopt(argc, argv, "c:h:m:s:H")) != -1) {
                switch (opt) {
                case 'c':
-                       nr_vcpus = atoi(optarg);
+                       nr_vcpus = atoi_paranoid(optarg);
                        TEST_ASSERT(nr_vcpus > 0, "number of vcpus must be >0");
                        break;
                case 'm':
-                       max_mem = atoi(optarg) * size_1gb;
+                       max_mem = atoi_paranoid(optarg) * size_1gb;
                        TEST_ASSERT(max_mem > 0, "memory size must be >0");
                        break;
                case 's':
-                       slot_size = atoi(optarg) * size_1gb;
+                       slot_size = atoi_paranoid(optarg) * size_1gb;
                        TEST_ASSERT(slot_size > 0, "slot size must be >0");
                        break;
                case 'H':
index bb1d17a1171bc1e198f4a392b94d090b9be232d8..7d19a27d80d2d78fd97ca525f6582cb896ec85c7 100644 (file)
@@ -158,7 +158,7 @@ int main(int argc, char *argv[])
                        guest_modes_cmdline(optarg);
                        break;
                case 'd':
-                       p.memslot_modification_delay = strtoul(optarg, NULL, 0);
+                       p.memslot_modification_delay = atoi_paranoid(optarg);
                        TEST_ASSERT(p.memslot_modification_delay >= 0,
                                    "A negative delay is not supported.");
                        break;
@@ -166,7 +166,7 @@ int main(int argc, char *argv[])
                        guest_percpu_mem_size = parse_size(optarg);
                        break;
                case 'v':
-                       nr_vcpus = atoi(optarg);
+                       nr_vcpus = atoi_paranoid(optarg);
                        TEST_ASSERT(nr_vcpus > 0 && nr_vcpus <= max_vcpus,
                                    "Invalid number of vcpus, must be between 1 and %d",
                                    max_vcpus);
@@ -175,7 +175,7 @@ int main(int argc, char *argv[])
                        p.partition_vcpu_memory_access = false;
                        break;
                case 'i':
-                       p.nr_memslot_modifications = atoi(optarg);
+                       p.nr_memslot_modifications = atoi_paranoid(optarg);
                        break;
                case 'h':
                default:
index 44995446d942a7e91b2da8d5015e2a061dca9e11..4bae9e3f5ca1e123b9cb7e135ef046f52f8f531c 100644 (file)
@@ -885,21 +885,21 @@ static bool parse_args(int argc, char *argv[],
                        map_unmap_verify = true;
                        break;
                case 's':
-                       targs->nslots = atoi(optarg);
+                       targs->nslots = atoi_paranoid(optarg);
                        if (targs->nslots <= 0 && targs->nslots != -1) {
                                pr_info("Slot count cap has to be positive or -1 for no cap\n");
                                return false;
                        }
                        break;
                case 'f':
-                       targs->tfirst = atoi(optarg);
+                       targs->tfirst = atoi_paranoid(optarg);
                        if (targs->tfirst < 0) {
                                pr_info("First test to run has to be non-negative\n");
                                return false;
                        }
                        break;
                case 'e':
-                       targs->tlast = atoi(optarg);
+                       targs->tlast = atoi_paranoid(optarg);
                        if (targs->tlast < 0 || targs->tlast >= NTESTS) {
                                pr_info("Last test to run has to be non-negative and less than %zu\n",
                                        NTESTS);
@@ -907,14 +907,14 @@ static bool parse_args(int argc, char *argv[],
                        }
                        break;
                case 'l':
-                       targs->seconds = atoi(optarg);
+                       targs->seconds = atoi_paranoid(optarg);
                        if (targs->seconds < 0) {
                                pr_info("Test length in seconds has to be non-negative\n");
                                return false;
                        }
                        break;
                case 'r':
-                       targs->runs = atoi(optarg);
+                       targs->runs = atoi_paranoid(optarg);
                        if (targs->runs <= 0) {
                                pr_info("Runs per test has to be positive\n");
                                return false;
index 0d55f508d595381d7e97ce4ca24744dc2391bfe1..c366949c8362c8588f2f0481a4968a78e30580ff 100644 (file)
@@ -407,7 +407,7 @@ int main(int argc, char *argv[])
 
 #ifdef __x86_64__
        if (argc > 1)
-               loops = atoi(argv[1]);
+               loops = atoi_paranoid(argv[1]);
        else
                loops = 10;
 
index 59ffe7fd354f8919c1ff7f4fe2e9bb9a95fc6ed8..354b6902849c5d64302b91049cd7c257a36122db 100644 (file)
@@ -241,10 +241,10 @@ int main(int argc, char **argv)
        while ((opt = getopt(argc, argv, "hp:t:r")) != -1) {
                switch (opt) {
                case 'p':
-                       reclaim_period_ms = atoi(optarg);
+                       reclaim_period_ms = atoi_paranoid(optarg);
                        break;
                case 't':
-                       token = atoi(optarg);
+                       token = atoi_paranoid(optarg);
                        break;
                case 'r':
                        reboot_permissions = true;