xref: /aosp_15_r20/art/odrefresh/odr_compilation_log_test.cc (revision 795d594fd825385562da6b089ea9b2033f3abf5a)
1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker  * Copyright (C) 2021 The Android Open Source Project
3*795d594fSAndroid Build Coastguard Worker  *
4*795d594fSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*795d594fSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*795d594fSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*795d594fSAndroid Build Coastguard Worker  *
8*795d594fSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*795d594fSAndroid Build Coastguard Worker  *
10*795d594fSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*795d594fSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*795d594fSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*795d594fSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*795d594fSAndroid Build Coastguard Worker  * limitations under the License.
15*795d594fSAndroid Build Coastguard Worker  */
16*795d594fSAndroid Build Coastguard Worker 
17*795d594fSAndroid Build Coastguard Worker #include <odr_compilation_log.h>
18*795d594fSAndroid Build Coastguard Worker 
19*795d594fSAndroid Build Coastguard Worker #include <time.h>
20*795d594fSAndroid Build Coastguard Worker 
21*795d594fSAndroid Build Coastguard Worker #include <cstdint>
22*795d594fSAndroid Build Coastguard Worker #include <ctime>
23*795d594fSAndroid Build Coastguard Worker #include <iosfwd>
24*795d594fSAndroid Build Coastguard Worker #include <istream>
25*795d594fSAndroid Build Coastguard Worker #include <limits>
26*795d594fSAndroid Build Coastguard Worker #include <ostream>
27*795d594fSAndroid Build Coastguard Worker #include <sstream>
28*795d594fSAndroid Build Coastguard Worker #include <string>
29*795d594fSAndroid Build Coastguard Worker #include <vector>
30*795d594fSAndroid Build Coastguard Worker 
31*795d594fSAndroid Build Coastguard Worker #include "android-base/file.h"
32*795d594fSAndroid Build Coastguard Worker #include "base/common_art_test.h"
33*795d594fSAndroid Build Coastguard Worker #include "odrefresh/odrefresh.h"
34*795d594fSAndroid Build Coastguard Worker #include "odr_metrics.h"
35*795d594fSAndroid Build Coastguard Worker 
36*795d594fSAndroid Build Coastguard Worker namespace art {
37*795d594fSAndroid Build Coastguard Worker namespace odrefresh {
38*795d594fSAndroid Build Coastguard Worker 
39*795d594fSAndroid Build Coastguard Worker const time_t kSecondsPerDay = 86'400;
40*795d594fSAndroid Build Coastguard Worker 
41*795d594fSAndroid Build Coastguard Worker class OdrCompilationLogTest : public CommonArtTest {};
42*795d594fSAndroid Build Coastguard Worker 
TEST(OdrCompilationLogEntry,Equality)43*795d594fSAndroid Build Coastguard Worker TEST(OdrCompilationLogEntry, Equality) {
44*795d594fSAndroid Build Coastguard Worker   OdrCompilationLogEntry a{1, 2, 3, 4, 5};
45*795d594fSAndroid Build Coastguard Worker 
46*795d594fSAndroid Build Coastguard Worker   ASSERT_EQ(a, (OdrCompilationLogEntry{1, 2, 3, 4, 5}));
47*795d594fSAndroid Build Coastguard Worker   ASSERT_NE(a, (OdrCompilationLogEntry{9, 2, 3, 4, 5}));
48*795d594fSAndroid Build Coastguard Worker   ASSERT_NE(a, (OdrCompilationLogEntry{1, 9, 3, 4, 5}));
49*795d594fSAndroid Build Coastguard Worker   ASSERT_NE(a, (OdrCompilationLogEntry{1, 2, 9, 4, 5}));
50*795d594fSAndroid Build Coastguard Worker   ASSERT_NE(a, (OdrCompilationLogEntry{2, 2, 3, 9, 5}));
51*795d594fSAndroid Build Coastguard Worker   ASSERT_NE(a, (OdrCompilationLogEntry{2, 2, 3, 5, 9}));
52*795d594fSAndroid Build Coastguard Worker }
53*795d594fSAndroid Build Coastguard Worker 
TEST(OdrCompilationLogEntry,InputOutput)54*795d594fSAndroid Build Coastguard Worker TEST(OdrCompilationLogEntry, InputOutput) {
55*795d594fSAndroid Build Coastguard Worker   const OdrCompilationLogEntry entries[] = {
56*795d594fSAndroid Build Coastguard Worker       {1, 2, 3, 4, 5},
57*795d594fSAndroid Build Coastguard Worker       {std::numeric_limits<int64_t>::min(),
58*795d594fSAndroid Build Coastguard Worker        std::numeric_limits<int64_t>::min(),
59*795d594fSAndroid Build Coastguard Worker        std::numeric_limits<int32_t>::min(),
60*795d594fSAndroid Build Coastguard Worker        std::numeric_limits<time_t>::min(),
61*795d594fSAndroid Build Coastguard Worker        std::numeric_limits<int32_t>::min()},
62*795d594fSAndroid Build Coastguard Worker       {std::numeric_limits<int64_t>::max(),
63*795d594fSAndroid Build Coastguard Worker        std::numeric_limits<int64_t>::max(),
64*795d594fSAndroid Build Coastguard Worker        std::numeric_limits<int32_t>::max(),
65*795d594fSAndroid Build Coastguard Worker        std::numeric_limits<time_t>::max(),
66*795d594fSAndroid Build Coastguard Worker        std::numeric_limits<int32_t>::max()},
67*795d594fSAndroid Build Coastguard Worker        {0, 0, 0, 0, 0},
68*795d594fSAndroid Build Coastguard Worker       {0x7fedcba9'87654321, 0x5a5a5a5a'5a5a5a5a, 0x12345678, 0x2346789, 0x76543210}
69*795d594fSAndroid Build Coastguard Worker   };
70*795d594fSAndroid Build Coastguard Worker   for (const auto& entry : entries) {
71*795d594fSAndroid Build Coastguard Worker     std::stringstream ss;
72*795d594fSAndroid Build Coastguard Worker     ss << entry;
73*795d594fSAndroid Build Coastguard Worker     OdrCompilationLogEntry actual;
74*795d594fSAndroid Build Coastguard Worker     ss >> actual;
75*795d594fSAndroid Build Coastguard Worker     ASSERT_EQ(entry, actual);
76*795d594fSAndroid Build Coastguard Worker   }
77*795d594fSAndroid Build Coastguard Worker }
78*795d594fSAndroid Build Coastguard Worker 
TEST(OdrCompilationLogEntry,TruncatedInput)79*795d594fSAndroid Build Coastguard Worker TEST(OdrCompilationLogEntry, TruncatedInput) {
80*795d594fSAndroid Build Coastguard Worker   std::stringstream ss;
81*795d594fSAndroid Build Coastguard Worker   ss << "1 2";
82*795d594fSAndroid Build Coastguard Worker 
83*795d594fSAndroid Build Coastguard Worker   OdrCompilationLogEntry entry;
84*795d594fSAndroid Build Coastguard Worker   ss >> entry;
85*795d594fSAndroid Build Coastguard Worker 
86*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(ss.fail());
87*795d594fSAndroid Build Coastguard Worker   ASSERT_FALSE(ss.bad());
88*795d594fSAndroid Build Coastguard Worker }
89*795d594fSAndroid Build Coastguard Worker 
TEST(OdrCompilationLogEntry,ReadMultiple)90*795d594fSAndroid Build Coastguard Worker TEST(OdrCompilationLogEntry, ReadMultiple) {
91*795d594fSAndroid Build Coastguard Worker   std::stringstream ss;
92*795d594fSAndroid Build Coastguard Worker   ss << "0 1 2 3 4\n5 6 7 8 9\n";
93*795d594fSAndroid Build Coastguard Worker 
94*795d594fSAndroid Build Coastguard Worker   OdrCompilationLogEntry entry0, entry1;
95*795d594fSAndroid Build Coastguard Worker   ss >> entry0 >> entry1;
96*795d594fSAndroid Build Coastguard Worker   ASSERT_EQ(entry0, (OdrCompilationLogEntry{0, 1, 2, 3, 4}));
97*795d594fSAndroid Build Coastguard Worker   ASSERT_EQ(entry1, (OdrCompilationLogEntry{5, 6, 7, 8, 9}));
98*795d594fSAndroid Build Coastguard Worker 
99*795d594fSAndroid Build Coastguard Worker   ASSERT_FALSE(ss.fail());
100*795d594fSAndroid Build Coastguard Worker   ASSERT_FALSE(ss.bad());
101*795d594fSAndroid Build Coastguard Worker }
102*795d594fSAndroid Build Coastguard Worker 
TEST(OdrCompilationLog,ShouldAttemptCompile)103*795d594fSAndroid Build Coastguard Worker TEST(OdrCompilationLog, ShouldAttemptCompile) {
104*795d594fSAndroid Build Coastguard Worker   OdrCompilationLog ocl(/*compilation_log_path=*/nullptr);
105*795d594fSAndroid Build Coastguard Worker 
106*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, 0));
107*795d594fSAndroid Build Coastguard Worker 
108*795d594fSAndroid Build Coastguard Worker   ocl.Log(
109*795d594fSAndroid Build Coastguard Worker       /*apex_version=*/1,
110*795d594fSAndroid Build Coastguard Worker       /*last_update_millis=*/762,
111*795d594fSAndroid Build Coastguard Worker       OdrMetrics::Trigger::kApexVersionMismatch,
112*795d594fSAndroid Build Coastguard Worker       ExitCode::kCompilationFailed);
113*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kApexVersionMismatch));
114*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kDexFilesChanged));
115*795d594fSAndroid Build Coastguard Worker   ASSERT_FALSE(ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown));
116*795d594fSAndroid Build Coastguard Worker }
117*795d594fSAndroid Build Coastguard Worker 
TEST(OdrCompilationLog,BackOffNoHistory)118*795d594fSAndroid Build Coastguard Worker TEST(OdrCompilationLog, BackOffNoHistory) {
119*795d594fSAndroid Build Coastguard Worker   time_t start_time;
120*795d594fSAndroid Build Coastguard Worker   time(&start_time);
121*795d594fSAndroid Build Coastguard Worker 
122*795d594fSAndroid Build Coastguard Worker   OdrCompilationLog ocl(/*compilation_log_path=*/nullptr);
123*795d594fSAndroid Build Coastguard Worker 
124*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time));
125*795d594fSAndroid Build Coastguard Worker 
126*795d594fSAndroid Build Coastguard Worker   // Start log
127*795d594fSAndroid Build Coastguard Worker   ocl.Log(/*apex_version=*/1,
128*795d594fSAndroid Build Coastguard Worker           /*last_update_millis=*/0,
129*795d594fSAndroid Build Coastguard Worker           OdrMetrics::Trigger::kApexVersionMismatch,
130*795d594fSAndroid Build Coastguard Worker           start_time,
131*795d594fSAndroid Build Coastguard Worker           ExitCode::kCompilationFailed);
132*795d594fSAndroid Build Coastguard Worker   ASSERT_FALSE(ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time));
133*795d594fSAndroid Build Coastguard Worker   ASSERT_FALSE(
134*795d594fSAndroid Build Coastguard Worker       ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + kSecondsPerDay / 2));
135*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + kSecondsPerDay));
136*795d594fSAndroid Build Coastguard Worker 
137*795d594fSAndroid Build Coastguard Worker   // Add one more log entry
138*795d594fSAndroid Build Coastguard Worker   ocl.Log(/*apex_version=*/1,
139*795d594fSAndroid Build Coastguard Worker           /*last_update_millis=*/0,
140*795d594fSAndroid Build Coastguard Worker           OdrMetrics::Trigger::kApexVersionMismatch,
141*795d594fSAndroid Build Coastguard Worker           start_time,
142*795d594fSAndroid Build Coastguard Worker           ExitCode::kCompilationFailed);
143*795d594fSAndroid Build Coastguard Worker   ASSERT_FALSE(
144*795d594fSAndroid Build Coastguard Worker       ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + kSecondsPerDay));
145*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(
146*795d594fSAndroid Build Coastguard Worker       ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + 2 * kSecondsPerDay));
147*795d594fSAndroid Build Coastguard Worker 
148*795d594fSAndroid Build Coastguard Worker   // One more.
149*795d594fSAndroid Build Coastguard Worker   ocl.Log(/*apex_version=*/1,
150*795d594fSAndroid Build Coastguard Worker           /*last_update_millis=*/0,
151*795d594fSAndroid Build Coastguard Worker           OdrMetrics::Trigger::kApexVersionMismatch,
152*795d594fSAndroid Build Coastguard Worker           start_time,
153*795d594fSAndroid Build Coastguard Worker           ExitCode::kCompilationFailed);
154*795d594fSAndroid Build Coastguard Worker   ASSERT_FALSE(
155*795d594fSAndroid Build Coastguard Worker       ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + 3 * kSecondsPerDay));
156*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(
157*795d594fSAndroid Build Coastguard Worker       ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + 4 * kSecondsPerDay));
158*795d594fSAndroid Build Coastguard Worker 
159*795d594fSAndroid Build Coastguard Worker   // And one for the road.
160*795d594fSAndroid Build Coastguard Worker   ocl.Log(/*apex_version=*/1,
161*795d594fSAndroid Build Coastguard Worker           /*last_update_millis=*/0,
162*795d594fSAndroid Build Coastguard Worker           OdrMetrics::Trigger::kApexVersionMismatch,
163*795d594fSAndroid Build Coastguard Worker           start_time,
164*795d594fSAndroid Build Coastguard Worker           ExitCode::kCompilationFailed);
165*795d594fSAndroid Build Coastguard Worker   ASSERT_FALSE(
166*795d594fSAndroid Build Coastguard Worker       ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + 7 * kSecondsPerDay));
167*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(
168*795d594fSAndroid Build Coastguard Worker       ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + 8 * kSecondsPerDay));
169*795d594fSAndroid Build Coastguard Worker }
170*795d594fSAndroid Build Coastguard Worker 
TEST(OdrCompilationLog,BackOffHappyHistory)171*795d594fSAndroid Build Coastguard Worker TEST(OdrCompilationLog, BackOffHappyHistory) {
172*795d594fSAndroid Build Coastguard Worker   time_t start_time;
173*795d594fSAndroid Build Coastguard Worker   time(&start_time);
174*795d594fSAndroid Build Coastguard Worker 
175*795d594fSAndroid Build Coastguard Worker   OdrCompilationLog ocl(/*compilation_log_path=*/nullptr);
176*795d594fSAndroid Build Coastguard Worker 
177*795d594fSAndroid Build Coastguard Worker   // Start log with a successful entry.
178*795d594fSAndroid Build Coastguard Worker   ocl.Log(/*apex_version=*/1,
179*795d594fSAndroid Build Coastguard Worker           /*last_update_millis=*/0,
180*795d594fSAndroid Build Coastguard Worker           OdrMetrics::Trigger::kApexVersionMismatch,
181*795d594fSAndroid Build Coastguard Worker           start_time,
182*795d594fSAndroid Build Coastguard Worker           ExitCode::kCompilationSuccess);
183*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time));
184*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(
185*795d594fSAndroid Build Coastguard Worker       ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + kSecondsPerDay / 4));
186*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(
187*795d594fSAndroid Build Coastguard Worker       ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + kSecondsPerDay / 2));
188*795d594fSAndroid Build Coastguard Worker 
189*795d594fSAndroid Build Coastguard Worker   // Add a log entry for a failed compilation.
190*795d594fSAndroid Build Coastguard Worker   ocl.Log(/*apex_version=*/1,
191*795d594fSAndroid Build Coastguard Worker           /*last_update_millis=*/0,
192*795d594fSAndroid Build Coastguard Worker           OdrMetrics::Trigger::kApexVersionMismatch,
193*795d594fSAndroid Build Coastguard Worker           start_time,
194*795d594fSAndroid Build Coastguard Worker           ExitCode::kCompilationFailed);
195*795d594fSAndroid Build Coastguard Worker   ASSERT_FALSE(
196*795d594fSAndroid Build Coastguard Worker       ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + kSecondsPerDay / 2));
197*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + kSecondsPerDay));
198*795d594fSAndroid Build Coastguard Worker }
199*795d594fSAndroid Build Coastguard Worker 
TEST_F(OdrCompilationLogTest,LogNumberOfEntriesAndPeek)200*795d594fSAndroid Build Coastguard Worker TEST_F(OdrCompilationLogTest, LogNumberOfEntriesAndPeek) {
201*795d594fSAndroid Build Coastguard Worker   OdrCompilationLog ocl(/*compilation_log_path=*/nullptr);
202*795d594fSAndroid Build Coastguard Worker 
203*795d594fSAndroid Build Coastguard Worker   std::vector<OdrCompilationLogEntry> entries = {
204*795d594fSAndroid Build Coastguard Worker     { 0, 1, 2, 3, 4 },
205*795d594fSAndroid Build Coastguard Worker     { 1, 2, 3, 4, 5 },
206*795d594fSAndroid Build Coastguard Worker     { 2, 3, 4, 5, 6 },
207*795d594fSAndroid Build Coastguard Worker     { 3, 4, 5, 6, 7 },
208*795d594fSAndroid Build Coastguard Worker     { 4, 5, 6, 7, 8 },
209*795d594fSAndroid Build Coastguard Worker     { 5, 6, 7, 8, 9 },
210*795d594fSAndroid Build Coastguard Worker     { 6, 7, 8, 9, 10 }
211*795d594fSAndroid Build Coastguard Worker   };
212*795d594fSAndroid Build Coastguard Worker 
213*795d594fSAndroid Build Coastguard Worker   for (size_t i = 0; i < entries.size(); ++i) {
214*795d594fSAndroid Build Coastguard Worker     OdrCompilationLogEntry& e = entries[i];
215*795d594fSAndroid Build Coastguard Worker     ocl.Log(e.apex_version,
216*795d594fSAndroid Build Coastguard Worker             e.last_update_millis,
217*795d594fSAndroid Build Coastguard Worker             static_cast<OdrMetrics::Trigger>(e.trigger),
218*795d594fSAndroid Build Coastguard Worker             e.when,
219*795d594fSAndroid Build Coastguard Worker             static_cast<ExitCode>(e.exit_code));
220*795d594fSAndroid Build Coastguard Worker     if (i < OdrCompilationLog::kMaxLoggedEntries) {
221*795d594fSAndroid Build Coastguard Worker       ASSERT_EQ(i + 1, ocl.NumberOfEntries());
222*795d594fSAndroid Build Coastguard Worker     } else {
223*795d594fSAndroid Build Coastguard Worker       ASSERT_EQ(OdrCompilationLog::kMaxLoggedEntries, ocl.NumberOfEntries());
224*795d594fSAndroid Build Coastguard Worker     }
225*795d594fSAndroid Build Coastguard Worker 
226*795d594fSAndroid Build Coastguard Worker     for (size_t j = 0; j < ocl.NumberOfEntries(); ++j) {
227*795d594fSAndroid Build Coastguard Worker       const OdrCompilationLogEntry* logged = ocl.Peek(j);
228*795d594fSAndroid Build Coastguard Worker       ASSERT_TRUE(logged != nullptr);
229*795d594fSAndroid Build Coastguard Worker       const OdrCompilationLogEntry& expected = entries[i + 1 - ocl.NumberOfEntries() + j];
230*795d594fSAndroid Build Coastguard Worker       ASSERT_EQ(expected, *logged);
231*795d594fSAndroid Build Coastguard Worker     }
232*795d594fSAndroid Build Coastguard Worker   }
233*795d594fSAndroid Build Coastguard Worker }
234*795d594fSAndroid Build Coastguard Worker 
TEST_F(OdrCompilationLogTest,LogReadWrite)235*795d594fSAndroid Build Coastguard Worker TEST_F(OdrCompilationLogTest, LogReadWrite) {
236*795d594fSAndroid Build Coastguard Worker   std::vector<OdrCompilationLogEntry> entries = {
237*795d594fSAndroid Build Coastguard Worker     { 0, 1, 2, 3, 4 },
238*795d594fSAndroid Build Coastguard Worker     { 1, 2, 3, 4, 5 },
239*795d594fSAndroid Build Coastguard Worker     { 2, 3, 4, 5, 6 },
240*795d594fSAndroid Build Coastguard Worker     { 3, 4, 5, 6, 7 },
241*795d594fSAndroid Build Coastguard Worker     { 4, 5, 6, 7, 8 },
242*795d594fSAndroid Build Coastguard Worker     { 5, 6, 7, 8, 9 },
243*795d594fSAndroid Build Coastguard Worker     { 6, 7, 8, 9, 10 }
244*795d594fSAndroid Build Coastguard Worker   };
245*795d594fSAndroid Build Coastguard Worker 
246*795d594fSAndroid Build Coastguard Worker   ScratchFile scratch_file;
247*795d594fSAndroid Build Coastguard Worker   scratch_file.Close();
248*795d594fSAndroid Build Coastguard Worker 
249*795d594fSAndroid Build Coastguard Worker   for (size_t i = 0; i < entries.size(); ++i) {
250*795d594fSAndroid Build Coastguard Worker     {
251*795d594fSAndroid Build Coastguard Worker       OdrCompilationLog ocl(scratch_file.GetFilename().c_str());
252*795d594fSAndroid Build Coastguard Worker       OdrCompilationLogEntry& e = entries[i];
253*795d594fSAndroid Build Coastguard Worker       ocl.Log(e.apex_version,
254*795d594fSAndroid Build Coastguard Worker               e.last_update_millis,
255*795d594fSAndroid Build Coastguard Worker               static_cast<OdrMetrics::Trigger>(e.trigger),
256*795d594fSAndroid Build Coastguard Worker               e.when,
257*795d594fSAndroid Build Coastguard Worker               static_cast<ExitCode>(e.exit_code));
258*795d594fSAndroid Build Coastguard Worker     }
259*795d594fSAndroid Build Coastguard Worker 
260*795d594fSAndroid Build Coastguard Worker     {
261*795d594fSAndroid Build Coastguard Worker       OdrCompilationLog ocl(scratch_file.GetFilename().c_str());
262*795d594fSAndroid Build Coastguard Worker       if (i < OdrCompilationLog::kMaxLoggedEntries) {
263*795d594fSAndroid Build Coastguard Worker         ASSERT_EQ(i + 1, ocl.NumberOfEntries());
264*795d594fSAndroid Build Coastguard Worker       } else {
265*795d594fSAndroid Build Coastguard Worker         ASSERT_EQ(OdrCompilationLog::kMaxLoggedEntries, ocl.NumberOfEntries());
266*795d594fSAndroid Build Coastguard Worker       }
267*795d594fSAndroid Build Coastguard Worker 
268*795d594fSAndroid Build Coastguard Worker       for (size_t j = 0; j < ocl.NumberOfEntries(); ++j) {
269*795d594fSAndroid Build Coastguard Worker         const OdrCompilationLogEntry* logged = ocl.Peek(j);
270*795d594fSAndroid Build Coastguard Worker         ASSERT_TRUE(logged != nullptr);
271*795d594fSAndroid Build Coastguard Worker         const OdrCompilationLogEntry& expected = entries[i + 1 - ocl.NumberOfEntries() + j];
272*795d594fSAndroid Build Coastguard Worker         ASSERT_EQ(expected, *logged);
273*795d594fSAndroid Build Coastguard Worker       }
274*795d594fSAndroid Build Coastguard Worker     }
275*795d594fSAndroid Build Coastguard Worker   }
276*795d594fSAndroid Build Coastguard Worker }
277*795d594fSAndroid Build Coastguard Worker 
TEST_F(OdrCompilationLogTest,BackoffBasedOnLog)278*795d594fSAndroid Build Coastguard Worker TEST_F(OdrCompilationLogTest, BackoffBasedOnLog) {
279*795d594fSAndroid Build Coastguard Worker   time_t start_time;
280*795d594fSAndroid Build Coastguard Worker   time(&start_time);
281*795d594fSAndroid Build Coastguard Worker 
282*795d594fSAndroid Build Coastguard Worker   ScratchFile scratch_file;
283*795d594fSAndroid Build Coastguard Worker   scratch_file.Close();
284*795d594fSAndroid Build Coastguard Worker 
285*795d594fSAndroid Build Coastguard Worker   const char* log_path = scratch_file.GetFilename().c_str();
286*795d594fSAndroid Build Coastguard Worker   {
287*795d594fSAndroid Build Coastguard Worker     OdrCompilationLog ocl(log_path);
288*795d594fSAndroid Build Coastguard Worker 
289*795d594fSAndroid Build Coastguard Worker     ASSERT_TRUE(ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time));
290*795d594fSAndroid Build Coastguard Worker   }
291*795d594fSAndroid Build Coastguard Worker 
292*795d594fSAndroid Build Coastguard Worker   {
293*795d594fSAndroid Build Coastguard Worker     OdrCompilationLog ocl(log_path);
294*795d594fSAndroid Build Coastguard Worker 
295*795d594fSAndroid Build Coastguard Worker     // Start log
296*795d594fSAndroid Build Coastguard Worker     ocl.Log(/*apex_version=*/1,
297*795d594fSAndroid Build Coastguard Worker             /*last_update_millis=*/0,
298*795d594fSAndroid Build Coastguard Worker             OdrMetrics::Trigger::kApexVersionMismatch,
299*795d594fSAndroid Build Coastguard Worker             start_time,
300*795d594fSAndroid Build Coastguard Worker             ExitCode::kCompilationFailed);
301*795d594fSAndroid Build Coastguard Worker   }
302*795d594fSAndroid Build Coastguard Worker 
303*795d594fSAndroid Build Coastguard Worker   {
304*795d594fSAndroid Build Coastguard Worker     OdrCompilationLog ocl(log_path);
305*795d594fSAndroid Build Coastguard Worker     ASSERT_FALSE(ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time));
306*795d594fSAndroid Build Coastguard Worker     ASSERT_FALSE(
307*795d594fSAndroid Build Coastguard Worker         ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + kSecondsPerDay / 2));
308*795d594fSAndroid Build Coastguard Worker     ASSERT_TRUE(
309*795d594fSAndroid Build Coastguard Worker         ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + kSecondsPerDay));
310*795d594fSAndroid Build Coastguard Worker   }
311*795d594fSAndroid Build Coastguard Worker 
312*795d594fSAndroid Build Coastguard Worker   {
313*795d594fSAndroid Build Coastguard Worker     // Add one more log entry
314*795d594fSAndroid Build Coastguard Worker     OdrCompilationLog ocl(log_path);
315*795d594fSAndroid Build Coastguard Worker     ocl.Log(/*apex_version=*/1,
316*795d594fSAndroid Build Coastguard Worker             /*last_update_millis=*/0,
317*795d594fSAndroid Build Coastguard Worker             OdrMetrics::Trigger::kApexVersionMismatch,
318*795d594fSAndroid Build Coastguard Worker             start_time,
319*795d594fSAndroid Build Coastguard Worker             ExitCode::kCompilationFailed);
320*795d594fSAndroid Build Coastguard Worker   }
321*795d594fSAndroid Build Coastguard Worker 
322*795d594fSAndroid Build Coastguard Worker   {
323*795d594fSAndroid Build Coastguard Worker     OdrCompilationLog ocl(log_path);
324*795d594fSAndroid Build Coastguard Worker 
325*795d594fSAndroid Build Coastguard Worker     ASSERT_FALSE(
326*795d594fSAndroid Build Coastguard Worker         ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + kSecondsPerDay));
327*795d594fSAndroid Build Coastguard Worker     ASSERT_TRUE(
328*795d594fSAndroid Build Coastguard Worker         ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + 2 * kSecondsPerDay));
329*795d594fSAndroid Build Coastguard Worker   }
330*795d594fSAndroid Build Coastguard Worker 
331*795d594fSAndroid Build Coastguard Worker   {
332*795d594fSAndroid Build Coastguard Worker     // One more log entry.
333*795d594fSAndroid Build Coastguard Worker     OdrCompilationLog ocl(log_path);
334*795d594fSAndroid Build Coastguard Worker     ocl.Log(/*apex_version=*/1,
335*795d594fSAndroid Build Coastguard Worker             /*last_update_millis=*/0,
336*795d594fSAndroid Build Coastguard Worker             OdrMetrics::Trigger::kApexVersionMismatch,
337*795d594fSAndroid Build Coastguard Worker             start_time,
338*795d594fSAndroid Build Coastguard Worker             ExitCode::kCompilationFailed);
339*795d594fSAndroid Build Coastguard Worker   }
340*795d594fSAndroid Build Coastguard Worker 
341*795d594fSAndroid Build Coastguard Worker   {
342*795d594fSAndroid Build Coastguard Worker     OdrCompilationLog ocl(log_path);
343*795d594fSAndroid Build Coastguard Worker     ASSERT_FALSE(
344*795d594fSAndroid Build Coastguard Worker         ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + 3 * kSecondsPerDay));
345*795d594fSAndroid Build Coastguard Worker     ASSERT_TRUE(
346*795d594fSAndroid Build Coastguard Worker         ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + 4 * kSecondsPerDay));
347*795d594fSAndroid Build Coastguard Worker   }
348*795d594fSAndroid Build Coastguard Worker 
349*795d594fSAndroid Build Coastguard Worker   {
350*795d594fSAndroid Build Coastguard Worker     // And one for the road.
351*795d594fSAndroid Build Coastguard Worker     OdrCompilationLog ocl(log_path);
352*795d594fSAndroid Build Coastguard Worker     ocl.Log(/*apex_version=*/1,
353*795d594fSAndroid Build Coastguard Worker             /*last_update_millis=*/0,
354*795d594fSAndroid Build Coastguard Worker             OdrMetrics::Trigger::kApexVersionMismatch,
355*795d594fSAndroid Build Coastguard Worker             start_time,
356*795d594fSAndroid Build Coastguard Worker             ExitCode::kCompilationFailed);
357*795d594fSAndroid Build Coastguard Worker   }
358*795d594fSAndroid Build Coastguard Worker 
359*795d594fSAndroid Build Coastguard Worker   {
360*795d594fSAndroid Build Coastguard Worker     OdrCompilationLog ocl(log_path);
361*795d594fSAndroid Build Coastguard Worker     ASSERT_FALSE(
362*795d594fSAndroid Build Coastguard Worker         ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + 7 * kSecondsPerDay));
363*795d594fSAndroid Build Coastguard Worker     ASSERT_TRUE(
364*795d594fSAndroid Build Coastguard Worker         ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time + 8 * kSecondsPerDay));
365*795d594fSAndroid Build Coastguard Worker   }
366*795d594fSAndroid Build Coastguard Worker }
367*795d594fSAndroid Build Coastguard Worker 
TEST_F(OdrCompilationLogTest,NewLogVersionTriggersCompilation)368*795d594fSAndroid Build Coastguard Worker TEST_F(OdrCompilationLogTest, NewLogVersionTriggersCompilation) {
369*795d594fSAndroid Build Coastguard Worker   static const int64_t kApexVersion = 1066;
370*795d594fSAndroid Build Coastguard Worker   static const int64_t kLastUpdateMillis = 777;
371*795d594fSAndroid Build Coastguard Worker   time_t start_time;
372*795d594fSAndroid Build Coastguard Worker   time(&start_time);
373*795d594fSAndroid Build Coastguard Worker 
374*795d594fSAndroid Build Coastguard Worker   ScratchFile scratch_file;
375*795d594fSAndroid Build Coastguard Worker   scratch_file.Close();
376*795d594fSAndroid Build Coastguard Worker 
377*795d594fSAndroid Build Coastguard Worker   // Generate a compilation log.
378*795d594fSAndroid Build Coastguard Worker   {
379*795d594fSAndroid Build Coastguard Worker     OdrCompilationLog ocl(scratch_file.GetFilename().c_str());
380*795d594fSAndroid Build Coastguard Worker     for (size_t i = 0; i < OdrCompilationLog::kMaxLoggedEntries; ++i) {
381*795d594fSAndroid Build Coastguard Worker       ocl.Log(kApexVersion,
382*795d594fSAndroid Build Coastguard Worker               kLastUpdateMillis,
383*795d594fSAndroid Build Coastguard Worker               OdrMetrics::Trigger::kApexVersionMismatch,
384*795d594fSAndroid Build Coastguard Worker               start_time,
385*795d594fSAndroid Build Coastguard Worker               ExitCode::kCompilationFailed);
386*795d594fSAndroid Build Coastguard Worker       ASSERT_FALSE(ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time));
387*795d594fSAndroid Build Coastguard Worker     }
388*795d594fSAndroid Build Coastguard Worker   }
389*795d594fSAndroid Build Coastguard Worker 
390*795d594fSAndroid Build Coastguard Worker   // Replace version string in the compilation log.
391*795d594fSAndroid Build Coastguard Worker   std::string log_text;
392*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(android::base::ReadFileToString(scratch_file.GetFilename(), &log_text));
393*795d594fSAndroid Build Coastguard Worker   std::string new_log_version = std::string(OdrCompilationLog::kLogVersion) + "a";
394*795d594fSAndroid Build Coastguard Worker   log_text.replace(0, new_log_version.size() - 1, new_log_version);
395*795d594fSAndroid Build Coastguard Worker   ASSERT_TRUE(android::base::WriteStringToFile(log_text, scratch_file.GetFilename()));
396*795d594fSAndroid Build Coastguard Worker 
397*795d594fSAndroid Build Coastguard Worker   // Read log with updated version entry, check it is treated as out-of-date.
398*795d594fSAndroid Build Coastguard Worker   {
399*795d594fSAndroid Build Coastguard Worker     OdrCompilationLog ocl(scratch_file.GetFilename().c_str());
400*795d594fSAndroid Build Coastguard Worker     ASSERT_TRUE(ocl.ShouldAttemptCompile(OdrMetrics::Trigger::kUnknown, start_time));
401*795d594fSAndroid Build Coastguard Worker     ASSERT_EQ(0u, ocl.NumberOfEntries());
402*795d594fSAndroid Build Coastguard Worker   }
403*795d594fSAndroid Build Coastguard Worker }
404*795d594fSAndroid Build Coastguard Worker 
405*795d594fSAndroid Build Coastguard Worker }  // namespace odrefresh
406*795d594fSAndroid Build Coastguard Worker }  // namespace art
407