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

Side by Side Diff: net/disk_cache/blockfile/file_win.cc

Issue 992733002: Remove //net (except for Android test stuff) and sdch (Closed) Base URL: git@github.com:domokit/mojo.git@master
Patch Set: Created 5 years, 9 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 unified diff | Download patch
« no previous file with comments | « net/disk_cache/blockfile/file_posix.cc ('k') | net/disk_cache/blockfile/histogram_macros.h » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
(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 "net/disk_cache/blockfile/file.h"
6
7 #include "base/files/file_path.h"
8 #include "base/lazy_instance.h"
9 #include "base/message_loop/message_loop.h"
10 #include "net/base/net_errors.h"
11 #include "net/disk_cache/disk_cache.h"
12
13 namespace {
14
15 // Structure used for asynchronous operations.
16 struct MyOverlapped {
17 MyOverlapped(disk_cache::File* file, size_t offset,
18 disk_cache::FileIOCallback* callback);
19 ~MyOverlapped() {}
20 OVERLAPPED* overlapped() {
21 return &context_.overlapped;
22 }
23
24 base::MessageLoopForIO::IOContext context_;
25 scoped_refptr<disk_cache::File> file_;
26 disk_cache::FileIOCallback* callback_;
27 };
28
29 static_assert(offsetof(MyOverlapped, context_) == 0,
30 "should start with overlapped");
31
32 // Helper class to handle the IO completion notifications from the message loop.
33 class CompletionHandler : public base::MessageLoopForIO::IOHandler {
34 virtual void OnIOCompleted(base::MessageLoopForIO::IOContext* context,
35 DWORD actual_bytes,
36 DWORD error);
37 };
38
39 static base::LazyInstance<CompletionHandler> g_completion_handler =
40 LAZY_INSTANCE_INITIALIZER;
41
42 void CompletionHandler::OnIOCompleted(
43 base::MessageLoopForIO::IOContext* context,
44 DWORD actual_bytes,
45 DWORD error) {
46 MyOverlapped* data = reinterpret_cast<MyOverlapped*>(context);
47
48 if (error) {
49 DCHECK(!actual_bytes);
50 actual_bytes = static_cast<DWORD>(net::ERR_CACHE_READ_FAILURE);
51 NOTREACHED();
52 }
53
54 if (data->callback_)
55 data->callback_->OnFileIOComplete(static_cast<int>(actual_bytes));
56
57 delete data;
58 }
59
60 MyOverlapped::MyOverlapped(disk_cache::File* file, size_t offset,
61 disk_cache::FileIOCallback* callback) {
62 memset(this, 0, sizeof(*this));
63 context_.handler = g_completion_handler.Pointer();
64 context_.overlapped.Offset = static_cast<DWORD>(offset);
65 file_ = file;
66 callback_ = callback;
67 }
68
69 } // namespace
70
71 namespace disk_cache {
72
73 File::File(base::File file)
74 : init_(true),
75 mixed_(true),
76 sync_base_file_(file.Pass()) {
77 }
78
79 bool File::Init(const base::FilePath& name) {
80 DCHECK(!init_);
81 if (init_)
82 return false;
83
84 DWORD sharing = FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE;
85 DWORD access = GENERIC_READ | GENERIC_WRITE | DELETE;
86 base_file_ =
87 base::File(CreateFile(name.value().c_str(), access, sharing, NULL,
88 OPEN_EXISTING, FILE_FLAG_OVERLAPPED, NULL));
89
90 if (!base_file_.IsValid())
91 return false;
92
93 base::MessageLoopForIO::current()->RegisterIOHandler(
94 base_file_.GetPlatformFile(), g_completion_handler.Pointer());
95
96 init_ = true;
97 sync_base_file_ =
98 base::File(CreateFile(name.value().c_str(), access, sharing, NULL,
99 OPEN_EXISTING, 0, NULL));
100
101 if (!sync_base_file_.IsValid())
102 return false;
103
104 return true;
105 }
106
107 bool File::IsValid() const {
108 if (!init_)
109 return false;
110 return base_file_.IsValid() || sync_base_file_.IsValid();
111 }
112
113 bool File::Read(void* buffer, size_t buffer_len, size_t offset) {
114 DCHECK(init_);
115 if (buffer_len > ULONG_MAX || offset > LONG_MAX)
116 return false;
117
118 int ret = sync_base_file_.Read(offset, static_cast<char*>(buffer),
119 buffer_len);
120 return static_cast<int>(buffer_len) == ret;
121 }
122
123 bool File::Write(const void* buffer, size_t buffer_len, size_t offset) {
124 DCHECK(init_);
125 if (buffer_len > ULONG_MAX || offset > ULONG_MAX)
126 return false;
127
128 int ret = sync_base_file_.Write(offset, static_cast<const char*>(buffer),
129 buffer_len);
130 return static_cast<int>(buffer_len) == ret;
131 }
132
133 // We have to increase the ref counter of the file before performing the IO to
134 // prevent the completion to happen with an invalid handle (if the file is
135 // closed while the IO is in flight).
136 bool File::Read(void* buffer, size_t buffer_len, size_t offset,
137 FileIOCallback* callback, bool* completed) {
138 DCHECK(init_);
139 if (!callback) {
140 if (completed)
141 *completed = true;
142 return Read(buffer, buffer_len, offset);
143 }
144
145 if (buffer_len > ULONG_MAX || offset > ULONG_MAX)
146 return false;
147
148 MyOverlapped* data = new MyOverlapped(this, offset, callback);
149 DWORD size = static_cast<DWORD>(buffer_len);
150
151 DWORD actual;
152 if (!ReadFile(base_file_.GetPlatformFile(), buffer, size, &actual,
153 data->overlapped())) {
154 *completed = false;
155 if (GetLastError() == ERROR_IO_PENDING)
156 return true;
157 delete data;
158 return false;
159 }
160
161 // The operation completed already. We'll be called back anyway.
162 *completed = (actual == size);
163 DCHECK_EQ(size, actual);
164 data->callback_ = NULL;
165 data->file_ = NULL; // There is no reason to hold on to this anymore.
166 return *completed;
167 }
168
169 bool File::Write(const void* buffer, size_t buffer_len, size_t offset,
170 FileIOCallback* callback, bool* completed) {
171 DCHECK(init_);
172 if (!callback) {
173 if (completed)
174 *completed = true;
175 return Write(buffer, buffer_len, offset);
176 }
177
178 return AsyncWrite(buffer, buffer_len, offset, callback, completed);
179 }
180
181 File::~File() {
182 }
183
184 base::PlatformFile File::platform_file() const {
185 DCHECK(init_);
186 return base_file_.IsValid() ? base_file_.GetPlatformFile() :
187 sync_base_file_.GetPlatformFile();
188 }
189
190 bool File::AsyncWrite(const void* buffer, size_t buffer_len, size_t offset,
191 FileIOCallback* callback, bool* completed) {
192 DCHECK(init_);
193 DCHECK(callback);
194 DCHECK(completed);
195 if (buffer_len > ULONG_MAX || offset > ULONG_MAX)
196 return false;
197
198 MyOverlapped* data = new MyOverlapped(this, offset, callback);
199 DWORD size = static_cast<DWORD>(buffer_len);
200
201 DWORD actual;
202 if (!WriteFile(base_file_.GetPlatformFile(), buffer, size, &actual,
203 data->overlapped())) {
204 *completed = false;
205 if (GetLastError() == ERROR_IO_PENDING)
206 return true;
207 delete data;
208 return false;
209 }
210
211 // The operation completed already. We'll be called back anyway.
212 *completed = (actual == size);
213 DCHECK_EQ(size, actual);
214 data->callback_ = NULL;
215 data->file_ = NULL; // There is no reason to hold on to this anymore.
216 return *completed;
217 }
218
219 bool File::SetLength(size_t length) {
220 DCHECK(init_);
221 if (length > ULONG_MAX)
222 return false;
223
224 DWORD size = static_cast<DWORD>(length);
225 HANDLE file = platform_file();
226 if (INVALID_SET_FILE_POINTER == SetFilePointer(file, size, NULL, FILE_BEGIN))
227 return false;
228
229 return TRUE == SetEndOfFile(file);
230 }
231
232 size_t File::GetLength() {
233 DCHECK(init_);
234 LARGE_INTEGER size;
235 HANDLE file = platform_file();
236 if (!GetFileSizeEx(file, &size))
237 return 0;
238 if (size.HighPart)
239 return ULONG_MAX;
240
241 return static_cast<size_t>(size.LowPart);
242 }
243
244 // Static.
245 void File::WaitForPendingIO(int* num_pending_io) {
246 while (*num_pending_io) {
247 // Asynchronous IO operations may be in flight and the completion may end
248 // up calling us back so let's wait for them.
249 base::MessageLoopForIO::IOHandler* handler = g_completion_handler.Pointer();
250 base::MessageLoopForIO::current()->WaitForIOCompletion(100, handler);
251 }
252 }
253
254 // Static.
255 void File::DropPendingIO() {
256 }
257
258 } // namespace disk_cache
OLDNEW
« no previous file with comments | « net/disk_cache/blockfile/file_posix.cc ('k') | net/disk_cache/blockfile/histogram_macros.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698