/* * Copyright (C) 2014 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 android.annotation.Nullable; import android.os.Parcel; import android.os.ParcelFileDescriptor; import android.os.Parcelable; import android.util.Slog; import android.util.proto.ProtoOutputStream; import java.io.IOException; import java.util.Objects; /** * System private API for passing profiler settings. * * {@hide} */ public class ProfilerInfo implements Parcelable { // Version of the profiler output public static final int OUTPUT_VERSION_DEFAULT = 1; // CLOCK_TYPE_DEFAULT chooses the default used by ART. ART uses CLOCK_TYPE_DUAL by default (see // kDefaultTraceClockSource in art/runtime/runtime_globals.h). public static final int CLOCK_TYPE_DEFAULT = 0x000; // The values of these constants are chosen such that they correspond to the flags passed to // VMDebug.startMethodTracing to choose the corresponding clock type (see // core/java/android/app/ActivityThread.java). // The flag values are defined in ART (see TraceFlag in art/runtime/trace.h). public static final int CLOCK_TYPE_WALL = 0x010; public static final int CLOCK_TYPE_THREAD_CPU = 0x100; public static final int CLOCK_TYPE_DUAL = 0x110; // The second and third bits of the flags field specify the trace format version. This should // match with kTraceFormatVersionShift defined in art/runtime/trace.h. public static final int TRACE_FORMAT_VERSION_SHIFT = 1; private static final String TAG = "ProfilerInfo"; /* Name of profile output file. */ public final String profileFile; /* File descriptor for profile output file, can be null. */ public ParcelFileDescriptor profileFd; /* Indicates sample profiling when nonzero, interval in microseconds. */ public final int samplingInterval; /* Automatically stop the profiler when the app goes idle. */ public final boolean autoStopProfiler; /* * Indicates whether to stream the profiling info to the out file continuously. */ public final boolean streamingOutput; /** * Denotes an agent (and its parameters) to attach for profiling. */ public final String agent; /** * Whether the {@link agent} should be attached early (before bind-application) or during * bind-application. Agents attached prior to binding cannot be loaded from the app's APK * directly and must be given as an absolute path (or available in the default LD_LIBRARY_PATH). * Agents attached during bind-application will miss early setup (e.g., resource initialization * and classloader generation), but are searched in the app's library search path. */ public final boolean attachAgentDuringBind; /** * Indicates the clock source to be used for profiling. The source could be wallclock, thread * cpu or both */ public final int clockType; /** * Indicates the version of profiler output. */ public final int profilerOutputVersion; public ProfilerInfo(String filename, ParcelFileDescriptor fd, int interval, boolean autoStop, boolean streaming, String agent, boolean attachAgentDuringBind, int clockType, int profilerOutputVersion) { profileFile = filename; profileFd = fd; samplingInterval = interval; autoStopProfiler = autoStop; streamingOutput = streaming; this.clockType = clockType; this.agent = agent; this.attachAgentDuringBind = attachAgentDuringBind; this.profilerOutputVersion = profilerOutputVersion; } public ProfilerInfo(ProfilerInfo in) { profileFile = in.profileFile; profileFd = in.profileFd; samplingInterval = in.samplingInterval; autoStopProfiler = in.autoStopProfiler; streamingOutput = in.streamingOutput; agent = in.agent; attachAgentDuringBind = in.attachAgentDuringBind; clockType = in.clockType; profilerOutputVersion = in.profilerOutputVersion; } /** * Get the value for the clock type corresponding to the option string passed to the activity * manager. am profile start / am start-activity start-profiler commands accept clock-type * option to choose the source of timestamps when profiling. This function maps the option * string to the value of flags that is used when calling VMDebug.startMethodTracing */ public static int getClockTypeFromString(String type) { if ("thread-cpu".equals(type)) { return CLOCK_TYPE_THREAD_CPU; } else if ("wall".equals(type)) { return CLOCK_TYPE_WALL; } else if ("dual".equals(type)) { return CLOCK_TYPE_DUAL; } else { return CLOCK_TYPE_DEFAULT; } } /** * Get the flags that need to be passed to VMDebug.startMethodTracing to specify the desired * output format. */ public static int getFlagsForOutputVersion(int version) { // Only two version 1 and version 2 are supported. Just use the default if we see an unknown // version. if (version != 1 || version != 2) { version = OUTPUT_VERSION_DEFAULT; } // The encoded version in the flags starts from 0, where as the version that we read from // user starts from 1. So, subtract one before encoding it in the flags. return (version - 1) << TRACE_FORMAT_VERSION_SHIFT; } /** * Return a new ProfilerInfo instance, with fields populated from this object, * and {@link agent} and {@link attachAgentDuringBind} as given. */ public ProfilerInfo setAgent(String agent, boolean attachAgentDuringBind) { return new ProfilerInfo(this.profileFile, this.profileFd, this.samplingInterval, this.autoStopProfiler, this.streamingOutput, agent, attachAgentDuringBind, this.clockType, this.profilerOutputVersion); } /** * Close profileFd, if it is open. The field will be null after a call to this function. */ public void closeFd() { if (profileFd != null) { try { profileFd.close(); } catch (IOException e) { Slog.w(TAG, "Failure closing profile fd", e); } profileFd = null; } } @Override public int describeContents() { if (profileFd != null) { return profileFd.describeContents(); } else { return 0; } } @Override public void writeToParcel(Parcel out, int flags) { out.writeString(profileFile); if (profileFd != null) { out.writeInt(1); profileFd.writeToParcel(out, flags); } else { out.writeInt(0); } out.writeInt(samplingInterval); out.writeInt(autoStopProfiler ? 1 : 0); out.writeInt(streamingOutput ? 1 : 0); out.writeString(agent); out.writeBoolean(attachAgentDuringBind); out.writeInt(clockType); out.writeInt(profilerOutputVersion); } /** @hide */ public void dumpDebug(ProtoOutputStream proto, long fieldId) { final long token = proto.start(fieldId); proto.write(ProfilerInfoProto.PROFILE_FILE, profileFile); if (profileFd != null) { proto.write(ProfilerInfoProto.PROFILE_FD, profileFd.getFd()); } proto.write(ProfilerInfoProto.SAMPLING_INTERVAL, samplingInterval); proto.write(ProfilerInfoProto.AUTO_STOP_PROFILER, autoStopProfiler); proto.write(ProfilerInfoProto.STREAMING_OUTPUT, streamingOutput); proto.write(ProfilerInfoProto.AGENT, agent); proto.write(ProfilerInfoProto.CLOCK_TYPE, clockType); proto.write(ProfilerInfoProto.PROFILER_OUTPUT_VERSION, profilerOutputVersion); proto.end(token); } public static final @android.annotation.NonNull Parcelable.Creator CREATOR = new Parcelable.Creator() { @Override public ProfilerInfo createFromParcel(Parcel in) { return new ProfilerInfo(in); } @Override public ProfilerInfo[] newArray(int size) { return new ProfilerInfo[size]; } }; private ProfilerInfo(Parcel in) { profileFile = in.readString(); profileFd = in.readInt() != 0 ? ParcelFileDescriptor.CREATOR.createFromParcel(in) : null; samplingInterval = in.readInt(); autoStopProfiler = in.readInt() != 0; streamingOutput = in.readInt() != 0; agent = in.readString(); attachAgentDuringBind = in.readBoolean(); clockType = in.readInt(); profilerOutputVersion = in.readInt(); } @Override public boolean equals(@Nullable Object o) { if (this == o) { return true; } if (o == null || getClass() != o.getClass()) { return false; } final ProfilerInfo other = (ProfilerInfo) o; // TODO: Also check #profileFd for equality. return Objects.equals(profileFile, other.profileFile) && autoStopProfiler == other.autoStopProfiler && samplingInterval == other.samplingInterval && streamingOutput == other.streamingOutput && Objects.equals(agent, other.agent) && clockType == other.clockType && profilerOutputVersion == other.profilerOutputVersion; } @Override public int hashCode() { int result = 17; result = 31 * result + Objects.hashCode(profileFile); result = 31 * result + samplingInterval; result = 31 * result + (autoStopProfiler ? 1 : 0); result = 31 * result + (streamingOutput ? 1 : 0); result = 31 * result + Objects.hashCode(agent); result = 31 * result + clockType; result = 31 * result + profilerOutputVersion; return result; } }