| /* |
| * Copyright (C) 2015 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 ART_RUNTIME_GC_SCOPED_GC_CRITICAL_SECTION_H_ |
| #define ART_RUNTIME_GC_SCOPED_GC_CRITICAL_SECTION_H_ |
| |
| #include "base/locks.h" |
| #include "collector_type.h" |
| #include "gc_cause.h" |
| |
| namespace art { |
| |
| class Thread; |
| |
| namespace gc { |
| |
| // The use of ScopedGCCriticalSection should be preferred whenever possible. |
| class GCCriticalSection { |
| public: |
| GCCriticalSection(Thread* self, const char* name) |
| : self_(self), section_name_(name) {} |
| ~GCCriticalSection() {} |
| |
| // Starts a GCCriticalSection. Returns the previous no-suspension reason. |
| const char* Enter(GcCause cause, CollectorType type) ACQUIRE(Roles::uninterruptible_); |
| |
| // Ends a GCCriticalSection. Takes the old no-suspension reason. |
| void Exit(const char* old_reason) RELEASE(Roles::uninterruptible_); |
| |
| private: |
| Thread* const self_; |
| const char* section_name_; |
| }; |
| |
| // Wait until the GC is finished and then prevent the GC from starting until the destructor. Used |
| // to prevent deadlocks in places where we call ClassLinker::VisitClass with all the threads |
| // suspended. |
| class ScopedGCCriticalSection { |
| public: |
| ScopedGCCriticalSection(Thread* self, GcCause cause, CollectorType collector_type) |
| ACQUIRE(Roles::uninterruptible_); |
| ~ScopedGCCriticalSection() RELEASE(Roles::uninterruptible_); |
| |
| private: |
| GCCriticalSection critical_section_; |
| const char* old_no_suspend_reason_; |
| }; |
| |
| // The use of ScopedGCCriticalSection should be preferred whenever possible. |
| // This class allows thread suspension but should never be used with allocations because of the |
| // deadlock risk. TODO: Add a new thread role for "no allocations" that still allows suspension. |
| class ScopedInterruptibleGCCriticalSection { |
| public: |
| ScopedInterruptibleGCCriticalSection(Thread* self, GcCause cause, CollectorType type); |
| ~ScopedInterruptibleGCCriticalSection(); |
| |
| private: |
| Thread* const self_; |
| }; |
| |
| |
| } // namespace gc |
| } // namespace art |
| |
| #endif // ART_RUNTIME_GC_SCOPED_GC_CRITICAL_SECTION_H_ |