langtools/src/jdk.jdeps/share/classes/com/sun/tools/jdeps/ModuleAnalyzer.java
author mchung
Wed, 15 Feb 2017 16:18:18 -0800
changeset 43873 705d732d3715
parent 43767 9cff98a149cb
permissions -rw-r--r--
8173374: Update GenGraphs tool to generate dot graph with requires transitive edges Reviewed-by: dfuchs, redestad
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
     1
/*
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
     2
 * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
     4
 *
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    10
 *
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    15
 * accompanied this code).
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    16
 *
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    20
 *
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    23
 * questions.
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    24
 */
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    25
package com.sun.tools.jdeps;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    26
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    27
import static com.sun.tools.jdeps.Graph.*;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    28
import static com.sun.tools.jdeps.JdepsFilter.DEFAULT_FILTER;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    29
import static com.sun.tools.jdeps.Module.*;
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    30
import static java.lang.module.ModuleDescriptor.Requires.Modifier.*;
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    31
import static java.util.stream.Collectors.*;
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    32
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    33
import com.sun.tools.classfile.Dependency;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    34
import com.sun.tools.jdeps.JdepsTask.BadArgs;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    35
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    36
import java.io.IOException;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    37
import java.io.OutputStream;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    38
import java.io.PrintWriter;
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    39
import java.lang.module.ModuleDescriptor;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    40
import java.nio.file.Files;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    41
import java.nio.file.Path;
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    42
import java.util.Collections;
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    43
import java.util.Comparator;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    44
import java.util.HashMap;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    45
import java.util.HashSet;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    46
import java.util.Map;
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    47
import java.util.Optional;
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    48
import java.util.Set;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    49
import java.util.function.Function;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    50
import java.util.stream.Collectors;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    51
import java.util.stream.Stream;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    52
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    53
/**
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    54
 * Analyze module dependences and compare with module descriptor.
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    55
 * Also identify any qualified exports not used by the target module.
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    56
 */
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    57
public class ModuleAnalyzer {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    58
    private static final String JAVA_BASE = "java.base";
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    59
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    60
    private final JdepsConfiguration configuration;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    61
    private final PrintWriter log;
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    62
    private final DependencyFinder dependencyFinder;
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    63
    private final Map<Module, ModuleDeps> modules;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    64
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    65
    public ModuleAnalyzer(JdepsConfiguration config,
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    66
                          PrintWriter log,
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    67
                          Set<String> names) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    68
        this.configuration = config;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    69
        this.log = log;
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    70
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    71
        this.dependencyFinder = new DependencyFinder(config, DEFAULT_FILTER);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    72
        if (names.isEmpty()) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    73
            this.modules = configuration.rootModules().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    74
                .collect(toMap(Function.identity(), ModuleDeps::new));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    75
        } else {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    76
            this.modules = names.stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    77
                .map(configuration::findModule)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    78
                .flatMap(Optional::stream)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    79
                .collect(toMap(Function.identity(), ModuleDeps::new));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    80
        }
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    81
    }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    82
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    83
    public boolean run() throws IOException {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    84
        try {
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
    85
            // compute "requires transitive" dependences
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
    86
            modules.values().forEach(ModuleDeps::computeRequiresTransitive);
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    87
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    88
            modules.values().forEach(md -> {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    89
                // compute "requires" dependences
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    90
                md.computeRequires();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    91
                // apply transitive reduction and reports recommended requires.
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    92
                md.analyzeDeps();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    93
            });
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    94
        } finally {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    95
            dependencyFinder.shutdown();
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    96
        }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    97
        return true;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    98
    }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    99
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   100
    class ModuleDeps {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   101
        final Module root;
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   102
        Set<Module> requiresTransitive;
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   103
        Set<Module> requires;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   104
        Map<String, Set<String>> unusedQualifiedExports;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   105
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   106
        ModuleDeps(Module root) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   107
            this.root = root;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   108
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   109
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   110
        /**
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   111
         * Compute 'requires transitive' dependences by analyzing API dependencies
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   112
         */
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   113
        private void computeRequiresTransitive() {
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   114
            // record requires transitive
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   115
            this.requiresTransitive = computeRequires(true)
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   116
                .filter(m -> !m.name().equals(JAVA_BASE))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   117
                .collect(toSet());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   118
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   119
            trace("requires transitive: %s%n", requiresTransitive);
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   120
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   121
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   122
        private void computeRequires() {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   123
            this.requires = computeRequires(false).collect(toSet());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   124
            trace("requires: %s%n", requires);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   125
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   126
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   127
        private Stream<Module> computeRequires(boolean apionly) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   128
            // analyze all classes
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   129
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   130
            if (apionly) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   131
                dependencyFinder.parseExportedAPIs(Stream.of(root));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   132
            } else {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   133
                dependencyFinder.parse(Stream.of(root));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   134
            }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   135
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   136
            // find the modules of all the dependencies found
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   137
            return dependencyFinder.getDependences(root)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   138
                        .map(Archive::getModule);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   139
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   140
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   141
        ModuleDescriptor descriptor() {
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   142
            return descriptor(requiresTransitive, requires);
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   143
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   144
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   145
        private ModuleDescriptor descriptor(Set<Module> requiresTransitive,
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   146
                                            Set<Module> requires) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   147
43767
9cff98a149cb 8173393: Module system implementation refresh (2/2017)
alanb
parents: 42407
diff changeset
   148
            ModuleDescriptor.Builder builder = ModuleDescriptor.newModule(root.name());
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   149
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   150
            if (!root.name().equals(JAVA_BASE))
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   151
                builder.requires(Set.of(MANDATED), JAVA_BASE);
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   152
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   153
            requiresTransitive.stream()
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   154
                .filter(m -> !m.name().equals(JAVA_BASE))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   155
                .map(Module::name)
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   156
                .forEach(mn -> builder.requires(Set.of(TRANSITIVE), mn));
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   157
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   158
            requires.stream()
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   159
                .filter(m -> !requiresTransitive.contains(m))
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   160
                .filter(m -> !m.name().equals(JAVA_BASE))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   161
                .map(Module::name)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   162
                .forEach(mn -> builder.requires(mn));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   163
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   164
            return builder.build();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   165
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   166
41860
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   167
        private Graph<Module> buildReducedGraph() {
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   168
            ModuleGraphBuilder rpBuilder = new ModuleGraphBuilder(configuration);
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   169
            rpBuilder.addModule(root);
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   170
            requiresTransitive.stream()
41860
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   171
                          .forEach(m -> rpBuilder.addEdge(root, m));
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   172
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   173
            // requires transitive graph
41860
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   174
            Graph<Module> rbg = rpBuilder.build().reduce();
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   175
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   176
            ModuleGraphBuilder gb = new ModuleGraphBuilder(configuration);
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   177
            gb.addModule(root);
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   178
            requires.stream()
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   179
                    .forEach(m -> gb.addEdge(root, m));
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   180
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   181
            // transitive reduction
41860
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   182
            Graph<Module> newGraph = gb.buildGraph().reduce(rbg);
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   183
            if (DEBUG) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   184
                System.err.println("after transitive reduction: ");
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   185
                newGraph.printGraph(log);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   186
            }
41860
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   187
            return newGraph;
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   188
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   189
41860
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   190
        /**
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   191
         * Apply the transitive reduction on the module graph
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   192
         * and returns the corresponding ModuleDescriptor
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   193
         */
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   194
        ModuleDescriptor reduced() {
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   195
            Graph<Module> g = buildReducedGraph();
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   196
            return descriptor(requiresTransitive, g.adjacentNodes(root));
41860
906670ff49c7 8167057: jdeps option to list modules and internal APIs for @modules for test dev
mchung
parents: 40308
diff changeset
   197
        }
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   198
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   199
        /**
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   200
         * Apply transitive reduction on the resulting graph and reports
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   201
         * recommended requires.
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   202
         */
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   203
        private void analyzeDeps() {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   204
            printModuleDescriptor(log, root);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   205
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   206
            ModuleDescriptor analyzedDescriptor = descriptor();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   207
            if (!matches(root.descriptor(), analyzedDescriptor)) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   208
                log.format("  [Suggested module descriptor for %s]%n", root.name());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   209
                analyzedDescriptor.requires()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   210
                    .stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   211
                    .sorted(Comparator.comparing(ModuleDescriptor.Requires::name))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   212
                    .forEach(req -> log.format("    requires %s;%n", req));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   213
            }
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   214
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   215
            ModuleDescriptor reduced = reduced();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   216
            if (!matches(root.descriptor(), reduced)) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   217
                log.format("  [Transitive reduced graph for %s]%n", root.name());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   218
                reduced.requires()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   219
                    .stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   220
                    .sorted(Comparator.comparing(ModuleDescriptor.Requires::name))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   221
                    .forEach(req -> log.format("    requires %s;%n", req));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   222
            }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   223
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   224
            checkQualifiedExports();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   225
            log.println();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   226
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   227
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   228
        private void checkQualifiedExports() {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   229
            // detect any qualified exports not used by the target module
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   230
            unusedQualifiedExports = unusedQualifiedExports();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   231
            if (!unusedQualifiedExports.isEmpty())
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   232
                log.format("  [Unused qualified exports in %s]%n", root.name());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   233
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   234
            unusedQualifiedExports.keySet().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   235
                .sorted()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   236
                .forEach(pn -> log.format("    exports %s to %s%n", pn,
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   237
                    unusedQualifiedExports.get(pn).stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   238
                        .sorted()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   239
                        .collect(joining(","))));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   240
        }
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   241
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   242
        private void printModuleDescriptor(PrintWriter out, Module module) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   243
            ModuleDescriptor descriptor = module.descriptor();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   244
            out.format("%s (%s)%n", descriptor.name(), module.location());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   245
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   246
            if (descriptor.name().equals(JAVA_BASE))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   247
                return;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   248
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   249
            out.println("  [Module descriptor]");
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   250
            descriptor.requires()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   251
                .stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   252
                .sorted(Comparator.comparing(ModuleDescriptor.Requires::name))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   253
                .forEach(req -> out.format("    requires %s;%n", req));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   254
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   255
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   256
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   257
        /**
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   258
         * Detects any qualified exports not used by the target module.
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   259
         */
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   260
        private Map<String, Set<String>> unusedQualifiedExports() {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   261
            Map<String, Set<String>> unused = new HashMap<>();
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   262
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   263
            // build the qualified exports map
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   264
            Map<String, Set<String>> qualifiedExports =
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   265
                root.exports().entrySet().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   266
                    .filter(e -> !e.getValue().isEmpty())
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   267
                    .map(Map.Entry::getKey)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   268
                    .collect(toMap(Function.identity(), _k -> new HashSet<>()));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   269
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   270
            Set<Module> mods = new HashSet<>();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   271
            root.exports().values()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   272
                .stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   273
                .flatMap(Set::stream)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   274
                .forEach(target -> configuration.findModule(target)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   275
                    .ifPresentOrElse(mods::add,
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   276
                        () -> log.format("Warning: %s not found%n", target))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   277
                );
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   278
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   279
            // parse all target modules
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   280
            dependencyFinder.parse(mods.stream());
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   281
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   282
            // adds to the qualified exports map if a module references it
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   283
            mods.stream().forEach(m ->
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   284
                m.getDependencies()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   285
                    .map(Dependency.Location::getPackageName)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   286
                    .filter(qualifiedExports::containsKey)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   287
                    .forEach(pn -> qualifiedExports.get(pn).add(m.name())));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   288
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   289
            // compare with the exports from ModuleDescriptor
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   290
            Set<String> staleQualifiedExports =
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   291
                qualifiedExports.keySet().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   292
                    .filter(pn -> !qualifiedExports.get(pn).equals(root.exports().get(pn)))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   293
                    .collect(toSet());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   294
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   295
            if (!staleQualifiedExports.isEmpty()) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   296
                for (String pn : staleQualifiedExports) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   297
                    Set<String> targets = new HashSet<>(root.exports().get(pn));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   298
                    targets.removeAll(qualifiedExports.get(pn));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   299
                    unused.put(pn, targets);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   300
                }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   301
            }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   302
            return unused;
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   303
        }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   304
    }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   305
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   306
    private boolean matches(ModuleDescriptor md, ModuleDescriptor other) {
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   307
        // build requires transitive from ModuleDescriptor
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   308
        Set<ModuleDescriptor.Requires> reqTransitive = md.requires().stream()
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   309
            .filter(req -> req.modifiers().contains(TRANSITIVE))
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   310
            .collect(toSet());
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   311
        Set<ModuleDescriptor.Requires> otherReqTransitive = other.requires().stream()
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   312
            .filter(req -> req.modifiers().contains(TRANSITIVE))
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   313
            .collect(toSet());
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   314
42407
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   315
        if (!reqTransitive.equals(otherReqTransitive)) {
f3702cff2933 8169069: Module system implementation refresh (11/2016)
alanb
parents: 41997
diff changeset
   316
            trace("mismatch requires transitive: %s%n", reqTransitive);
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   317
            return false;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   318
        }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   319
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   320
        Set<ModuleDescriptor.Requires> unused = md.requires().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   321
            .filter(req -> !other.requires().contains(req))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   322
            .collect(Collectors.toSet());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   323
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   324
        if (!unused.isEmpty()) {
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   325
            trace("mismatch requires: %s%n", unused);
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   326
            return false;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   327
        }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   328
        return true;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   329
    }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   330
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   331
    // ---- for testing purpose
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   332
    public ModuleDescriptor[] descriptors(String name) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   333
        ModuleDeps moduleDeps = modules.keySet().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   334
            .filter(m -> m.name().equals(name))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   335
            .map(modules::get)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   336
            .findFirst().get();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   337
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   338
        ModuleDescriptor[] descriptors = new ModuleDescriptor[3];
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   339
        descriptors[0] = moduleDeps.root.descriptor();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   340
        descriptors[1] = moduleDeps.descriptor();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   341
        descriptors[2] = moduleDeps.reduced();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   342
        return descriptors;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   343
    }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   344
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   345
    public Map<String, Set<String>> unusedQualifiedExports(String name) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   346
        ModuleDeps moduleDeps = modules.keySet().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   347
            .filter(m -> m.name().equals(name))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   348
            .map(modules::get)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   349
            .findFirst().get();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   350
        return moduleDeps.unusedQualifiedExports;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   351
    }
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   352
}