📄 pepeercontrolimpl.java
字号:
/*
* Created by Olivier Chalouhi
* Modified Apr 13, 2004 by Alon Rohter
* Heavily modified Sep 2005 by Joseph Bridgewater
* Copyright (C) 2004, 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 org.gudy.azureus2.core3.peer.impl.control;
import java.nio.ByteBuffer;
import java.util.*;
import org.gudy.azureus2.core3.config.*;
import org.gudy.azureus2.core3.disk.*;
import org.gudy.azureus2.core3.ipfilter.*;
import org.gudy.azureus2.core3.logging.*;
import org.gudy.azureus2.core3.peer.*;
import org.gudy.azureus2.core3.peer.impl.*;
import org.gudy.azureus2.core3.peer.util.*;
import org.gudy.azureus2.core3.torrent.TOTorrentException;
import org.gudy.azureus2.core3.tracker.client.*;
import org.gudy.azureus2.core3.util.*;
import com.aelitis.azureus.core.networkmanager.LimitedRateGroup;
import com.aelitis.azureus.core.networkmanager.impl.ConnectDisconnectManager;
import com.aelitis.azureus.core.peermanager.PeerManager;
import com.aelitis.azureus.core.peermanager.control.*;
import com.aelitis.azureus.core.peermanager.peerdb.*;
import com.aelitis.azureus.core.peermanager.piecepicker.*;
import com.aelitis.azureus.core.peermanager.unchoker.*;
/**
* manages all peer transports for a torrent
*
* @author MjrTom
* 2005/Oct/08: Numerous changes for new piece-picking. Also
* a few optimizations and multi-thread cleanups
* 2006/Jan/02: refactoring piece picking related code
*/
public class
PEPeerControlImpl
extends LogRelation
implements PEPeerControl, ParameterListener, DiskManagerWriteRequestListener, PeerControlInstance,
DiskManagerCheckRequestListener, IPFilterListener
{
private static final LogIDs LOGID = LogIDs.PEER;
private static final int WARNINGS_LIMIT = 2;
private static final int CHECK_REASON_DOWNLOADED = 1;
private static final int CHECK_REASON_COMPLETE = 2;
private static final int CHECK_REASON_SCAN = 3;
private static boolean disconnect_seeds_when_seeding = COConfigurationManager.getBooleanParameter("Disconnect Seed", true);
private static IpFilter ip_filter = IpFilterManagerFactory.getSingleton().getIPFilter();
private volatile boolean is_running = false;
private volatile ArrayList peer_transports_cow = new ArrayList(); // Copy on write!
private AEMonitor peer_transports_mon = new AEMonitor( "PEPeerControl:PT");
protected PEPeerManagerAdapter adapter;
private DiskManager disk_mgr;
private DiskManagerPiece[] dm_pieces;
private PiecePicker piecePicker;
private long lastNeededUndonePieceChange;
private boolean seeding_mode;
private boolean restart_initiated;
protected int _nbPieces =-1; //how many pieces in the torrent
private PEPieceImpl[] pePieces; //pieces that are currently in progress
private PeerIdentityDataID _hash;
private byte[] _myPeerId;
protected PEPeerManagerStats _stats;
//private final TRTrackerAnnouncer _tracker;
// private int _maxUploads;
private int _seeds, _peers,_remotes;
private long last_remote_time;
private long _timeStarted;
private long _timeStartedSeeding = -1;
private long _timeFinished;
private Average _averageReceptionSpeed;
private long mainloop_loop_count;
private static final int MAINLOOP_ONE_SECOND_INTERVAL = 1000 / PeerControlScheduler.SCHEDULE_PERIOD_MILLIS;
private static final int MAINLOOP_FIVE_SECOND_INTERVAL = MAINLOOP_ONE_SECOND_INTERVAL * 5;
private static final int MAINLOOP_TEN_SECOND_INTERVAL = MAINLOOP_ONE_SECOND_INTERVAL * 10;
private static final int MAINLOOP_THIRTY_SECOND_INTERVAL = MAINLOOP_ONE_SECOND_INTERVAL * 30;
private static final int MAINLOOP_SIXTY_SECOND_INTERVAL = MAINLOOP_ONE_SECOND_INTERVAL * 60;
private static final int MAINLOOP_TEN_MINUTE_INTERVAL = MAINLOOP_SIXTY_SECOND_INTERVAL * 10;
private volatile ArrayList peer_manager_listeners_cow = new ArrayList(); //copy on write
private List piece_check_result_list = new ArrayList();
private AEMonitor piece_check_result_list_mon = new AEMonitor( "PEPeerControl:PCRL");
private boolean superSeedMode;
private int superSeedModeCurrentPiece;
private int superSeedModeNumberOfAnnounces;
private SuperSeedPiece[] superSeedPieces;
private AEMonitor this_mon = new AEMonitor( "PEPeerControl");
private long ip_filter_last_update_time;
private Map user_data;
private Unchoker unchoker;
private PeerDatabase peer_database;
private int next_rescan_piece = -1;
private long rescan_piece_time = -1;
private long last_eta;
private long last_eta_calculation;
private final LimitedRateGroup upload_limited_rate_group = new LimitedRateGroup() {
public int getRateLimitBytesPerSecond() {
return adapter.getUploadRateLimitBytesPerSecond();
}
};
private final LimitedRateGroup download_limited_rate_group = new LimitedRateGroup() {
public int getRateLimitBytesPerSecond() {
return adapter.getDownloadRateLimitBytesPerSecond();
}
};
public
PEPeerControlImpl(
byte[] _peer_id,
PEPeerManagerAdapter _adapter,
DiskManager diskManager)
{
_myPeerId = _peer_id;
adapter = _adapter;
disk_mgr = diskManager;
_nbPieces =disk_mgr.getNbPieces();
piecePicker = PiecePickerFactory.create( this );
COConfigurationManager.addParameterListener("Ip Filter Enabled", this);
COConfigurationManager.addParameterListener( "Disconnect Seed", this );
ip_filter.addListener( this );
}
public void
start()
{
// This torrent Hash
try
{
_hash =PeerIdentityManager.createDataID(disk_mgr.getTorrent().getHash());
} catch (TOTorrentException e)
{
// this should never happen
Debug.printStackTrace(e);
_hash =PeerIdentityManager.createDataID(new byte[20]);
}
// setup the diskManager
dm_pieces =disk_mgr.getPieces();
// the recovered active pieces
pePieces =new PEPieceImpl[_nbPieces];
for (int i =0; i <_nbPieces; i++ )
{
final DiskManagerPiece dmPiece =dm_pieces[i];
if (!dmPiece.isDone() &&dmPiece.getNbWritten() >0)
{
addPiece(new PEPieceImpl(this, dmPiece, 0), i);
}
}
// The peer connections
peer_transports_cow =new ArrayList();
// BtManager is threaded, this variable represents the
// current loop iteration. It's used by some components only called
// at some specific times.
mainloop_loop_count =0;
// The current tracker state
// this could be start or update
_averageReceptionSpeed =Average.getInstance(1000, 30);
// the stats
_stats =new PEPeerManagerStatsImpl(this);
superSeedMode =(COConfigurationManager.getBooleanParameter("Use Super Seeding") &&this.getRemaining() ==0);
superSeedModeCurrentPiece =0;
if (superSeedMode)
{
initialiseSuperSeedMode();
}
peer_database =PeerDatabaseFactory.createPeerDatabase();
// register as legacy controller
PeerManager.getSingleton().registerLegacyManager(this);
// initial check on finished state - future checks are driven by piece check results
// Moved out of mainLoop() so that it runs immediately, possibly changing
// the state to seeding.
checkFinished(true);
PeerControlSchedulerFactory.getSingleton().register(this);
lastNeededUndonePieceChange =Long.MIN_VALUE;
_timeStarted =SystemTime.getCurrentTime();
is_running =true;
}
public void stopAll()
{
is_running =false;
PeerControlSchedulerFactory.getSingleton().unregister(this);
peer_database =null;
// remove legacy controller registration
PeerManager.getSingleton().deregisterLegacyManager(this);
closeAndRemoveAllPeers("download stopped", false);
// clear pieces
for (int i =0; i <_nbPieces; i++ )
{
if (pePieces[i] !=null)
removePiece(pePieces[i], i);
}
// 5. Remove listeners
COConfigurationManager.removeParameterListener("Ip Filter Enabled", this);
COConfigurationManager.removeParameterListener("Disconnect Seed", this);
ip_filter.removeListener(this);
}
public DiskManager getDiskManager() { return disk_mgr; }
public PiecePicker getPiecePicker()
{
return piecePicker;
}
public PEPeerManagerAdapter getAdapter(){ return( adapter ); }
public String getDisplayName(){ return( adapter.getDisplayName()); }
public void
schedule()
{
try {
updateTrackerAnnounceInterval();
doConnectionChecks();
processPieceChecks();
checkCompletedPieces(); //check to see if we've completed anything else
updateStats();
checkInterested(); // see if need to recheck Interested on all peers
piecePicker.updateAvailability();
boolean forcenoseeds = disconnect_seeds_when_seeding;
if (!seeding_mode)
{ // if we're not finished
checkRequests(); //check the requests
// if we have no downloadable pieces (due to "do not download") then
// we disconnect seeds and avoid calling these methods to save CPU.
forcenoseeds =!piecePicker.checkDownloadPossible(); //download blocks if possible
checkRescan();
checkSpeedAndReserved();
}
checkSeeds( forcenoseeds );
updatePeersInSuperSeedMode();
doUnchokes();
}catch (Throwable e) {
Debug.printStackTrace( e );
}
mainloop_loop_count++;
}
/**
* A private method that does analysis of the result sent by the tracker.
* It will mainly open new connections with peers provided
* and set the timeToWait variable according to the tracker response.
* @param tracker_response
*/
private void
analyseTrackerResponse(
TRTrackerAnnouncerResponse tracker_response )
{
// tracker_response.print();
TRTrackerAnnouncerResponsePeer[] peers = tracker_response.getPeers();
if ( peers != null ){
addPeersFromTracker( tracker_response.getPeers());
}
Map extensions = tracker_response.getExtensions();
if (extensions != null ){
addExtendedPeersFromTracker( extensions );
}
}
public void
processTrackerResponse(
TRTrackerAnnouncerResponse response )
{
// only process new peers if we're still running
if ( is_running ){
analyseTrackerResponse( response );
}
}
private void
addExtendedPeersFromTracker(
Map extensions )
{
Map protocols = (Map)extensions.get("protocols");
if ( protocols != null ){
System.out.println( "PEPeerControl: tracker response contained protocol extensions");
Iterator protocol_it = protocols.keySet().iterator();
while( protocol_it.hasNext()){
String protocol_name = (String)protocol_it.next();
Map protocol = (Map)protocols.get(protocol_name);
List transports = PEPeerTransportFactory.createExtendedTransports( this, protocol_name, protocol );
for (int i=0;i<transports.size();i++){
PEPeer transport = (PEPeer)transports.get(i);
addPeer( transport );
}
}
}
}
public List
getPeers()
{
return( new ArrayList( peer_transports_cow ));
}
public void
addPeer(
PEPeer _transport )
{
if ( !( _transport instanceof PEPeerTransport )){
throw( new RuntimeException("invalid class"));
}
PEPeerTransport transport = (PEPeerTransport)_transport;
if (!ip_filter.isInRange(transport.getIp(), adapter.getDisplayName())) {
ArrayList peer_transports = peer_transports_cow;
if ( !peer_transports.contains(transport)){
addToPeerTransports( transport );
}else{
Debug.out( "addPeer():: peer_transports.contains(transport): SHOULD NEVER HAPPEN !" );
transport.closeConnection( "already connected" );
}
}else{
transport.closeConnection( "IP address blocked by filters" );
}
}
public void
removePeer(
PEPeer _transport )
{
if ( !( _transport instanceof PEPeerTransport )){
throw( new RuntimeException("invalid class"));
}
PEPeerTransport transport = (PEPeerTransport)_transport;
closeAndRemovePeer( transport, "remove peer", true );
}
private void closeAndRemovePeer( PEPeerTransport peer, String reason, boolean log_if_not_found ) {
boolean removed = false;
// copy-on-write semantics
try{
peer_transports_mon.enter();
if ( peer_transports_cow.contains( peer )){
ArrayList new_peer_transports = new ArrayList( peer_transports_cow );
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -