| 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 "net/sdch/sdch_owner.h" | 5 #include "net/sdch/sdch_owner.h" | 
| 6 | 6 | 
| 7 #include <utility> | 7 #include <utility> | 
| 8 | 8 | 
| 9 #include "base/bind.h" | 9 #include "base/bind.h" | 
| 10 #include "base/debug/alias.h" | 10 #include "base/debug/alias.h" | 
| 11 #include "base/logging.h" | 11 #include "base/logging.h" | 
| 12 #include "base/macros.h" | 12 #include "base/macros.h" | 
| 13 #include "base/metrics/histogram_macros.h" | 13 #include "base/metrics/histogram_macros.h" | 
| 14 #include "base/prefs/persistent_pref_store.h" |  | 
| 15 #include "base/prefs/value_map_pref_store.h" |  | 
| 16 #include "base/strings/string_util.h" | 14 #include "base/strings/string_util.h" | 
| 17 #include "base/time/default_clock.h" | 15 #include "base/time/default_clock.h" | 
| 18 #include "base/values.h" | 16 #include "base/values.h" | 
| 19 #include "net/base/sdch_manager.h" | 17 #include "net/base/sdch_manager.h" | 
| 20 #include "net/base/sdch_net_log_params.h" | 18 #include "net/base/sdch_net_log_params.h" | 
| 21 | 19 | 
| 22 namespace net { | 20 namespace net { | 
| 23 | 21 | 
| 24 namespace { | 22 namespace { | 
| 25 | 23 | 
| 26 enum PersistenceFailureReason { |  | 
| 27   // File didn't exist; is being created. |  | 
| 28   PERSISTENCE_FAILURE_REASON_NO_FILE = 1, |  | 
| 29 |  | 
| 30   // Error reading in information, but should be able to write. |  | 
| 31   PERSISTENCE_FAILURE_REASON_READ_FAILED = 2, |  | 
| 32 |  | 
| 33   // Error leading to abort on attempted persistence. |  | 
| 34   PERSISTENCE_FAILURE_REASON_WRITE_FAILED = 3, |  | 
| 35 |  | 
| 36   PERSISTENCE_FAILURE_REASON_MAX = 4 |  | 
| 37 }; |  | 
| 38 |  | 
| 39 // Dictionaries that haven't been touched in 24 hours may be evicted | 24 // Dictionaries that haven't been touched in 24 hours may be evicted | 
| 40 // to make room for new dictionaries. | 25 // to make room for new dictionaries. | 
| 41 const int kFreshnessLifetimeHours = 24; | 26 const int kFreshnessLifetimeHours = 24; | 
| 42 | 27 | 
| 43 // Dictionaries that have never been used only stay fresh for one hour. | 28 // Dictionaries that have never been used only stay fresh for one hour. | 
| 44 const int kNeverUsedFreshnessLifetimeHours = 1; | 29 const int kNeverUsedFreshnessLifetimeHours = 1; | 
| 45 | 30 | 
| 46 void RecordPersistenceFailure(PersistenceFailureReason failure_reason) { | 31 void RecordPersistenceFailure( | 
| 47   UMA_HISTOGRAM_ENUMERATION("Sdch3.PersistenceFailureReason", failure_reason, | 32     SdchOwner::PrefStorage::ReadError failure_reason) { | 
| 48                             PERSISTENCE_FAILURE_REASON_MAX); | 33   UMA_HISTOGRAM_ENUMERATION( | 
|  | 34       "Sdch3.PersistenceFailureReason", failure_reason, | 
|  | 35       SdchOwner::PrefStorage::PERSISTENCE_FAILURE_REASON_MAX); | 
| 49 } | 36 } | 
| 50 | 37 | 
| 51 // Schema specifications and access routines. | 38 // Schema specifications and access routines. | 
| 52 | 39 | 
| 53 // The persistent prefs store is conceptually shared with any other network | 40 // The persistent prefs store is conceptually shared with any other network | 
| 54 // stack systems that want to persist data over browser restarts, and so | 41 // stack systems that want to persist data over browser restarts, and so | 
| 55 // use of it must be namespace restricted. | 42 // use of it must be namespace restricted. | 
| 56 // Schema: | 43 // Schema: | 
| 57 //      pref_store_->GetValue(kPreferenceName) -> Dictionary { | 44 //      pref_store_->GetValue() -> Dictionary { | 
| 58 //          'version' -> 2 [int] | 45 //          'version' -> 2 [int] | 
| 59 //          'dictionaries' -> Dictionary { | 46 //          'dictionaries' -> Dictionary { | 
| 60 //              server_hash -> { | 47 //              server_hash -> { | 
| 61 //                  'url' -> URL [string] | 48 //                  'url' -> URL [string] | 
| 62 //                  'last_used' -> seconds since unix epoch [double] | 49 //                  'last_used' -> seconds since unix epoch [double] | 
| 63 //                  'created_time' -> seconds since unix epoch [double] | 50 //                  'created_time' -> seconds since unix epoch [double] | 
| 64 //                  'use_count' -> use count [int] | 51 //                  'use_count' -> use count [int] | 
| 65 //                  'size' -> size [int] | 52 //                  'size' -> size [int] | 
| 66 //          } | 53 //          } | 
| 67 //      } | 54 //      } | 
| 68 const char kPreferenceName[] = "SDCH"; |  | 
| 69 const char kVersionKey[] = "version"; | 55 const char kVersionKey[] = "version"; | 
| 70 const char kDictionariesKey[] = "dictionaries"; | 56 const char kDictionariesKey[] = "dictionaries"; | 
| 71 const char kDictionaryUrlKey[] = "url"; | 57 const char kDictionaryUrlKey[] = "url"; | 
| 72 const char kDictionaryLastUsedKey[] = "last_used"; | 58 const char kDictionaryLastUsedKey[] = "last_used"; | 
| 73 const char kDictionaryCreatedTimeKey[] = "created_time"; | 59 const char kDictionaryCreatedTimeKey[] = "created_time"; | 
| 74 const char kDictionaryUseCountKey[] = "use_count"; | 60 const char kDictionaryUseCountKey[] = "use_count"; | 
| 75 const char kDictionarySizeKey[] = "size"; | 61 const char kDictionarySizeKey[] = "size"; | 
| 76 | 62 | 
| 77 const int kVersion = 2; | 63 const int kVersion = 2; | 
| 78 | 64 | 
|  | 65 // A simple implementation of pref storage that just stores the value in | 
|  | 66 // memory. | 
|  | 67 class ValueMapPrefStorage : public SdchOwner::PrefStorage { | 
|  | 68  public: | 
|  | 69   ValueMapPrefStorage() {} | 
|  | 70   ~ValueMapPrefStorage() override {} | 
|  | 71 | 
|  | 72   ReadError GetReadError() const override { return PERSISTENCE_FAILURE_NONE; } | 
|  | 73 | 
|  | 74   bool GetValue(const base::DictionaryValue** result) const override { | 
|  | 75     *result = &storage_; | 
|  | 76     return true; | 
|  | 77   } | 
|  | 78   bool GetMutableValue(base::DictionaryValue** result) override { | 
|  | 79     *result = &storage_; | 
|  | 80     return true; | 
|  | 81   } | 
|  | 82   void SetValue(scoped_ptr<base::DictionaryValue> value) override { | 
|  | 83     storage_.Clear(); | 
|  | 84     storage_.MergeDictionary(value.get()); | 
|  | 85   } | 
|  | 86 | 
|  | 87   void ReportValueChanged() override {} | 
|  | 88 | 
|  | 89   // This storage class requires no special initialization. | 
|  | 90   bool IsInitializationComplete() override { return true; } | 
|  | 91   void StartObservingInit(SdchOwner* observer) override {} | 
|  | 92   void StopObservingInit() override {} | 
|  | 93 | 
|  | 94  private: | 
|  | 95   base::DictionaryValue storage_; | 
|  | 96 | 
|  | 97   DISALLOW_COPY_AND_ASSIGN(ValueMapPrefStorage); | 
|  | 98 }; | 
|  | 99 | 
| 79 // This function returns store[kPreferenceName/kDictionariesKey].  The caller | 100 // This function returns store[kPreferenceName/kDictionariesKey].  The caller | 
| 80 // is responsible for making sure any needed calls to | 101 // is responsible for making sure any needed calls to | 
| 81 // |store->ReportValueChanged()| occur. | 102 // |store->ReportValueChanged()| occur. | 
| 82 base::DictionaryValue* GetPersistentStoreDictionaryMap( | 103 base::DictionaryValue* GetPersistentStoreDictionaryMap( | 
| 83     WriteablePrefStore* store) { | 104     SdchOwner::PrefStorage* store) { | 
| 84   base::Value* result = nullptr; | 105   base::DictionaryValue* preference_dictionary = nullptr; | 
| 85   bool success = store->GetMutableValue(kPreferenceName, &result); | 106   bool success = store->GetMutableValue(&preference_dictionary); | 
| 86   DCHECK(success); | 107   DCHECK(success); | 
| 87 | 108 | 
| 88   base::DictionaryValue* preference_dictionary = nullptr; |  | 
| 89   success = result->GetAsDictionary(&preference_dictionary); |  | 
| 90   DCHECK(success); |  | 
| 91   DCHECK(preference_dictionary); |  | 
| 92 |  | 
| 93   base::DictionaryValue* dictionary_list_dictionary = nullptr; | 109   base::DictionaryValue* dictionary_list_dictionary = nullptr; | 
| 94   success = preference_dictionary->GetDictionary(kDictionariesKey, | 110   success = preference_dictionary->GetDictionary(kDictionariesKey, | 
| 95                                                  &dictionary_list_dictionary); | 111                                                  &dictionary_list_dictionary); | 
| 96   DCHECK(success); | 112   DCHECK(success); | 
| 97   DCHECK(dictionary_list_dictionary); | 113   DCHECK(dictionary_list_dictionary); | 
| 98 | 114 | 
| 99   return dictionary_list_dictionary; | 115   return dictionary_list_dictionary; | 
| 100 } | 116 } | 
| 101 | 117 | 
| 102 // This function initializes a pref store with an empty version of the | 118 // This function initializes a pref store with an empty version of the | 
| 103 // above schema, removing anything previously in the store under | 119 // above schema, removing anything previously in the store under | 
| 104 // kPreferenceName. | 120 // kPreferenceName. | 
| 105 void InitializePrefStore(WriteablePrefStore* store) { | 121 void InitializePrefStore(SdchOwner::PrefStorage* store) { | 
| 106   scoped_ptr<base::DictionaryValue> empty_store(new base::DictionaryValue); | 122   scoped_ptr<base::DictionaryValue> empty_store(new base::DictionaryValue); | 
| 107   empty_store->SetInteger(kVersionKey, kVersion); | 123   empty_store->SetInteger(kVersionKey, kVersion); | 
| 108   empty_store->Set(kDictionariesKey, | 124   empty_store->Set(kDictionariesKey, | 
| 109                    make_scoped_ptr(new base::DictionaryValue)); | 125                    make_scoped_ptr(new base::DictionaryValue)); | 
| 110   store->SetValue(kPreferenceName, std::move(empty_store), | 126   store->SetValue(std::move(empty_store)); | 
| 111                   WriteablePrefStore::DEFAULT_PREF_WRITE_FLAGS); |  | 
| 112 } | 127 } | 
| 113 | 128 | 
| 114 // A class to allow iteration over all dictionaries in the pref store, and | 129 // A class to allow iteration over all dictionaries in the pref store, and | 
| 115 // easy lookup of the information associated with those dictionaries. | 130 // easy lookup of the information associated with those dictionaries. | 
| 116 // Note that this is an "Iterator" in the same sense (and for the same | 131 // Note that this is an "Iterator" in the same sense (and for the same | 
| 117 // reasons) that base::Dictionary::Iterator is an iterator--it allows | 132 // reasons) that base::Dictionary::Iterator is an iterator--it allows | 
| 118 // iterating over all the dictionaries in the preference store, but it | 133 // iterating over all the dictionaries in the preference store, but it | 
| 119 // does not allow use as an STL iterator because the container it | 134 // does not allow use as an STL iterator because the container it | 
| 120 // is iterating over does not export begin()/end() methods. This iterator can | 135 // is iterating over does not export begin()/end() methods. This iterator can | 
| 121 // only be safely used on sanitized pref stores that are known to conform to the | 136 // only be safely used on sanitized pref stores that are known to conform to the | 
| 122 // pref store schema. | 137 // pref store schema. | 
| 123 class DictionaryPreferenceIterator { | 138 class DictionaryPreferenceIterator { | 
| 124  public: | 139  public: | 
| 125   explicit DictionaryPreferenceIterator(WriteablePrefStore* pref_store); | 140   explicit DictionaryPreferenceIterator(SdchOwner::PrefStorage* pref_store); | 
| 126 | 141 | 
| 127   bool IsAtEnd() const; | 142   bool IsAtEnd() const; | 
| 128   void Advance(); | 143   void Advance(); | 
| 129 | 144 | 
| 130   const std::string& server_hash() const { return server_hash_; } | 145   const std::string& server_hash() const { return server_hash_; } | 
| 131   const GURL& url() const { return url_; } | 146   const GURL& url() const { return url_; } | 
| 132   base::Time last_used() const { return last_used_; } | 147   base::Time last_used() const { return last_used_; } | 
| 133   base::Time created_time() const { return created_time_; } | 148   base::Time created_time() const { return created_time_; } | 
| 134   int use_count() const { return use_count_; } | 149   int use_count() const { return use_count_; } | 
| 135   int size() const { return size_; } | 150   int size() const { return size_; } | 
| 136 | 151 | 
| 137  private: | 152  private: | 
| 138   // Load Dictionary silently skipping any that are malformed. | 153   // Load Dictionary silently skipping any that are malformed. | 
| 139   void LoadNextDictionary(); | 154   void LoadNextDictionary(); | 
| 140   // Try to load Dictionary from current iterator's position. Returns true if | 155   // Try to load Dictionary from current iterator's position. Returns true if | 
| 141   // succeeded. | 156   // succeeded. | 
| 142   bool TryLoadDictionary(); | 157   bool TryLoadDictionary(); | 
| 143 | 158 | 
| 144   std::string server_hash_; | 159   std::string server_hash_; | 
| 145   GURL url_; | 160   GURL url_; | 
| 146   base::Time last_used_; | 161   base::Time last_used_; | 
| 147   base::Time created_time_; | 162   base::Time created_time_; | 
| 148   int use_count_; | 163   int use_count_; | 
| 149   int size_; | 164   int size_; | 
| 150 | 165 | 
| 151   base::DictionaryValue::Iterator dictionary_iterator_; | 166   base::DictionaryValue::Iterator dictionary_iterator_; | 
| 152 }; | 167 }; | 
| 153 | 168 | 
| 154 DictionaryPreferenceIterator::DictionaryPreferenceIterator( | 169 DictionaryPreferenceIterator::DictionaryPreferenceIterator( | 
| 155     WriteablePrefStore* pref_store) | 170     SdchOwner::PrefStorage* pref_store) | 
| 156     : use_count_(0), | 171     : use_count_(0), | 
| 157       size_(0), | 172       size_(0), | 
| 158       dictionary_iterator_(*GetPersistentStoreDictionaryMap(pref_store)) { | 173       dictionary_iterator_(*GetPersistentStoreDictionaryMap(pref_store)) { | 
| 159   LoadNextDictionary(); | 174   LoadNextDictionary(); | 
| 160 } | 175 } | 
| 161 | 176 | 
| 162 bool DictionaryPreferenceIterator::IsAtEnd() const { | 177 bool DictionaryPreferenceIterator::IsAtEnd() const { | 
| 163   return dictionary_iterator_.IsAtEnd(); | 178   return dictionary_iterator_.IsAtEnd(); | 
| 164 } | 179 } | 
| 165 | 180 | 
| (...skipping 42 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 208 | 223 | 
| 209   double created_time_seconds = 0; | 224   double created_time_seconds = 0; | 
| 210   success = dict->GetDouble(kDictionaryCreatedTimeKey, &created_time_seconds); | 225   success = dict->GetDouble(kDictionaryCreatedTimeKey, &created_time_seconds); | 
| 211   if (!success) | 226   if (!success) | 
| 212     return false; | 227     return false; | 
| 213   created_time_ = base::Time::FromDoubleT(created_time_seconds); | 228   created_time_ = base::Time::FromDoubleT(created_time_seconds); | 
| 214 | 229 | 
| 215   return true; | 230   return true; | 
| 216 } | 231 } | 
| 217 | 232 | 
| 218 // Triggers a ReportValueChanged() on the specified WriteablePrefStore | 233 // Triggers a ReportValueChanged() when the object goes out of scope. | 
| 219 // when the object goes out of scope. |  | 
| 220 class ScopedPrefNotifier { | 234 class ScopedPrefNotifier { | 
| 221  public: | 235  public: | 
| 222   // Caller must guarantee lifetime of |*pref_store| exceeds the | 236   // Caller must guarantee lifetime of |*pref_store| exceeds the | 
| 223   // lifetime of this object. | 237   // lifetime of this object. | 
| 224   ScopedPrefNotifier(WriteablePrefStore* pref_store) | 238   ScopedPrefNotifier(SdchOwner::PrefStorage* pref_store) | 
| 225       : pref_store_(pref_store) {} | 239       : pref_store_(pref_store) { | 
| 226   ~ScopedPrefNotifier() { | 240     DCHECK(pref_store); | 
| 227     pref_store_->ReportValueChanged( |  | 
| 228         kPreferenceName, WriteablePrefStore::DEFAULT_PREF_WRITE_FLAGS); |  | 
| 229   } | 241   } | 
|  | 242   ~ScopedPrefNotifier() { pref_store_->ReportValueChanged(); } | 
| 230 | 243 | 
| 231  private: | 244  private: | 
| 232   WriteablePrefStore* pref_store_; | 245   SdchOwner::PrefStorage* pref_store_; | 
| 233 | 246 | 
| 234   DISALLOW_COPY_AND_ASSIGN(ScopedPrefNotifier); | 247   DISALLOW_COPY_AND_ASSIGN(ScopedPrefNotifier); | 
| 235 }; | 248 }; | 
| 236 | 249 | 
| 237 }  // namespace | 250 }  // namespace | 
| 238 | 251 | 
| 239 // Adjust SDCH limits downwards for mobile. | 252 // Adjust SDCH limits downwards for mobile. | 
| 240 #if defined(OS_ANDROID) || defined(OS_IOS) | 253 #if defined(OS_ANDROID) || defined(OS_IOS) | 
| 241 // static | 254 // static | 
| 242 const size_t SdchOwner::kMaxTotalDictionarySize = 2 * 500 * 1000; | 255 const size_t SdchOwner::kMaxTotalDictionarySize = 2 * 500 * 1000; | 
| 243 #else | 256 #else | 
| 244 // static | 257 // static | 
| 245 const size_t SdchOwner::kMaxTotalDictionarySize = 20 * 1000 * 1000; | 258 const size_t SdchOwner::kMaxTotalDictionarySize = 20 * 1000 * 1000; | 
| 246 #endif | 259 #endif | 
| 247 | 260 | 
|  | 261 SdchOwner::PrefStorage::~PrefStorage() {} | 
|  | 262 | 
| 248 // Somewhat arbitrary, but we assume a dictionary smaller than | 263 // Somewhat arbitrary, but we assume a dictionary smaller than | 
| 249 // 50K isn't going to do anyone any good.  Note that this still doesn't | 264 // 50K isn't going to do anyone any good.  Note that this still doesn't | 
| 250 // prevent download and addition unless there is less than this | 265 // prevent download and addition unless there is less than this | 
| 251 // amount of space available in storage. | 266 // amount of space available in storage. | 
| 252 const size_t SdchOwner::kMinSpaceForDictionaryFetch = 50 * 1000; | 267 const size_t SdchOwner::kMinSpaceForDictionaryFetch = 50 * 1000; | 
| 253 | 268 | 
| 254 void SdchOwner::RecordDictionaryFate(enum DictionaryFate fate) { | 269 void SdchOwner::RecordDictionaryFate(enum DictionaryFate fate) { | 
| 255   UMA_HISTOGRAM_ENUMERATION("Sdch3.DictionaryFate", fate, DICTIONARY_FATE_MAX); | 270   UMA_HISTOGRAM_ENUMERATION("Sdch3.DictionaryFate", fate, DICTIONARY_FATE_MAX); | 
| 256 } | 271 } | 
| 257 | 272 | 
| (...skipping 22 matching lines...) Expand all  Loading... | 
| 280       total_dictionary_bytes_(0), | 295       total_dictionary_bytes_(0), | 
| 281       clock_(new base::DefaultClock), | 296       clock_(new base::DefaultClock), | 
| 282       max_total_dictionary_size_(kMaxTotalDictionarySize), | 297       max_total_dictionary_size_(kMaxTotalDictionarySize), | 
| 283       min_space_for_dictionary_fetch_(kMinSpaceForDictionaryFetch), | 298       min_space_for_dictionary_fetch_(kMinSpaceForDictionaryFetch), | 
| 284       memory_pressure_listener_( | 299       memory_pressure_listener_( | 
| 285           base::Bind(&SdchOwner::OnMemoryPressure, | 300           base::Bind(&SdchOwner::OnMemoryPressure, | 
| 286                      // Because |memory_pressure_listener_| is owned by | 301                      // Because |memory_pressure_listener_| is owned by | 
| 287                      // SdchOwner, the SdchOwner object will be available | 302                      // SdchOwner, the SdchOwner object will be available | 
| 288                      // for the lifetime of |memory_pressure_listener_|. | 303                      // for the lifetime of |memory_pressure_listener_|. | 
| 289                      base::Unretained(this))), | 304                      base::Unretained(this))), | 
| 290       in_memory_pref_store_(new ValueMapPrefStore()), | 305       in_memory_pref_store_(new ValueMapPrefStorage()), | 
| 291       external_pref_store_(nullptr), | 306       external_pref_store_(nullptr), | 
| 292       pref_store_(in_memory_pref_store_.get()), | 307       pref_store_(in_memory_pref_store_.get()), | 
| 293       creation_time_(clock_->Now()) { | 308       creation_time_(clock_->Now()) { | 
| 294   manager_->AddObserver(this); | 309   manager_->AddObserver(this); | 
| 295   InitializePrefStore(pref_store_); | 310   InitializePrefStore(pref_store_); | 
| 296 } | 311 } | 
| 297 | 312 | 
| 298 SdchOwner::~SdchOwner() { | 313 SdchOwner::~SdchOwner() { | 
| 299   for (DictionaryPreferenceIterator it(pref_store_); !it.IsAtEnd(); | 314   for (DictionaryPreferenceIterator it(pref_store_); !it.IsAtEnd(); | 
| 300        it.Advance()) { | 315        it.Advance()) { | 
| 301     int new_uses = it.use_count() - use_counts_at_load_[it.server_hash()]; | 316     int new_uses = it.use_count() - use_counts_at_load_[it.server_hash()]; | 
| 302     DictionaryFate fate = IsPersistingDictionaries() ? | 317     DictionaryFate fate = IsPersistingDictionaries() ? | 
| 303                           DICTIONARY_FATE_UNLOAD_FOR_DESTRUCTION : | 318                           DICTIONARY_FATE_UNLOAD_FOR_DESTRUCTION : | 
| 304                           DICTIONARY_FATE_EVICT_FOR_DESTRUCTION; | 319                           DICTIONARY_FATE_EVICT_FOR_DESTRUCTION; | 
| 305     RecordDictionaryEvictionOrUnload(it.server_hash(), it.size(), new_uses, | 320     RecordDictionaryEvictionOrUnload(it.server_hash(), it.size(), new_uses, | 
| 306                                      fate); | 321                                      fate); | 
| 307   } | 322   } | 
| 308   manager_->RemoveObserver(this); | 323   manager_->RemoveObserver(this); | 
| 309 | 324 | 
| 310   // This object only observes the external store during loading, | 325   // This object only observes the external store during loading, | 
| 311   // i.e. before it's made the default preferences store. | 326   // i.e. before it's made the default preferences store. | 
| 312   if (external_pref_store_) | 327   if (external_pref_store_ && pref_store_ != external_pref_store_.get()) | 
| 313     external_pref_store_->RemoveObserver(this); | 328     external_pref_store_->StopObservingInit(); | 
| 314 | 329 | 
| 315   int64_t object_lifetime = (clock_->Now() - creation_time_).InMilliseconds(); | 330   int64_t object_lifetime = (clock_->Now() - creation_time_).InMilliseconds(); | 
| 316   for (const auto& val : consumed_byte_seconds_) { | 331   for (const auto& val : consumed_byte_seconds_) { | 
| 317     if (object_lifetime > 0) { | 332     if (object_lifetime > 0) { | 
| 318       // Objects that are created and immediately destroyed don't add any memory | 333       // Objects that are created and immediately destroyed don't add any memory | 
| 319       // pressure over time (and also cause a crash here). | 334       // pressure over time (and also cause a crash here). | 
| 320       UMA_HISTOGRAM_MEMORY_KB("Sdch3.TimeWeightedMemoryUse", | 335       UMA_HISTOGRAM_MEMORY_KB("Sdch3.TimeWeightedMemoryUse", | 
| 321                               val / object_lifetime); | 336                               val / object_lifetime); | 
| 322     } | 337     } | 
| 323   } | 338   } | 
| 324 |  | 
| 325 } | 339 } | 
| 326 | 340 | 
| 327 void SdchOwner::EnablePersistentStorage(PersistentPrefStore* pref_store) { | 341 void SdchOwner::EnablePersistentStorage(scoped_ptr<PrefStorage> pref_store) { | 
| 328   DCHECK(!external_pref_store_); | 342   DCHECK(!external_pref_store_); | 
| 329   external_pref_store_ = pref_store; | 343   DCHECK(pref_store); | 
| 330   external_pref_store_->AddObserver(this); | 344   external_pref_store_ = std::move(pref_store); | 
|  | 345   external_pref_store_->StartObservingInit(this); | 
| 331 | 346 | 
| 332   if (external_pref_store_->IsInitializationComplete()) | 347   if (external_pref_store_->IsInitializationComplete()) | 
| 333     OnInitializationCompleted(true); | 348     OnPrefStorageInitializationComplete(true); | 
| 334 } | 349 } | 
| 335 | 350 | 
| 336 void SdchOwner::SetMaxTotalDictionarySize(size_t max_total_dictionary_size) { | 351 void SdchOwner::SetMaxTotalDictionarySize(size_t max_total_dictionary_size) { | 
| 337   max_total_dictionary_size_ = max_total_dictionary_size; | 352   max_total_dictionary_size_ = max_total_dictionary_size; | 
| 338 } | 353 } | 
| 339 | 354 | 
| 340 void SdchOwner::SetMinSpaceForDictionaryFetch( | 355 void SdchOwner::SetMinSpaceForDictionaryFetch( | 
| 341     size_t min_space_for_dictionary_fetch) { | 356     size_t min_space_for_dictionary_fetch) { | 
| 342   min_space_for_dictionary_fetch_ = min_space_for_dictionary_fetch; | 357   min_space_for_dictionary_fetch_ = min_space_for_dictionary_fetch; | 
| 343 } | 358 } | 
| (...skipping 230 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 574                  base::Unretained(this), base::Time(), base::Time::Now(), 0)); | 589                  base::Unretained(this), base::Time(), base::Time::Now(), 0)); | 
| 575 } | 590 } | 
| 576 | 591 | 
| 577 void SdchOwner::OnClearDictionaries() { | 592 void SdchOwner::OnClearDictionaries() { | 
| 578   total_dictionary_bytes_ = 0; | 593   total_dictionary_bytes_ = 0; | 
| 579   fetcher_->Cancel(); | 594   fetcher_->Cancel(); | 
| 580 | 595 | 
| 581   InitializePrefStore(pref_store_); | 596   InitializePrefStore(pref_store_); | 
| 582 } | 597 } | 
| 583 | 598 | 
| 584 void SdchOwner::OnPrefValueChanged(const std::string& key) { | 599 void SdchOwner::OnPrefStorageInitializationComplete(bool succeeded) { | 
| 585 } | 600   PrefStorage::ReadError error = external_pref_store_->GetReadError(); | 
| 586 |  | 
| 587 void SdchOwner::OnInitializationCompleted(bool succeeded) { |  | 
| 588   PersistentPrefStore::PrefReadError error = |  | 
| 589       external_pref_store_->GetReadError(); |  | 
| 590   // Errors on load are self-correcting; if dictionaries were not | 601   // Errors on load are self-correcting; if dictionaries were not | 
| 591   // persisted from the last instance of the browser, they will be | 602   // persisted from the last instance of the browser, they will be | 
| 592   // faulted in by user action over time.  However, if a load error | 603   // faulted in by user action over time.  However, if a load error | 
| 593   // means that the dictionary information won't be able to be persisted, | 604   // means that the dictionary information won't be able to be persisted, | 
| 594   // the in memory pref store is left in place. | 605   // the in memory pref store is left in place. | 
| 595   if (!succeeded) { | 606   if (!succeeded) { | 
| 596     // Failure means a write failed, since read failures are recoverable. | 607     // Failure means a write failed, since read failures are recoverable. | 
| 597     DCHECK_NE( | 608     external_pref_store_->StopObservingInit(); | 
| 598         error, |  | 
| 599         PersistentPrefStore::PREF_READ_ERROR_ASYNCHRONOUS_TASK_INCOMPLETE); |  | 
| 600     DCHECK_NE(error, |  | 
| 601               PersistentPrefStore::PREF_READ_ERROR_MAX_ENUM); |  | 
| 602 |  | 
| 603     LOG(ERROR) << "Pref store write failed: " << error; |  | 
| 604     external_pref_store_->RemoveObserver(this); |  | 
| 605     external_pref_store_ = nullptr; | 609     external_pref_store_ = nullptr; | 
| 606     RecordPersistenceFailure(PERSISTENCE_FAILURE_REASON_WRITE_FAILED); | 610     RecordPersistenceFailure( | 
|  | 611         PrefStorage::PERSISTENCE_FAILURE_REASON_WRITE_FAILED); | 
| 607     return; | 612     return; | 
| 608   } | 613   } | 
| 609   switch (external_pref_store_->GetReadError()) { |  | 
| 610     case PersistentPrefStore::PREF_READ_ERROR_NONE: |  | 
| 611         break; |  | 
| 612 | 614 | 
| 613     case PersistentPrefStore::PREF_READ_ERROR_NO_FILE: | 615   if (error != PrefStorage::PERSISTENCE_FAILURE_NONE) | 
| 614       // First time reading; the file will be created. | 616     RecordPersistenceFailure(error); | 
| 615       RecordPersistenceFailure(PERSISTENCE_FAILURE_REASON_NO_FILE); |  | 
| 616       break; |  | 
| 617 |  | 
| 618     case PersistentPrefStore::PREF_READ_ERROR_JSON_PARSE: |  | 
| 619     case PersistentPrefStore::PREF_READ_ERROR_JSON_TYPE: |  | 
| 620     case PersistentPrefStore::PREF_READ_ERROR_FILE_OTHER: |  | 
| 621     case PersistentPrefStore::PREF_READ_ERROR_FILE_LOCKED: |  | 
| 622     case PersistentPrefStore::PREF_READ_ERROR_JSON_REPEAT: |  | 
| 623       RecordPersistenceFailure(PERSISTENCE_FAILURE_REASON_READ_FAILED); |  | 
| 624       break; |  | 
| 625 |  | 
| 626     case PersistentPrefStore::PREF_READ_ERROR_ACCESS_DENIED: |  | 
| 627     case PersistentPrefStore::PREF_READ_ERROR_FILE_NOT_SPECIFIED: |  | 
| 628     case PersistentPrefStore::PREF_READ_ERROR_ASYNCHRONOUS_TASK_INCOMPLETE: |  | 
| 629     case PersistentPrefStore::PREF_READ_ERROR_MAX_ENUM: |  | 
| 630       // Shouldn't ever happen. ACCESS_DENIED and FILE_NOT_SPECIFIED should |  | 
| 631       // imply !succeeded, and TASK_INCOMPLETE should never be delivered. |  | 
| 632       NOTREACHED(); |  | 
| 633       break; |  | 
| 634   } |  | 
| 635 |  | 
| 636 | 617 | 
| 637   // Load in what was stored before chrome exited previously. | 618   // Load in what was stored before chrome exited previously. | 
| 638   const base::Value* sdch_persistence_value = nullptr; |  | 
| 639   const base::DictionaryValue* sdch_persistence_dictionary = nullptr; | 619   const base::DictionaryValue* sdch_persistence_dictionary = nullptr; | 
| 640 | 620 | 
| 641   // The GetPersistentStore() routine above assumes data formatted | 621   // The GetPersistentStore() routine above assumes data formatted | 
| 642   // according to the schema described at the top of this file.  Since | 622   // according to the schema described at the top of this file.  Since | 
| 643   // this data comes from disk, to avoid disk corruption resulting in | 623   // this data comes from disk, to avoid disk corruption resulting in | 
| 644   // persistent chrome errors this code avoids those assupmtions. | 624   // persistent chrome errors this code avoids those assupmtions. | 
| 645   if (external_pref_store_->GetValue(kPreferenceName, | 625   if (external_pref_store_->GetValue(&sdch_persistence_dictionary)) | 
| 646                                      &sdch_persistence_value) && |  | 
| 647       sdch_persistence_value->GetAsDictionary(&sdch_persistence_dictionary)) { |  | 
| 648     SchedulePersistedDictionaryLoads(*sdch_persistence_dictionary); | 626     SchedulePersistedDictionaryLoads(*sdch_persistence_dictionary); | 
| 649   } |  | 
| 650 | 627 | 
| 651   // Reset the persistent store and update it with the accumulated | 628   // Reset the persistent store and update it with the accumulated | 
| 652   // information from the local store. | 629   // information from the local store. | 
| 653   InitializePrefStore(external_pref_store_); | 630   InitializePrefStore(external_pref_store_.get()); | 
| 654 | 631 | 
| 655   ScopedPrefNotifier scoped_pref_notifier(external_pref_store_); | 632   ScopedPrefNotifier scoped_pref_notifier(external_pref_store_.get()); | 
| 656   GetPersistentStoreDictionaryMap(external_pref_store_) | 633   GetPersistentStoreDictionaryMap(external_pref_store_.get()) | 
| 657       ->Swap(GetPersistentStoreDictionaryMap(in_memory_pref_store_.get())); | 634       ->Swap(GetPersistentStoreDictionaryMap(in_memory_pref_store_.get())); | 
| 658 | 635 | 
| 659   // This object can stop waiting on (i.e. observing) the external preference | 636   // This object can stop waiting on (i.e. observing) the external preference | 
| 660   // store and switch over to using it as the primary preference store. | 637   // store and switch over to using it as the primary preference store. | 
| 661   pref_store_ = external_pref_store_; | 638   pref_store_ = external_pref_store_.get(); | 
| 662   external_pref_store_->RemoveObserver(this); | 639   external_pref_store_->StopObservingInit(); | 
| 663   external_pref_store_ = nullptr; |  | 
| 664   in_memory_pref_store_ = nullptr; | 640   in_memory_pref_store_ = nullptr; | 
| 665 } | 641 } | 
| 666 | 642 | 
| 667 void SdchOwner::SetClockForTesting(scoped_ptr<base::Clock> clock) { | 643 void SdchOwner::SetClockForTesting(scoped_ptr<base::Clock> clock) { | 
| 668   clock_ = std::move(clock); | 644   clock_ = std::move(clock); | 
| 669 } | 645 } | 
| 670 | 646 | 
| 671 int SdchOwner::GetDictionaryCountForTesting() const { | 647 int SdchOwner::GetDictionaryCountForTesting() const { | 
| 672   int count = 0; | 648   int count = 0; | 
| 673   for (DictionaryPreferenceIterator it(pref_store_); !it.IsAtEnd(); | 649   for (DictionaryPreferenceIterator it(pref_store_); !it.IsAtEnd(); | 
| (...skipping 86 matching lines...) Expand 10 before | Expand all | Expand 10 after  Loading... | 
| 760   } | 736   } | 
| 761 | 737 | 
| 762   return true; | 738   return true; | 
| 763 } | 739 } | 
| 764 | 740 | 
| 765 bool SdchOwner::IsPersistingDictionaries() const { | 741 bool SdchOwner::IsPersistingDictionaries() const { | 
| 766   return in_memory_pref_store_.get() != nullptr; | 742   return in_memory_pref_store_.get() != nullptr; | 
| 767 } | 743 } | 
| 768 | 744 | 
| 769 }  // namespace net | 745 }  // namespace net | 
| OLD | NEW | 
|---|