Add myself as libcli/dns maintainer
[Samba/gebeck_regimport.git] / testprogs / win32 / midltests / midltests_tcp.c
blob3046f6debc6db97e6ef8252c202bd4ce584bc298
1 /*
2 MIDLTESTS client.
4 Copyright (C) Stefan Metzmacher 2008
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 3 of the License, or
9 (at your option) any later version.
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with this program. If not, see <http://www.gnu.org/licenses/>.
20 #include <stdio.h>
21 #include <stdlib.h>
22 #include <ctype.h>
23 #include <winsock.h>
24 #include "midltests.h"
26 #ifndef _M_AMD64
27 #error "please run 'vcvarsall.bat amd64' -midltests_tcp needs 64-bit support!"
28 #endif
30 #define MIDLTESTS_C_CODE 1
31 #include "midltests.idl"
33 #ifndef LISTEN_IP
34 #define LISTEN_IP "127.0.0.1"
35 #endif
37 #ifndef FORWARD_IP
38 #define FORWARD_IP "127.0.0.1"
39 #endif
41 #ifndef CONNECT_IP
42 #define CONNECT_IP "127.0.0.1"
43 #endif
45 struct NDRTcpThreadCtx;
47 struct NDRProxyThreadCtx {
48 const struct NDRTcpThreadCtx *ctx;
49 SOCKET InSocket;
50 SOCKET OutSocket;
51 DWORD dwThreadId;
52 HANDLE hThread;
55 struct NDRTcpThreadCtx {
56 const char *name;
57 short listen_port;
58 short client_port;
59 BOOL ndr64;
60 BOOL stop;
63 struct dcerpc_header {
64 BYTE rpc_vers; /* RPC version */
65 BYTE rpc_vers_minor; /* Minor version */
66 BYTE ptype; /* Packet type */
67 BYTE pfc_flags; /* Fragmentation flags */
68 BYTE drep[4]; /* NDR data representation */
69 short frag_length; /* Total length of fragment */
70 short auth_length; /* authenticator length */
71 DWORD call_id; /* Call identifier */
74 static void dump_packet(const char *ctx, const char *direction,
75 const unsigned char *buf, int len)
77 struct dcerpc_header *hdr = (struct dcerpc_header *)buf;
79 if (len < sizeof(struct dcerpc_header)) {
80 printf("%s:%s: invalid dcerpc pdu len(%d)\n",
81 ctx, direction, len);
82 fflush(stdout);
83 return;
86 if (hdr->rpc_vers != 5 || hdr->rpc_vers_minor != 0) {
87 printf("%s:%s: invalid dcerpc pdu len(%d) ver:%d min:%d\n",
88 ctx, direction, len,
89 hdr->rpc_vers, hdr->rpc_vers_minor);
90 fflush(stdout);
91 return;
94 if (hdr->frag_length != len) {
95 printf("%s:%s: invalid dcerpc pdu len(%d) should be (%d)\n",
96 ctx, direction, len, hdr->frag_length);
97 fflush(stdout);
98 return;
102 switch (hdr->ptype) {
103 case 0: /* request */
104 printf("%s:%s: ptype[request] flen[%d] plen[%d] ahint[%d]\n\n",
105 ctx, direction, hdr->frag_length,
106 len - 24, *(DWORD *)(&buf[0x10]));
107 dump_data(buf + 24, len - 24);
108 printf("\n");
109 fflush(stdout);
110 break;
112 case 2: /* response */
113 printf("\n%s:%s: ptype[response] flen[%d] plen[%d] ahint[%d]\n\n",
114 ctx, direction, hdr->frag_length,
115 len - 24, *(DWORD *)(&buf[0x10]));
116 dump_data(buf + 24, len - 24);
117 printf("\n");
118 fflush(stdout);
119 break;
121 case 11: /* bind */
122 #if 0
123 printf("%s:%s: ptype[bind] flen[%d] call[%d] contexts[%d]\n\n"
124 ctx, direction, hdr->frag_length, hdr->call_id,
125 buf[24]);
126 dump_data(buf + 24, len - 24);
127 printf("\n");
128 fflush(stdout);
129 #endif
130 break;
132 case 12: /* bind ack */
133 #if 0
134 printf("%s:%s: ptype[bind_ack] flen[%d] call[%d]\n\n",
135 ctx, direction, hdr->frag_length, hdr->call_id);
136 fflush(stdout);
137 #endif
138 break;
140 case 14: /* alter_req */
141 #if 1
142 printf("%s:%s: ptype[alter_req] flen[%d] call[%d] contexts[%d]\n\n",
143 ctx, direction, hdr->frag_length, hdr->call_id,
144 buf[24]);
145 //dump_data(buf + 24, len - 24);
146 printf("\n");
147 fflush(stdout);
148 #endif
149 break;
151 case 15: /* alter_ack */
152 #if 1
153 printf("%s:%s: ptype[alter_ack] flen[%d] call[%d]\n\n",
154 ctx, direction, hdr->frag_length, hdr->call_id);
155 fflush(stdout);
156 #endif
157 break;
159 default:
160 printf("%s:%s: ptype[%d] flen[%d] call[%d]\n\n",
161 ctx, direction, hdr->ptype, hdr->frag_length, hdr->call_id);
162 fflush(stdout);
163 break;
167 static void change_packet(const char *ctx, BOOL ndr64,
168 unsigned char *buf, int len)
170 struct dcerpc_header *hdr = (struct dcerpc_header *)buf;
171 BOOL is_ndr64 = FALSE;
172 const unsigned char ndr64_buf[] = {
173 0x33, 0x05, 0x71, 0x71, 0xBA, 0xBE, 0x37, 0x49,
174 0x83, 0x19, 0xB5, 0xDB, 0xEF, 0x9C, 0xCC, 0x36
177 if (len < sizeof(struct dcerpc_header)) {
178 printf("%s: invalid dcerpc pdu len(%d)\n",
179 ctx, len);
180 fflush(stdout);
181 return;
184 if (hdr->rpc_vers != 5 || hdr->rpc_vers_minor != 0) {
185 printf("%s: invalid dcerpc pdu len(%d) ver:%d min:%d\n",
186 ctx, len,
187 hdr->rpc_vers, hdr->rpc_vers_minor);
188 fflush(stdout);
189 return;
192 if (hdr->frag_length != len) {
193 printf("%s: invalid dcerpc pdu len(%d) should be (%d)\n",
194 ctx, len, hdr->frag_length);
195 fflush(stdout);
196 return;
199 switch (hdr->ptype) {
200 case 11: /* bind */
201 case 14: /* alter_req */
203 if (buf[24] >= 2) {
204 int ret;
206 ret = memcmp(&buf[0x60], ndr64_buf, 16);
207 if (ret == 0) {
208 is_ndr64 = TRUE;
212 if (is_ndr64 && !ndr64) {
213 buf[24+0x48] = 0xFF;
214 memset(&buf[0x60], 0xFF, 16);
215 printf("%s: disable NDR64\n\n", ctx);
216 } else if (!is_ndr64 && ndr64) {
217 printf("\n%s: got NDR32 downgrade\n\n", ctx);
218 #ifndef DONOT_FORCE_NDR64
219 printf("\n\tERROR!!!\n\n");
220 memset(&buf[0x34], 0xFF, 16);
221 printf("You may need to run 'vcvarsall.bat amd64' before 'nmake tcp'\n");
222 #endif
223 printf("\n");
224 } else if (is_ndr64) {
225 printf("%s: got NDR64\n\n", ctx);
226 } else {
227 printf("%s: got NDR32\n\n", ctx);
229 //printf("%s: bind with %u pres\n", ctx, buf[24]);
230 fflush(stdout);
231 break;
235 static int sock_pending(SOCKET s)
237 int ret, error;
238 int value = 0;
239 int len;
241 ret = ioctlsocket(s, FIONREAD, &value);
242 if (ret == -1) {
243 return ret;
246 if (ret != 0) {
247 /* this should not be reached */
248 return -1;
251 if (value != 0) {
252 return value;
255 error = 0;
256 len = sizeof(error);
259 * if no data is available check if the socket is in error state. For
260 * dgram sockets it's the way to return ICMP error messages of
261 * connected sockets to the caller.
263 ret = getsockopt(s, SOL_SOCKET, SO_ERROR, (char *)&error, &len);
264 if (ret == -1) {
265 return ret;
267 if (error != 0) {
268 return -1;
270 return 0;
273 DWORD WINAPI NDRProxyThread(LPVOID lpParameter)
275 struct NDRProxyThreadCtx *p = (struct NDRProxyThreadCtx *)lpParameter;
277 while (!p->ctx->stop) {
278 int r, s;
279 int ret = -1;
280 BYTE buf[5840];
282 Sleep(250);
284 ret = sock_pending(p->InSocket);
285 if (ret == 0) {
286 goto out;
289 r = recv(p->InSocket, buf, sizeof(buf), 0);
290 if (r <= 0) {
291 ret = WSAGetLastError();
292 printf("%s: recv(in) failed[%d][%d]\n", p->ctx->name, r, ret);
293 fflush(stdout);
294 goto stop;
297 change_packet(p->ctx->name, p->ctx->ndr64, buf, r);
298 fflush(stdout);
300 dump_packet(p->ctx->name, "in => out", buf, r);
301 fflush(stdout);
303 out:
304 s = send(p->OutSocket, buf, r, 0);
305 if (s <= 0) {
306 ret = WSAGetLastError();
307 printf("%s: send(out) failed[%d][%d]\n", p->ctx->name, s, ret);
308 fflush(stdout);
309 goto stop;
312 ret = sock_pending(p->OutSocket);
313 if (ret == 0) {
314 goto next;
317 r = recv(p->OutSocket, buf, sizeof(buf), 0);
318 if (r <= 0) {
319 ret = WSAGetLastError();
320 printf("%s: recv(out) failed[%d][%d]\n", p->ctx->name, r, ret);
321 fflush(stdout);
322 goto stop;
325 dump_packet(p->ctx->name, "out => in", buf, r);
326 fflush(stdout);
328 s = send(p->InSocket, buf, r, 0);
329 if (s <= 0) {
330 ret = WSAGetLastError();
331 printf("%s: send(in) failed[%d][%d]\n", p->ctx->name, s, ret);
332 fflush(stdout);
333 goto stop;
335 next:
336 continue;
338 stop:
339 closesocket(p->InSocket);
340 closesocket(p->OutSocket);
342 printf("NDRTcpThread[%s] stop\n", p->ctx->name);
343 fflush(stdout);
344 return 0;
347 DWORD WINAPI NDRTcpThread(LPVOID lpParameter)
349 struct NDRTcpThreadCtx *ctx = (struct NDRTcpThreadCtx *)lpParameter;
350 int ret = -1;
351 SOCKET ListenSocket;
352 struct sockaddr_in saServer;
353 struct sockaddr_in saClient;
355 //printf("NDRTcpThread[%s] start\n", ctx->name);
356 fflush(stdout);
358 ListenSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
359 if (ListenSocket == INVALID_SOCKET) {
360 ret = WSAGetLastError();
361 printf("socket() failed[%d][%d]\n", ListenSocket, ret);
362 fflush(stdout);
363 goto failed;
366 saServer.sin_family = AF_INET;
367 saServer.sin_addr.s_addr = inet_addr(LISTEN_IP);
368 saServer.sin_port = htons(ctx->listen_port);
370 saClient.sin_family = AF_INET;
371 saClient.sin_addr.s_addr = inet_addr(FORWARD_IP);
372 saClient.sin_port = htons(ctx->client_port);
374 ret = bind(ListenSocket, (SOCKADDR*)&saServer, sizeof(saServer));
375 if (ret == SOCKET_ERROR) {
376 ret = WSAGetLastError();
377 printf("bind() failed[%d]\n", ret);
378 fflush(stdout);
379 goto failed;
382 ret = listen(ListenSocket, 10);
383 if (ret == SOCKET_ERROR) {
384 ret = WSAGetLastError();
385 printf("listen() failed[%d]\n", ret);
386 fflush(stdout);
387 goto failed;
390 while (!ctx->stop) {
391 struct sockaddr_in sa;
392 int sa_len = sizeof(sa);
393 struct NDRProxyThreadCtx *p = malloc(sizeof(*p));
394 p->ctx = ctx;
396 p->InSocket = accept(ListenSocket, (SOCKADDR *)&sa, &sa_len);
397 if (p->InSocket == INVALID_SOCKET) {
398 ret = WSAGetLastError();
399 printf("%s: accept() failed[%d][%d]\n", p->ctx->name, p->InSocket, ret);
400 fflush(stdout);
401 continue;
404 p->OutSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
405 if (p->OutSocket == INVALID_SOCKET) {
406 ret = WSAGetLastError();
407 printf("%s: socket(out) failed[%d][%d]\n", p->ctx->name, p->OutSocket, ret);
408 fflush(stdout);
409 closesocket(p->InSocket);
410 continue;
413 ret = connect(p->OutSocket, (SOCKADDR*)&saClient, sizeof(saClient));
414 if (ret == SOCKET_ERROR) {
415 ret = WSAGetLastError();
416 printf("%s: connect() failed[%d]\n", p->ctx->name, ret);
417 fflush(stdout);
418 closesocket(p->InSocket);
419 closesocket(p->OutSocket);
420 continue;
423 p->hThread = CreateThread(
424 NULL, // default security attributes
425 0, // use default stack size
426 NDRProxyThread, // thread function name
427 p, // argument to thread function
428 0, // use default creation flags
429 &p->dwThreadId);// returns the thread identifier
430 if (p->hThread == NULL) {
431 printf("failed to create thread ndr32\n");
432 fflush(stdout);
433 return -1;
437 //printf("NDRTcpThread[%s] stop\n", ctx->name);
438 fflush(stdout);
439 return 0;
440 failed:
441 printf("NDRTcpThread[%s] failed[%d]\n", ctx->name, ret);
442 fflush(stdout);
443 return ret;
446 struct NDRRpcThreadCtx {
447 const char *name;
448 short listen_port;
451 DWORD WINAPI NDRRpcThread(LPVOID lpParameter)
453 struct NDRRpcThreadCtx *ctx = (struct NDRRpcThreadCtx *)lpParameter;
454 int ret = -1;
455 RPC_STATUS status;
456 RPC_BINDING_VECTOR *pBindingVector;
458 #define RPC_MIN_CALLS 1
459 #define RPC_MAX_CALLS 20
461 //printf("NDRRpcThread[%s] start\n", ctx->name);
462 fflush(stdout);
463 status = RpcServerUseProtseqEp("ncacn_ip_tcp", RPC_MAX_CALLS, "5055", NULL);
464 if (status) {
465 printf("Failed to register ncacn_ip_tcp endpoint\n");
466 fflush(stdout);
467 return status;
470 status = RpcServerInqBindings(&pBindingVector);
471 if (status) {
472 printf("Failed RpcServerInqBindings\n");
473 fflush(stdout);
474 return status;
477 #if 0
478 status = RpcEpRegister(srv_midltests_v0_0_s_ifspec, pBindingVector, NULL, "midltests server");
479 if (status) {
480 printf("Failed RpcEpRegister\n");
481 fflush(stdout);
482 return status;
484 #endif
485 status = RpcServerRegisterIf(srv_midltests_v0_0_s_ifspec, NULL, NULL);
486 if (status) {
487 printf("Failed to register interface\n");
488 fflush(stdout);
489 return status;
492 status = RpcServerListen(RPC_MIN_CALLS, RPC_MAX_CALLS, FALSE);
493 if (status) {
494 printf("RpcServerListen returned error %d\n", status);
495 fflush(stdout);
496 return status;
499 printf("NDRRpcThread[%s] stop\n", ctx->name);
500 fflush(stdout);
501 return 0;
502 failed:
503 printf("NDRRpcThread[%s] failed[%d]\n", ctx->name, ret);
504 fflush(stdout);
505 return ret;
508 int main(int argc, char **argv)
510 int ret;
511 struct NDRTcpThreadCtx ctx_ndr32;
512 struct NDRTcpThreadCtx ctx_ndr64;
513 struct NDRRpcThreadCtx ctx_rpc;
514 DWORD dwThreadIdArray[3];
515 HANDLE hThreadArray[3];
516 WORD wVersionRequested = MAKEWORD(2, 2);
517 WSADATA wsaData;
518 char *binding;
519 RPC_STATUS status;
521 ret = WSAStartup(wVersionRequested, &wsaData);
522 if (ret != 0) {
523 printf("WSAStartup failed with error: %d\n", ret);
524 fflush(stdout);
525 return -1;
528 ctx_ndr32.name = "ndr32";
529 ctx_ndr32.listen_port = 5032;
530 ctx_ndr32.client_port = 5055;
531 ctx_ndr32.ndr64 = FALSE;
532 ctx_ndr32.stop = FALSE;
533 hThreadArray[0] = CreateThread(
534 NULL, // default security attributes
535 0, // use default stack size
536 NDRTcpThread, // thread function name
537 &ctx_ndr32, // argument to thread function
538 0, // use default creation flags
539 &dwThreadIdArray[0]); // returns the thread identifier
540 if (hThreadArray[0] == NULL) {
541 printf("failed to create thread ndr32\n");
542 fflush(stdout);
543 return -1;
546 ctx_ndr64.name = "ndr64";
547 ctx_ndr64.listen_port = 5064;
548 ctx_ndr64.client_port = 5055;
549 ctx_ndr64.ndr64 = TRUE;
550 ctx_ndr64.stop = FALSE;
551 hThreadArray[1] = CreateThread(
552 NULL, // default security attributes
553 0, // use default stack size
554 NDRTcpThread, // thread function name
555 &ctx_ndr64, // argument to thread function
556 0, // use default creation flags
557 &dwThreadIdArray[1]); // returns the thread identifier
558 if (hThreadArray[1] == NULL) {
559 printf("failed to create thread ndr64\n");
560 fflush(stdout);
561 return -1;
564 ctx_rpc.name = "rpc";
565 ctx_rpc.listen_port = 5050;
566 hThreadArray[2] = CreateThread(
567 NULL, // default security attributes
568 0, // use default stack size
569 NDRRpcThread, // thread function name
570 &ctx_rpc, // argument to thread function
571 0, // use default creation flags
572 &dwThreadIdArray[2]); // returns the thread identifier
573 if (hThreadArray[2] == NULL) {
574 printf("failed to create thread rpc\n");
575 fflush(stdout);
576 return -1;
579 printf("Wait for setup of server threads\n");
580 fflush(stdout);
581 ret = WaitForMultipleObjects(3, hThreadArray, TRUE, 3000);
582 if (ret == WAIT_TIMEOUT) {
583 /* OK */
584 } else {
585 printf("Failed to setup of server threads %d:%d\n",
586 ret, GetLastError());
587 fflush(stdout);
588 return -1;
590 ret = 0;
592 printf("\nTest NDR32\n\n");
593 fflush(stdout);
594 binding = "ncacn_ip_tcp:" CONNECT_IP "[5032]";
595 status = RpcBindingFromStringBinding(
596 binding,
597 &midltests_IfHandle);
598 if (status) {
599 printf("RpcBindingFromStringBinding returned %d\n", status);
600 fflush(stdout);
601 return status;
604 RpcTryExcept {
605 midltests();
606 } RpcExcept(1) {
607 ret = RpcExceptionCode();
608 printf("NDR32 Runtime error 0x%x\n", ret);
609 fflush(stdout);
610 } RpcEndExcept
611 ctx_ndr32.stop = TRUE;
613 Sleep(250);
615 printf("\nTest NDR64\n\n");
616 binding = "ncacn_ip_tcp:" CONNECT_IP "[5064]";
617 status = RpcBindingFromStringBinding(
618 binding,
619 &midltests_IfHandle);
620 if (status) {
621 printf("RpcBindingFromStringBinding returned %d\n", status);
622 fflush(stdout);
623 return status;
626 RpcTryExcept {
627 midltests();
628 } RpcExcept(1) {
629 ret = RpcExceptionCode();
630 printf("Runtime error 0x%x\n", ret);
631 fflush(stdout);
632 } RpcEndExcept
633 ctx_ndr64.stop = TRUE;
635 WaitForMultipleObjects(3, hThreadArray, TRUE, 2000);
637 if (ret == 0) {
638 printf("\nTest OK\n");
639 fflush(stdout);
640 } else {
641 printf("\nTest FAILED[%d]\n", ret);
642 fflush(stdout);
645 return ret;