blob: f777f2823fd4e3e5e07eae8fa2e788725db5bea1 [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 SIMONNETbd3505e2015-08-04 14:04:51 -070016#include "constants.h"
Bertrand SIMONNET4b915ae2015-07-28 15:38:14 -070017#include "serialization/metric_sample.h"
18#include "serialization/serialization_utils.h"
Chris Masonee10b5482013-02-14 12:15:35 -080019
Luigi Semenzato32684222013-03-13 10:53:55 -070020static const char kCrosEventHistogramName[] = "Platform.CrOSEvent";
21static const int kCrosEventHistogramMax = 100;
Darin Petkov65b01462010-04-14 13:32:20 -070022
Chih-Chung Chang6844c062013-04-01 14:27:39 +080023/* Add new cros events here.
24 *
25 * The index of the event is sent in the message, so please do not
26 * reorder the names.
27 */
28static const char *kCrosEventNames[] = {
29 "ModemManagerCommandSendFailure", // 0
30 "HwWatchdogReboot", // 1
31 "Cras.NoCodecsFoundAtBoot", // 2
Darren Krahn6e55c1152013-07-19 14:09:50 -070032 "Chaps.DatabaseCorrupted", // 3
33 "Chaps.DatabaseRepairFailure", // 4
34 "Chaps.DatabaseCreateFailure", // 5
Darren Krahn86830ba2013-07-26 13:37:20 -070035 "Attestation.OriginSpecificExhausted", // 6
Luigi Semenzatoe57398a2013-11-11 14:24:44 -080036 "SpringPowerSupply.Original.High", // 7
37 "SpringPowerSupply.Other.High", // 8
Luigi Semenzatoe8fd9682013-11-13 16:28:43 -080038 "SpringPowerSupply.Original.Low", // 9
39 "SpringPowerSupply.ChargerIdle", // 10
Darren Krahn09a15fa2014-02-07 16:51:15 -080040 "TPM.NonZeroDictionaryAttackCounter", // 11
Luigi Semenzato538e2092015-03-19 17:18:24 -070041 "TPM.EarlyResetDuringCommand", // 12
Chih-Chung Chang6844c062013-04-01 14:27:39 +080042};
43
Ken Mixter4c5daa42010-08-26 18:35:06 -070044time_t MetricsLibrary::cached_enabled_time_ = 0;
45bool MetricsLibrary::cached_enabled_ = false;
46
Bertrand SIMONNETbd3505e2015-08-04 14:04:51 -070047MetricsLibrary::MetricsLibrary() : consent_file_(metrics::kConsentFilePath) {}
Daniel Eratfd158292014-03-09 21:39:08 -070048MetricsLibrary::~MetricsLibrary() {}
49
Ken Mixtereafbbdf2010-10-01 15:38:42 -070050// We take buffer and buffer_size as parameters in order to simplify testing
51// of various alignments of the |device_name| with |buffer_size|.
52bool MetricsLibrary::IsDeviceMounted(const char* device_name,
53 const char* mounts_file,
54 char* buffer,
55 int buffer_size,
56 bool* result) {
Alex Vakulenko14595032014-08-28 14:59:56 -070057 if (buffer == nullptr || buffer_size < 1)
Ken Mixtereafbbdf2010-10-01 15:38:42 -070058 return false;
59 int mounts_fd = open(mounts_file, O_RDONLY);
60 if (mounts_fd < 0)
61 return false;
62 // match_offset describes:
63 // -1 -- not beginning of line
64 // 0..strlen(device_name)-1 -- this offset in device_name is next to match
65 // strlen(device_name) -- matched full name, just need a space.
66 int match_offset = 0;
67 bool match = false;
68 while (!match) {
69 int read_size = read(mounts_fd, buffer, buffer_size);
70 if (read_size <= 0) {
71 if (errno == -EINTR)
72 continue;
73 break;
74 }
75 for (int i = 0; i < read_size; ++i) {
76 if (buffer[i] == '\n') {
77 match_offset = 0;
78 continue;
79 }
80 if (match_offset < 0) {
81 continue;
82 }
83 if (device_name[match_offset] == '\0') {
84 if (buffer[i] == ' ') {
85 match = true;
86 break;
87 }
88 match_offset = -1;
89 continue;
90 }
91
92 if (buffer[i] == device_name[match_offset]) {
93 ++match_offset;
94 } else {
95 match_offset = -1;
96 }
97 }
98 }
99 close(mounts_fd);
100 *result = match;
101 return true;
102}
103
104bool MetricsLibrary::IsGuestMode() {
105 char buffer[256];
106 bool result = false;
107 if (!IsDeviceMounted("guestfs",
108 "/proc/mounts",
109 buffer,
110 sizeof(buffer),
111 &result)) {
112 return false;
113 }
Arkaitz Ruiz Alvarez9f1a7742011-05-26 12:22:22 -0700114 return result && (access("/var/run/state/logged-in", F_OK) == 0);
Ken Mixtereafbbdf2010-10-01 15:38:42 -0700115}
116
Ken Mixter4c5daa42010-08-26 18:35:06 -0700117bool MetricsLibrary::AreMetricsEnabled() {
Julian Pastarmov70b7abd2011-08-02 16:10:49 +0200118 static struct stat stat_buffer;
Alex Vakulenko14595032014-08-28 14:59:56 -0700119 time_t this_check_time = time(nullptr);
Ken Mixter4c5daa42010-08-26 18:35:06 -0700120 if (this_check_time != cached_enabled_time_) {
121 cached_enabled_time_ = this_check_time;
Bertrand SIMONNET5db66c32015-07-28 15:42:42 -0700122 cached_enabled_ = stat(consent_file_.c_str(), &stat_buffer) >= 0;
Ken Mixter4c5daa42010-08-26 18:35:06 -0700123 }
124 return cached_enabled_;
125}
Darin Petkov11b8eb32010-05-18 11:00:59 -0700126
Darin Petkovfc91b422010-05-12 13:05:45 -0700127void MetricsLibrary::Init() {
Bertrand SIMONNETbd3505e2015-08-04 14:04:51 -0700128 uma_events_file_ = metrics::kMetricsEventsFilePath;
Darin Petkovfc91b422010-05-12 13:05:45 -0700129}
130
Luigi Semenzato41c54502014-05-13 15:16:24 -0700131bool MetricsLibrary::SendToUMA(const std::string& name,
132 int sample,
133 int min,
134 int max,
135 int nbuckets) {
Bertrand SIMONNETd83ca802014-07-09 16:34:29 -0700136 return metrics::SerializationUtils::WriteMetricToFile(
137 *metrics::MetricSample::HistogramSample(name, sample, min, max, nbuckets)
138 .get(),
Bertrand SIMONNETbd3505e2015-08-04 14:04:51 -0700139 metrics::kMetricsEventsFilePath);
Darin Petkov65b01462010-04-14 13:32:20 -0700140}
Darin Petkov5b7dce12010-04-21 15:45:10 -0700141
Darin Petkov21cd2c52010-05-12 15:26:16 -0700142bool MetricsLibrary::SendEnumToUMA(const std::string& name, int sample,
143 int max) {
Bertrand SIMONNETd83ca802014-07-09 16:34:29 -0700144 return metrics::SerializationUtils::WriteMetricToFile(
145 *metrics::MetricSample::LinearHistogramSample(name, sample, max).get(),
Bertrand SIMONNETbd3505e2015-08-04 14:04:51 -0700146 metrics::kMetricsEventsFilePath);
Darin Petkoved824852011-01-06 10:51:47 -0800147}
148
Luigi Semenzatoa7ebeb32013-03-19 15:02:42 -0700149bool MetricsLibrary::SendSparseToUMA(const std::string& name, int sample) {
Bertrand SIMONNETd83ca802014-07-09 16:34:29 -0700150 return metrics::SerializationUtils::WriteMetricToFile(
151 *metrics::MetricSample::SparseHistogramSample(name, sample).get(),
Bertrand SIMONNETbd3505e2015-08-04 14:04:51 -0700152 metrics::kMetricsEventsFilePath);
Luigi Semenzatoa7ebeb32013-03-19 15:02:42 -0700153}
154
Darin Petkoved824852011-01-06 10:51:47 -0800155bool MetricsLibrary::SendUserActionToUMA(const std::string& action) {
Bertrand SIMONNETd83ca802014-07-09 16:34:29 -0700156 return metrics::SerializationUtils::WriteMetricToFile(
Bertrand SIMONNETbd3505e2015-08-04 14:04:51 -0700157 *metrics::MetricSample::UserActionSample(action).get(), metrics::kMetricsEventsFilePath);
Darin Petkov5b7dce12010-04-21 15:45:10 -0700158}
Ken Mixterbe2e13b2011-01-22 06:15:56 -0800159
160bool MetricsLibrary::SendCrashToUMA(const char *crash_kind) {
Bertrand SIMONNETd83ca802014-07-09 16:34:29 -0700161 return metrics::SerializationUtils::WriteMetricToFile(
Bertrand SIMONNETbd3505e2015-08-04 14:04:51 -0700162 *metrics::MetricSample::CrashSample(crash_kind).get(), metrics::kMetricsEventsFilePath);
Ken Mixterbe2e13b2011-01-22 06:15:56 -0800163}
Ken Mixterb2f17092011-07-22 14:59:51 -0700164
Luigi Semenzato32684222013-03-13 10:53:55 -0700165bool MetricsLibrary::SendCrosEventToUMA(const std::string& event) {
Alex Vakulenko788d3b62014-12-11 09:48:46 -0800166 for (size_t i = 0; i < arraysize(kCrosEventNames); i++) {
Chih-Chung Chang6844c062013-04-01 14:27:39 +0800167 if (strcmp(event.c_str(), kCrosEventNames[i]) == 0) {
168 return SendEnumToUMA(kCrosEventHistogramName, i, kCrosEventHistogramMax);
169 }
Luigi Semenzato32684222013-03-13 10:53:55 -0700170 }
Chih-Chung Chang6844c062013-04-01 14:27:39 +0800171 return false;
Luigi Semenzato32684222013-03-13 10:53:55 -0700172}