blob: 32405c6ec72091abeac2b7c62444e13c0fa9bdf5 [file] [log] [blame]
/*
* Copyright (C) 2010 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "IDBTransactionBackendImpl.h"
#if ENABLE(INDEXED_DATABASE)
#include "IDBBackingStore.h"
#include "IDBCursorBackendImpl.h"
#include "IDBDatabaseBackendImpl.h"
#include "IDBDatabaseException.h"
#include "IDBObjectStoreBackendImpl.h"
#include "IDBTracing.h"
#include "IDBTransactionCoordinator.h"
#include "ScriptExecutionContext.h"
namespace WebCore {
PassRefPtr<IDBTransactionBackendImpl> IDBTransactionBackendImpl::create(const Vector<int64_t>& objectStoreIds, unsigned short mode, IDBDatabaseBackendImpl* database)
{
return adoptRef(new IDBTransactionBackendImpl(objectStoreIds, mode, database));
}
IDBTransactionBackendImpl::IDBTransactionBackendImpl(const Vector<int64_t>& objectStoreIds, unsigned short mode, IDBDatabaseBackendImpl* database)
: m_objectStoreIds(objectStoreIds)
, m_mode(mode)
, m_state(Unused)
, m_database(database)
, m_transaction(database->backingStore().get())
, m_taskTimer(this, &IDBTransactionBackendImpl::taskTimerFired)
, m_taskEventTimer(this, &IDBTransactionBackendImpl::taskEventTimerFired)
, m_pendingPreemptiveEvents(0)
, m_pendingEvents(0)
{
m_database->transactionCoordinator()->didCreateTransaction(this);
}
IDBTransactionBackendImpl::~IDBTransactionBackendImpl()
{
// It shouldn't be possible for this object to get deleted until it's either complete or aborted.
ASSERT(m_state == Finished);
}
PassRefPtr<IDBObjectStoreBackendInterface> IDBTransactionBackendImpl::objectStore(int64_t id, ExceptionCode& ec)
{
if (m_state == Finished) {
ec = IDBDatabaseException::IDB_INVALID_STATE_ERR;
return 0;
}
RefPtr<IDBObjectStoreBackendImpl> objectStore = m_database->objectStore(id);
ASSERT(objectStore);
return objectStore.release();
}
bool IDBTransactionBackendImpl::scheduleTask(TaskType type, PassOwnPtr<ScriptExecutionContext::Task> task, PassOwnPtr<ScriptExecutionContext::Task> abortTask)
{
if (m_state == Finished)
return false;
if (type == NormalTask)
m_taskQueue.append(task);
else
m_preemptiveTaskQueue.append(task);
if (abortTask)
m_abortTaskQueue.prepend(abortTask);
if (m_state == Unused)
start();
return true;
}
void IDBTransactionBackendImpl::abort()
{
abort(IDBDatabaseError::create(IDBDatabaseException::UNKNOWN_ERR, "Internal error."));
}
void IDBTransactionBackendImpl::abort(PassRefPtr<IDBDatabaseError> error)
{
IDB_TRACE("IDBTransactionBackendImpl::abort");
if (m_state == Finished)
return;
bool wasRunning = m_state == Running;
// The last reference to this object may be released while performing the
// abort steps below. We therefore take a self reference to keep ourselves
// alive while executing this method.
RefPtr<IDBTransactionBackendImpl> self(this);
m_state = Finished;
m_taskTimer.stop();
m_taskEventTimer.stop();
if (wasRunning)
m_transaction.rollback();
// Run the abort tasks, if any.
while (!m_abortTaskQueue.isEmpty()) {
OwnPtr<ScriptExecutionContext::Task> task(m_abortTaskQueue.first().release());
m_abortTaskQueue.removeFirst();
task->performTask(0);
}
// Backing store resources (held via cursors) must be released before script callbacks
// are fired, as the script callbacks may release references and allow the backing store
// itself to be released, and order is critical.
closeOpenCursors();
m_transaction = 0;
// Transactions must also be marked as completed before the front-end is notified, as
// the transaction completion unblocks operations like closing connections.
m_database->transactionCoordinator()->didFinishTransaction(this);
ASSERT(!m_database->transactionCoordinator()->isActive(this));
m_database->transactionFinished(this);
if (m_callbacks)
m_callbacks->onAbort(error);
m_database->transactionFinishedAndAbortFired(this);
m_database = 0;
}
bool IDBTransactionBackendImpl::isTaskQueueEmpty() const
{
return m_preemptiveTaskQueue.isEmpty() && m_taskQueue.isEmpty();
}
void IDBTransactionBackendImpl::registerOpenCursor(IDBCursorBackendImpl* cursor)
{
m_openCursors.add(cursor);
}
void IDBTransactionBackendImpl::unregisterOpenCursor(IDBCursorBackendImpl* cursor)
{
m_openCursors.remove(cursor);
}
void IDBTransactionBackendImpl::addPendingEvents(int n)
{
m_pendingEvents += n;
ASSERT(m_pendingEvents >= 0);
}
void IDBTransactionBackendImpl::didCompleteTaskEvents()
{
if (m_state == Finished)
return;
ASSERT(m_state == Running);
ASSERT(m_pendingEvents);
m_pendingEvents--;
if (!m_taskEventTimer.isActive())
m_taskEventTimer.startOneShot(0);
}
void IDBTransactionBackendImpl::run()
{
ASSERT(m_state == StartPending || m_state == Running);
ASSERT(!m_taskTimer.isActive());
m_taskTimer.startOneShot(0);
}
void IDBTransactionBackendImpl::start()
{
ASSERT(m_state == Unused);
m_state = StartPending;
m_database->transactionCoordinator()->didStartTransaction(this);
m_database->transactionStarted(this);
}
void IDBTransactionBackendImpl::commit()
{
IDB_TRACE("IDBTransactionBackendImpl::commit");
// The last reference to this object may be released while performing the
// commit steps below. We therefore take a self reference to keep ourselves
// alive while executing this method.
RefPtr<IDBTransactionBackendImpl> self(this);
ASSERT(m_state == Unused || m_state == Running);
ASSERT(isTaskQueueEmpty());
bool unused = m_state == Unused;
m_state = Finished;
bool committed = unused || m_transaction.commit();
// Backing store resources (held via cursors) must be released before script callbacks
// are fired, as the script callbacks may release references and allow the backing store
// itself to be released, and order is critical.
closeOpenCursors();
m_transaction = 0;
// Transactions must also be marked as completed before the front-end is notified, as
// the transaction completion unblocks operations like closing connections.
if (!unused)
m_database->transactionCoordinator()->didFinishTransaction(this);
m_database->transactionFinished(this);
if (committed) {
m_callbacks->onComplete();
m_database->transactionFinishedAndCompleteFired(this);
} else {
m_callbacks->onAbort(IDBDatabaseError::create(IDBDatabaseException::UNKNOWN_ERR, "Internal error."));
m_database->transactionFinishedAndAbortFired(this);
}
m_database = 0;
}
void IDBTransactionBackendImpl::taskTimerFired(Timer<IDBTransactionBackendImpl>*)
{
IDB_TRACE("IDBTransactionBackendImpl::taskTimerFired");
ASSERT(!isTaskQueueEmpty());
if (m_state == StartPending) {
m_transaction.begin();
m_state = Running;
}
// Just process a single event here, in case the event itself
// changes which queue should be processed next.
TaskQueue& taskQueue = m_pendingPreemptiveEvents ? m_preemptiveTaskQueue : m_taskQueue;
if (!taskQueue.isEmpty() && m_state != Finished) {
ASSERT(m_state == Running);
OwnPtr<ScriptExecutionContext::Task> task(taskQueue.first().release());
taskQueue.removeFirst();
m_pendingEvents++;
task->performTask(0);
}
}
void IDBTransactionBackendImpl::taskEventTimerFired(Timer<IDBTransactionBackendImpl>*)
{
IDB_TRACE("IDBTransactionBackendImpl::taskEventTimerFired");
ASSERT(m_state == Running);
if (!m_pendingEvents && isTaskQueueEmpty()) {
// The last task event has completed and the task
// queue is empty. Commit the transaction.
commit();
return;
}
// We are still waiting for other events to complete. However,
// the task queue is non-empty and the timer is inactive.
// We can therfore schedule the timer again.
if (!isTaskQueueEmpty() && !m_taskTimer.isActive())
m_taskTimer.startOneShot(0);
}
void IDBTransactionBackendImpl::closeOpenCursors()
{
for (HashSet<IDBCursorBackendImpl*>::iterator i = m_openCursors.begin(); i != m_openCursors.end(); ++i)
(*i)->close();
m_openCursors.clear();
}
};
#endif // ENABLE(INDEXED_DATABASE)