mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-26 06:11:37 +00:00
572 lines
15 KiB
C++
572 lines
15 KiB
C++
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim: set ts=2 et sw=2 tw=80: */
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (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.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is Indexed Database.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* The Mozilla Foundation.
|
|
* Portions created by the Initial Developer are Copyright (C) 2010
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Ben Turner <bent.mozilla@gmail.com>
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
#include "AsyncConnectionHelper.h"
|
|
|
|
#include "mozilla/storage.h"
|
|
#include "nsComponentManagerUtils.h"
|
|
#include "nsContentUtils.h"
|
|
#include "nsProxyRelease.h"
|
|
#include "nsThreadUtils.h"
|
|
|
|
#include "IDBEvents.h"
|
|
#include "IDBTransaction.h"
|
|
#include "IndexedDatabaseManager.h"
|
|
#include "TransactionThreadPool.h"
|
|
|
|
using mozilla::TimeStamp;
|
|
using mozilla::TimeDuration;
|
|
|
|
USING_INDEXEDDB_NAMESPACE
|
|
|
|
namespace {
|
|
|
|
IDBTransaction* gCurrentTransaction = nsnull;
|
|
|
|
const PRUint32 kProgressHandlerGranularity = 1000;
|
|
const PRUint32 kDefaultTimeoutMS = 30000;
|
|
|
|
NS_STACK_CLASS
|
|
class TransactionPoolEventTarget : public nsIEventTarget
|
|
{
|
|
public:
|
|
NS_DECL_ISUPPORTS
|
|
NS_DECL_NSIEVENTTARGET
|
|
|
|
TransactionPoolEventTarget(IDBTransaction* aTransaction)
|
|
: mTransaction(aTransaction)
|
|
{ }
|
|
|
|
private:
|
|
IDBTransaction* mTransaction;
|
|
};
|
|
|
|
// This inline is just so that we always clear aBuffers appropriately even if
|
|
// something fails.
|
|
inline
|
|
nsresult
|
|
ConvertCloneBuffersToArrayInternal(
|
|
JSContext* aCx,
|
|
nsTArray<JSAutoStructuredCloneBuffer>& aBuffers,
|
|
jsval* aResult)
|
|
{
|
|
JSObject* array = JS_NewArrayObject(aCx, 0, nsnull);
|
|
if (!array) {
|
|
NS_WARNING("Failed to make array!");
|
|
return NS_ERROR_DOM_INDEXEDDB_UNKNOWN_ERR;
|
|
}
|
|
|
|
if (!aBuffers.IsEmpty()) {
|
|
if (!JS_SetArrayLength(aCx, array, jsuint(aBuffers.Length()))) {
|
|
NS_WARNING("Failed to set array length!");
|
|
return NS_ERROR_DOM_INDEXEDDB_UNKNOWN_ERR;
|
|
}
|
|
|
|
for (uint32 index = 0, count = aBuffers.Length(); index < count; index++) {
|
|
JSAutoStructuredCloneBuffer& buffer = aBuffers[index];
|
|
|
|
jsval val;
|
|
if (!IDBObjectStore::DeserializeValue(aCx, buffer, &val)) {
|
|
NS_WARNING("Failed to decode!");
|
|
return NS_ERROR_DOM_DATA_CLONE_ERR;
|
|
}
|
|
|
|
if (!JS_SetElement(aCx, array, index, &val)) {
|
|
NS_WARNING("Failed to set array element!");
|
|
return NS_ERROR_DOM_INDEXEDDB_UNKNOWN_ERR;
|
|
}
|
|
}
|
|
}
|
|
|
|
*aResult = OBJECT_TO_JSVAL(array);
|
|
return NS_OK;
|
|
}
|
|
|
|
} // anonymous namespace
|
|
|
|
HelperBase::~HelperBase()
|
|
{
|
|
if (!NS_IsMainThread()) {
|
|
IDBRequest* request;
|
|
mRequest.forget(&request);
|
|
|
|
if (request) {
|
|
nsCOMPtr<nsIThread> mainThread;
|
|
NS_GetMainThread(getter_AddRefs(mainThread));
|
|
NS_WARN_IF_FALSE(mainThread, "Couldn't get the main thread!");
|
|
|
|
if (mainThread) {
|
|
NS_ProxyRelease(mainThread, static_cast<nsIDOMEventTarget*>(request));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
nsresult
|
|
HelperBase::WrapNative(JSContext* aCx,
|
|
nsISupports* aNative,
|
|
jsval* aResult)
|
|
{
|
|
NS_ASSERTION(aCx, "Null context!");
|
|
NS_ASSERTION(aNative, "Null pointer!");
|
|
NS_ASSERTION(aResult, "Null pointer!");
|
|
NS_ASSERTION(mRequest, "Null request!");
|
|
|
|
JSObject* global = mRequest->ScriptContext()->GetNativeGlobal();
|
|
NS_ENSURE_TRUE(global, NS_ERROR_DOM_INDEXEDDB_UNKNOWN_ERR);
|
|
|
|
nsresult rv =
|
|
nsContentUtils::WrapNative(aCx, global, aNative, aResult);
|
|
NS_ENSURE_SUCCESS(rv, NS_ERROR_DOM_INDEXEDDB_UNKNOWN_ERR);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
HelperBase::ReleaseMainThreadObjects()
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
|
|
mRequest = nsnull;
|
|
}
|
|
|
|
AsyncConnectionHelper::AsyncConnectionHelper(IDBDatabase* aDatabase,
|
|
IDBRequest* aRequest)
|
|
: HelperBase(aRequest),
|
|
mDatabase(aDatabase),
|
|
mTimeoutDuration(TimeDuration::FromMilliseconds(kDefaultTimeoutMS)),
|
|
mResultCode(NS_OK),
|
|
mDispatched(false)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
}
|
|
|
|
AsyncConnectionHelper::AsyncConnectionHelper(IDBTransaction* aTransaction,
|
|
IDBRequest* aRequest)
|
|
: HelperBase(aRequest),
|
|
mDatabase(aTransaction->mDatabase),
|
|
mTransaction(aTransaction),
|
|
mTimeoutDuration(TimeDuration::FromMilliseconds(kDefaultTimeoutMS)),
|
|
mResultCode(NS_OK),
|
|
mDispatched(false)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
}
|
|
|
|
AsyncConnectionHelper::~AsyncConnectionHelper()
|
|
{
|
|
if (!NS_IsMainThread()) {
|
|
IDBDatabase* database;
|
|
mDatabase.forget(&database);
|
|
|
|
IDBTransaction* transaction;
|
|
mTransaction.forget(&transaction);
|
|
|
|
nsCOMPtr<nsIThread> mainThread;
|
|
NS_GetMainThread(getter_AddRefs(mainThread));
|
|
NS_WARN_IF_FALSE(mainThread, "Couldn't get the main thread!");
|
|
|
|
if (mainThread) {
|
|
if (database) {
|
|
NS_ProxyRelease(mainThread, static_cast<nsIIDBDatabase*>(database));
|
|
}
|
|
if (transaction) {
|
|
NS_ProxyRelease(mainThread,
|
|
static_cast<nsIIDBTransaction*>(transaction));
|
|
}
|
|
}
|
|
}
|
|
|
|
NS_ASSERTION(!mOldProgressHandler, "Should not have anything here!");
|
|
}
|
|
|
|
NS_IMPL_THREADSAFE_ISUPPORTS2(AsyncConnectionHelper, nsIRunnable,
|
|
mozIStorageProgressHandler)
|
|
|
|
NS_IMETHODIMP
|
|
AsyncConnectionHelper::Run()
|
|
{
|
|
if (NS_IsMainThread()) {
|
|
if (mTransaction &&
|
|
mTransaction->IsAborted() &&
|
|
NS_SUCCEEDED(mResultCode)) {
|
|
// Don't fire success events if the transaction has since been aborted.
|
|
// Instead convert to an error event.
|
|
mResultCode = NS_ERROR_DOM_INDEXEDDB_ABORT_ERR;
|
|
}
|
|
|
|
IDBTransaction* oldTransaction = gCurrentTransaction;
|
|
gCurrentTransaction = mTransaction;
|
|
|
|
if (mRequest) {
|
|
nsresult rv = mRequest->NotifyHelperCompleted(this);
|
|
if (NS_SUCCEEDED(mResultCode) && NS_FAILED(rv)) {
|
|
mResultCode = rv;
|
|
}
|
|
}
|
|
|
|
// Call OnError if the database had an error or if the OnSuccess handler
|
|
// has an error.
|
|
if (NS_FAILED(mResultCode) ||
|
|
NS_FAILED((mResultCode = OnSuccess()))) {
|
|
OnError();
|
|
}
|
|
|
|
NS_ASSERTION(gCurrentTransaction == mTransaction, "Should be unchanged!");
|
|
gCurrentTransaction = oldTransaction;
|
|
|
|
if (mDispatched && mTransaction) {
|
|
mTransaction->OnRequestFinished();
|
|
}
|
|
|
|
ReleaseMainThreadObjects();
|
|
|
|
NS_ASSERTION(!(mDatabase || mTransaction || mRequest), "Subclass didn't "
|
|
"call AsyncConnectionHelper::ReleaseMainThreadObjects!");
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult rv = NS_OK;
|
|
nsCOMPtr<mozIStorageConnection> connection;
|
|
|
|
if (mTransaction) {
|
|
rv = mTransaction->GetOrCreateConnection(getter_AddRefs(connection));
|
|
if (NS_SUCCEEDED(rv)) {
|
|
NS_ASSERTION(connection, "This should never be null!");
|
|
}
|
|
}
|
|
|
|
if (connection) {
|
|
rv = connection->SetProgressHandler(kProgressHandlerGranularity, this,
|
|
getter_AddRefs(mOldProgressHandler));
|
|
NS_WARN_IF_FALSE(NS_SUCCEEDED(rv), "SetProgressHandler failed!");
|
|
if (NS_SUCCEEDED(rv)) {
|
|
mStartTime = TimeStamp::Now();
|
|
}
|
|
}
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
bool hasSavepoint = false;
|
|
if (mDatabase) {
|
|
IndexedDatabaseManager::SetCurrentDatabase(mDatabase);
|
|
|
|
// Make the first savepoint.
|
|
if (mTransaction) {
|
|
if (!(hasSavepoint = mTransaction->StartSavepoint())) {
|
|
NS_WARNING("Failed to make savepoint!");
|
|
}
|
|
}
|
|
}
|
|
|
|
mResultCode = DoDatabaseWork(connection);
|
|
|
|
if (mDatabase) {
|
|
IndexedDatabaseManager::SetCurrentDatabase(nsnull);
|
|
|
|
// Release or roll back the savepoint depending on the error code.
|
|
if (hasSavepoint) {
|
|
NS_ASSERTION(mTransaction, "Huh?!");
|
|
if (NS_SUCCEEDED(mResultCode)) {
|
|
mTransaction->ReleaseSavepoint();
|
|
}
|
|
else {
|
|
mTransaction->RollbackSavepoint();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
// NS_ERROR_NOT_AVAILABLE is our special code for "database is invalidated"
|
|
// and we should fail with RECOVERABLE_ERR.
|
|
if (rv == NS_ERROR_NOT_AVAILABLE) {
|
|
mResultCode = NS_ERROR_DOM_INDEXEDDB_RECOVERABLE_ERR;
|
|
}
|
|
else {
|
|
mResultCode = NS_ERROR_DOM_INDEXEDDB_UNKNOWN_ERR;
|
|
}
|
|
}
|
|
|
|
if (!mStartTime.IsNull()) {
|
|
nsCOMPtr<mozIStorageProgressHandler> handler;
|
|
rv = connection->RemoveProgressHandler(getter_AddRefs(handler));
|
|
NS_WARN_IF_FALSE(NS_SUCCEEDED(rv), "RemoveProgressHandler failed!");
|
|
#ifdef DEBUG
|
|
if (NS_SUCCEEDED(rv)) {
|
|
nsCOMPtr<nsISupports> handlerSupports(do_QueryInterface(handler));
|
|
nsCOMPtr<nsISupports> thisSupports = do_QueryObject(this);
|
|
NS_ASSERTION(thisSupports == handlerSupports, "Mismatch!");
|
|
}
|
|
#endif
|
|
mStartTime = TimeStamp();
|
|
}
|
|
|
|
return NS_DispatchToMainThread(this, NS_DISPATCH_NORMAL);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
AsyncConnectionHelper::OnProgress(mozIStorageConnection* aConnection,
|
|
bool* _retval)
|
|
{
|
|
if (mDatabase && mDatabase->IsInvalidated()) {
|
|
// Someone is trying to delete the database file. Exit lightningfast!
|
|
*_retval = true;
|
|
return NS_OK;
|
|
}
|
|
|
|
TimeDuration elapsed = TimeStamp::Now() - mStartTime;
|
|
if (elapsed >= mTimeoutDuration) {
|
|
*_retval = true;
|
|
return NS_OK;
|
|
}
|
|
|
|
if (mOldProgressHandler) {
|
|
return mOldProgressHandler->OnProgress(aConnection, _retval);
|
|
}
|
|
|
|
*_retval = false;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
AsyncConnectionHelper::Dispatch(nsIEventTarget* aDatabaseThread)
|
|
{
|
|
#ifdef DEBUG
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
{
|
|
bool sameThread;
|
|
nsresult rv = aDatabaseThread->IsOnCurrentThread(&sameThread);
|
|
NS_ASSERTION(NS_SUCCEEDED(rv), "IsOnCurrentThread failed!");
|
|
NS_ASSERTION(!sameThread, "Dispatching to main thread not supported!");
|
|
}
|
|
#endif
|
|
|
|
nsresult rv = Init();
|
|
if (NS_FAILED(rv)) {
|
|
return rv;
|
|
}
|
|
|
|
rv = aDatabaseThread->Dispatch(this, NS_DISPATCH_NORMAL);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
if (mTransaction) {
|
|
mTransaction->OnNewRequest();
|
|
}
|
|
|
|
mDispatched = true;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
AsyncConnectionHelper::DispatchToTransactionPool()
|
|
{
|
|
NS_ASSERTION(mTransaction, "Only ok to call this with a transaction!");
|
|
TransactionPoolEventTarget target(mTransaction);
|
|
return Dispatch(&target);
|
|
}
|
|
|
|
// static
|
|
IDBTransaction*
|
|
AsyncConnectionHelper::GetCurrentTransaction()
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
|
|
return gCurrentTransaction;
|
|
}
|
|
|
|
nsresult
|
|
AsyncConnectionHelper::Init()
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
already_AddRefed<nsDOMEvent>
|
|
AsyncConnectionHelper::CreateSuccessEvent()
|
|
{
|
|
return CreateGenericEvent(NS_LITERAL_STRING(SUCCESS_EVT_STR));
|
|
}
|
|
|
|
nsresult
|
|
AsyncConnectionHelper::OnSuccess()
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
NS_ASSERTION(mRequest, "Null request!");
|
|
|
|
nsRefPtr<nsDOMEvent> event = CreateSuccessEvent();
|
|
if (!event) {
|
|
NS_ERROR("Failed to create event!");
|
|
return NS_ERROR_DOM_INDEXEDDB_UNKNOWN_ERR;
|
|
}
|
|
|
|
bool dummy;
|
|
nsresult rv = mRequest->DispatchEvent(event, &dummy);
|
|
NS_ENSURE_SUCCESS(rv, NS_ERROR_DOM_INDEXEDDB_UNKNOWN_ERR);
|
|
|
|
nsEvent* internalEvent = event->GetInternalNSEvent();
|
|
NS_ASSERTION(internalEvent, "This should never be null!");
|
|
|
|
NS_ASSERTION(!mTransaction ||
|
|
mTransaction->IsOpen() ||
|
|
mTransaction->IsAborted(),
|
|
"How else can this be closed?!");
|
|
|
|
if ((internalEvent->flags & NS_EVENT_FLAG_EXCEPTION_THROWN) &&
|
|
mTransaction &&
|
|
mTransaction->IsOpen()) {
|
|
rv = mTransaction->Abort();
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
AsyncConnectionHelper::OnError()
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
NS_ASSERTION(mRequest, "Null request!");
|
|
|
|
// Make an error event and fire it at the target.
|
|
nsRefPtr<nsDOMEvent> event =
|
|
CreateGenericEvent(NS_LITERAL_STRING(ERROR_EVT_STR), true);
|
|
if (!event) {
|
|
NS_ERROR("Failed to create event!");
|
|
return;
|
|
}
|
|
|
|
bool doDefault;
|
|
nsresult rv = mRequest->DispatchEvent(event, &doDefault);
|
|
if (NS_SUCCEEDED(rv)) {
|
|
NS_ASSERTION(!mTransaction ||
|
|
mTransaction->IsOpen() ||
|
|
mTransaction->IsAborted(),
|
|
"How else can this be closed?!");
|
|
|
|
if (doDefault &&
|
|
mTransaction &&
|
|
mTransaction->IsOpen() &&
|
|
NS_FAILED(mTransaction->Abort())) {
|
|
NS_WARNING("Failed to abort transaction!");
|
|
}
|
|
}
|
|
else {
|
|
NS_WARNING("DispatchEvent failed!");
|
|
}
|
|
}
|
|
|
|
nsresult
|
|
AsyncConnectionHelper::GetSuccessResult(JSContext* aCx,
|
|
jsval* aVal)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
|
|
*aVal = JSVAL_VOID;
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
AsyncConnectionHelper::ReleaseMainThreadObjects()
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Wrong thread!");
|
|
|
|
mDatabase = nsnull;
|
|
mTransaction = nsnull;
|
|
|
|
HelperBase::ReleaseMainThreadObjects();
|
|
}
|
|
|
|
// static
|
|
nsresult
|
|
AsyncConnectionHelper::ConvertCloneBuffersToArray(
|
|
JSContext* aCx,
|
|
nsTArray<JSAutoStructuredCloneBuffer>& aBuffers,
|
|
jsval* aResult)
|
|
{
|
|
NS_ASSERTION(aCx, "Null context!");
|
|
NS_ASSERTION(aResult, "Null pointer!");
|
|
|
|
JSAutoRequest ar(aCx);
|
|
|
|
nsresult rv = ConvertCloneBuffersToArrayInternal(aCx, aBuffers, aResult);
|
|
|
|
for (PRUint32 index = 0; index < aBuffers.Length(); index++) {
|
|
aBuffers[index].clear();
|
|
}
|
|
aBuffers.Clear();
|
|
|
|
return rv;
|
|
}
|
|
|
|
NS_IMETHODIMP_(nsrefcnt)
|
|
TransactionPoolEventTarget::AddRef()
|
|
{
|
|
NS_NOTREACHED("Don't call me!");
|
|
return 2;
|
|
}
|
|
|
|
NS_IMETHODIMP_(nsrefcnt)
|
|
TransactionPoolEventTarget::Release()
|
|
{
|
|
NS_NOTREACHED("Don't call me!");
|
|
return 1;
|
|
}
|
|
|
|
NS_IMPL_QUERY_INTERFACE1(TransactionPoolEventTarget, nsIEventTarget)
|
|
|
|
NS_IMETHODIMP
|
|
TransactionPoolEventTarget::Dispatch(nsIRunnable* aRunnable,
|
|
PRUint32 aFlags)
|
|
{
|
|
NS_ASSERTION(aRunnable, "Null pointer!");
|
|
NS_ASSERTION(aFlags == NS_DISPATCH_NORMAL, "Unsupported!");
|
|
|
|
TransactionThreadPool* pool = TransactionThreadPool::GetOrCreate();
|
|
return pool->Dispatch(mTransaction, aRunnable, false, nsnull);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
TransactionPoolEventTarget::IsOnCurrentThread(bool* aResult)
|
|
{
|
|
*aResult = false;
|
|
return NS_OK;
|
|
}
|