| // Copyright 2013 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef CONTENT_CHILD_INDEXED_DB_INDEXED_DB_DISPATCHER_H_ |
| #define CONTENT_CHILD_INDEXED_DB_INDEXED_DB_DISPATCHER_H_ |
| |
| #include <stddef.h> |
| #include <stdint.h> |
| |
| #include <map> |
| #include <string> |
| #include <vector> |
| |
| #include "base/gtest_prod_util.h" |
| #include "base/id_map.h" |
| #include "base/macros.h" |
| #include "base/memory/ref_counted.h" |
| #include "base/strings/nullable_string16.h" |
| #include "content/child/indexed_db/indexed_db_callbacks_impl.h" |
| #include "content/child/indexed_db/indexed_db_database_callbacks_impl.h" |
| #include "content/common/content_export.h" |
| #include "content/common/indexed_db/indexed_db_constants.h" |
| #include "content/public/child/worker_thread.h" |
| #include "ipc/ipc_sync_message_filter.h" |
| #include "third_party/WebKit/public/platform/WebBlobInfo.h" |
| #include "third_party/WebKit/public/platform/modules/indexeddb/WebIDBCallbacks.h" |
| #include "third_party/WebKit/public/platform/modules/indexeddb/WebIDBObserver.h" |
| #include "third_party/WebKit/public/platform/modules/indexeddb/WebIDBTypes.h" |
| #include "url/origin.h" |
| |
| struct IndexedDBMsg_CallbacksSuccessCursorContinue_Params; |
| struct IndexedDBMsg_CallbacksSuccessCursorPrefetch_Params; |
| struct IndexedDBMsg_CallbacksSuccessIDBCursor_Params; |
| struct IndexedDBMsg_CallbacksSuccessArray_Params; |
| struct IndexedDBMsg_CallbacksSuccessValue_Params; |
| struct IndexedDBMsg_Observation; |
| struct IndexedDBMsg_ObserverChanges; |
| |
| namespace blink { |
| class WebData; |
| struct WebIDBObservation; |
| } |
| |
| namespace content { |
| class IndexedDBKey; |
| class IndexedDBKeyRange; |
| class WebIDBCursorImpl; |
| class WebIDBDatabaseImpl; |
| class ThreadSafeSender; |
| |
| // Handle the indexed db related communication for this context thread - the |
| // main thread and each worker thread have their own copies. |
| class CONTENT_EXPORT IndexedDBDispatcher : public WorkerThread::Observer { |
| public: |
| // Constructor made public to allow RenderThreadImpl to own a copy without |
| // failing a NOTREACHED in ThreadSpecificInstance in tests that instantiate |
| // two copies of RenderThreadImpl on the same thread. Everyone else probably |
| // wants to use ThreadSpecificInstance(). |
| explicit IndexedDBDispatcher(ThreadSafeSender* thread_safe_sender); |
| ~IndexedDBDispatcher() override; |
| |
| // |thread_safe_sender| needs to be passed in because if the call leads to |
| // construction it will be needed. |
| static IndexedDBDispatcher* ThreadSpecificInstance( |
| ThreadSafeSender* thread_safe_sender); |
| |
| // WorkerThread::Observer implementation. |
| void WillStopCurrentWorkerThread() override; |
| |
| static std::vector<blink::WebIDBObservation> ConvertObservations( |
| const std::vector<IndexedDBMsg_Observation>& idb_observation); |
| |
| void OnMessageReceived(const IPC::Message& msg); |
| |
| // This method is virtual so it can be overridden in unit tests. |
| virtual bool Send(IPC::Message* msg); |
| |
| int32_t RegisterObserver(std::unique_ptr<blink::WebIDBObserver> observer); |
| |
| // Removes observers from our local map observers_. |
| void RemoveObservers(const std::vector<int32_t>& observer_ids_to_remove); |
| |
| // This method is virtual so it can be overridden in unit tests. |
| virtual void RequestIDBCursorAdvance(unsigned long count, |
| blink::WebIDBCallbacks* callbacks_ptr, |
| int32_t ipc_cursor_id, |
| int64_t transaction_id); |
| |
| // This method is virtual so it can be overridden in unit tests. |
| virtual void RequestIDBCursorContinue(const IndexedDBKey& key, |
| const IndexedDBKey& primary_key, |
| blink::WebIDBCallbacks* callbacks_ptr, |
| int32_t ipc_cursor_id, |
| int64_t transaction_id); |
| |
| // This method is virtual so it can be overridden in unit tests. |
| virtual void RequestIDBCursorPrefetch(int n, |
| blink::WebIDBCallbacks* callbacks_ptr, |
| int32_t ipc_cursor_id); |
| |
| // This method is virtual so it can be overridden in unit tests. |
| virtual void RequestIDBCursorPrefetchReset(int used_prefetches, |
| int unused_prefetches, |
| int32_t ipc_cursor_id); |
| |
| void RegisterCursor(int32_t ipc_cursor_id, WebIDBCursorImpl* cursor); |
| |
| virtual void CursorDestroyed(int32_t ipc_cursor_id); |
| |
| enum { kAllCursors = -1 }; |
| |
| // Reset cursor prefetch caches for all cursors except exception_cursor_id. |
| void ResetCursorPrefetchCaches(int64_t transaction_id, |
| int32_t ipc_exception_cursor_id); |
| |
| void RegisterMojoOwnedCallbacks( |
| IndexedDBCallbacksImpl::InternalState* callback_state); |
| void UnregisterMojoOwnedCallbacks( |
| IndexedDBCallbacksImpl::InternalState* callback_state); |
| void RegisterMojoOwnedDatabaseCallbacks( |
| blink::WebIDBDatabaseCallbacks* callback_state); |
| void UnregisterMojoOwnedDatabaseCallbacks( |
| blink::WebIDBDatabaseCallbacks* callback_state); |
| |
| private: |
| FRIEND_TEST_ALL_PREFIXES(IndexedDBDispatcherTest, CursorReset); |
| FRIEND_TEST_ALL_PREFIXES(IndexedDBDispatcherTest, CursorTransactionId); |
| |
| static int32_t CurrentWorkerId() { return WorkerThread::GetCurrentId(); } |
| |
| template <typename T> |
| void init_params(T* params, blink::WebIDBCallbacks* callbacks_ptr) { |
| std::unique_ptr<blink::WebIDBCallbacks> callbacks(callbacks_ptr); |
| params->ipc_thread_id = CurrentWorkerId(); |
| params->ipc_callbacks_id = pending_callbacks_.Add(callbacks.release()); |
| } |
| |
| // IDBCallback message handlers. |
| void OnSuccessCursorContinue( |
| const IndexedDBMsg_CallbacksSuccessCursorContinue_Params& p); |
| void OnSuccessCursorPrefetch( |
| const IndexedDBMsg_CallbacksSuccessCursorPrefetch_Params& p); |
| void OnSuccessValue(const IndexedDBMsg_CallbacksSuccessValue_Params& p); |
| void OnSuccessInteger(int32_t ipc_thread_id, |
| int32_t ipc_callbacks_id, |
| int64_t value); |
| void OnError(int32_t ipc_thread_id, |
| int32_t ipc_callbacks_id, |
| int code, |
| const base::string16& message); |
| void OnDatabaseChanges(int32_t ipc_thread_id, |
| const IndexedDBMsg_ObserverChanges&); |
| |
| scoped_refptr<ThreadSafeSender> thread_safe_sender_; |
| |
| // Careful! WebIDBCallbacks wraps non-threadsafe data types. It must be |
| // destroyed and used on the same thread it was created on. |
| IDMap<blink::WebIDBCallbacks, IDMapOwnPointer> pending_callbacks_; |
| IDMap<blink::WebIDBObserver, IDMapOwnPointer> observers_; |
| |
| // Holds pointers to the worker-thread owned state of IndexedDBCallbacksImpl |
| // and IndexedDBDatabaseCallbacksImpl objects to makes sure that it is |
| // destroyed on thread exit if the Mojo pipe is not yet closed. Otherwise the |
| // object will leak because the thread's task runner is no longer executing |
| // tasks. |
| std::unordered_set<IndexedDBCallbacksImpl::InternalState*> |
| mojo_owned_callback_state_; |
| std::unordered_set<blink::WebIDBDatabaseCallbacks*> |
| mojo_owned_database_callback_state_; |
| |
| // Maps the ipc_callback_id from an open cursor request to the request's |
| // transaction_id. Used to assign the transaction_id to the WebIDBCursorImpl |
| // when it is created. |
| std::map<int32_t, int64_t> cursor_transaction_ids_; |
| |
| // Map from cursor id to WebIDBCursorImpl. |
| std::map<int32_t, WebIDBCursorImpl*> cursors_; |
| |
| DISALLOW_COPY_AND_ASSIGN(IndexedDBDispatcher); |
| }; |
| |
| } // namespace content |
| |
| #endif // CONTENT_CHILD_INDEXED_DB_INDEXED_DB_DISPATCHER_H_ |