8308995: Update Network IO JFR events to be static mirror events
Reviewed-by: egahlin, alanb
This commit is contained in:
parent
e1870d360e
commit
b275bdd9b5
@ -1,5 +1,5 @@
|
|||||||
#
|
#
|
||||||
# Copyright (c) 2018, 2022, Oracle and/or its affiliates. All rights reserved.
|
# Copyright (c) 2018, 2023, Oracle and/or its affiliates. All rights reserved.
|
||||||
# DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
# DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
#
|
#
|
||||||
# This code is free software; you can redistribute it and/or modify it
|
# This code is free software; you can redistribute it and/or modify it
|
||||||
@ -105,6 +105,7 @@ $(eval $(call SetupJavaCompilation, BUILD_JDK_MICROBENCHMARK, \
|
|||||||
--add-exports java.base/jdk.internal.org.objectweb.asm=ALL-UNNAMED \
|
--add-exports java.base/jdk.internal.org.objectweb.asm=ALL-UNNAMED \
|
||||||
--add-exports java.base/jdk.internal.org.objectweb.asm.tree=ALL-UNNAMED \
|
--add-exports java.base/jdk.internal.org.objectweb.asm.tree=ALL-UNNAMED \
|
||||||
--add-exports java.base/jdk.internal.vm=ALL-UNNAMED \
|
--add-exports java.base/jdk.internal.vm=ALL-UNNAMED \
|
||||||
|
--add-exports java.base/jdk.internal.event=ALL-UNNAMED \
|
||||||
--enable-preview, \
|
--enable-preview, \
|
||||||
JAVA_FLAGS := --add-modules jdk.unsupported --limit-modules java.management \
|
JAVA_FLAGS := --add-modules jdk.unsupported --limit-modules java.management \
|
||||||
--add-exports java.base/jdk.internal.vm=ALL-UNNAMED \
|
--add-exports java.base/jdk.internal.vm=ALL-UNNAMED \
|
||||||
|
@ -25,6 +25,8 @@
|
|||||||
|
|
||||||
package java.net;
|
package java.net;
|
||||||
|
|
||||||
|
import jdk.internal.event.SocketReadEvent;
|
||||||
|
import jdk.internal.event.SocketWriteEvent;
|
||||||
import sun.security.util.SecurityConstants;
|
import sun.security.util.SecurityConstants;
|
||||||
|
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
@ -1073,9 +1075,6 @@ public class Socket implements java.io.Closeable {
|
|||||||
/**
|
/**
|
||||||
* An InputStream that delegates read/available operations to an underlying
|
* An InputStream that delegates read/available operations to an underlying
|
||||||
* input stream. The close method is overridden to close the Socket.
|
* input stream. The close method is overridden to close the Socket.
|
||||||
*
|
|
||||||
* This class is instrumented by Java Flight Recorder (JFR) to get socket
|
|
||||||
* I/O events.
|
|
||||||
*/
|
*/
|
||||||
private static class SocketInputStream extends InputStream {
|
private static class SocketInputStream extends InputStream {
|
||||||
private final Socket parent;
|
private final Socket parent;
|
||||||
@ -1092,6 +1091,16 @@ public class Socket implements java.io.Closeable {
|
|||||||
}
|
}
|
||||||
@Override
|
@Override
|
||||||
public int read(byte[] b, int off, int len) throws IOException {
|
public int read(byte[] b, int off, int len) throws IOException {
|
||||||
|
if (!SocketReadEvent.enabled()) {
|
||||||
|
return implRead(b, off, len);
|
||||||
|
}
|
||||||
|
long start = SocketReadEvent.timestamp();
|
||||||
|
int nbytes = implRead(b, off, len);
|
||||||
|
SocketReadEvent.offer(start, nbytes, parent.getRemoteSocketAddress(), getSoTimeout());
|
||||||
|
return nbytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
private int implRead(byte[] b, int off, int len) throws IOException {
|
||||||
try {
|
try {
|
||||||
return in.read(b, off, len);
|
return in.read(b, off, len);
|
||||||
} catch (SocketTimeoutException e) {
|
} catch (SocketTimeoutException e) {
|
||||||
@ -1105,6 +1114,16 @@ public class Socket implements java.io.Closeable {
|
|||||||
throw e;
|
throw e;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private int getSoTimeout() {
|
||||||
|
try {
|
||||||
|
return parent.getSoTimeout();
|
||||||
|
} catch (SocketException e) {
|
||||||
|
// ignored - avoiding exceptions in jfr event data gathering
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int available() throws IOException {
|
public int available() throws IOException {
|
||||||
return in.available();
|
return in.available();
|
||||||
@ -1169,9 +1188,6 @@ public class Socket implements java.io.Closeable {
|
|||||||
/**
|
/**
|
||||||
* An OutputStream that delegates write operations to an underlying output
|
* An OutputStream that delegates write operations to an underlying output
|
||||||
* stream. The close method is overridden to close the Socket.
|
* stream. The close method is overridden to close the Socket.
|
||||||
*
|
|
||||||
* This class is instrumented by Java Flight Recorder (JFR) to get socket
|
|
||||||
* I/O events.
|
|
||||||
*/
|
*/
|
||||||
private static class SocketOutputStream extends OutputStream {
|
private static class SocketOutputStream extends OutputStream {
|
||||||
private final Socket parent;
|
private final Socket parent;
|
||||||
@ -1187,6 +1203,16 @@ public class Socket implements java.io.Closeable {
|
|||||||
}
|
}
|
||||||
@Override
|
@Override
|
||||||
public void write(byte[] b, int off, int len) throws IOException {
|
public void write(byte[] b, int off, int len) throws IOException {
|
||||||
|
if (!SocketWriteEvent.enabled()) {
|
||||||
|
implWrite(b, off, len);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
long start = SocketWriteEvent.timestamp();
|
||||||
|
implWrite(b, off, len);
|
||||||
|
SocketWriteEvent.offer(start, len, parent.getRemoteSocketAddress());
|
||||||
|
}
|
||||||
|
|
||||||
|
private void implWrite(byte[] b, int off, int len) throws IOException {
|
||||||
try {
|
try {
|
||||||
out.write(b, off, len);
|
out.write(b, off, len);
|
||||||
} catch (InterruptedIOException e) {
|
} catch (InterruptedIOException e) {
|
||||||
|
@ -0,0 +1,134 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2023, 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. Oracle designates this
|
||||||
|
* particular file as subject to the "Classpath" exception as provided
|
||||||
|
* by Oracle in the LICENSE file that accompanied this code.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package jdk.internal.event;
|
||||||
|
|
||||||
|
|
||||||
|
import java.net.InetSocketAddress;
|
||||||
|
import java.net.SocketAddress;
|
||||||
|
import java.net.UnixDomainSocketAddress;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A JFR event for socket read operations. This event is mirrored in
|
||||||
|
* {@code jdk.jfr.events.SocketReadEvent } where the metadata for the event is
|
||||||
|
* provided with annotations. Some of the methods are replaced by generated
|
||||||
|
* methods when jfr is enabled. Note that the order of the arguments of the
|
||||||
|
* {@link #commit(long, long, String, String, int, long, long, boolean)} method
|
||||||
|
* must be the same as the order of the fields.
|
||||||
|
*/
|
||||||
|
public class SocketReadEvent extends Event {
|
||||||
|
|
||||||
|
// THE ORDER OF THE FOLLOWING FIELDS IS IMPORTANT!
|
||||||
|
// The order must match the argument order of the generated commit method.
|
||||||
|
public String host;
|
||||||
|
public String address;
|
||||||
|
public int port;
|
||||||
|
public long timeout;
|
||||||
|
public long bytesRead;
|
||||||
|
public boolean endOfStream;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Actually commit a socket read event. The implementation
|
||||||
|
* of this method is generated automatically if jfr is enabled.
|
||||||
|
* The order of the fields must be the same as the parameters in this method.
|
||||||
|
* {@code commit(..., String, String, int, long, long, boolean)}
|
||||||
|
*
|
||||||
|
* @param start timestamp of the start of the operation
|
||||||
|
* @param duration time in nanoseconds to complete the operation
|
||||||
|
* @param host remote host of the transfer
|
||||||
|
* @param address remote address of the transfer
|
||||||
|
* @param port remote port of the transfer
|
||||||
|
* @param timeout timeout setting for the read
|
||||||
|
* @param bytes number of bytes that were transferred
|
||||||
|
* @param endOfStream has the end of the stream been reached
|
||||||
|
*/
|
||||||
|
public static void commit(long start, long duration, String host, String address, int port, long timeout, long bytes, boolean endOfStream) {
|
||||||
|
// Generated by JFR
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine if an event should be emitted. The duration of the operation
|
||||||
|
* must exceed some threshold in order to commit the event. The implementation
|
||||||
|
* of this method is generated automatically if jfr is enabled.
|
||||||
|
*
|
||||||
|
* @param duration time in nanoseconds to complete the operation
|
||||||
|
* @return true if the event should be commited
|
||||||
|
*/
|
||||||
|
public static boolean shouldCommit(long duration) {
|
||||||
|
// Generated by JFR
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine if this kind of event is enabled. The implementation
|
||||||
|
* of this method is generated automatically if jfr is enabled.
|
||||||
|
*
|
||||||
|
* @return true if socket read events are enabled, false otherwise
|
||||||
|
*/
|
||||||
|
public static boolean enabled() {
|
||||||
|
// Generated by JFR
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fetch the current timestamp in nanoseconds. This method is used
|
||||||
|
* to determine the start and end of an operation. The implementation
|
||||||
|
* of this method is generated automatically if jfr is enabled.
|
||||||
|
*
|
||||||
|
* @return the current timestamp value
|
||||||
|
*/
|
||||||
|
public static long timestamp() {
|
||||||
|
// Generated by JFR
|
||||||
|
return 0L;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helper method to offer the data needed to potentially commit an event.
|
||||||
|
* The duration of the operation is computed using the current
|
||||||
|
* timestamp and the given start time. If the duration is meets
|
||||||
|
* or exceeds the configured value (determined by calling the generated method
|
||||||
|
* {@link #shouldCommit(long)}), an event will be emitted by calling
|
||||||
|
* {@link #commit(long, long, String, String, int, long, long, boolean)}.
|
||||||
|
*
|
||||||
|
* @param start the start time
|
||||||
|
* @param nbytes how many bytes were transferred
|
||||||
|
* @param remote the address of the remote socket
|
||||||
|
* @param timeout maximum time to wait
|
||||||
|
*/
|
||||||
|
public static void offer(long start, long nbytes, SocketAddress remote, long timeout) {
|
||||||
|
long duration = timestamp() - start;
|
||||||
|
if (shouldCommit(duration)) {
|
||||||
|
boolean eof = nbytes < 0 ? true : false;
|
||||||
|
nbytes = nbytes < 0 ? 0 : nbytes;
|
||||||
|
if (remote instanceof InetSocketAddress isa) {
|
||||||
|
commit(start, duration, isa.getHostString(), isa.getAddress().getHostAddress(), isa.getPort(), timeout, nbytes, eof);
|
||||||
|
} else if (remote instanceof UnixDomainSocketAddress udsa) {
|
||||||
|
String path = "[" + udsa.getPath().toString() + "]";
|
||||||
|
commit(start, duration, "Unix domain socket", path, 0, timeout, nbytes, eof);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,126 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2023, 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. Oracle designates this
|
||||||
|
* particular file as subject to the "Classpath" exception as provided
|
||||||
|
* by Oracle in the LICENSE file that accompanied this code.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package jdk.internal.event;
|
||||||
|
|
||||||
|
|
||||||
|
import java.net.InetSocketAddress;
|
||||||
|
import java.net.SocketAddress;
|
||||||
|
import java.net.UnixDomainSocketAddress;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A JFR event for socket write operations. This event is mirrored in
|
||||||
|
* {@code jdk.jfr.events.SocketWriteEvent } where the metadata for the event is
|
||||||
|
* provided with annotations. Some of the methods are replaced by generated
|
||||||
|
* methods when jfr is enabled. Note that the order of the arguments of the
|
||||||
|
* {@link #commit(long, long, String, String, int, long)} method
|
||||||
|
* must be the same as the order of the fields.
|
||||||
|
*/
|
||||||
|
public class SocketWriteEvent extends Event {
|
||||||
|
|
||||||
|
// THE ORDER OF THE FOLLOWING FIELDS IS IMPORTANT!
|
||||||
|
// The order must match the argument order of the generated commit method.
|
||||||
|
public String host;
|
||||||
|
public String address;
|
||||||
|
public int port;
|
||||||
|
public long bytesWritten;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Actually commit a socket write event. This is generated automatically.
|
||||||
|
* The order of the fields must be the same as the parameters in this method.
|
||||||
|
* {@code commit(..., String, String, int, long)}
|
||||||
|
*
|
||||||
|
* @param start timestamp of the start of the operation
|
||||||
|
* @param duration time in nanoseconds to complete the operation
|
||||||
|
* @param host remote host of the transfer
|
||||||
|
* @param address remote address of the transfer
|
||||||
|
* @param port remote port of the transfer
|
||||||
|
* @param bytes number of bytes that were transferred
|
||||||
|
*/
|
||||||
|
public static void commit(long start, long duration, String host, String address, int port, long bytes) {
|
||||||
|
// Generated by JFR
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine if an event should be emitted. The duration of the operation
|
||||||
|
* must exceed some threshold in order to commit the event. The implementation
|
||||||
|
* of this method is generated automatically if jfr is enabled.
|
||||||
|
*
|
||||||
|
* @param duration time in nanoseconds to complete the operation
|
||||||
|
* @return true if the event should be commited
|
||||||
|
*/
|
||||||
|
public static boolean shouldCommit(long duration) {
|
||||||
|
// Generated by JFR
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine if this kind of event is enabled. The implementation
|
||||||
|
* of this method is generated automatically if jfr is enabled.
|
||||||
|
*
|
||||||
|
* @return true if socket write events are enabled, false otherwise
|
||||||
|
*/
|
||||||
|
public static boolean enabled() {
|
||||||
|
// Generated by JFR
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fetch the current timestamp in nanoseconds. This method is used
|
||||||
|
* to determine the start and end of an operation. The implementation
|
||||||
|
* of this method is generated automatically if jfr is enabled.
|
||||||
|
*
|
||||||
|
* @return the current timestamp value
|
||||||
|
*/
|
||||||
|
public static long timestamp() {
|
||||||
|
// Generated by JFR
|
||||||
|
return 0L;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helper method to offer the data needed to potentially commit an event.
|
||||||
|
* The duration of the operation is computed using the current
|
||||||
|
* timestamp and the given start time. If the duration is meets
|
||||||
|
* or exceeds the configured value (determined by calling the generated method
|
||||||
|
* {@link #shouldCommit(long)}), an event will be emitted by calling
|
||||||
|
* {@link #commit(long, long, String, String, int, long)}.
|
||||||
|
*
|
||||||
|
* @param start the start time
|
||||||
|
* @param bytesWritten how many bytes were sent
|
||||||
|
* @param remote the address of the remote socket being written to
|
||||||
|
*/
|
||||||
|
public static void offer(long start, long bytesWritten, SocketAddress remote) {
|
||||||
|
long duration = timestamp() - start;
|
||||||
|
if (shouldCommit(duration)) {
|
||||||
|
long bytes = bytesWritten < 0 ? 0 : bytesWritten;
|
||||||
|
if (remote instanceof InetSocketAddress isa) {
|
||||||
|
commit(start, duration, isa.getHostString(), isa.getAddress().getHostAddress(), isa.getPort(), bytes);
|
||||||
|
} else if (remote instanceof UnixDomainSocketAddress udsa) {
|
||||||
|
String path = "[" + udsa.getPath().toString() + "]";
|
||||||
|
commit(start, duration, "Unix domain socket", path, 0, bytes);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -59,6 +59,8 @@ import static java.net.StandardProtocolFamily.INET;
|
|||||||
import static java.net.StandardProtocolFamily.INET6;
|
import static java.net.StandardProtocolFamily.INET6;
|
||||||
import static java.net.StandardProtocolFamily.UNIX;
|
import static java.net.StandardProtocolFamily.UNIX;
|
||||||
|
|
||||||
|
import jdk.internal.event.SocketReadEvent;
|
||||||
|
import jdk.internal.event.SocketWriteEvent;
|
||||||
import sun.net.ConnectionResetException;
|
import sun.net.ConnectionResetException;
|
||||||
import sun.net.NetHooks;
|
import sun.net.NetHooks;
|
||||||
import sun.net.ext.ExtendedSocketOptions;
|
import sun.net.ext.ExtendedSocketOptions;
|
||||||
@ -401,8 +403,7 @@ class SocketChannelImpl
|
|||||||
throw new SocketException("Connection reset");
|
throw new SocketException("Connection reset");
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
private int implRead(ByteBuffer buf) throws IOException {
|
||||||
public int read(ByteBuffer buf) throws IOException {
|
|
||||||
Objects.requireNonNull(buf);
|
Objects.requireNonNull(buf);
|
||||||
|
|
||||||
readLock.lock();
|
readLock.lock();
|
||||||
@ -443,8 +444,7 @@ class SocketChannelImpl
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
private long implRead(ByteBuffer[] dsts, int offset, int length)
|
||||||
public long read(ByteBuffer[] dsts, int offset, int length)
|
|
||||||
throws IOException
|
throws IOException
|
||||||
{
|
{
|
||||||
Objects.checkFromIndexSize(offset, length, dsts.length);
|
Objects.checkFromIndexSize(offset, length, dsts.length);
|
||||||
@ -487,6 +487,31 @@ class SocketChannelImpl
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int read(ByteBuffer buf) throws IOException {
|
||||||
|
if (!SocketReadEvent.enabled()) {
|
||||||
|
return implRead(buf);
|
||||||
|
}
|
||||||
|
long start = SocketReadEvent.timestamp();
|
||||||
|
int nbytes = implRead(buf);
|
||||||
|
SocketReadEvent.offer(start, nbytes, remoteAddress(), 0);
|
||||||
|
return nbytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public long read(ByteBuffer[] dsts, int offset, int length)
|
||||||
|
throws IOException
|
||||||
|
{
|
||||||
|
if (!SocketReadEvent.enabled()) {
|
||||||
|
return implRead(dsts, offset, length);
|
||||||
|
}
|
||||||
|
long start = SocketReadEvent.timestamp();
|
||||||
|
long nbytes = implRead(dsts, offset, length);
|
||||||
|
SocketReadEvent.offer(start, nbytes, remoteAddress(), 0);
|
||||||
|
return nbytes;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Marks the beginning of a write operation that might block.
|
* Marks the beginning of a write operation that might block.
|
||||||
*
|
*
|
||||||
@ -528,8 +553,7 @@ class SocketChannelImpl
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
private int implWrite(ByteBuffer buf) throws IOException {
|
||||||
public int write(ByteBuffer buf) throws IOException {
|
|
||||||
Objects.requireNonNull(buf);
|
Objects.requireNonNull(buf);
|
||||||
writeLock.lock();
|
writeLock.lock();
|
||||||
try {
|
try {
|
||||||
@ -557,8 +581,7 @@ class SocketChannelImpl
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
private long implWrite(ByteBuffer[] srcs, int offset, int length)
|
||||||
public long write(ByteBuffer[] srcs, int offset, int length)
|
|
||||||
throws IOException
|
throws IOException
|
||||||
{
|
{
|
||||||
Objects.checkFromIndexSize(offset, length, srcs.length);
|
Objects.checkFromIndexSize(offset, length, srcs.length);
|
||||||
@ -589,6 +612,30 @@ class SocketChannelImpl
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int write(ByteBuffer buf) throws IOException {
|
||||||
|
if (!SocketWriteEvent.enabled()) {
|
||||||
|
return implWrite(buf);
|
||||||
|
}
|
||||||
|
long start = SocketWriteEvent.timestamp();
|
||||||
|
int nbytes = implWrite(buf);
|
||||||
|
SocketWriteEvent.offer(start, nbytes, remoteAddress());
|
||||||
|
return nbytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public long write(ByteBuffer[] srcs, int offset, int length)
|
||||||
|
throws IOException
|
||||||
|
{
|
||||||
|
if (!SocketWriteEvent.enabled()) {
|
||||||
|
return implWrite(srcs, offset, length);
|
||||||
|
}
|
||||||
|
long start = SocketWriteEvent.timestamp();
|
||||||
|
long nbytes = implWrite(srcs, offset, length);
|
||||||
|
SocketWriteEvent.offer(start, nbytes, remoteAddress());
|
||||||
|
return nbytes;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Writes a byte of out of band data.
|
* Writes a byte of out of band data.
|
||||||
*/
|
*/
|
||||||
|
@ -28,8 +28,6 @@ import jdk.jfr.internal.JVMSupport;
|
|||||||
import jdk.jfr.internal.event.EventConfiguration;
|
import jdk.jfr.internal.event.EventConfiguration;
|
||||||
|
|
||||||
public final class EventConfigurations {
|
public final class EventConfigurations {
|
||||||
public static final EventConfiguration SOCKET_READ = JVMSupport.getConfiguration(SocketReadEvent.class);
|
|
||||||
public static final EventConfiguration SOCKET_WRITE = JVMSupport.getConfiguration(SocketWriteEvent.class);
|
|
||||||
public static final EventConfiguration FILE_READ = JVMSupport.getConfiguration(FileReadEvent.class);
|
public static final EventConfiguration FILE_READ = JVMSupport.getConfiguration(FileReadEvent.class);
|
||||||
public static final EventConfiguration FILE_WRITE = JVMSupport.getConfiguration(FileWriteEvent.class);
|
public static final EventConfiguration FILE_WRITE = JVMSupport.getConfiguration(FileWriteEvent.class);
|
||||||
public static final EventConfiguration FILE_FORCE = JVMSupport.getConfiguration(FileForceEvent.class);
|
public static final EventConfiguration FILE_FORCE = JVMSupport.getConfiguration(FileForceEvent.class);
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2012, 2022, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2012, 2023, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -31,17 +31,16 @@ import jdk.jfr.Label;
|
|||||||
import jdk.jfr.DataAmount;
|
import jdk.jfr.DataAmount;
|
||||||
import jdk.jfr.Name;
|
import jdk.jfr.Name;
|
||||||
import jdk.jfr.Timespan;
|
import jdk.jfr.Timespan;
|
||||||
|
import jdk.jfr.internal.MirrorEvent;
|
||||||
import jdk.jfr.internal.Type;
|
import jdk.jfr.internal.Type;
|
||||||
|
|
||||||
@Name(Type.EVENT_NAME_PREFIX + "SocketRead")
|
@Name(Type.EVENT_NAME_PREFIX + "SocketRead")
|
||||||
@Label("Socket Read")
|
@Label("Socket Read")
|
||||||
@Category("Java Application")
|
@Category("Java Application")
|
||||||
@Description("Reading data from a socket")
|
@Description("Reading data from a socket")
|
||||||
|
@MirrorEvent(className = "jdk.internal.event.SocketReadEvent")
|
||||||
public final class SocketReadEvent extends AbstractJDKEvent {
|
public final class SocketReadEvent extends AbstractJDKEvent {
|
||||||
|
|
||||||
// The order of these fields must be the same as the parameters in
|
|
||||||
// commit(..., String, String, int, long, long, boolean)
|
|
||||||
|
|
||||||
@Label("Remote Host")
|
@Label("Remote Host")
|
||||||
public String host;
|
public String host;
|
||||||
|
|
||||||
@ -64,7 +63,4 @@ public final class SocketReadEvent extends AbstractJDKEvent {
|
|||||||
@Description("If end of stream was reached")
|
@Description("If end of stream was reached")
|
||||||
public boolean endOfStream;
|
public boolean endOfStream;
|
||||||
|
|
||||||
public static void commit(long start, long duration, String host, String address, int port, long timeout, long byteRead, boolean endOfStream) {
|
|
||||||
// Generated
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2012, 2022, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2012, 2023, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -30,17 +30,16 @@ import jdk.jfr.Description;
|
|||||||
import jdk.jfr.Label;
|
import jdk.jfr.Label;
|
||||||
import jdk.jfr.DataAmount;
|
import jdk.jfr.DataAmount;
|
||||||
import jdk.jfr.Name;
|
import jdk.jfr.Name;
|
||||||
|
import jdk.jfr.internal.MirrorEvent;
|
||||||
import jdk.jfr.internal.Type;
|
import jdk.jfr.internal.Type;
|
||||||
|
|
||||||
@Name(Type.EVENT_NAME_PREFIX + "SocketWrite")
|
@Name(Type.EVENT_NAME_PREFIX + "SocketWrite")
|
||||||
@Label("Socket Write")
|
@Label("Socket Write")
|
||||||
@Category("Java Application")
|
@Category("Java Application")
|
||||||
@Description("Writing data to a socket")
|
@Description("Writing data to a socket")
|
||||||
|
@MirrorEvent(className = "jdk.internal.event.SocketWriteEvent")
|
||||||
public final class SocketWriteEvent extends AbstractJDKEvent {
|
public final class SocketWriteEvent extends AbstractJDKEvent {
|
||||||
|
|
||||||
// The order of these fields must be the same as the parameters in
|
|
||||||
// commit(..., String, String, int, long)
|
|
||||||
|
|
||||||
@Label("Remote Host")
|
@Label("Remote Host")
|
||||||
public String host;
|
public String host;
|
||||||
|
|
||||||
@ -55,7 +54,4 @@ public final class SocketWriteEvent extends AbstractJDKEvent {
|
|||||||
@DataAmount
|
@DataAmount
|
||||||
public long bytesWritten;
|
public long bytesWritten;
|
||||||
|
|
||||||
public static void commit(long start, long duration, String host, String address, int port, long bytes) {
|
|
||||||
// Generated
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -75,6 +75,8 @@ public final class JDKEvents {
|
|||||||
ProcessStartEvent.class,
|
ProcessStartEvent.class,
|
||||||
SecurityPropertyModificationEvent.class,
|
SecurityPropertyModificationEvent.class,
|
||||||
SecurityProviderServiceEvent.class,
|
SecurityProviderServiceEvent.class,
|
||||||
|
SocketReadEvent.class,
|
||||||
|
SocketWriteEvent.class,
|
||||||
ThreadSleepEvent.class,
|
ThreadSleepEvent.class,
|
||||||
TLSHandshakeEvent.class,
|
TLSHandshakeEvent.class,
|
||||||
VirtualThreadStartEvent.class,
|
VirtualThreadStartEvent.class,
|
||||||
@ -100,6 +102,8 @@ public final class JDKEvents {
|
|||||||
jdk.internal.event.ProcessStartEvent.class,
|
jdk.internal.event.ProcessStartEvent.class,
|
||||||
jdk.internal.event.SecurityPropertyModificationEvent.class,
|
jdk.internal.event.SecurityPropertyModificationEvent.class,
|
||||||
jdk.internal.event.SecurityProviderServiceEvent.class,
|
jdk.internal.event.SecurityProviderServiceEvent.class,
|
||||||
|
jdk.internal.event.SocketReadEvent.class,
|
||||||
|
jdk.internal.event.SocketWriteEvent.class,
|
||||||
jdk.internal.event.ThreadSleepEvent.class,
|
jdk.internal.event.ThreadSleepEvent.class,
|
||||||
jdk.internal.event.TLSHandshakeEvent.class,
|
jdk.internal.event.TLSHandshakeEvent.class,
|
||||||
jdk.internal.event.VirtualThreadStartEvent.class,
|
jdk.internal.event.VirtualThreadStartEvent.class,
|
||||||
@ -118,10 +122,7 @@ public final class JDKEvents {
|
|||||||
FileInputStreamInstrumentor.class,
|
FileInputStreamInstrumentor.class,
|
||||||
FileOutputStreamInstrumentor.class,
|
FileOutputStreamInstrumentor.class,
|
||||||
RandomAccessFileInstrumentor.class,
|
RandomAccessFileInstrumentor.class,
|
||||||
FileChannelImplInstrumentor.class,
|
FileChannelImplInstrumentor.class
|
||||||
SocketInputStreamInstrumentor.class,
|
|
||||||
SocketOutputStreamInstrumentor.class,
|
|
||||||
SocketChannelImplInstrumentor.class
|
|
||||||
};
|
};
|
||||||
|
|
||||||
private static final Class<?>[] targetClasses = new Class<?>[instrumentationClasses.length];
|
private static final Class<?>[] targetClasses = new Class<?>[instrumentationClasses.length];
|
||||||
|
@ -1,201 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2013, 2022, 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. Oracle designates this
|
|
||||||
* particular file as subject to the "Classpath" exception as provided
|
|
||||||
* by Oracle in the LICENSE file that accompanied this code.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package jdk.jfr.internal.instrument;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.net.InetSocketAddress;
|
|
||||||
import java.net.SocketAddress;
|
|
||||||
import java.net.UnixDomainSocketAddress;
|
|
||||||
import java.nio.ByteBuffer;
|
|
||||||
import jdk.jfr.events.EventConfigurations;
|
|
||||||
import jdk.jfr.events.SocketReadEvent;
|
|
||||||
import jdk.jfr.events.SocketWriteEvent;
|
|
||||||
import jdk.jfr.internal.event.EventConfiguration;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* See {@link JITracer} for an explanation of this code.
|
|
||||||
*/
|
|
||||||
@JIInstrumentationTarget("sun.nio.ch.SocketChannelImpl")
|
|
||||||
final class SocketChannelImplInstrumentor {
|
|
||||||
|
|
||||||
private SocketChannelImplInstrumentor() {
|
|
||||||
}
|
|
||||||
|
|
||||||
@JIInstrumentationMethod
|
|
||||||
public int read(ByteBuffer dst) throws IOException {
|
|
||||||
EventConfiguration eventConfiguration = EventConfigurations.SOCKET_READ;
|
|
||||||
if (!eventConfiguration.isEnabled()) {
|
|
||||||
return read(dst);
|
|
||||||
}
|
|
||||||
int bytesRead = 0;
|
|
||||||
long start = 0;
|
|
||||||
try {
|
|
||||||
start = EventConfiguration.timestamp();;
|
|
||||||
bytesRead = read(dst);
|
|
||||||
} finally {
|
|
||||||
long duration = EventConfiguration.timestamp() - start;
|
|
||||||
if (eventConfiguration.shouldCommit(duration)) {
|
|
||||||
SocketAddress remoteAddress = getRemoteAddress();
|
|
||||||
if (remoteAddress instanceof InetSocketAddress isa) {
|
|
||||||
String hostString = isa.getAddress().toString();
|
|
||||||
int delimiterIndex = hostString.lastIndexOf('/');
|
|
||||||
|
|
||||||
String host = hostString.substring(0, delimiterIndex);
|
|
||||||
String address = hostString.substring(delimiterIndex + 1);
|
|
||||||
int port = isa.getPort();
|
|
||||||
if (bytesRead < 0) {
|
|
||||||
SocketReadEvent.commit(start, duration, host, address, port, 0, 0L, true);
|
|
||||||
} else {
|
|
||||||
SocketReadEvent.commit(start, duration, host, address, port, 0, bytesRead, false);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
UnixDomainSocketAddress udsa = (UnixDomainSocketAddress) remoteAddress;
|
|
||||||
String path = "[" + udsa.getPath().toString() + "]";
|
|
||||||
if (bytesRead < 0) {
|
|
||||||
SocketReadEvent.commit(start, duration, "Unix domain socket", path, 0, 0, 0L, true);
|
|
||||||
} else {
|
|
||||||
SocketReadEvent.commit(start, duration, "Unix domain socket", path, 0, 0, bytesRead, false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return bytesRead;
|
|
||||||
}
|
|
||||||
|
|
||||||
@JIInstrumentationMethod
|
|
||||||
public long read(ByteBuffer[] dsts, int offset, int length) throws IOException {
|
|
||||||
EventConfiguration eventConfiguration = EventConfigurations.SOCKET_READ;
|
|
||||||
if (!eventConfiguration.isEnabled()) {
|
|
||||||
return read(dsts, offset, length);
|
|
||||||
}
|
|
||||||
long bytesRead = 0;
|
|
||||||
long start = 0;
|
|
||||||
try {
|
|
||||||
start = EventConfiguration.timestamp();
|
|
||||||
bytesRead = read(dsts, offset, length);
|
|
||||||
} finally {
|
|
||||||
long duration = EventConfiguration.timestamp() - start;
|
|
||||||
if (eventConfiguration.shouldCommit(duration)) {
|
|
||||||
SocketAddress remoteAddress = getRemoteAddress();
|
|
||||||
if (remoteAddress instanceof InetSocketAddress isa) {
|
|
||||||
String hostString = isa.getAddress().toString();
|
|
||||||
int delimiterIndex = hostString.lastIndexOf('/');
|
|
||||||
|
|
||||||
String host = hostString.substring(0, delimiterIndex);
|
|
||||||
String address = hostString.substring(delimiterIndex + 1);
|
|
||||||
int port = isa.getPort();
|
|
||||||
if (bytesRead < 0) {
|
|
||||||
SocketReadEvent.commit(start, duration, host, address, port, 0, 0L, true);
|
|
||||||
} else {
|
|
||||||
SocketReadEvent.commit(start, duration, host, address, port, 0, bytesRead, false);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
UnixDomainSocketAddress udsa = (UnixDomainSocketAddress) remoteAddress;
|
|
||||||
String path = "[" + udsa.getPath().toString() + "]";
|
|
||||||
if (bytesRead < 0) {
|
|
||||||
SocketReadEvent.commit(start, duration, "Unix domain socket", path, 0, 0, 0L, true);
|
|
||||||
} else {
|
|
||||||
SocketReadEvent.commit(start, duration, "Unix domain socket", path, 0, 0, bytesRead, false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return bytesRead;
|
|
||||||
}
|
|
||||||
|
|
||||||
@JIInstrumentationMethod
|
|
||||||
public int write(ByteBuffer buf) throws IOException {
|
|
||||||
EventConfiguration eventConfiguration = EventConfigurations.SOCKET_WRITE;
|
|
||||||
if (!eventConfiguration.isEnabled()) {
|
|
||||||
return write(buf);
|
|
||||||
}
|
|
||||||
int bytesWritten = 0;
|
|
||||||
long start = 0;
|
|
||||||
try {
|
|
||||||
start = EventConfiguration.timestamp();
|
|
||||||
bytesWritten = write(buf);
|
|
||||||
} finally {
|
|
||||||
long duration = EventConfiguration.timestamp() - start;
|
|
||||||
if (eventConfiguration.shouldCommit(duration)) {
|
|
||||||
long bytes = bytesWritten < 0 ? 0 : bytesWritten;
|
|
||||||
SocketAddress remoteAddress = getRemoteAddress();
|
|
||||||
if (remoteAddress instanceof InetSocketAddress isa) {
|
|
||||||
String hostString = isa.getAddress().toString();
|
|
||||||
int delimiterIndex = hostString.lastIndexOf('/');
|
|
||||||
|
|
||||||
String host = hostString.substring(0, delimiterIndex);
|
|
||||||
String address = hostString.substring(delimiterIndex + 1);
|
|
||||||
int port = isa.getPort();
|
|
||||||
SocketWriteEvent.commit(start, duration, host, address, port, bytes);
|
|
||||||
} else {
|
|
||||||
UnixDomainSocketAddress udsa = (UnixDomainSocketAddress) remoteAddress;
|
|
||||||
String path = "[" + udsa.getPath().toString() + "]";
|
|
||||||
SocketWriteEvent.commit(start, duration, "Unix domain socket", path, 0, bytes);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return bytesWritten;
|
|
||||||
}
|
|
||||||
|
|
||||||
public SocketAddress getRemoteAddress() throws IOException {
|
|
||||||
// gets replaced by call to instrumented class
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@JIInstrumentationMethod
|
|
||||||
public long write(ByteBuffer[] srcs, int offset, int length) throws IOException {
|
|
||||||
EventConfiguration eventConfiguration = EventConfigurations.SOCKET_WRITE;
|
|
||||||
if (!eventConfiguration.isEnabled()) {
|
|
||||||
return write(srcs, offset, length);
|
|
||||||
}
|
|
||||||
long bytesWritten = 0;
|
|
||||||
long start = 0;
|
|
||||||
try {
|
|
||||||
start = EventConfiguration.timestamp();
|
|
||||||
bytesWritten = write(srcs, offset, length);
|
|
||||||
} finally {
|
|
||||||
long duration = EventConfiguration.timestamp() - start;
|
|
||||||
if (eventConfiguration.shouldCommit(duration)) {
|
|
||||||
long bytes = bytesWritten < 0 ? 0 : bytesWritten;
|
|
||||||
SocketAddress remoteAddress = getRemoteAddress();
|
|
||||||
if (remoteAddress instanceof InetSocketAddress isa) {
|
|
||||||
String hostString = isa.getAddress().toString();
|
|
||||||
int delimiterIndex = hostString.lastIndexOf('/');
|
|
||||||
|
|
||||||
String host = hostString.substring(0, delimiterIndex);
|
|
||||||
String address = hostString.substring(delimiterIndex + 1);
|
|
||||||
int port = isa.getPort();
|
|
||||||
SocketWriteEvent.commit(start, duration, host, address, port, bytes);
|
|
||||||
} else {
|
|
||||||
UnixDomainSocketAddress udsa = (UnixDomainSocketAddress) remoteAddress;
|
|
||||||
String path = "[" + udsa.getPath().toString() + "]";
|
|
||||||
SocketWriteEvent.commit(start, duration, "Unix domain socket", path, 0, bytes);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return bytesWritten;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,76 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2013, 2022, 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. Oracle designates this
|
|
||||||
* particular file as subject to the "Classpath" exception as provided
|
|
||||||
* by Oracle in the LICENSE file that accompanied this code.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package jdk.jfr.internal.instrument;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.net.InetAddress;
|
|
||||||
import java.net.Socket;
|
|
||||||
|
|
||||||
import jdk.jfr.events.EventConfigurations;
|
|
||||||
import jdk.jfr.events.SocketReadEvent;
|
|
||||||
import jdk.jfr.internal.event.EventConfiguration;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* See {@link JITracer} for an explanation of this code.
|
|
||||||
*/
|
|
||||||
@JIInstrumentationTarget("java.net.Socket$SocketInputStream")
|
|
||||||
final class SocketInputStreamInstrumentor {
|
|
||||||
|
|
||||||
private SocketInputStreamInstrumentor() {
|
|
||||||
}
|
|
||||||
|
|
||||||
@JIInstrumentationMethod
|
|
||||||
public int read(byte b[], int off, int length) throws IOException {
|
|
||||||
EventConfiguration eventConfiguration = EventConfigurations.SOCKET_READ;
|
|
||||||
if (!eventConfiguration.isEnabled()) {
|
|
||||||
return read(b, off, length);
|
|
||||||
}
|
|
||||||
int bytesRead = 0;
|
|
||||||
long start = 0;
|
|
||||||
try {
|
|
||||||
start = EventConfiguration.timestamp();
|
|
||||||
bytesRead = read(b, off, length);
|
|
||||||
} finally {
|
|
||||||
long duration = EventConfiguration.timestamp() - start;
|
|
||||||
if (eventConfiguration.shouldCommit(duration)) {
|
|
||||||
InetAddress remote = parent.getInetAddress();
|
|
||||||
String host = remote.getHostName();
|
|
||||||
String address = remote.getHostAddress();
|
|
||||||
int port = parent.getPort();
|
|
||||||
int timeout = parent.getSoTimeout();
|
|
||||||
if (bytesRead < 0) {
|
|
||||||
SocketReadEvent.commit(start, duration, host, address, port, timeout, 0L, true);
|
|
||||||
} else {
|
|
||||||
SocketReadEvent.commit(start, duration, host, address, port, timeout, bytesRead, false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return bytesRead;
|
|
||||||
}
|
|
||||||
|
|
||||||
// private field in java.net.Socket$SocketInputStream
|
|
||||||
private Socket parent;
|
|
||||||
}
|
|
@ -1,75 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2013, 2022, 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. Oracle designates this
|
|
||||||
* particular file as subject to the "Classpath" exception as provided
|
|
||||||
* by Oracle in the LICENSE file that accompanied this code.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package jdk.jfr.internal.instrument;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.net.InetAddress;
|
|
||||||
import java.net.Socket;
|
|
||||||
|
|
||||||
import jdk.jfr.events.EventConfigurations;
|
|
||||||
import jdk.jfr.events.SocketWriteEvent;
|
|
||||||
import jdk.jfr.internal.event.EventConfiguration;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* See {@link JITracer} for an explanation of this code.
|
|
||||||
*/
|
|
||||||
@JIInstrumentationTarget("java.net.Socket$SocketOutputStream")
|
|
||||||
final class SocketOutputStreamInstrumentor {
|
|
||||||
|
|
||||||
private SocketOutputStreamInstrumentor() {
|
|
||||||
}
|
|
||||||
|
|
||||||
@JIInstrumentationMethod
|
|
||||||
public void write(byte b[], int off, int len) throws IOException {
|
|
||||||
EventConfiguration eventConfiguration = EventConfigurations.SOCKET_WRITE;
|
|
||||||
if (!eventConfiguration.isEnabled()) {
|
|
||||||
write(b, off, len);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
int bytesWritten = 0;
|
|
||||||
long start = 0;
|
|
||||||
try {
|
|
||||||
start = EventConfiguration.timestamp();
|
|
||||||
write(b, off, len);
|
|
||||||
bytesWritten = len;
|
|
||||||
} finally {
|
|
||||||
long duration = EventConfiguration.timestamp() - start;
|
|
||||||
if (eventConfiguration.shouldCommit(duration)) {
|
|
||||||
InetAddress remote = parent.getInetAddress();
|
|
||||||
SocketWriteEvent.commit(
|
|
||||||
start,
|
|
||||||
duration,
|
|
||||||
remote.getHostName(),
|
|
||||||
remote.getHostAddress(),
|
|
||||||
parent.getPort(),
|
|
||||||
bytesWritten);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// private field in java.net.Socket$SocketOutputStream
|
|
||||||
private Socket parent;
|
|
||||||
}
|
|
167
test/micro/org/openjdk/bench/java/net/SocketEventOverhead.java
Normal file
167
test/micro/org/openjdk/bench/java/net/SocketEventOverhead.java
Normal file
@ -0,0 +1,167 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2023, 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.openjdk.bench.java.net;
|
||||||
|
|
||||||
|
import jdk.internal.event.SocketReadEvent;
|
||||||
|
import jdk.internal.event.SocketWriteEvent;
|
||||||
|
import org.openjdk.jmh.annotations.*;
|
||||||
|
import org.openjdk.jmh.runner.Runner;
|
||||||
|
import org.openjdk.jmh.runner.options.Options;
|
||||||
|
import org.openjdk.jmh.runner.options.OptionsBuilder;
|
||||||
|
|
||||||
|
import java.net.InetSocketAddress;
|
||||||
|
import java.net.SocketAddress;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test the overhead of the handling jfr events SocketReadEvent and
|
||||||
|
* SocketWriteEvent without the latencies of the actual I/O code.
|
||||||
|
*/
|
||||||
|
@BenchmarkMode(Mode.AverageTime)
|
||||||
|
@OutputTimeUnit(TimeUnit.NANOSECONDS)
|
||||||
|
@Warmup(iterations = 10, time = 1, timeUnit = TimeUnit.SECONDS)
|
||||||
|
@Measurement(iterations = 5, time = 2, timeUnit = TimeUnit.SECONDS)
|
||||||
|
@State(Scope.Thread)
|
||||||
|
public class SocketEventOverhead {
|
||||||
|
|
||||||
|
@Fork(value = 1, jvmArgsAppend = {
|
||||||
|
"--add-exports",
|
||||||
|
"java.base/jdk.internal.event=ALL-UNNAMED" })
|
||||||
|
@Benchmark
|
||||||
|
public int socketWriteJFRDisabled(SkeletonFixture fixture) {
|
||||||
|
return fixture.write();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Fork(value = 1, jvmArgsAppend = {
|
||||||
|
"--add-exports",
|
||||||
|
"java.base/jdk.internal.event=ALL-UNNAMED",
|
||||||
|
"-XX:StartFlightRecording:jdk.SocketWrite#enabled=false"})
|
||||||
|
@Benchmark
|
||||||
|
public int socketWriteJFREnabledEventDisabled(SkeletonFixture fixture) {
|
||||||
|
return fixture.write();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Fork(value = 1, jvmArgsAppend = {
|
||||||
|
"--add-exports",
|
||||||
|
"java.base/jdk.internal.event=ALL-UNNAMED",
|
||||||
|
"-XX:StartFlightRecording:jdk.SocketWrite#enabled=true,jdk.SocketWrite#threshold=1s"})
|
||||||
|
@Benchmark
|
||||||
|
public int socketWriteJFREnabledEventNotEmitted(SkeletonFixture fixture) {
|
||||||
|
return fixture.write();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Fork(value = 1, jvmArgsAppend = {
|
||||||
|
"--add-exports","java.base/jdk.internal.event=ALL-UNNAMED",
|
||||||
|
"-XX:StartFlightRecording:jdk.SocketWrite#enabled=true,jdk.SocketWrite#threshold=0ms,disk=false,jdk.SocketWrite#stackTrace=false"})
|
||||||
|
@Benchmark
|
||||||
|
public int socketWriteJFREnabledEventEmitted(SkeletonFixture fixture) {
|
||||||
|
return fixture.write();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Fork(value = 1, jvmArgsAppend = {
|
||||||
|
"--add-exports",
|
||||||
|
"java.base/jdk.internal.event=ALL-UNNAMED" })
|
||||||
|
@Benchmark
|
||||||
|
public int socketReadJFRDisabled(SkeletonFixture fixture) {
|
||||||
|
return fixture.read();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Fork(value = 1, jvmArgsAppend = {
|
||||||
|
"--add-exports",
|
||||||
|
"java.base/jdk.internal.event=ALL-UNNAMED",
|
||||||
|
"-XX:StartFlightRecording:jdk.SocketRead#enabled=false"})
|
||||||
|
@Benchmark
|
||||||
|
public int socketReadJFREnabledEventDisabled(SkeletonFixture fixture) {
|
||||||
|
return fixture.read();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Fork(value = 1, jvmArgsAppend = {
|
||||||
|
"--add-exports",
|
||||||
|
"java.base/jdk.internal.event=ALL-UNNAMED",
|
||||||
|
"-XX:StartFlightRecording:jdk.SocketRead#enabled=true,jdk.SocketRead#threshold=1s"})
|
||||||
|
@Benchmark
|
||||||
|
public int socketReadJFREnabledEventNotEmitted(SkeletonFixture fixture) {
|
||||||
|
return fixture.read();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Fork(value = 1, jvmArgsAppend = {
|
||||||
|
"--add-exports","java.base/jdk.internal.event=ALL-UNNAMED",
|
||||||
|
"-XX:StartFlightRecording:jdk.SocketRead#enabled=true,jdk.SocketRead#threshold=0ms,disk=false,jdk.SocketRead#stackTrace=false"})
|
||||||
|
@Benchmark
|
||||||
|
public int socketReadJFREnabledEventEmitted(SkeletonFixture fixture) {
|
||||||
|
return fixture.read();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fixture with fake read/write operations that have only the JFR event
|
||||||
|
* boilerplate code for managing jfr events. No actual transfer is done
|
||||||
|
* to eliminate the I/O portion and measure the overhead of JFR event
|
||||||
|
* handling in it's various states.
|
||||||
|
*/
|
||||||
|
@State(Scope.Thread)
|
||||||
|
public static class SkeletonFixture {
|
||||||
|
|
||||||
|
private final InetSocketAddress remote = new InetSocketAddress("localhost",5000);
|
||||||
|
|
||||||
|
public SocketAddress getRemoteAddress() {
|
||||||
|
return remote;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int write() {
|
||||||
|
if (! SocketWriteEvent.enabled()) {
|
||||||
|
return write0();
|
||||||
|
}
|
||||||
|
int nbytes = 0;
|
||||||
|
long start = SocketWriteEvent.timestamp();
|
||||||
|
try {
|
||||||
|
nbytes = write0();
|
||||||
|
} finally {
|
||||||
|
SocketWriteEvent.offer(start, nbytes, getRemoteAddress());
|
||||||
|
}
|
||||||
|
return nbytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
private int write0() {
|
||||||
|
return 1024;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int read() {
|
||||||
|
if (! SocketReadEvent.enabled()) {
|
||||||
|
return read0();
|
||||||
|
}
|
||||||
|
int nbytes = 0;
|
||||||
|
long start = SocketReadEvent.timestamp();
|
||||||
|
try {
|
||||||
|
nbytes = read0();
|
||||||
|
} finally {
|
||||||
|
SocketReadEvent.offer(start, nbytes, getRemoteAddress(), 0);
|
||||||
|
}
|
||||||
|
return nbytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
private int read0() {
|
||||||
|
return 1024;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user