1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
|
/*
* 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.
*/
package com.android.shell;
import android.annotation.NonNull;
import android.content.ContentProvider;
import android.content.ContentValues;
import android.database.Cursor;
import android.net.Uri;
import android.os.Binder;
import android.os.ParcelFileDescriptor;
import android.os.Process;
import java.io.File;
import java.io.FileNotFoundException;
/** ContentProvider to write and access heap dumps. */
public class HeapDumpProvider extends ContentProvider {
private static final String FILENAME_SUFFIX = "_javaheap.bin";
private static final Object sLock = new Object();
private File mRoot;
@Override
public boolean onCreate() {
synchronized (sLock) {
mRoot = new File(getContext().createCredentialProtectedStorageContext().getFilesDir(),
"heapdumps");
return mRoot.mkdir();
}
}
@Override
public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs,
String sortOrder) {
return null;
}
@Override
public String getType(Uri uri) {
return "application/octet-stream";
}
@Override
public Uri insert(Uri uri, ContentValues values) {
throw new UnsupportedOperationException("Insert not allowed.");
}
@Override
public int delete(Uri uri, String selection, String[] selectionArgs) {
String path = sanitizePath(uri.getEncodedPath());
String tag = Uri.decode(path);
return (new File(mRoot, tag)).delete() ? 1 : 0;
}
@Override
public int update(Uri uri, ContentValues values, String selection, String[] selectionArgs) {
throw new UnsupportedOperationException("Update not allowed.");
}
@Override
public ParcelFileDescriptor openFile(Uri uri, String mode) throws FileNotFoundException {
String path = sanitizePath(uri.getEncodedPath());
String tag = Uri.decode(path);
final int pMode;
if (Binder.getCallingUid() == Process.SYSTEM_UID) {
pMode = ParcelFileDescriptor.MODE_CREATE
| ParcelFileDescriptor.MODE_TRUNCATE
| ParcelFileDescriptor.MODE_WRITE_ONLY;
} else {
pMode = ParcelFileDescriptor.MODE_READ_ONLY;
}
synchronized (sLock) {
return ParcelFileDescriptor.open(new File(mRoot, tag), pMode);
}
}
@NonNull
static Uri makeUri(@NonNull String procName) {
return Uri.parse("content://com.android.shell.heapdump/" + procName + FILENAME_SUFFIX);
}
private String sanitizePath(String path) {
return path.replaceAll("[^a-zA-Z0-9_.]", "");
}
}
|