langtools/src/jdk.jdeps/share/classes/com/sun/tools/jdeps/InverseDepsAnalyzer.java
author mchung
Mon, 19 Dec 2016 12:30:39 -0800
changeset 42840 dfe1a03d4db4
parent 38525 a2169f8fa712
permissions -rw-r--r--
8171418: Remove jdeps internal --include-system-modules option Reviewed-by: dfuchs, lancea
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
38525
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
     1
/*
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
     2
 * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
     4
 *
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    10
 *
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    15
 * accompanied this code).
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    16
 *
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    20
 *
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    23
 * questions.
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    24
 */
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    25
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    26
package com.sun.tools.jdeps;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    27
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    28
import static com.sun.tools.jdeps.JdepsFilter.DEFAULT_FILTER;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    29
import static com.sun.tools.jdeps.Module.trace;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    30
import static com.sun.tools.jdeps.Graph.*;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    31
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    32
import java.lang.module.ModuleDescriptor.Requires;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    33
import java.io.IOException;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    34
import java.util.Collections;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    35
import java.util.Deque;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    36
import java.util.HashMap;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    37
import java.util.HashSet;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    38
import java.util.LinkedList;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    39
import java.util.Map;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    40
import java.util.Optional;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    41
import java.util.Set;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    42
import java.util.stream.Collectors;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    43
import java.util.stream.Stream;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    44
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    45
/**
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    46
 * Inverse transitive dependency analysis (compile-time view)
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    47
 */
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    48
public class InverseDepsAnalyzer extends DepsAnalyzer {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    49
    // the end points for the resulting paths to be reported
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    50
    private final Map<Archive, Set<Archive>> endPoints = new HashMap<>();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    51
    // target archives for inverse transitive dependence analysis
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    52
    private final Set<Archive> targets = new HashSet<>();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    53
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    54
    public InverseDepsAnalyzer(JdepsConfiguration config,
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    55
                               JdepsFilter filter,
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    56
                               JdepsWriter writer,
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    57
                               Analyzer.Type verbose,
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    58
                               boolean apiOnly) {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    59
        super(config, filter, writer, verbose, apiOnly);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    60
    }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    61
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    62
    public boolean run() throws IOException {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    63
        try {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    64
            if (apiOnly) {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    65
                finder.parseExportedAPIs(rootArchives.stream());
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    66
            } else {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    67
                finder.parse(rootArchives.stream());
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    68
            }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    69
            archives.addAll(rootArchives);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    70
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    71
            Set<Archive> archives = archives();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    72
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    73
            // If -package or -regex is specified, the archives that reference
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    74
            // the matching types are used as the targets for inverse
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    75
            // transitive analysis.  If -requires is specified, the
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    76
            // specified modules are the targets.
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    77
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    78
            if (filter.requiresFilter().isEmpty()) {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    79
                targets.addAll(archives);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    80
            } else {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    81
                filter.requiresFilter().stream()
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    82
                      .map(configuration::findModule)
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    83
                      .flatMap(Optional::stream)
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    84
                      .forEach(targets::add);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    85
            }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    86
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    87
            // If -package or -regex is specified, the end points are
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    88
            // the matching archives.  If -requires is specified,
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    89
            // the end points are the modules specified in -requires.
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    90
            if (filter.requiresFilter().isEmpty()) {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    91
                Map<Archive, Set<Archive>> dependences = finder.dependences();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    92
                targets.forEach(source -> endPoints.put(source, dependences.get(source)));
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    93
            } else {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    94
                targets.forEach(t -> endPoints.put(t, Collections.emptySet()));
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    95
            }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    96
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    97
            analyzer.run(archives, finder.locationToArchive());
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    98
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
    99
            // print the first-level of dependencies
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   100
            if (writer != null) {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   101
                writer.generateOutput(archives, analyzer);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   102
            }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   103
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   104
        } finally {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   105
            finder.shutdown();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   106
        }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   107
        return true;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   108
    }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   109
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   110
    /**
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   111
     * Returns the target archives determined from the dependency analysis.
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   112
     *
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   113
     * Inverse transitive dependency will find all nodes that depend
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   114
     * upon the returned set of archives directly and indirectly.
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   115
     */
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   116
    public Set<Archive> targets() {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   117
        return Collections.unmodifiableSet(targets);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   118
    }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   119
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   120
    /**
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   121
     * Finds all inverse transitive dependencies using the given requires set
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   122
     * as the targets, if non-empty.  If the given requires set is empty,
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   123
     * use the archives depending the packages specified in -regex or -p options.
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   124
     */
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   125
    public Set<Deque<Archive>> inverseDependences() throws IOException {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   126
        // create a new dependency finder to do the analysis
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   127
        DependencyFinder dependencyFinder = new DependencyFinder(configuration, DEFAULT_FILTER);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   128
        try {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   129
            // parse all archives in unnamed module to get compile-time dependences
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   130
            Stream<Archive> archives =
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   131
                Stream.concat(configuration.initialArchives().stream(),
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   132
                              configuration.classPathArchives().stream());
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   133
            if (apiOnly) {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   134
                dependencyFinder.parseExportedAPIs(archives);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   135
            } else {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   136
                dependencyFinder.parse(archives);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   137
            }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   138
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   139
            Graph.Builder<Archive> builder = new Graph.Builder<>();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   140
            // include all target nodes
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   141
            targets().forEach(builder::addNode);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   142
42840
dfe1a03d4db4 8171418: Remove jdeps internal --include-system-modules option
mchung
parents: 38525
diff changeset
   143
            // transpose the module graph
38525
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   144
            configuration.getModules().values().stream()
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   145
                .forEach(m -> {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   146
                    builder.addNode(m);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   147
                    m.descriptor().requires().stream()
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   148
                        .map(Requires::name)
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   149
                        .map(configuration::findModule)  // must be present
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   150
                        .forEach(v -> builder.addEdge(v.get(), m));
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   151
                });
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   152
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   153
            // add the dependences from the analysis
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   154
            Map<Archive, Set<Archive>> dependences = dependencyFinder.dependences();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   155
            dependences.entrySet().stream()
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   156
                .forEach(e -> {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   157
                    Archive u = e.getKey();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   158
                    builder.addNode(u);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   159
                    e.getValue().forEach(v -> builder.addEdge(v, u));
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   160
                });
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   161
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   162
            // transposed dependence graph.
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   163
            Graph<Archive> graph = builder.build();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   164
            trace("targets: %s%n", targets());
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   165
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   166
            // Traverse from the targets and find all paths
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   167
            // rebuild a graph with all nodes that depends on targets
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   168
            // targets directly and indirectly
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   169
            return targets().stream()
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   170
                .map(t -> findPaths(graph, t))
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   171
                .flatMap(Set::stream)
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   172
                .collect(Collectors.toSet());
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   173
        } finally {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   174
            dependencyFinder.shutdown();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   175
        }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   176
    }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   177
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   178
    /**
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   179
     * Returns all paths reachable from the given targets.
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   180
     */
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   181
    private Set<Deque<Archive>> findPaths(Graph<Archive> graph, Archive target) {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   182
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   183
        // path is in reversed order
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   184
        Deque<Archive> path = new LinkedList<>();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   185
        path.push(target);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   186
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   187
        Set<Edge<Archive>> visited = new HashSet<>();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   188
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   189
        Deque<Edge<Archive>> deque = new LinkedList<>();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   190
        deque.addAll(graph.edgesFrom(target));
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   191
        if (deque.isEmpty()) {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   192
            return makePaths(path).collect(Collectors.toSet());
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   193
        }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   194
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   195
        Set<Deque<Archive>> allPaths = new HashSet<>();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   196
        while (!deque.isEmpty()) {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   197
            Edge<Archive> edge = deque.pop();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   198
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   199
            if (visited.contains(edge))
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   200
                continue;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   201
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   202
            Archive node = edge.v;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   203
            path.addLast(node);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   204
            visited.add(edge);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   205
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   206
            Set<Edge<Archive>> unvisitedDeps = graph.edgesFrom(node)
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   207
                    .stream()
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   208
                    .filter(e -> !visited.contains(e))
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   209
                    .collect(Collectors.toSet());
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   210
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   211
            trace("visiting %s %s (%s)%n", edge, path, unvisitedDeps);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   212
            if (unvisitedDeps.isEmpty()) {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   213
                makePaths(path).forEach(allPaths::add);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   214
                path.removeLast();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   215
            }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   216
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   217
            // push unvisited adjacent edges
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   218
            unvisitedDeps.stream().forEach(deque::push);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   219
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   220
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   221
            // when the adjacent edges of a node are visited, pop it from the path
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   222
            while (!path.isEmpty()) {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   223
                if (visited.containsAll(graph.edgesFrom(path.peekLast())))
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   224
                    path.removeLast();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   225
                else
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   226
                    break;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   227
            }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   228
        }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   229
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   230
       return allPaths;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   231
    }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   232
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   233
    /**
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   234
     * Prepend end point to the path
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   235
     */
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   236
    private Stream<Deque<Archive>> makePaths(Deque<Archive> path) {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   237
        Set<Archive> nodes = endPoints.get(path.peekFirst());
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   238
        if (nodes == null || nodes.isEmpty()) {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   239
            return Stream.of(new LinkedList<>(path));
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   240
        } else {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   241
            return nodes.stream().map(n -> {
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   242
                Deque<Archive> newPath = new LinkedList<>();
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   243
                newPath.addFirst(n);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   244
                newPath.addAll(path);
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   245
                return newPath;
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   246
            });
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   247
        }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   248
    }
a2169f8fa712 8156575: Add jdeps -addmods, -system, -inverse options
mchung
parents:
diff changeset
   249
}