summaryrefslogtreecommitdiff
path: root/Tests/UserspaceEmulator
diff options
context:
space:
mode:
authorBrian Gianforcaro <bgianf@serenityos.org>2021-05-06 01:14:50 -0700
committerAndreas Kling <kling@serenityos.org>2021-05-06 17:54:28 +0200
commitfd0dbd1ebfbcbc29d46393061daa49dc7390caa7 (patch)
tree278ea94a46900e47ff7dae46b1017cd31095971a /Tests/UserspaceEmulator
parent6e641fadfa61d4b890db52fb60cc3709352336b6 (diff)
downloadserenity-fd0dbd1ebfbcbc29d46393061daa49dc7390caa7.zip
Tests: Establish root Tests directory, move Userland/Tests there
With the goal of centralizing all tests in the system, this is a first step to establish a Tests sub-tree. It will contain all of the unit tests and test harnesses for the various components in the system.
Diffstat (limited to 'Tests/UserspaceEmulator')
-rw-r--r--Tests/UserspaceEmulator/CMakeLists.txt9
-rw-r--r--Tests/UserspaceEmulator/write-oob.cpp80
2 files changed, 89 insertions, 0 deletions
diff --git a/Tests/UserspaceEmulator/CMakeLists.txt b/Tests/UserspaceEmulator/CMakeLists.txt
new file mode 100644
index 0000000000..3fbbbc4120
--- /dev/null
+++ b/Tests/UserspaceEmulator/CMakeLists.txt
@@ -0,0 +1,9 @@
+file(GLOB CMD_SOURCES CONFIGURE_DEPENDS "*.cpp")
+
+# FIXME: These tests do not use LibTest
+foreach(CMD_SRC ${CMD_SOURCES})
+ get_filename_component(CMD_NAME ${CMD_SRC} NAME_WE)
+ add_executable(${CMD_NAME} ${CMD_SRC})
+ target_link_libraries(${CMD_NAME} LibCore)
+ install(TARGETS ${CMD_NAME} RUNTIME DESTINATION usr/Tests/UserEmulator)
+endforeach()
diff --git a/Tests/UserspaceEmulator/write-oob.cpp b/Tests/UserspaceEmulator/write-oob.cpp
new file mode 100644
index 0000000000..77620a7105
--- /dev/null
+++ b/Tests/UserspaceEmulator/write-oob.cpp
@@ -0,0 +1,80 @@
+/*
+ * Copyright (c) 2021, Ben Wiederhake <BenWiederhake.GitHub@gmx.de>
+ *
+ * SPDX-License-Identifier: BSD-2-Clause
+ */
+
+#include <AK/Assertions.h>
+#include <LibCore/ArgsParser.h>
+#include <mman.h>
+#include <stdint.h>
+#include <stdio.h>
+#include <stdlib.h>
+
+static void write8(void* ptr) { *(volatile uint8_t*)ptr = 1; }
+static void write16(void* ptr) { *(volatile uint16_t*)ptr = 1; }
+static void write32(void* ptr) { *(volatile uint32_t*)ptr = 1; }
+static void write64(void* ptr) { *(volatile double*)ptr = 1.0; }
+// A u64 write might be translated by the compiler as a 32-then-32-bit write:
+// static void write64_bad(void* ptr) { *(volatile uint64_t*)ptr = 1.0; }
+// Let's hope this won't be translated like that.
+// Godbolt says yes: https://godbolt.org/z/1b9WGo
+
+static void run_test(void* region, ssize_t offset, size_t bits)
+{
+ void* ptr = (char*)region + offset;
+ printf("Writing to %p\n", ptr);
+ switch (bits) {
+ case 8:
+ write8(ptr);
+ break;
+ case 16:
+ write16(ptr);
+ break;
+ case 32:
+ write32(ptr);
+ break;
+ case 64:
+ write64(ptr);
+ break;
+ default:
+ VERIFY_NOT_REACHED();
+ }
+}
+
+int main(int argc, char** argv)
+{
+ bool do_static = false;
+ int size = 10 * PAGE_SIZE;
+ int offset = 10 * PAGE_SIZE - 1;
+ int bits = 16;
+
+ auto args_parser = Core::ArgsParser();
+ args_parser.set_general_help(
+ "Access out of bounds memory; a great testcase for UserEmulator.");
+ args_parser.add_option(do_static, "Use a static region instead of an mmap'ed region. Fixes 'size' to 10*PAGESIZE = 40960. (Default: false)", "static", 'S');
+ args_parser.add_option(size, "The size of the region to allocate. (Default: 10*PAGESIZE = 40960)", "size", 's', "size");
+ args_parser.add_option(offset, "The signed offset at which to start writing. (Default: 10*PAGESIZE-1 = 40959)", "offset", 'o', "offset");
+ args_parser.add_option(bits, "Amount of bits to write in a single instruction. (Default: 16)", "bits", 'b', "bits");
+ args_parser.parse(argc, argv);
+
+ if (do_static)
+ size = 10 * PAGE_SIZE;
+
+ printf("Writing %d bits to %s region of size %d at offset %d.\n",
+ bits, do_static ? "static" : "MMAP", size, offset);
+
+ if (do_static) {
+ // Let's just hope the linker puts nothing after it!
+ static unsigned char region[PAGE_SIZE * 10] = { 0 };
+
+ run_test(region, offset, 64);
+ } else {
+ void* region = mmap(nullptr, size, PROT_READ | PROT_WRITE, MAP_ANONYMOUS | MAP_PRIVATE, 0, 0);
+ VERIFY(region);
+ run_test(region, offset, bits);
+ }
+
+ printf("FAIL (should have caused SIGSEGV)\n");
+ return 1;
+}