| 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 "base/bind.h" | |
| 6 #include "base/memory/weak_ptr.h" | |
| 7 #include "base/message_loop.h" | |
| 8 #include "webkit/chromeos/fileapi/memory_file_util.h" | |
| 9 | |
| 10 namespace { | |
| 11 const int kDefaultReadDirectoryBufferSize = 100; | |
| 12 } // namespace | |
| 13 | |
| 14 namespace fileapi { | |
| 15 | |
| 16 // In-memory implementation of AsyncFileStream. | |
| 17 class MemoryFileUtilAsyncFileStream : public AsyncFileStream { | |
| 18 public: | |
| 19 // |file_entry| is owned by MemoryFileUtil. | |
| 20 MemoryFileUtilAsyncFileStream(MemoryFileUtil::FileEntry* file_entry, | |
| 21 int flags) | |
| 22 : file_entry_(file_entry), | |
| 23 flags_(flags), | |
| 24 offset_(0), | |
| 25 weak_ptr_factory_(ALLOW_THIS_IN_INITIALIZER_LIST(this)) { | |
| 26 } | |
| 27 | |
| 28 // AsyncFileStream override. | |
| 29 virtual void Read(char* buffer, | |
| 30 int64 length, | |
| 31 const ReadWriteCallback& callback) OVERRIDE { | |
| 32 MessageLoop::current()->PostTask( | |
| 33 FROM_HERE, | |
| 34 base::Bind(&MemoryFileUtilAsyncFileStream::DoRead, | |
| 35 weak_ptr_factory_.GetWeakPtr(), | |
| 36 buffer, length, callback)); | |
| 37 } | |
| 38 | |
| 39 // AsyncFileStream override. | |
| 40 virtual void Write(const char* buffer, | |
| 41 int64 length, | |
| 42 const ReadWriteCallback& callback) OVERRIDE { | |
| 43 MessageLoop::current()->PostTask( | |
| 44 FROM_HERE, | |
| 45 base::Bind(&MemoryFileUtilAsyncFileStream::DoWrite, | |
| 46 weak_ptr_factory_.GetWeakPtr(), | |
| 47 buffer, length, callback)); | |
| 48 } | |
| 49 | |
| 50 // AsyncFileStream override. | |
| 51 virtual void Seek(int64 offset, | |
| 52 const SeekCallback& callback) OVERRIDE { | |
| 53 MessageLoop::current()->PostTask( | |
| 54 FROM_HERE, | |
| 55 base::Bind(&MemoryFileUtilAsyncFileStream::DoSeek, | |
| 56 weak_ptr_factory_.GetWeakPtr(), | |
| 57 offset, callback)); | |
| 58 } | |
| 59 | |
| 60 private: | |
| 61 // Callback used for Read(). | |
| 62 void DoRead(char* buffer, | |
| 63 int64 length, | |
| 64 const ReadWriteCallback& callback) { | |
| 65 | |
| 66 if ((flags_ & base::PLATFORM_FILE_READ) == 0) { | |
| 67 callback.Run(base::PLATFORM_FILE_ERROR_INVALID_OPERATION, 0); | |
| 68 return; | |
| 69 } | |
| 70 | |
| 71 // Shorten the length so the read does not overrun. | |
| 72 length = std::min(length, file_size() - offset_); | |
| 73 | |
| 74 const std::string& contents = file_entry_->contents; | |
| 75 std::copy(contents.begin() + offset_, | |
| 76 contents.begin() + offset_ + length, | |
| 77 buffer); | |
| 78 offset_ += length; | |
| 79 | |
| 80 callback.Run(base::PLATFORM_FILE_OK, length); | |
| 81 } | |
| 82 | |
| 83 // Callback used for Write(). | |
| 84 void DoWrite(const char* buffer, | |
| 85 int64 length, | |
| 86 const ReadWriteCallback& callback) { | |
| 87 if ((flags_ & base::PLATFORM_FILE_WRITE) == 0) { | |
| 88 callback.Run(base::PLATFORM_FILE_ERROR_INVALID_OPERATION, 0); | |
| 89 return; | |
| 90 } | |
| 91 | |
| 92 // Extend the contents if needed. | |
| 93 std::string* contents = &file_entry_->contents; | |
| 94 if (offset_ + length > file_size()) | |
| 95 contents->resize(offset_ + length, 0); // Fill with 0. | |
| 96 | |
| 97 std::copy(buffer, buffer + length, | |
| 98 contents->begin() + offset_); | |
| 99 file_entry_->last_modified = base::Time::Now(); | |
| 100 offset_ += length; | |
| 101 | |
| 102 callback.Run(base::PLATFORM_FILE_OK, length); | |
| 103 } | |
| 104 | |
| 105 // Callback used for Seek(). | |
| 106 void DoSeek(int64 offset, | |
| 107 const SeekCallback& callback) { | |
| 108 if (offset > file_size()) { | |
| 109 // Unlike lseek(2), we don't allow an offset larger than the file | |
| 110 // size for this file implementation. | |
| 111 callback.Run(base::PLATFORM_FILE_ERROR_INVALID_OPERATION); | |
| 112 return; | |
| 113 } | |
| 114 | |
| 115 offset_ = offset; | |
| 116 callback.Run(base::PLATFORM_FILE_OK); | |
| 117 } | |
| 118 | |
| 119 // Returns the file size as int64. | |
| 120 int64 file_size() const { | |
| 121 return static_cast<int64>(file_entry_->contents.size()); | |
| 122 } | |
| 123 | |
| 124 MemoryFileUtil::FileEntry* file_entry_; | |
| 125 const int flags_; | |
| 126 int64 offset_; | |
| 127 base::WeakPtrFactory<MemoryFileUtilAsyncFileStream> weak_ptr_factory_; | |
| 128 }; | |
| 129 | |
| 130 MemoryFileUtil::FileEntry::FileEntry() | |
| 131 : is_directory(false) { | |
| 132 } | |
| 133 | |
| 134 MemoryFileUtil::FileEntry::~FileEntry() { | |
| 135 } | |
| 136 | |
| 137 MemoryFileUtil::MemoryFileUtil(const base::FilePath& root_path) | |
| 138 : read_directory_buffer_size_(kDefaultReadDirectoryBufferSize) { | |
| 139 FileEntry root; | |
| 140 root.is_directory = true; | |
| 141 root.last_modified = base::Time::Now(); | |
| 142 | |
| 143 files_[root_path] = root; | |
| 144 } | |
| 145 | |
| 146 MemoryFileUtil::~MemoryFileUtil() { | |
| 147 } | |
| 148 | |
| 149 // Depending on the flags value the flow of file opening will be one of | |
| 150 // the following: | |
| 151 // | |
| 152 // PLATFORM_FILE_OPEN: | |
| 153 // - GetFileInfo | |
| 154 // - DidGetFileInfoForOpen | |
| 155 // - OpenVerifiedFile | |
| 156 // | |
| 157 // PLATFORM_FILE_CREATE: | |
| 158 // - Create | |
| 159 // - DidCreateOrTruncateForOpen | |
| 160 // - OpenVerifiedFile | |
| 161 // | |
| 162 // PLATFORM_FILE_OPEN_ALWAYS: | |
| 163 // - GetFileInfo | |
| 164 // - DidGetFileInfoForOpen | |
| 165 // - OpenVerifiedFile OR Create | |
| 166 // DidCreateOrTruncateForOpen | |
| 167 // OpenVerifiedFile | |
| 168 // | |
| 169 // PLATFORM_FILE_CREATE_ALWAYS: | |
| 170 // - Truncate | |
| 171 // - OpenTruncatedFileOrCreate | |
| 172 // - OpenVerifiedFile OR Create | |
| 173 // DidCreateOrTruncateForOpen | |
| 174 // OpenVerifiedFile | |
| 175 // | |
| 176 // PLATFORM_FILE_OPEN_TRUNCATED: | |
| 177 // - Truncate | |
| 178 // - DidCreateOrTruncateForOpen | |
| 179 // - OpenVerifiedFile | |
| 180 // | |
| 181 void MemoryFileUtil::Open( | |
| 182 const base::FilePath& file_path, | |
| 183 int flags, | |
| 184 const OpenCallback& callback) { | |
| 185 int create_flag = flags & (base::PLATFORM_FILE_OPEN | | |
| 186 base::PLATFORM_FILE_CREATE | | |
| 187 base::PLATFORM_FILE_OPEN_ALWAYS | | |
| 188 base::PLATFORM_FILE_CREATE_ALWAYS | | |
| 189 base::PLATFORM_FILE_OPEN_TRUNCATED); | |
| 190 switch (create_flag) { | |
| 191 case base::PLATFORM_FILE_OPEN: | |
| 192 case base::PLATFORM_FILE_OPEN_ALWAYS: | |
| 193 GetFileInfo( | |
| 194 file_path, | |
| 195 base::Bind(&MemoryFileUtil::DidGetFileInfoForOpen, | |
| 196 base::Unretained(this), file_path, flags, callback)); | |
| 197 | |
| 198 break; | |
| 199 | |
| 200 case base::PLATFORM_FILE_CREATE: | |
| 201 Create( | |
| 202 file_path, | |
| 203 base::Bind(&MemoryFileUtil::DidCreateOrTruncateForOpen, | |
| 204 base::Unretained(this), file_path, flags, 0, callback)); | |
| 205 break; | |
| 206 | |
| 207 case base::PLATFORM_FILE_CREATE_ALWAYS: | |
| 208 Truncate( | |
| 209 file_path, | |
| 210 0, | |
| 211 base::Bind(&MemoryFileUtil::OpenTruncatedFileOrCreate, | |
| 212 base::Unretained(this), file_path, flags, callback)); | |
| 213 break; | |
| 214 | |
| 215 case base::PLATFORM_FILE_OPEN_TRUNCATED: | |
| 216 Truncate( | |
| 217 file_path, | |
| 218 0, | |
| 219 base::Bind(&MemoryFileUtil::DidCreateOrTruncateForOpen, | |
| 220 base::Unretained(this), file_path, flags, 0, callback)); | |
| 221 break; | |
| 222 | |
| 223 default: | |
| 224 MessageLoop::current()->PostTask( | |
| 225 FROM_HERE, | |
| 226 base::Bind(callback, | |
| 227 base::PLATFORM_FILE_ERROR_INVALID_OPERATION, | |
| 228 static_cast<AsyncFileStream*>(NULL))); | |
| 229 } | |
| 230 } | |
| 231 | |
| 232 void MemoryFileUtil::GetFileInfo( | |
| 233 const base::FilePath& file_path, | |
| 234 const GetFileInfoCallback& callback) { | |
| 235 MessageLoop::current()->PostTask( | |
| 236 FROM_HERE, | |
| 237 base::Bind(&MemoryFileUtil::DoGetFileInfo, base::Unretained(this), | |
| 238 file_path.StripTrailingSeparators(), callback)); | |
| 239 } | |
| 240 | |
| 241 void MemoryFileUtil::Create( | |
| 242 const base::FilePath& file_path, | |
| 243 const StatusCallback& callback) { | |
| 244 MessageLoop::current()->PostTask( | |
| 245 FROM_HERE, | |
| 246 base::Bind(&MemoryFileUtil::DoCreate, base::Unretained(this), | |
| 247 file_path.StripTrailingSeparators(), false, callback)); | |
| 248 } | |
| 249 | |
| 250 void MemoryFileUtil::Truncate( | |
| 251 const base::FilePath& file_path, | |
| 252 int64 length, | |
| 253 const StatusCallback& callback) { | |
| 254 MessageLoop::current()->PostTask( | |
| 255 FROM_HERE, | |
| 256 base::Bind(&MemoryFileUtil::DoTruncate, base::Unretained(this), | |
| 257 file_path.StripTrailingSeparators(), length, callback)); | |
| 258 } | |
| 259 | |
| 260 void MemoryFileUtil::Touch( | |
| 261 const base::FilePath& file_path, | |
| 262 const base::Time& last_access_time, | |
| 263 const base::Time& last_modified_time, | |
| 264 const StatusCallback& callback) { | |
| 265 MessageLoop::current()->PostTask( | |
| 266 FROM_HERE, | |
| 267 base::Bind(&MemoryFileUtil::DoTouch, base::Unretained(this), | |
| 268 file_path.StripTrailingSeparators(), | |
| 269 last_modified_time, callback)); | |
| 270 } | |
| 271 | |
| 272 void MemoryFileUtil::Remove( | |
| 273 const base::FilePath& file_path, | |
| 274 bool recursive, | |
| 275 const StatusCallback& callback) { | |
| 276 if (recursive) { | |
| 277 MessageLoop::current()->PostTask( | |
| 278 FROM_HERE, | |
| 279 base::Bind(&MemoryFileUtil::DoRemoveRecursive, | |
| 280 base::Unretained(this), file_path.StripTrailingSeparators(), | |
| 281 callback)); | |
| 282 } else { | |
| 283 MessageLoop::current()->PostTask( | |
| 284 FROM_HERE, | |
| 285 base::Bind(&MemoryFileUtil::DoRemoveSingleFile, | |
| 286 base::Unretained(this), file_path.StripTrailingSeparators(), | |
| 287 callback)); | |
| 288 } | |
| 289 } | |
| 290 | |
| 291 void MemoryFileUtil::CreateDirectory( | |
| 292 const base::FilePath& dir_path, | |
| 293 const StatusCallback& callback) { | |
| 294 MessageLoop::current()->PostTask( | |
| 295 FROM_HERE, | |
| 296 base::Bind(&MemoryFileUtil::DoCreate, | |
| 297 base::Unretained(this), dir_path.StripTrailingSeparators(), | |
| 298 true, callback)); | |
| 299 } | |
| 300 | |
| 301 void MemoryFileUtil::ReadDirectory( | |
| 302 const base::FilePath& dir_path, | |
| 303 const ReadDirectoryCallback& callback) { | |
| 304 MessageLoop::current()->PostTask( | |
| 305 FROM_HERE, | |
| 306 base::Bind(&MemoryFileUtil::DoReadDirectory, | |
| 307 base::Unretained(this), dir_path.StripTrailingSeparators(), | |
| 308 base::FilePath(), callback)); | |
| 309 } | |
| 310 | |
| 311 void MemoryFileUtil::DoGetFileInfo(const base::FilePath& file_path, | |
| 312 const GetFileInfoCallback& callback) { | |
| 313 base::PlatformFileInfo file_info; | |
| 314 | |
| 315 FileIterator file_it = files_.find(file_path); | |
| 316 | |
| 317 if (file_it == files_.end()) { | |
| 318 callback.Run(base::PLATFORM_FILE_ERROR_NOT_FOUND, file_info); | |
| 319 return; | |
| 320 } | |
| 321 const FileEntry& file_entry = file_it->second; | |
| 322 | |
| 323 file_info.size = file_entry.contents.size(); | |
| 324 file_info.is_directory = file_entry.is_directory; | |
| 325 file_info.is_symbolic_link = false; | |
| 326 | |
| 327 // In this file system implementation we store only one datetime. Many | |
| 328 // popular file systems do the same. | |
| 329 file_info.last_modified = file_entry.last_modified; | |
| 330 file_info.last_accessed = file_entry.last_modified; | |
| 331 file_info.creation_time = file_entry.last_modified; | |
| 332 | |
| 333 callback.Run(base::PLATFORM_FILE_OK, file_info); | |
| 334 } | |
| 335 | |
| 336 void MemoryFileUtil::DoCreate( | |
| 337 const base::FilePath& file_path, | |
| 338 bool is_directory, | |
| 339 const StatusCallback& callback) { | |
| 340 if (FileExists(file_path)) { | |
| 341 callback.Run(base::PLATFORM_FILE_ERROR_EXISTS); | |
| 342 return; | |
| 343 } | |
| 344 | |
| 345 if (!IsDirectory(file_path.DirName())) { | |
| 346 callback.Run(base::PLATFORM_FILE_ERROR_FAILED); | |
| 347 return; | |
| 348 } | |
| 349 | |
| 350 FileEntry file; | |
| 351 file.is_directory = is_directory; | |
| 352 file.last_modified = base::Time::Now(); | |
| 353 | |
| 354 files_[file_path] = file; | |
| 355 callback.Run(base::PLATFORM_FILE_OK); | |
| 356 } | |
| 357 | |
| 358 void MemoryFileUtil::DoTruncate( | |
| 359 const base::FilePath& file_path, | |
| 360 int64 length, | |
| 361 const StatusCallback& callback) { | |
| 362 FileIterator file_it = files_.find(file_path); | |
| 363 if (file_it == files_.end()) { | |
| 364 callback.Run(base::PLATFORM_FILE_ERROR_NOT_FOUND); | |
| 365 return; | |
| 366 } | |
| 367 | |
| 368 FileEntry& file = file_it->second; | |
| 369 | |
| 370 // Fill the extended part with 0 if |length| is larger than the original | |
| 371 // contents size. | |
| 372 file.contents.resize(length, 0); | |
| 373 callback.Run(base::PLATFORM_FILE_OK); | |
| 374 } | |
| 375 | |
| 376 void MemoryFileUtil::DoTouch( | |
| 377 const base::FilePath& file_path, | |
| 378 const base::Time& last_modified_time, | |
| 379 const StatusCallback& callback) { | |
| 380 FileIterator file_it = files_.find(file_path); | |
| 381 if (file_it == files_.end()) { | |
| 382 callback.Run(base::PLATFORM_FILE_ERROR_NOT_FOUND); | |
| 383 return; | |
| 384 } | |
| 385 | |
| 386 FileEntry& file = file_it->second; | |
| 387 | |
| 388 file.last_modified = last_modified_time; | |
| 389 callback.Run(base::PLATFORM_FILE_OK); | |
| 390 } | |
| 391 | |
| 392 void MemoryFileUtil::DoRemoveSingleFile( | |
| 393 const base::FilePath& file_path, | |
| 394 const StatusCallback& callback) { | |
| 395 FileIterator file_it = files_.find(file_path); | |
| 396 if (file_it == files_.end()) { | |
| 397 callback.Run(base::PLATFORM_FILE_ERROR_NOT_FOUND); | |
| 398 return; | |
| 399 } | |
| 400 | |
| 401 FileEntry& file = file_it->second; | |
| 402 if (file.is_directory) { | |
| 403 // Check if the directory is empty. This can be done by checking if | |
| 404 // the next file is present under the directory. Note that |files_| is | |
| 405 // a map hence the file names are sorted by names. | |
| 406 FileIterator tmp_it = file_it; | |
| 407 ++tmp_it; | |
| 408 if (tmp_it != files_.end() && file_path.IsParent(tmp_it->first)) { | |
| 409 callback.Run(base::PLATFORM_FILE_ERROR_NOT_A_FILE); | |
| 410 return; | |
| 411 } | |
| 412 } | |
| 413 | |
| 414 files_.erase(file_it); | |
| 415 callback.Run(base::PLATFORM_FILE_OK); | |
| 416 } | |
| 417 | |
| 418 void MemoryFileUtil::DoRemoveRecursive( | |
| 419 const base::FilePath& file_path, | |
| 420 const StatusCallback& callback) { | |
| 421 FileIterator file_it = files_.find(file_path); | |
| 422 if (file_it == files_.end()) { | |
| 423 callback.Run(base::PLATFORM_FILE_ERROR_NOT_FOUND); | |
| 424 return; | |
| 425 } | |
| 426 | |
| 427 FileEntry& file = file_it->second; | |
| 428 if (!file.is_directory) { | |
| 429 files_.erase(file_it); | |
| 430 callback.Run(base::PLATFORM_FILE_OK); | |
| 431 return; | |
| 432 } | |
| 433 | |
| 434 // Remove the directory itself. | |
| 435 files_.erase(file_it++); | |
| 436 // Remove files under the directory. | |
| 437 while (file_it != files_.end()) { | |
| 438 if (file_path.IsParent(file_it->first)) { | |
| 439 files_.erase(file_it++); | |
| 440 } else { | |
| 441 break; | |
| 442 } | |
| 443 } | |
| 444 callback.Run(base::PLATFORM_FILE_OK); | |
| 445 } | |
| 446 | |
| 447 void MemoryFileUtil::DoReadDirectory( | |
| 448 const base::FilePath& dir_path, | |
| 449 const base::FilePath& in_from, | |
| 450 const ReadDirectoryCallback& callback) { | |
| 451 base::FilePath from = in_from; | |
| 452 read_directory_buffer_.clear(); | |
| 453 | |
| 454 if (!FileExists(dir_path)) { | |
| 455 callback.Run(base::PLATFORM_FILE_ERROR_NOT_FOUND, | |
| 456 read_directory_buffer_, true); | |
| 457 return; | |
| 458 } | |
| 459 | |
| 460 if (!IsDirectory(dir_path)) { | |
| 461 callback.Run(base::PLATFORM_FILE_ERROR_NOT_A_DIRECTORY, | |
| 462 read_directory_buffer_, true); | |
| 463 return; | |
| 464 } | |
| 465 | |
| 466 if (from.empty()) | |
| 467 from = dir_path; | |
| 468 | |
| 469 bool completed = true; | |
| 470 | |
| 471 // Here we iterate over all paths in files_ starting with the prefix |from|. | |
| 472 // It is not very efficient in case of a deep tree with many files in | |
| 473 // subdirectories. If ever we'll need efficiency from this implementation of | |
| 474 // FS, this should be changed. | |
| 475 for (ConstFileIterator it = files_.lower_bound(from); | |
| 476 it != files_.end(); | |
| 477 ++it) { | |
| 478 if (dir_path == it->first) // Skip the directory itself. | |
| 479 continue; | |
| 480 if (!dir_path.IsParent(it->first)) // Not in the directory. | |
| 481 break; | |
| 482 if (it->first.DirName() != dir_path) // a file in subdirectory | |
| 483 continue; | |
| 484 | |
| 485 if (read_directory_buffer_.size() == read_directory_buffer_size_) { | |
| 486 from = it->first; | |
| 487 completed = false; | |
| 488 break; | |
| 489 } | |
| 490 | |
| 491 const FileEntry& file = it->second; | |
| 492 DirectoryEntry entry; | |
| 493 entry.name = it->first.BaseName().value(); | |
| 494 entry.is_directory = file.is_directory; | |
| 495 entry.size = file.contents.size(); | |
| 496 entry.last_modified_time = file.last_modified; | |
| 497 | |
| 498 read_directory_buffer_.push_back(entry); | |
| 499 } | |
| 500 | |
| 501 callback.Run(base::PLATFORM_FILE_OK, read_directory_buffer_, completed); | |
| 502 | |
| 503 if (!completed) { | |
| 504 MessageLoop::current()->PostTask( | |
| 505 FROM_HERE, | |
| 506 base::Bind(&MemoryFileUtil::DoReadDirectory, | |
| 507 base::Unretained(this), dir_path, | |
| 508 from, callback)); | |
| 509 } | |
| 510 } | |
| 511 | |
| 512 void MemoryFileUtil::OpenVerifiedFile( | |
| 513 const base::FilePath& file_path, | |
| 514 int flags, | |
| 515 const OpenCallback& callback) { | |
| 516 FileIterator file_it = files_.find(file_path); | |
| 517 // The existence of the file is guranteed here. | |
| 518 DCHECK(file_it != files_.end()); | |
| 519 | |
| 520 FileEntry* file_entry = &file_it->second; | |
| 521 callback.Run(base::PLATFORM_FILE_OK, | |
| 522 new MemoryFileUtilAsyncFileStream(file_entry, flags)); | |
| 523 } | |
| 524 | |
| 525 void MemoryFileUtil::DidGetFileInfoForOpen( | |
| 526 const base::FilePath& file_path, | |
| 527 int flags, | |
| 528 const OpenCallback& callback, | |
| 529 PlatformFileError get_info_result, | |
| 530 const base::PlatformFileInfo& file_info) { | |
| 531 if (get_info_result == base::PLATFORM_FILE_OK && file_info.is_directory) { | |
| 532 callback.Run(base::PLATFORM_FILE_ERROR_NOT_A_FILE, NULL); | |
| 533 return; | |
| 534 } | |
| 535 | |
| 536 if (get_info_result == base::PLATFORM_FILE_OK) { | |
| 537 OpenVerifiedFile(file_path, flags, callback); | |
| 538 return; | |
| 539 } | |
| 540 | |
| 541 if (get_info_result == base::PLATFORM_FILE_ERROR_NOT_FOUND && | |
| 542 flags & base::PLATFORM_FILE_CREATE_ALWAYS) { | |
| 543 Create(file_path, | |
| 544 base::Bind(&MemoryFileUtil::DidCreateOrTruncateForOpen, | |
| 545 base::Unretained(this), file_path, flags, 0, callback)); | |
| 546 return; | |
| 547 } | |
| 548 | |
| 549 callback.Run(get_info_result, NULL); | |
| 550 } | |
| 551 | |
| 552 void MemoryFileUtil::OpenTruncatedFileOrCreate( | |
| 553 const base::FilePath& file_path, | |
| 554 int flags, | |
| 555 const OpenCallback& callback, | |
| 556 PlatformFileError result) { | |
| 557 if (result == base::PLATFORM_FILE_OK) { | |
| 558 OpenVerifiedFile(file_path, flags, callback); | |
| 559 return; | |
| 560 } | |
| 561 | |
| 562 if (result == base::PLATFORM_FILE_ERROR_NOT_FOUND) { | |
| 563 Create( | |
| 564 file_path, | |
| 565 base::Bind(&MemoryFileUtil::DidCreateOrTruncateForOpen, | |
| 566 base::Unretained(this), file_path, flags, 0, callback)); | |
| 567 return; | |
| 568 } | |
| 569 | |
| 570 callback.Run(result, NULL); | |
| 571 } | |
| 572 | |
| 573 void MemoryFileUtil::DidCreateOrTruncateForOpen( | |
| 574 const base::FilePath& file_path, | |
| 575 int flags, | |
| 576 int64 size, | |
| 577 const OpenCallback& callback, | |
| 578 PlatformFileError result) { | |
| 579 if (result != base::PLATFORM_FILE_OK) { | |
| 580 callback.Run(result, NULL); | |
| 581 return; | |
| 582 } | |
| 583 | |
| 584 OpenVerifiedFile(file_path, flags, callback); | |
| 585 } | |
| 586 | |
| 587 } // namespace fileapi | |
| OLD | NEW |