ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/IO-FDPass/FDPass.xs
Revision: 1.1
Committed: Fri Apr 5 04:10:51 2013 UTC (11 years, 1 month ago) by root
Branch: MAIN
Log Message:
*** empty log message ***

File Contents

# Content
1 /* GetCurrentProcessId is XP and up, which means in all supported versions */
2 /* but older SDK's might need this */
3 #define _WIN32_WINNT NTDDI_WINXP
4
5 #ifdef __sun
6 #define _XOPEN_SOURCE 1
7 #define _XOPEN_SOURCE_EXTENDED 1
8 #define __EXTENSIONS__ 1
9 #endif
10
11 #include "EXTERN.h"
12 #include "perl.h"
13 #include "XSUB.h"
14
15 #if WIN32
16
17 /* perl probably did this already */
18 #include <windows.h>
19
20 #elif __CYGWIN__
21
22 #include <windows.h>
23 #include <io.h>
24 #include <sys/cygwin.h>
25
26 #define ioctlsocket(a,b,c) ioctl (a, b, c)
27 #define _open_osfhandle(h,m) cygwin_attach_handle_to_fd ("/dev/pipe", -1, (HANDLE)h, 1, GENERIC_READ | GENERIC_WRITE)
28 typedef int SOCKET;
29
30 #else
31
32 #include <stddef.h> // needed by broken bsds for NULL used in sys/uio.h
33 #include <stdlib.h>
34 #include <errno.h>
35
36 /* send_fd/recv_fd taken from libptytty */
37 #include <sys/types.h>
38 #include <sys/uio.h>
39 #include <sys/socket.h>
40
41 #ifndef CMSG_SPACE
42 # define CMSG_SPACE(len) (sizeof (cmsghdr) + len)
43 #endif
44
45 #ifndef CMSG_LEN
46 # define CMSG_LEN(len) (sizeof (cmsghdr) + len)
47 #endif
48
49 #endif
50
51 #if defined(WIN32)
52 /* the rub is this: win32 doesn't seem to have a way to query whether a socket */
53 /* is non-blocking or not. so we assume it is blocking, make it so if it isn't */
54 /* and reset it afterwards */
55 static int
56 rw (int wr, int fd, char *buf, int len)
57 {
58 u_long nbio = 0;
59 int got = 0;
60
61 while (got != len)
62 {
63 int sze = wr
64 ? send ((SOCKET)fd, buf, len) /* we assume send and recv are macros with arguments */
65 : recv ((SOCKET)fd, buf, len); /* to be on the safe side */
66
67 if (sze <= 0)
68 {
69 if (errno == EAGAIN || errno == EWOULDBLOCK || errno == WSAWOULDBLOCK)
70 {
71 ioctlsocket (s, FIONBIO, &nbio);
72 nbio = 1;
73 }
74 else
75 break;
76 }
77 else
78 got += sze;
79 }
80
81 if (nbio)
82 ioctlsocket (s, FIONBIO, &nbio);
83
84 return got == len;
85 }
86 #endif
87
88 static int
89 fd_send (int socket, int fd)
90 {
91 #if defined(WIN32)
92 DWORD pid;
93 HANDLE target, h;
94
95 /* seriously, there is no way to query whether a socket is non-blocking?? */
96 if (rw (0, socket, (char *)&pid, sizeof (pid), 0) != sizeof (pid))
97 return 0;
98
99 target = OpenProcess (PROCESS_DUP_HANDLE, FALSE, pid);
100 if (!target)
101 croak ("AnyEvent::ProcessPool::fd_recv: OpenProcess failed");
102
103 if (!DuplicateHandle ((HANDLE)-1, (HANDLE)_get_osfhandle (fd), target, &h, 0, FALSE, DUPLICATE_SAME_ACCESS))
104 croak ("AnyEvent::ProcessPool::fd_recv: DuplicateHandle failed");
105
106 CloseHandle (target);
107
108 if (rw (1, socket, (char *)&h, sizeof (h), 0) != sizeof (h))
109 return 0;
110
111 return 1;
112
113 #else
114 void *buf = malloc (CMSG_SPACE (sizeof (int)));
115
116 if (!buf)
117 return 0;
118
119 struct msghdr msg;
120 struct iovec iov;
121 struct cmsghdr *cmsg;
122 char data = 0;
123
124 iov.iov_base = &data;
125 iov.iov_len = 1;
126
127 msg.msg_name = 0;
128 msg.msg_namelen = 0;
129 msg.msg_iov = &iov;
130 msg.msg_iovlen = 1;
131 msg.msg_control = buf;
132 msg.msg_controllen = CMSG_SPACE (sizeof (int));
133
134 cmsg = CMSG_FIRSTHDR (&msg);
135 cmsg->cmsg_level = SOL_SOCKET;
136 cmsg->cmsg_type = SCM_RIGHTS;
137 cmsg->cmsg_len = CMSG_LEN (sizeof (int));
138
139 *(int *)CMSG_DATA (cmsg) = fd;
140
141 ssize_t result = sendmsg (socket, &msg, 0);
142
143 free (buf);
144
145 return result >= 0;
146 #endif
147 }
148
149 static int
150 fd_recv (int socket)
151 {
152 #if defined(WIN32)
153 DWORD pid = GetCurrentProcessId ();
154 HANDLE h;
155
156 if (rw (1, socket, (char *)&pid, sizeof (pid), 0) != sizeof (pid))
157 return -1;
158
159 if (rw (0, socket, (char *)&h, sizeof (h), 0) != sizeof (h))
160 return -1;
161
162 return _open_osfhandle ((intptr_t)h, 0);
163 #else
164 void *buf = malloc (CMSG_SPACE (sizeof (int)));
165
166 if (!buf)
167 return -1;
168
169 struct msghdr msg;
170 struct iovec iov;
171 char data = 1;
172
173 iov.iov_base = &data;
174 iov.iov_len = 1;
175
176 msg.msg_name = 0;
177 msg.msg_namelen = 0;
178 msg.msg_iov = &iov;
179 msg.msg_iovlen = 1;
180 msg.msg_control = buf;
181 msg.msg_controllen = CMSG_SPACE (sizeof (int));
182
183 if (recvmsg (socket, &msg, 0) <= 0)
184 return -1;
185
186 int fd = -1;
187
188 errno = EDOM;
189
190 if (
191 data == 0
192 #if __OpenBSD__
193 && msg.msg_controllen >= CMSG_LEN (sizeof (int)) /* work around a bug in at least openbsd 4.5 and 4.8 */
194 #else
195 && msg.msg_controllen >= CMSG_SPACE (sizeof (int))
196 #endif
197 ) {
198 struct cmsghdr *cmsg = CMSG_FIRSTHDR (&msg);
199
200 if (cmsg->cmsg_level == SOL_SOCKET
201 && cmsg->cmsg_type == SCM_RIGHTS
202 && cmsg->cmsg_len >= CMSG_LEN (sizeof (int)))
203 fd = *(int *)CMSG_DATA (cmsg);
204 }
205
206 free (buf);
207
208 return fd;
209 #endif
210 }
211
212 MODULE = IO::FDPass PACKAGE = IO::FDPass PREFIX = fd_
213
214 PROTOTYPES: DISABLE
215
216 int
217 fd_send (int socket, int fd)
218
219 int
220 fd_recv (int socket)
221