blob: b8446df00b193c4d7aececdff05173e4e1fba800 [file] [log] [blame]
/*
* Copyright (C) 2021 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "odr_metrics.h"
#include <unistd.h>
#include <chrono>
#include <fstream>
#include <memory>
#include <string>
#include <thread>
#include "base/casts.h"
#include "base/common_art_test.h"
#include "odr_metrics_record.h"
namespace art {
namespace odrefresh {
using std::chrono_literals::operator""ms; // NOLINT
class OdrMetricsTest : public CommonArtTest {
public:
void SetUp() override {
CommonArtTest::SetUp();
scratch_dir_ = std::make_unique<ScratchDir>();
metrics_file_path_ = scratch_dir_->GetPath() + "/metrics.xml";
cache_directory_ = scratch_dir_->GetPath() + "/dir";
mkdir(cache_directory_.c_str(), S_IRWXU);
}
void TearDown() override {
scratch_dir_.reset();
}
bool MetricsFileExists() const {
const char* path = metrics_file_path_.c_str();
return OS::FileExists(path);
}
const std::string GetCacheDirectory() const { return cache_directory_; }
const std::string GetMetricsFilePath() const { return metrics_file_path_; }
protected:
std::unique_ptr<ScratchDir> scratch_dir_;
std::string metrics_file_path_;
std::string cache_directory_;
};
TEST_F(OdrMetricsTest, MetricsFileIsNotCreatedIfNotEnabled) {
// Metrics file is (potentially) written in OdrMetrics destructor.
{
OdrMetrics metrics(GetCacheDirectory(), GetMetricsFilePath());
metrics.SetArtApexVersion(99);
metrics.SetTrigger(OdrMetrics::Trigger::kApexVersionMismatch);
metrics.SetStage(OdrMetrics::Stage::kCheck);
metrics.SetStatus(OdrMetrics::Status::kNoSpace);
}
EXPECT_FALSE(MetricsFileExists());
}
TEST_F(OdrMetricsTest, MetricsFileIsCreatedIfEnabled) {
// Metrics file is (potentially) written in OdrMetrics destructor.
{
OdrMetrics metrics(GetCacheDirectory(), GetMetricsFilePath());
metrics.SetEnabled(true);
metrics.SetArtApexVersion(101);
metrics.SetTrigger(OdrMetrics::Trigger::kDexFilesChanged);
metrics.SetStage(OdrMetrics::Stage::kCheck);
metrics.SetStatus(OdrMetrics::Status::kNoSpace);
}
EXPECT_TRUE(MetricsFileExists());
}
TEST_F(OdrMetricsTest, TimeValuesAreRecorded) {
OdrMetrics metrics(GetCacheDirectory(), GetMetricsFilePath());
metrics.SetArtApexVersion(1999);
metrics.SetTrigger(OdrMetrics::Trigger::kMissingArtifacts);
metrics.SetStage(OdrMetrics::Stage::kCheck);
metrics.SetStatus(OdrMetrics::Status::kOK);
// Primary boot classpath compilation time.
{
metrics.SetStage(OdrMetrics::Stage::kPrimaryBootClasspath);
ScopedOdrCompilationTimer timer(metrics);
std::this_thread::sleep_for(100ms);
}
OdrMetricsRecord record = metrics.ToRecord();
EXPECT_EQ(enum_cast<OdrMetrics::Stage>(record.stage_reached),
OdrMetrics::Stage::kPrimaryBootClasspath);
EXPECT_GT(record.primary_bcp_compilation_millis, 0);
EXPECT_LT(record.primary_bcp_compilation_millis, 300);
EXPECT_EQ(record.secondary_bcp_compilation_millis, 0);
EXPECT_EQ(record.system_server_compilation_millis, 0);
// Secondary boot classpath compilation time.
{
metrics.SetStage(OdrMetrics::Stage::kSecondaryBootClasspath);
ScopedOdrCompilationTimer timer(metrics);
std::this_thread::sleep_for(100ms);
}
record = metrics.ToRecord();
EXPECT_EQ(OdrMetrics::Stage::kSecondaryBootClasspath,
enum_cast<OdrMetrics::Stage>(record.stage_reached));
EXPECT_GT(record.primary_bcp_compilation_millis, 0);
EXPECT_GT(record.secondary_bcp_compilation_millis, 0);
EXPECT_LT(record.secondary_bcp_compilation_millis, 300);
EXPECT_EQ(record.system_server_compilation_millis, 0);
// system_server classpath compilation time.
{
metrics.SetStage(OdrMetrics::Stage::kSystemServerClasspath);
ScopedOdrCompilationTimer timer(metrics);
std::this_thread::sleep_for(100ms);
}
record = metrics.ToRecord();
EXPECT_EQ(OdrMetrics::Stage::kSystemServerClasspath,
enum_cast<OdrMetrics::Stage>(record.stage_reached));
EXPECT_GT(record.primary_bcp_compilation_millis, 0);
EXPECT_GT(record.secondary_bcp_compilation_millis, 0);
EXPECT_GT(record.system_server_compilation_millis, 0);
EXPECT_LT(record.system_server_compilation_millis, 300);
}
TEST_F(OdrMetricsTest, CacheSpaceValuesAreUpdated) {
OdrMetrics metrics(GetCacheDirectory(), GetMetricsFilePath());
metrics.CaptureSpaceFreeEnd();
OdrMetricsRecord record = metrics.ToRecord();
EXPECT_GT(record.cache_space_free_start_mib, 0);
EXPECT_GT(record.cache_space_free_end_mib, 0);
}
TEST_F(OdrMetricsTest, PrimaryBcpResultWithValue) {
OdrMetrics metrics(GetCacheDirectory(), GetMetricsFilePath());
metrics.SetStage(OdrMetrics::Stage::kPrimaryBootClasspath);
metrics.SetDex2OatResult({
.status = ExecResult::Status::kExited,
.exit_code = 0,
.signal = 0
});
OdrMetricsRecord record = metrics.ToRecord();
EXPECT_EQ(record.primary_bcp_dex2oat_result.status, ExecResult::Status::kExited);
EXPECT_EQ(record.primary_bcp_dex2oat_result.exit_code, 0);
EXPECT_EQ(record.primary_bcp_dex2oat_result.signal, 0);
}
TEST_F(OdrMetricsTest, PrimaryBcpResultWithoutValue) {
OdrMetrics metrics(GetCacheDirectory(), GetMetricsFilePath());
OdrMetricsRecord record = metrics.ToRecord();
EXPECT_EQ(record.primary_bcp_dex2oat_result.status, kExecResultNotRun);
EXPECT_EQ(record.primary_bcp_dex2oat_result.exit_code, -1);
EXPECT_EQ(record.primary_bcp_dex2oat_result.signal, 0);
}
TEST_F(OdrMetricsTest, SecondaryBcpResultWithValue) {
OdrMetrics metrics(GetCacheDirectory(), GetMetricsFilePath());
metrics.SetStage(OdrMetrics::Stage::kPrimaryBootClasspath);
metrics.SetDex2OatResult({
.status = ExecResult::Status::kExited,
.exit_code = 0,
.signal = 0
});
metrics.SetStage(OdrMetrics::Stage::kSecondaryBootClasspath);
metrics.SetDex2OatResult({
.status = ExecResult::Status::kTimedOut,
.exit_code = 3,
.signal = 0
});
OdrMetricsRecord record = metrics.ToRecord();
EXPECT_EQ(record.primary_bcp_dex2oat_result.status, ExecResult::Status::kExited);
EXPECT_EQ(record.primary_bcp_dex2oat_result.exit_code, 0);
EXPECT_EQ(record.primary_bcp_dex2oat_result.signal, 0);
EXPECT_EQ(record.secondary_bcp_dex2oat_result.status, ExecResult::Status::kTimedOut);
EXPECT_EQ(record.secondary_bcp_dex2oat_result.exit_code, 3);
EXPECT_EQ(record.secondary_bcp_dex2oat_result.signal, 0);
}
TEST_F(OdrMetricsTest, SecondaryBcpResultWithoutValue) {
OdrMetrics metrics(GetCacheDirectory(), GetMetricsFilePath());
metrics.SetStage(OdrMetrics::Stage::kPrimaryBootClasspath);
metrics.SetDex2OatResult({
.status = ExecResult::Status::kExited,
.exit_code = 0,
.signal = 0
});
OdrMetricsRecord record = metrics.ToRecord();
EXPECT_EQ(record.primary_bcp_dex2oat_result.status, ExecResult::Status::kExited);
EXPECT_EQ(record.primary_bcp_dex2oat_result.exit_code, 0);
EXPECT_EQ(record.primary_bcp_dex2oat_result.signal, 0);
EXPECT_EQ(record.secondary_bcp_dex2oat_result.status, kExecResultNotRun);
EXPECT_EQ(record.secondary_bcp_dex2oat_result.exit_code, -1);
EXPECT_EQ(record.secondary_bcp_dex2oat_result.signal, 0);
}
TEST_F(OdrMetricsTest, SystemServerResultWithValue) {
OdrMetrics metrics(GetCacheDirectory(), GetMetricsFilePath());
metrics.SetStage(OdrMetrics::Stage::kPrimaryBootClasspath);
metrics.SetDex2OatResult({
.status = ExecResult::Status::kExited,
.exit_code = 0,
.signal = 0
});
metrics.SetStage(OdrMetrics::Stage::kSecondaryBootClasspath);
metrics.SetDex2OatResult({
.status = ExecResult::Status::kTimedOut,
.exit_code = 3,
.signal = 0
});
metrics.SetStage(OdrMetrics::Stage::kSystemServerClasspath);
metrics.SetDex2OatResult({
.status = ExecResult::Status::kSignaled,
.exit_code = 2,
.signal = 9
});
OdrMetricsRecord record = metrics.ToRecord();
EXPECT_EQ(record.primary_bcp_dex2oat_result.status, ExecResult::Status::kExited);
EXPECT_EQ(record.primary_bcp_dex2oat_result.exit_code, 0);
EXPECT_EQ(record.primary_bcp_dex2oat_result.signal, 0);
EXPECT_EQ(record.secondary_bcp_dex2oat_result.status, ExecResult::Status::kTimedOut);
EXPECT_EQ(record.secondary_bcp_dex2oat_result.exit_code, 3);
EXPECT_EQ(record.secondary_bcp_dex2oat_result.signal, 0);
EXPECT_EQ(record.system_server_dex2oat_result.status, ExecResult::Status::kSignaled);
EXPECT_EQ(record.system_server_dex2oat_result.exit_code, 2);
EXPECT_EQ(record.system_server_dex2oat_result.signal, 9);
}
TEST_F(OdrMetricsTest, SystemServerResultWithoutValue) {
OdrMetrics metrics(GetCacheDirectory(), GetMetricsFilePath());
metrics.SetStage(OdrMetrics::Stage::kPrimaryBootClasspath);
metrics.SetDex2OatResult({
.status = ExecResult::Status::kExited,
.exit_code = 0,
.signal = 0
});
metrics.SetStage(OdrMetrics::Stage::kSecondaryBootClasspath);
metrics.SetDex2OatResult({
.status = ExecResult::Status::kTimedOut,
.exit_code = 3,
.signal = 0
});
OdrMetricsRecord record = metrics.ToRecord();
EXPECT_EQ(record.primary_bcp_dex2oat_result.status, ExecResult::Status::kExited);
EXPECT_EQ(record.primary_bcp_dex2oat_result.exit_code, 0);
EXPECT_EQ(record.primary_bcp_dex2oat_result.signal, 0);
EXPECT_EQ(record.secondary_bcp_dex2oat_result.status, ExecResult::Status::kTimedOut);
EXPECT_EQ(record.secondary_bcp_dex2oat_result.exit_code, 3);
EXPECT_EQ(record.secondary_bcp_dex2oat_result.signal, 0);
EXPECT_EQ(record.system_server_dex2oat_result.status, kExecResultNotRun);
EXPECT_EQ(record.system_server_dex2oat_result.exit_code, -1);
EXPECT_EQ(record.system_server_dex2oat_result.signal, 0);
}
} // namespace odrefresh
} // namespace art