1006 lines
43 KiB
Java
1006 lines
43 KiB
Java
![]() |
/*
|
||
|
* Copyright 2017 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.
|
||
|
*/
|
||
|
package android.app;
|
||
|
|
||
|
import static android.Manifest.permission.DUMP;
|
||
|
import static android.Manifest.permission.PACKAGE_USAGE_STATS;
|
||
|
import static android.Manifest.permission.READ_RESTRICTED_STATS;
|
||
|
import static android.provider.DeviceConfig.NAMESPACE_STATSD_JAVA;
|
||
|
|
||
|
import android.annotation.CallbackExecutor;
|
||
|
import android.annotation.NonNull;
|
||
|
import android.annotation.Nullable;
|
||
|
import android.annotation.RequiresPermission;
|
||
|
import android.annotation.SystemApi;
|
||
|
import android.content.Context;
|
||
|
import android.os.Binder;
|
||
|
import android.os.Build;
|
||
|
import android.os.IPullAtomCallback;
|
||
|
import android.os.IPullAtomResultReceiver;
|
||
|
import android.os.IStatsManagerService;
|
||
|
import android.os.IStatsQueryCallback;
|
||
|
import android.os.OutcomeReceiver;
|
||
|
import android.os.ParcelFileDescriptor;
|
||
|
import android.os.RemoteException;
|
||
|
import android.os.StatsFrameworkInitializer;
|
||
|
import android.provider.DeviceConfig;
|
||
|
import android.util.AndroidException;
|
||
|
import android.util.Log;
|
||
|
import android.util.StatsEvent;
|
||
|
import android.util.StatsEventParcel;
|
||
|
|
||
|
import androidx.annotation.RequiresApi;
|
||
|
|
||
|
import com.android.internal.annotations.GuardedBy;
|
||
|
import com.android.internal.annotations.VisibleForTesting;
|
||
|
import com.android.modules.utils.build.SdkLevel;
|
||
|
|
||
|
import java.io.DataInputStream;
|
||
|
import java.io.FileInputStream;
|
||
|
import java.io.IOException;
|
||
|
import java.nio.BufferOverflowException;
|
||
|
import java.nio.ByteBuffer;
|
||
|
import java.util.ArrayList;
|
||
|
import java.util.List;
|
||
|
import java.util.concurrent.Executor;
|
||
|
|
||
|
/**
|
||
|
* API for statsd clients to send configurations and retrieve data.
|
||
|
*
|
||
|
* @hide
|
||
|
*/
|
||
|
@SystemApi
|
||
|
public final class StatsManager {
|
||
|
private static final String TAG = "StatsManager";
|
||
|
private static final boolean DEBUG = false;
|
||
|
|
||
|
private static final Object sLock = new Object();
|
||
|
private final Context mContext;
|
||
|
|
||
|
@GuardedBy("sLock")
|
||
|
private IStatsManagerService mStatsManagerService;
|
||
|
|
||
|
/**
|
||
|
* Long extra of uid that added the relevant stats config.
|
||
|
*/
|
||
|
public static final String EXTRA_STATS_CONFIG_UID = "android.app.extra.STATS_CONFIG_UID";
|
||
|
/**
|
||
|
* Long extra of the relevant stats config's configKey.
|
||
|
*/
|
||
|
public static final String EXTRA_STATS_CONFIG_KEY = "android.app.extra.STATS_CONFIG_KEY";
|
||
|
/**
|
||
|
* Long extra of the relevant statsd_config.proto's Subscription.id.
|
||
|
*/
|
||
|
public static final String EXTRA_STATS_SUBSCRIPTION_ID =
|
||
|
"android.app.extra.STATS_SUBSCRIPTION_ID";
|
||
|
/**
|
||
|
* Long extra of the relevant statsd_config.proto's Subscription.rule_id.
|
||
|
*/
|
||
|
public static final String EXTRA_STATS_SUBSCRIPTION_RULE_ID =
|
||
|
"android.app.extra.STATS_SUBSCRIPTION_RULE_ID";
|
||
|
/**
|
||
|
* List<String> of the relevant statsd_config.proto's BroadcastSubscriberDetails.cookie.
|
||
|
* Obtain using {@link android.content.Intent#getStringArrayListExtra(String)}.
|
||
|
*/
|
||
|
public static final String EXTRA_STATS_BROADCAST_SUBSCRIBER_COOKIES =
|
||
|
"android.app.extra.STATS_BROADCAST_SUBSCRIBER_COOKIES";
|
||
|
/**
|
||
|
* Extra of a {@link android.os.StatsDimensionsValue} representing sliced dimension value
|
||
|
* information.
|
||
|
*/
|
||
|
public static final String EXTRA_STATS_DIMENSIONS_VALUE =
|
||
|
"android.app.extra.STATS_DIMENSIONS_VALUE";
|
||
|
/**
|
||
|
* Long array extra of the active configs for the uid that added those configs.
|
||
|
*/
|
||
|
public static final String EXTRA_STATS_ACTIVE_CONFIG_KEYS =
|
||
|
"android.app.extra.STATS_ACTIVE_CONFIG_KEYS";
|
||
|
|
||
|
/**
|
||
|
* Long array extra of the restricted metric ids present for the client.
|
||
|
*/
|
||
|
public static final String EXTRA_STATS_RESTRICTED_METRIC_IDS =
|
||
|
"android.app.extra.STATS_RESTRICTED_METRIC_IDS";
|
||
|
|
||
|
/**
|
||
|
* Broadcast Action: Statsd has started.
|
||
|
* Configurations and PendingIntents can now be sent to it.
|
||
|
*/
|
||
|
public static final String ACTION_STATSD_STARTED = "android.app.action.STATSD_STARTED";
|
||
|
|
||
|
// Pull atom callback return codes.
|
||
|
/**
|
||
|
* Value indicating that this pull was successful and that the result should be used.
|
||
|
*
|
||
|
**/
|
||
|
public static final int PULL_SUCCESS = 0;
|
||
|
|
||
|
/**
|
||
|
* Value indicating that this pull was unsuccessful and that the result should not be used.
|
||
|
**/
|
||
|
public static final int PULL_SKIP = 1;
|
||
|
|
||
|
/**
|
||
|
* @hide
|
||
|
**/
|
||
|
@VisibleForTesting public static final long DEFAULT_COOL_DOWN_MILLIS = 1_000L; // 1 second.
|
||
|
|
||
|
/**
|
||
|
* @hide
|
||
|
**/
|
||
|
@VisibleForTesting public static final long DEFAULT_TIMEOUT_MILLIS = 1_500L; // 1.5 seconds.
|
||
|
|
||
|
/**
|
||
|
* Constructor for StatsManagerClient.
|
||
|
*
|
||
|
* @hide
|
||
|
*/
|
||
|
public StatsManager(Context context) {
|
||
|
mContext = context;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Adds the given configuration and associates it with the given configKey. If a config with the
|
||
|
* given configKey already exists for the caller's uid, it is replaced with the new one.
|
||
|
* This call can block on statsd.
|
||
|
*
|
||
|
* @param configKey An arbitrary integer that allows clients to track the configuration.
|
||
|
* @param config Wire-encoded StatsdConfig proto that specifies metrics (and all
|
||
|
* dependencies eg, conditions and matchers).
|
||
|
* @throws StatsUnavailableException if unsuccessful due to failing to connect to stats service
|
||
|
* @throws IllegalArgumentException if config is not a wire-encoded StatsdConfig proto
|
||
|
*/
|
||
|
@RequiresPermission(allOf = { DUMP, PACKAGE_USAGE_STATS })
|
||
|
public void addConfig(long configKey, byte[] config) throws StatsUnavailableException {
|
||
|
synchronized (sLock) {
|
||
|
try {
|
||
|
IStatsManagerService service = getIStatsManagerServiceLocked();
|
||
|
// can throw IllegalArgumentException
|
||
|
service.addConfiguration(configKey, config, mContext.getOpPackageName());
|
||
|
} catch (RemoteException e) {
|
||
|
Log.e(TAG, "Failed to connect to statsmanager when adding configuration");
|
||
|
throw new StatsUnavailableException("could not connect", e);
|
||
|
} catch (SecurityException e) {
|
||
|
throw new StatsUnavailableException(e.getMessage(), e);
|
||
|
} catch (IllegalStateException e) {
|
||
|
Log.e(TAG, "Failed to addConfig in statsmanager");
|
||
|
throw new StatsUnavailableException(e.getMessage(), e);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// TODO: Temporary for backwards compatibility. Remove.
|
||
|
/**
|
||
|
* @deprecated Use {@link #addConfig(long, byte[])}
|
||
|
*/
|
||
|
@Deprecated
|
||
|
@RequiresPermission(allOf = { DUMP, PACKAGE_USAGE_STATS })
|
||
|
public boolean addConfiguration(long configKey, byte[] config) {
|
||
|
try {
|
||
|
addConfig(configKey, config);
|
||
|
return true;
|
||
|
} catch (StatsUnavailableException | IllegalArgumentException e) {
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Remove a configuration from logging.
|
||
|
*
|
||
|
* This call can block on statsd.
|
||
|
*
|
||
|
* @param configKey Configuration key to remove.
|
||
|
* @throws StatsUnavailableException if unsuccessful due to failing to connect to stats service
|
||
|
*/
|
||
|
@RequiresPermission(allOf = { DUMP, PACKAGE_USAGE_STATS })
|
||
|
public void removeConfig(long configKey) throws StatsUnavailableException {
|
||
|
synchronized (sLock) {
|
||
|
try {
|
||
|
IStatsManagerService service = getIStatsManagerServiceLocked();
|
||
|
service.removeConfiguration(configKey, mContext.getOpPackageName());
|
||
|
} catch (RemoteException e) {
|
||
|
Log.e(TAG, "Failed to connect to statsmanager when removing configuration");
|
||
|
throw new StatsUnavailableException("could not connect", e);
|
||
|
} catch (SecurityException e) {
|
||
|
throw new StatsUnavailableException(e.getMessage(), e);
|
||
|
} catch (IllegalStateException e) {
|
||
|
Log.e(TAG, "Failed to removeConfig in statsmanager");
|
||
|
throw new StatsUnavailableException(e.getMessage(), e);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// TODO: Temporary for backwards compatibility. Remove.
|
||
|
/**
|
||
|
* @deprecated Use {@link #removeConfig(long)}
|
||
|
*/
|
||
|
@Deprecated
|
||
|
@RequiresPermission(allOf = { DUMP, PACKAGE_USAGE_STATS })
|
||
|
public boolean removeConfiguration(long configKey) {
|
||
|
try {
|
||
|
removeConfig(configKey);
|
||
|
return true;
|
||
|
} catch (StatsUnavailableException e) {
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Set the PendingIntent to be used when broadcasting subscriber information to the given
|
||
|
* subscriberId within the given config.
|
||
|
* <p>
|
||
|
* Suppose that the calling uid has added a config with key configKey, and that in this config
|
||
|
* it is specified that when a particular anomaly is detected, a broadcast should be sent to
|
||
|
* a BroadcastSubscriber with id subscriberId. This function links the given pendingIntent with
|
||
|
* that subscriberId (for that config), so that this pendingIntent is used to send the broadcast
|
||
|
* when the anomaly is detected.
|
||
|
* <p>
|
||
|
* When statsd sends the broadcast, the PendingIntent will used to send an intent with
|
||
|
* information of
|
||
|
* {@link #EXTRA_STATS_CONFIG_UID},
|
||
|
* {@link #EXTRA_STATS_CONFIG_KEY},
|
||
|
* {@link #EXTRA_STATS_SUBSCRIPTION_ID},
|
||
|
* {@link #EXTRA_STATS_SUBSCRIPTION_RULE_ID},
|
||
|
* {@link #EXTRA_STATS_BROADCAST_SUBSCRIBER_COOKIES}, and
|
||
|
* {@link #EXTRA_STATS_DIMENSIONS_VALUE}.
|
||
|
* <p>
|
||
|
* This function can only be called by the owner (uid) of the config. It must be called each
|
||
|
* time statsd starts. The config must have been added first (via {@link #addConfig}).
|
||
|
* This call can block on statsd.
|
||
|
*
|
||
|
* @param pendingIntent the PendingIntent to use when broadcasting info to the subscriber
|
||
|
* associated with the given subscriberId. May be null, in which case
|
||
|
* it undoes any previous setting of this subscriberId.
|
||
|
* @param configKey The integer naming the config to which this subscriber is attached.
|
||
|
* @param subscriberId ID of the subscriber, as used in the config.
|
||
|
* @throws StatsUnavailableException if unsuccessful due to failing to connect to stats service
|
||
|
*/
|
||
|
@RequiresPermission(allOf = { DUMP, PACKAGE_USAGE_STATS })
|
||
|
public void setBroadcastSubscriber(
|
||
|
PendingIntent pendingIntent, long configKey, long subscriberId)
|
||
|
throws StatsUnavailableException {
|
||
|
synchronized (sLock) {
|
||
|
try {
|
||
|
IStatsManagerService service = getIStatsManagerServiceLocked();
|
||
|
if (pendingIntent != null) {
|
||
|
service.setBroadcastSubscriber(configKey, subscriberId, pendingIntent,
|
||
|
mContext.getOpPackageName());
|
||
|
} else {
|
||
|
service.unsetBroadcastSubscriber(configKey, subscriberId,
|
||
|
mContext.getOpPackageName());
|
||
|
}
|
||
|
} catch (RemoteException e) {
|
||
|
Log.e(TAG, "Failed to connect to statsmanager when adding broadcast subscriber",
|
||
|
e);
|
||
|
throw new StatsUnavailableException("could not connect", e);
|
||
|
} catch (SecurityException e) {
|
||
|
throw new StatsUnavailableException(e.getMessage(), e);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// TODO: Temporary for backwards compatibility. Remove.
|
||
|
/**
|
||
|
* @deprecated Use {@link #setBroadcastSubscriber(PendingIntent, long, long)}
|
||
|
*/
|
||
|
@Deprecated
|
||
|
@RequiresPermission(allOf = { DUMP, PACKAGE_USAGE_STATS })
|
||
|
public boolean setBroadcastSubscriber(
|
||
|
long configKey, long subscriberId, PendingIntent pendingIntent) {
|
||
|
try {
|
||
|
setBroadcastSubscriber(pendingIntent, configKey, subscriberId);
|
||
|
return true;
|
||
|
} catch (StatsUnavailableException e) {
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Registers the operation that is called to retrieve the metrics data. This must be called
|
||
|
* each time statsd starts. The config must have been added first (via {@link #addConfig},
|
||
|
* although addConfig could have been called on a previous boot). This operation allows
|
||
|
* statsd to send metrics data whenever statsd determines that the metrics in memory are
|
||
|
* approaching the memory limits. The fetch operation should call {@link #getReports} to fetch
|
||
|
* the data, which also deletes the retrieved metrics from statsd's memory.
|
||
|
* This call can block on statsd.
|
||
|
*
|
||
|
* @param pendingIntent the PendingIntent to use when broadcasting info to the subscriber
|
||
|
* associated with the given subscriberId. May be null, in which case
|
||
|
* it removes any associated pending intent with this configKey.
|
||
|
* @param configKey The integer naming the config to which this operation is attached.
|
||
|
* @throws StatsUnavailableException if unsuccessful due to failing to connect to stats service
|
||
|
*/
|
||
|
@RequiresPermission(allOf = { DUMP, PACKAGE_USAGE_STATS })
|
||
|
public void setFetchReportsOperation(PendingIntent pendingIntent, long configKey)
|
||
|
throws StatsUnavailableException {
|
||
|
synchronized (sLock) {
|
||
|
try {
|
||
|
IStatsManagerService service = getIStatsManagerServiceLocked();
|
||
|
if (pendingIntent == null) {
|
||
|
service.removeDataFetchOperation(configKey, mContext.getOpPackageName());
|
||
|
} else {
|
||
|
service.setDataFetchOperation(configKey, pendingIntent,
|
||
|
mContext.getOpPackageName());
|
||
|
}
|
||
|
|
||
|
} catch (RemoteException e) {
|
||
|
Log.e(TAG, "Failed to connect to statsmanager when registering data listener.");
|
||
|
throw new StatsUnavailableException("could not connect", e);
|
||
|
} catch (SecurityException e) {
|
||
|
throw new StatsUnavailableException(e.getMessage(), e);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Registers the operation that is called whenever there is a change in which configs are
|
||
|
* active. This must be called each time statsd starts. This operation allows
|
||
|
* statsd to inform clients that they should pull data of the configs that are currently
|
||
|
* active. The activeConfigsChangedOperation should set periodic alarms to pull data of configs
|
||
|
* that are active and stop pulling data of configs that are no longer active.
|
||
|
* This call can block on statsd.
|
||
|
*
|
||
|
* @param pendingIntent the PendingIntent to use when broadcasting info to the subscriber
|
||
|
* associated with the given subscriberId. May be null, in which case
|
||
|
* it removes any associated pending intent for this client.
|
||
|
* @return A list of configs that are currently active for this client. If the pendingIntent is
|
||
|
* null, this will be an empty list.
|
||
|
* @throws StatsUnavailableException if unsuccessful due to failing to connect to stats service
|
||
|
*/
|
||
|
@RequiresPermission(allOf = { DUMP, PACKAGE_USAGE_STATS })
|
||
|
public @NonNull long[] setActiveConfigsChangedOperation(@Nullable PendingIntent pendingIntent)
|
||
|
throws StatsUnavailableException {
|
||
|
synchronized (sLock) {
|
||
|
try {
|
||
|
IStatsManagerService service = getIStatsManagerServiceLocked();
|
||
|
if (pendingIntent == null) {
|
||
|
service.removeActiveConfigsChangedOperation(mContext.getOpPackageName());
|
||
|
return new long[0];
|
||
|
} else {
|
||
|
return service.setActiveConfigsChangedOperation(pendingIntent,
|
||
|
mContext.getOpPackageName());
|
||
|
}
|
||
|
|
||
|
} catch (RemoteException e) {
|
||
|
Log.e(TAG, "Failed to connect to statsmanager "
|
||
|
+ "when registering active configs listener.");
|
||
|
throw new StatsUnavailableException("could not connect", e);
|
||
|
} catch (SecurityException e) {
|
||
|
throw new StatsUnavailableException(e.getMessage(), e);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Registers the operation that is called whenever there is a change in the restricted metrics
|
||
|
* for a specified config that are present for this client. This operation allows statsd to
|
||
|
* inform the client about the current restricted metric ids available to be queried for the
|
||
|
* specified config. This call can block on statsd.
|
||
|
*
|
||
|
* If there is no config in statsd that matches the provided config package and key, an empty
|
||
|
* list is returned. The pending intent will be tracked, and the operation will be called
|
||
|
* whenever a matching config is added.
|
||
|
*
|
||
|
* @param configKey The configKey passed by the package that added the config in
|
||
|
* StatsManager#addConfig
|
||
|
* @param configPackage The package that added the config in StatsManager#addConfig
|
||
|
* @param pendingIntent the PendingIntent to use when broadcasting info to caller.
|
||
|
* May be null, in which case it removes any associated pending intent
|
||
|
* for this client.
|
||
|
* @return A list of metric ids identifying the restricted metrics that are currently available
|
||
|
* to be queried for the specified config.
|
||
|
* If the pendingIntent is null, this will be an empty list.
|
||
|
* @throws StatsUnavailableException if unsuccessful due to failing to connect to stats service
|
||
|
*/
|
||
|
@RequiresPermission(READ_RESTRICTED_STATS)
|
||
|
@RequiresApi(Build.VERSION_CODES.UPSIDE_DOWN_CAKE)
|
||
|
public @NonNull long[] setRestrictedMetricsChangedOperation(long configKey,
|
||
|
@NonNull String configPackage,
|
||
|
@Nullable PendingIntent pendingIntent)
|
||
|
throws StatsUnavailableException {
|
||
|
synchronized (sLock) {
|
||
|
try {
|
||
|
IStatsManagerService service = getIStatsManagerServiceLocked();
|
||
|
if (pendingIntent == null) {
|
||
|
service.removeRestrictedMetricsChangedOperation(configKey, configPackage);
|
||
|
return new long[0];
|
||
|
} else {
|
||
|
return service.setRestrictedMetricsChangedOperation(pendingIntent,
|
||
|
configKey, configPackage);
|
||
|
}
|
||
|
|
||
|
} catch (RemoteException e) {
|
||
|
Log.e(TAG, "Failed to connect to statsmanager "
|
||
|
+ "when registering restricted metrics listener.");
|
||
|
throw new StatsUnavailableException("could not connect", e);
|
||
|
} catch (SecurityException e) {
|
||
|
throw new StatsUnavailableException(e.getMessage(), e);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Queries the underlying service based on query received and populates the OutcomeReceiver via
|
||
|
* callback. This call is blocking on statsd being available, but is otherwise nonblocking.
|
||
|
* i.e. the call can return before the query processing is done.
|
||
|
* <p>
|
||
|
* Two types of tables are supported: Metric tables and the device information table.
|
||
|
* </p>
|
||
|
* <p>
|
||
|
* The device information table is named device_info and contains the following columns:
|
||
|
* sdkVersion, model, product, hardware, device, osBuild, fingerprint, brand, manufacturer, and
|
||
|
* board. These columns correspond to {@link Build.VERSION.SDK_INT}, {@link Build.MODEL},
|
||
|
* {@link Build.PRODUCT}, {@link Build.HARDWARE}, {@link Build.DEVICE}, {@link Build.ID},
|
||
|
* {@link Build.FINGERPRINT}, {@link Build.BRAND}, {@link Build.MANUFACTURER},
|
||
|
* {@link Build.BOARD} respectively.
|
||
|
* </p>
|
||
|
* <p>
|
||
|
* The metric tables are named metric_METRIC_ID where METRIC_ID is the metric id that is part
|
||
|
* of the wire encoded config passed to {@link #addConfig(long, byte[])}. If the metric id is
|
||
|
* negative, then the '-' character is replaced with 'n' in the table name. Each metric table
|
||
|
* contains the 3 columns followed by n columns of the following form: atomId,
|
||
|
* elapsedTimestampNs, wallTimestampNs, field_1, field_2, field_3 ... field_n. These
|
||
|
* columns correspond to to the id of the atom from frameworks/proto_logging/stats/atoms.proto,
|
||
|
* time when the atom is recorded, and the data fields within each atom.
|
||
|
* </p>
|
||
|
* @param configKey The configKey passed by the package that added
|
||
|
* the config being queried in StatsManager#addConfig
|
||
|
* @param configPackage The package that added the config being queried in
|
||
|
* StatsManager#addConfig
|
||
|
* @param query the query object encapsulating a sql-string and necessary config to query
|
||
|
* underlying sql-based data store.
|
||
|
* @param executor the executor on which outcomeReceiver will be invoked.
|
||
|
* @param outcomeReceiver the receiver to be populated with cursor pointing to result data.
|
||
|
*/
|
||
|
@RequiresPermission(READ_RESTRICTED_STATS)
|
||
|
@RequiresApi(Build.VERSION_CODES.UPSIDE_DOWN_CAKE)
|
||
|
public void query(long configKey, @NonNull String configPackage, @NonNull StatsQuery query,
|
||
|
@NonNull @CallbackExecutor Executor executor,
|
||
|
@NonNull OutcomeReceiver<StatsCursor, StatsQueryException> outcomeReceiver)
|
||
|
throws StatsUnavailableException {
|
||
|
if(query.getSqlDialect() != StatsQuery.DIALECT_SQLITE) {
|
||
|
executor.execute(() -> {
|
||
|
outcomeReceiver.onError(new StatsQueryException("Unsupported Sql Dialect"));
|
||
|
});
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
StatsQueryCallbackInternal callbackInternal =
|
||
|
new StatsQueryCallbackInternal(outcomeReceiver, executor);
|
||
|
synchronized (sLock) {
|
||
|
try {
|
||
|
IStatsManagerService service = getIStatsManagerServiceLocked();
|
||
|
service.querySql(query.getRawSql(), query.getMinSqlClientVersion(),
|
||
|
query.getPolicyConfig(), callbackInternal, configKey,
|
||
|
configPackage);
|
||
|
} catch (RemoteException | IllegalStateException e) {
|
||
|
throw new StatsUnavailableException("could not connect", e);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
// TODO: Temporary for backwards compatibility. Remove.
|
||
|
/**
|
||
|
* @deprecated Use {@link #setFetchReportsOperation(PendingIntent, long)}
|
||
|
*/
|
||
|
@Deprecated
|
||
|
@RequiresPermission(allOf = { DUMP, PACKAGE_USAGE_STATS })
|
||
|
public boolean setDataFetchOperation(long configKey, PendingIntent pendingIntent) {
|
||
|
try {
|
||
|
setFetchReportsOperation(pendingIntent, configKey);
|
||
|
return true;
|
||
|
} catch (StatsUnavailableException e) {
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Request the data collected for the given configKey.
|
||
|
* This getter is destructive - it also clears the retrieved metrics from statsd's memory.
|
||
|
* This call can block on statsd.
|
||
|
*
|
||
|
* @param configKey Configuration key to retrieve data from.
|
||
|
* @return Serialized ConfigMetricsReportList proto.
|
||
|
* @throws StatsUnavailableException if unsuccessful due to failing to connect to stats service
|
||
|
*/
|
||
|
@RequiresPermission(allOf = { DUMP, PACKAGE_USAGE_STATS })
|
||
|
public byte[] getReports(long configKey) throws StatsUnavailableException {
|
||
|
synchronized (sLock) {
|
||
|
try {
|
||
|
IStatsManagerService service = getIStatsManagerServiceLocked();
|
||
|
if (getUseFileDescriptor()) {
|
||
|
return getDataWithFd(service, configKey, mContext.getOpPackageName());
|
||
|
} else {
|
||
|
return service.getData(configKey, mContext.getOpPackageName());
|
||
|
}
|
||
|
} catch (RemoteException e) {
|
||
|
Log.e(TAG, "Failed to connect to statsmanager when getting data");
|
||
|
throw new StatsUnavailableException("could not connect", e);
|
||
|
} catch (SecurityException e) {
|
||
|
throw new StatsUnavailableException(e.getMessage(), e);
|
||
|
} catch (IllegalStateException e) {
|
||
|
Log.e(TAG, "Failed to getReports in statsmanager");
|
||
|
throw new StatsUnavailableException(e.getMessage(), e);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// TODO: Temporary for backwards compatibility. Remove.
|
||
|
/**
|
||
|
* @deprecated Use {@link #getReports(long)}
|
||
|
*/
|
||
|
@Deprecated
|
||
|
@RequiresPermission(allOf = { DUMP, PACKAGE_USAGE_STATS })
|
||
|
public @Nullable byte[] getData(long configKey) {
|
||
|
try {
|
||
|
return getReports(configKey);
|
||
|
} catch (StatsUnavailableException e) {
|
||
|
return null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Clients can request metadata for statsd. Will contain stats across all configurations but not
|
||
|
* the actual metrics themselves (metrics must be collected via {@link #getReports(long)}.
|
||
|
* This getter is not destructive and will not reset any metrics/counters.
|
||
|
* This call can block on statsd.
|
||
|
*
|
||
|
* @return Serialized StatsdStatsReport proto.
|
||
|
* @throws StatsUnavailableException if unsuccessful due to failing to connect to stats service
|
||
|
*/
|
||
|
@RequiresPermission(allOf = { DUMP, PACKAGE_USAGE_STATS })
|
||
|
public byte[] getStatsMetadata() throws StatsUnavailableException {
|
||
|
synchronized (sLock) {
|
||
|
try {
|
||
|
IStatsManagerService service = getIStatsManagerServiceLocked();
|
||
|
return service.getMetadata(mContext.getOpPackageName());
|
||
|
} catch (RemoteException e) {
|
||
|
Log.e(TAG, "Failed to connect to statsmanager when getting metadata");
|
||
|
throw new StatsUnavailableException("could not connect", e);
|
||
|
} catch (SecurityException e) {
|
||
|
throw new StatsUnavailableException(e.getMessage(), e);
|
||
|
} catch (IllegalStateException e) {
|
||
|
Log.e(TAG, "Failed to getStatsMetadata in statsmanager");
|
||
|
throw new StatsUnavailableException(e.getMessage(), e);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// TODO: Temporary for backwards compatibility. Remove.
|
||
|
/**
|
||
|
* @deprecated Use {@link #getStatsMetadata()}
|
||
|
*/
|
||
|
@Deprecated
|
||
|
@RequiresPermission(allOf = { DUMP, PACKAGE_USAGE_STATS })
|
||
|
public @Nullable byte[] getMetadata() {
|
||
|
try {
|
||
|
return getStatsMetadata();
|
||
|
} catch (StatsUnavailableException e) {
|
||
|
return null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the experiments IDs registered with statsd, or an empty array if there aren't any.
|
||
|
*
|
||
|
* This call can block on statsd.
|
||
|
*
|
||
|
* @throws StatsUnavailableException if unsuccessful due to failing to connect to stats service
|
||
|
*/
|
||
|
@RequiresPermission(allOf = {DUMP, PACKAGE_USAGE_STATS})
|
||
|
public long[] getRegisteredExperimentIds()
|
||
|
throws StatsUnavailableException {
|
||
|
synchronized (sLock) {
|
||
|
try {
|
||
|
IStatsManagerService service = getIStatsManagerServiceLocked();
|
||
|
return service.getRegisteredExperimentIds();
|
||
|
} catch (RemoteException e) {
|
||
|
if (DEBUG) {
|
||
|
Log.d(TAG,
|
||
|
"Failed to connect to StatsManagerService when getting "
|
||
|
+ "registered experiment IDs");
|
||
|
}
|
||
|
throw new StatsUnavailableException("could not connect", e);
|
||
|
} catch (SecurityException e) {
|
||
|
throw new StatsUnavailableException(e.getMessage(), e);
|
||
|
} catch (IllegalStateException e) {
|
||
|
Log.e(TAG, "Failed to getRegisteredExperimentIds in statsmanager");
|
||
|
throw new StatsUnavailableException(e.getMessage(), e);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Sets a callback for an atom when that atom is to be pulled. The stats service will
|
||
|
* invoke pullData in the callback when the stats service determines that this atom needs to be
|
||
|
* pulled. This method should not be called by third-party apps.
|
||
|
*
|
||
|
* @param atomTag The tag of the atom for this puller callback.
|
||
|
* @param metadata Optional metadata specifying the timeout, cool down time, and
|
||
|
* additive fields for mapping isolated to host uids.
|
||
|
* @param executor The executor in which to run the callback.
|
||
|
* @param callback The callback to be invoked when the stats service pulls the atom.
|
||
|
*
|
||
|
*/
|
||
|
@RequiresPermission(android.Manifest.permission.REGISTER_STATS_PULL_ATOM)
|
||
|
public void setPullAtomCallback(int atomTag, @Nullable PullAtomMetadata metadata,
|
||
|
@NonNull @CallbackExecutor Executor executor,
|
||
|
@NonNull StatsPullAtomCallback callback) {
|
||
|
long coolDownMillis =
|
||
|
metadata == null ? DEFAULT_COOL_DOWN_MILLIS : metadata.mCoolDownMillis;
|
||
|
long timeoutMillis = metadata == null ? DEFAULT_TIMEOUT_MILLIS : metadata.mTimeoutMillis;
|
||
|
int[] additiveFields = metadata == null ? new int[0] : metadata.mAdditiveFields;
|
||
|
if (additiveFields == null) {
|
||
|
additiveFields = new int[0];
|
||
|
}
|
||
|
|
||
|
synchronized (sLock) {
|
||
|
try {
|
||
|
IStatsManagerService service = getIStatsManagerServiceLocked();
|
||
|
PullAtomCallbackInternal rec =
|
||
|
new PullAtomCallbackInternal(atomTag, callback, executor);
|
||
|
service.registerPullAtomCallback(
|
||
|
atomTag, coolDownMillis, timeoutMillis, additiveFields, rec);
|
||
|
} catch (RemoteException e) {
|
||
|
throw new RuntimeException("Unable to register pull callback", e);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Clears a callback for an atom when that atom is to be pulled. Note that any ongoing
|
||
|
* pulls will still occur. This method should not be called by third-party apps.
|
||
|
*
|
||
|
* @param atomTag The tag of the atom of which to unregister
|
||
|
*
|
||
|
*/
|
||
|
@RequiresPermission(android.Manifest.permission.REGISTER_STATS_PULL_ATOM)
|
||
|
public void clearPullAtomCallback(int atomTag) {
|
||
|
synchronized (sLock) {
|
||
|
try {
|
||
|
IStatsManagerService service = getIStatsManagerServiceLocked();
|
||
|
service.unregisterPullAtomCallback(atomTag);
|
||
|
} catch (RemoteException e) {
|
||
|
throw new RuntimeException("Unable to unregister pull atom callback");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private static class PullAtomCallbackInternal extends IPullAtomCallback.Stub {
|
||
|
public final int mAtomId;
|
||
|
public final StatsPullAtomCallback mCallback;
|
||
|
public final Executor mExecutor;
|
||
|
|
||
|
PullAtomCallbackInternal(int atomId, StatsPullAtomCallback callback, Executor executor) {
|
||
|
mAtomId = atomId;
|
||
|
mCallback = callback;
|
||
|
mExecutor = executor;
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public void onPullAtom(int atomTag, IPullAtomResultReceiver resultReceiver) {
|
||
|
final long token = Binder.clearCallingIdentity();
|
||
|
try {
|
||
|
mExecutor.execute(() -> {
|
||
|
List<StatsEvent> data = new ArrayList<>();
|
||
|
int successInt = mCallback.onPullAtom(atomTag, data);
|
||
|
boolean success = successInt == PULL_SUCCESS;
|
||
|
StatsEventParcel[] parcels = new StatsEventParcel[data.size()];
|
||
|
for (int i = 0; i < data.size(); i++) {
|
||
|
parcels[i] = new StatsEventParcel();
|
||
|
parcels[i].buffer = data.get(i).getBytes();
|
||
|
}
|
||
|
try {
|
||
|
resultReceiver.pullFinished(atomTag, success, parcels);
|
||
|
} catch (RemoteException e) {
|
||
|
Log.w(TAG, "StatsPullResultReceiver failed for tag " + mAtomId
|
||
|
+ " due to TransactionTooLarge. Calling pullFinish with no data");
|
||
|
StatsEventParcel[] emptyData = new StatsEventParcel[0];
|
||
|
try {
|
||
|
resultReceiver.pullFinished(atomTag, /*success=*/false, emptyData);
|
||
|
} catch (RemoteException nestedException) {
|
||
|
Log.w(TAG, "StatsPullResultReceiver failed for tag " + mAtomId
|
||
|
+ " with empty payload");
|
||
|
}
|
||
|
}
|
||
|
});
|
||
|
} finally {
|
||
|
Binder.restoreCallingIdentity(token);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Metadata required for registering a StatsPullAtomCallback.
|
||
|
* All fields are optional, and defaults will be used for fields that are unspecified.
|
||
|
*
|
||
|
*/
|
||
|
public static class PullAtomMetadata {
|
||
|
private final long mCoolDownMillis;
|
||
|
private final long mTimeoutMillis;
|
||
|
private final int[] mAdditiveFields;
|
||
|
|
||
|
// Private Constructor for builder
|
||
|
private PullAtomMetadata(long coolDownMillis, long timeoutMillis, int[] additiveFields) {
|
||
|
mCoolDownMillis = coolDownMillis;
|
||
|
mTimeoutMillis = timeoutMillis;
|
||
|
mAdditiveFields = additiveFields;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Builder for PullAtomMetadata.
|
||
|
*/
|
||
|
public static class Builder {
|
||
|
private long mCoolDownMillis;
|
||
|
private long mTimeoutMillis;
|
||
|
private int[] mAdditiveFields;
|
||
|
|
||
|
/**
|
||
|
* Returns a new PullAtomMetadata.Builder object for constructing PullAtomMetadata for
|
||
|
* StatsManager#registerPullAtomCallback
|
||
|
*/
|
||
|
public Builder() {
|
||
|
mCoolDownMillis = DEFAULT_COOL_DOWN_MILLIS;
|
||
|
mTimeoutMillis = DEFAULT_TIMEOUT_MILLIS;
|
||
|
mAdditiveFields = null;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Set the cool down time of the pull in milliseconds. If two successive pulls are
|
||
|
* issued within the cool down, a cached version of the first pull will be used for the
|
||
|
* second pull. The minimum allowed cool down is 1 second.
|
||
|
*/
|
||
|
@NonNull
|
||
|
public Builder setCoolDownMillis(long coolDownMillis) {
|
||
|
mCoolDownMillis = coolDownMillis;
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Set the maximum time the pull can take in milliseconds. The maximum allowed timeout
|
||
|
* is 10 seconds.
|
||
|
*/
|
||
|
@NonNull
|
||
|
public Builder setTimeoutMillis(long timeoutMillis) {
|
||
|
mTimeoutMillis = timeoutMillis;
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Set the additive fields of this pulled atom.
|
||
|
*
|
||
|
* This is only applicable for atoms which have a uid field. When tasks are run in
|
||
|
* isolated processes, the data will be attributed to the host uid. Additive fields
|
||
|
* will be combined when the non-additive fields are the same.
|
||
|
*/
|
||
|
@NonNull
|
||
|
public Builder setAdditiveFields(@NonNull int[] additiveFields) {
|
||
|
mAdditiveFields = additiveFields;
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Builds and returns a PullAtomMetadata object with the values set in the builder and
|
||
|
* defaults for unset fields.
|
||
|
*/
|
||
|
@NonNull
|
||
|
public PullAtomMetadata build() {
|
||
|
return new PullAtomMetadata(mCoolDownMillis, mTimeoutMillis, mAdditiveFields);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Return the cool down time of this pull in milliseconds.
|
||
|
*/
|
||
|
public long getCoolDownMillis() {
|
||
|
return mCoolDownMillis;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Return the maximum amount of time this pull can take in milliseconds.
|
||
|
*/
|
||
|
public long getTimeoutMillis() {
|
||
|
return mTimeoutMillis;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Return the additive fields of this pulled atom.
|
||
|
*
|
||
|
* This is only applicable for atoms that have a uid field. When tasks are run in
|
||
|
* isolated processes, the data will be attributed to the host uid. Additive fields
|
||
|
* will be combined when the non-additive fields are the same.
|
||
|
*/
|
||
|
@Nullable
|
||
|
public int[] getAdditiveFields() {
|
||
|
return mAdditiveFields;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Callback interface for pulling atoms requested by the stats service.
|
||
|
*
|
||
|
*/
|
||
|
public interface StatsPullAtomCallback {
|
||
|
/**
|
||
|
* Pull data for the specified atom tag, filling in the provided list of StatsEvent data.
|
||
|
* @return {@link #PULL_SUCCESS} if the pull was successful, or {@link #PULL_SKIP} if not.
|
||
|
*/
|
||
|
int onPullAtom(int atomTag, @NonNull List<StatsEvent> data);
|
||
|
}
|
||
|
|
||
|
@GuardedBy("sLock")
|
||
|
private IStatsManagerService getIStatsManagerServiceLocked() {
|
||
|
if (mStatsManagerService != null) {
|
||
|
return mStatsManagerService;
|
||
|
}
|
||
|
mStatsManagerService = IStatsManagerService.Stub.asInterface(
|
||
|
StatsFrameworkInitializer
|
||
|
.getStatsServiceManager()
|
||
|
.getStatsManagerServiceRegisterer()
|
||
|
.get());
|
||
|
return mStatsManagerService;
|
||
|
}
|
||
|
|
||
|
private static class StatsQueryCallbackInternal extends IStatsQueryCallback.Stub {
|
||
|
OutcomeReceiver<StatsCursor, StatsQueryException> queryCallback;
|
||
|
Executor mExecutor;
|
||
|
|
||
|
StatsQueryCallbackInternal(OutcomeReceiver<StatsCursor, StatsQueryException> queryCallback,
|
||
|
@NonNull @CallbackExecutor Executor executor) {
|
||
|
this.queryCallback = queryCallback;
|
||
|
this.mExecutor = executor;
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public void sendResults(String[] queryData, String[] columnNames, int[] columnTypes,
|
||
|
int rowCount) {
|
||
|
if (!SdkLevel.isAtLeastU()) {
|
||
|
throw new IllegalStateException(
|
||
|
"StatsManager#query is not available before Android U");
|
||
|
}
|
||
|
final long token = Binder.clearCallingIdentity();
|
||
|
try {
|
||
|
mExecutor.execute(() -> {
|
||
|
StatsCursor cursor = new StatsCursor(queryData, columnNames, columnTypes,
|
||
|
rowCount);
|
||
|
queryCallback.onResult(cursor);
|
||
|
});
|
||
|
} finally {
|
||
|
Binder.restoreCallingIdentity(token);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public void sendFailure(String error) {
|
||
|
if (!SdkLevel.isAtLeastU()) {
|
||
|
throw new IllegalStateException(
|
||
|
"StatsManager#query is not available before Android U");
|
||
|
}
|
||
|
final long token = Binder.clearCallingIdentity();
|
||
|
try {
|
||
|
mExecutor.execute(() -> {
|
||
|
queryCallback.onError(new StatsQueryException(error));
|
||
|
});
|
||
|
} finally {
|
||
|
Binder.restoreCallingIdentity(token);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Exception thrown when communication with the stats service fails (eg if it is not available).
|
||
|
* This might be thrown early during boot before the stats service has started or if it crashed.
|
||
|
*/
|
||
|
public static class StatsUnavailableException extends AndroidException {
|
||
|
public StatsUnavailableException(String reason) {
|
||
|
super("Failed to connect to statsd: " + reason);
|
||
|
}
|
||
|
|
||
|
public StatsUnavailableException(String reason, Throwable e) {
|
||
|
super("Failed to connect to statsd: " + reason, e);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Exception thrown when executing a query in statsd fails for any reason. This might be thrown
|
||
|
* if the query is malformed or if there is a database error when executing the query.
|
||
|
*/
|
||
|
public static class StatsQueryException extends AndroidException {
|
||
|
public StatsQueryException(@NonNull String reason) {
|
||
|
super("Failed to query statsd: " + reason);
|
||
|
}
|
||
|
|
||
|
public StatsQueryException(@NonNull String reason, @NonNull Throwable e) {
|
||
|
super("Failed to query statsd: " + reason, e);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private static boolean getUseFileDescriptor() {
|
||
|
return SdkLevel.isAtLeastT();
|
||
|
}
|
||
|
|
||
|
private static final int MAX_BUFFER_SIZE = 1024 * 1024 * 20; // 20MB
|
||
|
private static final int CHUNK_SIZE = 1024 * 64; // 64kB
|
||
|
|
||
|
/**
|
||
|
* Executes a binder transaction with file descriptors.
|
||
|
*/
|
||
|
private static byte[] getDataWithFd(IStatsManagerService service, long configKey,
|
||
|
String packageName) throws IllegalStateException, RemoteException {
|
||
|
ParcelFileDescriptor[] pipe;
|
||
|
try {
|
||
|
pipe = ParcelFileDescriptor.createPipe();
|
||
|
} catch (IOException e) {
|
||
|
Log.e(TAG, "Failed to create a pipe to receive reports.", e);
|
||
|
throw new IllegalStateException("Failed to create a pipe to receive reports.", e);
|
||
|
}
|
||
|
|
||
|
ParcelFileDescriptor readFd = pipe[0];
|
||
|
ParcelFileDescriptor writeFd = pipe[1];
|
||
|
|
||
|
// StatsManagerService write/flush will block until read() will start to consume data.
|
||
|
// OTOH read cannot start until binder sync operation is over.
|
||
|
// To decouple this dependency call to StatsManagerService should be async
|
||
|
service.getDataFd(configKey, packageName, writeFd);
|
||
|
try {
|
||
|
writeFd.close();
|
||
|
} catch (IOException e) {
|
||
|
Log.e(TAG, "Failed to pass FD to StatsManagerService", e);
|
||
|
throw new IllegalStateException("Failed to pass FD to StatsManagerService.", e);
|
||
|
}
|
||
|
|
||
|
try (FileInputStream inputStream = new ParcelFileDescriptor.AutoCloseInputStream(readFd);
|
||
|
DataInputStream dataInputStream = new DataInputStream(inputStream)) {
|
||
|
|
||
|
byte[] chunk = new byte[CHUNK_SIZE];
|
||
|
|
||
|
// read 4 bytes determining size of the reports
|
||
|
final int expectedReportSize = dataInputStream.readInt();
|
||
|
if (expectedReportSize > MAX_BUFFER_SIZE || expectedReportSize <= 0) {
|
||
|
Log.e(TAG, "expectedReportSize must be in a range (0, MAX_BUFFER_SIZE]: "
|
||
|
+ expectedReportSize);
|
||
|
throw new IllegalStateException("expectedReportSize > MAX_BUFFER_SIZE.");
|
||
|
}
|
||
|
|
||
|
ByteBuffer resultBuffer = ByteBuffer.allocate(expectedReportSize);
|
||
|
// read chunk-by-chunk, it will block until next chunk is ready or until EOF symbol
|
||
|
// is read. EOF symbol is written by FD close(), which happens when async binder
|
||
|
// transaction is over.
|
||
|
int chunkLen = 0;
|
||
|
int readBytes = 0;
|
||
|
// -1 denotes EOF
|
||
|
while ((chunkLen = dataInputStream.read(chunk, 0, CHUNK_SIZE)) != -1) {
|
||
|
try {
|
||
|
resultBuffer.put(chunk, 0, chunkLen);
|
||
|
} catch (BufferOverflowException e) {
|
||
|
Log.e(TAG, "Failed to store report chunk", e);
|
||
|
throw new IllegalStateException("Failed to store report chunk.", e);
|
||
|
}
|
||
|
readBytes += chunkLen;
|
||
|
}
|
||
|
if (readBytes != expectedReportSize) {
|
||
|
throw new IllegalStateException("Incomplete data read from StatsManagerService.");
|
||
|
}
|
||
|
return resultBuffer.array();
|
||
|
} catch (IOException e) {
|
||
|
Log.e(TAG, "Failed to read report data from StatsManagerService", e);
|
||
|
throw new IllegalStateException("Failed to read report data from StatsManagerService.",
|
||
|
e);
|
||
|
}
|
||
|
}
|
||
|
}
|