1 // Copyright 2014 The Chromium Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 #include "sandbox/linux/seccomp-bpf/bpf_tests.h" 6 7 #include <errno.h> 8 #include <sys/ptrace.h> 9 #include <sys/syscall.h> 10 #include <sys/types.h> 11 #include <unistd.h> 12 13 #include "base/logging.h" 14 #include "base/macros.h" 15 #include "base/memory/scoped_ptr.h" 16 #include "build/build_config.h" 17 #include "sandbox/linux/bpf_dsl/bpf_dsl.h" 18 #include "sandbox/linux/bpf_dsl/policy.h" 19 #include "sandbox/linux/seccomp-bpf/sandbox_bpf.h" 20 #include "sandbox/linux/services/syscall_wrappers.h" 21 #include "sandbox/linux/system_headers/linux_syscalls.h" 22 #include "sandbox/linux/tests/unit_tests.h" 23 #include "testing/gtest/include/gtest/gtest.h" 24 25 using sandbox::bpf_dsl::Allow; 26 using sandbox::bpf_dsl::Error; 27 using sandbox::bpf_dsl::ResultExpr; 28 29 namespace sandbox { 30 31 namespace { 32 33 class FourtyTwo { 34 public: 35 static const int kMagicValue = 42; 36 FourtyTwo() : value_(kMagicValue) {} 37 int value() { return value_; } 38 39 private: 40 int value_; 41 DISALLOW_COPY_AND_ASSIGN(FourtyTwo); 42 }; 43 44 class EmptyClassTakingPolicy : public bpf_dsl::Policy { 45 public: 46 explicit EmptyClassTakingPolicy(FourtyTwo* fourty_two) { 47 BPF_ASSERT(fourty_two); 48 BPF_ASSERT(FourtyTwo::kMagicValue == fourty_two->value()); 49 } 50 ~EmptyClassTakingPolicy() override {} 51 52 ResultExpr EvaluateSyscall(int sysno) const override { 53 DCHECK(SandboxBPF::IsValidSyscallNumber(sysno)); 54 return Allow(); 55 } 56 }; 57 58 BPF_TEST(BPFTest, 59 BPFAUXPointsToClass, 60 EmptyClassTakingPolicy, 61 FourtyTwo /* *BPF_AUX */) { 62 // BPF_AUX should point to an instance of FourtyTwo. 63 BPF_ASSERT(BPF_AUX); 64 BPF_ASSERT(FourtyTwo::kMagicValue == BPF_AUX->value()); 65 } 66 67 void DummyTestFunction(FourtyTwo *fourty_two) { 68 } 69 70 TEST(BPFTest, BPFTesterCompatibilityDelegateLeakTest) { 71 // Don't do anything, simply gives dynamic tools an opportunity to detect 72 // leaks. 73 { 74 BPFTesterCompatibilityDelegate<EmptyClassTakingPolicy, FourtyTwo> 75 simple_delegate(DummyTestFunction); 76 } 77 { 78 // Test polymorphism. 79 scoped_ptr<BPFTesterDelegate> simple_delegate( 80 new BPFTesterCompatibilityDelegate<EmptyClassTakingPolicy, FourtyTwo>( 81 DummyTestFunction)); 82 } 83 } 84 85 class EnosysPtracePolicy : public bpf_dsl::Policy { 86 public: 87 EnosysPtracePolicy() { my_pid_ = sys_getpid(); } 88 ~EnosysPtracePolicy() override { 89 // Policies should be able to bind with the process on which they are 90 // created. They should never be created in a parent process. 91 BPF_ASSERT_EQ(my_pid_, sys_getpid()); 92 } 93 94 ResultExpr EvaluateSyscall(int system_call_number) const override { 95 CHECK(SandboxBPF::IsValidSyscallNumber(system_call_number)); 96 if (system_call_number == __NR_ptrace) { 97 // The EvaluateSyscall function should run in the process that created 98 // the current object. 99 BPF_ASSERT_EQ(my_pid_, sys_getpid()); 100 return Error(ENOSYS); 101 } else { 102 return Allow(); 103 } 104 } 105 106 private: 107 pid_t my_pid_; 108 DISALLOW_COPY_AND_ASSIGN(EnosysPtracePolicy); 109 }; 110 111 class BasicBPFTesterDelegate : public BPFTesterDelegate { 112 public: 113 BasicBPFTesterDelegate() {} 114 ~BasicBPFTesterDelegate() override {} 115 116 scoped_ptr<bpf_dsl::Policy> GetSandboxBPFPolicy() override { 117 return scoped_ptr<bpf_dsl::Policy>(new EnosysPtracePolicy()); 118 } 119 void RunTestFunction() override { 120 errno = 0; 121 int ret = ptrace(PTRACE_TRACEME, -1, NULL, NULL); 122 BPF_ASSERT(-1 == ret); 123 BPF_ASSERT(ENOSYS == errno); 124 } 125 126 private: 127 DISALLOW_COPY_AND_ASSIGN(BasicBPFTesterDelegate); 128 }; 129 130 // This is the most powerful and complex way to create a BPF test, but it 131 // requires a full class definition (BasicBPFTesterDelegate). 132 BPF_TEST_D(BPFTest, BPFTestWithDelegateClass, BasicBPFTesterDelegate); 133 134 // This is the simplest form of BPF tests. 135 BPF_TEST_C(BPFTest, BPFTestWithInlineTest, EnosysPtracePolicy) { 136 errno = 0; 137 int ret = ptrace(PTRACE_TRACEME, -1, NULL, NULL); 138 BPF_ASSERT(-1 == ret); 139 BPF_ASSERT(ENOSYS == errno); 140 } 141 142 const char kHelloMessage[] = "Hello"; 143 144 BPF_DEATH_TEST_C(BPFTest, 145 BPFDeathTestWithInlineTest, 146 DEATH_MESSAGE(kHelloMessage), 147 EnosysPtracePolicy) { 148 LOG(ERROR) << kHelloMessage; 149 _exit(1); 150 } 151 152 } // namespace 153 154 } // namespace sandbox 155