OLD | NEW |
1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "chrome/browser/chromeos/file_system_provider/fileapi/file_stream_write
r.h" | 5 #include "chrome/browser/chromeos/file_system_provider/fileapi/file_stream_write
r.h" |
6 | 6 |
7 #include "base/debug/trace_event.h" | 7 #include "base/debug/trace_event.h" |
8 #include "base/memory/ref_counted.h" | 8 #include "base/memory/ref_counted.h" |
9 #include "base/thread_task_runner_handle.h" | 9 #include "base/thread_task_runner_handle.h" |
10 #include "chrome/browser/chromeos/file_system_provider/abort_callback.h" | 10 #include "chrome/browser/chromeos/file_system_provider/abort_callback.h" |
(...skipping 20 matching lines...) Expand all Loading... |
31 : public base::RefCountedThreadSafe<FileStreamWriter::OperationRunner> { | 31 : public base::RefCountedThreadSafe<FileStreamWriter::OperationRunner> { |
32 public: | 32 public: |
33 OperationRunner() : file_handle_(-1) {} | 33 OperationRunner() : file_handle_(-1) {} |
34 | 34 |
35 // Opens a file for writing and calls the completion callback. Must be called | 35 // Opens a file for writing and calls the completion callback. Must be called |
36 // on UI thread. | 36 // on UI thread. |
37 void OpenFileOnUIThread( | 37 void OpenFileOnUIThread( |
38 const storage::FileSystemURL& url, | 38 const storage::FileSystemURL& url, |
39 const storage::AsyncFileUtil::StatusCallback& callback) { | 39 const storage::AsyncFileUtil::StatusCallback& callback) { |
40 DCHECK_CURRENTLY_ON(BrowserThread::UI); | 40 DCHECK_CURRENTLY_ON(BrowserThread::UI); |
| 41 DCHECK(abort_callback_.is_null()); |
41 | 42 |
42 util::FileSystemURLParser parser(url); | 43 util::FileSystemURLParser parser(url); |
43 if (!parser.Parse()) { | 44 if (!parser.Parse()) { |
44 BrowserThread::PostTask( | 45 BrowserThread::PostTask( |
45 BrowserThread::IO, | 46 BrowserThread::IO, |
46 FROM_HERE, | 47 FROM_HERE, |
47 base::Bind(callback, base::File::FILE_ERROR_SECURITY)); | 48 base::Bind(callback, base::File::FILE_ERROR_SECURITY)); |
48 return; | 49 return; |
49 } | 50 } |
50 | 51 |
51 file_system_ = parser.file_system()->GetWeakPtr(); | 52 file_system_ = parser.file_system()->GetWeakPtr(); |
52 abort_callback_ = parser.file_system()->OpenFile( | 53 abort_callback_ = parser.file_system()->OpenFile( |
53 parser.file_path(), | 54 parser.file_path(), |
54 ProvidedFileSystemInterface::OPEN_FILE_MODE_WRITE, | 55 ProvidedFileSystemInterface::OPEN_FILE_MODE_WRITE, |
55 base::Bind( | 56 base::Bind( |
56 &OperationRunner::OnOpenFileCompletedOnUIThread, this, callback)); | 57 &OperationRunner::OnOpenFileCompletedOnUIThread, this, callback)); |
57 } | 58 } |
58 | 59 |
59 // Closes a file. Ignores result, since outlives the caller. Must be called on | 60 // Closes a file. Ignores result, since outlives the caller. Must be called on |
60 // UI thread. | 61 // UI thread. |
61 void CloseFileOnUIThread() { | 62 void CloseFileOnUIThread() { |
62 DCHECK_CURRENTLY_ON(BrowserThread::UI); | 63 DCHECK_CURRENTLY_ON(BrowserThread::UI); |
| 64 DCHECK(abort_callback_.is_null()); |
| 65 |
63 if (file_system_.get() && file_handle_ != -1) { | 66 if (file_system_.get() && file_handle_ != -1) { |
64 // Closing a file must not be aborted, since we could end up on files | 67 // Closing a file must not be aborted, since we could end up on files |
65 // which are never closed. | 68 // which are never closed. |
66 file_system_->CloseFile(file_handle_, base::Bind(&EmptyStatusCallback)); | 69 file_system_->CloseFile(file_handle_, base::Bind(&EmptyStatusCallback)); |
67 } | 70 } |
68 } | 71 } |
69 | 72 |
70 // Requests writing bytes to the file. In case of either success or a failure | 73 // Requests writing bytes to the file. In case of either success or a failure |
71 // |callback| is executed. Must be called on UI thread. | 74 // |callback| is executed. Must be called on UI thread. |
72 void WriteFileOnUIThread( | 75 void WriteFileOnUIThread( |
73 scoped_refptr<net::IOBuffer> buffer, | 76 scoped_refptr<net::IOBuffer> buffer, |
74 int64 offset, | 77 int64 offset, |
75 int length, | 78 int length, |
76 const storage::AsyncFileUtil::StatusCallback& callback) { | 79 const storage::AsyncFileUtil::StatusCallback& callback) { |
77 DCHECK_CURRENTLY_ON(BrowserThread::UI); | 80 DCHECK_CURRENTLY_ON(BrowserThread::UI); |
| 81 DCHECK(abort_callback_.is_null()); |
78 | 82 |
79 // If the file system got unmounted, then abort the writing operation. | 83 // If the file system got unmounted, then abort the writing operation. |
80 if (!file_system_.get()) { | 84 if (!file_system_.get()) { |
81 BrowserThread::PostTask( | 85 BrowserThread::PostTask( |
82 BrowserThread::IO, | 86 BrowserThread::IO, |
83 FROM_HERE, | 87 FROM_HERE, |
84 base::Bind(callback, base::File::FILE_ERROR_ABORT)); | 88 base::Bind(callback, base::File::FILE_ERROR_ABORT)); |
85 return; | 89 return; |
86 } | 90 } |
87 | 91 |
88 abort_callback_ = file_system_->WriteFile( | 92 abort_callback_ = file_system_->WriteFile( |
89 file_handle_, | 93 file_handle_, |
90 buffer.get(), | 94 buffer.get(), |
91 offset, | 95 offset, |
92 length, | 96 length, |
93 base::Bind( | 97 base::Bind( |
94 &OperationRunner::OnWriteFileCompletedOnUIThread, this, callback)); | 98 &OperationRunner::OnWriteFileCompletedOnUIThread, this, callback)); |
95 } | 99 } |
96 | 100 |
97 // Aborts the most recent operation (if exists), and calls the callback. | 101 // Aborts the most recent operation (if exists). |
98 void AbortOnUIThread(const storage::AsyncFileUtil::StatusCallback& callback) { | 102 void AbortOnUIThread() { |
99 DCHECK_CURRENTLY_ON(BrowserThread::UI); | 103 DCHECK_CURRENTLY_ON(BrowserThread::UI); |
100 | 104 if (abort_callback_.is_null()) |
101 if (abort_callback_.is_null()) { | |
102 // No operation to be cancelled. At most a callback call, which will be | |
103 // discarded. | |
104 BrowserThread::PostTask(BrowserThread::IO, | |
105 FROM_HERE, | |
106 base::Bind(callback, base::File::FILE_OK)); | |
107 return; | 105 return; |
108 } | |
109 | 106 |
110 const AbortCallback last_abort_callback = abort_callback_; | 107 const AbortCallback last_abort_callback = abort_callback_; |
111 abort_callback_ = AbortCallback(); | 108 abort_callback_ = AbortCallback(); |
112 last_abort_callback.Run(base::Bind( | 109 last_abort_callback.Run(); |
113 &OperationRunner::OnAbortCompletedOnUIThread, this, callback)); | |
114 } | 110 } |
115 | 111 |
116 private: | 112 private: |
117 friend class base::RefCountedThreadSafe<OperationRunner>; | 113 friend class base::RefCountedThreadSafe<OperationRunner>; |
118 | 114 |
119 virtual ~OperationRunner() {} | 115 virtual ~OperationRunner() {} |
120 | 116 |
121 // Remembers a file handle for further operations and forwards the result to | 117 // Remembers a file handle for further operations and forwards the result to |
122 // the IO thread. | 118 // the IO thread. |
123 void OnOpenFileCompletedOnUIThread( | 119 void OnOpenFileCompletedOnUIThread( |
124 const storage::AsyncFileUtil::StatusCallback& callback, | 120 const storage::AsyncFileUtil::StatusCallback& callback, |
125 int file_handle, | 121 int file_handle, |
126 base::File::Error result) { | 122 base::File::Error result) { |
127 DCHECK_CURRENTLY_ON(BrowserThread::UI); | 123 DCHECK_CURRENTLY_ON(BrowserThread::UI); |
128 | 124 |
| 125 abort_callback_ = AbortCallback(); |
129 if (result == base::File::FILE_OK) | 126 if (result == base::File::FILE_OK) |
130 file_handle_ = file_handle; | 127 file_handle_ = file_handle; |
131 | 128 |
132 BrowserThread::PostTask( | 129 BrowserThread::PostTask( |
133 BrowserThread::IO, FROM_HERE, base::Bind(callback, result)); | 130 BrowserThread::IO, FROM_HERE, base::Bind(callback, result)); |
134 } | 131 } |
135 | 132 |
136 // Forwards a response of writing to a file to the IO thread. | 133 // Forwards a response of writing to a file to the IO thread. |
137 void OnWriteFileCompletedOnUIThread( | 134 void OnWriteFileCompletedOnUIThread( |
138 const storage::AsyncFileUtil::StatusCallback& callback, | 135 const storage::AsyncFileUtil::StatusCallback& callback, |
139 base::File::Error result) { | 136 base::File::Error result) { |
140 DCHECK_CURRENTLY_ON(BrowserThread::UI); | 137 DCHECK_CURRENTLY_ON(BrowserThread::UI); |
| 138 |
| 139 abort_callback_ = AbortCallback(); |
141 BrowserThread::PostTask( | 140 BrowserThread::PostTask( |
142 BrowserThread::IO, FROM_HERE, base::Bind(callback, result)); | 141 BrowserThread::IO, FROM_HERE, base::Bind(callback, result)); |
143 } | 142 } |
144 | |
145 // Forwards a response of aborting an operation to the IO thread. | |
146 void OnAbortCompletedOnUIThread( | |
147 const storage::AsyncFileUtil::StatusCallback& callback, | |
148 base::File::Error result) { | |
149 DCHECK_CURRENTLY_ON(BrowserThread::UI); | |
150 BrowserThread::PostTask( | |
151 BrowserThread::IO, FROM_HERE, base::Bind(callback, result)); | |
152 } | |
153 | 143 |
154 AbortCallback abort_callback_; | 144 AbortCallback abort_callback_; |
155 base::WeakPtr<ProvidedFileSystemInterface> file_system_; | 145 base::WeakPtr<ProvidedFileSystemInterface> file_system_; |
156 int file_handle_; | 146 int file_handle_; |
157 | 147 |
158 DISALLOW_COPY_AND_ASSIGN(OperationRunner); | 148 DISALLOW_COPY_AND_ASSIGN(OperationRunner); |
159 }; | 149 }; |
160 | 150 |
161 FileStreamWriter::FileStreamWriter(const storage::FileSystemURL& url, | 151 FileStreamWriter::FileStreamWriter(const storage::FileSystemURL& url, |
162 int64 initial_offset) | 152 int64 initial_offset) |
(...skipping 82 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
245 break; | 235 break; |
246 | 236 |
247 case INITIALIZED: | 237 case INITIALIZED: |
248 WriteAfterInitialized(buffer, | 238 WriteAfterInitialized(buffer, |
249 buffer_length, | 239 buffer_length, |
250 base::Bind(&FileStreamWriter::OnWriteCompleted, | 240 base::Bind(&FileStreamWriter::OnWriteCompleted, |
251 weak_ptr_factory_.GetWeakPtr(), | 241 weak_ptr_factory_.GetWeakPtr(), |
252 callback)); | 242 callback)); |
253 break; | 243 break; |
254 | 244 |
| 245 case EXECUTING: |
255 case FAILED: | 246 case FAILED: |
256 NOTREACHED(); | 247 NOTREACHED(); |
257 break; | 248 break; |
258 } | 249 } |
259 | 250 |
260 return net::ERR_IO_PENDING; | 251 return net::ERR_IO_PENDING; |
261 } | 252 } |
262 | 253 |
263 int FileStreamWriter::Cancel(const net::CompletionCallback& callback) { | 254 int FileStreamWriter::Cancel(const net::CompletionCallback& callback) { |
264 DCHECK_CURRENTLY_ON(BrowserThread::IO); | 255 DCHECK_CURRENTLY_ON(BrowserThread::IO); |
265 | 256 |
| 257 if (state_ != INITIALIZING && state_ != EXECUTING) |
| 258 return net::ERR_UNEXPECTED; |
| 259 |
| 260 // Abort and Optimistically return an OK result code, as the aborting |
| 261 // operation is always forced and can't be cancelled. |
266 BrowserThread::PostTask( | 262 BrowserThread::PostTask( |
267 BrowserThread::UI, | 263 BrowserThread::UI, FROM_HERE, |
268 FROM_HERE, | 264 base::Bind(&OperationRunner::AbortOnUIThread, runner_)); |
269 base::Bind(&OperationRunner::AbortOnUIThread, | 265 base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, |
270 runner_, | 266 base::Bind(callback, net::OK)); |
271 base::Bind(&FileStreamWriter::OnAbortCompleted, | 267 |
272 weak_ptr_factory_.GetWeakPtr(), | |
273 callback))); | |
274 return net::ERR_IO_PENDING; | 268 return net::ERR_IO_PENDING; |
275 } | 269 } |
276 | 270 |
277 int FileStreamWriter::Flush(const net::CompletionCallback& callback) { | 271 int FileStreamWriter::Flush(const net::CompletionCallback& callback) { |
| 272 DCHECK_CURRENTLY_ON(BrowserThread::IO); |
278 base::ThreadTaskRunnerHandle::Get()->PostTask( | 273 base::ThreadTaskRunnerHandle::Get()->PostTask( |
279 FROM_HERE, | 274 FROM_HERE, |
280 base::Bind(callback, state_ == INITIALIZED ? net::OK : net::ERR_FAILED)); | 275 base::Bind(callback, state_ == INITIALIZED ? net::OK : net::ERR_FAILED)); |
281 | 276 |
282 return net::ERR_IO_PENDING; | 277 return net::ERR_IO_PENDING; |
283 } | 278 } |
284 | 279 |
285 void FileStreamWriter::OnWriteFileCompleted( | 280 void FileStreamWriter::OnWriteFileCompleted( |
286 int buffer_length, | 281 int buffer_length, |
287 const net::CompletionCallback& callback, | 282 const net::CompletionCallback& callback, |
288 base::File::Error result) { | 283 base::File::Error result) { |
289 DCHECK_CURRENTLY_ON(BrowserThread::IO); | 284 DCHECK_CURRENTLY_ON(BrowserThread::IO); |
290 DCHECK_EQ(INITIALIZED, state_); | 285 DCHECK_EQ(EXECUTING, state_); |
| 286 state_ = INITIALIZED; |
291 | 287 |
292 if (result != base::File::FILE_OK) { | 288 if (result != base::File::FILE_OK) { |
293 state_ = FAILED; | 289 state_ = FAILED; |
294 callback.Run(net::FileErrorToNetError(result)); | 290 callback.Run(net::FileErrorToNetError(result)); |
295 return; | 291 return; |
296 } | 292 } |
297 | 293 |
298 current_offset_ += buffer_length; | 294 current_offset_ += buffer_length; |
299 callback.Run(buffer_length); | 295 callback.Run(buffer_length); |
300 } | 296 } |
301 | 297 |
302 void FileStreamWriter::OnWriteCompleted(net::CompletionCallback callback, | 298 void FileStreamWriter::OnWriteCompleted(net::CompletionCallback callback, |
303 int result) { | 299 int result) { |
304 DCHECK_CURRENTLY_ON(BrowserThread::IO); | 300 DCHECK_CURRENTLY_ON(BrowserThread::IO); |
305 callback.Run(result); | 301 callback.Run(result); |
306 TRACE_EVENT_ASYNC_END0( | 302 TRACE_EVENT_ASYNC_END0( |
307 "file_system_provider", "FileStreamWriter::Write", this); | 303 "file_system_provider", "FileStreamWriter::Write", this); |
308 } | 304 } |
309 | 305 |
310 void FileStreamWriter::OnAbortCompleted(const net::CompletionCallback& callback, | |
311 base::File::Error result) { | |
312 DCHECK_CURRENTLY_ON(BrowserThread::IO); | |
313 | |
314 if (result != base::File::FILE_OK) | |
315 state_ = FAILED; | |
316 | |
317 callback.Run(net::FileErrorToNetError(result)); | |
318 } | |
319 | |
320 void FileStreamWriter::WriteAfterInitialized( | 306 void FileStreamWriter::WriteAfterInitialized( |
321 scoped_refptr<net::IOBuffer> buffer, | 307 scoped_refptr<net::IOBuffer> buffer, |
322 int buffer_length, | 308 int buffer_length, |
323 const net::CompletionCallback& callback) { | 309 const net::CompletionCallback& callback) { |
324 DCHECK_CURRENTLY_ON(BrowserThread::IO); | 310 DCHECK_CURRENTLY_ON(BrowserThread::IO); |
325 DCHECK_EQ(INITIALIZED, state_); | 311 DCHECK_EQ(INITIALIZED, state_); |
| 312 state_ = EXECUTING; |
326 | 313 |
327 BrowserThread::PostTask( | 314 BrowserThread::PostTask( |
328 BrowserThread::UI, | 315 BrowserThread::UI, |
329 FROM_HERE, | 316 FROM_HERE, |
330 base::Bind(&OperationRunner::WriteFileOnUIThread, | 317 base::Bind(&OperationRunner::WriteFileOnUIThread, |
331 runner_, | 318 runner_, |
332 buffer, | 319 buffer, |
333 current_offset_, | 320 current_offset_, |
334 buffer_length, | 321 buffer_length, |
335 base::Bind(&FileStreamWriter::OnWriteFileCompleted, | 322 base::Bind(&FileStreamWriter::OnWriteFileCompleted, |
336 weak_ptr_factory_.GetWeakPtr(), | 323 weak_ptr_factory_.GetWeakPtr(), |
337 buffer_length, | 324 buffer_length, |
338 callback))); | 325 callback))); |
339 } | 326 } |
340 | 327 |
341 } // namespace file_system_provider | 328 } // namespace file_system_provider |
342 } // namespace chromeos | 329 } // namespace chromeos |
OLD | NEW |