1 /* 2 * Copyright (C) 2011 Google Inc. All rights reserved. 3 * 4 * Redistribution and use in source and binary forms, with or without 5 * modification, are permitted provided that the following conditions are 6 * met: 7 * 8 * * Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * * Redistributions in binary form must reproduce the above 11 * copyright notice, this list of conditions and the following disclaimer 12 * in the documentation and/or other materials provided with the 13 * distribution. 14 * * Neither the name of Google Inc. nor the names of its 15 * contributors may be used to endorse or promote products derived from 16 * this software without specific prior written permission. 17 * 18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 19 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 20 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 21 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 22 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 23 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 24 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 25 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 26 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 28 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 29 */ 30 31 #include "config.h" 32 #include "modules/webdatabase/DatabaseTracker.h" 33 34 #include "core/dom/ExecutionContext.h" 35 #include "core/dom/ExecutionContextTask.h" 36 #include "modules/webdatabase/DatabaseBackendBase.h" 37 #include "modules/webdatabase/DatabaseClient.h" 38 #include "modules/webdatabase/DatabaseContext.h" 39 #include "modules/webdatabase/QuotaTracker.h" 40 #include "modules/webdatabase/sqlite/SQLiteFileSystem.h" 41 #include "platform/weborigin/DatabaseIdentifier.h" 42 #include "platform/weborigin/SecurityOrigin.h" 43 #include "platform/weborigin/SecurityOriginHash.h" 44 #include "public/platform/Platform.h" 45 #include "public/platform/WebDatabaseObserver.h" 46 #include "wtf/Assertions.h" 47 #include "wtf/StdLibExtras.h" 48 49 namespace WebCore { 50 51 static void databaseClosed(DatabaseBackendBase* database) 52 { 53 if (blink::Platform::current()->databaseObserver()) { 54 blink::Platform::current()->databaseObserver()->databaseClosed( 55 createDatabaseIdentifierFromSecurityOrigin(database->securityOrigin()), 56 database->stringIdentifier()); 57 } 58 } 59 60 DatabaseTracker& DatabaseTracker::tracker() 61 { 62 AtomicallyInitializedStatic(DatabaseTracker&, tracker = *new DatabaseTracker()); 63 return tracker; 64 } 65 66 DatabaseTracker::DatabaseTracker() 67 { 68 SQLiteFileSystem::registerSQLiteVFS(); 69 } 70 71 bool DatabaseTracker::canEstablishDatabase(DatabaseContext* databaseContext, const String& name, const String& displayName, unsigned long estimatedSize, DatabaseError& error) 72 { 73 ExecutionContext* executionContext = databaseContext->executionContext(); 74 bool success = DatabaseClient::from(executionContext)->allowDatabase(executionContext, name, displayName, estimatedSize); 75 if (!success) 76 error = DatabaseError::GenericSecurityError; 77 return success; 78 } 79 80 String DatabaseTracker::fullPathForDatabase(SecurityOrigin* origin, const String& name, bool) 81 { 82 return createDatabaseIdentifierFromSecurityOrigin(origin) + "/" + name + "#"; 83 } 84 85 void DatabaseTracker::addOpenDatabase(DatabaseBackendBase* database) 86 { 87 MutexLocker openDatabaseMapLock(m_openDatabaseMapGuard); 88 if (!m_openDatabaseMap) 89 m_openDatabaseMap = adoptPtr(new DatabaseOriginMap); 90 91 String originIdentifier = createDatabaseIdentifierFromSecurityOrigin(database->securityOrigin()); 92 DatabaseNameMap* nameMap = m_openDatabaseMap->get(originIdentifier); 93 if (!nameMap) { 94 nameMap = new DatabaseNameMap(); 95 m_openDatabaseMap->set(originIdentifier, nameMap); 96 } 97 98 String name(database->stringIdentifier()); 99 DatabaseSet* databaseSet = nameMap->get(name); 100 if (!databaseSet) { 101 databaseSet = new DatabaseSet(); 102 nameMap->set(name, databaseSet); 103 } 104 105 databaseSet->add(database); 106 } 107 108 class NotifyDatabaseObserverOnCloseTask : public ExecutionContextTask { 109 public: 110 static PassOwnPtr<NotifyDatabaseObserverOnCloseTask> create(PassRefPtr<DatabaseBackendBase> database) 111 { 112 return adoptPtr(new NotifyDatabaseObserverOnCloseTask(database)); 113 } 114 115 virtual void performTask(ExecutionContext* context) 116 { 117 databaseClosed(m_database.get()); 118 } 119 120 virtual bool isCleanupTask() const 121 { 122 return true; 123 } 124 125 private: 126 NotifyDatabaseObserverOnCloseTask(PassRefPtr<DatabaseBackendBase> database) 127 : m_database(database) 128 { 129 } 130 131 RefPtr<DatabaseBackendBase> m_database; 132 }; 133 134 void DatabaseTracker::removeOpenDatabase(DatabaseBackendBase* database) 135 { 136 String originIdentifier = createDatabaseIdentifierFromSecurityOrigin(database->securityOrigin()); 137 MutexLocker openDatabaseMapLock(m_openDatabaseMapGuard); 138 ASSERT(m_openDatabaseMap); 139 DatabaseNameMap* nameMap = m_openDatabaseMap->get(originIdentifier); 140 if (!nameMap) 141 return; 142 143 String name(database->stringIdentifier()); 144 DatabaseSet* databaseSet = nameMap->get(name); 145 if (!databaseSet) 146 return; 147 148 DatabaseSet::iterator found = databaseSet->find(database); 149 if (found == databaseSet->end()) 150 return; 151 152 databaseSet->remove(found); 153 if (databaseSet->isEmpty()) { 154 nameMap->remove(name); 155 delete databaseSet; 156 if (nameMap->isEmpty()) { 157 m_openDatabaseMap->remove(originIdentifier); 158 delete nameMap; 159 } 160 } 161 162 ExecutionContext* executionContext = database->databaseContext()->executionContext(); 163 if (!executionContext->isContextThread()) 164 executionContext->postTask(NotifyDatabaseObserverOnCloseTask::create(database)); 165 else 166 databaseClosed(database); 167 } 168 169 void DatabaseTracker::prepareToOpenDatabase(DatabaseBackendBase* database) 170 { 171 ASSERT(database->databaseContext()->executionContext()->isContextThread()); 172 if (blink::Platform::current()->databaseObserver()) { 173 blink::Platform::current()->databaseObserver()->databaseOpened( 174 createDatabaseIdentifierFromSecurityOrigin(database->securityOrigin()), 175 database->stringIdentifier(), 176 database->displayName(), 177 database->estimatedSize()); 178 } 179 } 180 181 void DatabaseTracker::failedToOpenDatabase(DatabaseBackendBase* database) 182 { 183 ExecutionContext* executionContext = database->databaseContext()->executionContext(); 184 if (!executionContext->isContextThread()) 185 executionContext->postTask(NotifyDatabaseObserverOnCloseTask::create(database)); 186 else 187 databaseClosed(database); 188 } 189 190 unsigned long long DatabaseTracker::getMaxSizeForDatabase(const DatabaseBackendBase* database) 191 { 192 unsigned long long spaceAvailable = 0; 193 unsigned long long databaseSize = 0; 194 QuotaTracker::instance().getDatabaseSizeAndSpaceAvailableToOrigin( 195 createDatabaseIdentifierFromSecurityOrigin(database->securityOrigin()), 196 database->stringIdentifier(), &databaseSize, &spaceAvailable); 197 return databaseSize + spaceAvailable; 198 } 199 200 void DatabaseTracker::interruptAllDatabasesForContext(const DatabaseContext* context) 201 { 202 MutexLocker openDatabaseMapLock(m_openDatabaseMapGuard); 203 204 if (!m_openDatabaseMap) 205 return; 206 207 DatabaseNameMap* nameMap = m_openDatabaseMap->get(createDatabaseIdentifierFromSecurityOrigin(context->securityOrigin())); 208 if (!nameMap) 209 return; 210 211 DatabaseNameMap::const_iterator dbNameMapEndIt = nameMap->end(); 212 for (DatabaseNameMap::const_iterator dbNameMapIt = nameMap->begin(); dbNameMapIt != dbNameMapEndIt; ++dbNameMapIt) { 213 DatabaseSet* databaseSet = dbNameMapIt->value; 214 DatabaseSet::const_iterator end = databaseSet->end(); 215 for (DatabaseSet::const_iterator it = databaseSet->begin(); it != end; ++it) { 216 if ((*it)->databaseContext() == context) 217 (*it)->interrupt(); 218 } 219 } 220 } 221 222 class DatabaseTracker::CloseOneDatabaseImmediatelyTask : public ExecutionContextTask { 223 public: 224 static PassOwnPtr<CloseOneDatabaseImmediatelyTask> create(const String& originIdentifier, const String& name, DatabaseBackendBase* database) 225 { 226 return adoptPtr(new CloseOneDatabaseImmediatelyTask(originIdentifier, name, database)); 227 } 228 229 virtual void performTask(ExecutionContext* context) 230 { 231 DatabaseTracker::tracker().closeOneDatabaseImmediately(m_originIdentifier, m_name, m_database); 232 } 233 234 private: 235 CloseOneDatabaseImmediatelyTask(const String& originIdentifier, const String& name, DatabaseBackendBase* database) 236 : m_originIdentifier(originIdentifier.isolatedCopy()) 237 , m_name(name.isolatedCopy()) 238 , m_database(database) 239 { 240 } 241 242 String m_originIdentifier; 243 String m_name; 244 DatabaseBackendBase* m_database; // Intentionally a raw pointer. 245 }; 246 247 void DatabaseTracker::closeDatabasesImmediately(const String& originIdentifier, const String& name) 248 { 249 MutexLocker openDatabaseMapLock(m_openDatabaseMapGuard); 250 if (!m_openDatabaseMap) 251 return; 252 253 DatabaseNameMap* nameMap = m_openDatabaseMap->get(originIdentifier); 254 if (!nameMap) 255 return; 256 257 DatabaseSet* databaseSet = nameMap->get(name); 258 if (!databaseSet) 259 return; 260 261 // We have to call closeImmediately() on the context thread and we cannot safely add a reference to 262 // the database in our collection when not on the context thread (which is always the case given 263 // current usage). 264 for (DatabaseSet::iterator it = databaseSet->begin(); it != databaseSet->end(); ++it) 265 (*it)->databaseContext()->executionContext()->postTask(CloseOneDatabaseImmediatelyTask::create(originIdentifier, name, *it)); 266 } 267 268 void DatabaseTracker::closeOneDatabaseImmediately(const String& originIdentifier, const String& name, DatabaseBackendBase* database) 269 { 270 // First we have to confirm the 'database' is still in our collection. 271 { 272 MutexLocker openDatabaseMapLock(m_openDatabaseMapGuard); 273 if (!m_openDatabaseMap) 274 return; 275 276 DatabaseNameMap* nameMap = m_openDatabaseMap->get(originIdentifier); 277 if (!nameMap) 278 return; 279 280 DatabaseSet* databaseSet = nameMap->get(name); 281 if (!databaseSet) 282 return; 283 284 DatabaseSet::iterator found = databaseSet->find(database); 285 if (found == databaseSet->end()) 286 return; 287 } 288 289 // And we have to call closeImmediately() without our collection lock being held. 290 database->closeImmediately(); 291 } 292 293 } 294