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
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
|
/*
* Copyright (C) 2008 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.
*/
import java.io.File;
import java.io.IOException;
import java.lang.reflect.Constructor;
/**
* DexFile tests (Dalvik-specific).
*/
public class Main {
private static final String CLASS_PATH = System.getenv("DEX_LOCATION") + "/071-dexfile-ex.jar";
private static final String ODEX_DIR = System.getenv("DEX_LOCATION");
//private static final String ODEX_DIR = ".";
private static final String ODEX_ALT = "/tmp";
private static final String LIB_DIR = "/nowhere/nothing/";
/**
* Prep the environment then run the test.
*/
public static void main(String[] args) {
Process p;
try {
/*
* Create a sub-process to see if the ProcessManager wait
* interferes with the dexopt invocation wait.
*
* /dev/random never hits EOF, so we're sure that we'll still
* be waiting for the process to complete. On the device it
* stops pretty quickly (which means the child won't be
* spinning).
*/
ProcessBuilder pb = new ProcessBuilder("cat", "/dev/random");
p = pb.start();
} catch (IOException ioe) {
System.err.println("cmd failed: " + ioe.getMessage());
p = null;
}
try {
testDexClassLoader();
} finally {
// shouldn't be necessary, but it's good to be tidy
if (p != null) {
p.destroy();
}
// let the ProcessManager's daemon thread finish before we shut down
// (avoids the occasional segmentation fault)
try {
Thread.sleep(500);
} catch (Exception ex) {}
}
System.out.println("done");
}
/**
* Create a class loader, explicitly specifying the source DEX and
* the location for the optimized DEX.
*/
private static void testDexClassLoader() {
ClassLoader dexClassLoader = getDexClassLoader();
Class anotherClass;
try {
anotherClass = dexClassLoader.loadClass("Another");
} catch (ClassNotFoundException cnfe) {
throw new RuntimeException("Another?", cnfe);
}
Object another;
try {
another = anotherClass.newInstance();
} catch (IllegalAccessException ie) {
throw new RuntimeException("new another", ie);
} catch (InstantiationException ie) {
throw new RuntimeException("new another", ie);
}
// not expected to work; just exercises the call
dexClassLoader.getResource("nonexistent");
}
/*
* Create an instance of DexClassLoader. The test harness doesn't
* have visibility into dalvik.system.*, so we do this through
* reflection.
*/
private static ClassLoader getDexClassLoader() {
String odexDir;
if (false) {
String androidData = System.getenv("ANDROID_DATA");
if (androidData == null) {
androidData = "";
}
odexDir = androidData + "/" + ODEX_DIR;
}
File test = new File(ODEX_DIR);
if (test.isDirectory()) {
odexDir = ODEX_DIR;
} else {
odexDir = ODEX_ALT;
}
if (false) {
System.out.println("Output dir is " + odexDir);
}
ClassLoader myLoader = Main.class.getClassLoader();
Class dclClass;
try {
dclClass = myLoader.loadClass("dalvik.system.DexClassLoader");
} catch (ClassNotFoundException cnfe) {
throw new RuntimeException("dalvik.system.DexClassLoader not found", cnfe);
}
Constructor ctor;
try {
ctor = dclClass.getConstructor(String.class, String.class,
String.class, ClassLoader.class);
} catch (NoSuchMethodException nsme) {
throw new RuntimeException("DCL ctor", nsme);
}
// create an instance, using the path we found
Object dclObj;
try {
dclObj = ctor.newInstance(CLASS_PATH, odexDir, LIB_DIR, myLoader);
} catch (Exception ex) {
throw new RuntimeException("DCL newInstance", ex);
}
return (ClassLoader) dclObj;
}
}
|