blob: c4188644de12e75cb22a41ce645de4a6105aade0 [file] [log] [blame]
Darin Petkov65b01462010-04-14 13:32:20 -07001// Copyright (c) 2010 The Chromium OS Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Bertrand SIMONNETd83ca802014-07-09 16:34:29 -07005#include "metrics/metrics_library.h"
Darin Petkov65b01462010-04-14 13:32:20 -07006
Luigi Semenzato41c54502014-05-13 15:16:24 -07007#include <base/logging.h>
8#include <base/strings/stringprintf.h>
Darin Petkov65b01462010-04-14 13:32:20 -07009#include <errno.h>
10#include <sys/file.h>
Ken Mixter4c5daa42010-08-26 18:35:06 -070011#include <sys/stat.h>
Darin Petkov4fcb2ac2010-04-15 16:40:23 -070012
Darin Petkov4fcb2ac2010-04-15 16:40:23 -070013#include <cstdio>
14#include <cstring>
Darin Petkov65b01462010-04-14 13:32:20 -070015
Bertrand SIMONNETd83ca802014-07-09 16:34:29 -070016#include "components/metrics/chromeos/metric_sample.h"
17#include "components/metrics/chromeos/serialization_utils.h"
Chris Masonee10b5482013-02-14 12:15:35 -080018
Ken Mixterb2f17092011-07-22 14:59:51 -070019#include "policy/device_policy.h"
Darin Petkov8842c8c2011-02-24 12:48:30 -080020
Chih-Chung Chang6844c062013-04-01 14:27:39 +080021#define ARRAY_SIZE(a) (sizeof(a) / sizeof((a)[0]))
Darin Petkov65b01462010-04-14 13:32:20 -070022
Ken Mixter4c5daa42010-08-26 18:35:06 -070023static const char kAutotestPath[] = "/var/log/metrics/autotest-events";
Luigi Semenzato31cda982014-05-19 08:30:42 -070024static const char kUMAEventsPath[] = "/var/run/metrics/uma-events";
Ken Mixter4c5daa42010-08-26 18:35:06 -070025static const char kConsentFile[] = "/home/chronos/Consent To Send Stats";
Darin Petkov4fcb2ac2010-04-15 16:40:23 -070026static const int32_t kBufferSize = 1024;
Luigi Semenzato32684222013-03-13 10:53:55 -070027static const char kCrosEventHistogramName[] = "Platform.CrOSEvent";
28static const int kCrosEventHistogramMax = 100;
Darin Petkov65b01462010-04-14 13:32:20 -070029
Chih-Chung Chang6844c062013-04-01 14:27:39 +080030/* Add new cros events here.
31 *
32 * The index of the event is sent in the message, so please do not
33 * reorder the names.
34 */
35static const char *kCrosEventNames[] = {
36 "ModemManagerCommandSendFailure", // 0
37 "HwWatchdogReboot", // 1
38 "Cras.NoCodecsFoundAtBoot", // 2
Darren Krahn6e55c1152013-07-19 14:09:50 -070039 "Chaps.DatabaseCorrupted", // 3
40 "Chaps.DatabaseRepairFailure", // 4
41 "Chaps.DatabaseCreateFailure", // 5
Darren Krahn86830ba2013-07-26 13:37:20 -070042 "Attestation.OriginSpecificExhausted", // 6
Luigi Semenzatoe57398a2013-11-11 14:24:44 -080043 "SpringPowerSupply.Original.High", // 7
44 "SpringPowerSupply.Other.High", // 8
Luigi Semenzatoe8fd9682013-11-13 16:28:43 -080045 "SpringPowerSupply.Original.Low", // 9
46 "SpringPowerSupply.ChargerIdle", // 10
Darren Krahn09a15fa2014-02-07 16:51:15 -080047 "TPM.NonZeroDictionaryAttackCounter", // 11
Chih-Chung Chang6844c062013-04-01 14:27:39 +080048};
49
Ken Mixter4c5daa42010-08-26 18:35:06 -070050time_t MetricsLibrary::cached_enabled_time_ = 0;
51bool MetricsLibrary::cached_enabled_ = false;
52
Luigi Semenzato41c54502014-05-13 15:16:24 -070053MetricsLibrary::MetricsLibrary() : consent_file_(kConsentFile) {}
Daniel Eratfd158292014-03-09 21:39:08 -070054MetricsLibrary::~MetricsLibrary() {}
55
Ken Mixtereafbbdf2010-10-01 15:38:42 -070056// We take buffer and buffer_size as parameters in order to simplify testing
57// of various alignments of the |device_name| with |buffer_size|.
58bool MetricsLibrary::IsDeviceMounted(const char* device_name,
59 const char* mounts_file,
60 char* buffer,
61 int buffer_size,
62 bool* result) {
63 if (buffer == NULL || buffer_size < 1)
64 return false;
65 int mounts_fd = open(mounts_file, O_RDONLY);
66 if (mounts_fd < 0)
67 return false;
68 // match_offset describes:
69 // -1 -- not beginning of line
70 // 0..strlen(device_name)-1 -- this offset in device_name is next to match
71 // strlen(device_name) -- matched full name, just need a space.
72 int match_offset = 0;
73 bool match = false;
74 while (!match) {
75 int read_size = read(mounts_fd, buffer, buffer_size);
76 if (read_size <= 0) {
77 if (errno == -EINTR)
78 continue;
79 break;
80 }
81 for (int i = 0; i < read_size; ++i) {
82 if (buffer[i] == '\n') {
83 match_offset = 0;
84 continue;
85 }
86 if (match_offset < 0) {
87 continue;
88 }
89 if (device_name[match_offset] == '\0') {
90 if (buffer[i] == ' ') {
91 match = true;
92 break;
93 }
94 match_offset = -1;
95 continue;
96 }
97
98 if (buffer[i] == device_name[match_offset]) {
99 ++match_offset;
100 } else {
101 match_offset = -1;
102 }
103 }
104 }
105 close(mounts_fd);
106 *result = match;
107 return true;
108}
109
110bool MetricsLibrary::IsGuestMode() {
111 char buffer[256];
112 bool result = false;
113 if (!IsDeviceMounted("guestfs",
114 "/proc/mounts",
115 buffer,
116 sizeof(buffer),
117 &result)) {
118 return false;
119 }
Arkaitz Ruiz Alvarez9f1a7742011-05-26 12:22:22 -0700120 return result && (access("/var/run/state/logged-in", F_OK) == 0);
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700121}
122
Ken Mixter4c5daa42010-08-26 18:35:06 -0700123bool MetricsLibrary::AreMetricsEnabled() {
Julian Pastarmov70b7abd2011-08-02 16:10:49 +0200124 static struct stat stat_buffer;
Ken Mixter4c5daa42010-08-26 18:35:06 -0700125 time_t this_check_time = time(NULL);
Ken Mixter4c5daa42010-08-26 18:35:06 -0700126 if (this_check_time != cached_enabled_time_) {
127 cached_enabled_time_ = this_check_time;
Julian Pastarmov70b7abd2011-08-02 16:10:49 +0200128
129 if (!policy_provider_.get())
130 policy_provider_.reset(new policy::PolicyProvider());
Julian Pastarmovd605a002013-02-04 17:58:14 +0100131 policy_provider_->Reload();
Julian Pastarmov70b7abd2011-08-02 16:10:49 +0200132 // We initialize with the default value which is false and will be preserved
133 // if the policy is not set.
134 bool enabled = false;
135 bool has_policy = false;
136 if (policy_provider_->device_policy_is_loaded()) {
137 has_policy =
138 policy_provider_->GetDevicePolicy().GetMetricsEnabled(&enabled);
139 }
140 // If policy couldn't be loaded or the metrics policy is not set we should
141 // still respect the consent file if it is present for migration purposes.
142 // TODO(pastarmovj)
143 if (!has_policy) {
Luigi Semenzato41c54502014-05-13 15:16:24 -0700144 enabled = stat(consent_file_.c_str(), &stat_buffer) >= 0;
Julian Pastarmov70b7abd2011-08-02 16:10:49 +0200145 }
146
Ken Mixterb2f17092011-07-22 14:59:51 -0700147 if (enabled && !IsGuestMode())
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700148 cached_enabled_ = true;
149 else
150 cached_enabled_ = false;
Ken Mixter4c5daa42010-08-26 18:35:06 -0700151 }
152 return cached_enabled_;
153}
Darin Petkov11b8eb32010-05-18 11:00:59 -0700154
Darin Petkovfc91b422010-05-12 13:05:45 -0700155void MetricsLibrary::Init() {
Darin Petkov11b8eb32010-05-18 11:00:59 -0700156 uma_events_file_ = kUMAEventsPath;
Darin Petkovfc91b422010-05-12 13:05:45 -0700157}
158
Luigi Semenzato41c54502014-05-13 15:16:24 -0700159bool MetricsLibrary::SendToAutotest(const std::string& name, int value) {
David James3b3add52010-06-04 15:01:19 -0700160 FILE* autotest_file = fopen(kAutotestPath, "a+");
Darin Petkov4fcb2ac2010-04-15 16:40:23 -0700161 if (autotest_file == NULL) {
Luigi Semenzato41c54502014-05-13 15:16:24 -0700162 PLOG(ERROR) << kAutotestPath << ": fopen";
Darin Petkov4fcb2ac2010-04-15 16:40:23 -0700163 return false;
164 }
165
166 fprintf(autotest_file, "%s=%d\n", name.c_str(), value);
167 fclose(autotest_file);
168 return true;
169}
170
Luigi Semenzato41c54502014-05-13 15:16:24 -0700171bool MetricsLibrary::SendToUMA(const std::string& name,
172 int sample,
173 int min,
174 int max,
175 int nbuckets) {
Bertrand SIMONNETd83ca802014-07-09 16:34:29 -0700176 return metrics::SerializationUtils::WriteMetricToFile(
177 *metrics::MetricSample::HistogramSample(name, sample, min, max, nbuckets)
178 .get(),
179 kUMAEventsPath);
Darin Petkov65b01462010-04-14 13:32:20 -0700180}
Darin Petkov5b7dce12010-04-21 15:45:10 -0700181
Darin Petkov21cd2c52010-05-12 15:26:16 -0700182bool MetricsLibrary::SendEnumToUMA(const std::string& name, int sample,
183 int max) {
Bertrand SIMONNETd83ca802014-07-09 16:34:29 -0700184 return metrics::SerializationUtils::WriteMetricToFile(
185 *metrics::MetricSample::LinearHistogramSample(name, sample, max).get(),
186 kUMAEventsPath);
Darin Petkoved824852011-01-06 10:51:47 -0800187}
188
Luigi Semenzatoa7ebeb32013-03-19 15:02:42 -0700189bool MetricsLibrary::SendSparseToUMA(const std::string& name, int sample) {
Bertrand SIMONNETd83ca802014-07-09 16:34:29 -0700190 return metrics::SerializationUtils::WriteMetricToFile(
191 *metrics::MetricSample::SparseHistogramSample(name, sample).get(),
192 kUMAEventsPath);
Luigi Semenzatoa7ebeb32013-03-19 15:02:42 -0700193}
194
Darin Petkoved824852011-01-06 10:51:47 -0800195bool MetricsLibrary::SendUserActionToUMA(const std::string& action) {
Bertrand SIMONNETd83ca802014-07-09 16:34:29 -0700196 return metrics::SerializationUtils::WriteMetricToFile(
197 *metrics::MetricSample::UserActionSample(action).get(), kUMAEventsPath);
Darin Petkov5b7dce12010-04-21 15:45:10 -0700198}
Ken Mixterbe2e13b2011-01-22 06:15:56 -0800199
200bool MetricsLibrary::SendCrashToUMA(const char *crash_kind) {
Bertrand SIMONNETd83ca802014-07-09 16:34:29 -0700201 return metrics::SerializationUtils::WriteMetricToFile(
202 *metrics::MetricSample::CrashSample(crash_kind).get(), kUMAEventsPath);
Ken Mixterbe2e13b2011-01-22 06:15:56 -0800203}
Ken Mixterb2f17092011-07-22 14:59:51 -0700204
205void MetricsLibrary::SetPolicyProvider(policy::PolicyProvider* provider) {
206 policy_provider_.reset(provider);
207}
Luigi Semenzato32684222013-03-13 10:53:55 -0700208
209bool MetricsLibrary::SendCrosEventToUMA(const std::string& event) {
Chih-Chung Chang6844c062013-04-01 14:27:39 +0800210 for (size_t i = 0; i < ARRAY_SIZE(kCrosEventNames); i++) {
211 if (strcmp(event.c_str(), kCrosEventNames[i]) == 0) {
212 return SendEnumToUMA(kCrosEventHistogramName, i, kCrosEventHistogramMax);
213 }
Luigi Semenzato32684222013-03-13 10:53:55 -0700214 }
Chih-Chung Chang6844c062013-04-01 14:27:39 +0800215 return false;
Luigi Semenzato32684222013-03-13 10:53:55 -0700216}