summaryrefslogtreecommitdiff
path: root/Meta/HeaderCheck
diff options
context:
space:
mode:
Diffstat (limited to 'Meta/HeaderCheck')
-rw-r--r--Meta/HeaderCheck/.gitignore1
-rw-r--r--Meta/HeaderCheck/CMakeLists.txt4
-rwxr-xr-xMeta/HeaderCheck/generate_all.py70
3 files changed, 75 insertions, 0 deletions
diff --git a/Meta/HeaderCheck/.gitignore b/Meta/HeaderCheck/.gitignore
new file mode 100644
index 0000000000..a0f00082c8
--- /dev/null
+++ b/Meta/HeaderCheck/.gitignore
@@ -0,0 +1 @@
+CMakeLists.txt
diff --git a/Meta/HeaderCheck/CMakeLists.txt b/Meta/HeaderCheck/CMakeLists.txt
new file mode 100644
index 0000000000..86e0d165e1
--- /dev/null
+++ b/Meta/HeaderCheck/CMakeLists.txt
@@ -0,0 +1,4 @@
+execute_process(COMMAND "${CMAKE_CURRENT_SOURCE_DIR}/generate_all.py" "${SERENITY_ARCH}" OUTPUT_VARIABLE SOURCES_STRING)
+string(REPLACE "\n" ";" SOURCES_LIST ${SOURCES_STRING})
+
+add_library(HeaderCheck OBJECT ${SOURCES_LIST})
diff --git a/Meta/HeaderCheck/generate_all.py b/Meta/HeaderCheck/generate_all.py
new file mode 100755
index 0000000000..df7c9e9142
--- /dev/null
+++ b/Meta/HeaderCheck/generate_all.py
@@ -0,0 +1,70 @@
+#!/usr/bin/env python3
+
+import os
+import sys
+import subprocess
+
+TEST_FILE_TEMPLATE = '''\
+#include <{filename}>
+// Check idempotency:
+#include <{filename}>
+'''
+
+
+def get_headers_here():
+ result = subprocess.run(['git', 'ls-files', 'Userland/Libraries/*.h'], check=True, capture_output=True, text=True)
+ assert result.stderr == ''
+ output = result.stdout.split('\n')
+ assert output[-1] == '' # Trailing newline
+ assert len(output) > 500, 'There should be well over a thousand headers, not only {}?!'.format(len(output))
+ return output[:-1]
+
+
+def as_filename(header_path):
+ return header_path.replace('/', '__') + '__test.cpp'
+
+
+def verbosely_write(path, new_content):
+ print(path)
+ # FIXME: Ensure directory exists
+ if os.path.exists(path):
+ with open(path, 'r') as fp:
+ old_data = fp.read()
+ if old_data == new_content:
+ # Fast path! Don't trigger ninja
+ return
+ with open(path, 'w') as fp:
+ fp.write(new_content)
+
+
+def generate_part(header):
+ content = TEST_FILE_TEMPLATE.format(filename=header)
+ if header.startswith('Kernel/'):
+ content += '#define KERNEL\n'
+ verbosely_write(as_filename(header), content)
+
+
+def run(root_path, arch):
+ os.chdir(root_path)
+ headers_list = get_headers_here()
+
+ generated_files_path = os.path.join(root_path, 'Build', arch, 'Meta', 'HeaderCheck')
+ if not os.path.exists(generated_files_path):
+ os.mkdir(generated_files_path)
+ os.chdir(generated_files_path)
+ for header in headers_list:
+ generate_part(header)
+
+
+if __name__ == '__main__':
+ if 'SERENITY_SOURCE_DIR' not in os.environ:
+ print('Must set SERENITY_SOURCE_DIR first!', file=sys.stderr)
+ exit(1)
+ if len(sys.argv) == 2:
+ with open('/tmp/the_arg', 'w') as fp:
+ fp.write(sys.argv[1])
+ run(os.environ['SERENITY_SOURCE_DIR'], sys.argv[1])
+ else:
+ print('Usage: SERENITY_SOURCE_DIR=/path/to/serenity {} SERENITY_BUILD_ARCH'
+ .format(sys.argv[0]), file=sys.stderr)
+ exit(1)