File: JdwpNetProps.java

package info (click to toggle)
openjdk-25 25.0.1%2B8-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 825,408 kB
  • sloc: java: 5,585,680; cpp: 1,333,948; xml: 1,321,242; ansic: 488,034; asm: 404,003; objc: 21,088; sh: 15,106; javascript: 13,265; python: 8,319; makefile: 2,518; perl: 357; awk: 351; pascal: 103; exp: 83; sed: 72; jsp: 24
file content (274 lines) | stat: -rw-r--r-- 11,944 bytes parent folder | download | duplicates (4)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
/*
 * Copyright (c) 2019, 2024, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

import com.sun.jdi.Bootstrap;
import com.sun.jdi.VirtualMachine;
import com.sun.jdi.connect.AttachingConnector;
import com.sun.jdi.connect.Connector;
import com.sun.jdi.connect.IllegalConnectorArgumentsException;
import lib.jdb.Debuggee;

import java.io.IOException;
import java.net.Inet4Address;
import java.net.Inet6Address;
import java.net.InetAddress;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Objects;

/*
 * @test
 * @bug 8184770 8313804
 * @summary Tests that JDWP agent honors jdk net properties
 * @library /test/lib
 *
 * @build HelloWorld JdwpNetProps
 * @run main/othervm -Djava.net.preferIPv6Addresses=system JdwpNetProps
 */
public class JdwpNetProps {

    // Set to true to allow testing of attach from wrong address (expected to fail).
    // It's off by default as it causes test interference (see JDK-8311990).
    private static boolean allowNegativeAttachTesting =
        "true".equalsIgnoreCase(System.getProperty("jdk.jdi.allowNegativeTesting"));

    public static void main(String[] args) throws Exception {
        InetAddress addrs[] = InetAddress.getAllByName("localhost");
        InetAddress ipv4Address = null;
        InetAddress ipv6Address = null;
        for (int i =  0; i < addrs.length; i++) {
            if (addrs[i] instanceof Inet4Address) {
                ipv4Address = addrs[i];
            } else if (addrs[i] instanceof Inet6Address) {
                ipv6Address = addrs[i];
            }
        }

        String preferIPv6Address = System.getProperty("java.net.preferIPv6Addresses");
        if (!Objects.equals(preferIPv6Address, "system")) {
          throw new AssertionError(
              "Expected -Djava.net.preferIPv6Address=system, was " + preferIPv6Address);
        }
        boolean systemPrefersIPv6 = addrs[0] instanceof Inet6Address;

        if (ipv4Address != null) {
            new ListenTest("localhost", ipv4Address)
                    .preferIPv4Stack(true)
                    .run(TestResult.Success);
            new ListenTest("localhost", ipv4Address)
                    .preferIPv4Stack(true)
                    .preferIPv6Addresses("true")
                    .run(TestResult.Success);
            new ListenTest("localhost", ipv4Address)
                    .preferIPv4Stack(true)
                    .preferIPv6Addresses("system")
                    .run(TestResult.Success);
            new ListenTest("localhost", ipv4Address)
                    .preferIPv4Stack(false)
                    .run(TestResult.Success);
            if (ipv6Address != null) {
                // - only IPv4, so connection from IPv6 should fail
                new ListenTest("localhost", ipv6Address)
                        .preferIPv4Stack(true)
                        .preferIPv6Addresses("true")
                        .run(TestResult.AttachFailed);
                new ListenTest("localhost", ipv6Address)
                        .preferIPv4Stack(true)
                        .preferIPv6Addresses("system")
                        .run(TestResult.AttachFailed);
                // - listen on IPv4
                new ListenTest("localhost", ipv6Address)
                        .preferIPv6Addresses("false")
                        .run(TestResult.AttachFailed);
                // - listen on IPv4 (preferIPv6Addresses defaults to false)
                new ListenTest("localhost", ipv6Address)
                        .run(TestResult.AttachFailed);
                // - listen on IPv6
                new ListenTest("localhost", ipv6Address)
                        .preferIPv6Addresses("true")
                        .run(TestResult.Success);
                new ListenTest("localhost", ipv6Address)
                        .preferIPv6Addresses("system")
                        .run(systemPrefersIPv6 ? TestResult.Success : TestResult.AttachFailed);
                // - listen on IPv6, connect from IPv4
                new ListenTest("localhost", ipv4Address)
                        .preferIPv4Stack(false)
                        .preferIPv6Addresses("true")
                        .run(TestResult.AttachFailed);
                // - listen on system preference, connect from IPv4
                new ListenTest("localhost", ipv4Address)
                        .preferIPv4Stack(false)
                        .preferIPv6Addresses("system")
                        .run(systemPrefersIPv6 ? TestResult.AttachFailed : TestResult.Success);
            }
        } else {
            if (!systemPrefersIPv6) {
                throw new AssertionError("The system is IPv6-only, but systemPrefersIPv6 was unexpectedly false");
            }

            // IPv6-only system - expected to fail on IPv4 address
            new ListenTest("localhost", ipv6Address)
                    .preferIPv4Stack(true)
                    .run(TestResult.ListenFailed);
            new ListenTest("localhost", ipv6Address)
                    .preferIPv4Stack(true)
                    .preferIPv6Addresses("system")
                    .run(TestResult.ListenFailed);
            new ListenTest("localhost", ipv6Address)
                    .preferIPv4Stack(true)
                    .preferIPv6Addresses("true")
                    .run(TestResult.ListenFailed);
            new ListenTest("localhost", ipv6Address)
                    .run(TestResult.Success);
            new ListenTest("localhost", ipv6Address)
                    .preferIPv6Addresses("system")
                    .run(TestResult.Success);
            new ListenTest("localhost", ipv6Address)
                    .preferIPv6Addresses("true")
                    .run(TestResult.Success);
        }
    }

