1f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project/*
2f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * Copyright (C) 2007 The Android Open Source Project
3f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project *
4f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
5f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * you may not use this file except in compliance with the License.
6f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * You may obtain a copy of the License at
7f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project *
8f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
9f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project *
10f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * Unless required by applicable law or agreed to in writing, software
11f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
12f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * See the License for the specific language governing permissions and
14f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * limitations under the License.
15f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project */
16f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
17f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Projectpackage dalvik.system;
18f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
19f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Projectimport java.io.BufferedReader;
20f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Projectimport java.io.InputStreamReader;
21f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Projectimport java.io.IOException;
22f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Projectimport java.io.File;
23f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Projectimport java.io.FilenameFilter;
24f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
25f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project/**
26f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * Induces optimization/verification of a set of DEX files.
27f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project *
28979baaeae3122b0c7ad5b24e7842472127b8a967Jesse Wilson * @deprecated this is an internal Dalvik class that is not appropriate for
29979baaeae3122b0c7ad5b24e7842472127b8a967Jesse Wilson *      general use. It will be removed from the public API in a future release.
30f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project */
31f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Projectpublic class TouchDex {
32f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
33f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /**
34f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * Forks a process, makes sure the DEX files are prepared, and returns
35f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * when everything is finished.
36f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * <p>
37f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * The filenames must be the same as will be used when the files are
38f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * actually opened, because the dalvik-cache filename is based upon
39f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * this filename.  (The absolute path to the JAR/ZIP/APK should work.)
40f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     *
41f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * @param dexFiles a colon-separated list of DEX files.
42f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * @return zero on success
43f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     *
44f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * @cts What about error cases?
45f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     */
46f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    public static int start(String dexFiles) {
47f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        return trampoline(dexFiles, System.getProperty("java.boot.class.path"));
48f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
49f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
50f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /**
51f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * This calls fork() and then, in the child, calls cont(dexFiles).
52f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     *
53f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * @param dexFiles Colon-separated list of DEX files.
54f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * @return zero on success
55f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     */
56f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    native private static int trampoline(String dexFiles, String bcp);
57f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
58f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /**
59f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * The entry point for the child process. args[0] can be a colon-separated
60f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * path list, or "-" to read from stdin.
61f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * <p>
62f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * Alternatively, if we're invoked directly from the command line we
63f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * just start here (skipping the fork/exec stuff).
64f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     *
65f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * @param args command line args
66f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     */
67f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    public static void main(String[] args) {
68f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
69f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        if ("-".equals(args[0])) {
70f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            BufferedReader in = new BufferedReader(
71f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                    new InputStreamReader(System.in), 256);
72f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
73f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            String line;
74f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            try {
75f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                while ((line = in.readLine()) != null) {
76f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                    prepFiles(line);
77f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                }
78f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            } catch (IOException ex) {
79f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                throw new RuntimeException ("Error processing stdin");
80f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            }
81f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        } else {
82f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            prepFiles(args[0]);
83f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
84f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
85f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        System.out.println(" Prep complete");
86f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
87f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
88f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
89f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    private static String expandDirectories(String dexPath) {
90f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        String[] parts = dexPath.split(":");
91f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        StringBuilder outPath = new StringBuilder(dexPath.length());
92f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
93f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        // A filename filter accepting *.jar and *.apk
94f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        FilenameFilter filter = new FilenameFilter() {
95f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            public boolean accept(File dir, String name) {
96f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                return name.endsWith(".jar") || name.endsWith(".apk");
97f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            }
98f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        };
99f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
100f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        for (String part: parts) {
101f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            File f = new File(part);
102f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
103f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            if (f.isFile()) {
104f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                outPath.append(part);
105f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                outPath.append(':');
106f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            } else if (f.isDirectory()) {
107f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                String[] filenames = f.list(filter);
108f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
109f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                if (filenames == null) {
110f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                    System.err.println("I/O error with directory: " + part);
111f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                    continue;
112f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                }
113f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
114f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                for (String filename: filenames) {
115f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                    outPath.append(part);
116f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                    outPath.append(File.separatorChar);
117f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                    outPath.append(filename);
118f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                    outPath.append(':');
119f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                }
120f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            } else {
121f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                System.err.println("File not found: " + part);
122f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            }
123f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
124f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
125f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
126f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        return outPath.toString();
127f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
128f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
129f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    private static void prepFiles(String dexPath) {
130f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
131f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        System.out.println(" Prepping: " + dexPath);
132f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
133f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        TouchDexLoader loader
134f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                = new TouchDexLoader(expandDirectories(dexPath), null);
135f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
136f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        try {
137f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            /* By looking for a nonexistent class, we'll trick TouchDexLoader
138f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project             * into trying to load something from every file on dexPath,
139f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project             * optimizing all of them as a side-effect.
140f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project             *
141f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project             * The optimization happens implicitly in the VM the first time
142f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project             * someone tries to load a class from an unoptimized dex file.
143f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project             */
144f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            loader.loadClass("com.google.NonexistentClassNeverFound");
145f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            throw new RuntimeException("nonexistent class loaded?!");
146f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        } catch (ClassNotFoundException cnfe) {
147f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            //System.out.println("got expected dnfe");
148f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
149f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
150f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project}
151f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
152