Skip to content

Commit 7066cc7

Browse files
committed
send/recvmsg() support for Windows
1 parent 8561680 commit 7066cc7

12 files changed

+300
-54
lines changed

ext/sockets/config.w32

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ if (PHP_SOCKETS != "no") {
77
if (CHECK_LIB("ws2_32.lib", "sockets", PHP_SOCKETS)
88
&& CHECK_LIB("Iphlpapi.lib", "sockets", PHP_SOCKETS)
99
&& CHECK_HEADER_ADD_INCLUDE("winsock.h", "CFLAGS_SOCKETS")) {
10-
EXTENSION('sockets', 'sockets.c multicast.c');
10+
EXTENSION('sockets', 'sockets.c multicast.c conversions.c sockaddr_conv.c sendrecvmsg.c');
1111
AC_DEFINE('HAVE_SOCKETS', 1);
1212
PHP_INSTALL_HEADERS("ext/sockets", "php_sockets.h");
1313
} else {

ext/sockets/conversions.c

Lines changed: 56 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -1,24 +1,67 @@
1-
#include "conversions.h"
21
#include "sockaddr_conv.h"
32
#include "conversions.h"
43
#include "sendrecvmsg.h" /* for ancillary registry */
4+
#include "windows_common.h"
55

66
#include <Zend/zend_llist.h>
77
#include <ext/standard/php_smart_str.h>
88

9-
#include <sys/types.h>
10-
#include <sys/socket.h>
11-
#include <arpa/inet.h>
12-
#include <netinet/in.h>
13-
#include <sys/un.h>
14-
15-
#include <sys/ioctl.h>
16-
#include <net/if.h>
9+
#ifndef PHP_WIN32
10+
# include <sys/types.h>
11+
# include <sys/socket.h>
12+
# include <arpa/inet.h>
13+
# include <netinet/in.h>
14+
# include <sys/un.h>
15+
# include <sys/ioctl.h>
16+
# include <net/if.h>
17+
#else
18+
# include <win32/php_stdint.h>
19+
#endif
1720

1821
#include <limits.h>
1922
#include <stdarg.h>
2023
#include <stddef.h>
2124

25+
#ifdef PHP_WIN32
26+
typedef unsigned short sa_family_t;
27+
# define msghdr _WSAMSG
28+
/*
29+
struct _WSAMSG {
30+
LPSOCKADDR name; //void *msg_name
31+
INT namelen; //socklen_t msg_namelen
32+
LPWSABUF lpBuffers; //struct iovec *msg_iov
33+
ULONG dwBufferCount; //size_t msg_iovlen
34+
WSABUF Control; //void *msg_control, size_t msg_controllen
35+
DWORD dwFlags; //int msg_flags
36+
}
37+
struct __WSABUF {
38+
u_long len; //size_t iov_len (2nd member)
39+
char FAR *buf; //void *iov_base (1st member)
40+
}
41+
struct _WSACMSGHDR {
42+
UINT cmsg_len; //socklen_t cmsg_len
43+
INT cmsg_level; //int cmsg_level
44+
INT cmsg_type; //int cmsg_type;
45+
followed by UCHAR cmsg_data[]
46+
}
47+
*/
48+
# define msg_name name
49+
# define msg_namelen namelen
50+
# define msg_iov lpBuffers
51+
# define msg_iovlen dwBufferCount
52+
# define msg_control Control.buf
53+
# define msg_controllen Control.len
54+
# define msg_flags dwFlags
55+
# define iov_base buf
56+
# define iov_len len
57+
58+
# define cmsghdr _WSACMSGHDR
59+
# ifdef CMSG_DATA
60+
# undef CMSG_DATA
61+
# endif
62+
# define CMSG_DATA WSA_CMSG_DATA
63+
#endif
64+
2265
#define MAX_USER_BUFF_SIZE ((size_t)(100*1024*1024))
2366
#define DEFAULT_BUFF_SIZE 8192
2467

@@ -132,7 +175,7 @@ static void do_from_to_zval_err(struct err_s *err,
132175
efree(user_msg);
133176
smart_str_free_ex(&path, 0);
134177
}
135-
__attribute__ ((format (printf, 2, 3)))
178+
ZEND_ATTRIBUTE_FORMAT(printf, 2 ,3)
136179
static void do_from_zval_err(ser_context *ctx, const char *fmt, ...)
137180
{
138181
va_list ap;
@@ -141,7 +184,7 @@ static void do_from_zval_err(ser_context *ctx, const char *fmt, ...)
141184
do_from_to_zval_err(&ctx->err, &ctx->keys, "user", fmt, ap);
142185
va_end(ap);
143186
}
144-
__attribute__ ((format (printf, 2, 3)))
187+
ZEND_ATTRIBUTE_FORMAT(printf, 2 ,3)
145188
static void do_to_zval_err(res_context *ctx, const char *fmt, ...)
146189
{
147190
va_list ap;
@@ -958,7 +1001,7 @@ static void to_zval_read_control_array(const char *msghdr_c, zval *zv, res_conte
9581001

9591002
for (cmsg = CMSG_FIRSTHDR(msg);
9601003
cmsg != NULL && !ctx->err.has_error;
961-
cmsg = CMSG_NXTHDR(msg,cmsg)) {
1004+
cmsg = CMSG_NXTHDR(msg, cmsg)) {
9621005
zval *elem;
9631006

9641007
ALLOC_INIT_ZVAL(elem);
@@ -1149,7 +1192,7 @@ static void to_zval_read_iov(const char *msghdr_c, zval *zv, res_context *ctx)
11491192

11501193
for (i = 0; bytes_left > 0 && i < (uint)iovlen; i++) {
11511194
zval *elem;
1152-
size_t len = MIN(msghdr->msg_iov[i].iov_len, bytes_left);
1195+
size_t len = MIN(msghdr->msg_iov[i].iov_len, (size_t)bytes_left);
11531196
char *buf = safe_emalloc(1, len, 1);
11541197

11551198
MAKE_STD_ZVAL(elem);

ext/sockets/conversions.h

Lines changed: 8 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,14 @@
22
#define PHP_SOCK_CONVERSIONS_H 1
33

44
#include <php.h>
5-
#include <netinet/in.h>
6-
#include <sys/socket.h>
5+
6+
#ifndef PHP_WIN32
7+
# include <netinet/in.h>
8+
# include <sys/socket.h>
9+
#else
10+
# include <Ws2tcpip.h>
11+
#endif
12+
713
#include "php_sockets.h"
814

915
/* TYPE DEFINITIONS */

ext/sockets/php_sockets.h

Lines changed: 3 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -26,11 +26,13 @@
2626

2727
#if HAVE_SOCKETS
2828

29+
#include <php.h>
30+
2931
extern zend_module_entry sockets_module_entry;
3032
#define phpext_sockets_ptr &sockets_module_entry
3133

3234
#ifdef PHP_WIN32
33-
#include <winsock.h>
35+
#include <Winsock2.h>
3436
#else
3537
#if HAVE_SYS_SOCKET_H
3638
#include <sys/socket.h>

ext/sockets/sendrecvmsg.c

Lines changed: 51 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,44 @@
3030
#define DEFAULT_BUFF_SIZE 8192
3131
#define MAX_ARRAY_KEY_SIZE 128
3232

33+
#ifdef PHP_WIN32
34+
#include "windows_common.h"
35+
#include <Mswsock.h>
36+
#define IPV6_RECVPKTINFO IPV6_PKTINFO
37+
#define IPV6_RECVHOPLIMIT IPV6_HOPLIMIT
38+
#define msghdr _WSAMSG
39+
40+
static GUID WSARecvMsg_GUID = WSAID_WSARECVMSG;
41+
static __declspec(thread) LPFN_WSARECVMSG WSARecvMsg = NULL;
42+
inline ssize_t recvmsg(int sockfd, struct msghdr *msg, int flags)
43+
{
44+
DWORD recvd = 0,
45+
bytesReturned;
46+
47+
if (WSARecvMsg == NULL) {
48+
int res = WSAIoctl((SOCKET) sockfd, SIO_GET_EXTENSION_FUNCTION_POINTER,
49+
&WSARecvMsg_GUID, sizeof(WSARecvMsg_GUID),
50+
&WSARecvMsg, sizeof(WSARecvMsg),
51+
&bytesReturned, NULL, NULL);
52+
if (res != 0) {
53+
return -1;
54+
}
55+
}
56+
57+
msg->dwFlags = (DWORD)flags;
58+
return WSARecvMsg((SOCKET)sockfd, msg, &recvd, NULL, NULL) == 0
59+
? (ssize_t)recvd
60+
: -1;
61+
}
62+
inline ssize_t sendmsg(int sockfd, const struct msghdr *msg, int flags)
63+
{
64+
DWORD sent = 0;
65+
return WSASendMsg((SOCKET)sockfd, (struct msghdr*)msg, (DWORD)flags, &sent, NULL, NULL) == 0
66+
? (ssize_t)sent
67+
: -1;
68+
}
69+
#endif
70+
3371
#define LONG_CHECK_VALID_INT(l) \
3472
do { \
3573
if ((l) < INT_MIN && (l) > INT_MAX) { \
@@ -158,9 +196,7 @@ PHP_FUNCTION(socket_sendmsg)
158196

159197
RETURN_LONG((long)res);
160198
} else {
161-
SOCKETS_G(last_error) = errno;
162-
php_error_docref(NULL TSRMLS_CC, E_WARNING, "error in sendmsg [%d]: %s",
163-
errno, sockets_strerror(errno TSRMLS_CC));
199+
PHP_SOCKET_ERROR(php_sock, "error in sendmsg", errno);
164200
RETURN_FALSE;
165201
}
166202
}
@@ -285,6 +321,18 @@ int php_do_setsockopt_ipv6_rfc3542(php_socket *php_sock, int level, int optname,
285321
switch (optname) {
286322
#ifdef IPV6_PKTINFO
287323
case IPV6_PKTINFO:
324+
#ifdef PHP_WIN32
325+
if (Z_TYPE_PP(arg4) == IS_ARRAY) {
326+
php_error_docref0(NULL TSRMLS_CC, E_WARNING, "Windows does not "
327+
"support sticky IPV6_PKTINFO");
328+
return FAILURE;
329+
} else {
330+
/* windows has no IPV6_RECVPKTINFO, and uses IPV6_PKTINFO
331+
* for the same effect. We define IPV6_RECVPKTINFO to be
332+
* IPV6_PKTINFO, so assume the assume user used IPV6_RECVPKTINFO */
333+
return 1;
334+
}
335+
#endif
288336
opt_ptr = from_zval_run_conversions(*arg4, php_sock, from_zval_write_in6_pktinfo,
289337
sizeof(struct in6_pktinfo), "in6_pktinfo", &allocations, &err);
290338
if (err.has_error) {

ext/sockets/sockaddr_conv.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
#include "php_sockets.h"
44

55
#ifdef PHP_WIN32
6-
#include <Ws2tcpip.h>
6+
#include "windows_common.h"
77
#else
88
#include <netdb.h>
99
#include <arpa/inet.h>

ext/sockets/sockaddr_conv.h

Lines changed: 9 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,16 @@
11
#ifndef PHP_SOCKADR_CONV_H
22
#define PHP_SOCKADR_CONV_H
33

4+
#define HAVE_SOCKETS 1
45
#include <php_network.h>
5-
#include "php_sockets.h"
6+
#include "php_sockets.h" /* php_socket */
7+
8+
#ifndef PHP_WIN32
9+
# include <netinet/in.h>
10+
#else
11+
# include <Winsock2.h>
12+
#endif
13+
614

715
/*
816
* Convert an IPv6 literal or a hostname info a sockaddr_in6.

ext/sockets/sockets.c

Lines changed: 7 additions & 23 deletions
Original file line numberDiff line numberDiff line change
@@ -35,32 +35,12 @@
3535
#include "ext/standard/info.h"
3636
#include "php_ini.h"
3737
#ifdef PHP_WIN32
38-
# include "win32/inet.h"
39-
# include <winsock2.h>
38+
# include "windows_common.h"
39+
# include <win32/inet.h>
4040
# include <windows.h>
4141
# include <Ws2tcpip.h>
4242
# include "php_sockets.h"
43-
# include "win32/sockets.h"
44-
# define IS_INVALID_SOCKET(a) (a->bsd_socket == INVALID_SOCKET)
45-
# ifdef EPROTONOSUPPORT
46-
# undef EPROTONOSUPPORT
47-
# endif
48-
# ifdef ECONNRESET
49-
# undef ECONNRESET
50-
# endif
51-
# define EPROTONOSUPPORT WSAEPROTONOSUPPORT
52-
# define ECONNRESET WSAECONNRESET
53-
# ifdef errno
54-
# undef errno
55-
# endif
56-
# define errno WSAGetLastError()
57-
# define h_errno WSAGetLastError()
58-
# define set_errno(a) WSASetLastError(a)
59-
# define close(a) closesocket(a)
60-
# include <IPHlpApi.h>
61-
# if _WIN32_WINNT >= 0x0600
62-
# define HAVE_IF_NAMETOINDEX 1
63-
# endif
43+
# include <win32/sockets.h>
6444
#else
6545
# include <sys/types.h>
6646
# include <sys/socket.h>
@@ -650,8 +630,12 @@ PHP_MINIT_FUNCTION(sockets)
650630
REGISTER_LONG_CONSTANT("MSG_TRUNC", MSG_TRUNC, CONST_CS | CONST_PERSISTENT);
651631
REGISTER_LONG_CONSTANT("MSG_PEEK", MSG_PEEK, CONST_CS | CONST_PERSISTENT);
652632
REGISTER_LONG_CONSTANT("MSG_DONTROUTE", MSG_DONTROUTE, CONST_CS | CONST_PERSISTENT);
633+
#ifdef MSG_EOR
653634
REGISTER_LONG_CONSTANT("MSG_EOR", MSG_EOR, CONST_CS | CONST_PERSISTENT);
635+
#endif
636+
#ifdef MSG_EOF
654637
REGISTER_LONG_CONSTANT("MSG_EOF", MSG_EOF, CONST_CS | CONST_PERSISTENT);
638+
#endif
655639

656640
#ifdef MSG_CONFIRM
657641
REGISTER_LONG_CONSTANT("MSG_CONFIRM", MSG_CONFIRM, CONST_CS | CONST_PERSISTENT);

0 commit comments

Comments
 (0)