OLD | NEW |
| (Empty) |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | |
2 // Use of this source code is governed by a BSD-style license that can be | |
3 // found in the LICENSE file. | |
4 | |
5 #include "content/common_child/webblobregistry_impl.h" | |
6 | |
7 #include "base/memory/ref_counted.h" | |
8 #include "base/message_loop.h" | |
9 #include "base/shared_memory.h" | |
10 #include "content/common/child_thread.h" | |
11 #include "content/common/fileapi/webblob_messages.h" | |
12 #include "content/common/thread_safe_sender.h" | |
13 #include "third_party/WebKit/public/platform/WebBlobData.h" | |
14 #include "third_party/WebKit/public/platform/WebString.h" | |
15 #include "third_party/WebKit/public/platform/WebURL.h" | |
16 #include "webkit/base/file_path_string_conversions.h" | |
17 #include "webkit/common/blob/blob_data.h" | |
18 | |
19 using WebKit::WebBlobData; | |
20 using WebKit::WebString; | |
21 using WebKit::WebURL; | |
22 | |
23 namespace content { | |
24 | |
25 WebBlobRegistryImpl::WebBlobRegistryImpl(ThreadSafeSender* sender) | |
26 : sender_(sender) { | |
27 } | |
28 | |
29 WebBlobRegistryImpl::~WebBlobRegistryImpl() { | |
30 } | |
31 | |
32 void WebBlobRegistryImpl::registerBlobURL( | |
33 const WebURL& url, WebBlobData& data) { | |
34 DCHECK(ChildThread::current()->message_loop() == | |
35 base::MessageLoop::current()); | |
36 const size_t kLargeThresholdBytes = 250 * 1024; | |
37 const size_t kMaxSharedMemoryBytes = 10 * 1024 * 1024; | |
38 | |
39 sender_->Send(new BlobHostMsg_StartBuildingBlob(url)); | |
40 size_t i = 0; | |
41 WebBlobData::Item data_item; | |
42 while (data.itemAt(i++, data_item)) { | |
43 webkit_blob::BlobData::Item item; | |
44 switch (data_item.type) { | |
45 case WebBlobData::Item::TypeData: { | |
46 // WebBlobData does not allow partial data items. | |
47 DCHECK(!data_item.offset && data_item.length == -1); | |
48 if (data_item.data.size() == 0) | |
49 break; | |
50 if (data_item.data.size() < kLargeThresholdBytes) { | |
51 item.SetToBytes(data_item.data.data(), data_item.data.size()); | |
52 sender_->Send(new BlobHostMsg_AppendBlobDataItem(url, item)); | |
53 } else { | |
54 // We handle larger amounts of data via SharedMemory instead of | |
55 // writing it directly to the IPC channel. | |
56 size_t data_size = data_item.data.size(); | |
57 const char* data_ptr = data_item.data.data(); | |
58 size_t shared_memory_size = std::min( | |
59 data_size, kMaxSharedMemoryBytes); | |
60 scoped_ptr<base::SharedMemory> shared_memory( | |
61 ChildThread::AllocateSharedMemory(shared_memory_size, | |
62 sender_.get())); | |
63 CHECK(shared_memory.get()); | |
64 while (data_size) { | |
65 size_t chunk_size = std::min(data_size, shared_memory_size); | |
66 memcpy(shared_memory->memory(), data_ptr, chunk_size); | |
67 sender_->Send(new BlobHostMsg_SyncAppendSharedMemory( | |
68 url, shared_memory->handle(), chunk_size)); | |
69 data_size -= chunk_size; | |
70 data_ptr += chunk_size; | |
71 } | |
72 } | |
73 break; | |
74 } | |
75 case WebBlobData::Item::TypeFile: | |
76 if (data_item.length) { | |
77 item.SetToFilePathRange( | |
78 webkit_base::WebStringToFilePath(data_item.filePath), | |
79 static_cast<uint64>(data_item.offset), | |
80 static_cast<uint64>(data_item.length), | |
81 base::Time::FromDoubleT(data_item.expectedModificationTime)); | |
82 sender_->Send(new BlobHostMsg_AppendBlobDataItem(url, item)); | |
83 } | |
84 break; | |
85 case WebBlobData::Item::TypeBlob: | |
86 if (data_item.length) { | |
87 item.SetToBlobUrlRange( | |
88 data_item.blobURL, | |
89 static_cast<uint64>(data_item.offset), | |
90 static_cast<uint64>(data_item.length)); | |
91 sender_->Send(new BlobHostMsg_AppendBlobDataItem(url, item)); | |
92 } | |
93 break; | |
94 case WebBlobData::Item::TypeURL: | |
95 if (data_item.length) { | |
96 // We only support filesystem URL as of now. | |
97 DCHECK(GURL(data_item.url).SchemeIsFileSystem()); | |
98 item.SetToFileSystemUrlRange( | |
99 data_item.url, | |
100 static_cast<uint64>(data_item.offset), | |
101 static_cast<uint64>(data_item.length), | |
102 base::Time::FromDoubleT(data_item.expectedModificationTime)); | |
103 sender_->Send(new BlobHostMsg_AppendBlobDataItem(url, item)); | |
104 } | |
105 break; | |
106 default: | |
107 NOTREACHED(); | |
108 } | |
109 } | |
110 sender_->Send(new BlobHostMsg_FinishBuildingBlob( | |
111 url, data.contentType().utf8().data())); | |
112 } | |
113 | |
114 void WebBlobRegistryImpl::registerBlobURL( | |
115 const WebURL& url, const WebURL& src_url) { | |
116 DCHECK(ChildThread::current()->message_loop() == | |
117 base::MessageLoop::current()); | |
118 sender_->Send(new BlobHostMsg_CloneBlob(url, src_url)); | |
119 } | |
120 | |
121 void WebBlobRegistryImpl::unregisterBlobURL(const WebURL& url) { | |
122 DCHECK(ChildThread::current()->message_loop() == | |
123 base::MessageLoop::current()); | |
124 sender_->Send(new BlobHostMsg_RemoveBlob(url)); | |
125 } | |
126 | |
127 } // namespace content | |
OLD | NEW |