| /* |
| * 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: |
| * |
| * * Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * * 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. |
| * * Neither the name of Google Inc. nor the names of its |
| * contributors may be used to endorse or promote products derived from |
| * this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND 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 THE COPYRIGHT |
| * OWNER OR 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 "platform/blob/BlobRegistry.h" |
| |
| #include "platform/ThreadSafeFunctional.h" |
| #include "platform/blob/BlobData.h" |
| #include "platform/blob/BlobURL.h" |
| #include "platform/weborigin/SecurityOrigin.h" |
| #include "platform/weborigin/SecurityOriginCache.h" |
| #include "public/platform/Platform.h" |
| #include "public/platform/WebBlobData.h" |
| #include "public/platform/WebBlobRegistry.h" |
| #include "public/platform/WebString.h" |
| #include "public/platform/WebTaskRunner.h" |
| #include "public/platform/WebTraceLocation.h" |
| #include "wtf/Assertions.h" |
| #include "wtf/HashMap.h" |
| #include "wtf/MainThread.h" |
| #include "wtf/RefPtr.h" |
| #include "wtf/ThreadSpecific.h" |
| #include "wtf/Threading.h" |
| #include "wtf/text/StringHash.h" |
| #include "wtf/text/WTFString.h" |
| |
| namespace blink { |
| |
| class BlobOriginCache : public SecurityOriginCache { |
| public: |
| BlobOriginCache(); |
| SecurityOrigin* cachedOrigin(const KURL&) override; |
| }; |
| |
| static WebBlobRegistry* blobRegistry() |
| { |
| return Platform::current()->blobRegistry(); |
| } |
| |
| typedef HashMap<String, RefPtr<SecurityOrigin>> BlobURLOriginMap; |
| static ThreadSpecific<BlobURLOriginMap>& originMap() |
| { |
| // We want to create the BlobOriginCache exactly once because it is shared by all the threads. |
| DEFINE_THREAD_SAFE_STATIC_LOCAL(BlobOriginCache, cache, new BlobOriginCache); |
| (void)cache; // BlobOriginCache's constructor does the interesting work. |
| |
| DEFINE_THREAD_SAFE_STATIC_LOCAL(ThreadSpecific<BlobURLOriginMap>, map, new ThreadSpecific<BlobURLOriginMap>); |
| return map; |
| } |
| |
| static void saveToOriginMap(SecurityOrigin* origin, const KURL& url) |
| { |
| // If the blob URL contains null origin, as in the context with unique |
| // security origin or file URL, save the mapping between url and origin so |
| // that the origin can be retrived when doing security origin check. |
| if (origin && BlobURL::getOrigin(url) == "null") |
| originMap()->add(url.getString(), origin); |
| } |
| |
| static void removeFromOriginMap(const KURL& url) |
| { |
| if (BlobURL::getOrigin(url) == "null") |
| originMap()->remove(url.getString()); |
| } |
| |
| void BlobRegistry::registerBlobData(const String& uuid, PassOwnPtr<BlobData> data) |
| { |
| blobRegistry()->registerBlobData(uuid, WebBlobData(std::move(data))); |
| } |
| |
| void BlobRegistry::addBlobDataRef(const String& uuid) |
| { |
| blobRegistry()->addBlobDataRef(uuid); |
| } |
| |
| void BlobRegistry::removeBlobDataRef(const String& uuid) |
| { |
| blobRegistry()->removeBlobDataRef(uuid); |
| } |
| |
| void BlobRegistry::registerPublicBlobURL(SecurityOrigin* origin, const KURL& url, PassRefPtr<BlobDataHandle> handle) |
| { |
| saveToOriginMap(origin, url); |
| blobRegistry()->registerPublicBlobURL(url, handle->uuid()); |
| } |
| |
| void BlobRegistry::revokePublicBlobURL(const KURL& url) |
| { |
| removeFromOriginMap(url); |
| blobRegistry()->revokePublicBlobURL(url); |
| } |
| |
| static void registerStreamURLTask(const KURL& url, const String& type) |
| { |
| if (WebBlobRegistry* registry = blobRegistry()) |
| registry->registerStreamURL(url, type); |
| } |
| |
| void BlobRegistry::registerStreamURL(const KURL& url, const String& type) |
| { |
| if (isMainThread()) |
| registerStreamURLTask(url, type); |
| else |
| Platform::current()->mainThread()->getWebTaskRunner()->postTask(BLINK_FROM_HERE, threadSafeBind(®isterStreamURLTask, url, type)); |
| } |
| |
| static void registerStreamURLFromTask(const KURL& url, const KURL& srcURL) |
| { |
| if (WebBlobRegistry* registry = blobRegistry()) |
| registry->registerStreamURL(url, srcURL); |
| } |
| |
| void BlobRegistry::registerStreamURL(SecurityOrigin* origin, const KURL& url, const KURL& srcURL) |
| { |
| saveToOriginMap(origin, url); |
| |
| if (isMainThread()) |
| registerStreamURLFromTask(url, srcURL); |
| else |
| Platform::current()->mainThread()->getWebTaskRunner()->postTask(BLINK_FROM_HERE, threadSafeBind(®isterStreamURLFromTask, url, srcURL)); |
| } |
| |
| static void addDataToStreamTask(const KURL& url, PassRefPtr<RawData> streamData) |
| { |
| if (WebBlobRegistry* registry = blobRegistry()) |
| registry->addDataToStream(url, streamData->data(), streamData->length()); |
| } |
| |
| void BlobRegistry::addDataToStream(const KURL& url, PassRefPtr<RawData> streamData) |
| { |
| if (isMainThread()) |
| addDataToStreamTask(url, streamData); |
| else |
| Platform::current()->mainThread()->getWebTaskRunner()->postTask(BLINK_FROM_HERE, threadSafeBind(&addDataToStreamTask, url, streamData)); |
| } |
| |
| static void flushStreamTask(const KURL& url) |
| { |
| if (WebBlobRegistry* registry = blobRegistry()) |
| registry->flushStream(url); |
| } |
| |
| void BlobRegistry::flushStream(const KURL& url) |
| { |
| if (isMainThread()) |
| flushStreamTask(url); |
| else |
| Platform::current()->mainThread()->getWebTaskRunner()->postTask(BLINK_FROM_HERE, threadSafeBind(&flushStreamTask, url)); |
| } |
| |
| static void finalizeStreamTask(const KURL& url) |
| { |
| if (WebBlobRegistry* registry = blobRegistry()) |
| registry->finalizeStream(url); |
| } |
| |
| void BlobRegistry::finalizeStream(const KURL& url) |
| { |
| if (isMainThread()) |
| finalizeStreamTask(url); |
| else |
| Platform::current()->mainThread()->getWebTaskRunner()->postTask(BLINK_FROM_HERE, threadSafeBind(&finalizeStreamTask, url)); |
| } |
| |
| static void abortStreamTask(const KURL& url) |
| { |
| if (WebBlobRegistry* registry = blobRegistry()) |
| registry->abortStream(url); |
| } |
| |
| void BlobRegistry::abortStream(const KURL& url) |
| { |
| if (isMainThread()) |
| abortStreamTask(url); |
| else |
| Platform::current()->mainThread()->getWebTaskRunner()->postTask(BLINK_FROM_HERE, threadSafeBind(&abortStreamTask, url)); |
| } |
| |
| static void unregisterStreamURLTask(const KURL& url) |
| { |
| if (WebBlobRegistry* registry = blobRegistry()) |
| registry->unregisterStreamURL(url); |
| } |
| |
| void BlobRegistry::unregisterStreamURL(const KURL& url) |
| { |
| removeFromOriginMap(url); |
| |
| if (isMainThread()) |
| unregisterStreamURLTask(url); |
| else |
| Platform::current()->mainThread()->getWebTaskRunner()->postTask(BLINK_FROM_HERE, threadSafeBind(&unregisterStreamURLTask, url)); |
| } |
| |
| BlobOriginCache::BlobOriginCache() |
| { |
| SecurityOrigin::setCache(this); |
| } |
| |
| SecurityOrigin* BlobOriginCache::cachedOrigin(const KURL& url) |
| { |
| if (url.protocolIs("blob")) |
| return originMap()->get(url.getString()); |
| return 0; |
| } |
| |
| } // namespace blink |