⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 psistackwinsock.pas

📁 一个delphi的p2p控件的源代码
💻 PAS
字号:
unit PsiStackWinsock;

//******************************************************************************
// The original software is under
// Copyright (c) 1993 - 2000, Chad Z. Hower (Kudzu)
//   and the Indy Pit Crew - http://www.nevrona.com/Indy/
//
// Amended : November 2000, by Michael M. Michalak MACS for use with
// MorphTek.com Inc Peer to Peer Open Source Components - http://www.morphtek.com
//
//******************************************************************************

interface

uses
  Classes,
  PsiStack, PsiStackConsts, PsiWinsock,
	Windows;

type

  TPsiStackVersionWinsock = class(TPsiStackVersion)
  public
    constructor create(InfoStruct: Pointer);  override;
  end;

  TPsiStackWinsock = class(TPsiStack)
  protected
    procedure PopulateLocalAddresses; override;
    function WSGetLocalAddress: string; override;
    function WSGetLocalAddresses: TStrings; override;
  public
    constructor Create; override;
    destructor Destroy; override;
    function TInAddrToString(var AInAddr): string; override;
    procedure TranslateStringToTInAddr(AIP: string; var AInAddr); override;
    //
    function WSAccept(ASocket: TPsiStackSocketHandle; var VIP: string; var VPort: Integer)
     : TPsiStackSocketHandle; override;
    function WSBind(ASocket: TPsiStackSocketHandle; const AFamily: Integer;
     const AIP: string; const APort: Integer): Integer; override;
    function WSCloseSocket(ASocket: TPsiStackSocketHandle): Integer; override;
    function WSConnect(const ASocket: TPsiStackSocketHandle; const AFamily: Integer;
     const AIP: string; const APort: Integer): Integer; override;
    function WSGetHostByAddr(const AAddress: string): string; override;
    function WSGetHostByName(const AHostName: string): string; override;
    function WSGetHostName: string; override;
    function WSGetServByName(const AServiceName: string): Integer; override;
    function WSGetServByPort(const APortNumber: Integer): TStrings; override;
    function WSHToNs(AHostShort: Word): Word; override;
    function WSListen(ASocket: TPsiStackSocketHandle; ABackLog: Integer): Integer; override;
    function WSNToHs(ANetShort: Word): Word; override;
    function WSHToNL(AHostLong: LongWord): LongWord; override;
    function WSNToHL(ANetLong: LongWord): LongWord; override;
    function WSRecv(ASocket: TPsiStackSocketHandle; var ABuffer; ABufferLength, AFlags: Integer)
     : integer; override;
    function WSRecvFrom(const ASocket: TPsiStackSocketHandle; var ABuffer;
     const ALength, AFlags: Integer; var VIP: string; var VPort: Integer): Integer; override;
    function WSSelect(ARead, AWrite, AErrors: TList; ATimeout: Integer): Integer; override;
    function WSSend(ASocket: TPsiStackSocketHandle; var ABuffer;
     const ABufferLength, AFlags: Integer): Integer; override;
    function WSSendTo(ASocket: TPsiStackSocketHandle; var ABuffer;
     const ABufferLength, AFlags: Integer; const AIP: string; const APort: integer): Integer;
     override;
    function WSSetSockOpt(ASocket: TPsiStackSocketHandle; ALevel, AOptName: Integer; AOptVal: PChar;
     AOptLen: Integer): Integer; override;
    function WSSocket(AFamily, AStruct, AProtocol: Integer): TPsiStackSocketHandle; override;
    function WSGetLastError: Integer; override;
  end;

implementation

uses
  PsiException,
  PsiGlobal, PsiResourceStrings,
  SysUtils;

constructor TPsiStackWinsock.Create;
var
  sData: TWSAData;
begin
  inherited;
  LoadWinsock;
  if WSAStartup($101, sData) = SOCKET_ERROR then begin
    raise EPsiException.Create(RSWinsockInitializationError);
  end;
  FStackVersion := TPsiStackVersionWinsock.Create(@sData);

  // TODO - Stack props
//	sStackDescription := StrPas(sData.szDescription);
//  fiMaxUDPSize := sData.iMaxUdpDg;
//  fiMaxSockets := sData.iMaxSockets;
end;

destructor TPsiStackWinsock.Destroy;
begin
  UnloadWinsock;
  FStackVersion.Free;
  inherited;
