1 /* |
|
2 * Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. |
|
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. |
|
4 * |
|
5 * This code is free software; you can redistribute it and/or modify it |
|
6 * under the terms of the GNU General Public License version 2 only, as |
|
7 * published by the Free Software Foundation. Oracle designates this |
|
8 * particular file as subject to the "Classpath" exception as provided |
|
9 * by Oracle in the LICENSE file that accompanied this code. |
|
10 * |
|
11 * This code is distributed in the hope that it will be useful, but WITHOUT |
|
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
|
13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License |
|
14 * version 2 for more details (a copy is included in the LICENSE file that |
|
15 * accompanied this code). |
|
16 * |
|
17 * You should have received a copy of the GNU General Public License version |
|
18 * 2 along with this work; if not, write to the Free Software Foundation, |
|
19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. |
|
20 * |
|
21 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
|
22 * or visit www.oracle.com if you need additional information or have any |
|
23 * questions. |
|
24 */ |
|
25 |
|
26 package jdk.internal.module; |
|
27 |
|
28 import java.io.ByteArrayInputStream; |
|
29 import java.io.IOException; |
|
30 import java.io.InputStream; |
|
31 import java.io.UncheckedIOException; |
|
32 import java.lang.module.ModuleDescriptor; |
|
33 import java.lang.module.ModuleFinder; |
|
34 import java.lang.module.ModuleReader; |
|
35 import java.lang.module.ModuleReference; |
|
36 import java.net.URI; |
|
37 import java.net.URLConnection; |
|
38 import java.nio.ByteBuffer; |
|
39 import java.util.ArrayDeque; |
|
40 import java.util.Collections; |
|
41 import java.util.Deque; |
|
42 import java.util.HashMap; |
|
43 import java.util.Iterator; |
|
44 import java.util.Map; |
|
45 import java.util.Map.Entry; |
|
46 import java.util.Objects; |
|
47 import java.util.Optional; |
|
48 import java.util.Set; |
|
49 import java.util.Spliterator; |
|
50 import java.util.function.Consumer; |
|
51 import java.util.function.Supplier; |
|
52 import java.util.stream.Stream; |
|
53 import java.util.stream.StreamSupport; |
|
54 |
|
55 import jdk.internal.jimage.ImageLocation; |
|
56 import jdk.internal.jimage.ImageReader; |
|
57 import jdk.internal.jimage.ImageReaderFactory; |
|
58 import jdk.internal.misc.JavaNetUriAccess; |
|
59 import jdk.internal.misc.SharedSecrets; |
|
60 import jdk.internal.module.ModuleHashes.HashSupplier; |
|
61 import jdk.internal.perf.PerfCounter; |
|
62 |
|
63 /** |
|
64 * A {@code ModuleFinder} that finds modules that are linked into the |
|
65 * run-time image. |
|
66 * |
|
67 * The modules linked into the run-time image are assumed to have the |
|
68 * Packages attribute. |
|
69 */ |
|
70 |
|
71 public class SystemModuleFinder implements ModuleFinder { |
|
72 |
|
73 private static final JavaNetUriAccess JNUA = SharedSecrets.getJavaNetUriAccess(); |
|
74 |
|
75 private static final PerfCounter initTime |
|
76 = PerfCounter.newPerfCounter("jdk.module.finder.jimage.initTime"); |
|
77 private static final PerfCounter moduleCount |
|
78 = PerfCounter.newPerfCounter("jdk.module.finder.jimage.modules"); |
|
79 private static final PerfCounter packageCount |
|
80 = PerfCounter.newPerfCounter("jdk.module.finder.jimage.packages"); |
|
81 private static final PerfCounter exportsCount |
|
82 = PerfCounter.newPerfCounter("jdk.module.finder.jimage.exports"); |
|
83 |
|
84 // singleton finder to find modules in the run-time images |
|
85 private static final SystemModuleFinder INSTANCE; |
|
86 |
|
87 public static SystemModuleFinder getInstance() { |
|
88 return INSTANCE; |
|
89 } |
|
90 |
|
91 /** |
|
92 * For now, the module references are created eagerly on the assumption |
|
93 * that service binding will require all modules to be located. |
|
94 */ |
|
95 static { |
|
96 long t0 = System.nanoTime(); |
|
97 |
|
98 INSTANCE = new SystemModuleFinder(); |
|
99 |
|
100 initTime.addElapsedTimeFrom(t0); |
|
101 } |
|
102 |
|
103 /** |
|
104 * Holder class for the ImageReader |
|
105 */ |
|
106 private static class SystemImage { |
|
107 static final ImageReader READER; |
|
108 static { |
|
109 long t0 = System.nanoTime(); |
|
110 READER = ImageReaderFactory.getImageReader(); |
|
111 initTime.addElapsedTimeFrom(t0); |
|
112 } |
|
113 |
|
114 static ImageReader reader() { |
|
115 return READER; |
|
116 } |
|
117 } |
|
118 |
|
119 private static boolean isFastPathSupported() { |
|
120 return SystemModules.MODULE_NAMES.length > 0; |
|
121 } |
|
122 |
|
123 private static String[] moduleNames() { |
|
124 if (isFastPathSupported()) |
|
125 // module names recorded at link time |
|
126 return SystemModules.MODULE_NAMES; |
|
127 |
|
128 // this happens when java.base is patched with java.base |
|
129 // from an exploded image |
|
130 return SystemImage.reader().getModuleNames(); |
|
131 } |
|
132 |
|
133 // the set of modules in the run-time image |
|
134 private final Set<ModuleReference> modules; |
|
135 |
|
136 // maps module name to module reference |
|
137 private final Map<String, ModuleReference> nameToModule; |
|
138 |
|
139 // module name to hashes |
|
140 private final Map<String, byte[]> hashes; |
|
141 |
|
142 private SystemModuleFinder() { |
|
143 String[] names = moduleNames(); |
|
144 int n = names.length; |
|
145 moduleCount.add(n); |
|
146 |
|
147 // fastpath is enabled by default. |
|
148 // It can be disabled for troubleshooting purpose. |
|
149 boolean disabled = |
|
150 System.getProperty("jdk.system.module.finder.disabledFastPath") != null; |
|
151 |
|
152 ModuleDescriptor[] descriptors; |
|
153 ModuleTarget[] targets; |
|
154 ModuleHashes[] recordedHashes; |
|
155 ModuleResolution[] moduleResolutions; |
|
156 |
|
157 // fast loading of ModuleDescriptor of system modules |
|
158 if (isFastPathSupported() && !disabled) { |
|
159 descriptors = SystemModules.descriptors(); |
|
160 targets = SystemModules.targets(); |
|
161 recordedHashes = SystemModules.hashes(); |
|
162 moduleResolutions = SystemModules.moduleResolutions(); |
|
163 } else { |
|
164 // if fast loading of ModuleDescriptors is disabled |
|
165 // fallback to read module-info.class |
|
166 descriptors = new ModuleDescriptor[n]; |
|
167 targets = new ModuleTarget[n]; |
|
168 recordedHashes = new ModuleHashes[n]; |
|
169 moduleResolutions = new ModuleResolution[n]; |
|
170 ImageReader imageReader = SystemImage.reader(); |
|
171 for (int i = 0; i < names.length; i++) { |
|
172 String mn = names[i]; |
|
173 ImageLocation loc = imageReader.findLocation(mn, "module-info.class"); |
|
174 ModuleInfo.Attributes attrs = |
|
175 ModuleInfo.read(imageReader.getResourceBuffer(loc), null); |
|
176 descriptors[i] = attrs.descriptor(); |
|
177 targets[i] = attrs.target(); |
|
178 recordedHashes[i] = attrs.recordedHashes(); |
|
179 moduleResolutions[i] = attrs.moduleResolution(); |
|
180 } |
|
181 } |
|
182 |
|
183 Map<String, byte[]> hashes = null; |
|
184 boolean secondSeen = false; |
|
185 // record the hashes to build HashSupplier |
|
186 for (ModuleHashes mh : recordedHashes) { |
|
187 if (mh != null) { |
|
188 // if only one module contain ModuleHashes, use it |
|
189 if (hashes == null) { |
|
190 hashes = mh.hashes(); |
|
191 } else { |
|
192 if (!secondSeen) { |
|
193 hashes = new HashMap<>(hashes); |
|
194 secondSeen = true; |
|
195 } |
|
196 hashes.putAll(mh.hashes()); |
|
197 } |
|
198 } |
|
199 } |
|
200 this.hashes = (hashes == null) ? Map.of() : hashes; |
|
201 |
|
202 ModuleReference[] mods = new ModuleReference[n]; |
|
203 |
|
204 @SuppressWarnings(value = {"rawtypes", "unchecked"}) |
|
205 Entry<String, ModuleReference>[] map |
|
206 = (Entry<String, ModuleReference>[])new Entry[n]; |
|
207 |
|
208 for (int i = 0; i < n; i++) { |
|
209 ModuleDescriptor md = descriptors[i]; |
|
210 |
|
211 // create the ModuleReference |
|
212 ModuleReference mref = toModuleReference(md, |
|
213 targets[i], |
|
214 recordedHashes[i], |
|
215 hashSupplier(names[i]), |
|
216 moduleResolutions[i]); |
|
217 mods[i] = mref; |
|
218 map[i] = Map.entry(names[i], mref); |
|
219 |
|
220 // counters |
|
221 packageCount.add(md.packages().size()); |
|
222 exportsCount.add(md.exports().size()); |
|
223 } |
|
224 |
|
225 modules = Set.of(mods); |
|
226 nameToModule = Map.ofEntries(map); |
|
227 } |
|
228 |
|
229 @Override |
|
230 public Optional<ModuleReference> find(String name) { |
|
231 Objects.requireNonNull(name); |
|
232 return Optional.ofNullable(nameToModule.get(name)); |
|
233 } |
|
234 |
|
235 @Override |
|
236 public Set<ModuleReference> findAll() { |
|
237 return modules; |
|
238 } |
|
239 |
|
240 private ModuleReference toModuleReference(ModuleDescriptor md, |
|
241 ModuleTarget target, |
|
242 ModuleHashes recordedHashes, |
|
243 HashSupplier hasher, |
|
244 ModuleResolution mres) { |
|
245 String mn = md.name(); |
|
246 URI uri = JNUA.create("jrt", "/".concat(mn)); |
|
247 |
|
248 Supplier<ModuleReader> readerSupplier = new Supplier<>() { |
|
249 @Override |
|
250 public ModuleReader get() { |
|
251 return new ImageModuleReader(mn, uri); |
|
252 } |
|
253 }; |
|
254 |
|
255 ModuleReference mref = new ModuleReferenceImpl(md, |
|
256 uri, |
|
257 readerSupplier, |
|
258 null, |
|
259 target, |
|
260 recordedHashes, |
|
261 hasher, |
|
262 mres); |
|
263 |
|
264 // may need a reference to a patched module if --patch-module specified |
|
265 mref = ModuleBootstrap.patcher().patchIfNeeded(mref); |
|
266 |
|
267 return mref; |
|
268 } |
|
269 |
|
270 private HashSupplier hashSupplier(String name) { |
|
271 if (!hashes.containsKey(name)) |
|
272 return null; |
|
273 |
|
274 return new HashSupplier() { |
|
275 @Override |
|
276 public byte[] generate(String algorithm) { |
|
277 return hashes.get(name); |
|
278 } |
|
279 }; |
|
280 } |
|
281 |
|
282 /** |
|
283 * A ModuleReader for reading resources from a module linked into the |
|
284 * run-time image. |
|
285 */ |
|
286 static class ImageModuleReader implements ModuleReader { |
|
287 private final String module; |
|
288 private volatile boolean closed; |
|
289 |
|
290 /** |
|
291 * If there is a security manager set then check permission to |
|
292 * connect to the run-time image. |
|
293 */ |
|
294 private static void checkPermissionToConnect(URI uri) { |
|
295 SecurityManager sm = System.getSecurityManager(); |
|
296 if (sm != null) { |
|
297 try { |
|
298 URLConnection uc = uri.toURL().openConnection(); |
|
299 sm.checkPermission(uc.getPermission()); |
|
300 } catch (IOException ioe) { |
|
301 throw new UncheckedIOException(ioe); |
|
302 } |
|
303 } |
|
304 } |
|
305 |
|
306 ImageModuleReader(String module, URI uri) { |
|
307 checkPermissionToConnect(uri); |
|
308 this.module = module; |
|
309 } |
|
310 |
|
311 /** |
|
312 * Returns the ImageLocation for the given resource, {@code null} |
|
313 * if not found. |
|
314 */ |
|
315 private ImageLocation findImageLocation(String name) throws IOException { |
|
316 Objects.requireNonNull(name); |
|
317 if (closed) |
|
318 throw new IOException("ModuleReader is closed"); |
|
319 ImageReader imageReader = SystemImage.reader(); |
|
320 if (imageReader != null) { |
|
321 return imageReader.findLocation(module, name); |
|
322 } else { |
|
323 // not an images build |
|
324 return null; |
|
325 } |
|
326 } |
|
327 |
|
328 @Override |
|
329 public Optional<URI> find(String name) throws IOException { |
|
330 ImageLocation location = findImageLocation(name); |
|
331 if (location != null) { |
|
332 URI u = URI.create("jrt:/" + module + "/" + name); |
|
333 return Optional.of(u); |
|
334 } else { |
|
335 return Optional.empty(); |
|
336 } |
|
337 } |
|
338 |
|
339 @Override |
|
340 public Optional<InputStream> open(String name) throws IOException { |
|
341 return read(name).map(this::toInputStream); |
|
342 } |
|
343 |
|
344 private InputStream toInputStream(ByteBuffer bb) { // ## -> ByteBuffer? |
|
345 try { |
|
346 int rem = bb.remaining(); |
|
347 byte[] bytes = new byte[rem]; |
|
348 bb.get(bytes); |
|
349 return new ByteArrayInputStream(bytes); |
|
350 } finally { |
|
351 release(bb); |
|
352 } |
|
353 } |
|
354 |
|
355 @Override |
|
356 public Optional<ByteBuffer> read(String name) throws IOException { |
|
357 ImageLocation location = findImageLocation(name); |
|
358 if (location != null) { |
|
359 return Optional.of(SystemImage.reader().getResourceBuffer(location)); |
|
360 } else { |
|
361 return Optional.empty(); |
|
362 } |
|
363 } |
|
364 |
|
365 @Override |
|
366 public void release(ByteBuffer bb) { |
|
367 Objects.requireNonNull(bb); |
|
368 ImageReader.releaseByteBuffer(bb); |
|
369 } |
|
370 |
|
371 @Override |
|
372 public Stream<String> list() throws IOException { |
|
373 if (closed) |
|
374 throw new IOException("ModuleReader is closed"); |
|
375 |
|
376 Spliterator<String> s = new ModuleContentSpliterator(module); |
|
377 return StreamSupport.stream(s, false); |
|
378 } |
|
379 |
|
380 @Override |
|
381 public void close() { |
|
382 // nothing else to do |
|
383 closed = true; |
|
384 } |
|
385 } |
|
386 |
|
387 /** |
|
388 * A Spliterator for traversing the resources of a module linked into the |
|
389 * run-time image. |
|
390 */ |
|
391 static class ModuleContentSpliterator implements Spliterator<String> { |
|
392 final String moduleRoot; |
|
393 final Deque<ImageReader.Node> stack; |
|
394 Iterator<ImageReader.Node> iterator; |
|
395 |
|
396 ModuleContentSpliterator(String module) throws IOException { |
|
397 moduleRoot = "/modules/" + module; |
|
398 stack = new ArrayDeque<>(); |
|
399 |
|
400 // push the root node to the stack to get started |
|
401 ImageReader.Node dir = SystemImage.reader().findNode(moduleRoot); |
|
402 if (dir == null || !dir.isDirectory()) |
|
403 throw new IOException(moduleRoot + " not a directory"); |
|
404 stack.push(dir); |
|
405 iterator = Collections.emptyIterator(); |
|
406 } |
|
407 |
|
408 /** |
|
409 * Returns the name of the next non-directory node or {@code null} if |
|
410 * there are no remaining nodes to visit. |
|
411 */ |
|
412 private String next() throws IOException { |
|
413 for (;;) { |
|
414 while (iterator.hasNext()) { |
|
415 ImageReader.Node node = iterator.next(); |
|
416 String name = node.getName(); |
|
417 if (node.isDirectory()) { |
|
418 // build node |
|
419 ImageReader.Node dir = SystemImage.reader().findNode(name); |
|
420 assert dir.isDirectory(); |
|
421 stack.push(dir); |
|
422 } else { |
|
423 // strip /modules/$MODULE/ prefix |
|
424 return name.substring(moduleRoot.length() + 1); |
|
425 } |
|
426 } |
|
427 |
|
428 if (stack.isEmpty()) { |
|
429 return null; |
|
430 } else { |
|
431 ImageReader.Node dir = stack.poll(); |
|
432 assert dir.isDirectory(); |
|
433 iterator = dir.getChildren().iterator(); |
|
434 } |
|
435 } |
|
436 } |
|
437 |
|
438 @Override |
|
439 public boolean tryAdvance(Consumer<? super String> action) { |
|
440 String next; |
|
441 try { |
|
442 next = next(); |
|
443 } catch (IOException ioe) { |
|
444 throw new UncheckedIOException(ioe); |
|
445 } |
|
446 if (next != null) { |
|
447 action.accept(next); |
|
448 return true; |
|
449 } else { |
|
450 return false; |
|
451 } |
|
452 } |
|
453 |
|
454 @Override |
|
455 public Spliterator<String> trySplit() { |
|
456 return null; |
|
457 } |
|
458 |
|
459 @Override |
|
460 public int characteristics() { |
|
461 return Spliterator.DISTINCT + Spliterator.NONNULL + Spliterator.IMMUTABLE; |
|
462 } |
|
463 |
|
464 @Override |
|
465 public long estimateSize() { |
|
466 return Long.MAX_VALUE; |
|
467 } |
|
468 } |
|
469 } |
|