    private enum TestResult {
        Success,
        ListenFailed,
        AttachFailed
    }

    private static class ListenTest {
        private final String listenAddress;
        private final InetAddress connectAddress;
        private Boolean preferIPv4Stack;
        private String preferIPv6Addresses;
        public ListenTest(String listenAddress, InetAddress connectAddress) {
            this.listenAddress = listenAddress;
            this.connectAddress = connectAddress;
        }
        public ListenTest preferIPv4Stack(Boolean value) {
            preferIPv4Stack = value;
            return this;
        }
        public ListenTest preferIPv6Addresses(String value) {
            preferIPv6Addresses = value;
            return this;
        }

        public void run(TestResult expectedResult) throws Exception {
            log("\nTest: listen at " + listenAddress + ", attaching to " + connectAddress
                + ", preferIPv4Stack = " + preferIPv4Stack
                + ", preferIPv6Addresses = " + preferIPv6Addresses
                + ", expectedResult = " + expectedResult);
            if (expectedResult == TestResult.AttachFailed && !allowNegativeAttachTesting) {
                log("SKIPPED: negative attach testing is disabled");
                return;
            }
            List<String> options = new LinkedList<>();
            if (preferIPv4Stack != null) {
                options.add("-Djava.net.preferIPv4Stack=" + preferIPv4Stack.toString());
            }
            if (preferIPv6Addresses != null) {
                options.add("-Djava.net.preferIPv6Addresses=" + preferIPv6Addresses);
            }
            log("Starting listening debuggee at " + listenAddress
                    + (expectedResult == TestResult.ListenFailed ? ": expected to fail" : ""));
            Exception error = null;
            try (Debuggee debuggee = Debuggee.launcher("HelloWorld")
                    .setAddress(listenAddress + ":0")
                    .addOptions(options).launch()) {
                log("Debuggee is listening on " + listenAddress + ":" + debuggee.getAddress());
                log("Connecting from " + connectAddress.getHostAddress()
                        + ", expected: " + (expectedResult == TestResult.Success ? "Success" : "Failure"));
                try {
                    VirtualMachine vm = attach(connectAddress.getHostAddress(), debuggee.getAddress());
                    vm.dispose();
                    if (expectedResult == TestResult.Success) {
                        log("Attached successfully (as expected)");
                    } else {
                        error = new RuntimeException("ERROR: attached successfully");
                    }
                } catch (Exception ex) {
                    if (expectedResult == TestResult.AttachFailed) {
                        log("Attach failed (as expected)");
                    } else {
                        error = new RuntimeException("ERROR: failed to attach", ex);
                    }
                }
            } catch (Exception ex) {
                if (expectedResult == TestResult.ListenFailed) {
                    log("Listen failed (as expected)");
                } else {
                    error = new RuntimeException("ERROR: listen failed", ex);
                }
            }
            if (error != null) {
                throw error;
            }
        }
    }

    private static String ATTACH_CONNECTOR = "com.sun.jdi.SocketAttach";
    // cache socket attaching connector
    private static AttachingConnector attachingConnector;

    private static VirtualMachine attach(String address, String port) throws IOException {
        if (attachingConnector == null) {
            attachingConnector = (AttachingConnector)getConnector(ATTACH_CONNECTOR);
        }
        Map<String, Connector.Argument> args = attachingConnector.defaultArguments();
        setConnectorArg(args, "hostname", address);
        setConnectorArg(args, "port", port);
        try {
            return attachingConnector.attach(args);
        } catch (IllegalConnectorArgumentsException e) {
            // unexpected.. wrap in RuntimeException
            throw new RuntimeException(e);
        }
    }

    private static Connector getConnector(String name) {
        List<Connector> connectors = Bootstrap.virtualMachineManager().allConnectors();
        for (Iterator<Connector> iter = connectors.iterator(); iter.hasNext(); ) {
            Connector connector = iter.next();
            if (connector.name().equalsIgnoreCase(name)) {
                return connector;
            }
        }
        throw new IllegalArgumentException("Connector " + name + " not found");
    }

    private static void setConnectorArg(Map<String, Connector.Argument> args, String name, String value) {
        Connector.Argument arg = args.get(name);
        if (arg == null) {
            throw new IllegalArgumentException("Argument " + name + " is not defined");
        }
        arg.setValue(value);
    }

    private static void log(Object o) {
        System.out.println(String.valueOf(o));
    }


}