Re-written by Scott Mansfield <thephantom@mac.com

git-svn-id: file:///srv/svn/repos/haiku/trunk/current@1549 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Philippe Houdoin 2002-10-16 22:54:57 +00:00
parent ae68bb3273
commit 61d3dba838

View File

@ -1,181 +1,585 @@
/****************************************************************************** /*=--------------------------------------------------------------------------=*
/ * NetAddress.cpp -- Implementation of the BNetAddress class.
/ File: NetAddress.cpp *
/ * Written by S.T. Mansfield (thephantom@mac.com)
/ Description: The Network API. *
/ * Remarks:
/ Copyright 2002, OpenBeOS Project, All Rights Reserved. * * In all accessors, address and port are converted from network to
/ * host byte order.
******************************************************************************/ * * In all mutators, address and port are converted from host to
* network byte order.
* * Michael, no trouts were harmed during the creation of this class.
*=--------------------------------------------------------------------------=*
* Copyright (c) 2002, The OpenBeOS project.
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, including without limitation
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
* and/or sell copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE.
*=--------------------------------------------------------------------------=*
*/
#include <string.h> #include <string.h>
#include <netdb.h> #include <netdb.h>
#include <sys/socket.h> #include <arpa/inet.h> // STM: #include <ByteOrder.h>???
#include <netinet/in.h> #include <netinet/in.h>
#include "NetAddress.h" #include "NetAddress.h"
/*
* AF_INET is 2 or 0 depending on whether or not you use the POSIX
* headers. Never never never compare by value, always use the
* #defined token.
*/
#define CTOR_INIT_LIST \
BArchivable( ), \
fInitialized( B_NO_INIT ), \
fFamily( AF_INET ), \
fPort( 0 ), \
fAddress( INADDR_ANY )
/* BNetAddress
*=--------------------------------------------------------------------------=*
* Purpose:
* Class constructors that have a direct compliment with one of the SetTo
* methods.
*
* Input parameters:
* Varies, see the method signature description for the corresponding
* SetTo() method descriptor.
*/
// This bad boy doubles up as the defalt ctor.
BNetAddress::BNetAddress( const char* hostname, unsigned short port )
: CTOR_INIT_LIST
{
fInitialized = SetTo( hostname, port );
}
BNetAddress::BNetAddress( const struct sockaddr_in& sa )
: CTOR_INIT_LIST
{
fInitialized = SetTo( sa );
}
BNetAddress::BNetAddress( in_addr addr, int port )
: CTOR_INIT_LIST
{
fInitialized = SetTo( addr, port );
}
BNetAddress::BNetAddress( uint32 addr, int port )
: CTOR_INIT_LIST
{
fInitialized = SetTo( addr, port );
}
BNetAddress::BNetAddress( const char* hostname, const char* protocol,
const char* service )
: CTOR_INIT_LIST
{
fInitialized = SetTo( hostname, protocol, service );
}
/* BNetAddress
*=--------------------------------------------------------------------------=*
* Purpose:
* Class copy ctor.
*
* Input parameter:
* refparam: Instance to copy.
*/
BNetAddress::BNetAddress( const BNetAddress& refparam )
: CTOR_INIT_LIST
{
fInitialized = clone( refparam );
}
/* BNetAddress
*=--------------------------------------------------------------------------=*
* Purpose:
* Ctor to instantiate from serialized data (BMessage).
*
* Input parameter:
* archive : Serialized object to instantiate from.
*/
BNetAddress::BNetAddress( BMessage* archive ) BNetAddress::BNetAddress( BMessage* archive )
: fStatus(B_OK), : CTOR_INIT_LIST
fName()
{ {
// TODO int8 int8_val;
fStatus = B_ERROR; int32 int32_val;
if ( archive->FindInt8( "bnaddr_family", &int8_val ) != B_OK )
{
return;
}
fFamily = int8_val;
if ( archive->FindInt8( "bnaddr_port", &int8_val ) != B_OK )
{
return;
}
fPort = int8_val;
if ( archive->FindInt32( "bnaddr_addr", &int32_val ) != B_OK )
{
return;
}
fAddress = int32_val;
fInitialized = B_OK;
} }
BNetAddress::~BNetAddress()
/* operator=
*=--------------------------------------------------------------------------=*
* Purpose:
* Class' assignment operator.
*
* Input parameter:
* refparam : Instance to assign from.
*/
BNetAddress& BNetAddress::operator=( const BNetAddress& refparam )
{ {
if ( clone( refparam ) == B_OK )
{
return *this;
}
} }
/* ~BNetAddress
*=--------------------------------------------------------------------------=*
* Purpose:
* Class dtor.
*/
BNetAddress::~BNetAddress( void )
{
fFamily = fPort = fAddress = 0;
fInitialized = B_NO_INIT;
}
/* GetAddr
*=--------------------------------------------------------------------------=*
* Purpose:
* Class accessor.
*
* Output parameters:
* hostname : Host name associated with this instance (default: NULL).
* In this particular implementation, hostname will be an
* ASCII-fied representation of an IP address.
* port : Port number associated with this instance
* (default: NULL). Will be converted to host byte order
* here, so it is not necessary to call ntohs() after
* calling this method.
*
* Returns:
* B_OK for success, B_NO_INIT if instance was not properly constructed.
*
* Remarks:
* Hostname and/or port can be NULL; although having them both NULL would
* be a pointless waste of CPU cycles. ;-)
*
* The hostname output parameter can be a variety of things, but it this
* method we convert the IP address to a string. See the relevant
* documentation about inet_ntoa() for details.
*
* Make sure hostname is large enough or you will step on someone
* else's toes. (Can you say "buffer overflow exploit" boys and girls?)
* You can generally be safe using the MAXHOSTNAMELEN define, which
* defaults to 64 bytes--don't forget to leave room for the NULL!
*/
status_t BNetAddress::GetAddr( char* hostname, unsigned short* port ) const
{
if ( fInitialized != B_OK )
{
return B_NO_INIT;
}
char* hnBuff;
struct in_addr ia;
ia.s_addr = fAddress;
if ( port != NULL )
{
*port = ( unsigned short )ntohs( fPort );
}
if ( hostname != NULL )
{
hnBuff = inet_ntoa( ia );
if ( hnBuff != NULL )
{
strcpy( hostname, hnBuff );
}
}
return B_OK;
}
/* GetAddr
*=--------------------------------------------------------------------------=*
* Purpose:
* Class accessor.
*
* Output parameter:
* sa : sockaddr_in struct to be filled.
*
* Returns:
* B_OK for success, B_NO_INIT if instance was not properly constructed.
*
* Remarks:
* This method fills in the sin_addr, sin_family, and sin_port fields of
* the output parameter, all other fields are untouched. All numeric
* values are in network byte order.
*/
status_t BNetAddress::GetAddr( struct sockaddr_in& sa ) const
{
if ( fInitialized != B_OK )
{
return B_NO_INIT;
}
sa.sin_family = ( uint8 )fFamily;
sa.sin_port = ( uint8 )fPort;
sa.sin_addr.s_addr = ( in_addr_t )fAddress;
return B_OK;
}
/* GetAddr
*=--------------------------------------------------------------------------=*
* Purpose:
* Class accessor.
*
* Output parameters:
* addr : in_addr struct to fill.
* port : optional port number to fill.
*
* Returns:
* B_OK for success, B_NO_INIT if instance was not properly constructed.
*
* Remarks:
* Output parameters will be in network byte order, so it is not
* necessary to call htons after calling this method.
*/
status_t BNetAddress::GetAddr( in_addr& addr, unsigned short* port ) const
{
if ( fInitialized != B_OK )
{
return B_NO_INIT;
}
addr.s_addr = fAddress;
if ( port != NULL )
{
*port = fPort;
}
return B_OK;
}
/* InitCheck
*=--------------------------------------------------------------------------=*
* Purpose:
* Determine whether or not this instance is properly initialized.
*
* Returns:
* B_OK if this instance is initialized, B_ERROR if not.
*/
status_t BNetAddress::InitCheck( void ) const
{
return ( fInitialized == B_OK ) ? B_OK : B_ERROR;
}
/* Archive
*=--------------------------------------------------------------------------=*
* Purpose:
* Serialize this instance into the passed BMessage parameter.
*
* Input parameter:
* deep : [ignored] default==true.
*
* Output parameter:
* into : BMessage object to serialize into.
*
* Returns:
* B_OK/BERROR on success/failure. Returns B_NO_INIT if instance not
* properly initialized.
*/
status_t BNetAddress::Archive( BMessage* into, bool deep ) const status_t BNetAddress::Archive( BMessage* into, bool deep ) const
{ {
// TODO if ( fInitialized != B_OK )
{
return B_NO_INIT;
}
//STM: When storing, should we store some kind of class name too...
//STM: Is there any way to deal with a complete failure here? (into
// not completely filled).
if ( into->AddInt8( "bnaddr_family", fFamily ) != B_OK )
{
return B_ERROR; return B_ERROR;
} }
if ( into->AddInt8( "bnaddr_port", fPort ) != B_OK )
{
return B_ERROR;
}
if ( into->AddInt32( "bnaddr_addr", fAddress ) != B_OK )
{
return B_ERROR;
}
return B_OK;
}
/* Instantiate
*=--------------------------------------------------------------------------=*
* Purpose:
* Un-serialize and instantiate from the passed BMessage parameter.
*
* Input parameter:
* archive : Archived BMessage object for (de)serialization.
*
* Returns:
* NULL if a BNetAddress instance can not be initialized, otherwise
* a new BNetAddress object instantiated from the BMessage parameter.
*/
BArchivable* BNetAddress::Instantiate( BMessage* archive ) BArchivable* BNetAddress::Instantiate( BMessage* archive )
{ {
// TODO if ( !validate_instantiation( archive, "BNetAddress" ) )
{
return NULL; return NULL;
} }
BNetAddress::BNetAddress(const char * hostname, unsigned short port) BNetAddress* bna = new BNetAddress( archive );
: fStatus(B_OK), if ( bna == NULL )
fName()
{ {
fStatus = SetTo(hostname, port); return NULL;
} }
BNetAddress::BNetAddress(const sockaddr_in & address) if ( bna->InitCheck( ) != B_OK )
: fStatus(B_OK),
fName()
{ {
fStatus = SetTo(address); delete bna;
return NULL;
} }
BNetAddress::BNetAddress(in_addr address, int port) return bna;
: fStatus(B_OK),
fName()
{
fStatus = SetTo(address, port);
}
BNetAddress::BNetAddress(uint32 address, int port)
: fStatus(B_OK),
fName()
{
fStatus = SetTo(address, port);
}
BNetAddress::BNetAddress(const BNetAddress & address)
: fStatus(B_OK),
fName(address.fName)
{
}
BNetAddress::BNetAddress(const char * hostname,
const char * protocol,
const char * service)
: fStatus(B_OK),
fName()
{
fStatus = SetTo(hostname, protocol, service);
}
BNetAddress & BNetAddress::operator = (const BNetAddress & address)
{
fName = address.fName;
return *this;
}
status_t BNetAddress::InitCheck() const
{
return fStatus;
} }
/* SetTo
*=--------------------------------------------------------------------------=*
* Purpose:
* Set hostname and port network address data.
*
* Input parameters:
* hostname : Can be one of three things:
* 1. An ASCII-string representation of an IP address.
* 2. A canonical hostname.
* 3. NULL. If NULL, then by default the address will be
* set to INADDR_ANY (0.0.0.0).
* port : Duh.
*
* Returns:
* B_OK/B_ERROR for success/failure.
*/
status_t BNetAddress::SetTo( const char* hostname, unsigned short port )
{
struct hostent* HostEnt;
in_addr_t ia;
ia = INADDR_ANY;
// Try like all git-out to set the address from the given hostname.
if ( hostname != NULL )
{
// See if the string is an ASCII-fied IP address.
ia = inet_addr( hostname );
if ( ia == INADDR_ANY || ia == ( unsigned long )-1 )
{
// See if we can resolve the hostname to an IP address.
HostEnt = gethostbyname( hostname );
if ( HostEnt != NULL )
{
ia = *( int * )HostEnt->h_addr_list[0];
}
else
{
//STM: Print and/or log an error message?
return B_ERROR;
}
}
}
fFamily = AF_INET;
fPort = htons( port );
fAddress = htonl( ia );
return B_OK;
}
/* SetTo
*=--------------------------------------------------------------------------=*
* Purpose:
* Set from passed in socket/address info, our preferred mutator.
*
* Input parameter:
* sa : Data specifying the host/client connection.
*
* Returns:
* B_OK.
*/
status_t BNetAddress::SetTo( const struct sockaddr_in& sa )
{
fFamily = sa.sin_family;
fPort = htons( sa.sin_port );
fAddress = htonl( sa.sin_addr.s_addr );
return B_OK;
}
/* SetTo
*=--------------------------------------------------------------------------=*
* Purpose:
* Set from passed in address and port.
*
* Input parameters:
* addr : IP address in network form.
* port : Optional port number.
*
* Returns:
* B_OK.
*/
status_t BNetAddress::SetTo( in_addr addr, int port )
{
fFamily = AF_INET;
fPort = htons( port );
fAddress = htonl( addr.s_addr );
return B_OK;
}
/* SetTo
*=--------------------------------------------------------------------------=*
* Purpose:
* Set from passed in address and port.
*
* Input parameters:
* addr : Optional IP address in network form.
* port : Optional port number.
*
* Returns:
* B_OK.
*/
status_t BNetAddress::SetTo( uint32 addr, int port )
{
fFamily = AF_INET;
fPort = htons( port );
fAddress = htonl( addr );
return B_OK;
}
/* SetTo
*=--------------------------------------------------------------------------=*
* Purpose:
* Set from passed in hostname and protocol/service information.
*
* Input parameters:
* hostname : Can be one of three things:
* 1. An ASCII-string representation of an IP address.
* 2. A canonical hostname.
* 3. NULL. If NULL, then by default the address will be
* set to INADDR_ANY (0.0.0.0).
* protocol : Datagram type, typically "TCP" or "UDP"
* service : The name of the service, such as http, ftp, et al. This
* must be one of the official service names listed in
* /etc/services -- but you already knew that because
* you're doing network/sockets programming, RIIIGHT???.
*
* Returns:
* B_OK/B_ERROR on success/failure.
*
* Remarks:
* The protocol and service input parameters must be one of the official
* types listed in /etc/services. We use these two parameters to
* determine the port number (see getservbyname(3)). This method will
* fail if the aforementioned precondition is not met.
*/
status_t BNetAddress::SetTo( const char* hostname, const char* protocol, status_t BNetAddress::SetTo( const char* hostname, const char* protocol,
const char* service ) const char* service )
{ {
struct hostent *host = gethostbyname(hostname); struct servent* ServiceEntry;
struct servent *serv = getservbyname(service, protocol);
if (host != NULL && serv != NULL) { ServiceEntry = getservbyname( service, protocol );
fName.sin_family = host->h_addrtype; if ( ServiceEntry == NULL )
fName.sin_port = serv->s_port; {
fName.sin_addr = *((in_addr *) host->h_addr); //STM: Print and/or log an error message?
return B_OK;
}
else {
fName.sin_family = AF_UNSPEC;
fName.sin_port = 0;
fName.sin_addr.s_addr = INADDR_ANY;
return B_ERROR; return B_ERROR;
} }
endservent( );
return SetTo( hostname, ServiceEntry->s_port );
} }
status_t BNetAddress::SetTo(const char * hostname, unsigned short port)
/* clone
*=--------------------------------------------------------------------------=*
* Purpose:
* Private copy helper method.
*
* Input parameter:
* RefParam: Instance to clone.
*
* Returns:
* B_OK for success, B_NO_INIT if RefParam was not properly constructed.
*/
status_t BNetAddress::clone( const BNetAddress& RefParam )
{ {
struct hostent *host; if ( !RefParam.fInitialized )
if (hostname != NULL && (host = gethostbyname(hostname)) != NULL) {
fName.sin_family = host->h_addrtype;
fName.sin_port = htons(port);
fName.sin_addr = *((in_addr *) host->h_addr);
return B_OK;
}
else {
fName.sin_family = AF_UNSPEC;
fName.sin_port = 0;
fName.sin_addr.s_addr = INADDR_ANY;
return B_ERROR;
}
}
status_t BNetAddress::SetTo(const sockaddr_in & address)
{ {
fName = address; return B_NO_INIT;
return (fName.sin_family != AF_UNSPEC ? B_OK : B_ERROR);
} }
status_t BNetAddress::SetTo(in_addr address, int port) fFamily = RefParam.fFamily;
{ fPort = RefParam.fPort;
fName.sin_family = AF_INET; fAddress = RefParam.fAddress;
fName.sin_port = port;
fName.sin_addr = address;
return B_OK; return B_OK;
} }
status_t BNetAddress::SetTo(uint32 address, int port) /*=------------------------------------------------------------------- End -=*/
{
fName.sin_family = AF_INET;
fName.sin_port = port;
fName.sin_addr.s_addr = address;
return B_OK;
}
status_t BNetAddress::GetAddr(char * hostname = 0, unsigned short * port = 0) const
{
struct hostent *host = gethostbyaddr((const char *) &fName.sin_addr,
sizeof(fName.sin_addr), fName.sin_family);
if (host != NULL) {
strcpy(hostname, host->h_name);
if (port != 0)
*port = ntohs(fName.sin_port);
return B_OK;
}
return B_ERROR;
}
status_t BNetAddress::GetAddr(sockaddr_in & address) const
{
address = fName;
return B_OK;
}
status_t BNetAddress::GetAddr(in_addr & address, unsigned short * port) const
{
address = fName.sin_addr;
if (port != 0)
*port = ntohs(fName.sin_port);
return B_OK;
}