Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(2405)

Unified Diff: WebCore/platform/network/BlobRegistryImpl.cpp

Issue 11192017: ********** WebCore blob hacking (Closed) Base URL: http://svn.webkit.org/repository/webkit/trunk/Source/
Patch Set: Created 7 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
« no previous file with comments | « WebCore/platform/network/BlobRegistryImpl.h ('k') | WebCore/platform/network/BlobResourceHandle.h » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: WebCore/platform/network/BlobRegistryImpl.cpp
===================================================================
--- WebCore/platform/network/BlobRegistryImpl.cpp (revision 140218)
+++ WebCore/platform/network/BlobRegistryImpl.cpp (working copy)
@@ -40,26 +40,28 @@
#include "ResourceLoader.h"
#include "ResourceRequest.h"
#include "ResourceResponse.h"
+#include "SecurityOrigin.h"
#include <wtf/MainThread.h>
#include <wtf/StdLibExtras.h>
namespace WebCore {
-#if !PLATFORM(CHROMIUM)
BlobRegistry& blobRegistry()
{
- ASSERT(isMainThread());
- DEFINE_STATIC_LOCAL(BlobRegistryImpl, instance, ());
- return instance;
+ // TODO: would be nice to ensure that this is first accessed on the main thread.
+ AtomicallyInitializedStatic(BlobRegistryImpl&, intance = *new BlobRegistryImpl);
+ return intance;
}
static PassRefPtr<ResourceHandle> createResourceHandle(const ResourceRequest& request, ResourceHandleClient* client)
{
+ ASSERT(isMainThread());
return static_cast<BlobRegistryImpl&>(blobRegistry()).createResourceHandle(request, client);
}
static void registerBlobResourceHandleConstructor()
{
+ ASSERT(isMainThread());
static bool didRegister = false;
if (!didRegister) {
ResourceHandle::registerBuiltinConstructor("blob", createResourceHandle);
@@ -67,15 +69,9 @@
}
}
-#else
-
-static void registerBlobResourceHandleConstructor()
-{
-}
-#endif
-
bool BlobRegistryImpl::shouldLoadResource(const ResourceRequest& request) const
{
+ ASSERT(isMainThread());
// If the resource is not fetched using the GET method, bail out.
if (!equalIgnoringCase(request.httpMethod(), "GET"))
return false;
@@ -88,7 +84,7 @@
if (!shouldLoadResource(request))
return 0;
- RefPtr<BlobResourceHandle> handle = BlobResourceHandle::create(m_blobs.get(request.url().string()), request, client);
+ RefPtr<BlobResourceHandle> handle = BlobResourceHandle::create(getBlobDataFromURL(request.url()), request, client);
handle->start();
return handle.release();
}
@@ -98,26 +94,10 @@
if (!shouldLoadResource(request))
return false;
- BlobResourceHandle::loadResourceSynchronously(m_blobs.get(request.url().string()), request, error, response, data);
+ BlobResourceHandle::loadResourceSynchronously(getBlobDataFromURL(request.url()), request, error, response, data);
return true;
}
-void BlobRegistryImpl::appendStorageItems(BlobStorageData* blobStorageData, const BlobDataItemList& items)
-{
- for (BlobDataItemList::const_iterator iter = items.begin(); iter != items.end(); ++iter) {
- if (iter->type == BlobDataItem::Data)
- blobStorageData->m_data.appendData(iter->data, iter->offset, iter->length);
-#if ENABLE(FILE_SYSTEM)
- else if (iter->type == BlobDataItem::URL)
- blobStorageData->m_data.appendURL(iter->url, iter->offset, iter->length, iter->expectedModificationTime);
-#endif
- else {
- ASSERT(iter->type == BlobDataItem::File);
- blobStorageData->m_data.appendFile(iter->path, iter->offset, iter->length, iter->expectedModificationTime);
- }
- }
-}
-
void BlobRegistryImpl::appendStorageItems(BlobStorageData* blobStorageData, const BlobDataItemList& items, long long offset, long long length)
{
ASSERT(length != BlobDataItem::toEndOfFile);
@@ -139,76 +119,156 @@
blobStorageData->m_data.appendData(iter->data, iter->offset + offset, newLength);
#if ENABLE(FILE_SYSTEM)
else if (iter->type == BlobDataItem::URL)
- blobStorageData->m_data.appendURL(iter->url, iter->offset + offset, newLength, iter->expectedModificationTime);
+ blobStorageData->m_data.appendURL(iter->url.copy(), iter->offset + offset, newLength, iter->expectedModificationTime);
#endif
else {
ASSERT(iter->type == BlobDataItem::File);
- blobStorageData->m_data.appendFile(iter->path, iter->offset + offset, newLength, iter->expectedModificationTime);
+ blobStorageData->m_data.appendFile(iter->path.isolatedCopy(), iter->offset + offset, newLength, iter->expectedModificationTime);
}
length -= newLength;
offset = 0;
}
}
-void BlobRegistryImpl::registerBlobURL(const KURL& url, PassOwnPtr<BlobData> blobData)
+void BlobRegistryImpl::registerBlobData(const String& uuid, PassOwnPtr<BlobData> blobData)
{
- ASSERT(isMainThread());
- registerBlobResourceHandleConstructor();
-
- RefPtr<BlobStorageData> blobStorageData = BlobStorageData::create(blobData->contentType(), blobData->contentDisposition());
-
// The blob data is stored in the "canonical" way. That is, it only contains a list of Data and File items.
// 1) The Data item is denoted by the raw data and the range.
// 2) The File item is denoted by the file path, the range and the expected modification time.
// 3) The URL item is denoted by the URL, the range and the expected modification time.
// All the Blob items in the passing blob data are resolved and expanded into a set of Data and File items.
+ RefPtr<BlobStorageData> blobStorageData = BlobStorageData::create(blobData->contentType(), blobData->contentDisposition());
for (BlobDataItemList::const_iterator iter = blobData->items().begin(); iter != blobData->items().end(); ++iter) {
switch (iter->type) {
case BlobDataItem::Data:
+ // FIXME: Can we avoid making this copy of the data?
blobStorageData->m_data.appendData(iter->data, 0, iter->data->length());
break;
case BlobDataItem::File:
- blobStorageData->m_data.appendFile(iter->path, iter->offset, iter->length, iter->expectedModificationTime);
+ blobStorageData->m_data.appendFile(iter->path.isolatedCopy(), iter->offset, iter->length, iter->expectedModificationTime);
break;
#if ENABLE(FILE_SYSTEM)
case BlobDataItem::URL:
- blobStorageData->m_data.appendURL(iter->url, iter->offset, iter->length, iter->expectedModificationTime);
+ blobStorageData->m_data.appendURL(iter->url.copy(), iter->offset, iter->length, iter->expectedModificationTime);
break;
#endif
- case BlobDataItem::Blob:
- if (m_blobs.contains(iter->url.string()))
- appendStorageItems(blobStorageData.get(), m_blobs.get(iter->url.string())->items(), iter->offset, iter->length);
+ case BlobDataItem::Blob: {
+ RefPtr<BlobStorageData> other = getBlobDataFromUUID(iter->blobDataHandle->uuid());
+ if (other)
+ appendStorageItems(blobStorageData.get(), other->items(), iter->offset, iter->length);
+ }
break;
}
}
- m_blobs.set(url.string(), blobStorageData);
+
+ MutexLocker locker(m_mutex);
+ ASSERT(m_blobs.find(uuid) == m_blobs.end());
+ m_blobs.set(uuid.isolatedCopy(), std::make_pair(1, blobStorageData));
}
-void BlobRegistryImpl::registerBlobURL(const KURL& url, const KURL& srcURL)
+void BlobRegistryImpl::addBlobDataRef(const String& uuid)
{
- ASSERT(isMainThread());
- registerBlobResourceHandleConstructor();
+ MutexLocker locker(m_mutex);
+ BlobMap::iterator found = m_blobs.find(uuid);
+ ASSERT(found != m_blobs.end());
+ found->value.first += 1;
+}
- RefPtr<BlobStorageData> src = m_blobs.get(srcURL.string());
- ASSERT(src);
- if (!src)
- return;
+void BlobRegistryImpl::removeBlobDataRef(const String& uuid)
+{
+ MutexLocker locker(m_mutex);
+ BlobMap::iterator found = m_blobs.find(uuid);
+ ASSERT(found != m_blobs.end());
+ found->value.first -= 1;
+ if (found->value.first == 0)
+ m_blobs.remove(found);
+}
- m_blobs.set(url.string(), src);
+// Helper to access the blob url map on the main thread only when registering/revoking urls.
+class BlobRegistryImpl::URLRegistrationHelper {
+public:
+ URLRegistrationHelper(BlobRegistryImpl* registry, const KURL& url, PassRefPtr<BlobDataHandle> dataHandle = 0)
+ : registry(registry)
+ , url(url.copy())
+ , dataHandle(dataHandle)
+ {
+ }
+
+ static void registerURLTask(void* context)
+ {
+ OwnPtr<URLRegistrationHelper> helper = adoptPtr(static_cast<URLRegistrationHelper*>(context));
+ helper->registerURL();
+ }
+
+ static void revokeURLTask(void* context)
+ {
+ OwnPtr<URLRegistrationHelper> helper = adoptPtr(static_cast<URLRegistrationHelper*>(context));
+ helper->revokeURL();
+ }
+
+ void registerURL()
+ {
+ ASSERT(isMainThread());
+ registerBlobResourceHandleConstructor();
+ registry->m_publicURLs.set(url, dataHandle);
+ }
+
+ void revokeURL()
+ {
+ ASSERT(isMainThread());
+ registry->m_publicURLs.remove(url);
+ }
+
+ BlobRegistryImpl* registry;
+ KURL url;
+ RefPtr<BlobDataHandle> dataHandle; // NULL for revoke tasks.
+};
+
+
+void BlobRegistryImpl::registerPublicBlobURL(SecurityOrigin* origin, const KURL& url, PassRefPtr<BlobDataHandle> dataHandle)
+{
+ // The cached origin has thread affinity so we must fiddle with it's registration on the current thread.
+ BlobRegistry::setCachedUniqueOrigin(url, origin);
+
+ // But the blob map is only accessed on the main thread.
+ OwnPtr<URLRegistrationHelper> helper = adoptPtr(new URLRegistrationHelper(this, url, dataHandle));
+ if (isMainThread())
+ helper->registerURL();
+ else
+ callOnMainThread(&URLRegistrationHelper::registerURLTask, helper.leakPtr());
}
-void BlobRegistryImpl::unregisterBlobURL(const KURL& url)
+void BlobRegistryImpl::revokePublicBlobURL(const KURL& url)
{
- ASSERT(isMainThread());
- m_blobs.remove(url.string());
+ BlobRegistry::clearCachedUniqueOrigin(url);
+
+ OwnPtr<URLRegistrationHelper> helper = adoptPtr(new URLRegistrationHelper(this, url));
+ if (isMainThread())
+ helper->revokeURL();
+ else
+ callOnMainThread(&URLRegistrationHelper::revokeURLTask, helper.leakPtr());
}
+PassRefPtr<SecurityOrigin> BlobRegistryImpl::cachedUniqueOrigin(const KURL& url)
+{
+ return BlobRegistry::getCachedUniqueOrigin(url);
+}
+
+PassRefPtr<BlobStorageData> BlobRegistryImpl::getBlobDataFromUUID(const String& uuid) const
+{
+ MutexLocker locker(m_mutex);
+ return m_blobs.get(uuid).second;
+}
+
PassRefPtr<BlobStorageData> BlobRegistryImpl::getBlobDataFromURL(const KURL& url) const
{
ASSERT(isMainThread());
- return m_blobs.get(url.string());
+ RefPtr<BlobDataHandle> dataHandle = m_publicURLs.get(url.string());
+ if (!dataHandle)
+ return 0;
+ return getBlobDataFromUUID(dataHandle->uuid());
}
} // namespace WebCore
« no previous file with comments | « WebCore/platform/network/BlobRegistryImpl.h ('k') | WebCore/platform/network/BlobResourceHandle.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698