langtools/src/jdk.jdeps/share/classes/com/sun/tools/jdeps/ModuleAnalyzer.java
author jjg
Wed, 10 Aug 2016 15:47:46 -0700
changeset 40308 274367a99f98
parent 38524 badd925c1d2f
child 41860 906670ff49c7
permissions -rw-r--r--
8136930: Simplify use of module-system options by custom launchers 8160489: Multiple -Xpatch lines ignored by javac 8156998: javac should support new option -XinheritRuntimeEnvironment Reviewed-by: jlahoda, ksrini
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.Deque;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    45
import java.util.HashMap;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    46
import java.util.HashSet;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    47
import java.util.LinkedList;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    48
import java.util.Map;
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    49
import java.util.Optional;
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    50
import java.util.Set;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    51
import java.util.function.Function;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    52
import java.util.stream.Collectors;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    53
import java.util.stream.Stream;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    54
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    55
/**
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    56
 * Analyze module dependences and compare with module descriptor.
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    57
 * Also identify any qualified exports not used by the target module.
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    58
 */
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    59
public class ModuleAnalyzer {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    60
    private static final String JAVA_BASE = "java.base";
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    61
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    62
    private final JdepsConfiguration configuration;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    63
    private final PrintWriter log;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    64
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    65
    private final DependencyFinder dependencyFinder;
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    66
    private final Map<Module, ModuleDeps> modules;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    67
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    68
    public ModuleAnalyzer(JdepsConfiguration config,
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    69
                          PrintWriter log) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    70
        this(config, log, Collections.emptySet());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    71
    }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    72
    public ModuleAnalyzer(JdepsConfiguration config,
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    73
                          PrintWriter log,
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    74
                          Set<String> names) {
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    75
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    76
        if (!config.initialArchives().isEmpty()) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    77
            String list = config.initialArchives().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    78
                .map(Archive::getPathName).collect(joining(" "));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    79
            throw new JdepsTask.UncheckedBadArgs(new BadArgs("err.invalid.module.option",
40308
274367a99f98 8136930: Simplify use of module-system options by custom launchers
jjg
parents: 38524
diff changeset
    80
                list, "--check"));
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    81
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    82
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    83
        this.configuration = config;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    84
        this.log = log;
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    85
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    86
        this.dependencyFinder = new DependencyFinder(config, DEFAULT_FILTER);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    87
        if (names.isEmpty()) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    88
            this.modules = configuration.rootModules().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    89
                .collect(toMap(Function.identity(), ModuleDeps::new));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    90
        } else {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    91
            this.modules = names.stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    92
                .map(configuration::findModule)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    93
                .flatMap(Optional::stream)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    94
                .collect(toMap(Function.identity(), ModuleDeps::new));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    95
        }
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    96
    }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
    97
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    98
    public boolean run() throws IOException {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
    99
        try {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   100
            // compute "requires public" dependences
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   101
            modules.values().forEach(ModuleDeps::computeRequiresPublic);
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   102
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   103
            modules.values().forEach(md -> {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   104
                // compute "requires" dependences
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   105
                md.computeRequires();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   106
                // apply transitive reduction and reports recommended requires.
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   107
                md.analyzeDeps();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   108
            });
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   109
        } finally {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   110
            dependencyFinder.shutdown();
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   111
        }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   112
        return true;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   113
    }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   114
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   115
    class ModuleDeps {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   116
        final Module root;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   117
        Set<Module> requiresPublic;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   118
        Set<Module> requires;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   119
        Map<String, Set<String>> unusedQualifiedExports;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   120
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   121
        ModuleDeps(Module root) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   122
            this.root = root;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   123
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   124
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   125
        /**
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   126
         * Compute 'requires public' dependences by analyzing API dependencies
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   127
         */
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   128
        private void computeRequiresPublic() {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   129
            // record requires public
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   130
            this.requiresPublic = computeRequires(true)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   131
                .filter(m -> !m.name().equals(JAVA_BASE))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   132
                .collect(toSet());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   133
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   134
            trace("requires public: %s%n", requiresPublic);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   135
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   136
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   137
        private void computeRequires() {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   138
            this.requires = computeRequires(false).collect(toSet());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   139
            trace("requires: %s%n", requires);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   140
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   141
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   142
        private Stream<Module> computeRequires(boolean apionly) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   143
            // analyze all classes
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   144
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   145
            if (apionly) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   146
                dependencyFinder.parseExportedAPIs(Stream.of(root));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   147
            } else {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   148
                dependencyFinder.parse(Stream.of(root));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   149
            }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   150
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   151
            // find the modules of all the dependencies found
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   152
            return dependencyFinder.getDependences(root)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   153
                        .map(Archive::getModule);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   154
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   155
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   156
        ModuleDescriptor descriptor() {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   157
            return descriptor(requiresPublic, requires);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   158
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   159
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   160
        private ModuleDescriptor descriptor(Set<Module> requiresPublic,
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   161
                                            Set<Module> requires) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   162
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   163
            ModuleDescriptor.Builder builder = new ModuleDescriptor.Builder(root.name());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   164
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   165
            if (!root.name().equals(JAVA_BASE))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   166
                builder.requires(MANDATED, JAVA_BASE);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   167
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   168
            requiresPublic.stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   169
                .filter(m -> !m.name().equals(JAVA_BASE))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   170
                .map(Module::name)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   171
                .forEach(mn -> builder.requires(PUBLIC, mn));
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   172
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   173
            requires.stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   174
                .filter(m -> !requiresPublic.contains(m))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   175
                .filter(m -> !m.name().equals(JAVA_BASE))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   176
                .map(Module::name)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   177
                .forEach(mn -> builder.requires(mn));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   178
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   179
            return builder.build();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   180
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   181
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   182
        ModuleDescriptor reduced() {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   183
            Graph.Builder<Module> bd = new Graph.Builder<>();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   184
            requiresPublic.stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   185
                .forEach(m -> {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   186
                    bd.addNode(m);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   187
                    bd.addEdge(root, m);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   188
                });
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   189
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   190
            // requires public graph
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   191
            Graph<Module> rbg = bd.build().reduce();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   192
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   193
            // transitive reduction
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   194
            Graph<Module> newGraph = buildGraph(requires).reduce(rbg);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   195
            if (DEBUG) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   196
                System.err.println("after transitive reduction: ");
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   197
                newGraph.printGraph(log);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   198
            }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   199
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   200
            return descriptor(requiresPublic, newGraph.adjacentNodes(root));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   201
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   202
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   203
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   204
        /**
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   205
         * Apply transitive reduction on the resulting graph and reports
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   206
         * recommended requires.
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   207
         */
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   208
        private void analyzeDeps() {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   209
            Graph.Builder<Module> builder = new Graph.Builder<>();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   210
            requiresPublic.stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   211
                .forEach(m -> {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   212
                    builder.addNode(m);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   213
                    builder.addEdge(root, m);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   214
                });
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   215
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   216
            // requires public graph
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   217
            Graph<Module> rbg = buildGraph(requiresPublic).reduce();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   218
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   219
            // transitive reduction
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   220
            Graph<Module> newGraph = buildGraph(requires).reduce(builder.build().reduce());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   221
            if (DEBUG) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   222
                System.err.println("after transitive reduction: ");
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   223
                newGraph.printGraph(log);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   224
            }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   225
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   226
            printModuleDescriptor(log, root);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   227
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   228
            ModuleDescriptor analyzedDescriptor = descriptor();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   229
            if (!matches(root.descriptor(), analyzedDescriptor)) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   230
                log.format("  [Suggested module descriptor for %s]%n", root.name());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   231
                analyzedDescriptor.requires()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   232
                    .stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   233
                    .sorted(Comparator.comparing(ModuleDescriptor.Requires::name))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   234
                    .forEach(req -> log.format("    requires %s;%n", req));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   235
            }
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   236
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   237
            ModuleDescriptor reduced = reduced();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   238
            if (!matches(root.descriptor(), reduced)) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   239
                log.format("  [Transitive reduced graph for %s]%n", root.name());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   240
                reduced.requires()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   241
                    .stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   242
                    .sorted(Comparator.comparing(ModuleDescriptor.Requires::name))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   243
                    .forEach(req -> log.format("    requires %s;%n", req));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   244
            }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   245
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   246
            checkQualifiedExports();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   247
            log.println();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   248
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   249
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   250
        private void checkQualifiedExports() {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   251
            // detect any qualified exports not used by the target module
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   252
            unusedQualifiedExports = unusedQualifiedExports();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   253
            if (!unusedQualifiedExports.isEmpty())
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   254
                log.format("  [Unused qualified exports in %s]%n", root.name());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   255
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   256
            unusedQualifiedExports.keySet().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   257
                .sorted()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   258
                .forEach(pn -> log.format("    exports %s to %s%n", pn,
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   259
                    unusedQualifiedExports.get(pn).stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   260
                        .sorted()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   261
                        .collect(joining(","))));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   262
        }
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   263
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   264
        private void printModuleDescriptor(PrintWriter out, Module module) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   265
            ModuleDescriptor descriptor = module.descriptor();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   266
            out.format("%s (%s)%n", descriptor.name(), module.location());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   267
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   268
            if (descriptor.name().equals(JAVA_BASE))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   269
                return;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   270
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   271
            out.println("  [Module descriptor]");
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   272
            descriptor.requires()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   273
                .stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   274
                .sorted(Comparator.comparing(ModuleDescriptor.Requires::name))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   275
                .forEach(req -> out.format("    requires %s;%n", req));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   276
        }
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
        /**
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   280
         * Returns a graph of modules required by the specified module.
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   281
         *
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   282
         * Requires public edges of the dependences are added to the graph.
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   283
         */
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   284
        private Graph<Module> buildGraph(Set<Module> deps) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   285
            Graph.Builder<Module> builder = new Graph.Builder<>();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   286
            builder.addNode(root);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   287
            Set<Module> visited = new HashSet<>();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   288
            visited.add(root);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   289
            Deque<Module> deque = new LinkedList<>();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   290
            deps.stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   291
                .forEach(m -> {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   292
                    deque.add(m);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   293
                    builder.addEdge(root, m);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   294
                });
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   295
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   296
            // read requires public from ModuleDescription
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   297
            Module source;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   298
            while ((source = deque.poll()) != null) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   299
                if (visited.contains(source))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   300
                    continue;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   301
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   302
                visited.add(source);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   303
                builder.addNode(source);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   304
                Module from = source;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   305
                source.descriptor().requires().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   306
                    .filter(req -> req.modifiers().contains(PUBLIC))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   307
                    .map(ModuleDescriptor.Requires::name)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   308
                    .map(configuration::findModule)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   309
                    .flatMap(Optional::stream)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   310
                    .forEach(m -> {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   311
                        deque.add(m);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   312
                        builder.addEdge(from, m);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   313
                    });
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   314
            }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   315
            return builder.build();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   316
        }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   317
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   318
        /**
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   319
         * Detects any qualified exports not used by the target module.
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   320
         */
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   321
        private Map<String, Set<String>> unusedQualifiedExports() {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   322
            Map<String, Set<String>> unused = new HashMap<>();
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   323
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   324
            // build the qualified exports map
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   325
            Map<String, Set<String>> qualifiedExports =
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   326
                root.exports().entrySet().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   327
                    .filter(e -> !e.getValue().isEmpty())
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   328
                    .map(Map.Entry::getKey)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   329
                    .collect(toMap(Function.identity(), _k -> new HashSet<>()));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   330
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   331
            Set<Module> mods = new HashSet<>();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   332
            root.exports().values()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   333
                .stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   334
                .flatMap(Set::stream)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   335
                .forEach(target -> configuration.findModule(target)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   336
                    .ifPresentOrElse(mods::add,
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   337
                        () -> log.format("Warning: %s not found%n", target))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   338
                );
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   339
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   340
            // parse all target modules
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   341
            dependencyFinder.parse(mods.stream());
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   342
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   343
            // adds to the qualified exports map if a module references it
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   344
            mods.stream().forEach(m ->
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   345
                m.getDependencies()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   346
                    .map(Dependency.Location::getPackageName)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   347
                    .filter(qualifiedExports::containsKey)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   348
                    .forEach(pn -> qualifiedExports.get(pn).add(m.name())));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   349
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   350
            // compare with the exports from ModuleDescriptor
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   351
            Set<String> staleQualifiedExports =
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   352
                qualifiedExports.keySet().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   353
                    .filter(pn -> !qualifiedExports.get(pn).equals(root.exports().get(pn)))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   354
                    .collect(toSet());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   355
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   356
            if (!staleQualifiedExports.isEmpty()) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   357
                for (String pn : staleQualifiedExports) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   358
                    Set<String> targets = new HashSet<>(root.exports().get(pn));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   359
                    targets.removeAll(qualifiedExports.get(pn));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   360
                    unused.put(pn, targets);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   361
                }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   362
            }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   363
            return unused;
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   364
        }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   365
    }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   366
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   367
    private boolean matches(ModuleDescriptor md, ModuleDescriptor other) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   368
        // build requires public from ModuleDescriptor
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   369
        Set<ModuleDescriptor.Requires> reqPublic = md.requires().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   370
            .filter(req -> req.modifiers().contains(PUBLIC))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   371
            .collect(toSet());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   372
        Set<ModuleDescriptor.Requires> otherReqPublic = other.requires().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   373
            .filter(req -> req.modifiers().contains(PUBLIC))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   374
            .collect(toSet());
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   375
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   376
        if (!reqPublic.equals(otherReqPublic)) {
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   377
            trace("mismatch requires public: %s%n", reqPublic);
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   378
            return false;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   379
        }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   380
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   381
        Set<ModuleDescriptor.Requires> unused = md.requires().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   382
            .filter(req -> !other.requires().contains(req))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   383
            .collect(Collectors.toSet());
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   384
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   385
        if (!unused.isEmpty()) {
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   386
            trace("mismatch requires: %s%n", unused);
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   387
            return false;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   388
        }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   389
        return true;
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   390
    }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   391
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   392
    /**
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   393
     * Generate dotfile from module descriptor
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   394
     *
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   395
     * @param dir output directory
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   396
     */
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   397
    public boolean genDotFiles(Path dir) throws IOException {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   398
        Files.createDirectories(dir);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   399
        for (Module m : modules.keySet()) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   400
            genDotFile(dir, m.name());
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   401
        }
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   402
        return true;
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   403
    }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   404
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   405
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   406
    private void genDotFile(Path dir, String name) throws IOException {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   407
        try (OutputStream os = Files.newOutputStream(dir.resolve(name + ".dot"));
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   408
             PrintWriter out = new PrintWriter(os)) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   409
            Set<Module> modules = configuration.resolve(Set.of(name))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   410
                .collect(Collectors.toSet());
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   411
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   412
            // transitive reduction
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   413
            Graph<String> graph = gengraph(modules);
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   414
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   415
            out.format("digraph \"%s\" {%n", name);
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   416
            DotGraph.printAttributes(out);
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   417
            DotGraph.printNodes(out, graph);
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   418
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   419
            modules.stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   420
                .map(Module::descriptor)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   421
                .sorted(Comparator.comparing(ModuleDescriptor::name))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   422
                .forEach(md -> {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   423
                    String mn = md.name();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   424
                    Set<String> requiresPublic = md.requires().stream()
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   425
                        .filter(d -> d.modifiers().contains(PUBLIC))
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   426
                        .map(d -> d.name())
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   427
                        .collect(toSet());
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   428
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   429
                    DotGraph.printEdges(out, graph, mn, requiresPublic);
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   430
                });
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   431
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   432
            out.println("}");
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   433
        }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   434
    }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   435
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   436
    /**
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   437
     * Returns a Graph of the given Configuration after transitive reduction.
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   438
     *
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   439
     * Transitive reduction of requires public edge and requires edge have
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   440
     * to be applied separately to prevent the requires public edges
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   441
     * (e.g. U -> V) from being reduced by a path (U -> X -> Y -> V)
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   442
     * in which  V would not be re-exported from U.
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   443
     */
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   444
    private Graph<String> gengraph(Set<Module> modules) {
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   445
        // build a Graph containing only requires public edges
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   446
        // with transitive reduction.
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   447
        Graph.Builder<String> rpgbuilder = new Graph.Builder<>();
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   448
        for (Module module : modules) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   449
            ModuleDescriptor md = module.descriptor();
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   450
            String mn = md.name();
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   451
            md.requires().stream()
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   452
                    .filter(d -> d.modifiers().contains(PUBLIC))
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   453
                    .map(d -> d.name())
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   454
                    .forEach(d -> rpgbuilder.addEdge(mn, d));
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   455
        }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   456
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   457
        Graph<String> rpg = rpgbuilder.build().reduce();
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   458
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   459
        // build the readability graph
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   460
        Graph.Builder<String> builder = new Graph.Builder<>();
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   461
        for (Module module : modules) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   462
            ModuleDescriptor md = module.descriptor();
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   463
            String mn = md.name();
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   464
            builder.addNode(mn);
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   465
            configuration.reads(module)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   466
                    .map(Module::name)
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   467
                    .forEach(d -> builder.addEdge(mn, d));
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   468
        }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   469
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   470
        // transitive reduction of requires edges
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   471
        return builder.build().reduce(rpg);
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   472
    }
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   473
38524
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   474
    // ---- for testing purpose
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   475
    public ModuleDescriptor[] descriptors(String name) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   476
        ModuleDeps moduleDeps = modules.keySet().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   477
            .filter(m -> m.name().equals(name))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   478
            .map(modules::get)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   479
            .findFirst().get();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   480
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   481
        ModuleDescriptor[] descriptors = new ModuleDescriptor[3];
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   482
        descriptors[0] = moduleDeps.root.descriptor();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   483
        descriptors[1] = moduleDeps.descriptor();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   484
        descriptors[2] = moduleDeps.reduced();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   485
        return descriptors;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   486
    }
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   487
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   488
    public Map<String, Set<String>> unusedQualifiedExports(String name) {
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   489
        ModuleDeps moduleDeps = modules.keySet().stream()
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   490
            .filter(m -> m.name().equals(name))
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   491
            .map(modules::get)
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   492
            .findFirst().get();
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   493
        return moduleDeps.unusedQualifiedExports;
badd925c1d2f 8156680: jdeps implementation refresh
mchung
parents: 36526
diff changeset
   494
    }
36526
3b41f1c69604 8142968: Module System implementation
alanb
parents:
diff changeset
   495
}