]>
git.saurik.com Git - wxWidgets.git/blob - src/unix/sockunix.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/unix/sockunix.cpp
3 // Purpose: wxSocketImpl implementation for Unix systems
4 // Authors: Guilhem Lavaux, Guillermo Rodriguez Garcia, David Elliott,
8 // Copyright: (c) 1997 Guilhem Lavaux
9 // (c) 2008 Vadim Zeitlin
10 // Licence: wxWindows licence
11 /////////////////////////////////////////////////////////////////////////////
14 #include "wx/wxprec.h"
18 #include "wx/private/fd.h"
19 #include "wx/private/socket.h"
20 #include "wx/unix/private/sockunix.h"
22 #if defined(__VISAGECPP__)
23 #define BSD_SELECT /* use Berkeley Sockets select */
26 #if defined(__WATCOMC__)
32 #include <sys/types.h>
37 #include <netinet/in.h>
40 #include <sys/ioctl.h>
42 #ifdef HAVE_SYS_SELECT_H
43 # include <sys/select.h>
50 u_char sun_len
; /* sockaddr len including null */
51 u_char sun_family
; /* AF_UNIX */
52 char sun_path
[108]; /* path name (gag) */
55 #include <sys/socket.h>
61 #include <netinet/in.h>
62 #include <arpa/inet.h>
69 #include <machine/endian.h>
75 #define EBADF SOCEBADF
81 #include <sys/socket.h>
82 #include <sys/ioctl.h>
83 #include <sys/select.h>
85 #define close(a) soclose(a)
86 #define select(a,b,c,d,e) bsdselect(a,b,c,d,e)
87 int _System
bsdselect(int,
92 int _System
soclose(int);
96 #include <sys/select.h>
104 # include <sys/filio.h>
107 # include <bstring.h>
110 # include <strings.h>
117 # define WX_SOCKLEN_T unsigned int
121 # define WX_SOCKLEN_T socklen_t
123 # elif defined(__WXMAC__)
124 # define WX_SOCKLEN_T socklen_t
126 # define WX_SOCKLEN_T int
130 #endif /* SOCKLEN_T */
133 #define SOCKOPTLEN_T WX_SOCKLEN_T
136 /* UnixWare reportedly needs this for FIONBIO definition */
138 #include <sys/filio.h>
142 * INADDR_BROADCAST is identical to INADDR_NONE which is not defined
143 * on all systems. INADDR_BROADCAST should be fine to indicate an error.
146 #define INADDR_NONE INADDR_BROADCAST
149 // ----------------------------------------------------------------------------
150 // implementation of thread-safe/reentrant functions if they're missing
151 // ----------------------------------------------------------------------------
153 #if wxUSE_THREADS && (defined(HAVE_GETHOSTBYNAME) || defined(HAVE_GETSERVBYNAME))
154 # include "wx/thread.h"
157 #if defined(HAVE_GETHOSTBYNAME)
158 static struct hostent
* deepCopyHostent(struct hostent
*h
,
159 const struct hostent
*he
,
160 char *buffer
, int size
, int *err
)
162 /* copy old structure */
163 memcpy(h
, he
, sizeof(struct hostent
));
166 int len
= strlen(h
->h_name
);
172 memcpy(buffer
, h
->h_name
, len
);
176 /* track position in the buffer */
179 /* reuse len to store address length */
182 /* ensure pointer alignment */
183 unsigned int misalign
= sizeof(char *) - pos%sizeof
(char *);
184 if(misalign
< sizeof(char *))
187 /* leave space for pointer list */
188 char **p
= h
->h_addr_list
, **q
;
189 char **h_addr_list
= (char **)(buffer
+ pos
);
191 pos
+= sizeof(char *);
193 /* copy addresses and fill new pointer list */
194 for (p
= h
->h_addr_list
, q
= h_addr_list
; *p
!= 0; p
++, q
++)
196 if (size
< pos
+ len
)
201 memcpy(buffer
+ pos
, *p
, len
); /* copy content */
202 *q
= buffer
+ pos
; /* set copied pointer to copied content */
205 *++q
= 0; /* null terminate the pointer list */
206 h
->h_addr_list
= h_addr_list
; /* copy pointer to pointers */
208 /* ensure word alignment of pointers */
209 misalign
= sizeof(char *) - pos%sizeof
(char *);
210 if(misalign
< sizeof(char *))
213 /* leave space for pointer list */
215 char **h_aliases
= (char **)(buffer
+ pos
);
217 pos
+= sizeof(char *);
219 /* copy aliases and fill new pointer list */
220 for (p
= h
->h_aliases
, q
= h_aliases
; *p
!= 0; p
++, q
++)
223 if (size
<= pos
+ len
)
228 memcpy(buffer
+ pos
, *p
, len
); /* copy content */
229 buffer
[pos
+ len
] = '\0';
230 *q
= buffer
+ pos
; /* set copied pointer to copied content */
233 *++q
= 0; /* null terminate the pointer list */
234 h
->h_aliases
= h_aliases
; /* copy pointer to pointers */
240 #if defined(HAVE_GETHOSTBYNAME) && wxUSE_THREADS
241 static wxMutex nameLock
;
243 struct hostent
* wxGethostbyname_r(const char *hostname
, struct hostent
*h
,
244 void *buffer
, int size
, int *err
)
247 struct hostent
*he
= NULL
;
249 #if defined(HAVE_FUNC_GETHOSTBYNAME_R_6)
250 if (gethostbyname_r(hostname
, h
, (char*)buffer
, size
, &he
, err
))
252 #elif defined(HAVE_FUNC_GETHOSTBYNAME_R_5)
253 he
= gethostbyname_r(hostname
, h
, (char*)buffer
, size
, err
);
254 #elif defined(HAVE_FUNC_GETHOSTBYNAME_R_3)
255 if (gethostbyname_r(hostname
, h
, (struct hostent_data
*) buffer
))
262 #elif defined(HAVE_GETHOSTBYNAME)
264 wxMutexLocker
locker(nameLock
);
266 he
= gethostbyname(hostname
);
270 he
= deepCopyHostent(h
, he
, (char*)buffer
, size
, err
);
275 #if defined(HAVE_GETHOSTBYNAME) && wxUSE_THREADS
276 static wxMutex addrLock
;
278 struct hostent
* wxGethostbyaddr_r(const char *addr_buf
, int buf_size
,
279 int proto
, struct hostent
*h
,
280 void *buffer
, int size
, int *err
)
282 struct hostent
*he
= NULL
;
284 #if defined(HAVE_FUNC_GETHOSTBYNAME_R_6)
285 if (gethostbyaddr_r(addr_buf
, buf_size
, proto
, h
,
286 (char*)buffer
, size
, &he
, err
))
288 #elif defined(HAVE_FUNC_GETHOSTBYNAME_R_5)
289 he
= gethostbyaddr_r(addr_buf
, buf_size
, proto
, h
, (char*)buffer
, size
, err
);
290 #elif defined(HAVE_FUNC_GETHOSTBYNAME_R_3)
291 if (gethostbyaddr_r(addr_buf
, buf_size
, proto
, h
,
292 (struct hostent_data
*) buffer
))
299 #elif defined(HAVE_GETHOSTBYNAME)
301 wxMutexLocker
locker(addrLock
);
303 he
= gethostbyaddr(addr_buf
, buf_size
, proto
);
307 he
= deepCopyHostent(h
, he
, (char*)buffer
, size
, err
);
312 #if defined(HAVE_GETSERVBYNAME)
313 static struct servent
* deepCopyServent(struct servent
*s
,
314 const struct servent
*se
,
315 char *buffer
, int size
)
317 /* copy plain old structure */
318 memcpy(s
, se
, sizeof(struct servent
));
321 int len
= strlen(s
->s_name
);
326 memcpy(buffer
, s
->s_name
, len
);
330 /* track position in the buffer */
334 len
= strlen(s
->s_proto
);
335 if (pos
+ len
>= size
)
339 memcpy(buffer
+ pos
, s
->s_proto
, len
);
340 buffer
[pos
+ len
] = '\0';
341 s
->s_proto
= buffer
+ pos
;
343 /* track position in the buffer */
346 /* ensure pointer alignment */
347 unsigned int misalign
= sizeof(char *) - pos%sizeof
(char *);
348 if(misalign
< sizeof(char *))
351 /* leave space for pointer list */
352 char **p
= s
->s_aliases
, **q
;
353 char **s_aliases
= (char **)(buffer
+ pos
);
355 pos
+= sizeof(char *);
357 /* copy addresses and fill new pointer list */
358 for (p
= s
->s_aliases
, q
= s_aliases
; *p
!= 0; p
++, q
++){
360 if (size
<= pos
+ len
)
364 memcpy(buffer
+ pos
, *p
, len
); /* copy content */
365 buffer
[pos
+ len
] = '\0';
366 *q
= buffer
+ pos
; /* set copied pointer to copied content */
369 *++q
= 0; /* null terminate the pointer list */
370 s
->s_aliases
= s_aliases
; /* copy pointer to pointers */
375 #if defined(HAVE_GETSERVBYNAME) && wxUSE_THREADS
376 static wxMutex servLock
;
378 struct servent
*wxGetservbyname_r(const char *port
, const char *protocol
,
379 struct servent
*serv
, void *buffer
, int size
)
381 struct servent
*se
= NULL
;
382 #if defined(HAVE_FUNC_GETSERVBYNAME_R_6)
383 if (getservbyname_r(port
, protocol
, serv
, (char*)buffer
, size
, &se
))
385 #elif defined(HAVE_FUNC_GETSERVBYNAME_R_5)
386 se
= getservbyname_r(port
, protocol
, serv
, (char*)buffer
, size
);
387 #elif defined(HAVE_FUNC_GETSERVBYNAME_R_4)
388 if (getservbyname_r(port
, protocol
, serv
, (struct servent_data
*) buffer
))
392 #elif defined(HAVE_GETSERVBYNAME)
394 wxMutexLocker
locker(servLock
);
396 se
= getservbyname(port
, protocol
);
398 se
= deepCopyServent(serv
, se
, (char*)buffer
, size
);
403 // ============================================================================
404 // wxSocketImpl implementation
405 // ============================================================================
408 wxSocketImpl
*wxSocketImpl::Create(wxSocketBase
& wxsocket
)
410 return new wxSocketImplUnix(wxsocket
);
414 wxSocketError
wxSocketImplUnix::GetLastError() const
419 return wxSOCKET_NOERROR
;
422 return wxSOCKET_INVSOCK
;
424 // unfortunately EAGAIN only has the "would block" meaning for read(),
425 // not for connect() for which it means something rather different but
426 // we can't distinguish between these two situations currently...
428 // also notice that EWOULDBLOCK can be different from EAGAIN on some
429 // systems (HP-UX being the only known example) while it's defined as
430 // EAGAIN on most others (e.g. Linux)
433 #if EWOULDBLOCK != EAGAIN
436 #endif // EWOULDBLOCK
438 return wxSOCKET_WOULDBLOCK
;
441 return wxSOCKET_IOERR
;
445 void wxSocketImplUnix::DoEnableEvents(bool flag
)
447 wxSocketManager
* const manager
= wxSocketManager::Get();
450 manager
->Install_Callback(this, wxSOCKET_INPUT
);
451 manager
->Install_Callback(this, wxSOCKET_OUTPUT
);
455 manager
->Uninstall_Callback(this, wxSOCKET_INPUT
);
456 manager
->Uninstall_Callback(this, wxSOCKET_OUTPUT
);
461 void wxSocketImplUnix::OnStateChange(wxSocketNotify event
)
463 NotifyOnStateChange(event
);
465 if ( event
== wxSOCKET_LOST
)
469 void wxSocketImplUnix::OnReadWaiting()
473 if (m_fd
== INVALID_SOCKET
)
478 int num
= recv(m_fd
, &c
, 1, MSG_PEEK
);
482 OnStateChange(wxSOCKET_INPUT
);
486 if (m_server
&& m_stream
)
488 OnStateChange(wxSOCKET_CONNECTION
);
494 /* graceful shutdown */
495 OnStateChange(wxSOCKET_LOST
);
499 /* Empty datagram received */
500 OnStateChange(wxSOCKET_INPUT
);
505 /* Do not throw a lost event in cases where the socket isn't really lost */
506 if ((errno
== EWOULDBLOCK
) || (errno
== EAGAIN
) || (errno
== EINTR
))
508 OnStateChange(wxSOCKET_INPUT
);
512 OnStateChange(wxSOCKET_LOST
);
518 void wxSocketImplUnix::OnWriteWaiting()
520 if (m_establishing
&& !m_server
)
523 SOCKOPTLEN_T len
= sizeof(error
);
525 m_establishing
= false;
527 getsockopt(m_fd
, SOL_SOCKET
, SO_ERROR
, (char*)&error
, &len
);
531 OnStateChange(wxSOCKET_LOST
);
535 OnStateChange(wxSOCKET_CONNECTION
);
536 /* We have to fire this event by hand because CONNECTION (for clients)
537 * and OUTPUT are internally the same and we just disabled CONNECTION
538 * events with the above macro.
540 OnStateChange(wxSOCKET_OUTPUT
);
545 OnStateChange(wxSOCKET_OUTPUT
);
549 void wxSocketImplUnix::OnExceptionWaiting()
551 wxFAIL_MSG( "not supposed to be called" );
555 * -------------------------------------------------------------------------
557 * -------------------------------------------------------------------------
560 /* CHECK_ADDRESS verifies that the current address family is either
561 * wxSOCKET_NOFAMILY or wxSOCKET_*family*, and if it is wxSOCKET_NOFAMILY, it
562 * initalizes it to be a wxSOCKET_*family*. In other cases, it returns
563 * an appropiate error code.
565 * CHECK_ADDRESS_RETVAL does the same but returning 'retval' on error.
567 #define CHECK_ADDRESS(address, family) \
569 if (address->m_family == wxSOCKET_NOFAMILY) \
570 if (_GAddress_Init_##family(address) != wxSOCKET_NOERROR) \
571 return address->m_error; \
572 if (address->m_family != wxSOCKET_##family) \
574 address->m_error = wxSOCKET_INVADDR; \
575 return wxSOCKET_INVADDR; \
579 #define CHECK_ADDRESS_RETVAL(address, family, retval) \
581 if (address->m_family == wxSOCKET_NOFAMILY) \
582 if (_GAddress_Init_##family(address) != wxSOCKET_NOERROR) \
584 if (address->m_family != wxSOCKET_##family) \
586 address->m_error = wxSOCKET_INVADDR; \
592 GAddress
*GAddress_new(void)
596 if ((address
= (GAddress
*) malloc(sizeof(GAddress
))) == NULL
)
599 address
->m_family
= wxSOCKET_NOFAMILY
;
600 address
->m_addr
= NULL
;
606 GAddress
*GAddress_copy(GAddress
*address
)
610 assert(address
!= NULL
);
612 if ((addr2
= (GAddress
*) malloc(sizeof(GAddress
))) == NULL
)
615 memcpy(addr2
, address
, sizeof(GAddress
));
617 if (address
->m_addr
&& address
->m_len
> 0)
619 addr2
->m_addr
= (struct sockaddr
*)malloc(addr2
->m_len
);
620 if (addr2
->m_addr
== NULL
)
625 memcpy(addr2
->m_addr
, address
->m_addr
, addr2
->m_len
);
631 void GAddress_destroy(GAddress
*address
)
633 assert(address
!= NULL
);
636 free(address
->m_addr
);
641 void GAddress_SetFamily(GAddress
*address
, GAddressType type
)
643 assert(address
!= NULL
);
645 address
->m_family
= type
;
648 GAddressType
GAddress_GetFamily(GAddress
*address
)
650 assert(address
!= NULL
);
652 return address
->m_family
;
655 wxSocketError
_GAddress_translate_from(GAddress
*address
,
656 struct sockaddr
*addr
, int len
)
658 address
->m_realfamily
= addr
->sa_family
;
659 switch (addr
->sa_family
)
662 address
->m_family
= wxSOCKET_INET
;
665 address
->m_family
= wxSOCKET_UNIX
;
669 address
->m_family
= wxSOCKET_INET6
;
674 address
->m_error
= wxSOCKET_INVOP
;
675 return wxSOCKET_INVOP
;
680 free(address
->m_addr
);
682 address
->m_len
= len
;
683 address
->m_addr
= (struct sockaddr
*)malloc(len
);
685 if (address
->m_addr
== NULL
)
687 address
->m_error
= wxSOCKET_MEMERR
;
688 return wxSOCKET_MEMERR
;
691 memcpy(address
->m_addr
, addr
, len
);
693 return wxSOCKET_NOERROR
;
696 wxSocketError
_GAddress_translate_to(GAddress
*address
,
697 struct sockaddr
**addr
, int *len
)
699 if (!address
->m_addr
)
701 address
->m_error
= wxSOCKET_INVADDR
;
702 return wxSOCKET_INVADDR
;
705 *len
= address
->m_len
;
706 *addr
= (struct sockaddr
*)malloc(address
->m_len
);
709 address
->m_error
= wxSOCKET_MEMERR
;
710 return wxSOCKET_MEMERR
;
713 memcpy(*addr
, address
->m_addr
, address
->m_len
);
714 return wxSOCKET_NOERROR
;
718 * -------------------------------------------------------------------------
719 * Internet address family
720 * -------------------------------------------------------------------------
723 wxSocketError
_GAddress_Init_INET(GAddress
*address
)
725 address
->m_len
= sizeof(struct sockaddr_in
);
726 address
->m_addr
= (struct sockaddr
*) malloc(address
->m_len
);
727 if (address
->m_addr
== NULL
)
729 address
->m_error
= wxSOCKET_MEMERR
;
730 return wxSOCKET_MEMERR
;
733 address
->m_family
= wxSOCKET_INET
;
734 address
->m_realfamily
= PF_INET
;
735 ((struct sockaddr_in
*)address
->m_addr
)->sin_family
= AF_INET
;
736 ((struct sockaddr_in
*)address
->m_addr
)->sin_addr
.s_addr
= INADDR_ANY
;
738 return wxSOCKET_NOERROR
;
741 wxSocketError
GAddress_INET_SetHostName(GAddress
*address
, const char *hostname
)
744 struct in_addr
*addr
;
746 assert(address
!= NULL
);
748 CHECK_ADDRESS(address
, INET
);
750 addr
= &(((struct sockaddr_in
*)address
->m_addr
)->sin_addr
);
752 /* If it is a numeric host name, convert it now */
753 #if defined(HAVE_INET_ATON)
754 if (inet_aton(hostname
, addr
) == 0)
756 #elif defined(HAVE_INET_ADDR)
757 if ( (addr
->s_addr
= inet_addr(hostname
)) == (unsigned)-1 )
760 /* Use gethostbyname by default */
762 int val
= 1; /* VA doesn't like constants in conditional expressions */
767 struct in_addr
*array_addr
;
769 /* It is a real name, we solve it */
771 #if defined(HAVE_FUNC_GETHOSTBYNAME_R_3)
772 struct hostent_data buffer
;
777 he
= wxGethostbyname_r(hostname
, &h
, (void*)&buffer
, sizeof(buffer
), &err
);
780 /* Reset to invalid address */
781 addr
->s_addr
= INADDR_NONE
;
782 address
->m_error
= wxSOCKET_NOHOST
;
783 return wxSOCKET_NOHOST
;
786 array_addr
= (struct in_addr
*) *(he
->h_addr_list
);
787 addr
->s_addr
= array_addr
[0].s_addr
;
790 return wxSOCKET_NOERROR
;
794 wxSocketError
GAddress_INET_SetBroadcastAddress(GAddress
*address
)
796 return GAddress_INET_SetHostAddress(address
, INADDR_BROADCAST
);
799 wxSocketError
GAddress_INET_SetAnyAddress(GAddress
*address
)
801 return GAddress_INET_SetHostAddress(address
, INADDR_ANY
);
804 wxSocketError
GAddress_INET_SetHostAddress(GAddress
*address
,
805 unsigned long hostaddr
)
807 struct in_addr
*addr
;
809 assert(address
!= NULL
);
811 CHECK_ADDRESS(address
, INET
);
813 addr
= &(((struct sockaddr_in
*)address
->m_addr
)->sin_addr
);
814 addr
->s_addr
= htonl(hostaddr
);
816 return wxSOCKET_NOERROR
;
819 wxSocketError
GAddress_INET_SetPortName(GAddress
*address
, const char *port
,
820 const char *protocol
)
823 struct sockaddr_in
*addr
;
825 assert(address
!= NULL
);
826 CHECK_ADDRESS(address
, INET
);
830 address
->m_error
= wxSOCKET_INVPORT
;
831 return wxSOCKET_INVPORT
;
834 #if defined(HAVE_FUNC_GETSERVBYNAME_R_4)
835 struct servent_data buffer
;
840 se
= wxGetservbyname_r(port
, protocol
, &serv
,
841 (void*)&buffer
, sizeof(buffer
));
844 /* the cast to int suppresses compiler warnings about subscript having the
846 if (isdigit((int)port
[0]))
850 port_int
= atoi(port
);
851 addr
= (struct sockaddr_in
*)address
->m_addr
;
852 addr
->sin_port
= htons(port_int
);
853 return wxSOCKET_NOERROR
;
856 address
->m_error
= wxSOCKET_INVPORT
;
857 return wxSOCKET_INVPORT
;
860 addr
= (struct sockaddr_in
*)address
->m_addr
;
861 addr
->sin_port
= se
->s_port
;
863 return wxSOCKET_NOERROR
;
866 wxSocketError
GAddress_INET_SetPort(GAddress
*address
, unsigned short port
)
868 struct sockaddr_in
*addr
;
870 assert(address
!= NULL
);
871 CHECK_ADDRESS(address
, INET
);
873 addr
= (struct sockaddr_in
*)address
->m_addr
;
874 addr
->sin_port
= htons(port
);
876 return wxSOCKET_NOERROR
;
879 wxSocketError
GAddress_INET_GetHostName(GAddress
*address
, char *hostname
, size_t sbuf
)
883 struct sockaddr_in
*addr
;
885 assert(address
!= NULL
);
886 CHECK_ADDRESS(address
, INET
);
888 addr
= (struct sockaddr_in
*)address
->m_addr
;
889 addr_buf
= (char *)&(addr
->sin_addr
);
891 struct hostent temphost
;
892 #if defined(HAVE_FUNC_GETHOSTBYNAME_R_3)
893 struct hostent_data buffer
;
898 he
= wxGethostbyaddr_r(addr_buf
, sizeof(addr
->sin_addr
), AF_INET
, &temphost
,
899 (void*)&buffer
, sizeof(buffer
), &err
);
902 address
->m_error
= wxSOCKET_NOHOST
;
903 return wxSOCKET_NOHOST
;
906 strncpy(hostname
, he
->h_name
, sbuf
);
908 return wxSOCKET_NOERROR
;
911 unsigned long GAddress_INET_GetHostAddress(GAddress
*address
)
913 struct sockaddr_in
*addr
;
915 assert(address
!= NULL
);
916 CHECK_ADDRESS_RETVAL(address
, INET
, 0);
918 addr
= (struct sockaddr_in
*)address
->m_addr
;
920 return ntohl(addr
->sin_addr
.s_addr
);
923 unsigned short GAddress_INET_GetPort(GAddress
*address
)
925 struct sockaddr_in
*addr
;
927 assert(address
!= NULL
);
928 CHECK_ADDRESS_RETVAL(address
, INET
, 0);
930 addr
= (struct sockaddr_in
*)address
->m_addr
;
931 return ntohs(addr
->sin_port
);
936 * -------------------------------------------------------------------------
937 * Internet IPv6 address family
938 * -------------------------------------------------------------------------
941 wxSocketError
_GAddress_Init_INET6(GAddress
*address
)
943 struct in6_addr any_address
= IN6ADDR_ANY_INIT
;
944 address
->m_len
= sizeof(struct sockaddr_in6
);
945 address
->m_addr
= (struct sockaddr
*) malloc(address
->m_len
);
946 if (address
->m_addr
== NULL
)
948 address
->m_error
= wxSOCKET_MEMERR
;
949 return wxSOCKET_MEMERR
;
951 memset(address
->m_addr
,0,address
->m_len
);
953 address
->m_family
= wxSOCKET_INET6
;
954 address
->m_realfamily
= AF_INET6
;
955 ((struct sockaddr_in6
*)address
->m_addr
)->sin6_family
= AF_INET6
;
956 ((struct sockaddr_in6
*)address
->m_addr
)->sin6_addr
= any_address
;
958 return wxSOCKET_NOERROR
;
961 wxSocketError
GAddress_INET6_SetHostName(GAddress
*address
, const char *hostname
)
963 assert(address
!= NULL
);
964 CHECK_ADDRESS(address
, INET6
);
967 memset( & hints
, 0, sizeof( hints
) );
968 hints
.ai_family
= AF_INET6
;
970 if ( getaddrinfo( hostname
, "0", & hints
, & info
) || ! info
)
972 address
->m_error
= wxSOCKET_NOHOST
;
973 return wxSOCKET_NOHOST
;
976 memcpy( address
->m_addr
, info
->ai_addr
, info
->ai_addrlen
);
977 freeaddrinfo( info
);
978 return wxSOCKET_NOERROR
;
981 wxSocketError
GAddress_INET6_SetAnyAddress(GAddress
*address
)
983 assert(address
!= NULL
);
985 CHECK_ADDRESS(address
, INET6
);
987 struct in6_addr addr
;
988 memset( & addr
, 0, sizeof( addr
) );
989 return GAddress_INET6_SetHostAddress(address
, addr
);
991 wxSocketError
GAddress_INET6_SetHostAddress(GAddress
*address
,
992 struct in6_addr hostaddr
)
994 assert(address
!= NULL
);
996 CHECK_ADDRESS(address
, INET6
);
998 ((struct sockaddr_in6
*)address
->m_addr
)->sin6_addr
= hostaddr
;
1000 return wxSOCKET_NOERROR
;
1003 wxSocketError
GAddress_INET6_SetPortName(GAddress
*address
, const char *port
,
1004 const char *protocol
)
1007 struct sockaddr_in6
*addr
;
1009 assert(address
!= NULL
);
1010 CHECK_ADDRESS(address
, INET6
);
1014 address
->m_error
= wxSOCKET_INVPORT
;
1015 return wxSOCKET_INVPORT
;
1018 se
= getservbyname(port
, protocol
);
1021 if (isdigit(port
[0]))
1025 port_int
= atoi(port
);
1026 addr
= (struct sockaddr_in6
*)address
->m_addr
;
1027 addr
->sin6_port
= htons((u_short
) port_int
);
1028 return wxSOCKET_NOERROR
;
1031 address
->m_error
= wxSOCKET_INVPORT
;
1032 return wxSOCKET_INVPORT
;
1035 addr
= (struct sockaddr_in6
*)address
->m_addr
;
1036 addr
->sin6_port
= se
->s_port
;
1038 return wxSOCKET_NOERROR
;
1041 wxSocketError
GAddress_INET6_SetPort(GAddress
*address
, unsigned short port
)
1043 struct sockaddr_in6
*addr
;
1045 assert(address
!= NULL
);
1046 CHECK_ADDRESS(address
, INET6
);
1048 addr
= (struct sockaddr_in6
*)address
->m_addr
;
1049 addr
->sin6_port
= htons(port
);
1051 return wxSOCKET_NOERROR
;
1054 wxSocketError
GAddress_INET6_GetHostName(GAddress
*address
, char *hostname
, size_t sbuf
)
1058 struct sockaddr_in6
*addr
;
1060 assert(address
!= NULL
);
1061 CHECK_ADDRESS(address
, INET6
);
1063 addr
= (struct sockaddr_in6
*)address
->m_addr
;
1064 addr_buf
= (char *)&(addr
->sin6_addr
);
1066 he
= gethostbyaddr(addr_buf
, sizeof(addr
->sin6_addr
), AF_INET6
);
1069 address
->m_error
= wxSOCKET_NOHOST
;
1070 return wxSOCKET_NOHOST
;
1073 strncpy(hostname
, he
->h_name
, sbuf
);
1075 return wxSOCKET_NOERROR
;
1078 wxSocketError
GAddress_INET6_GetHostAddress(GAddress
*address
,struct in6_addr
*hostaddr
)
1080 assert(address
!= NULL
);
1081 assert(hostaddr
!= NULL
);
1082 CHECK_ADDRESS_RETVAL(address
, INET6
, wxSOCKET_INVADDR
);
1083 *hostaddr
= ( (struct sockaddr_in6
*)address
->m_addr
)->sin6_addr
;
1084 return wxSOCKET_NOERROR
;
1087 unsigned short GAddress_INET6_GetPort(GAddress
*address
)
1089 assert(address
!= NULL
);
1090 CHECK_ADDRESS_RETVAL(address
, INET6
, 0);
1092 return ntohs( ((struct sockaddr_in6
*)address
->m_addr
)->sin6_port
);
1095 #endif // wxUSE_IPV6
1098 * -------------------------------------------------------------------------
1099 * Unix address family
1100 * -------------------------------------------------------------------------
1103 #ifndef __VISAGECPP__
1104 wxSocketError
_GAddress_Init_UNIX(GAddress
*address
)
1106 address
->m_len
= sizeof(struct sockaddr_un
);
1107 address
->m_addr
= (struct sockaddr
*)malloc(address
->m_len
);
1108 if (address
->m_addr
== NULL
)
1110 address
->m_error
= wxSOCKET_MEMERR
;
1111 return wxSOCKET_MEMERR
;
1114 address
->m_family
= wxSOCKET_UNIX
;
1115 address
->m_realfamily
= PF_UNIX
;
1116 ((struct sockaddr_un
*)address
->m_addr
)->sun_family
= AF_UNIX
;
1117 ((struct sockaddr_un
*)address
->m_addr
)->sun_path
[0] = 0;
1119 return wxSOCKET_NOERROR
;
1122 #define UNIX_SOCK_PATHLEN (sizeof(addr->sun_path)/sizeof(addr->sun_path[0]))
1124 wxSocketError
GAddress_UNIX_SetPath(GAddress
*address
, const char *path
)
1126 struct sockaddr_un
*addr
;
1128 assert(address
!= NULL
);
1130 CHECK_ADDRESS(address
, UNIX
);
1132 addr
= ((struct sockaddr_un
*)address
->m_addr
);
1133 strncpy(addr
->sun_path
, path
, UNIX_SOCK_PATHLEN
);
1134 addr
->sun_path
[UNIX_SOCK_PATHLEN
- 1] = '\0';
1136 return wxSOCKET_NOERROR
;
1139 wxSocketError
GAddress_UNIX_GetPath(GAddress
*address
, char *path
, size_t sbuf
)
1141 struct sockaddr_un
*addr
;
1143 assert(address
!= NULL
);
1144 CHECK_ADDRESS(address
, UNIX
);
1146 addr
= (struct sockaddr_un
*)address
->m_addr
;
1148 strncpy(path
, addr
->sun_path
, sbuf
);
1150 return wxSOCKET_NOERROR
;
1152 #endif /* !defined(__VISAGECPP__) */
1154 #endif /* wxUSE_SOCKETS */