end;

//function TPsiStackWinsock.TInAddrToString(AInAddr: TInAddr): string;
function TPsiStackWinsock.TInAddrToString(var AInAddr): string;
begin
  with TInAddr(AInAddr).S_un_b do begin
    result := IntToStr(Ord(s_b1)) + '.' + IntToStr(Ord(s_b2)) + '.' + IntToStr(Ord(s_b3)) + '.'
     + IntToStr(Ord(s_b4));
  end;
end;

function TPsiStackWinsock.WSAccept(ASocket: TPsiStackSocketHandle;
  var VIP: string; var VPort: Integer): TPsiStackSocketHandle;
var
	i: Integer;
  Addr: TSockAddr;
begin
	i := SizeOf(addr);
  result := Accept(ASocket, @addr, @i);
  VIP := TInAddrToString(Addr.sin_addr);
  VPort := NToHs(Addr.sin_port);
end;

function TPsiStackWinsock.WSBind(ASocket: TPsiStackSocketHandle;
  const AFamily: Integer; const AIP: string;
  const APort: Integer): Integer;
var
  Addr: TSockAddrIn;
begin
  Addr.sin_family := AFamily;
  if length(AIP) = 0 then begin
    Addr.sin_addr.s_addr := INADDR_ANY;
  end else begin
    Addr.sin_addr := TInAddr(StringToTInAddr(AIP));
  end;
  Addr.sin_port := HToNS(APort);
  result := Bind(ASocket, addr, SizeOf(Addr));
end;

function TPsiStackWinsock.WSCloseSocket(ASocket: TPsiStackSocketHandle): Integer;
begin
  result := CloseSocket(ASocket);
end;

function TPsiStackWinsock.WSConnect(const ASocket: TPsiStackSocketHandle;
  const AFamily: Integer; const AIP: string;
  const APort: Integer): Integer;
var
  Addr: TSockAddrIn;
begin
  Addr.sin_family := AFamily;
  Addr.sin_addr := TInAddr(StringToTInAddr(AIP));
  Addr.sin_port := HToNS(APort);
  result := Connect(ASocket, Addr, SizeOf(Addr));
end;

function TPsiStackWinsock.WSGetHostByName(const AHostName: string): string;
var
  pa: PChar;
  sa: TInAddr;
  Host: PHostEnt;
begin
  Host := GetHostByName(PChar(AHostName));
  if Host = nil then begin
    CheckForSocketError(SOCKET_ERROR);
  end else begin
    pa := Host^.h_addr_list^;
    sa.S_un_b.s_b1 := pa[0];
    sa.S_un_b.s_b2 := pa[1];
    sa.S_un_b.s_b3 := pa[2];
    sa.S_un_b.s_b4 := pa[3];
    result := TInAddrToString(sa);
  end;
end;

function TPsiStackWinsock.WSGetHostByAddr(const AAddress: string): string;
var
  Host: PHostEnt;
  LAddr: Longint;
begin
  LAddr := inet_addr(PChar(AAddress));
  Host := GetHostByAddr(@LAddr, SizeOf(LAddr), AF_INET);
  if Host = nil then begin
    CheckForSocketError(SOCKET_ERROR);
  end else begin
    result := Host^.h_name;
  end;
end;



function TPsiStackWinsock.WSGetHostName: string;
begin
  SetLength(result, 250);
  GetHostName(PChar(result), Length(result));
  Result := String(PChar(result));
end;

function TPsiStackWinsock.WSListen(ASocket: TPsiStackSocketHandle;
  ABackLog: Integer): Integer;
begin
  result := Listen(ASocket, ABacklog);
end;

function TPsiStackWinsock.WSRecv(ASocket: TPsiStackSocketHandle; var ABuffer; ABufferLength
 , AFlags: Integer): integer;
begin
  result := Recv(ASocket, ABuffer, ABufferLength, AFlags);
end;

function TPsiStackWinsock.WSRecvFrom(const ASocket: TPsiStackSocketHandle;
  var ABuffer; const ALength, AFlags: Integer; var VIP: string;
  var VPort: Integer): Integer;
var
  iSize: integer;
  Addr: TSockAddrIn;
begin
  iSize := SizeOf(Addr);
  result := RecvFrom(ASocket, ABuffer, ALength, AFlags, Addr, iSize);
  VIP := TInAddrToString(Addr.sin_addr);
  VPort := NToHs(Addr.sin_port);
