/* * Copyright (C) 2014 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. */ #include "transaction.h" #include "base/stl_util.h" #include "base/logging.h" #include "gc/accounting/card_table-inl.h" #include "intern_table.h" #include "mirror/class-inl.h" #include "mirror/object-inl.h" #include "mirror/object_array-inl.h" #include namespace art { // TODO: remove (only used for debugging purpose). static constexpr bool kEnableTransactionStats = false; Transaction::Transaction() : log_lock_("transaction log lock", kTransactionLogLock), aborted_(false) { CHECK(Runtime::Current()->IsAotCompiler()); } Transaction::~Transaction() { if (kEnableTransactionStats) { MutexLock mu(Thread::Current(), log_lock_); size_t objects_count = object_logs_.size(); size_t field_values_count = 0; for (auto it : object_logs_) { field_values_count += it.second.Size(); } size_t array_count = array_logs_.size(); size_t array_values_count = 0; for (auto it : array_logs_) { array_values_count += it.second.Size(); } size_t string_count = intern_string_logs_.size(); LOG(INFO) << "Transaction::~Transaction" << ": objects_count=" << objects_count << ", field_values_count=" << field_values_count << ", array_count=" << array_count << ", array_values_count=" << array_values_count << ", string_count=" << string_count; } } void Transaction::Abort(const std::string& abort_message) { MutexLock mu(Thread::Current(), log_lock_); // We may abort more than once if the exception thrown at the time of the // previous abort has been caught during execution of a class initializer. // We just keep the message of the first abort because it will cause the // transaction to be rolled back anyway. if (!aborted_) { aborted_ = true; abort_message_ = abort_message; } } void Transaction::ThrowAbortError(Thread* self, const std::string* abort_message) { const bool rethrow = (abort_message == nullptr); if (kIsDebugBuild && rethrow) { CHECK(IsAborted()) << "Rethrow " << Transaction::kAbortExceptionDescriptor << " while transaction is not aborted"; } if (rethrow) { // Rethrow an exception with the earlier abort message stored in the transaction. self->ThrowNewWrappedException(Transaction::kAbortExceptionSignature, GetAbortMessage().c_str()); } else { // Throw an exception with the given abort message. self->ThrowNewWrappedException(Transaction::kAbortExceptionSignature, abort_message->c_str()); } } bool Transaction::IsAborted() { MutexLock mu(Thread::Current(), log_lock_); return aborted_; } const std::string& Transaction::GetAbortMessage() { MutexLock mu(Thread::Current(), log_lock_); return abort_message_; } void Transaction::RecordWriteFieldBoolean(mirror::Object* obj, MemberOffset field_offset, uint8_t value, bool is_volatile) { DCHECK(obj != nullptr); MutexLock mu(Thread::Current(), log_lock_); ObjectLog& object_log = object_logs_[obj]; object_log.LogBooleanValue(field_offset, value, is_volatile); } void Transaction::RecordWriteFieldByte(mirror::Object* obj, MemberOffset field_offset, int8_t value, bool is_volatile) { DCHECK(obj != nullptr); MutexLock mu(Thread::Current(), log_lock_); ObjectLog& object_log = object_logs_[obj]; object_log.LogByteValue(field_offset, value, is_volatile); } void Transaction::RecordWriteFieldChar(mirror::Object* obj, MemberOffset field_offset, uint16_t value, bool is_volatile) { DCHECK(obj != nullptr); MutexLock mu(Thread::Current(), log_lock_); ObjectLog& object_log = object_logs_[obj]; object_log.LogCharValue(field_offset, value, is_volatile); } void Transaction::RecordWriteFieldShort(mirror::Object* obj, MemberOffset field_offset, int16_t value, bool is_volatile) { DCHECK(obj != nullptr); MutexLock mu(Thread::Current(), log_lock_); ObjectLog& object_log = object_logs_[obj]; object_log.LogShortValue(field_offset, value, is_volatile); } void Transaction::RecordWriteField32(mirror::Object* obj, MemberOffset field_offset, uint32_t value, bool is_volatile) { DCHECK(obj != nullptr); MutexLock mu(Thread::Current(), log_lock_); ObjectLog& object_log = object_logs_[obj]; object_log.Log32BitsValue(field_offset, value, is_volatile); } void Transaction::RecordWriteField64(mirror::Object* obj, MemberOffset field_offset, uint64_t value, bool is_volatile) { DCHECK(obj != nullptr); MutexLock mu(Thread::Current(), log_lock_); ObjectLog& object_log = object_logs_[obj]; object_log.Log64BitsValue(field_offset, value, is_volatile); } void Transaction::RecordWriteFieldReference(mirror::Object* obj, MemberOffset field_offset, mirror::Object* value, bool is_volatile) { DCHECK(obj != nullptr); MutexLock mu(Thread::Current(), log_lock_); ObjectLog& object_log = object_logs_[obj]; object_log.LogReferenceValue(field_offset, value, is_volatile); } void Transaction::RecordWriteArray(mirror::Array* array, size_t index, uint64_t value) { DCHECK(array != nullptr); DCHECK(array->IsArrayInstance()); DCHECK(!array->IsObjectArray()); MutexLock mu(Thread::Current(), log_lock_); ArrayLog& array_log = array_logs_[array]; array_log.LogValue(index, value); } void Transaction::RecordStrongStringInsertion(mirror::String* s) { InternStringLog log(s, InternStringLog::kStrongString, InternStringLog::kInsert); LogInternedString(log); } void Transaction::RecordWeakStringInsertion(mirror::String* s) { InternStringLog log(s, InternStringLog::kWeakString, InternStringLog::kInsert); LogInternedString(log); } void Transaction::RecordStrongStringRemoval(mirror::String* s) { InternStringLog log(s, InternStringLog::kStrongString, InternStringLog::kRemove); LogInternedString(log); } void Transaction::RecordWeakStringRemoval(mirror::String* s) { InternStringLog log(s, InternStringLog::kWeakString, InternStringLog::kRemove); LogInternedString(log); } void Transaction::LogInternedString(const InternStringLog& log) { Locks::intern_table_lock_->AssertExclusiveHeld(Thread::Current()); MutexLock mu(Thread::Current(), log_lock_); intern_string_logs_.push_front(log); } void Transaction::Rollback() { CHECK(!Runtime::Current()->IsActiveTransaction()); Thread* self = Thread::Current(); self->AssertNoPendingException(); MutexLock mu1(self, *Locks::intern_table_lock_); MutexLock mu2(self, log_lock_); UndoObjectModifications(); UndoArrayModifications(); UndoInternStringTableModifications(); } void Transaction::UndoObjectModifications() { // TODO we may not need to restore objects allocated during this transaction. Or we could directly // remove them from the heap. for (auto it : object_logs_) { it.second.Undo(it.first); } object_logs_.clear(); } void Transaction::UndoArrayModifications() { // TODO we may not need to restore array allocated during this transaction. Or we could directly // remove them from the heap. for (auto it : array_logs_) { it.second.Undo(it.first); } array_logs_.clear(); } void Transaction::UndoInternStringTableModifications() { InternTable* const intern_table = Runtime::Current()->GetInternTable(); // We want to undo each operation from the most recent to the oldest. List has been filled so the // most recent operation is at list begin so just have to iterate over it. for (InternStringLog& string_log : intern_string_logs_) { string_log.Undo(intern_table); } intern_string_logs_.clear(); } void Transaction::VisitRoots(RootVisitor* visitor) { MutexLock mu(Thread::Current(), log_lock_); VisitObjectLogs(visitor); VisitArrayLogs(visitor); VisitStringLogs(visitor); } void Transaction::VisitObjectLogs(RootVisitor* visitor) { // List of moving roots. typedef std::pair ObjectPair; std::list moving_roots; // Visit roots. for (auto it : object_logs_) { it.second.VisitRoots(visitor); mirror::Object* old_root = it.first; mirror::Object* new_root = old_root; visitor->VisitRoot(&new_root, RootInfo(kRootUnknown)); if (new_root != old_root) { moving_roots.push_back(std::make_pair(old_root, new_root)); } } // Update object logs with moving roots. for (const ObjectPair& pair : moving_roots) { mirror::Object* old_root = pair.first; mirror::Object* new_root = pair.second; auto old_root_it = object_logs_.find(old_root); CHECK(old_root_it != object_logs_.end()); CHECK(object_logs_.find(new_root) == object_logs_.end()); object_logs_.insert(std::make_pair(new_root, old_root_it->second)); object_logs_.erase(old_root_it); } } void Transaction::VisitArrayLogs(RootVisitor* visitor) { // List of moving roots. typedef std::pair ArrayPair; std::list moving_roots; for (auto it : array_logs_) { mirror::Array* old_root = it.first; CHECK(!old_root->IsObjectArray()); mirror::Array* new_root = old_root; visitor->VisitRoot(reinterpret_cast(&new_root), RootInfo(kRootUnknown)); if (new_root != old_root) { moving_roots.push_back(std::make_pair(old_root, new_root)); } } // Update array logs with moving roots. for (const ArrayPair& pair : moving_roots) { mirror::Array* old_root = pair.first; mirror::Array* new_root = pair.second; auto old_root_it = array_logs_.find(old_root); CHECK(old_root_it != array_logs_.end()); CHECK(array_logs_.find(new_root) == array_logs_.end()); array_logs_.insert(std::make_pair(new_root, old_root_it->second)); array_logs_.erase(old_root_it); } } void Transaction::VisitStringLogs(RootVisitor* visitor) { for (InternStringLog& log : intern_string_logs_) { log.VisitRoots(visitor); } } void Transaction::ObjectLog::LogBooleanValue(MemberOffset offset, uint8_t value, bool is_volatile) { LogValue(ObjectLog::kBoolean, offset, value, is_volatile); } void Transaction::ObjectLog::LogByteValue(MemberOffset offset, int8_t value, bool is_volatile) { LogValue(ObjectLog::kByte, offset, value, is_volatile); } void Transaction::ObjectLog::LogCharValue(MemberOffset offset, uint16_t value, bool is_volatile) { LogValue(ObjectLog::kChar, offset, value, is_volatile); } void Transaction::ObjectLog::LogShortValue(MemberOffset offset, int16_t value, bool is_volatile) { LogValue(ObjectLog::kShort, offset, value, is_volatile); } void Transaction::ObjectLog::Log32BitsValue(MemberOffset offset, uint32_t value, bool is_volatile) { LogValue(ObjectLog::k32Bits, offset, value, is_volatile); } void Transaction::ObjectLog::Log64BitsValue(MemberOffset offset, uint64_t value, bool is_volatile) { LogValue(ObjectLog::k64Bits, offset, value, is_volatile); } void Transaction::ObjectLog::LogReferenceValue(MemberOffset offset, mirror::Object* obj, bool is_volatile) { LogValue(ObjectLog::kReference, offset, reinterpret_cast(obj), is_volatile); } void Transaction::ObjectLog::LogValue(ObjectLog::FieldValueKind kind, MemberOffset offset, uint64_t value, bool is_volatile) { auto it = field_values_.find(offset.Uint32Value()); if (it == field_values_.end()) { ObjectLog::FieldValue field_value; field_value.value = value; field_value.is_volatile = is_volatile; field_value.kind = kind; field_values_.insert(std::make_pair(offset.Uint32Value(), field_value)); } } void Transaction::ObjectLog::Undo(mirror::Object* obj) { for (auto& it : field_values_) { // Garbage collector needs to access object's class and array's length. So we don't rollback // these values. MemberOffset field_offset(it.first); if (field_offset.Uint32Value() == mirror::Class::ClassOffset().Uint32Value()) { // Skip Object::class field. continue; } if (obj->IsArrayInstance() && field_offset.Uint32Value() == mirror::Array::LengthOffset().Uint32Value()) { // Skip Array::length field. continue; } FieldValue& field_value = it.second; UndoFieldWrite(obj, field_offset, field_value); } } void Transaction::ObjectLog::UndoFieldWrite(mirror::Object* obj, MemberOffset field_offset, const FieldValue& field_value) { // TODO We may want to abort a transaction while still being in transaction mode. In this case, // we'd need to disable the check. constexpr bool kCheckTransaction = true; switch (field_value.kind) { case kBoolean: if (UNLIKELY(field_value.is_volatile)) { obj->SetFieldBooleanVolatile(field_offset, static_cast(field_value.value)); } else { obj->SetFieldBoolean(field_offset, static_cast(field_value.value)); } break; case kByte: if (UNLIKELY(field_value.is_volatile)) { obj->SetFieldByteVolatile(field_offset, static_cast(field_value.value)); } else { obj->SetFieldByte(field_offset, static_cast(field_value.value)); } break; case kChar: if (UNLIKELY(field_value.is_volatile)) { obj->SetFieldCharVolatile(field_offset, static_cast(field_value.value)); } else { obj->SetFieldChar(field_offset, static_cast(field_value.value)); } break; case kShort: if (UNLIKELY(field_value.is_volatile)) { obj->SetFieldShortVolatile(field_offset, static_cast(field_value.value)); } else { obj->SetFieldShort(field_offset, static_cast(field_value.value)); } break; case k32Bits: if (UNLIKELY(field_value.is_volatile)) { obj->SetField32Volatile(field_offset, static_cast(field_value.value)); } else { obj->SetField32(field_offset, static_cast(field_value.value)); } break; case k64Bits: if (UNLIKELY(field_value.is_volatile)) { obj->SetField64Volatile(field_offset, field_value.value); } else { obj->SetField64(field_offset, field_value.value); } break; case kReference: if (UNLIKELY(field_value.is_volatile)) { obj->SetFieldObjectVolatile(field_offset, reinterpret_cast(field_value.value)); } else { obj->SetFieldObject(field_offset, reinterpret_cast(field_value.value)); } break; default: LOG(FATAL) << "Unknown value kind " << static_cast(field_value.kind); break; } } void Transaction::ObjectLog::VisitRoots(RootVisitor* visitor) { for (auto it : field_values_) { FieldValue& field_value = it.second; if (field_value.kind == ObjectLog::kReference) { visitor->VisitRootIfNonNull(reinterpret_cast(&field_value.value), RootInfo(kRootUnknown)); } } } void Transaction::InternStringLog::Undo(InternTable* intern_table) { DCHECK(intern_table != nullptr); switch (string_op_) { case InternStringLog::kInsert: { switch (string_kind_) { case InternStringLog::kStrongString: intern_table->RemoveStrongFromTransaction(str_); break; case InternStringLog::kWeakString: intern_table->RemoveWeakFromTransaction(str_); break; default: LOG(FATAL) << "Unknown interned string kind"; break; } break; } case InternStringLog::kRemove: { switch (string_kind_) { case InternStringLog::kStrongString: intern_table->InsertStrongFromTransaction(str_); break; case InternStringLog::kWeakString: intern_table->InsertWeakFromTransaction(str_); break; default: LOG(FATAL) << "Unknown interned string kind"; break; } break; } default: LOG(FATAL) << "Unknown interned string op"; break; } } void Transaction::InternStringLog::VisitRoots(RootVisitor* visitor) { visitor->VisitRoot(reinterpret_cast(&str_), RootInfo(kRootInternedString)); } void Transaction::ArrayLog::LogValue(size_t index, uint64_t value) { auto it = array_values_.find(index); if (it == array_values_.end()) { array_values_.insert(std::make_pair(index, value)); } } void Transaction::ArrayLog::Undo(mirror::Array* array) { DCHECK(array != nullptr); DCHECK(array->IsArrayInstance()); Primitive::Type type = array->GetClass()->GetComponentType()->GetPrimitiveType(); for (auto it : array_values_) { UndoArrayWrite(array, type, it.first, it.second); } } void Transaction::ArrayLog::UndoArrayWrite(mirror::Array* array, Primitive::Type array_type, size_t index, uint64_t value) { // TODO We may want to abort a transaction while still being in transaction mode. In this case, // we'd need to disable the check. switch (array_type) { case Primitive::kPrimBoolean: array->AsBooleanArray()->SetWithoutChecks(index, static_cast(value)); break; case Primitive::kPrimByte: array->AsByteArray()->SetWithoutChecks(index, static_cast(value)); break; case Primitive::kPrimChar: array->AsCharArray()->SetWithoutChecks(index, static_cast(value)); break; case Primitive::kPrimShort: array->AsShortArray()->SetWithoutChecks(index, static_cast(value)); break; case Primitive::kPrimInt: array->AsIntArray()->SetWithoutChecks(index, static_cast(value)); break; case Primitive::kPrimFloat: array->AsFloatArray()->SetWithoutChecks(index, static_cast(value)); break; case Primitive::kPrimLong: array->AsLongArray()->SetWithoutChecks(index, static_cast(value)); break; case Primitive::kPrimDouble: array->AsDoubleArray()->SetWithoutChecks(index, static_cast(value)); break; case Primitive::kPrimNot: LOG(FATAL) << "ObjectArray should be treated as Object"; break; default: LOG(FATAL) << "Unsupported type " << array_type; } } } // namespace art