mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
b3c31c0c49
--HG-- rename : dom/indexedDB/ipc/SerializationHelpers.h => dom/indexedDB/SerializationHelpers.h rename : dom/indexedDB/ipc/unit/head.js => dom/indexedDB/test/unit/xpcshell-head-child-process.js rename : dom/indexedDB/test/unit/head.js => dom/indexedDB/test/unit/xpcshell-head-parent-process.js rename : dom/ipc/Blob.h => dom/ipc/BlobParent.h rename : dom/ipc/FileDescriptorSetChild.cpp => ipc/glue/FileDescriptorSetChild.cpp rename : dom/ipc/FileDescriptorSetChild.h => ipc/glue/FileDescriptorSetChild.h rename : dom/ipc/FileDescriptorSetParent.cpp => ipc/glue/FileDescriptorSetParent.cpp rename : dom/ipc/FileDescriptorSetParent.h => ipc/glue/FileDescriptorSetParent.h rename : dom/ipc/PFileDescriptorSet.ipdl => ipc/glue/PFileDescriptorSet.ipdl
855 lines
22 KiB
C++
855 lines
22 KiB
C++
/* -*- 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 "IDBTransaction.h"
|
|
|
|
#include "BackgroundChildImpl.h"
|
|
#include "IDBDatabase.h"
|
|
#include "IDBEvents.h"
|
|
#include "IDBObjectStore.h"
|
|
#include "IDBRequest.h"
|
|
#include "mozilla/ErrorResult.h"
|
|
#include "mozilla/EventDispatcher.h"
|
|
#include "mozilla/dom/DOMError.h"
|
|
#include "mozilla/dom/DOMStringList.h"
|
|
#include "mozilla/ipc/BackgroundChild.h"
|
|
#include "nsIAppShell.h"
|
|
#include "nsIDOMFile.h"
|
|
#include "nsPIDOMWindow.h"
|
|
#include "nsServiceManagerUtils.h"
|
|
#include "nsTHashtable.h"
|
|
#include "nsWidgetsCID.h"
|
|
#include "ProfilerHelpers.h"
|
|
#include "ReportInternalError.h"
|
|
|
|
// Include this last to avoid path problems on Windows.
|
|
#include "ActorsChild.h"
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
namespace indexedDB {
|
|
|
|
namespace {
|
|
|
|
NS_DEFINE_CID(kAppShellCID, NS_APPSHELL_CID);
|
|
|
|
} // anonymous namespace
|
|
|
|
IDBTransaction::IDBTransaction(IDBDatabase* aDatabase,
|
|
const nsTArray<nsString>& aObjectStoreNames,
|
|
Mode aMode)
|
|
: IDBWrapperCache(aDatabase)
|
|
, mDatabase(aDatabase)
|
|
, mObjectStoreNames(aObjectStoreNames)
|
|
, mNextObjectStoreId(0)
|
|
, mNextIndexId(0)
|
|
, mAbortCode(NS_OK)
|
|
, mPendingRequestCount(0)
|
|
, mReadyState(IDBTransaction::INITIAL)
|
|
, mMode(aMode)
|
|
, mCreating(false)
|
|
, mAbortedByScript(false)
|
|
#ifdef DEBUG
|
|
, mSentCommitOrAbort(false)
|
|
, mFiredCompleteOrAbort(false)
|
|
#endif
|
|
{
|
|
MOZ_ASSERT(aDatabase);
|
|
aDatabase->AssertIsOnOwningThread();
|
|
|
|
mBackgroundActor.mNormalBackgroundActor = nullptr;
|
|
|
|
#ifdef MOZ_ENABLE_PROFILER_SPS
|
|
{
|
|
using namespace mozilla::ipc;
|
|
BackgroundChildImpl::ThreadLocal* threadLocal =
|
|
BackgroundChildImpl::GetThreadLocalForCurrentThread();
|
|
MOZ_ASSERT(threadLocal);
|
|
|
|
mSerialNumber = threadLocal->mNextTransactionSerialNumber++;
|
|
}
|
|
#endif
|
|
|
|
#ifdef DEBUG
|
|
if (!aObjectStoreNames.IsEmpty()) {
|
|
nsTArray<nsString> sortedNames(aObjectStoreNames);
|
|
sortedNames.Sort();
|
|
|
|
const uint32_t count = sortedNames.Length();
|
|
MOZ_ASSERT(count == aObjectStoreNames.Length());
|
|
|
|
// Make sure the array is properly sorted.
|
|
for (uint32_t index = 0; index < count; index++) {
|
|
MOZ_ASSERT(aObjectStoreNames[index] == sortedNames[index]);
|
|
}
|
|
|
|
// Make sure there are no duplicates in our objectStore names.
|
|
for (uint32_t index = 0; index < count - 1; index++) {
|
|
MOZ_ASSERT(sortedNames[index] != sortedNames[index + 1]);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
IDBTransaction::~IDBTransaction()
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(!mPendingRequestCount);
|
|
MOZ_ASSERT(!mCreating);
|
|
MOZ_ASSERT(mSentCommitOrAbort);
|
|
MOZ_ASSERT_IF(mMode == VERSION_CHANGE &&
|
|
mBackgroundActor.mVersionChangeBackgroundActor,
|
|
mFiredCompleteOrAbort);
|
|
MOZ_ASSERT_IF(mMode != VERSION_CHANGE &&
|
|
mBackgroundActor.mNormalBackgroundActor,
|
|
mFiredCompleteOrAbort);
|
|
|
|
mDatabase->UnregisterTransaction(this);
|
|
|
|
if (mMode == VERSION_CHANGE) {
|
|
if (mBackgroundActor.mVersionChangeBackgroundActor) {
|
|
mBackgroundActor.mVersionChangeBackgroundActor->SendDeleteMeInternal();
|
|
MOZ_ASSERT(!mBackgroundActor.mVersionChangeBackgroundActor,
|
|
"SendDeleteMeInternal should have cleared!");
|
|
}
|
|
} else if (mBackgroundActor.mNormalBackgroundActor) {
|
|
mBackgroundActor.mNormalBackgroundActor->SendDeleteMeInternal();
|
|
MOZ_ASSERT(!mBackgroundActor.mNormalBackgroundActor,
|
|
"SendDeleteMeInternal should have cleared!");
|
|
}
|
|
}
|
|
|
|
// static
|
|
already_AddRefed<IDBTransaction>
|
|
IDBTransaction::CreateVersionChange(
|
|
IDBDatabase* aDatabase,
|
|
BackgroundVersionChangeTransactionChild* aActor,
|
|
int64_t aNextObjectStoreId,
|
|
int64_t aNextIndexId)
|
|
{
|
|
MOZ_ASSERT(aDatabase);
|
|
aDatabase->AssertIsOnOwningThread();
|
|
MOZ_ASSERT(aActor);
|
|
MOZ_ASSERT(aNextObjectStoreId > 0);
|
|
MOZ_ASSERT(aNextIndexId > 0);
|
|
|
|
nsTArray<nsString> emptyObjectStoreNames;
|
|
|
|
nsRefPtr<IDBTransaction> transaction =
|
|
new IDBTransaction(aDatabase, emptyObjectStoreNames, VERSION_CHANGE);
|
|
|
|
transaction->SetScriptOwner(aDatabase->GetScriptOwner());
|
|
transaction->mBackgroundActor.mVersionChangeBackgroundActor = aActor;
|
|
transaction->mNextObjectStoreId = aNextObjectStoreId;
|
|
transaction->mNextIndexId = aNextIndexId;
|
|
|
|
// XXX Fix!
|
|
MOZ_ASSERT(NS_IsMainThread(), "This won't work on non-main threads!");
|
|
|
|
nsCOMPtr<nsIAppShell> appShell = do_GetService(kAppShellCID);
|
|
if (NS_WARN_IF(!appShell) ||
|
|
NS_WARN_IF(NS_FAILED(appShell->RunBeforeNextEvent(transaction)))) {
|
|
return nullptr;
|
|
}
|
|
|
|
transaction->mCreating = true;
|
|
|
|
aDatabase->RegisterTransaction(transaction);
|
|
|
|
return transaction.forget();
|
|
}
|
|
|
|
// static
|
|
already_AddRefed<IDBTransaction>
|
|
IDBTransaction::Create(IDBDatabase* aDatabase,
|
|
const nsTArray<nsString>& aObjectStoreNames,
|
|
Mode aMode)
|
|
{
|
|
MOZ_ASSERT(aDatabase);
|
|
aDatabase->AssertIsOnOwningThread();
|
|
MOZ_ASSERT(!aObjectStoreNames.IsEmpty());
|
|
MOZ_ASSERT(aMode == READ_ONLY || aMode == READ_WRITE);
|
|
|
|
nsRefPtr<IDBTransaction> transaction =
|
|
new IDBTransaction(aDatabase, aObjectStoreNames, aMode);
|
|
|
|
transaction->SetScriptOwner(aDatabase->GetScriptOwner());
|
|
|
|
// XXX Fix!
|
|
MOZ_ASSERT(NS_IsMainThread(), "This won't work on non-main threads!");
|
|
|
|
nsCOMPtr<nsIAppShell> appShell = do_GetService(kAppShellCID);
|
|
if (NS_WARN_IF(!appShell) ||
|
|
NS_WARN_IF(NS_FAILED(appShell->RunBeforeNextEvent(transaction)))) {
|
|
return nullptr;
|
|
}
|
|
|
|
transaction->mCreating = true;
|
|
|
|
aDatabase->RegisterTransaction(transaction);
|
|
|
|
return transaction.forget();
|
|
}
|
|
|
|
// static
|
|
IDBTransaction*
|
|
IDBTransaction::GetCurrent()
|
|
{
|
|
using namespace mozilla::ipc;
|
|
|
|
MOZ_ASSERT(BackgroundChild::GetForCurrentThread());
|
|
|
|
BackgroundChildImpl::ThreadLocal* threadLocal =
|
|
BackgroundChildImpl::GetThreadLocalForCurrentThread();
|
|
MOZ_ASSERT(threadLocal);
|
|
|
|
return threadLocal->mCurrentTransaction;
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
|
|
void
|
|
IDBTransaction::AssertIsOnOwningThread() const
|
|
{
|
|
MOZ_ASSERT(mDatabase);
|
|
mDatabase->AssertIsOnOwningThread();
|
|
}
|
|
|
|
#endif // DEBUG
|
|
|
|
void
|
|
IDBTransaction::SetBackgroundActor(BackgroundTransactionChild* aBackgroundActor)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(aBackgroundActor);
|
|
MOZ_ASSERT(!mBackgroundActor.mNormalBackgroundActor);
|
|
MOZ_ASSERT(mMode != VERSION_CHANGE);
|
|
|
|
mBackgroundActor.mNormalBackgroundActor = aBackgroundActor;
|
|
}
|
|
|
|
void
|
|
IDBTransaction::StartRequest(BackgroundRequestChild* aBackgroundActor,
|
|
const RequestParams& aParams)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(aBackgroundActor);
|
|
MOZ_ASSERT(aParams.type() != RequestParams::T__None);
|
|
|
|
if (mMode == VERSION_CHANGE) {
|
|
MOZ_ASSERT(mBackgroundActor.mVersionChangeBackgroundActor);
|
|
|
|
mBackgroundActor.mVersionChangeBackgroundActor->
|
|
SendPBackgroundIDBRequestConstructor(aBackgroundActor, aParams);
|
|
} else {
|
|
MOZ_ASSERT(mBackgroundActor.mNormalBackgroundActor);
|
|
|
|
mBackgroundActor.mNormalBackgroundActor->
|
|
SendPBackgroundIDBRequestConstructor(aBackgroundActor, aParams);
|
|
}
|
|
}
|
|
|
|
void
|
|
IDBTransaction::OpenCursor(BackgroundCursorChild* aBackgroundActor,
|
|
const OpenCursorParams& aParams)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(aBackgroundActor);
|
|
MOZ_ASSERT(aParams.type() != OpenCursorParams::T__None);
|
|
|
|
if (mMode == VERSION_CHANGE) {
|
|
MOZ_ASSERT(mBackgroundActor.mVersionChangeBackgroundActor);
|
|
|
|
mBackgroundActor.mVersionChangeBackgroundActor->
|
|
SendPBackgroundIDBCursorConstructor(aBackgroundActor, aParams);
|
|
} else {
|
|
MOZ_ASSERT(mBackgroundActor.mNormalBackgroundActor);
|
|
|
|
mBackgroundActor.mNormalBackgroundActor->
|
|
SendPBackgroundIDBCursorConstructor(aBackgroundActor, aParams);
|
|
}
|
|
|
|
// Balanced in BackgroundCursorChild::RecvResponse().
|
|
OnNewRequest();
|
|
}
|
|
|
|
void
|
|
IDBTransaction::RefreshSpec(bool aMayDelete)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
|
|
for (uint32_t count = mObjectStores.Length(), index = 0;
|
|
index < count;
|
|
index++) {
|
|
mObjectStores[index]->RefreshSpec(aMayDelete);
|
|
}
|
|
|
|
for (uint32_t count = mDeletedObjectStores.Length(), index = 0;
|
|
index < count;
|
|
index++) {
|
|
mDeletedObjectStores[index]->RefreshSpec(false);
|
|
}
|
|
}
|
|
|
|
void
|
|
IDBTransaction::OnNewRequest()
|
|
{
|
|
AssertIsOnOwningThread();
|
|
|
|
if (!mPendingRequestCount) {
|
|
MOZ_ASSERT(INITIAL == mReadyState);
|
|
mReadyState = LOADING;
|
|
}
|
|
|
|
++mPendingRequestCount;
|
|
}
|
|
|
|
void
|
|
IDBTransaction::OnRequestFinished()
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(mPendingRequestCount);
|
|
|
|
--mPendingRequestCount;
|
|
|
|
if (!mPendingRequestCount && !mDatabase->IsInvalidated()) {
|
|
mReadyState = COMMITTING;
|
|
|
|
if (NS_SUCCEEDED(mAbortCode)) {
|
|
SendCommit();
|
|
} else {
|
|
SendAbort(mAbortCode);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
IDBTransaction::SendCommit()
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(NS_SUCCEEDED(mAbortCode));
|
|
MOZ_ASSERT(IsFinished());
|
|
MOZ_ASSERT(!mSentCommitOrAbort);
|
|
|
|
if (mMode == VERSION_CHANGE) {
|
|
MOZ_ASSERT(mBackgroundActor.mVersionChangeBackgroundActor);
|
|
mBackgroundActor.mVersionChangeBackgroundActor->SendCommit();
|
|
} else {
|
|
MOZ_ASSERT(mBackgroundActor.mNormalBackgroundActor);
|
|
mBackgroundActor.mNormalBackgroundActor->SendCommit();
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
mSentCommitOrAbort = true;
|
|
#endif
|
|
}
|
|
|
|
void
|
|
IDBTransaction::SendAbort(nsresult aResultCode)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(NS_FAILED(aResultCode));
|
|
MOZ_ASSERT(IsFinished());
|
|
MOZ_ASSERT(!mSentCommitOrAbort);
|
|
|
|
if (mMode == VERSION_CHANGE) {
|
|
MOZ_ASSERT(mBackgroundActor.mVersionChangeBackgroundActor);
|
|
mBackgroundActor.mVersionChangeBackgroundActor->SendAbort(aResultCode);
|
|
} else {
|
|
MOZ_ASSERT(mBackgroundActor.mNormalBackgroundActor);
|
|
mBackgroundActor.mNormalBackgroundActor->SendAbort(aResultCode);
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
mSentCommitOrAbort = true;
|
|
#endif
|
|
}
|
|
|
|
bool
|
|
IDBTransaction::IsOpen() const
|
|
{
|
|
AssertIsOnOwningThread();
|
|
|
|
// If we haven't started anything then we're open.
|
|
if (mReadyState == IDBTransaction::INITIAL) {
|
|
return true;
|
|
}
|
|
|
|
// If we've already started then we need to check to see if we still have the
|
|
// mCreating flag set. If we do (i.e. we haven't returned to the event loop
|
|
// from the time we were created) then we are open. Otherwise check the
|
|
// currently running transaction to see if it's the same. We only allow other
|
|
// requests to be made if this transaction is currently running.
|
|
if (mReadyState == IDBTransaction::LOADING &&
|
|
(mCreating || GetCurrent() == this)) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
already_AddRefed<IDBObjectStore>
|
|
IDBTransaction::CreateObjectStore(const ObjectStoreSpec& aSpec)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(aSpec.metadata().id());
|
|
MOZ_ASSERT(VERSION_CHANGE == mMode);
|
|
MOZ_ASSERT(mBackgroundActor.mVersionChangeBackgroundActor);
|
|
MOZ_ASSERT(IsOpen());
|
|
|
|
#ifdef DEBUG
|
|
{
|
|
const nsString& name = aSpec.metadata().name();
|
|
|
|
for (uint32_t count = mObjectStores.Length(), index = 0;
|
|
index < count;
|
|
index++) {
|
|
MOZ_ASSERT(mObjectStores[index]->Name() != name);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
MOZ_ALWAYS_TRUE(mBackgroundActor.mVersionChangeBackgroundActor->
|
|
SendCreateObjectStore(aSpec.metadata()));
|
|
|
|
nsRefPtr<IDBObjectStore> objectStore = IDBObjectStore::Create(this, aSpec);
|
|
MOZ_ASSERT(objectStore);
|
|
|
|
mObjectStores.AppendElement(objectStore);
|
|
|
|
return objectStore.forget();
|
|
}
|
|
|
|
void
|
|
IDBTransaction::DeleteObjectStore(int64_t aObjectStoreId)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(aObjectStoreId);
|
|
MOZ_ASSERT(VERSION_CHANGE == mMode);
|
|
MOZ_ASSERT(mBackgroundActor.mVersionChangeBackgroundActor);
|
|
MOZ_ASSERT(IsOpen());
|
|
|
|
MOZ_ALWAYS_TRUE(mBackgroundActor.mVersionChangeBackgroundActor->
|
|
SendDeleteObjectStore(aObjectStoreId));
|
|
|
|
for (uint32_t count = mObjectStores.Length(), index = 0;
|
|
index < count;
|
|
index++) {
|
|
nsRefPtr<IDBObjectStore>& objectStore = mObjectStores[index];
|
|
|
|
if (objectStore->Id() == aObjectStoreId) {
|
|
objectStore->NoteDeletion();
|
|
|
|
nsRefPtr<IDBObjectStore>* deletedObjectStore =
|
|
mDeletedObjectStores.AppendElement();
|
|
deletedObjectStore->swap(mObjectStores[index]);
|
|
|
|
mObjectStores.RemoveElementAt(index);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
IDBTransaction::CreateIndex(IDBObjectStore* aObjectStore,
|
|
const IndexMetadata& aMetadata)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(aObjectStore);
|
|
MOZ_ASSERT(aMetadata.id());
|
|
MOZ_ASSERT(VERSION_CHANGE == mMode);
|
|
MOZ_ASSERT(mBackgroundActor.mVersionChangeBackgroundActor);
|
|
MOZ_ASSERT(IsOpen());
|
|
|
|
MOZ_ALWAYS_TRUE(mBackgroundActor.mVersionChangeBackgroundActor->
|
|
SendCreateIndex(aObjectStore->Id(), aMetadata));
|
|
}
|
|
|
|
void
|
|
IDBTransaction::DeleteIndex(IDBObjectStore* aObjectStore,
|
|
int64_t aIndexId)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(aObjectStore);
|
|
MOZ_ASSERT(aIndexId);
|
|
MOZ_ASSERT(VERSION_CHANGE == mMode);
|
|
MOZ_ASSERT(mBackgroundActor.mVersionChangeBackgroundActor);
|
|
MOZ_ASSERT(IsOpen());
|
|
|
|
MOZ_ALWAYS_TRUE(mBackgroundActor.mVersionChangeBackgroundActor->
|
|
SendDeleteIndex(aObjectStore->Id(), aIndexId));
|
|
}
|
|
|
|
void
|
|
IDBTransaction::AbortInternal(nsresult aAbortCode,
|
|
already_AddRefed<DOMError> aError)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(NS_FAILED(aAbortCode));
|
|
|
|
nsRefPtr<DOMError> error = aError;
|
|
|
|
if (IsFinished()) {
|
|
// Already finished, nothing to do here.
|
|
return;
|
|
}
|
|
|
|
const bool isVersionChange = mMode == VERSION_CHANGE;
|
|
const bool isInvalidated = mDatabase->IsInvalidated();
|
|
bool needToSendAbort = mReadyState == INITIAL && !isInvalidated;
|
|
|
|
#ifdef DEBUG
|
|
if (isInvalidated) {
|
|
mSentCommitOrAbort = true;
|
|
}
|
|
#endif
|
|
|
|
mAbortCode = aAbortCode;
|
|
mReadyState = DONE;
|
|
mError = error.forget();
|
|
|
|
if (isVersionChange) {
|
|
// If a version change transaction is aborted, we must revert the world
|
|
// back to its previous state unless we're being invalidated after the
|
|
// transaction already completed.
|
|
if (!isInvalidated) {
|
|
mDatabase->RevertToPreviousState();
|
|
}
|
|
|
|
const nsTArray<ObjectStoreSpec>& specArray =
|
|
mDatabase->Spec()->objectStores();
|
|
|
|
if (specArray.IsEmpty()) {
|
|
mObjectStores.Clear();
|
|
mDeletedObjectStores.Clear();
|
|
} else {
|
|
nsTHashtable<nsUint64HashKey> validIds(specArray.Length());
|
|
|
|
for (uint32_t specCount = specArray.Length(), specIndex = 0;
|
|
specIndex < specCount;
|
|
specIndex++) {
|
|
const int64_t objectStoreId = specArray[specIndex].metadata().id();
|
|
MOZ_ASSERT(objectStoreId);
|
|
|
|
validIds.PutEntry(uint64_t(objectStoreId));
|
|
}
|
|
|
|
for (uint32_t objCount = mObjectStores.Length(), objIndex = 0;
|
|
objIndex < objCount;
|
|
/* incremented conditionally */) {
|
|
const int64_t objectStoreId = mObjectStores[objIndex]->Id();
|
|
MOZ_ASSERT(objectStoreId);
|
|
|
|
if (validIds.Contains(uint64_t(objectStoreId))) {
|
|
objIndex++;
|
|
} else {
|
|
mObjectStores.RemoveElementAt(objIndex);
|
|
objCount--;
|
|
}
|
|
}
|
|
|
|
if (!mDeletedObjectStores.IsEmpty()) {
|
|
for (uint32_t objCount = mDeletedObjectStores.Length(), objIndex = 0;
|
|
objIndex < objCount;
|
|
objIndex++) {
|
|
const int64_t objectStoreId = mDeletedObjectStores[objIndex]->Id();
|
|
MOZ_ASSERT(objectStoreId);
|
|
|
|
if (validIds.Contains(uint64_t(objectStoreId))) {
|
|
nsRefPtr<IDBObjectStore>* objectStore =
|
|
mObjectStores.AppendElement();
|
|
objectStore->swap(mDeletedObjectStores[objIndex]);
|
|
}
|
|
}
|
|
mDeletedObjectStores.Clear();
|
|
}
|
|
}
|
|
}
|
|
|
|
// Fire the abort event if there are no outstanding requests. Otherwise the
|
|
// abort event will be fired when all outstanding requests finish.
|
|
if (needToSendAbort) {
|
|
SendAbort(aAbortCode);
|
|
}
|
|
|
|
if (isVersionChange) {
|
|
mDatabase->Close();
|
|
}
|
|
}
|
|
|
|
void
|
|
IDBTransaction::Abort(IDBRequest* aRequest)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(aRequest);
|
|
|
|
ErrorResult rv;
|
|
nsRefPtr<DOMError> error = aRequest->GetError(rv);
|
|
|
|
AbortInternal(aRequest->GetErrorCode(), error.forget());
|
|
}
|
|
|
|
void
|
|
IDBTransaction::Abort(nsresult aErrorCode)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
|
|
nsRefPtr<DOMError> error = new DOMError(GetOwner(), aErrorCode);
|
|
AbortInternal(aErrorCode, error.forget());
|
|
}
|
|
|
|
void
|
|
IDBTransaction::Abort(ErrorResult& aRv)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
|
|
if (IsFinished()) {
|
|
aRv = NS_ERROR_DOM_INDEXEDDB_NOT_ALLOWED_ERR;
|
|
return;
|
|
}
|
|
|
|
AbortInternal(NS_ERROR_DOM_INDEXEDDB_ABORT_ERR, nullptr);
|
|
|
|
MOZ_ASSERT(!mAbortedByScript);
|
|
mAbortedByScript = true;
|
|
}
|
|
|
|
void
|
|
IDBTransaction::FireCompleteOrAbortEvents(nsresult aResult)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(!mFiredCompleteOrAbort);
|
|
|
|
IDB_PROFILER_MARK("IndexedDB Transaction %llu: Complete (rv = %lu)",
|
|
"IDBTransaction[%llu] MT Complete",
|
|
mTransaction->GetSerialNumber(), mAbortCode);
|
|
|
|
mReadyState = DONE;
|
|
|
|
#ifdef DEBUG
|
|
mFiredCompleteOrAbort = true;
|
|
#endif
|
|
|
|
nsCOMPtr<nsIDOMEvent> event;
|
|
if (NS_SUCCEEDED(aResult)) {
|
|
event = CreateGenericEvent(this,
|
|
nsDependentString(kCompleteEventType),
|
|
eDoesNotBubble,
|
|
eNotCancelable);
|
|
} else {
|
|
if (!mError && !mAbortedByScript) {
|
|
mError = new DOMError(GetOwner(), aResult);
|
|
}
|
|
|
|
event = CreateGenericEvent(this,
|
|
nsDependentString(kAbortEventType),
|
|
eDoesBubble,
|
|
eNotCancelable);
|
|
}
|
|
|
|
if (NS_WARN_IF(!event)) {
|
|
return;
|
|
}
|
|
|
|
bool dummy;
|
|
if (NS_FAILED(DispatchEvent(event, &dummy))) {
|
|
NS_WARNING("DispatchEvent failed!");
|
|
}
|
|
|
|
mDatabase->DelayedMaybeExpireFileActors();
|
|
}
|
|
|
|
int64_t
|
|
IDBTransaction::NextObjectStoreId()
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(VERSION_CHANGE == mMode);
|
|
|
|
return mNextObjectStoreId++;
|
|
}
|
|
|
|
int64_t
|
|
IDBTransaction::NextIndexId()
|
|
{
|
|
AssertIsOnOwningThread();
|
|
MOZ_ASSERT(VERSION_CHANGE == mMode);
|
|
|
|
return mNextIndexId++;
|
|
}
|
|
|
|
nsPIDOMWindow*
|
|
IDBTransaction::GetParentObject() const
|
|
{
|
|
AssertIsOnOwningThread();
|
|
|
|
return mDatabase->GetParentObject();
|
|
}
|
|
|
|
IDBTransactionMode
|
|
IDBTransaction::GetMode(ErrorResult& aRv) const
|
|
{
|
|
AssertIsOnOwningThread();
|
|
|
|
switch (mMode) {
|
|
case READ_ONLY:
|
|
return IDBTransactionMode::Readonly;
|
|
|
|
case READ_WRITE:
|
|
return IDBTransactionMode::Readwrite;
|
|
|
|
case VERSION_CHANGE:
|
|
return IDBTransactionMode::Versionchange;
|
|
|
|
case MODE_INVALID:
|
|
default:
|
|
MOZ_CRASH("Bad mode!");
|
|
}
|
|
}
|
|
|
|
DOMError*
|
|
IDBTransaction::GetError() const
|
|
{
|
|
AssertIsOnOwningThread();
|
|
|
|
return mError;
|
|
}
|
|
|
|
already_AddRefed<DOMStringList>
|
|
IDBTransaction::ObjectStoreNames()
|
|
{
|
|
AssertIsOnOwningThread();
|
|
|
|
if (mMode == IDBTransaction::VERSION_CHANGE) {
|
|
return mDatabase->ObjectStoreNames();
|
|
}
|
|
|
|
nsRefPtr<DOMStringList> list = new DOMStringList();
|
|
list->StringArray() = mObjectStoreNames;
|
|
return list.forget();
|
|
}
|
|
|
|
already_AddRefed<IDBObjectStore>
|
|
IDBTransaction::ObjectStore(const nsAString& aName, ErrorResult& aRv)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
|
|
if (IsFinished()) {
|
|
aRv.Throw(NS_ERROR_DOM_INVALID_STATE_ERR);
|
|
return nullptr;
|
|
}
|
|
|
|
const ObjectStoreSpec* spec = nullptr;
|
|
|
|
if (IDBTransaction::VERSION_CHANGE == mMode ||
|
|
mObjectStoreNames.Contains(aName)) {
|
|
const nsTArray<ObjectStoreSpec>& objectStores =
|
|
mDatabase->Spec()->objectStores();
|
|
|
|
for (uint32_t count = objectStores.Length(), index = 0;
|
|
index < count;
|
|
index++) {
|
|
const ObjectStoreSpec& objectStore = objectStores[index];
|
|
if (objectStore.metadata().name() == aName) {
|
|
spec = &objectStore;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!spec) {
|
|
aRv.Throw(NS_ERROR_DOM_INDEXEDDB_NOT_FOUND_ERR);
|
|
return nullptr;
|
|
}
|
|
|
|
const int64_t desiredId = spec->metadata().id();
|
|
|
|
nsRefPtr<IDBObjectStore> objectStore;
|
|
|
|
for (uint32_t count = mObjectStores.Length(), index = 0;
|
|
index < count;
|
|
index++) {
|
|
nsRefPtr<IDBObjectStore>& existingObjectStore = mObjectStores[index];
|
|
|
|
if (existingObjectStore->Id() == desiredId) {
|
|
objectStore = existingObjectStore;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!objectStore) {
|
|
objectStore = IDBObjectStore::Create(this, *spec);
|
|
MOZ_ASSERT(objectStore);
|
|
|
|
mObjectStores.AppendElement(objectStore);
|
|
}
|
|
|
|
return objectStore.forget();
|
|
}
|
|
|
|
NS_IMPL_ADDREF_INHERITED(IDBTransaction, IDBWrapperCache)
|
|
NS_IMPL_RELEASE_INHERITED(IDBTransaction, IDBWrapperCache)
|
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION_INHERITED(IDBTransaction)
|
|
NS_INTERFACE_MAP_ENTRY(nsIRunnable)
|
|
NS_INTERFACE_MAP_END_INHERITING(IDBWrapperCache)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CLASS(IDBTransaction)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(IDBTransaction,
|
|
IDBWrapperCache)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mDatabase)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mError)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mObjectStores)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mDeletedObjectStores)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(IDBTransaction, IDBWrapperCache)
|
|
// Don't unlink mDatabase!
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK(mError)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK(mObjectStores)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK(mDeletedObjectStores)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
|
|
|
|
JSObject*
|
|
IDBTransaction::WrapObject(JSContext* aCx)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
|
|
return IDBTransactionBinding::Wrap(aCx, this);
|
|
}
|
|
|
|
nsresult
|
|
IDBTransaction::PreHandleEvent(EventChainPreVisitor& aVisitor)
|
|
{
|
|
AssertIsOnOwningThread();
|
|
|
|
aVisitor.mCanHandle = true;
|
|
aVisitor.mParentTarget = mDatabase;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
IDBTransaction::Run()
|
|
{
|
|
AssertIsOnOwningThread();
|
|
|
|
// We're back at the event loop, no longer newborn.
|
|
mCreating = false;
|
|
|
|
// Maybe commit if there were no requests generated.
|
|
if (mReadyState == IDBTransaction::INITIAL) {
|
|
mReadyState = DONE;
|
|
|
|
SendCommit();
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
} // namespace indexedDB
|
|
} // namespace dom
|
|
} // namespace mozilla
|