2016-01-14 23:35:40 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2016 The Android Open Source Project
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef LIBMEMUNREACHABLE_SCOPED_DISABLE_MALLOC_H_
|
|
|
|
#define LIBMEMUNREACHABLE_SCOPED_DISABLE_MALLOC_H_
|
|
|
|
|
|
|
|
#include <memory>
|
|
|
|
|
|
|
|
#include "android-base/macros.h"
|
|
|
|
|
|
|
|
#include "bionic.h"
|
|
|
|
#include "log.h"
|
|
|
|
#include "ScopedAlarm.h"
|
|
|
|
|
|
|
|
class DisableMallocGuard{
|
|
|
|
public:
|
|
|
|
DisableMallocGuard() : disabled_(false){}
|
|
|
|
~DisableMallocGuard() {
|
|
|
|
Enable();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Disable() {
|
|
|
|
if (!disabled_) {
|
|
|
|
malloc_disable();
|
|
|
|
disabled_ = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Enable() {
|
|
|
|
if (disabled_) {
|
|
|
|
malloc_enable();
|
|
|
|
disabled_ = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(DisableMallocGuard);
|
|
|
|
bool disabled_;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Any calls to malloc or free from this thread will deadlock as long as this
|
|
|
|
// object is in scope. Calls to malloc from other threads may succeed (for
|
|
|
|
// example if the allocation is satisfied out of the thread's tcache), or may
|
|
|
|
// block until the object is destroyed.
|
|
|
|
//
|
|
|
|
// Don't call fork() while malloc is disabled, it needs the same locks held
|
|
|
|
// here.
|
|
|
|
class ScopedDisableMalloc {
|
|
|
|
public:
|
|
|
|
ScopedDisableMalloc() {
|
|
|
|
disable_malloc_.Disable();
|
|
|
|
}
|
|
|
|
|
|
|
|
~ScopedDisableMalloc() {
|
|
|
|
disable_malloc_.Enable();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(ScopedDisableMalloc);
|
|
|
|
DisableMallocGuard disable_malloc_;
|
|
|
|
};
|
|
|
|
|
|
|
|
class ScopedDisableMallocTimeout {
|
|
|
|
public:
|
2016-07-12 20:50:44 +00:00
|
|
|
explicit ScopedDisableMallocTimeout(std::chrono::milliseconds timeout = std::chrono::milliseconds(2000)) :
|
2016-01-14 23:35:40 +00:00
|
|
|
timeout_(timeout), timed_out_(false), disable_malloc_() {
|
|
|
|
Disable();
|
|
|
|
}
|
|
|
|
|
|
|
|
~ScopedDisableMallocTimeout() {
|
|
|
|
Enable();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool timed_out() {
|
|
|
|
return timed_out_;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Enable() {
|
|
|
|
disable_malloc_.Enable();
|
|
|
|
alarm_ = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Disable() {
|
|
|
|
// set up the alarm before disabling malloc so unique_ptr can be used
|
|
|
|
alarm_ = std::make_unique<ScopedAlarm>(timeout_, [&]() {
|
|
|
|
disable_malloc_.Enable();
|
|
|
|
timed_out_ = true;
|
|
|
|
});
|
|
|
|
|
|
|
|
disable_malloc_.Disable();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(ScopedDisableMallocTimeout);
|
|
|
|
std::chrono::milliseconds timeout_;
|
|
|
|
bool timed_out_;
|
|
|
|
std::unique_ptr<ScopedAlarm> alarm_;
|
|
|
|
DisableMallocGuard disable_malloc_;
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif // LIBMEMUNREACHABLE_SCOPED_DISABLE_MALLOC_H_
|