1 /* 2 * Copyright (C) 2016 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #ifndef LIBMEMUNREACHABLE_SCOPED_DISABLE_MALLOC_H_ 18 #define LIBMEMUNREACHABLE_SCOPED_DISABLE_MALLOC_H_ 19 20 #include <memory> 21 22 #include "android-base/macros.h" 23 24 #include "ScopedAlarm.h" 25 #include "bionic.h" 26 #include "log.h" 27 28 namespace android { 29 30 class DisableMallocGuard { 31 public: 32 DisableMallocGuard() : disabled_(false) {} 33 ~DisableMallocGuard() { Enable(); } 34 35 void Disable() { 36 if (!disabled_) { 37 disabled_ = true; 38 malloc_disable(); 39 } 40 } 41 42 void Enable() { 43 if (disabled_) { 44 malloc_enable(); 45 disabled_ = false; 46 } 47 } 48 49 private: 50 DISALLOW_COPY_AND_ASSIGN(DisableMallocGuard); 51 bool disabled_; 52 }; 53 54 // Any calls to malloc or free from this thread will deadlock as long as this 55 // object is in scope. Calls to malloc from other threads may succeed (for 56 // example if the allocation is satisfied out of the thread's tcache), or may 57 // block until the object is destroyed. 58 // 59 // Don't call fork() while malloc is disabled, it needs the same locks held 60 // here. 61 class ScopedDisableMalloc { 62 public: 63 ScopedDisableMalloc() { disable_malloc_.Disable(); } 64 65 ~ScopedDisableMalloc() { disable_malloc_.Enable(); } 66 67 private: 68 DISALLOW_COPY_AND_ASSIGN(ScopedDisableMalloc); 69 DisableMallocGuard disable_malloc_; 70 }; 71 72 class ScopedDisableMallocTimeout { 73 public: 74 explicit ScopedDisableMallocTimeout(std::chrono::milliseconds timeout = std::chrono::seconds(10)) 75 : timeout_(timeout), timed_out_(false), disable_malloc_() { 76 Disable(); 77 } 78 79 ~ScopedDisableMallocTimeout() { Enable(); } 80 81 bool timed_out() { return timed_out_; } 82 83 void Enable() { 84 disable_malloc_.Enable(); 85 alarm_ = nullptr; 86 } 87 88 void Disable() { 89 // set up the alarm before disabling malloc so unique_ptr can be used 90 alarm_ = std::make_unique<ScopedAlarm>(timeout_, [&]() { 91 disable_malloc_.Enable(); 92 timed_out_ = true; 93 }); 94 95 disable_malloc_.Disable(); 96 } 97 98 private: 99 DISALLOW_COPY_AND_ASSIGN(ScopedDisableMallocTimeout); 100 std::chrono::milliseconds timeout_; 101 bool timed_out_; 102 std::unique_ptr<ScopedAlarm> alarm_; 103 DisableMallocGuard disable_malloc_; 104 }; 105 106 } // namespace android 107 108 #endif // LIBMEMUNREACHABLE_SCOPED_DISABLE_MALLOC_H_ 109