123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488 |
- #!/usr/bin/env python3
- import os
- import unittest
- import subprocess
- from regression import (
- HAS_SGX,
- RegressionTestCase,
- )
- class TC_00_Unittests(RegressionTestCase):
- def test_000_spinlock(self):
- stdout, _ = self.run_binary(['spinlock'])
- self.assertIn('Test successful!', stdout)
- class TC_01_Bootstrap(RegressionTestCase):
- def test_100_basic_bootstrapping(self):
- stdout, _ = self.run_binary(['bootstrap'])
- # Basic Bootstrapping
- self.assertIn('User Program Started', stdout)
- # One Argument Given
- self.assertIn('# of Arguments: 1', stdout)
- self.assertIn('argv[0] = bootstrap', stdout)
- def test_101_basic_bootstrapping_five_arguments(self):
- # Five Arguments Given
- stdout, _ = self.run_binary(['bootstrap', 'a', 'b', 'c', 'd'])
- self.assertIn('# of Arguments: 5', stdout)
- self.assertIn('argv[0] = bootstrap', stdout)
- self.assertIn('argv[1] = a', stdout)
- self.assertIn('argv[2] = b', stdout)
- self.assertIn('argv[3] = c', stdout)
- self.assertIn('argv[4] = d', stdout)
- @unittest.skipUnless(HAS_SGX,
- 'This test is only meaningful on SGX PAL because only SGX catches raw '
- 'syscalls and redirects to Graphene\'s LibOS. If we will add seccomp to '
- 'Linux PAL, then we should allow this test on Linux PAL as well.')
- def test_102_basic_bootstrapping_static(self):
- # bootstrap_static
- stdout, _ = self.run_binary(['bootstrap_static'])
- self.assertIn('Hello world (bootstrap_static)!', stdout)
- def test_103_basic_bootstrapping_pie(self):
- # bootstrap_pie
- stdout, _ = self.run_binary(['bootstrap_pie'])
- self.assertIn('User program started', stdout)
- self.assertIn('Local Address in Executable: 0x', stdout)
- self.assertIn('argv[0] = bootstrap_pie', stdout)
- def test_110_basic_bootstrapping_cxx(self):
- stdout, _ = self.run_binary(['bootstrap-c++'])
- # Basic Bootstrapping (C++)
- self.assertIn('User Program Started', stdout)
- def test_200_exec(self):
- stdout, _ = self.run_binary(['exec'])
- # 2 page child binary
- self.assertIn(
- '00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 '
- '000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ',
- stdout)
- def test_201_exec_same(self):
- stdout, _ = self.run_binary(['exec_same'])
- self.assertIn('hello from execv process', stdout)
- def test_202_fork_and_exec(self):
- stdout, _ = self.run_binary(['fork_and_exec'])
- # fork and exec 2 page child binary
- self.assertIn('child exited with status: 0', stdout)
- self.assertIn('test completed successfully', stdout)
- def test_203_vfork_and_exec(self):
- stdout, _ = self.run_binary(['vfork_and_exec'])
- # vfork and exec 2 page child binary
- self.assertIn('child exited with status: 0', stdout)
- self.assertIn('test completed successfully', stdout)
- def test_210_exec_invalid_args(self):
- stdout, _ = self.run_binary(['exec_invalid_args'])
- # Execve with invalid pointers in arguments
- self.assertIn(
- 'execve(invalid-path) correctly returned error', stdout)
- self.assertIn(
- 'execve(invalid-argv-ptr) correctly returned error', stdout)
- self.assertIn(
- 'execve(invalid-envp-ptr) correctly returned error', stdout)
- self.assertIn(
- 'execve(invalid-argv) correctly returned error', stdout)
- self.assertIn(
- 'execve(invalid-envp) correctly returned error', stdout)
- def test_300_shared_object(self):
- stdout, _ = self.run_binary(['shared_object'])
- # Shared Object
- self.assertIn('Hello world', stdout)
- def test_400_exit(self):
- with self.expect_returncode(113):
- self.run_binary(['exit'])
- def test_401_exit_group(self):
- try:
- self.run_binary(['exit_group'])
- except subprocess.CalledProcessError as e:
- self.assertTrue(1 <= e.returncode and e.returncode <= 4)
- def test_402_signalexit(self):
- with self.expect_returncode(134):
- self.run_binary(['abort'])
- def test_403_signalexit_multithread(self):
- with self.expect_returncode(134):
- self.run_binary(['abort_multithread'])
- def test_404_sigprocmask(self):
- with self.expect_returncode(113):
- self.run_binary(['sigprocmask'])
- def test_500_init_fail(self):
- try:
- self.run_binary(['init_fail'])
- self.fail('expected to return nonzero (and != 42)')
- except subprocess.CalledProcessError as e:
- self.assertNotEqual(e.returncode, 42, 'expected returncode != 42')
- def test_600_multi_pthread(self):
- stdout, _ = self.run_binary(['multi_pthread'])
- # Multiple thread creation
- self.assertIn('128 Threads Created', stdout)
- @unittest.skipUnless(HAS_SGX,
- 'This test is only meaningful on SGX PAL because only SGX catches raw '
- 'syscalls and redirects to Graphene\'s LibOS. If we will add seccomp to '
- 'Linux PAL, then we should allow this test on Linux PAL as well.')
- class TC_02_OpenMP(RegressionTestCase):
- def test_000_simple_for_loop(self):
- stdout, _ = self.run_binary(['openmp'])
- # OpenMP simple for loop
- self.assertIn('first: 0, last: 9', stdout)
- @unittest.skipUnless(HAS_SGX,
- 'This test is only meaningful on SGX PAL because file-check-policy is '
- 'only relevant to SGX.')
- class TC_03_FileCheckPolicy(RegressionTestCase):
- def test_000_strict_success(self):
- manifest = self.get_manifest('file_check_policy_strict')
- stdout, _ = self.run_binary([manifest, 'trusted_testfile'])
- self.assertIn('file_check_policy succeeded', stdout)
- def test_001_strict_fail(self):
- manifest = self.get_manifest('file_check_policy_strict')
- with self.expect_returncode(2):
- self.run_binary([manifest, 'unknown_testfile'])
- def test_002_allow_all_but_log_success(self):
- manifest = self.get_manifest('file_check_policy_allow_all_but_log')
- stdout, stderr = self.run_binary([manifest, 'unknown_testfile'])
- self.assertIn('Allowing access to an unknown file due to file_check_policy settings: file:unknown_testfile', stderr)
- self.assertIn('file_check_policy succeeded', stdout)
- def test_003_allow_all_but_log_fail(self):
- manifest = self.get_manifest('file_check_policy_allow_all_but_log')
- stdout, stderr = self.run_binary([manifest, 'trusted_testfile'])
- self.assertNotIn('Allowing access to an unknown file due to file_check_policy settings: file:trusted_testfile', stderr)
- self.assertIn('file_check_policy succeeded', stdout)
- class TC_30_Syscall(RegressionTestCase):
- def test_000_getcwd(self):
- stdout, _ = self.run_binary(['getcwd'])
- # Getcwd syscall
- self.assertIn('[bss_cwd_buf] getcwd succeeded: /', stdout)
- self.assertIn('[mmapped_cwd_buf] getcwd succeeded: /', stdout)
- def test_010_stat_invalid_args(self):
- stdout, _ = self.run_binary(['stat_invalid_args'])
- # Stat with invalid arguments
- self.assertIn('stat(invalid-path-ptr) correctly returned error', stdout)
- self.assertIn('stat(invalid-buf-ptr) correctly returned error', stdout)
- self.assertIn('lstat(invalid-path-ptr) correctly returned error', stdout)
- self.assertIn('lstat(invalid-buf-ptr) correctly returned error', stdout)
- def test_011_fstat_cwd(self):
- stdout, _ = self.run_binary(['fstat_cwd'])
- # fstat on a directory
- self.assertIn('fstat returned the fd type as S_IFDIR', stdout)
- def test_020_getdents(self):
- # This doesn't catch extraneous entries, but should be fine
- # until the LTP test can be run (need symlink support)
- stdout, _ = self.run_binary(['getdents'])
- self.assertIn('getdents: setup ok', stdout)
- # Directory listing (32-bit)
- self.assertIn('getdents32: . [0x4]', stdout)
- self.assertIn('getdents32: .. [0x4]', stdout)
- self.assertIn('getdents32: file1 [0x8]', stdout)
- self.assertIn('getdents32: file2 [0x8]', stdout)
- self.assertIn('getdents32: dir3 [0x4]', stdout)
- # Directory listing (64-bit)
- self.assertIn('getdents64: . [0x4]', stdout)
- self.assertIn('getdents64: .. [0x4]', stdout)
- self.assertIn('getdents64: file1 [0x8]', stdout)
- self.assertIn('getdents64: file2 [0x8]', stdout)
- self.assertIn('getdents64: dir3 [0x4]', stdout)
- def test_021_getdents_large_dir(self):
- stdout, _ = self.run_binary(['large_dir_read', 'tmp/large_dir', '3000'])
- self.assertIn('Success!', stdout)
- def test_022_host_root_fs(self):
- stdout, _ = self.run_binary(['host_root_fs'])
- self.assertIn('Test was successful', stdout)
- def test_030_fopen(self):
- if os.path.exists("tmp/filecreatedbygraphene"):
- os.remove("tmp/filecreatedbygraphene")
- stdout, _ = self.run_binary(['fopen_cornercases'])
- # fopen corner cases
- self.assertIn('Successfully read from file: Hello World', stdout)
- def test_031_readdir(self):
- stdout, _ = self.run_binary(['readdir'])
- self.assertIn('test completed successfully', stdout)
- def test_032_file_size(self):
- stdout, _ = self.run_binary(['file_size'])
- self.assertIn('test completed successfully', stdout)
- def test_040_futex_bitset(self):
- stdout, _ = self.run_binary(['futex_bitset'])
- # Futex Wake Test
- self.assertIn('Woke all kiddos', stdout)
- def test_041_futex_timeout(self):
- stdout, _ = self.run_binary(['futex_timeout'])
- # Futex Timeout Test
- self.assertIn('futex correctly timed out', stdout)
- def test_042_futex_requeue(self):
- stdout, _ = self.run_binary(['futex_requeue'])
- self.assertIn('Test successful!', stdout)
- def test_043_futex_wake_op(self):
- stdout, _ = self.run_binary(['futex_wake_op'])
- self.assertIn('Test successful!', stdout)
- def test_050_mmap(self):
- stdout, _ = self.run_binary(['mmap-file'], timeout=60)
- # Private mmap beyond file range
- self.assertIn('mmap test 6 passed', stdout)
- self.assertIn('mmap test 7 passed', stdout)
- # Private mmap beyond file range (after fork)
- self.assertIn('mmap test 1 passed', stdout)
- self.assertIn('mmap test 2 passed', stdout)
- self.assertIn('mmap test 3 passed', stdout)
- self.assertIn('mmap test 4 passed', stdout)
- @unittest.skipIf(HAS_SGX,
- 'On SGX, SIGBUS isn\'t always implemented correctly, for lack '
- 'of memory protection. For now, some of these cases won\'t work.')
- def test_051_mmap_sgx(self):
- stdout, _ = self.run_binary(['mmap-file'], timeout=60)
- # SIGBUS test
- self.assertIn('mmap test 5 passed', stdout)
- self.assertIn('mmap test 8 passed', stdout)
- def test_052_large_mmap(self):
- stdout, _ = self.run_binary(['large-mmap'], timeout=480)
- # Ftruncate
- self.assertIn('large-mmap: ftruncate OK', stdout)
- # Large mmap
- self.assertIn('large-mmap: mmap 1 completed OK', stdout)
- self.assertIn('large-mmap: mmap 2 completed OK', stdout)
- def test_053_mprotect_file_fork(self):
- stdout, _ = self.run_binary(['mprotect_file_fork'])
- self.assertIn('Test successful!', stdout)
- @unittest.skip('sigaltstack isn\'t correctly implemented')
- def test_060_sigaltstack(self):
- stdout, _ = self.run_binary(['sigaltstack'])
- # Sigaltstack Test
- self.assertIn('OK on sigaltstack in main thread before alarm', stdout)
- self.assertIn('&act == 0x', stdout)
- self.assertIn('sig 14 count 1 goes off with sp=0x', stdout)
- self.assertIn('OK on signal stack', stdout)
- self.assertIn('OK on sigaltstack in handler', stdout)
- self.assertIn('sig 14 count 2 goes off with sp=0x', stdout)
- self.assertIn('OK on signal stack', stdout)
- self.assertIn('OK on sigaltstack in handler', stdout)
- self.assertIn('sig 14 count 3 goes off with sp=0x', stdout)
- self.assertIn('OK on signal stack', stdout)
- self.assertIn('OK on sigaltstack in handler', stdout)
- self.assertIn('OK on sigaltstack in main thread', stdout)
- self.assertIn('done exiting', stdout)
- def test_070_eventfd(self):
- stdout, _ = self.run_binary(['eventfd'])
- # Eventfd Test
- self.assertIn('eventfd_using_poll completed successfully', stdout)
- self.assertIn('eventfd_using_various_flags completed successfully', stdout)
- self.assertIn('eventfd_using_fork completed successfully', stdout)
- def test_080_sched(self):
- stdout, _ = self.run_binary(['sched'])
- # Scheduling Syscalls Test
- self.assertIn('Test completed successfully', stdout)
- @unittest.skipUnless(HAS_SGX,
- 'This test is only meaningful on SGX PAL because only SGX catches raw '
- 'syscalls and redirects to Graphene\'s LibOS. If we will add seccomp to '
- 'Linux PAL, then we should allow this test on Linux PAL as well.')
- class TC_31_SyscallSGX(RegressionTestCase):
- def test_000_syscall_redirect(self):
- stdout, _ = self.run_binary(['syscall'])
- # Syscall Instruction Redirection
- self.assertIn('Hello world', stdout)
- class TC_40_FileSystem(RegressionTestCase):
- def test_000_base(self):
- stdout, _ = self.run_binary(['proc'])
- # Base /proc files present
- self.assertIn('/proc/1/..', stdout)
- self.assertIn('/proc/1/cwd', stdout)
- self.assertIn('/proc/1/exe', stdout)
- self.assertIn('/proc/1/root', stdout)
- self.assertIn('/proc/1/fd', stdout)
- self.assertIn('/proc/1/maps', stdout)
- self.assertIn('/proc/.', stdout)
- self.assertIn('/proc/1', stdout)
- self.assertIn('/proc/self', stdout)
- self.assertIn('/proc/meminfo', stdout)
- self.assertIn('/proc/cpuinfo', stdout)
- def test_010_path(self):
- stdout, _ = self.run_binary(['proc-path'])
- # Base /proc path present
- self.assertIn('proc path test success', stdout)
- def test_020_cpuinfo(self):
- stdout, _ = self.run_binary(['proc_cpuinfo'], timeout=50)
- # proc/cpuinfo Linux-based formatting
- self.assertIn('cpuinfo test passed', stdout)
- def test_030_fdleak(self):
- stdout, _ = self.run_binary(['fdleak'], timeout=10)
- self.assertIn("Test succeeded.", stdout)
- def test_040_str_close_leak(self):
- stdout, _ = self.run_binary(['str_close_leak'], timeout=60)
- self.assertIn("Success", stdout)
- class TC_80_Socket(RegressionTestCase):
- def test_000_getsockopt(self):
- stdout, _ = self.run_binary(['getsockopt'])
- self.assertIn('getsockopt: Got socket type OK', stdout)
- self.assertIn('getsockopt: Got TCP_NODELAY flag OK', stdout)
- def test_010_epoll_wait_timeout(self):
- stdout, _ = self.run_binary(['epoll_wait_timeout', '8000'],
- timeout=50)
- # epoll_wait timeout
- self.assertIn('epoll_wait test passed', stdout)
- def test_020_poll(self):
- stdout, _ = self.run_binary(['poll'])
- self.assertIn('poll(POLLOUT) returned 1 file descriptors', stdout)
- self.assertIn('poll(POLLIN) returned 1 file descriptors', stdout)
- def test_021_poll_many_types(self):
- stdout, _ = self.run_binary(['poll_many_types'])
- self.assertIn('poll(POLLIN) returned 3 file descriptors', stdout)
- def test_030_ppoll(self):
- stdout, _ = self.run_binary(['ppoll'])
- self.assertIn('ppoll(POLLOUT) returned 1 file descriptors', stdout)
- self.assertIn('ppoll(POLLIN) returned 1 file descriptors', stdout)
- def test_040_select(self):
- stdout, _ = self.run_binary(['select'])
- self.assertIn('select() on write event returned 1 file descriptors', stdout)
- self.assertIn('select() on read event returned 1 file descriptors', stdout)
- def test_050_pselect(self):
- stdout, _ = self.run_binary(['pselect'])
- self.assertIn('pselect() on write event returned 1 file descriptors', stdout)
- self.assertIn('pselect() on read event returned 1 file descriptors', stdout)
- def test_100_socket_unix(self):
- stdout, _ = self.run_binary(['unix'])
- self.assertIn('Data: This is packet 0', stdout)
- self.assertIn('Data: This is packet 1', stdout)
- self.assertIn('Data: This is packet 2', stdout)
- self.assertIn('Data: This is packet 3', stdout)
- self.assertIn('Data: This is packet 4', stdout)
- self.assertIn('Data: This is packet 5', stdout)
- self.assertIn('Data: This is packet 6', stdout)
- self.assertIn('Data: This is packet 7', stdout)
- self.assertIn('Data: This is packet 8', stdout)
- self.assertIn('Data: This is packet 9', stdout)
- def test_200_socket_udp(self):
- stdout, _ = self.run_binary(['udp'], timeout=50)
- self.assertIn('Data: This is packet 0', stdout)
- self.assertIn('Data: This is packet 1', stdout)
- self.assertIn('Data: This is packet 2', stdout)
- self.assertIn('Data: This is packet 3', stdout)
- self.assertIn('Data: This is packet 4', stdout)
- self.assertIn('Data: This is packet 5', stdout)
- self.assertIn('Data: This is packet 6', stdout)
- self.assertIn('Data: This is packet 7', stdout)
- self.assertIn('Data: This is packet 8', stdout)
- self.assertIn('Data: This is packet 9', stdout)
- def test_300_socket_tcp_msg_peek(self):
- stdout, _ = self.run_binary(['tcp_msg_peek'], timeout=50)
- self.assertIn('[client] receiving with MSG_PEEK: Hello from server!', stdout)
- self.assertIn('[client] receiving without MSG_PEEK: Hello from server!', stdout)
- self.assertIn('[client] checking how many bytes are left unread: 0', stdout)
- self.assertIn('[client] done', stdout)
- self.assertIn('[server] done', stdout)
- def test_310_socket_tcp_ipv6_v6only(self):
- stdout, _ = self.run_binary(['tcp_ipv6_v6only'], timeout=50)
- self.assertIn('test completed successfully', stdout)
- @unittest.skipUnless(HAS_SGX,
- 'This test is only meaningful on SGX PAL because only SGX emulates CPUID.')
- class TC_90_CpuidSGX(RegressionTestCase):
- def test_000_cpuid(self):
- stdout, _ = self.run_binary(['cpuid'])
- self.assertIn('CPUID test passed.', stdout)
|