end;

function TPsiStackWinsock.WSSelect(ARead, AWrite, AErrors: TList; ATimeout: Integer): Integer;
var
  tmTo: TTimeVal;
  FDRead, FDWrite, FDError: TFDSet;

  procedure GetFDSet(AList: TList; var ASet: TFDSet);
  var
    i: Integer;
  begin
    if assigned( AList ) then
    begin
      if ASet.fd_count > 0 then
      begin
        AList.Clear;
        for i := 0 to ASet.fd_count - 1 do
        begin
          AList.Add(TObject(ASet.fd_array[i]));
        end;
      end;
    end;
  end;

  procedure SetFDSet(AList: TList; var ASet: TFDSet);
  var
    i: integer;
  begin
    if AList <> nil then begin
      if AList.Count > FD_SETSIZE then begin
        raise EPsiException.Create(RSSetSizeExceeded);
      end;
      for i := 0 to AList.Count - 1 do begin
        ASet.fd_array[i] := TPsiStackSocketHandle(AList[i]);
      end;
      ASet.fd_count := AList.Count;
    end;
  end;

begin
  FillChar(FDRead, SizeOf(FDRead), 0);
  FillChar(FDWrite, SizeOf(FDWrite), 0);
  FillChar(FDError, SizeOf(FDError), 0);
  SetFDSet(ARead, FDRead);
  SetFDSet(AWrite, FDWrite);
  SetFDSet(AErrors, FDError);
  if ATimeout = PsiTimeoutInfinite then begin
    Result := Select(0, @FDRead, @FDWrite, @FDError, nil);
  end else begin
    tmTo.tv_sec := ATimeout div 1000;
    tmTo.tv_usec := (ATimeout mod 1000) * 1000;
    Result := Select(0, @FDRead, @FDWrite, @FDError, @tmTO);
  end;
  GetFDSet(ARead, FDRead);
  GetFDSet(AWrite, FDWrite);
  GetFDSet(AErrors, FDError);
end;

function TPsiStackWinsock.WSSend(ASocket: TPsiStackSocketHandle;
  var ABuffer; const ABufferLength, AFlags: Integer): Integer;
begin
  result := Send(ASocket, ABuffer, ABufferLength, AFlags);
end;

function TPsiStackWinsock.WSSendTo(ASocket: TPsiStackSocketHandle;
  var ABuffer; const ABufferLength, AFlags: Integer; const AIP: string;
  const APort: integer): Integer;
var
  Addr: TSockAddrIn;
begin
  FillChar(Addr, SizeOf(Addr), 0);
  with Addr do
  begin
    sin_family := Psi_PF_INET;
    sin_addr := TInAddr(StringToTInAddr(AIP));
    sin_port := HToNs(APort);
  end;
  result := SendTo(ASocket, ABuffer, ABufferLength, AFlags, Addr, SizeOf(Addr));
end;

function TPsiStackWinsock.WSSetSockOpt(ASocket: TPsiStackSocketHandle;
  ALevel, AOptName: Integer; AOptVal: PChar; AOptLen: Integer): Integer;
begin
  result := SetSockOpt(ASocket, ALevel, AOptName, AOptVal, AOptLen);
end;

function TPsiStackWinsock.WSGetLocalAddresses: TStrings;
begin
  if FLocalAddresses = nil then
  begin
    FLocalAddresses := TStringList.Create;
  end;
  PopulateLocalAddresses;
  Result := FLocalAddresses;
end;

function TPsiStackWinsock.WSGetLastError: Integer;
begin
  result := WSAGetLastError;
end;

function TPsiStackWinsock.WSSocket(AFamily, AStruct, AProtocol: Integer): TPsiStackSocketHandle;
begin
  result := Socket(AFamily, AStruct, AProtocol);
end;

function TPsiStackWinsock.WSHToNs(AHostShort: Word): Word;
begin
  result := HToNs(AHostShort);
end;

function TPsiStackWinsock.WSNToHs(ANetShort: Word): Word;
begin
  result := NToHs(ANetShort);
end;


function TPsiStackWinsock.WSGetServByName(const AServiceName: string): Integer;
var
  ps: PServEnt;
