mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-11-28 16:26:10 +03:00
488 lines
18 KiB
C
488 lines
18 KiB
C
|
// Copyright (c) 2013 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.
|
||
|
|
||
|
#ifndef NET_DISK_CACHE_SIMPLE_SIMPLE_SYNCHRONOUS_ENTRY_H_
|
||
|
#define NET_DISK_CACHE_SIMPLE_SIMPLE_SYNCHRONOUS_ENTRY_H_
|
||
|
|
||
|
#include <stdint.h>
|
||
|
|
||
|
#include <algorithm>
|
||
|
#include <map>
|
||
|
#include <memory>
|
||
|
#include <string>
|
||
|
#include <utility>
|
||
|
#include <vector>
|
||
|
|
||
|
#include "base/feature_list.h"
|
||
|
#include "base/files/file.h"
|
||
|
#include "base/files/file_path.h"
|
||
|
#include "base/gtest_prod_util.h"
|
||
|
#include "base/memory/ref_counted.h"
|
||
|
#include "base/strings/string_piece_forward.h"
|
||
|
#include "base/time/time.h"
|
||
|
#include "net/base/cache_type.h"
|
||
|
#include "net/base/net_export.h"
|
||
|
#include "net/disk_cache/simple/simple_entry_format.h"
|
||
|
#include "net/disk_cache/simple/simple_file_tracker.h"
|
||
|
|
||
|
namespace net {
|
||
|
class GrowableIOBuffer;
|
||
|
class IOBuffer;
|
||
|
}
|
||
|
|
||
|
FORWARD_DECLARE_TEST(DiskCacheBackendTest, SimpleCacheEnumerationLongKeys);
|
||
|
|
||
|
namespace disk_cache {
|
||
|
|
||
|
NET_EXPORT_PRIVATE extern const base::Feature kSimpleCachePrefetchExperiment;
|
||
|
NET_EXPORT_PRIVATE extern const char kSimplePrefetchBytesParam[];
|
||
|
|
||
|
// Returns how large a file would get prefetched on reading the entry.
|
||
|
// If the experiment is disabled, returns 0.
|
||
|
NET_EXPORT_PRIVATE int GetSimpleCachePrefetchSize();
|
||
|
|
||
|
class SimpleSynchronousEntry;
|
||
|
|
||
|
// This class handles the passing of data about the entry between
|
||
|
// SimpleEntryImplementation and SimpleSynchronousEntry and the computation of
|
||
|
// file offsets based on the data size for all streams.
|
||
|
class NET_EXPORT_PRIVATE SimpleEntryStat {
|
||
|
public:
|
||
|
SimpleEntryStat(base::Time last_used,
|
||
|
base::Time last_modified,
|
||
|
const int32_t data_size[],
|
||
|
const int32_t sparse_data_size);
|
||
|
|
||
|
int GetOffsetInFile(size_t key_length, int offset, int stream_index) const;
|
||
|
int GetEOFOffsetInFile(size_t key_length, int stream_index) const;
|
||
|
int GetLastEOFOffsetInFile(size_t key_length, int file_index) const;
|
||
|
int64_t GetFileSize(size_t key_length, int file_index) const;
|
||
|
|
||
|
base::Time last_used() const { return last_used_; }
|
||
|
base::Time last_modified() const { return last_modified_; }
|
||
|
void set_last_used(base::Time last_used) { last_used_ = last_used; }
|
||
|
void set_last_modified(base::Time last_modified) {
|
||
|
last_modified_ = last_modified;
|
||
|
}
|
||
|
|
||
|
int32_t data_size(int stream_index) const { return data_size_[stream_index]; }
|
||
|
void set_data_size(int stream_index, int data_size) {
|
||
|
data_size_[stream_index] = data_size;
|
||
|
}
|
||
|
|
||
|
int32_t sparse_data_size() const { return sparse_data_size_; }
|
||
|
void set_sparse_data_size(int32_t sparse_data_size) {
|
||
|
sparse_data_size_ = sparse_data_size;
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
base::Time last_used_;
|
||
|
base::Time last_modified_;
|
||
|
int32_t data_size_[kSimpleEntryStreamCount];
|
||
|
int32_t sparse_data_size_;
|
||
|
};
|
||
|
|
||
|
struct SimpleStreamPrefetchData {
|
||
|
SimpleStreamPrefetchData();
|
||
|
~SimpleStreamPrefetchData();
|
||
|
|
||
|
scoped_refptr<net::GrowableIOBuffer> data;
|
||
|
uint32_t stream_crc32;
|
||
|
};
|
||
|
|
||
|
struct SimpleEntryCreationResults {
|
||
|
explicit SimpleEntryCreationResults(SimpleEntryStat entry_stat);
|
||
|
~SimpleEntryCreationResults();
|
||
|
|
||
|
SimpleSynchronousEntry* sync_entry;
|
||
|
|
||
|
// Expectation is that [0] will always be filled in, but [1] might not be.
|
||
|
SimpleStreamPrefetchData stream_prefetch_data[2];
|
||
|
|
||
|
SimpleEntryStat entry_stat;
|
||
|
int result;
|
||
|
};
|
||
|
|
||
|
// Worker thread interface to the very simple cache. This interface is not
|
||
|
// thread safe, and callers must ensure that it is only ever accessed from
|
||
|
// a single thread between synchronization points.
|
||
|
class SimpleSynchronousEntry {
|
||
|
public:
|
||
|
struct CRCRecord {
|
||
|
CRCRecord();
|
||
|
CRCRecord(int index_p, bool has_crc32_p, uint32_t data_crc32_p);
|
||
|
|
||
|
int index;
|
||
|
bool has_crc32;
|
||
|
uint32_t data_crc32;
|
||
|
};
|
||
|
|
||
|
struct ReadRequest {
|
||
|
// Also sets request_update_crc to false.
|
||
|
ReadRequest(int index_p, int offset_p, int buf_len_p);
|
||
|
int index;
|
||
|
int offset;
|
||
|
int buf_len;
|
||
|
|
||
|
// Partial CRC of data immediately preceeding this read. Only relevant if
|
||
|
// request_update_crc is set.
|
||
|
uint32_t previous_crc32;
|
||
|
bool request_update_crc;
|
||
|
bool request_verify_crc; // only relevant if request_update_crc is set
|
||
|
};
|
||
|
|
||
|
struct ReadResult {
|
||
|
ReadResult()
|
||
|
: crc_updated(false),
|
||
|
crc_performed_verify(false),
|
||
|
crc_verify_ok(false) {}
|
||
|
int result;
|
||
|
uint32_t updated_crc32; // only relevant if crc_updated set
|
||
|
bool crc_updated;
|
||
|
bool crc_performed_verify; // only relevant if crc_updated set
|
||
|
bool crc_verify_ok; // only relevant if crc_updated set
|
||
|
};
|
||
|
|
||
|
struct WriteRequest {
|
||
|
WriteRequest(int index_p,
|
||
|
int offset_p,
|
||
|
int buf_len_p,
|
||
|
uint32_t previous_crc32_p,
|
||
|
bool truncate_p,
|
||
|
bool doomed_p,
|
||
|
bool request_update_crc_p);
|
||
|
int index;
|
||
|
int offset;
|
||
|
int buf_len;
|
||
|
uint32_t previous_crc32;
|
||
|
bool truncate;
|
||
|
bool doomed;
|
||
|
bool request_update_crc;
|
||
|
};
|
||
|
|
||
|
struct WriteResult {
|
||
|
WriteResult() : crc_updated(false) {}
|
||
|
int result;
|
||
|
uint32_t updated_crc32; // only relevant if crc_updated set
|
||
|
bool crc_updated;
|
||
|
};
|
||
|
|
||
|
struct SparseRequest {
|
||
|
SparseRequest(int64_t sparse_offset_p, int buf_len_p);
|
||
|
|
||
|
int64_t sparse_offset;
|
||
|
int buf_len;
|
||
|
};
|
||
|
|
||
|
// Opens a disk cache entry on disk. The |key| parameter is optional, if empty
|
||
|
// the operation may be slower. The |entry_hash| parameter is required.
|
||
|
// |had_index| is provided only for histograms.
|
||
|
// |time_enqueued| is when this operation was added to the I/O thread pool,
|
||
|
// and is provided only for histograms.
|
||
|
static void OpenEntry(net::CacheType cache_type,
|
||
|
const base::FilePath& path,
|
||
|
const std::string& key,
|
||
|
uint64_t entry_hash,
|
||
|
bool had_index,
|
||
|
const base::TimeTicks& time_enqueued,
|
||
|
SimpleFileTracker* file_tracker,
|
||
|
SimpleEntryCreationResults* out_results);
|
||
|
|
||
|
static void CreateEntry(net::CacheType cache_type,
|
||
|
const base::FilePath& path,
|
||
|
const std::string& key,
|
||
|
uint64_t entry_hash,
|
||
|
bool had_index,
|
||
|
const base::TimeTicks& time_enqueued,
|
||
|
SimpleFileTracker* file_tracker,
|
||
|
SimpleEntryCreationResults* out_results);
|
||
|
|
||
|
// Renames the entry on the file system, making it no longer possible to open
|
||
|
// it again, but allowing operations to continue to be executed through that
|
||
|
// instance. The renamed file will be removed once the entry is closed.
|
||
|
// Returns a net error code.
|
||
|
int Doom();
|
||
|
|
||
|
// Deletes an entry from the file system. This variant should only be used
|
||
|
// if there is no actual open instance around, as it doesn't account for
|
||
|
// possibility of it having been renamed to a non-standard name.
|
||
|
static int DeleteEntryFiles(const base::FilePath& path,
|
||
|
net::CacheType cache_type,
|
||
|
uint64_t entry_hash);
|
||
|
|
||
|
// Like |DeleteEntryFiles()| above, except that it truncates the entry files
|
||
|
// rather than deleting them. Used when dooming entries after the backend has
|
||
|
// shutdown. See implementation of |SimpleEntryImpl::DoomEntryInternal()| for
|
||
|
// more.
|
||
|
static int TruncateEntryFiles(const base::FilePath& path,
|
||
|
uint64_t entry_hash);
|
||
|
|
||
|
// Like |DeleteEntryFiles()| above. Deletes all entries corresponding to the
|
||
|
// |key_hashes|. Succeeds only when all entries are deleted. Returns a net
|
||
|
// error code.
|
||
|
static int DeleteEntrySetFiles(const std::vector<uint64_t>* key_hashes,
|
||
|
const base::FilePath& path);
|
||
|
|
||
|
// N.B. ReadData(), WriteData(), CheckEOFRecord(), ReadSparseData(),
|
||
|
// WriteSparseData() and Close() may block on IO.
|
||
|
//
|
||
|
// All of these methods will put the //net return value into |*out_result|.
|
||
|
|
||
|
void ReadData(const ReadRequest& in_entry_op,
|
||
|
SimpleEntryStat* entry_stat,
|
||
|
net::IOBuffer* out_buf,
|
||
|
ReadResult* out_result);
|
||
|
void WriteData(const WriteRequest& in_entry_op,
|
||
|
net::IOBuffer* in_buf,
|
||
|
SimpleEntryStat* out_entry_stat,
|
||
|
WriteResult* out_write_result);
|
||
|
int CheckEOFRecord(base::File* file,
|
||
|
int stream_index,
|
||
|
const SimpleEntryStat& entry_stat,
|
||
|
uint32_t expected_crc32);
|
||
|
|
||
|
void ReadSparseData(const SparseRequest& in_entry_op,
|
||
|
net::IOBuffer* out_buf,
|
||
|
base::Time* out_last_used,
|
||
|
int* out_result);
|
||
|
void WriteSparseData(const SparseRequest& in_entry_op,
|
||
|
net::IOBuffer* in_buf,
|
||
|
uint64_t max_sparse_data_size,
|
||
|
SimpleEntryStat* out_entry_stat,
|
||
|
int* out_result);
|
||
|
void GetAvailableRange(const SparseRequest& in_entry_op,
|
||
|
int64_t* out_start,
|
||
|
int* out_result);
|
||
|
|
||
|
// Close all streams, and add write EOF records to streams indicated by the
|
||
|
// CRCRecord entries in |crc32s_to_write|.
|
||
|
void Close(const SimpleEntryStat& entry_stat,
|
||
|
std::unique_ptr<std::vector<CRCRecord>> crc32s_to_write,
|
||
|
net::GrowableIOBuffer* stream_0_data);
|
||
|
|
||
|
const base::FilePath& path() const { return path_; }
|
||
|
std::string key() const { return key_; }
|
||
|
const SimpleFileTracker::EntryFileKey& entry_file_key() const {
|
||
|
return entry_file_key_;
|
||
|
}
|
||
|
|
||
|
NET_EXPORT_PRIVATE base::FilePath GetFilenameForSubfile(
|
||
|
SimpleFileTracker::SubFile sub_file) const;
|
||
|
|
||
|
private:
|
||
|
FRIEND_TEST_ALL_PREFIXES(::DiskCacheBackendTest,
|
||
|
SimpleCacheEnumerationLongKeys);
|
||
|
friend class SimpleFileTrackerTest;
|
||
|
|
||
|
enum CreateEntryResult {
|
||
|
CREATE_ENTRY_SUCCESS = 0,
|
||
|
CREATE_ENTRY_PLATFORM_FILE_ERROR = 1,
|
||
|
CREATE_ENTRY_CANT_WRITE_HEADER = 2,
|
||
|
CREATE_ENTRY_CANT_WRITE_KEY = 3,
|
||
|
CREATE_ENTRY_MAX = 4,
|
||
|
};
|
||
|
|
||
|
enum FileRequired {
|
||
|
FILE_NOT_REQUIRED,
|
||
|
FILE_REQUIRED
|
||
|
};
|
||
|
|
||
|
struct SparseRange {
|
||
|
int64_t offset;
|
||
|
int64_t length;
|
||
|
uint32_t data_crc32;
|
||
|
int64_t file_offset;
|
||
|
|
||
|
bool operator<(const SparseRange& other) const {
|
||
|
return offset < other.offset;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// When opening an entry without knowing the key, the header must be read
|
||
|
// without knowing the size of the key. This is how much to read initially, to
|
||
|
// make it likely the entire key is read.
|
||
|
static const size_t kInitialHeaderRead = 64 * 1024;
|
||
|
|
||
|
NET_EXPORT_PRIVATE SimpleSynchronousEntry(
|
||
|
net::CacheType cache_type,
|
||
|
const base::FilePath& path,
|
||
|
const std::string& key,
|
||
|
uint64_t entry_hash,
|
||
|
bool had_index,
|
||
|
SimpleFileTracker* simple_file_tracker);
|
||
|
|
||
|
// Like Entry, the SimpleSynchronousEntry self releases when Close() is
|
||
|
// called.
|
||
|
NET_EXPORT_PRIVATE ~SimpleSynchronousEntry();
|
||
|
|
||
|
// Tries to open one of the cache entry files. Succeeds if the open succeeds
|
||
|
// or if the file was not found and is allowed to be omitted if the
|
||
|
// corresponding stream is empty.
|
||
|
bool MaybeOpenFile(int file_index,
|
||
|
base::File::Error* out_error);
|
||
|
// Creates one of the cache entry files if necessary. If the file is allowed
|
||
|
// to be omitted if the corresponding stream is empty, and if |file_required|
|
||
|
// is FILE_NOT_REQUIRED, then the file is not created; otherwise, it is.
|
||
|
bool MaybeCreateFile(int file_index,
|
||
|
FileRequired file_required,
|
||
|
base::File::Error* out_error);
|
||
|
bool OpenFiles(SimpleEntryStat* out_entry_stat);
|
||
|
bool CreateFiles(SimpleEntryStat* out_entry_stat);
|
||
|
void CloseFile(int index);
|
||
|
void CloseFiles();
|
||
|
|
||
|
// Read the header and key at the beginning of the file, and validate that
|
||
|
// they are correct. If this entry was opened with a key, the key is checked
|
||
|
// for a match. If not, then the |key_| member is set based on the value in
|
||
|
// this header. Records histograms if any check is failed.
|
||
|
bool CheckHeaderAndKey(base::File* file, int file_index);
|
||
|
|
||
|
// Returns a net error, i.e. net::OK on success.
|
||
|
int InitializeForOpen(SimpleEntryStat* out_entry_stat,
|
||
|
SimpleStreamPrefetchData stream_prefetch_data[2]);
|
||
|
|
||
|
// Writes the header and key to a newly-created stream file. |index| is the
|
||
|
// index of the stream. Returns true on success; returns false and sets
|
||
|
// |*out_result| on failure.
|
||
|
bool InitializeCreatedFile(int index, CreateEntryResult* out_result);
|
||
|
|
||
|
// Returns a net error, including net::OK on success and net::FILE_EXISTS
|
||
|
// when the entry already exists.
|
||
|
int InitializeForCreate(SimpleEntryStat* out_entry_stat);
|
||
|
|
||
|
// Allocates and fills a buffer with stream 0 data in |stream_0_data|, then
|
||
|
// checks its crc32. May also optionally read in |stream_1_data| and its
|
||
|
// crc, but might decide not to.
|
||
|
int ReadAndValidateStream0AndMaybe1(
|
||
|
int file_size,
|
||
|
SimpleEntryStat* out_entry_stat,
|
||
|
SimpleStreamPrefetchData stream_prefetch_data[2]);
|
||
|
|
||
|
// Reads the EOF record located at |file_offset| in file |file_index|,
|
||
|
// with |file_0_prefetch| potentially having prefetched file 0 content.
|
||
|
// Puts the result into |*eof_record| and sanity-checks it.
|
||
|
// Returns net status, and records any failures to UMA.
|
||
|
int GetEOFRecordData(base::File* file,
|
||
|
base::StringPiece file_0_prefetch,
|
||
|
int file_index,
|
||
|
int file_offset,
|
||
|
SimpleFileEOF* eof_record);
|
||
|
|
||
|
// Reads either from |file_0_prefetch| or |file|.
|
||
|
// Range-checks all the in-memory reads.
|
||
|
bool ReadFromFileOrPrefetched(base::File* file,
|
||
|
base::StringPiece file_0_prefetch,
|
||
|
int file_index,
|
||
|
int offset,
|
||
|
int size,
|
||
|
char* dest);
|
||
|
|
||
|
// Extracts out the payload of stream |stream_index|, reading either from
|
||
|
// |file_0_prefetch|, if available, or |file|. |entry_stat| will be used to
|
||
|
// determine file layout, though |extra_size| additional bytes will be read
|
||
|
// past the stream payload end.
|
||
|
//
|
||
|
// |*stream_data| will be pointed to a fresh buffer with the results,
|
||
|
// and |*out_crc32| will get the checksum, which will be verified against
|
||
|
// |eof_record|.
|
||
|
int PreReadStreamPayload(base::File* file,
|
||
|
base::StringPiece file_0_prefetch,
|
||
|
int stream_index,
|
||
|
int extra_size,
|
||
|
const SimpleEntryStat& entry_stat,
|
||
|
const SimpleFileEOF& eof_record,
|
||
|
SimpleStreamPrefetchData* out);
|
||
|
|
||
|
// Opens the sparse data file and scans it if it exists.
|
||
|
bool OpenSparseFileIfExists(int32_t* out_sparse_data_size);
|
||
|
|
||
|
// Creates and initializes the sparse data file.
|
||
|
bool CreateSparseFile();
|
||
|
|
||
|
// Closes the sparse data file.
|
||
|
void CloseSparseFile();
|
||
|
|
||
|
// Writes the header to the (newly-created) sparse file.
|
||
|
bool InitializeSparseFile(base::File* file);
|
||
|
|
||
|
// Removes all but the header of the sparse file.
|
||
|
bool TruncateSparseFile(base::File* sparse_file);
|
||
|
|
||
|
// Scans the existing ranges in the sparse file. Populates |sparse_ranges_|
|
||
|
// and sets |*out_sparse_data_size| to the total size of all the ranges (not
|
||
|
// including headers).
|
||
|
bool ScanSparseFile(base::File* sparse_file, int32_t* out_sparse_data_size);
|
||
|
|
||
|
// Reads from a single sparse range. If asked to read the entire range, also
|
||
|
// verifies the CRC32.
|
||
|
bool ReadSparseRange(base::File* sparse_file,
|
||
|
const SparseRange* range,
|
||
|
int offset,
|
||
|
int len,
|
||
|
char* buf);
|
||
|
|
||
|
// Writes to a single (existing) sparse range. If asked to write the entire
|
||
|
// range, also updates the CRC32; otherwise, invalidates it.
|
||
|
bool WriteSparseRange(base::File* sparse_file,
|
||
|
SparseRange* range,
|
||
|
int offset,
|
||
|
int len,
|
||
|
const char* buf);
|
||
|
|
||
|
// Appends a new sparse range to the sparse data file.
|
||
|
bool AppendSparseRange(base::File* sparse_file,
|
||
|
int64_t offset,
|
||
|
int len,
|
||
|
const char* buf);
|
||
|
|
||
|
static bool DeleteFileForEntryHash(const base::FilePath& path,
|
||
|
uint64_t entry_hash,
|
||
|
int file_index);
|
||
|
static bool DeleteFilesForEntryHash(const base::FilePath& path,
|
||
|
uint64_t entry_hash);
|
||
|
static bool TruncateFilesForEntryHash(const base::FilePath& path,
|
||
|
uint64_t entry_hash);
|
||
|
|
||
|
void RecordSyncCreateResult(CreateEntryResult result, bool had_index);
|
||
|
|
||
|
base::FilePath GetFilenameFromFileIndex(int file_index) const;
|
||
|
|
||
|
bool sparse_file_open() const { return sparse_file_open_; }
|
||
|
|
||
|
const net::CacheType cache_type_;
|
||
|
const base::FilePath path_;
|
||
|
SimpleFileTracker::EntryFileKey entry_file_key_;
|
||
|
const bool had_index_;
|
||
|
std::string key_;
|
||
|
|
||
|
bool have_open_files_;
|
||
|
bool initialized_;
|
||
|
|
||
|
// Normally false. This is set to true when an entry is opened without
|
||
|
// checking the file headers. Any subsequent read will perform the check
|
||
|
// before completing.
|
||
|
bool header_and_key_check_needed_[kSimpleEntryNormalFileCount] = {
|
||
|
false,
|
||
|
};
|
||
|
|
||
|
SimpleFileTracker* file_tracker_;
|
||
|
|
||
|
// True if the corresponding stream is empty and therefore no on-disk file
|
||
|
// was created to store it.
|
||
|
bool empty_file_omitted_[kSimpleEntryNormalFileCount];
|
||
|
|
||
|
typedef std::map<int64_t, SparseRange> SparseRangeOffsetMap;
|
||
|
typedef SparseRangeOffsetMap::iterator SparseRangeIterator;
|
||
|
SparseRangeOffsetMap sparse_ranges_;
|
||
|
bool sparse_file_open_;
|
||
|
|
||
|
// Offset of the end of the sparse file (where the next sparse range will be
|
||
|
// written).
|
||
|
int64_t sparse_tail_offset_;
|
||
|
};
|
||
|
|
||
|
} // namespace disk_cache
|
||
|
|
||
|
#endif // NET_DISK_CACHE_SIMPLE_SIMPLE_SYNCHRONOUS_ENTRY_H_
|