File: file_system_access_observer_quota_manager.cc

package info (click to toggle)
chromium 138.0.7204.183-1~deb12u1
  • links: PTS, VCS
  • area: main
  • in suites: bookworm-proposed-updates
  • size: 6,080,960 kB
  • sloc: cpp: 34,937,079; ansic: 7,176,967; javascript: 4,110,704; python: 1,419,954; asm: 946,768; xml: 739,971; pascal: 187,324; sh: 89,623; perl: 88,663; objc: 79,944; sql: 50,304; cs: 41,786; fortran: 24,137; makefile: 21,811; php: 13,980; tcl: 13,166; yacc: 8,925; ruby: 7,485; awk: 3,720; lisp: 3,096; lex: 1,327; ada: 727; jsp: 228; sed: 36
file content (108 lines) | stat: -rw-r--r-- 4,164 bytes parent folder | download | duplicates (5)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
// Copyright 2024 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "content/browser/file_system_access/file_system_access_observer_quota_manager.h"

#include "base/memory/scoped_refptr.h"
#include "base/metrics/histogram_functions.h"
#include "content/browser/file_system_access/file_system_access_watcher_manager.h"
#include "services/metrics/public/cpp/metrics_utils.h"
#include "services/metrics/public/cpp/ukm_builders.h"

namespace content {
FileSystemAccessObserverQuotaManager::Handle::Handle(
    scoped_refptr<FileSystemAccessObserverQuotaManager> quota_manager)
    : quota_manager_(std::move(quota_manager)) {}

FileSystemAccessObserverQuotaManager::Handle::~Handle() = default;

FileSystemAccessObserverQuotaManager::Handle::Handle(Handle&&) = default;
FileSystemAccessObserverQuotaManager::Handle&
FileSystemAccessObserverQuotaManager::Handle::operator=(Handle&&) = default;

FileSystemAccessObserverQuotaManager::UsageChangeResult
FileSystemAccessObserverQuotaManager::Handle::OnUsageChange(size_t usage) {
  if (errored_) {
    return UsageChangeResult::kQuotaUnavailable;
  }

  UsageChangeResult result = quota_manager_->OnUsageChange(old_usage_, usage);

  old_usage_ = usage;
  errored_ = result == UsageChangeResult::kQuotaUnavailable;

  return result;
}

FileSystemAccessObserverQuotaManager::FileSystemAccessObserverQuotaManager(
    const blink::StorageKey& storage_key,
    ukm::SourceId ukm_source_id,
    FileSystemAccessWatcherManager& watcher_manager)
    : base::RefCountedDeleteOnSequence<FileSystemAccessObserverQuotaManager>(
          base::SequencedTaskRunner::GetCurrentDefault()),
      storage_key_(storage_key),
      ukm_source_id_(ukm_source_id),
      watcher_manager_(watcher_manager) {}

FileSystemAccessObserverQuotaManager::~FileSystemAccessObserverQuotaManager() {
  CHECK(FileSystemAccessChangeSource::quota_limit() > 0);
  // The percentile value, rounded down to the nearest integer.
  size_t usage_rate = 100 * high_water_mark_usage_ /
                      FileSystemAccessChangeSource::quota_limit();

  // UMA logging.
  if (high_water_mark_usage_ > 0) {
    base::UmaHistogramCounts100000("Storage.FileSystemAccess.ObserverUsage",
                                   high_water_mark_usage_);
    base::UmaHistogramPercentage("Storage.FileSystemAccess.ObserverUsageRate",
                                 usage_rate);
  }
  base::UmaHistogramBoolean(
      "Storage.FileSystemAccess.ObserverUsageQuotaExceeded",
      reached_quota_limit_);

  // UKM logging.
  if (ukm_source_id_ != ukm::kInvalidSourceId) {
    auto ukm_builder = ukm::builders::FileSystemObserver_Usage(ukm_source_id_);
    if (high_water_mark_usage_ > 0) {
      ukm_builder
          .SetHighWaterMark(ukm::GetExponentialBucketMin(
              high_water_mark_usage_, kHighWaterMarkBucketSpacing))
          .SetHighWaterMarkPercentage(usage_rate);
    }
    ukm_builder.SetQuotaExceeded(reached_quota_limit_)
        .Record(ukm::UkmRecorder::Get());
  }

  watcher_manager_->RemoveQuotaManager(storage_key_);
}

FileSystemAccessObserverQuotaManager::Handle
FileSystemAccessObserverQuotaManager::CreateHandle() {
  return Handle(base::WrapRefCounted(this));
}

FileSystemAccessObserverQuotaManager::UsageChangeResult
FileSystemAccessObserverQuotaManager::OnUsageChange(size_t old_usage,
                                                    size_t new_usage) {
  // The caller should have reported this `old_usage` in its last call, so that
  // `total_usage_` is equal to the sum of `old_usage` plus possibly other
  // observation group usages.
  CHECK_GE(total_usage_, old_usage);

  size_t updated_total_usage = total_usage_ + new_usage - old_usage;
  if (updated_total_usage > FileSystemAccessChangeSource::quota_limit()) {
    total_usage_ -= old_usage;
    reached_quota_limit_ = true;
    return UsageChangeResult::kQuotaUnavailable;
  }

  if (updated_total_usage > high_water_mark_usage_) {
    high_water_mark_usage_ = updated_total_usage;
  }
  total_usage_ = updated_total_usage;
  return UsageChangeResult::kOk;
}

}  // namespace content