blob: 0fedb1d878c622f9c3264f16732b65f5b7270447 [file] [log] [blame]
// Copyright 2020 The Chromium OS 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 DIAGNOSTICS_CROS_HEALTHD_FETCH_AGGREGATOR_H_
#define DIAGNOSTICS_CROS_HEALTHD_FETCH_AGGREGATOR_H_
#include <cstdint>
#include <map>
#include <memory>
#include <set>
#include <vector>
#include <base/memory/weak_ptr.h>
#include <base/synchronization/lock.h>
#include "diagnostics/cros_healthd/fetchers/audio_fetcher.h"
#include "diagnostics/cros_healthd/fetchers/backlight_fetcher.h"
#include "diagnostics/cros_healthd/fetchers/battery_fetcher.h"
#include "diagnostics/cros_healthd/fetchers/bluetooth_fetcher.h"
#include "diagnostics/cros_healthd/fetchers/boot_performance_fetcher.h"
#include "diagnostics/cros_healthd/fetchers/bus_fetcher.h"
#include "diagnostics/cros_healthd/fetchers/cpu_fetcher.h"
#include "diagnostics/cros_healthd/fetchers/disk_fetcher.h"
#include "diagnostics/cros_healthd/fetchers/fan_fetcher.h"
#include "diagnostics/cros_healthd/fetchers/memory_fetcher.h"
#include "diagnostics/cros_healthd/fetchers/network_fetcher.h"
#include "diagnostics/cros_healthd/fetchers/stateful_partition_fetcher.h"
#include "diagnostics/cros_healthd/fetchers/system_fetcher.h"
#include "diagnostics/cros_healthd/fetchers/timezone_fetcher.h"
#include "diagnostics/cros_healthd/system/context.h"
#include "mojo/cros_healthd.mojom.h"
#include "mojo/cros_healthd_probe.mojom.h"
namespace diagnostics {
// This class is responsible for aggregating probe data from various fetchers,
// some of which may be asynchronous, and running the given callback when all
// probe data has been fetched.
class FetchAggregator final {
public:
explicit FetchAggregator(Context* context);
FetchAggregator(const FetchAggregator&) = delete;
FetchAggregator& operator=(const FetchAggregator&) = delete;
~FetchAggregator();
// Runs the aggregator, which will collect all relevant data and then run the
// callback.
void Run(const std::vector<chromeos::cros_healthd::mojom::ProbeCategoryEnum>&
categories_to_probe,
chromeos::cros_healthd::mojom::CrosHealthdProbeService::
ProbeTelemetryInfoCallback callback);
private:
// Holds all state related to a single call to Run().
struct ProbeState {
// Contains requested categories which have not been fetched yet.
std::set<chromeos::cros_healthd::mojom::ProbeCategoryEnum>
remaining_categories;
// Callback which will be run once all requested categories have been
// fetched.
chromeos::cros_healthd::mojom::CrosHealthdProbeService::
ProbeTelemetryInfoCallback callback;
// Holds all fetched data.
chromeos::cros_healthd::mojom::TelemetryInfo fetched_data;
};
// Wraps a fetch operation from either a synchronous or asynchronous fetcher.
template <class T>
void WrapFetchProbeData(
chromeos::cros_healthd::mojom::ProbeCategoryEnum category,
std::map<uint32_t, std::unique_ptr<ProbeState>>::iterator itr,
T* response_data,
T fetched_data);
// Returns the next available key in |pending_calls_|.
uint32_t GetNextAvailableKey();
// Maps call state to individual calls to Run(). This allows a single
// FetchAggregator instance to have multiple pending asychronous fetches
// corresponding to distinct Run() calls.
std::map<uint32_t, std::unique_ptr<ProbeState>> pending_calls_;
// Protects against one fetcher setting the last bit of a |fetched_data| map
// to true while another fetcher reads it. Without the lock, the reading
// fetcher would then take ownership of the callback and run it, then the
// setting fetcher would later attempt to run the invalid callback a second
// time.
base::Lock lock_;
AudioFetcher audio_fetcher_;
BacklightFetcher backlight_fetcher_;
BatteryFetcher battery_fetcher_;
BluetoothFetcher bluetooth_fetcher_;
BootPerformanceFetcher boot_performance_fetcher_;
BusFetcher bus_fetcher_;
CpuFetcher cpu_fetcher_;
DiskFetcher disk_fetcher_;
FanFetcher fan_fetcher_;
MemoryFetcher memory_fetcher_;
NetworkFetcher network_fetcher_;
StatefulPartitionFetcher stateful_partition_fetcher_;
SystemFetcher system_fetcher_;
TimezoneFetcher timezone_fetcher_;
// Must be the last member of the class.
base::WeakPtrFactory<FetchAggregator> weak_factory_{this};
};
} // namespace diagnostics
#endif // DIAGNOSTICS_CROS_HEALTHD_FETCH_AGGREGATOR_H_