diff options
Diffstat (limited to 'webkit/appcache/appcache_response.cc')
-rw-r--r-- | webkit/appcache/appcache_response.cc | 262 |
1 files changed, 262 insertions, 0 deletions
diff --git a/webkit/appcache/appcache_response.cc b/webkit/appcache/appcache_response.cc new file mode 100644 index 0000000..3952d59 --- /dev/null +++ b/webkit/appcache/appcache_response.cc @@ -0,0 +1,262 @@ +// Copyright (c) 2009 The Chromium Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#include "webkit/appcache/appcache_response.h" + +#include "base/logging.h" +#include "base/message_loop.h" +#include "base/pickle.h" +#include "base/string_util.h" +#include "net/base/net_errors.h" +#include "net/base/io_buffer.h" +#include "net/disk_cache/disk_cache.h" +#include "webkit/appcache/appcache_service.h" + +using disk_cache::Entry; + +namespace appcache { + +namespace { + +// Disk cache entry data indices. +enum { + kResponseInfoIndex, + kResponseContentIndex +}; + +// Disk cache entry keys. +std::string response_key(int64 response_id) { + return Int64ToString(response_id); +} + +// An IOBuffer that wraps a pickle's data. Ownership of the +// pickle is transfered to the WrappedPickleIOBuffer object. +class WrappedPickleIOBuffer : public net::WrappedIOBuffer { + public: + explicit WrappedPickleIOBuffer(const Pickle* pickle) : + net::WrappedIOBuffer(reinterpret_cast<const char*>(pickle->data())), + pickle_(pickle) { + DCHECK(pickle->data()); + } + + private: + scoped_ptr<const Pickle> pickle_; +}; + +} // anon namespace + + +// AppCacheResponseInfo ---------------------------------------------- + +AppCacheResponseInfo::AppCacheResponseInfo( + AppCacheService* service, int64 response_id, + net::HttpResponseInfo* http_info) + : response_id_(response_id), http_response_info_(http_info), + service_(service) { + DCHECK(http_info); + DCHECK(response_id != kNoResponseId); + service_->storage()->working_set()->AddResponseInfo(this); +} + +AppCacheResponseInfo::~AppCacheResponseInfo() { + service_->storage()->working_set()->RemoveResponseInfo(this); +} + + +// AppCacheResponseIO ---------------------------------------------- + +AppCacheResponseIO::AppCacheResponseIO( + int64 response_id, disk_cache::Backend* disk_cache) + : response_id_(response_id), disk_cache_(disk_cache), + entry_(NULL), user_callback_(NULL), + ALLOW_THIS_IN_INITIALIZER_LIST(method_factory_(this)), + ALLOW_THIS_IN_INITIALIZER_LIST(raw_callback_( + new net::CancelableCompletionCallback<AppCacheResponseIO>( + this, &AppCacheResponseIO::OnRawIOComplete))) { +} + +AppCacheResponseIO::~AppCacheResponseIO() { + raw_callback_->Cancel(); + if (entry_) + entry_->Close(); +} + +void AppCacheResponseIO::ScheduleIOCompletionCallback(int result) { + MessageLoop::current()->PostTask(FROM_HERE, + method_factory_.NewRunnableMethod( + &AppCacheResponseIO::OnIOComplete, result)); +} + +void AppCacheResponseIO::InvokeUserCompletionCallback(int result) { + // Clear the user callback and buffers prior to invoking the callback + // so the caller can schedule additional operations in the callback. + buffer_ = NULL; + info_buffer_ = NULL; + net::CompletionCallback* temp_user_callback = user_callback_; + user_callback_ = NULL; + temp_user_callback->Run(result); +} + +void AppCacheResponseIO::ReadRaw(int index, int offset, + net::IOBuffer* buf, int buf_len) { + DCHECK(entry_); + raw_callback_->AddRef(); // Balanced in OnRawIOComplete. + int rv = entry_->ReadData(index, offset, buf, buf_len, raw_callback_); + if (rv != net::ERR_IO_PENDING) { + raw_callback_->Release(); + ScheduleIOCompletionCallback(rv); + } +} + +void AppCacheResponseIO::WriteRaw(int index, int offset, + net::IOBuffer* buf, int buf_len) { + DCHECK(entry_); + const bool kTruncate = true; + raw_callback_->AddRef(); // Balanced in OnRawIOComplete. + int rv = entry_->WriteData(index, offset, buf, buf_len, raw_callback_, + kTruncate); + if (rv != net::ERR_IO_PENDING) { + raw_callback_->Release(); + ScheduleIOCompletionCallback(rv); + } +} + +void AppCacheResponseIO::OnRawIOComplete(int result) { + raw_callback_->Release(); // Balance the AddRefs + OnIOComplete(result); +} + + +// AppCacheResponseReader ---------------------------------------------- + +void AppCacheResponseReader::ReadInfo(HttpResponseInfoIOBuffer* info_buf, + net::CompletionCallback* callback) { + DCHECK(callback && !IsReadPending()); + DCHECK(info_buf && !info_buf->http_info.get()); + DCHECK(!buffer_.get() && !info_buffer_.get()); + + user_callback_ = callback; // cleared on completion + + if (!OpenEntryIfNeeded()) { + ScheduleIOCompletionCallback(net::ERR_CACHE_MISS); + return; + } + + int size = entry_->GetDataSize(kResponseInfoIndex); + info_buffer_ = info_buf; + buffer_ = new net::IOBuffer(size); + ReadRaw(kResponseInfoIndex, 0, buffer_.get(), size); +} + +void AppCacheResponseReader::ReadData(net::IOBuffer* buf, int buf_len, + net::CompletionCallback* callback) { + DCHECK(callback && !IsReadPending()); + DCHECK(buf && (buf_len >= 0)); + DCHECK(!buffer_.get() && !info_buffer_.get()); + + user_callback_ = callback; // cleared on completion + + if (!OpenEntryIfNeeded()) { + ScheduleIOCompletionCallback(net::ERR_CACHE_MISS); + return; + } + + buffer_ = buf; + if (read_position_ + buf_len > range_length_) { + // TODO(michaeln): What about integer overflows? + DCHECK(range_length_ >= read_position_); + buf_len = range_length_ - read_position_; + } + ReadRaw(kResponseContentIndex, range_offset_ + read_position_, + buf, buf_len); +} + +void AppCacheResponseReader::SetReadRange(int offset, int length) { + DCHECK(!IsReadPending() && !read_position_); + range_offset_ = offset; + range_length_ = length; +} + +void AppCacheResponseReader::OnIOComplete(int result) { + if (result >= 0) { + if (info_buffer_.get()) { + // Allocate and deserialize the http info structure. + Pickle pickle(buffer_->data(), result); + bool response_truncated = false; + info_buffer_->http_info.reset(new net::HttpResponseInfo); + info_buffer_->http_info->InitFromPickle(pickle, &response_truncated); + DCHECK(!response_truncated); + } else { + read_position_ += result; + } + } + InvokeUserCompletionCallback(result); +} + +bool AppCacheResponseReader::OpenEntryIfNeeded() { + if (!entry_) + disk_cache_->OpenEntry(response_key(response_id_), &entry_); + return entry_ ? true : false; +} + + +// AppCacheResponseWriter ---------------------------------------------- + +void AppCacheResponseWriter::WriteInfo(HttpResponseInfoIOBuffer* info_buf, + net::CompletionCallback* callback) { + DCHECK(callback && !IsWritePending()); + DCHECK(info_buf && info_buf->http_info.get()); + DCHECK(!buffer_.get() && !info_buffer_.get()); + + user_callback_ = callback; // cleared on completion + + if (!CreateEntryIfNeeded()) { + ScheduleIOCompletionCallback(net::ERR_FAILED); + return; + } + + const bool kSkipTransientHeaders = true; + const bool kTruncated = false; + Pickle* pickle = new Pickle; + info_buf->http_info->Persist(pickle, kSkipTransientHeaders, kTruncated); + write_amount_ = static_cast<int>(pickle->size()); + buffer_ = new WrappedPickleIOBuffer(pickle); // takes ownership of pickle + WriteRaw(kResponseInfoIndex, 0, buffer_, write_amount_); +} + +void AppCacheResponseWriter::WriteData(net::IOBuffer* buf, int buf_len, + net::CompletionCallback* callback) { + DCHECK(callback && !IsWritePending()); + DCHECK(buf && (buf_len >= 0)); + DCHECK(!buffer_.get() && !info_buffer_.get()); + + user_callback_ = callback; // cleared on completion + + if (!CreateEntryIfNeeded()) { + ScheduleIOCompletionCallback(net::ERR_FAILED); + return; + } + + buffer_ = buf; + write_amount_ = buf_len; + WriteRaw(kResponseContentIndex, write_position_, buf, buf_len); +} + +void AppCacheResponseWriter::OnIOComplete(int result) { + if (result >= 0) { + DCHECK(write_amount_ == result); + if (!info_buffer_.get()) + write_position_ += result; + } + InvokeUserCompletionCallback(result); +} + +bool AppCacheResponseWriter::CreateEntryIfNeeded() { + if (!entry_) + disk_cache_->CreateEntry(response_key(response_id_), &entry_); + return entry_ ? true : false; +} + +} // namespace appcache + |