begin
  ps := GetServByName(PChar(AServiceName), nil);
  if ps <> nil then
  begin
    Result := Ntohs(ps^.s_port);
  end
  else
  begin
    try
      Result := StrToInt(AServiceName);
    except
      on EConvertError do raise EPsiException.CreateFmt(RSInvalidServiceName, [AServiceName]);
    end;
  end;
end;

function TPsiStackWinsock.WSGetServByPort(
  const APortNumber: Integer): TStrings;
var
  ps: PServEnt;
  i: integer;
  p: array of PChar;
begin
  Result := TStringList.Create;
  p := nil;
  try
    ps := GetServByPort(HToNs(APortNumber), nil);
    if ps <> nil then
    begin
      Result.Add(ps^.s_name);
      i := 0;
      p := pointer(ps^.s_aliases);
      while p[i] <> nil do
      begin
        Result.Add(PChar(p[i]));
        inc(i);
      end;
    end;
  except
    Result.Free;
  end;
end;

function TPsiStackWinsock.WSHToNL(AHostLong: LongWord): LongWord;
begin
  Result := HToNL(AHostLong);
end;

function TPsiStackWinsock.WSNToHL(ANetLong: LongWord): LongWord;
begin
  Result := NToHL(ANetLong);
end;

procedure TPsiStackWinsock.PopulateLocalAddresses;
type
  TaPInAddr = Array[0..250] of PInAddr;
  PaPInAddr = ^TaPInAddr;
var
  i: integer;
  AHost: PHostEnt;
  PAdrPtr: PaPInAddr;
begin
  FLocalAddresses.Clear ;
  AHost := GetHostByName(PChar(WSGetHostName));
  if AHost = nil then
  begin
    CheckForSocketError(SOCKET_ERROR);
  end
  else
  begin
    PAdrPtr := PAPInAddr(AHost^.h_addr_list);
    i := 0;
    while PAdrPtr^[i] <> nil do
    begin
      FLocalAddresses.Add(TInAddrToString(PAdrPtr^[I]^));
      Inc(I);
    end;
  end;
end;

function TPsiStackWinsock.WSGetLocalAddress: string;
begin
  Result := LocalAddresses[0];
end;

{ TPsiStackVersionWinsock }

constructor TPsiStackVersionWinsock.create(InfoStruct: Pointer);
var
  aData: PWSAData;
  procedure SetProp(var PropHolder: string; Value: PChar);
  begin
    if value <> nil then
      PropHolder := Value
    else
      PropHolder := '';
  end;
begin
  aData := InfoStruct;
  FMaxUdpDg     := aData^.iMaxUdpDg;
  FMaxSockets   := aData^.iMaxSockets;
  FVersion      := aData^.wHighVersion;
  FLowVersion   := aData^.wVersion;

  Setprop(FDescription ,aData^.szDescription);
//  if (Data^.lpVendorInfo <> nil) and (PChar(Data^.lpVendorInfo^) <> nil) then
//    Setprop(FVendorInfo ,PChar(Data^.lpVendorInfo^))
//  else
//    FVendorInfo := '';
  Setprop(FSystemStatus ,aData^.szSystemStatus);
  FName          := RSWSockStack;
end;


function ServeFile(ASocket: TPsiStackSocketHandle; AFileName: string): cardinal;
var
  LFileHandle: THandle;
begin
  result := 0;
  LFileHandle := CreateFile(PChar(AFileName), GENERIC_READ, FILE_SHARE_READ, nil, OPEN_EXISTING
   , FILE_ATTRIBUTE_NORMAL or FILE_FLAG_SEQUENTIAL_SCAN, 0); try
    if TransmitFile(ASocket, LFileHandle, 0, 0, nil, nil, 0) then begin
      result := getFileSize(LFileHandle, nil);
    end;
  finally CloseHandle(LFileHandle); end;
end;

procedure TPsiStackWinsock.TranslateStringToTInAddr(AIP: string; var AInAddr);
begin
  with TInAddr(AInAddr).S_un_b do begin
    s_b1 := Chr(StrToInt(Fetch(AIP, '.')));
    s_b2 := Chr(StrToInt(Fetch(AIP, '.')));
    s_b3 := Chr(StrToInt(Fetch(AIP, '.')));
    s_b4 := Chr(StrToInt(Fetch(AIP, '.')));
  end;
end;

initialization
  // Check if we are running under windows NT
  if (SysUtils.Win32Platform = VER_PLATFORM_WIN32_NT) then begin
    GServeFileProc := ServeFile;
  end;
end.

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -