/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=2 et sw=2 tw=80: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "IDBRequest.h" #include "BackgroundChildImpl.h" #include "IDBCursor.h" #include "IDBDatabase.h" #include "IDBEvents.h" #include "IDBFactory.h" #include "IDBIndex.h" #include "IDBObjectStore.h" #include "IDBTransaction.h" #include "mozilla/ContentEvents.h" #include "mozilla/ErrorResult.h" #include "mozilla/EventDispatcher.h" #include "mozilla/dom/ErrorEventBinding.h" #include "mozilla/dom/IDBOpenDBRequestBinding.h" #include "mozilla/dom/ScriptSettings.h" #include "mozilla/dom/UnionTypes.h" #include "nsCOMPtr.h" #include "nsContentUtils.h" #include "nsIScriptContext.h" #include "nsJSUtils.h" #include "nsPIDOMWindow.h" #include "nsString.h" #include "ReportInternalError.h" namespace mozilla { namespace dom { namespace indexedDB { using namespace mozilla::ipc; IDBRequest::IDBRequest(IDBDatabase* aDatabase) : IDBWrapperCache(aDatabase) { MOZ_ASSERT(aDatabase); aDatabase->AssertIsOnOwningThread(); InitMembers(); } IDBRequest::IDBRequest(nsPIDOMWindow* aOwner) : IDBWrapperCache(aOwner) { InitMembers(); } IDBRequest::~IDBRequest() { AssertIsOnOwningThread(); } #ifdef DEBUG void IDBRequest::AssertIsOnOwningThread() const { MOZ_ASSERT(mOwningThread); MOZ_ASSERT(PR_GetCurrentThread() == mOwningThread); } #endif // DEBUG void IDBRequest::InitMembers() { #ifdef DEBUG mOwningThread = PR_GetCurrentThread(); #endif AssertIsOnOwningThread(); mResultVal.setUndefined(); mErrorCode = NS_OK; mLineNo = 0; mHaveResultOrErrorCode = false; #ifdef MOZ_ENABLE_PROFILER_SPS { BackgroundChildImpl::ThreadLocal* threadLocal = BackgroundChildImpl::GetThreadLocalForCurrentThread(); MOZ_ASSERT(threadLocal); mSerialNumber = threadLocal->mNextRequestSerialNumber++; } #endif } // static already_AddRefed IDBRequest::Create(IDBDatabase* aDatabase, IDBTransaction* aTransaction) { MOZ_ASSERT(aDatabase); aDatabase->AssertIsOnOwningThread(); nsRefPtr request = new IDBRequest(aDatabase); request->mTransaction = aTransaction; request->SetScriptOwner(aDatabase->GetScriptOwner()); request->CaptureCaller(); return request.forget(); } // static already_AddRefed IDBRequest::Create(IDBObjectStore* aSourceAsObjectStore, IDBDatabase* aDatabase, IDBTransaction* aTransaction) { MOZ_ASSERT(aSourceAsObjectStore); aSourceAsObjectStore->AssertIsOnOwningThread(); nsRefPtr request = Create(aDatabase, aTransaction); request->mSourceAsObjectStore = aSourceAsObjectStore; return request.forget(); } // static already_AddRefed IDBRequest::Create(IDBIndex* aSourceAsIndex, IDBDatabase* aDatabase, IDBTransaction* aTransaction) { MOZ_ASSERT(aSourceAsIndex); aSourceAsIndex->AssertIsOnOwningThread(); nsRefPtr request = Create(aDatabase, aTransaction); request->mSourceAsIndex = aSourceAsIndex; return request.forget(); } void IDBRequest::GetSource( Nullable& aSource) const { AssertIsOnOwningThread(); MOZ_ASSERT_IF(mSourceAsObjectStore, !mSourceAsIndex); MOZ_ASSERT_IF(mSourceAsIndex, !mSourceAsObjectStore); MOZ_ASSERT_IF(mSourceAsCursor, mSourceAsObjectStore || mSourceAsIndex); // Always check cursor first since cursor requests hold both the cursor and // the objectStore or index the cursor came from. if (mSourceAsCursor) { aSource.SetValue().SetAsIDBCursor() = mSourceAsCursor; } else if (mSourceAsObjectStore) { aSource.SetValue().SetAsIDBObjectStore() = mSourceAsObjectStore; } else if (mSourceAsIndex) { aSource.SetValue().SetAsIDBIndex() = mSourceAsIndex; } else { aSource.SetNull(); } } void IDBRequest::Reset() { AssertIsOnOwningThread(); mResultVal.setUndefined(); mHaveResultOrErrorCode = false; mError = nullptr; } void IDBRequest::DispatchNonTransactionError(nsresult aErrorCode) { AssertIsOnOwningThread(); MOZ_ASSERT(NS_FAILED(aErrorCode)); MOZ_ASSERT(NS_ERROR_GET_MODULE(aErrorCode) == NS_ERROR_MODULE_DOM_INDEXEDDB); SetError(aErrorCode); // Make an error event and fire it at the target. nsCOMPtr event = CreateGenericEvent(this, nsDependentString(kErrorEventType), eDoesBubble, eCancelable); if (NS_WARN_IF(!event)) { return; } bool ignored; if (NS_FAILED(DispatchEvent(event, &ignored))) { NS_WARNING("Failed to dispatch event!"); } } void IDBRequest::SetError(nsresult aRv) { AssertIsOnOwningThread(); MOZ_ASSERT(NS_FAILED(aRv)); MOZ_ASSERT(NS_ERROR_GET_MODULE(aRv) == NS_ERROR_MODULE_DOM_INDEXEDDB); MOZ_ASSERT(!mError); mHaveResultOrErrorCode = true; mError = new DOMError(GetOwner(), aRv); mErrorCode = aRv; mResultVal.setUndefined(); } #ifdef DEBUG nsresult IDBRequest::GetErrorCode() const { AssertIsOnOwningThread(); MOZ_ASSERT(mHaveResultOrErrorCode); return mErrorCode; } #endif // DEBUG void IDBRequest::CaptureCaller() { AutoJSContext cx; const char* filename = nullptr; uint32_t lineNo = 0; if (!nsJSUtils::GetCallingLocation(cx, &filename, &lineNo)) { return; } mFilename.Assign(NS_ConvertUTF8toUTF16(filename)); mLineNo = lineNo; } void IDBRequest::FillScriptErrorEvent(ErrorEventInit& aEventInit) const { aEventInit.mLineno = mLineNo; aEventInit.mFilename = mFilename; } IDBRequestReadyState IDBRequest::ReadyState() const { AssertIsOnOwningThread(); return IsPending() ? IDBRequestReadyState::Pending : IDBRequestReadyState::Done; } void IDBRequest::SetSource(IDBCursor* aSource) { AssertIsOnOwningThread(); MOZ_ASSERT(aSource); MOZ_ASSERT(mSourceAsObjectStore || mSourceAsIndex); MOZ_ASSERT(!mSourceAsCursor); mSourceAsCursor = aSource; } JSObject* IDBRequest::WrapObject(JSContext* aCx) { return IDBRequestBinding::Wrap(aCx, this); } void IDBRequest::GetResult(JS::MutableHandle aResult, ErrorResult& aRv) const { AssertIsOnOwningThread(); if (!mHaveResultOrErrorCode) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); return; } JS::ExposeValueToActiveJS(mResultVal); aResult.set(mResultVal); } void IDBRequest::SetResultCallback(ResultCallback* aCallback) { AssertIsOnOwningThread(); MOZ_ASSERT(aCallback); MOZ_ASSERT(!mHaveResultOrErrorCode); MOZ_ASSERT(mResultVal.isUndefined()); MOZ_ASSERT(!mError); // See if our window is still valid. if (NS_WARN_IF(NS_FAILED(CheckInnerWindowCorrectness()))) { IDB_REPORT_INTERNAL_ERR(); SetError(NS_ERROR_DOM_INDEXEDDB_UNKNOWN_ERR); return; } AutoJSAPI autoJS; Maybe ac; if (GetScriptOwner()) { // If we have a script owner we want the SafeJSContext and then to enter the // script owner's compartment. autoJS.Init(); ac.emplace(autoJS.cx(), GetScriptOwner()); } else { // Otherwise our owner is a window and we use that to initialize. MOZ_ASSERT(GetOwner()); if (!autoJS.InitWithLegacyErrorReporting(GetOwner())) { IDB_WARNING("Failed to initialize AutoJSAPI!"); SetError(NS_ERROR_DOM_INDEXEDDB_UNKNOWN_ERR); return; } } JSContext* cx = autoJS.cx(); AssertIsRooted(); JS::Rooted result(cx); nsresult rv = aCallback->GetResult(cx, &result); if (NS_WARN_IF(NS_FAILED(rv))) { SetError(rv); mResultVal.setUndefined(); } else { mError = nullptr; mResultVal = result; } mHaveResultOrErrorCode = true; } DOMError* IDBRequest::GetError(ErrorResult& aRv) { AssertIsOnOwningThread(); if (!mHaveResultOrErrorCode) { aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR); return nullptr; } return mError; } NS_IMPL_CYCLE_COLLECTION_CLASS(IDBRequest) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(IDBRequest, IDBWrapperCache) // Don't need NS_IMPL_CYCLE_COLLECTION_TRAVERSE_SCRIPT_OBJECTS because // DOMEventTargetHelper does it for us. NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mSourceAsObjectStore) NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mSourceAsIndex) NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mSourceAsCursor) NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mTransaction) NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mError) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(IDBRequest, IDBWrapperCache) tmp->mResultVal.setUndefined(); NS_IMPL_CYCLE_COLLECTION_UNLINK(mSourceAsObjectStore) NS_IMPL_CYCLE_COLLECTION_UNLINK(mSourceAsIndex) NS_IMPL_CYCLE_COLLECTION_UNLINK(mSourceAsCursor) NS_IMPL_CYCLE_COLLECTION_UNLINK(mTransaction) NS_IMPL_CYCLE_COLLECTION_UNLINK(mError) NS_IMPL_CYCLE_COLLECTION_UNLINK_END NS_IMPL_CYCLE_COLLECTION_TRACE_BEGIN_INHERITED(IDBRequest, IDBWrapperCache) // Don't need NS_IMPL_CYCLE_COLLECTION_TRACE_PRESERVED_WRAPPER because // DOMEventTargetHelper does it for us. NS_IMPL_CYCLE_COLLECTION_TRACE_JSVAL_MEMBER_CALLBACK(mResultVal) NS_IMPL_CYCLE_COLLECTION_TRACE_END NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION_INHERITED(IDBRequest) NS_INTERFACE_MAP_END_INHERITING(IDBWrapperCache) NS_IMPL_ADDREF_INHERITED(IDBRequest, IDBWrapperCache) NS_IMPL_RELEASE_INHERITED(IDBRequest, IDBWrapperCache) nsresult IDBRequest::PreHandleEvent(EventChainPreVisitor& aVisitor) { AssertIsOnOwningThread(); aVisitor.mCanHandle = true; aVisitor.mParentTarget = mTransaction; return NS_OK; } IDBOpenDBRequest::IDBOpenDBRequest(IDBFactory* aFactory, nsPIDOMWindow* aOwner) : IDBRequest(aOwner) , mFactory(aFactory) { AssertIsOnOwningThread(); MOZ_ASSERT(aFactory); // aOwner may be null. } IDBOpenDBRequest::~IDBOpenDBRequest() { AssertIsOnOwningThread(); } // static already_AddRefed IDBOpenDBRequest::CreateForWindow(IDBFactory* aFactory, nsPIDOMWindow* aOwner, JS::Handle aScriptOwner) { MOZ_ASSERT(aFactory); aFactory->AssertIsOnOwningThread(); MOZ_ASSERT(aOwner); MOZ_ASSERT(aScriptOwner); nsRefPtr request = new IDBOpenDBRequest(aFactory, aOwner); request->CaptureCaller(); request->SetScriptOwner(aScriptOwner); return request.forget(); } // static already_AddRefed IDBOpenDBRequest::CreateForJS(IDBFactory* aFactory, JS::Handle aScriptOwner) { MOZ_ASSERT(aFactory); aFactory->AssertIsOnOwningThread(); MOZ_ASSERT(aScriptOwner); nsRefPtr request = new IDBOpenDBRequest(aFactory, nullptr); request->CaptureCaller(); request->SetScriptOwner(aScriptOwner); return request.forget(); } void IDBOpenDBRequest::SetTransaction(IDBTransaction* aTransaction) { AssertIsOnOwningThread(); MOZ_ASSERT(!aTransaction || !mTransaction); mTransaction = aTransaction; } NS_IMPL_CYCLE_COLLECTION_CLASS(IDBOpenDBRequest) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(IDBOpenDBRequest, IDBRequest) NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mFactory) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(IDBOpenDBRequest, IDBRequest) // Don't unlink mFactory! NS_IMPL_CYCLE_COLLECTION_UNLINK_END NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION_INHERITED(IDBOpenDBRequest) NS_INTERFACE_MAP_END_INHERITING(IDBRequest) NS_IMPL_ADDREF_INHERITED(IDBOpenDBRequest, IDBRequest) NS_IMPL_RELEASE_INHERITED(IDBOpenDBRequest, IDBRequest) nsresult IDBOpenDBRequest::PostHandleEvent(EventChainPostVisitor& aVisitor) { // XXX Fix me! MOZ_ASSERT(NS_IsMainThread()); return IndexedDatabaseManager::FireWindowOnError(GetOwner(), aVisitor); } JSObject* IDBOpenDBRequest::WrapObject(JSContext* aCx) { AssertIsOnOwningThread(); return IDBOpenDBRequestBinding::Wrap(aCx, this); } } // namespace indexedDB } // namespace dom } // namespace mozilla