📄 tcptransporthelper.java
字号:
/*
* Created on Nov 1, 2005
* Created by Alon Rohter
* Copyright (C) 2005, 2006 Aelitis, All Rights Reserved.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
* This program 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 for more details.
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* AELITIS, SAS au capital de 46,603.30 euros
* 8 Allee Lenotre, La Grille Royale, 78600 Le Mesnil le Roi, France.
*
*/
package com.aelitis.azureus.core.networkmanager.impl.tcp;
import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.SocketChannel;
import java.util.*;
import org.gudy.azureus2.core3.logging.*;
import org.gudy.azureus2.core3.util.*;
import com.aelitis.azureus.core.networkmanager.VirtualChannelSelector;
import com.aelitis.azureus.core.networkmanager.VirtualChannelSelector.VirtualSelectorListener;
import com.aelitis.azureus.core.networkmanager.impl.TransportHelper;
/**
*
*/
public class
TCPTransportHelper
implements TransportHelper
{
public static final int READ_TIMEOUT = 10*1000;
public static final int CONNECT_TIMEOUT = 60*1000;
private static boolean enable_efficient_io = !Constants.JAVA_VERSION.startsWith("1.4");
private final SocketChannel channel;
private Map user_data;
public TCPTransportHelper( SocketChannel _channel ) {
channel = _channel;
}
public InetSocketAddress
getAddress()
{
return( new InetSocketAddress( channel.socket().getInetAddress(), channel.socket().getPort()));
}
public String
getName()
{
// default is TCP so don't clutter up views with this info
return( "" );
}
public boolean
minimiseOverheads()
{
return( false );
}
public int
getConnectTimeout()
{
return( CONNECT_TIMEOUT );
}
public int
getReadTimeout()
{
return( READ_TIMEOUT );
}
public int write( ByteBuffer buffer, boolean partial_write ) throws IOException {
if( channel == null ) {
Debug.out( "channel == null" );
return 0;
}
return( channel.write( buffer ));
}
public long write( ByteBuffer[] buffers, int array_offset, int length ) throws IOException {
if( channel == null ) {
Debug.out( "channel == null" );
return 0;
}
if( enable_efficient_io ) {
try {
return channel.write( buffers, array_offset, length );
}
catch( IOException ioe ) {
//a bug only fixed in Tiger (1.5 series):
//http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=4854354
String msg = ioe.getMessage();
if( msg != null && msg.equals( "A non-blocking socket operation could not be completed immediately" ) ) {
enable_efficient_io = false;
Logger.log(new LogAlert(LogAlert.UNREPEATABLE, LogAlert.AT_WARNING,
"WARNING: Multi-buffer socket write failed; "
+ "switching to single-buffer mode.\n"
+ "Upgrade to JRE 1.5 (5.0) series to fix this problem!"));
}
throw ioe;
}
}
//single-buffer mode
long written_sofar = 0;
for( int i=array_offset; i < (array_offset + length); i++ ) {
int data_length = buffers[ i ].remaining();
int written = channel.write( buffers[ i ] );
written_sofar += written;
if( written < data_length ) {
break;
}
}
return written_sofar;
}
public int read( ByteBuffer buffer ) throws IOException {
if( channel == null ) {
Debug.out( "channel == null" );
return 0;
}
return( channel.read( buffer ));
}
public long read( ByteBuffer[] buffers, int array_offset, int length ) throws IOException {
if( channel == null ) {
Debug.out( "channel == null" );
return 0;
}
if( buffers == null ) {
Debug.out( "read: buffers == null" );
return 0;
}
long bytes_read = 0;
if( enable_efficient_io ) {
try{
bytes_read = channel.read( buffers, array_offset, length );
}
catch( IOException ioe ) {
//a bug only fixed in Tiger (1.5 series):
//http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=4854354
String msg = ioe.getMessage();
if( msg != null && msg.equals( "A non-blocking socket operation could not be completed immediately" ) ) {
enable_efficient_io = false;
Logger.log(new LogAlert(LogAlert.UNREPEATABLE, LogAlert.AT_WARNING,
"WARNING: Multi-buffer socket read failed; switching to single-buffer mode.\n"
+ "Upgrade to JRE 1.5 (5.0) series to fix this problem!"));
}
throw ioe;
}
}
else {
//single-buffer mode
for( int i=array_offset; i < (array_offset + length); i++ ) {
int data_length = buffers[ i ].remaining();
int read = channel.read( buffers[ i ] );
bytes_read += read;
if( read < data_length ) {
break;
}
}
}
if( bytes_read < 0 ) {
throw new IOException( "end of stream on socket read" );
}
return bytes_read;
}
public void
registerForReadSelects(
final selectListener listener,
Object attachment )
{
TCPNetworkManager.getSingleton().getReadSelector().register(
channel,
new VirtualSelectorListener()
{
public boolean
selectSuccess(
VirtualChannelSelector selector,
SocketChannel sc,
Object attachment )
{
return( listener.selectSuccess( TCPTransportHelper.this, attachment ));
}
public void
selectFailure(
VirtualChannelSelector selector,
SocketChannel sc,
Object attachment,
Throwable msg)
{
listener.selectFailure( TCPTransportHelper.this, attachment, msg );
}
},
attachment );
}
public void
registerForWriteSelects(
final selectListener listener,
Object attachment )
{
TCPNetworkManager.getSingleton().getWriteSelector().register(
channel,
new VirtualSelectorListener()
{
public boolean
selectSuccess(
VirtualChannelSelector selector,
SocketChannel sc,
Object attachment )
{
return( listener.selectSuccess( TCPTransportHelper.this, attachment ));
}
public void
selectFailure(
VirtualChannelSelector selector,
SocketChannel sc,
Object attachment,
Throwable msg)
{
listener.selectFailure( TCPTransportHelper.this, attachment, msg );
}
},
attachment );
}
public void
cancelReadSelects()
{
TCPNetworkManager.getSingleton().getReadSelector().cancel( channel );
}
public void
cancelWriteSelects()
{
TCPNetworkManager.getSingleton().getWriteSelector().cancel( channel );
}
public void
resumeReadSelects()
{
TCPNetworkManager.getSingleton().getReadSelector().resumeSelects( channel );
}
public void
resumeWriteSelects()
{
TCPNetworkManager.getSingleton().getWriteSelector().resumeSelects( channel );
}
public void
pauseReadSelects()
{
TCPNetworkManager.getSingleton().getReadSelector().pauseSelects( channel );
}
public void
pauseWriteSelects()
{
TCPNetworkManager.getSingleton().getWriteSelector().pauseSelects( channel );
}
public void
close( String reason )
{
TCPNetworkManager.getSingleton().getReadSelector().cancel( channel );
TCPNetworkManager.getSingleton().getWriteSelector().cancel( channel );
TCPNetworkManager.getSingleton().getConnectDisconnectManager().closeConnection( channel );
}
public void
failed(
Throwable reason )
{
close( Debug.getNestedExceptionMessage( reason ));
}
public SocketChannel getSocketChannel(){ return channel; }
public synchronized void
setUserData(
Object key,
Object data )
{
if ( user_data == null ){
user_data = new HashMap();
}
user_data.put( key, data );
}
public synchronized Object
getUserData(
Object key )
{
if ( user_data == null ){
return(null);
}
return( user_data.get( key ));
}
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -