• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2017 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #ifndef ART_RUNTIME_CLASS_LOADER_CONTEXT_H_
18 #define ART_RUNTIME_CLASS_LOADER_CONTEXT_H_
19 
20 #include <string>
21 #include <vector>
22 #include <set>
23 
24 #include "arch/instruction_set.h"
25 #include "base/dchecked_vector.h"
26 #include "dex/dex_file.h"
27 #include "handle_scope.h"
28 #include "mirror/class_loader.h"
29 #include "oat_file.h"
30 #include "scoped_thread_state_change.h"
31 
32 namespace art {
33 
34 class DexFile;
35 class OatFile;
36 
37 // Utility class which holds the class loader context used during compilation/verification.
38 class ClassLoaderContext {
39  public:
40   enum class VerificationResult {
41     kVerifies,
42     kMismatch,
43   };
44 
45   enum ClassLoaderType {
46     kInvalidClassLoader = 0,
47     kPathClassLoader = 1,
48     kDelegateLastClassLoader = 2,
49     kInMemoryDexClassLoader = 3
50   };
51 
52   // Special encoding used to denote a foreign ClassLoader was found when trying to encode class
53   // loader contexts for each classpath element in a ClassLoader. See
54   // EncodeClassPathContextsForClassLoader. Keep in sync with PackageDexUsage in the framework
55   // (frameworks/base/services/core/java/com/android/server/pm/dex/PackageDexUsage.java) and
56   // DexUseManager in ART Services
57   // (art/libartservice/service/java/com/android/server/art/DexUseManager.java).
58   static constexpr const char* kUnsupportedClassLoaderContextEncoding =
59       "=UnsupportedClassLoaderContext=";
60 
61   ~ClassLoaderContext();
62 
63   // Opens requested class path files and appends them to ClassLoaderInfo::opened_dex_files.
64   // If the dex files have been stripped, the method opens them from their oat files which are added
65   // to ClassLoaderInfo::opened_oat_files. The 'classpath_dir' argument specifies the directory to
66   // use for the relative class paths.
67   // Returns true if all dex files where successfully opened.
68   // It may be called only once per ClassLoaderContext. Subsequent calls will return the same
69   // result without doing anything.
70   // If `context_fds` is an empty vector, files will be opened using the class path locations as
71   // filenames. Otherwise `context_fds` is expected to contain file descriptors to class path dex
72   // files, following the order of dex file locations in a flattened class loader context. If their
73   // number (size of `context_fds`) does not match the number of dex files, OpenDexFiles will fail.
74   //
75   // This will replace the class path locations with the locations of the opened dex files.
76   // (Note that one dex file can contain multidexes. Each multidex will be added to the classpath
77   // separately.)
78   //
79   // only_read_checksums controls whether or not we only read the dex locations and the checksums
80   // from the apk instead of fully opening the dex files.
81   //
82   // This method is not thread safe.
83   //
84   // Note that a "false" return could mean that either an apk/jar contained no dex files or
85   // that we hit a I/O or checksum mismatch error.
86   // TODO(calin): Currently there's no easy way to tell the difference.
87   //
88   // TODO(calin): we're forced to complicate the flow in this class with a different
89   // OpenDexFiles step because the current dex2oat flow requires the dex files be opened before
90   // the class loader is created. Consider reworking the dex2oat part.
91   bool OpenDexFiles(const std::string& classpath_dir = "",
92                     const std::vector<int>& context_fds = std::vector<int>(),
93                     bool only_read_checksums = false);
94 
95   // Remove the specified compilation sources from all classpaths present in this context.
96   // Should only be called before the first call to OpenDexFiles().
97   bool RemoveLocationsFromClassPaths(const dchecked_vector<std::string>& compilation_sources);
98 
99   // Creates the entire class loader hierarchy according to the current context.
100   // Returns the first class loader from the chain.
101   //
102   // For example: if the context was built from the spec
103   // "ClassLoaderType1[ClasspathElem1:ClasspathElem2...];ClassLoaderType2[...]..."
104   // the method returns the class loader correponding to ClassLoader1. The parent chain will be
105   // ClassLoader1 --> ClassLoader2 --> ... --> BootClassLoader.
106   //
107   // The compilation sources are appended to the classpath of the first class loader (in the above
108   // example ClassLoader1).
109   //
110   // If the context is empty, this method only creates a single PathClassLoader with the
111   // given compilation_sources.
112   //
113   // Shared libraries found in the chain will be canonicalized based on the dex files they
114   // contain.
115   //
116   // Implementation notes:
117   //   1) the objects are not completely set up. Do not use this outside of tests and the compiler.
118   //   2) should only be called before the first call to OpenDexFiles().
119   jobject CreateClassLoader(const std::vector<const DexFile*>& compilation_sources) const;
120 
121   // Encodes the context as a string suitable to be added in oat files.
122   // (so that it can be read and verified at runtime against the actual class
123   // loader hierarchy).
124   // Should only be called if OpenDexFiles() returned true.
125   // If stored context is non-null, the stored names are overwritten by the class path from the
126   // stored context.
127   // E.g. if the context is PCL[a.dex:b.dex] this will return
128   // "PCL[a.dex*a_checksum*b.dex*a_checksum]".
129   std::string EncodeContextForOatFile(const std::string& base_dir,
130                                       ClassLoaderContext* stored_context = nullptr) const;
131 
132   // Encodes the context as a string suitable to be passed to dex2oat.
133   // This is the same as EncodeContextForOatFile but without adding the checksums
134   // and only adding each dex files once (no multidex).
135   // Should only be called if OpenDexFiles() returned true.
136   std::string EncodeContextForDex2oat(const std::string& base_dir) const;
137 
138   // Encodes the contexts for each of the classpath elements in the child-most
139   // classloader. Under the hood EncodeContextForDex2oat is used, so no checksums
140   // will be encoded.
141   // Should only be called if the dex files are opened (either via OpenDexFiles() or by creating the
142   // context from a live class loader).
143   // Notably, for each classpath element the encoded classloader context will contain only the
144   // elements that appear before it in the containing classloader. E.g. if `this` contains
145   // (from child to parent):
146   //
147   // PathClassLoader { multidex.apk!classes.dex, multidex.apk!classes2.dex, foo.dex, bar.dex } ->
148   //    PathClassLoader { baz.dex } -> BootClassLoader
149   //
150   // then the return value will look like:
151   //
152   // `{ "multidex.apk": "PCL[];PCL[baz.dex]",
153   //    "foo.dex"     : "PCL[multidex.apk];PCL[baz.dex]",
154   //    "bar.dex"     : "PCL[multidex.apk:foo.dex];PCL[baz.dex]" }`
155   std::map<std::string, std::string> EncodeClassPathContexts(const std::string& base_dir) const;
156 
157   // Flattens the opened dex files into the given vector.
158   // Should only be called if OpenDexFiles() returned true.
159   std::vector<const DexFile*> FlattenOpenedDexFiles() const;
160 
161   // Return a list of dex file locations from this class loader context after flattening.
162   std::vector<std::string> FlattenDexPaths() const;
163 
164   // Verifies that the current context is identical to the context encoded as `context_spec`.
165   // Identical means:
166   //    - the number and type of the class loaders from the chain matches
167   //    - the class loader from the same position have the same classpath
168   //      (the order and checksum of the dex files matches)
169   // This should be called after OpenDexFiles() with only_read_checksums=true. There's no
170   // need to fully open the dex files if the only thing that needs to be done is to verify
171   // the context.
172   //
173   // Names are only verified if verify_names is true.
174   // Checksums are only verified if verify_checksums is true.
175   VerificationResult VerifyClassLoaderContextMatch(const std::string& context_spec,
176                                                    bool verify_names = true,
177                                                    bool verify_checksums = true) const;
178 
179   // Checks if any of the given dex files is already loaded in the current class loader context.
180   // It only checks the first class loader.
181   // Returns the list of duplicate dex files (empty if there are no duplicates).
182   std::set<const DexFile*> CheckForDuplicateDexFiles(
183       const std::vector<const DexFile*>& dex_files);
184 
185   // Creates the class loader context from the given string.
186   // The format: ClassLoaderType1[ClasspathElem1:ClasspathElem2...];ClassLoaderType2[...]...
187   // ClassLoaderType is either "PCL" (PathClassLoader) or "DLC" (DelegateLastClassLoader).
188   // ClasspathElem is the path of dex/jar/apk file.
189   //
190   // The spec represents a class loader chain with the natural interpretation:
191   // ClassLoader1 has ClassLoader2 as parent which has ClassLoader3 as a parent and so on.
192   // The last class loader is assumed to have the BootClassLoader as a parent.
193   //
194   // Note that we allowed class loaders with an empty class path in order to support a custom
195   // class loader for the source dex files.
196   static std::unique_ptr<ClassLoaderContext> Create(const std::string& spec);
197 
198   // Creates a context for the given class_loader and dex_elements.
199   // The method will walk the parent chain starting from `class_loader` and add their dex files
200   // to the current class loaders chain. The `dex_elements` will be added at the end of the
201   // classpath belonging to the `class_loader` argument.
202   // The ownership of the opened dex files will be retained by the given `class_loader`.
203   // If there are errors in processing the class loader chain (e.g. unsupported elements) the
204   // method returns null.
205   static std::unique_ptr<ClassLoaderContext> CreateContextForClassLoader(jobject class_loader,
206                                                                          jobjectArray dex_elements);
207 
208   // Returns the default class loader context to be used when none is specified.
209   // This will return a context with a single and empty PathClassLoader.
210   static std::unique_ptr<ClassLoaderContext> Default();
211 
212   // Encodes the contexts for each of the classpath elements in `class_loader`. See
213   // ClassLoaderContext::EncodeClassPathContexts for more information about the return value.
214   //
215   // If `class_loader` does not derive from BaseDexClassLoader then an empty map is returned.
216   // Otherwise if a foreign ClassLoader is found in the class loader chain then the results values
217   // will all be ClassLoaderContext::kUnsupportedClassLoaderContextEncoding.
218   static std::map<std::string, std::string> EncodeClassPathContextsForClassLoader(
219       jobject class_loader);
220 
221   // Returns whether `encoded_class_loader_context` is a valid encoded ClassLoaderContext or
222   // EncodedUnsupportedClassLoaderContext.
223   static bool IsValidEncoding(const std::string& possible_encoded_class_loader_context);
224 
225   struct ClassLoaderInfo {
226     // The type of this class loader.
227     ClassLoaderType type;
228     // Shared libraries this context has.
229     std::vector<std::unique_ptr<ClassLoaderInfo>> shared_libraries;
230     // Shared libraries that will be loaded after apks code that this context has.
231     std::vector<std::unique_ptr<ClassLoaderInfo>> shared_libraries_after;
232     // The list of class path elements that this loader loads.
233     // Note that this list may contain relative paths.
234     std::vector<std::string> classpath;
235     // Original opened class path (ignoring multidex).
236     std::vector<std::string> original_classpath;
237     // The list of class path elements checksums.
238     // May be empty if the checksums are not given when the context is created.
239     std::vector<uint32_t> checksums;
240     // After OpenDexFiles is called this holds the opened dex files.
241     std::vector<std::unique_ptr<const DexFile>> opened_dex_files;
242     // After OpenDexFiles, in case some of the dex files were opened from their oat files
243     // this holds the list of opened oat files.
244     std::vector<std::unique_ptr<OatFile>> opened_oat_files;
245     // The parent class loader.
246     std::unique_ptr<ClassLoaderInfo> parent;
247 
ClassLoaderInfoClassLoaderInfo248     explicit ClassLoaderInfo(ClassLoaderType cl_type) : type(cl_type) {}
249   };
250 
251  private:
252   // Creates an empty context (with no class loaders).
253   ClassLoaderContext();
254 
255   // Get the parent of the class loader chain at depth `index`.
GetParent(size_t index)256   ClassLoaderInfo* GetParent(size_t index) const {
257     ClassLoaderInfo* result = class_loader_chain_.get();
258     while ((result != nullptr) && (index-- != 0)) {
259       result = result->parent.get();
260     }
261     return result;
262   }
263 
GetParentChainSize()264   size_t GetParentChainSize() const {
265     size_t result = 0;
266     ClassLoaderInfo* info = class_loader_chain_.get();
267     while (info != nullptr) {
268       ++result;
269       info = info->parent.get();
270     }
271     return result;
272   }
273 
274   // Constructs an empty context.
275   // `owns_the_dex_files` specifies whether or not the context will own the opened dex files
276   // present in the class loader chain. If `owns_the_dex_files` is true then OpenDexFiles cannot
277   // be called on this context (dex_files_open_attempted_ and dex_files_open_result_ will be set
278   // to true as well)
279   explicit ClassLoaderContext(bool owns_the_dex_files);
280 
281   // Reads the class loader spec in place and returns true if the spec is valid and the
282   // compilation context was constructed.
283   bool Parse(const std::string& spec, bool parse_checksums = false);
284   ClassLoaderInfo* ParseInternal(const std::string& spec, bool parse_checksums);
285 
286   // Attempts to parse a single class loader spec.
287   // Returns the ClassLoaderInfo abstraction for this spec, or null if it cannot be parsed.
288   std::unique_ptr<ClassLoaderInfo> ParseClassLoaderSpec(
289       const std::string& class_loader_spec,
290       bool parse_checksums = false);
291 
292   // CHECKs that the dex files were opened (OpenDexFiles was called and set dex_files_open_result_
293   // to true). Aborts if not. The `calling_method` is used in the log message to identify the source
294   // of the call.
295   void CheckDexFilesOpened(const std::string& calling_method) const;
296 
297   // Creates the `ClassLoaderInfo` representing`class_loader` and attach it to `this`.
298   // The dex file present in `dex_elements` array (if not null) will be added at the end of
299   // the classpath.
300   bool CreateInfoFromClassLoader(ScopedObjectAccessAlreadyRunnable& soa,
301                                  Handle<mirror::ClassLoader> class_loader,
302                                  Handle<mirror::ObjectArray<mirror::Object>> dex_elements,
303                                  ClassLoaderInfo* child_info,
304                                  bool is_shared_library,
305                                  bool is_after)
306     REQUIRES_SHARED(Locks::mutator_lock_);
307 
308   // Encodes the context as a string suitable to be passed to dex2oat or to be added to the
309   // oat file as the class path key.
310   // If for_dex2oat is true, the encoding adds each file once (i.e. it does not add multidex
311   // location). Otherwise, for oat files, the encoding adds all the dex files (including multidex)
312   // together with their checksums.
313   // Should only be called if OpenDexFiles() returned true.
314   std::string EncodeContext(const std::string& base_dir,
315                             bool for_dex2oat,
316                             ClassLoaderContext* stored_context) const;
317 
318   // Internal version of `EncodeContext`, which will be called recursively
319   // on the parent and shared libraries.
320   void EncodeContextInternal(const ClassLoaderInfo& info,
321                              const std::string& base_dir,
322                              bool for_dex2oat,
323                              ClassLoaderInfo* stored_info,
324                              std::ostringstream& out) const;
325 
326   // Encodes e.g. PCL[foo.dex:bar.dex]
327   void EncodeClassPath(const std::string& base_dir,
328                        const std::vector<std::string>& dex_locations,
329                        const std::vector<uint32_t>& checksums,
330                        ClassLoaderType type,
331                        std::ostringstream& out) const;
332 
333   // Encodes the shared libraries classloaders and the parent classloader if
334   // either are present in info, e.g. {PCL[foo.dex]#PCL[bar.dex]};PCL[baz.dex]
335   void EncodeSharedLibAndParent(const ClassLoaderInfo& info,
336                                 const std::string& base_dir,
337                                 bool for_dex2oat,
338                                 ClassLoaderInfo* stored_info,
339                                 std::ostringstream& out) const;
340 
341   bool ClassLoaderInfoMatch(const ClassLoaderInfo& info,
342                             const ClassLoaderInfo& expected_info,
343                             const std::string& context_spec,
344                             bool verify_names,
345                             bool verify_checksums) const;
346 
347   // Extracts the class loader type from the given spec.
348   // Return ClassLoaderContext::kInvalidClassLoader if the class loader type is not
349   // recognized.
350   static ClassLoaderType ExtractClassLoaderType(const std::string& class_loader_spec);
351 
352   // Returns the string representation of the class loader type.
353   // The returned format can be used when parsing a context spec.
354   static const char* GetClassLoaderTypeName(ClassLoaderType type);
355 
356   // Encodes the state of processing the dex files associated with the context.
357   enum ContextDexFilesState {
358     // The dex files are not opened.
359     kDexFilesNotOpened = 1,
360     // The dex checksums/locations were read from the apk/dex but the dex files were not opened.
361     kDexFilesChecksumsRead = 2,
362     // The dex files are opened (either because we called OpenDexFiles, or we used a class loader
363     // to create the context). This implies kDexFilesChecksumsRead.
364     kDexFilesOpened = 3,
365     // We failed to open the dex files or read the checksums.
366     kDexFilesOpenFailed = 4
367   };
368 
369   // The class loader chain.
370   std::unique_ptr<ClassLoaderInfo> class_loader_chain_;
371 
372   // The opening state of the dex files.
373   ContextDexFilesState dex_files_state_;
374 
375   // Whether or not the context owns the opened dex and oat files.
376   // If true, the opened dex files will be de-allocated when the context is destructed.
377   // If false, the objects will continue to be alive.
378   // Note that for convenience the the opened dex/oat files are stored as unique pointers
379   // which will release their ownership in the destructor based on this flag.
380   const bool owns_the_dex_files_;
381 
382   friend class ClassLoaderContextTest;
383 
384   DISALLOW_COPY_AND_ASSIGN(ClassLoaderContext);
385 };
386 
387 }  // namespace art
388 #endif  // ART_RUNTIME_CLASS_LOADER_CONTEXT_H_
389