1 /*
2  * Copyright (C) 2015 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.am;
18 
19 import android.app.ActivityOptions;
20 import android.app.PendingIntent;
21 import android.content.Context;
22 import android.content.Intent;
23 import android.os.Bundle;
24 import android.os.SystemClock;
25 import android.util.ArrayMap;
26 import android.util.MutableLong;
27 import android.util.TimeUtils;
28 import android.util.proto.ProtoOutputStream;
29 import android.util.proto.ProtoUtils;
30 
31 import java.io.PrintWriter;
32 
33 /**
34  * Tracks the time a user spent in an app.
35  */
36 public class AppTimeTracker {
37     private final PendingIntent mReceiver;
38 
39     private long mTotalTime;
40     private final ArrayMap<String, MutableLong> mPackageTimes = new ArrayMap<>();
41 
42     private long mStartedTime;
43     private String mStartedPackage;
44     private MutableLong mStartedPackageTime;
45 
AppTimeTracker(PendingIntent receiver)46     public AppTimeTracker(PendingIntent receiver) {
47         mReceiver = receiver;
48     }
49 
start(String packageName)50     public void start(String packageName) {
51         long now = SystemClock.elapsedRealtime();
52         if (mStartedTime == 0) {
53             mStartedTime = now;
54         }
55         if (!packageName.equals(mStartedPackage)) {
56             if (mStartedPackageTime != null) {
57                 long elapsedTime = now - mStartedTime;
58                 mStartedPackageTime.value += elapsedTime;
59                 mTotalTime += elapsedTime;
60             }
61             mStartedPackage = packageName;
62             mStartedPackageTime = mPackageTimes.get(packageName);
63             if (mStartedPackageTime == null) {
64                 mStartedPackageTime = new MutableLong(0);
65                 mPackageTimes.put(packageName, mStartedPackageTime);
66             }
67         }
68     }
69 
stop()70     public void stop() {
71         if (mStartedTime != 0) {
72             long elapsedTime = SystemClock.elapsedRealtime() - mStartedTime;
73             mTotalTime += elapsedTime;
74             if (mStartedPackageTime != null) {
75                 mStartedPackageTime.value += elapsedTime;
76             }
77             mStartedPackage = null;
78             mStartedPackageTime = null;
79         }
80     }
81 
deliverResult(Context context)82     public void deliverResult(Context context) {
83         stop();
84         Bundle extras = new Bundle();
85         extras.putLong(ActivityOptions.EXTRA_USAGE_TIME_REPORT, mTotalTime);
86         Bundle pkgs = new Bundle();
87         for (int i=mPackageTimes.size()-1; i>=0; i--) {
88             pkgs.putLong(mPackageTimes.keyAt(i), mPackageTimes.valueAt(i).value);
89         }
90         extras.putBundle(ActivityOptions.EXTRA_USAGE_TIME_REPORT_PACKAGES, pkgs);
91         Intent fillinIntent = new Intent();
92         fillinIntent.putExtras(extras);
93         try {
94             mReceiver.send(context, 0, fillinIntent);
95         } catch (PendingIntent.CanceledException e) {
96         }
97     }
98 
dumpWithHeader(PrintWriter pw, String prefix, boolean details)99     public void dumpWithHeader(PrintWriter pw, String prefix, boolean details) {
100         pw.print(prefix); pw.print("AppTimeTracker #");
101         pw.print(Integer.toHexString(System.identityHashCode(this)));
102         pw.println(":");
103         dump(pw, prefix + "  ", details);
104     }
105 
dump(PrintWriter pw, String prefix, boolean details)106     public void dump(PrintWriter pw, String prefix, boolean details) {
107         pw.print(prefix); pw.print("mReceiver="); pw.println(mReceiver);
108         pw.print(prefix); pw.print("mTotalTime=");
109         TimeUtils.formatDuration(mTotalTime, pw);
110         pw.println();
111         for (int i = 0; i < mPackageTimes.size(); i++) {
112             pw.print(prefix); pw.print("mPackageTime:"); pw.print(mPackageTimes.keyAt(i));
113             pw.print("=");
114             TimeUtils.formatDuration(mPackageTimes.valueAt(i).value, pw);
115             pw.println();
116         }
117         if (details && mStartedTime != 0) {
118             pw.print(prefix); pw.print("mStartedTime=");
119             TimeUtils.formatDuration(SystemClock.elapsedRealtime(), mStartedTime, pw);
120             pw.println();
121             pw.print(prefix); pw.print("mStartedPackage="); pw.println(mStartedPackage);
122         }
123     }
124 
writeToProto(ProtoOutputStream proto, long fieldId, boolean details)125     void writeToProto(ProtoOutputStream proto, long fieldId, boolean details) {
126         final long token = proto.start(fieldId);
127         proto.write(AppTimeTrackerProto.RECEIVER, mReceiver.toString());
128         proto.write(AppTimeTrackerProto.TOTAL_DURATION_MS, mTotalTime);
129         for (int i=0; i<mPackageTimes.size(); i++) {
130             final long ptoken = proto.start(AppTimeTrackerProto.PACKAGE_TIMES);
131             proto.write(AppTimeTrackerProto.PackageTime.PACKAGE, mPackageTimes.keyAt(i));
132             proto.write(AppTimeTrackerProto.PackageTime.DURATION_MS, mPackageTimes.valueAt(i).value);
133             proto.end(ptoken);
134         }
135         if (details && mStartedTime != 0) {
136             ProtoUtils.toDuration(proto, AppTimeTrackerProto.STARTED_TIME,
137                     mStartedTime, SystemClock.elapsedRealtime());
138             proto.write(AppTimeTrackerProto.STARTED_PACKAGE, mStartedPackage);
139         }
140         proto.end(token);
141     }
142 }
143