| /* |
| * Copyright (C) 2019 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 "src/trace_processor/clock_tracker.h" |
| |
| #include "perfetto/ext/base/optional.h" |
| #include "src/trace_processor/trace_processor_context.h" |
| #include "src/trace_processor/trace_storage.h" |
| #include "test/gtest_and_gmock.h" |
| |
| #include "protos/perfetto/trace/clock_snapshot.pbzero.h" |
| |
| namespace perfetto { |
| namespace trace_processor { |
| namespace { |
| |
| using ::testing::NiceMock; |
| using Clock = protos::pbzero::ClockSnapshot::Clock; |
| |
| constexpr auto REALTIME = Clock::REALTIME; |
| constexpr auto BOOTTIME = Clock::BOOTTIME; |
| constexpr auto MONOTONIC = Clock::MONOTONIC; |
| constexpr auto MONOTONIC_COARSE = Clock::MONOTONIC_COARSE; |
| constexpr auto MONOTONIC_RAW = Clock::MONOTONIC_RAW; |
| |
| class ClockTrackerTest : public ::testing::Test { |
| public: |
| ClockTrackerTest() { context_.storage.reset(new TraceStorage()); } |
| |
| TraceProcessorContext context_; |
| ClockTracker ct_{&context_}; |
| }; |
| |
| TEST_F(ClockTrackerTest, ClockDomainConversions) { |
| EXPECT_EQ(ct_.ToTraceTime(Clock::REALTIME, 0), base::nullopt); |
| |
| ct_.AddSnapshot({{REALTIME, 10}, {BOOTTIME, 10010}}); |
| ct_.AddSnapshot({{REALTIME, 20}, {BOOTTIME, 20220}}); |
| ct_.AddSnapshot({{REALTIME, 30}, {BOOTTIME, 30030}}); |
| ct_.AddSnapshot({{MONOTONIC, 1000}, {BOOTTIME, 100000}}); |
| |
| EXPECT_EQ(ct_.ToTraceTime(REALTIME, 0), 10000); |
| EXPECT_EQ(ct_.ToTraceTime(REALTIME, 1), 10001); |
| EXPECT_EQ(ct_.ToTraceTime(REALTIME, 9), 10009); |
| EXPECT_EQ(ct_.ToTraceTime(REALTIME, 10), 10010); |
| EXPECT_EQ(ct_.ToTraceTime(REALTIME, 11), 10011); |
| EXPECT_EQ(ct_.ToTraceTime(REALTIME, 19), 10019); |
| EXPECT_EQ(ct_.ToTraceTime(REALTIME, 20), 20220); |
| EXPECT_EQ(ct_.ToTraceTime(REALTIME, 21), 20221); |
| EXPECT_EQ(ct_.ToTraceTime(REALTIME, 29), 20229); |
| EXPECT_EQ(ct_.ToTraceTime(REALTIME, 30), 30030); |
| EXPECT_EQ(ct_.ToTraceTime(REALTIME, 40), 30040); |
| |
| EXPECT_EQ(ct_.ToTraceTime(MONOTONIC, 0), 100000 - 1000); |
| EXPECT_EQ(ct_.ToTraceTime(MONOTONIC, 999), 100000 - 1); |
| EXPECT_EQ(ct_.ToTraceTime(MONOTONIC, 1000), 100000); |
| EXPECT_EQ(ct_.ToTraceTime(MONOTONIC, 1e6), |
| static_cast<int64_t>(100000 - 1000 + 1e6)); |
| } |
| |
| // When a clock moves backwards conversions *from* that clock are forbidden |
| // but conversions *to* that clock should still work. |
| // Think to the case of REALTIME going backwards from 3AM to 2AM during DST day. |
| // You can't convert 2.10AM REALTIME to BOOTTIME because there are two possible |
| // answers, but you can still unambiguosly convert BOOTTIME into REALTIME. |
| TEST_F(ClockTrackerTest, RealTimeClockMovingBackwards) { |
| ct_.AddSnapshot({{BOOTTIME, 10010}, {REALTIME, 10}}); |
| |
| // At this point conversions are still possible in both ways because we |
| // haven't broken monotonicity yet. |
| EXPECT_EQ(ct_.Convert(REALTIME, 11, BOOTTIME), 10011); |
| |
| ct_.AddSnapshot({{BOOTTIME, 10020}, {REALTIME, 20}}); |
| ct_.AddSnapshot({{BOOTTIME, 30040}, {REALTIME, 40}}); |
| ct_.AddSnapshot({{BOOTTIME, 40030}, {REALTIME, 30}}); |
| |
| // Now only BOOTIME -> REALTIME conversion should be possible. |
| EXPECT_FALSE(ct_.Convert(REALTIME, 11, BOOTTIME)); |
| EXPECT_EQ(ct_.Convert(BOOTTIME, 10011, REALTIME), 11); |
| EXPECT_EQ(ct_.Convert(BOOTTIME, 10029, REALTIME), 29); |
| EXPECT_EQ(ct_.Convert(BOOTTIME, 40030, REALTIME), 30); |
| EXPECT_EQ(ct_.Convert(BOOTTIME, 40040, REALTIME), 40); |
| |
| ct_.AddSnapshot({{BOOTTIME, 50000}, {REALTIME, 50}}); |
| EXPECT_EQ(ct_.Convert(BOOTTIME, 50005, REALTIME), 55); |
| |
| ct_.AddSnapshot({{BOOTTIME, 60020}, {REALTIME, 20}}); |
| EXPECT_EQ(ct_.Convert(BOOTTIME, 60020, REALTIME), 20); |
| } |
| |
| // Simulate the following scenario: |
| // MONOTONIC = MONOTONIC_COARSE + 10 |
| // BOOTTIME = MONOTONIC + 1000 (until T=200) |
| // BOOTTIME = MONOTONIC + 2000 (from T=200) |
| // Then resolve MONOTONIC_COARSE. This requires a two-level resolution: |
| // MONOTONIC_COARSE -> MONOTONIC -> BOOTTIME. |
| TEST_F(ClockTrackerTest, ChainedResolutionSimple) { |
| ct_.AddSnapshot({{MONOTONIC_COARSE, 1}, {MONOTONIC, 11}}); |
| ct_.AddSnapshot({{MONOTONIC, 100}, {BOOTTIME, 1100}}); |
| ct_.AddSnapshot({{MONOTONIC, 200}, {BOOTTIME, 2200}}); |
| |
| EXPECT_EQ(ct_.Convert(MONOTONIC, 100, MONOTONIC_COARSE), 90); |
| EXPECT_EQ(ct_.Convert(MONOTONIC_COARSE, 20, MONOTONIC), 30); |
| |
| // MONOTONIC_COARSE@100 == MONOTONIC@110 == BOOTTIME@1100. |
| EXPECT_EQ(ct_.ToTraceTime(MONOTONIC, 110), 1110); |
| EXPECT_EQ(*ct_.ToTraceTime(MONOTONIC_COARSE, 100), 100 + 10 + 1000); |
| EXPECT_EQ(*ct_.ToTraceTime(MONOTONIC_COARSE, 202), 202 + 10 + 2000); |
| } |
| |
| TEST_F(ClockTrackerTest, ChainedResolutionHard) { |
| // MONOTONIC_COARSE = MONOTONIC_RAW - 1. |
| ct_.AddSnapshot({{MONOTONIC_RAW, 10}, {MONOTONIC_COARSE, 9}}); |
| |
| // MONOTONIC = MONOTONIC_COARSE - 50. |
| ct_.AddSnapshot({{MONOTONIC_COARSE, 100}, {MONOTONIC, 50}}); |
| |
| // BOOTTIME = MONOTONIC + 1000 until T=100 (see below). |
| ct_.AddSnapshot({{MONOTONIC, 1}, {BOOTTIME, 1001}, {REALTIME, 10001}}); |
| |
| // BOOTTIME = MONOTONIC + 2000 from T=100. |
| // At the same time, REALTIME goes backwards. |
| ct_.AddSnapshot({{MONOTONIC, 101}, {BOOTTIME, 2101}, {REALTIME, 9101}}); |
| |
| // 1-hop conversions. |
| EXPECT_EQ(ct_.Convert(MONOTONIC_RAW, 2, MONOTONIC_COARSE), 1); |
| EXPECT_EQ(ct_.Convert(MONOTONIC_COARSE, 1, MONOTONIC_RAW), 2); |
| EXPECT_EQ(ct_.Convert(MONOTONIC_RAW, 100001, MONOTONIC_COARSE), 100000); |
| EXPECT_EQ(ct_.Convert(MONOTONIC_COARSE, 100000, MONOTONIC_RAW), 100001); |
| |
| // 2-hop conversions (MONOTONIC_RAW <-> MONOTONIC_COARSE <-> MONOTONIC). |
| // From above, MONOTONIC = (MONOTONIC_RAW - 1) - 50. |
| EXPECT_EQ(ct_.Convert(MONOTONIC_RAW, 53, MONOTONIC), 53 - 1 - 50); |
| EXPECT_EQ(ct_.Convert(MONOTONIC, 2, MONOTONIC_RAW), 2 + 1 + 50); |
| |
| // 3-hop conversions (as above + BOOTTIME) |
| EXPECT_EQ(ct_.Convert(MONOTONIC_RAW, 53, BOOTTIME), 53 - 1 - 50 + 1000); |
| EXPECT_EQ(ct_.Convert(BOOTTIME, 1002, MONOTONIC_RAW), 1002 - 1000 + 1 + 50); |
| |
| EXPECT_EQ(*ct_.Convert(MONOTONIC_RAW, 753, BOOTTIME), 753 - 1 - 50 + 2000); |
| EXPECT_EQ(ct_.Convert(BOOTTIME, 2702, MONOTONIC_RAW), 2702 - 2000 + 1 + 50); |
| |
| // 3-hop conversion to REALTIME, one way only (REALTIME goes backwards). |
| EXPECT_EQ(*ct_.Convert(MONOTONIC_RAW, 53, REALTIME), 53 - 1 - 50 + 10000); |
| EXPECT_EQ(*ct_.Convert(MONOTONIC_RAW, 753, REALTIME), 753 - 1 - 50 + 9000); |
| } |
| |
| TEST_F(ClockTrackerTest, SequenceScopedClocks) { |
| ct_.AddSnapshot({{MONOTONIC, 1000}, {BOOTTIME, 100000}}); |
| |
| ClockTracker::ClockId c64_1 = ct_.SeqScopedClockIdToGlobal(1, 64); |
| ClockTracker::ClockId c65_1 = ct_.SeqScopedClockIdToGlobal(1, 65); |
| ClockTracker::ClockId c66_1 = ct_.SeqScopedClockIdToGlobal(1, 66); |
| ClockTracker::ClockId c66_2 = ct_.SeqScopedClockIdToGlobal(2, 64); |
| |
| ct_.AddSnapshot( |
| {{MONOTONIC, 10000}, |
| {c64_1, 100000}, |
| {c65_1, 100, /*unit_multiplier_ns=*/1000, /*is_incremental=*/false}, |
| {c66_1, 10, /*unit_multiplier_ns=*/1000, /*is_incremental=*/true}}); |
| |
| // c64_1 is non-incremental and in nanos. |
| EXPECT_EQ(*ct_.Convert(c64_1, 150000, MONOTONIC), 60000); |
| EXPECT_EQ(*ct_.Convert(c64_1, 150000, BOOTTIME), 159000); |
| EXPECT_EQ(*ct_.ToTraceTime(c64_1, 150000), 159000); |
| |
| // c65_1 is non-incremental and in micros. |
| EXPECT_EQ(*ct_.Convert(c65_1, 150, MONOTONIC), 60000); |
| EXPECT_EQ(*ct_.Convert(c65_1, 150, BOOTTIME), 159000); |
| EXPECT_EQ(*ct_.ToTraceTime(c65_1, 150), 159000); |
| |
| // c66_1 is incremental and in micros. |
| EXPECT_EQ(*ct_.Convert(c66_1, 1 /* abs 11 */, MONOTONIC), 11000); |
| EXPECT_EQ(*ct_.Convert(c66_1, 1 /* abs 12 */, MONOTONIC), 12000); |
| EXPECT_EQ(*ct_.Convert(c66_1, 1 /* abs 13 */, BOOTTIME), 112000); |
| EXPECT_EQ(*ct_.ToTraceTime(c66_1, 2 /* abs 15 */), 114000); |
| |
| ct_.AddSnapshot( |
| {{MONOTONIC, 20000}, |
| {c66_1, 20, /*unit_multiplier_ns=*/1000, /*is_incremental=*/true}}); |
| ct_.AddSnapshot( |
| {{MONOTONIC, 20000}, |
| {c66_2, 20, /*unit_multiplier_ns=*/1000, /*is_incremental=*/true}}); |
| |
| // c66_1 and c66_2 are both incremental and in micros, but shouldn't affect |
| // each other. |
| EXPECT_EQ(*ct_.Convert(c66_1, 1 /* abs 21 */, MONOTONIC), 21000); |
| EXPECT_EQ(*ct_.Convert(c66_2, 2 /* abs 22 */, MONOTONIC), 22000); |
| EXPECT_EQ(*ct_.Convert(c66_1, 1 /* abs 22 */, MONOTONIC), 22000); |
| EXPECT_EQ(*ct_.Convert(c66_2, 2 /* abs 24 */, MONOTONIC), 24000); |
| EXPECT_EQ(*ct_.Convert(c66_1, 1 /* abs 23 */, BOOTTIME), 122000); |
| EXPECT_EQ(*ct_.Convert(c66_2, 2 /* abs 26 */, BOOTTIME), 125000); |
| EXPECT_EQ(*ct_.ToTraceTime(c66_1, 2 /* abs 25 */), 124000); |
| EXPECT_EQ(*ct_.ToTraceTime(c66_2, 4 /* abs 30 */), 129000); |
| } |
| |
| } // namespace |
| } // namespace trace_processor |
| } // namespace perfetto |