1 /*
2  * Copyright (C) 2020 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.server.people.data;
18 
19 import android.annotation.IntDef;
20 import android.annotation.NonNull;
21 import android.annotation.Nullable;
22 import android.text.format.DateFormat;
23 import android.util.Range;
24 import android.util.Slog;
25 import android.util.proto.ProtoInputStream;
26 import android.util.proto.ProtoOutputStream;
27 
28 import com.android.internal.annotations.VisibleForTesting;
29 import com.android.server.people.PeopleEventIndexProto;
30 
31 import java.io.IOException;
32 import java.lang.annotation.Retention;
33 import java.lang.annotation.RetentionPolicy;
34 import java.time.Instant;
35 import java.time.LocalDateTime;
36 import java.time.ZoneId;
37 import java.time.temporal.ChronoUnit;
38 import java.util.ArrayList;
39 import java.util.Arrays;
40 import java.util.Collections;
41 import java.util.List;
42 import java.util.Objects;
43 import java.util.TimeZone;
44 import java.util.function.Function;
45 
46 /**
47  * The index of {@link Event}s. It is used for quickly looking up the time distribution of
48  * {@link Event}s based on {@code Event#getTimestamp()}.
49  *
50  * <p>The 64-bits {code long} is used as the bitmap index. Each bit is to denote whether there are
51  * any events in a specified time slot. The least significant bit is for the most recent time slot.
52  * And the most significant bit is for the oldest time slot.
53  *
54  * <p>Multiple {code long}s are used to index the events in different time grains. For the recent
55  * events, the fine-grained bitmap index can provide the narrower time range. For the older events,
56  * the coarse-grained bitmap index can cover longer period but can only provide wider time range.
57  *
58  * <p>E.g. the below chart shows how the bitmap indexes index the events in the past 24 hours:
59  * <pre>
60  * 2020/1/3                                                             2020/1/4
61  *   0:00        4:00        8:00       12:00       16:00       20:00        0:00
62  *  --+-----------------------------------------------------------------------+-  1 day per bit
63  *  --+-----------+-----------+-----------+-----------+-----------+-----------+-  4 hours per bit
64  *  --+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+--+-  1 hour per bit
65  *                                                                     +++++++++  2 minutes per bit
66  *  </pre>
67  */
68 public class EventIndex {
69     private static final String TAG = EventIndex.class.getSimpleName();
70 
71     private static final int RETENTION_DAYS = 63;
72 
73     private static final int TIME_SLOT_ONE_DAY = 0;
74 
75     private static final int TIME_SLOT_FOUR_HOURS = 1;
76 
77     private static final int TIME_SLOT_ONE_HOUR = 2;
78 
79     private static final int TIME_SLOT_TWO_MINUTES = 3;
80 
81     @IntDef(prefix = {"TIME_SLOT_"}, value = {
82             TIME_SLOT_ONE_DAY,
83             TIME_SLOT_FOUR_HOURS,
84             TIME_SLOT_ONE_HOUR,
85             TIME_SLOT_TWO_MINUTES,
86     })
87     @Retention(RetentionPolicy.SOURCE)
88     private @interface TimeSlotType {
89     }
90 
91     private static final int TIME_SLOT_TYPES_COUNT = 4;
92 
93     static final EventIndex EMPTY = new EventIndex();
94 
95     private static final List<Function<Long, Range<Long>>> TIME_SLOT_FACTORIES =
96             Collections.unmodifiableList(
97                     Arrays.asList(
98                             EventIndex::createOneDayLongTimeSlot,
99                             EventIndex::createFourHoursLongTimeSlot,
100                             EventIndex::createOneHourLongTimeSlot,
101                             EventIndex::createTwoMinutesLongTimeSlot
102                     )
103             );
104 
105     /** Combines the two {@link EventIndex} objects and returns the combined result. */
combine(EventIndex lhs, EventIndex rhs)106     static EventIndex combine(EventIndex lhs, EventIndex rhs) {
107         EventIndex older = lhs.mLastUpdatedTime < rhs.mLastUpdatedTime ? lhs : rhs;
108         EventIndex younger = lhs.mLastUpdatedTime >= rhs.mLastUpdatedTime ? lhs : rhs;
109 
110         EventIndex combined = new EventIndex(older);
111         combined.updateEventBitmaps(younger.mLastUpdatedTime);
112 
113         for (int slotType = 0; slotType < TIME_SLOT_TYPES_COUNT; slotType++) {
114             combined.mEventBitmaps[slotType] |= younger.mEventBitmaps[slotType];
115         }
116         return combined;
117     }
118 
119     private final long[] mEventBitmaps;
120 
121     private long mLastUpdatedTime;
122 
123     private final Object mLock = new Object();
124 
125     private final Injector mInjector;
126 
EventIndex()127     EventIndex() {
128         this(new Injector());
129     }
130 
EventIndex(@onNull EventIndex from)131     EventIndex(@NonNull EventIndex from) {
132         this(from.mInjector, from.mEventBitmaps, from.mLastUpdatedTime);
133     }
134 
135     @VisibleForTesting
EventIndex(@onNull Injector injector)136     EventIndex(@NonNull Injector injector) {
137         this(injector, new long[]{0L, 0L, 0L, 0L}, injector.currentTimeMillis());
138     }
139 
EventIndex(@onNull Injector injector, long[] eventBitmaps, long lastUpdatedTime)140     private EventIndex(@NonNull Injector injector, long[] eventBitmaps, long lastUpdatedTime) {
141         mInjector = injector;
142         mEventBitmaps = Arrays.copyOf(eventBitmaps, TIME_SLOT_TYPES_COUNT);
143         mLastUpdatedTime = lastUpdatedTime;
144     }
145 
146     /**
147      * Gets the most recent active time slot. A time slot is active if there is at least one event
148      * occurred in that time slot.
149      */
150     @Nullable
getMostRecentActiveTimeSlot()151     public Range<Long> getMostRecentActiveTimeSlot() {
152         synchronized (mLock) {
153             for (int slotType = TIME_SLOT_TYPES_COUNT - 1; slotType >= 0; slotType--) {
154                 if (mEventBitmaps[slotType] == 0L) {
155                     continue;
156                 }
157                 Range<Long> lastTimeSlot =
158                         TIME_SLOT_FACTORIES.get(slotType).apply(mLastUpdatedTime);
159                 int numberOfTrailingZeros = Long.numberOfTrailingZeros(mEventBitmaps[slotType]);
160                 long offset = getDuration(lastTimeSlot) * numberOfTrailingZeros;
161                 return Range.create(lastTimeSlot.getLower() - offset,
162                         lastTimeSlot.getUpper() - offset);
163             }
164         }
165         return null;
166     }
167 
168     /**
169      * Gets the active time slots. A time slot is active if there is at least one event occurred
170      * in that time slot.
171      *
172      * @return active time slots in chronological order.
173      */
174     @NonNull
getActiveTimeSlots()175     public List<Range<Long>> getActiveTimeSlots() {
176         List<Range<Long>> activeTimeSlots = new ArrayList<>();
177         synchronized (mLock) {
178             for (int slotType = 0; slotType < TIME_SLOT_TYPES_COUNT; slotType++) {
179                 activeTimeSlots = combineTimeSlotLists(activeTimeSlots,
180                         getActiveTimeSlotsForType(slotType));
181             }
182         }
183         Collections.reverse(activeTimeSlots);
184         return activeTimeSlots;
185     }
186 
187     /** Returns whether this {@link EventIndex} instance is empty. */
isEmpty()188     public boolean isEmpty() {
189         synchronized (mLock) {
190             for (int slotType = 0; slotType < TIME_SLOT_TYPES_COUNT; slotType++) {
191                 if (mEventBitmaps[slotType] != 0L) {
192                     return false;
193                 }
194             }
195         }
196         return true;
197     }
198 
199     /**
200      * Adds an event to this index with the given event time. Before the new event is recorded, the
201      * index is updated first with the current timestamp.
202      */
addEvent(long eventTime)203     void addEvent(long eventTime) {
204         if (EMPTY == this) {
205             throw new IllegalStateException("EMPTY instance is immutable");
206         }
207         synchronized (mLock) {
208             long currentTime = mInjector.currentTimeMillis();
209             updateEventBitmaps(currentTime);
210             for (int slotType = 0; slotType < TIME_SLOT_TYPES_COUNT; slotType++) {
211                 int offset = diffTimeSlots(slotType, eventTime, currentTime);
212                 if (offset < Long.SIZE) {
213                     mEventBitmaps[slotType] |= (1L << offset);
214                 }
215             }
216         }
217     }
218 
219     /** Updates to make all bitmaps up to date. */
update()220     void update() {
221         updateEventBitmaps(mInjector.currentTimeMillis());
222     }
223 
224     @Override
toString()225     public String toString() {
226         StringBuilder sb = new StringBuilder();
227         sb.append("EventIndex {");
228         sb.append("perDayEventBitmap=0b");
229         sb.append(Long.toBinaryString(mEventBitmaps[TIME_SLOT_ONE_DAY]));
230         sb.append(", perFourHoursEventBitmap=0b");
231         sb.append(Long.toBinaryString(mEventBitmaps[TIME_SLOT_FOUR_HOURS]));
232         sb.append(", perHourEventBitmap=0b");
233         sb.append(Long.toBinaryString(mEventBitmaps[TIME_SLOT_ONE_HOUR]));
234         sb.append(", perTwoMinutesEventBitmap=0b");
235         sb.append(Long.toBinaryString(mEventBitmaps[TIME_SLOT_TWO_MINUTES]));
236         sb.append(", lastUpdatedTime=");
237         sb.append(DateFormat.format("yyyy-MM-dd HH:mm:ss", mLastUpdatedTime));
238         sb.append("}");
239         return sb.toString();
240     }
241 
242     @Override
equals(Object obj)243     public boolean equals(Object obj) {
244         if (this == obj) {
245             return true;
246         }
247         if (!(obj instanceof EventIndex)) {
248             return false;
249         }
250         EventIndex other = (EventIndex) obj;
251         return mLastUpdatedTime == other.mLastUpdatedTime
252                 && Arrays.equals(mEventBitmaps, other.mEventBitmaps);
253     }
254 
255     @Override
hashCode()256     public int hashCode() {
257         return Objects.hash(mLastUpdatedTime, Arrays.hashCode(mEventBitmaps));
258     }
259 
writeToProto(@onNull ProtoOutputStream protoOutputStream)260     synchronized void writeToProto(@NonNull ProtoOutputStream protoOutputStream) {
261         for (long bitmap : mEventBitmaps) {
262             protoOutputStream.write(PeopleEventIndexProto.EVENT_BITMAPS, bitmap);
263         }
264         protoOutputStream.write(PeopleEventIndexProto.LAST_UPDATED_TIME, mLastUpdatedTime);
265     }
266 
267     /** Shifts the event bitmaps to make them up-to-date. */
updateEventBitmaps(long currentTimeMillis)268     private void updateEventBitmaps(long currentTimeMillis) {
269         for (int slotType = 0; slotType < TIME_SLOT_TYPES_COUNT; slotType++) {
270             int offset = diffTimeSlots(slotType, mLastUpdatedTime, currentTimeMillis);
271             if (offset < Long.SIZE) {
272                 mEventBitmaps[slotType] <<= offset;
273             } else {
274                 mEventBitmaps[slotType] = 0L;
275             }
276         }
277 
278         int bitsToClear = Long.SIZE - RETENTION_DAYS;
279         mEventBitmaps[TIME_SLOT_ONE_DAY] <<= bitsToClear;
280         mEventBitmaps[TIME_SLOT_ONE_DAY] >>>= bitsToClear;
281         mLastUpdatedTime = currentTimeMillis;
282     }
283 
readFromProto(@onNull ProtoInputStream protoInputStream)284     static EventIndex readFromProto(@NonNull ProtoInputStream protoInputStream) throws IOException {
285         int bitmapIndex = 0;
286         long[] eventBitmaps = new long[TIME_SLOT_TYPES_COUNT];
287         long lastUpdated = 0L;
288         while (protoInputStream.nextField() != ProtoInputStream.NO_MORE_FIELDS) {
289             switch (protoInputStream.getFieldNumber()) {
290                 case (int) PeopleEventIndexProto.EVENT_BITMAPS:
291                     eventBitmaps[bitmapIndex++] = protoInputStream.readLong(
292                             PeopleEventIndexProto.EVENT_BITMAPS);
293                     break;
294                 case (int) PeopleEventIndexProto.LAST_UPDATED_TIME:
295                     lastUpdated = protoInputStream.readLong(
296                             PeopleEventIndexProto.LAST_UPDATED_TIME);
297                     break;
298                 default:
299                     Slog.e(TAG, "Could not read undefined field: "
300                             + protoInputStream.getFieldNumber());
301             }
302         }
303         return new EventIndex(new Injector(), eventBitmaps, lastUpdated);
304     }
305 
toLocalDateTime(long epochMilli)306     private static LocalDateTime toLocalDateTime(long epochMilli) {
307         return LocalDateTime.ofInstant(
308                 Instant.ofEpochMilli(epochMilli), TimeZone.getDefault().toZoneId());
309     }
310 
toEpochMilli(LocalDateTime localDateTime)311     private static long toEpochMilli(LocalDateTime localDateTime) {
312         return localDateTime.atZone(ZoneId.systemDefault()).toInstant().toEpochMilli();
313     }
314 
getDuration(Range<Long> timeSlot)315     private static long getDuration(Range<Long> timeSlot) {
316         return timeSlot.getUpper() - timeSlot.getLower();
317     }
318 
319     /**
320      * Finds the time slots for the given two timestamps and returns the distance (in the number
321      * of time slots) between these two time slots.
322      */
diffTimeSlots(@imeSlotType int timeSlotType, long fromTime, long toTime)323     private static int diffTimeSlots(@TimeSlotType int timeSlotType, long fromTime, long toTime) {
324         Function<Long, Range<Long>> timeSlotFactory = TIME_SLOT_FACTORIES.get(timeSlotType);
325         Range<Long> fromSlot = timeSlotFactory.apply(fromTime);
326         Range<Long> toSlot = timeSlotFactory.apply(toTime);
327         return (int) ((toSlot.getLower() - fromSlot.getLower()) / getDuration(fromSlot));
328     }
329 
330     /**
331      * Returns the active time slots for a specified type. The returned time slots are in
332      * reverse-chronological order.
333      */
getActiveTimeSlotsForType(@imeSlotType int timeSlotType)334     private List<Range<Long>> getActiveTimeSlotsForType(@TimeSlotType int timeSlotType) {
335         long eventBitmap = mEventBitmaps[timeSlotType];
336         Range<Long> latestTimeSlot = TIME_SLOT_FACTORIES.get(timeSlotType).apply(mLastUpdatedTime);
337         long startTime = latestTimeSlot.getLower();
338         final long duration = getDuration(latestTimeSlot);
339         List<Range<Long>> timeSlots = new ArrayList<>();
340         while (eventBitmap != 0) {
341             int trailingZeros = Long.numberOfTrailingZeros(eventBitmap);
342             if (trailingZeros > 0) {
343                 startTime -= duration * trailingZeros;
344                 eventBitmap >>>= trailingZeros;
345             }
346             if (eventBitmap != 0) {
347                 timeSlots.add(Range.create(startTime, startTime + duration));
348                 startTime -= duration;
349                 eventBitmap >>>= 1;
350             }
351         }
352         return timeSlots;
353     }
354 
355     /**
356      * Combines two lists of time slots into one. If one longer time slot covers one or multiple
357      * shorter time slots, the smaller time slot(s) will be added to the result and the longer one
358      * will be dropped. This ensures the returned list does not contain any overlapping time slots.
359      */
combineTimeSlotLists(List<Range<Long>> longerSlots, List<Range<Long>> shorterSlots)360     private static List<Range<Long>> combineTimeSlotLists(List<Range<Long>> longerSlots,
361             List<Range<Long>> shorterSlots) {
362         List<Range<Long>> result = new ArrayList<>();
363         int i = 0;
364         int j = 0;
365         while (i < longerSlots.size() && j < shorterSlots.size()) {
366             Range<Long> longerSlot = longerSlots.get(i);
367             Range<Long> shorterSlot = shorterSlots.get(j);
368             if (longerSlot.contains(shorterSlot)) {
369                 result.add(shorterSlot);
370                 i++;
371                 j++;
372             } else if (longerSlot.getLower() < shorterSlot.getLower()) {
373                 result.add(shorterSlot);
374                 j++;
375             } else {
376                 result.add(longerSlot);
377                 i++;
378             }
379         }
380         if (i < longerSlots.size()) {
381             result.addAll(longerSlots.subList(i, longerSlots.size()));
382         } else if (j < shorterSlots.size()) {
383             result.addAll(shorterSlots.subList(j, shorterSlots.size()));
384         }
385         return result;
386     }
387 
388     /**
389      * Finds and creates the time slot (duration = 1 day) that the given time falls into.
390      */
391     @NonNull
createOneDayLongTimeSlot(long time)392     private static Range<Long> createOneDayLongTimeSlot(long time) {
393         LocalDateTime beginTime = toLocalDateTime(time).truncatedTo(ChronoUnit.DAYS);
394         return Range.create(toEpochMilli(beginTime), toEpochMilli(beginTime.plusDays(1)));
395     }
396 
397     /**
398      * Finds and creates the time slot (duration = 4 hours) that the given time falls into.
399      */
400     @NonNull
createFourHoursLongTimeSlot(long time)401     private static Range<Long> createFourHoursLongTimeSlot(long time) {
402         int hourOfDay = toLocalDateTime(time).getHour();
403         LocalDateTime beginTime =
404                 toLocalDateTime(time).truncatedTo(ChronoUnit.HOURS).minusHours(hourOfDay % 4);
405         return Range.create(toEpochMilli(beginTime), toEpochMilli(beginTime.plusHours(4)));
406     }
407 
408     /**
409      * Finds and creates the time slot (duration = 1 hour) that the given time falls into.
410      */
411     @NonNull
createOneHourLongTimeSlot(long time)412     private static Range<Long> createOneHourLongTimeSlot(long time) {
413         LocalDateTime beginTime = toLocalDateTime(time).truncatedTo(ChronoUnit.HOURS);
414         return Range.create(toEpochMilli(beginTime), toEpochMilli(beginTime.plusHours(1)));
415     }
416 
417     /**
418      * Finds and creates the time slot (duration = 2 minutes) that the given time falls into.
419      */
420     @NonNull
createTwoMinutesLongTimeSlot(long time)421     private static Range<Long> createTwoMinutesLongTimeSlot(long time) {
422         int minuteOfHour = toLocalDateTime(time).getMinute();
423         LocalDateTime beginTime = toLocalDateTime(time).truncatedTo(
424                 ChronoUnit.MINUTES).minusMinutes(minuteOfHour % 2);
425         return Range.create(toEpochMilli(beginTime), toEpochMilli(beginTime.plusMinutes(2)));
426     }
427 
428     @VisibleForTesting
429     static class Injector {
430         /** This should be the only way to get the current timestamp in {@code EventIndex}. */
currentTimeMillis()431         long currentTimeMillis() {
432             return System.currentTimeMillis();
433         }
434     }
435 }
436