/nsprpub/pr/tests/peek.c

https://github.com/rillian/firefox · C · 360 lines · 314 code · 25 blank · 21 comment · 106 complexity · 3c7581191786a8536caf0505a0c423af MD5 · raw file

  1. /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
  2. /* This Source Code Form is subject to the terms of the Mozilla Public
  3. * License, v. 2.0. If a copy of the MPL was not distributed with this
  4. * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
  5. /*
  6. * A test case for the PR_MSG_PEEK flag of PR_Recv().
  7. *
  8. * Test both blocking and non-blocking sockets.
  9. */
  10. #include "nspr.h"
  11. #include <stdio.h>
  12. #include <stdlib.h>
  13. #include <string.h>
  14. #define BUFFER_SIZE 1024
  15. static int iterations = 10;
  16. /*
  17. * In iteration i, recv_amount[i] is the number of bytes we
  18. * wish to receive, and send_amount[i] is the number of bytes
  19. * we actually send. Therefore, the number of elements in the
  20. * recv_amount or send_amount array should equal to 'iterations'.
  21. * For this test to pass we need to ensure that
  22. * recv_amount[i] <= BUFFER_SIZE,
  23. * send_amount[i] <= BUFFER_SIZE,
  24. * send_amount[i] <= recv_amount[i].
  25. */
  26. static PRInt32 recv_amount[10] = {
  27. 16, 128, 256, 1024, 512, 512, 128, 256, 32, 32};
  28. static PRInt32 send_amount[10] = {
  29. 16, 64, 128, 1024, 512, 256, 128, 64, 16, 32};
  30. /* Blocking I/O */
  31. static void ServerB(void *arg)
  32. {
  33. PRFileDesc *listenSock = (PRFileDesc *) arg;
  34. PRFileDesc *sock;
  35. char buf[BUFFER_SIZE];
  36. PRInt32 nbytes;
  37. int i;
  38. int j;
  39. sock = PR_Accept(listenSock, NULL, PR_INTERVAL_NO_TIMEOUT);
  40. if (NULL == sock) {
  41. fprintf(stderr, "PR_Accept failed\n");
  42. exit(1);
  43. }
  44. for (i = 0; i < iterations; i++) {
  45. memset(buf, 0, sizeof(buf));
  46. nbytes = PR_Recv(sock, buf, recv_amount[i],
  47. PR_MSG_PEEK, PR_INTERVAL_NO_TIMEOUT);
  48. if (-1 == nbytes) {
  49. fprintf(stderr, "PR_Recv failed\n");
  50. exit(1);
  51. }
  52. if (send_amount[i] != nbytes) {
  53. fprintf(stderr, "PR_Recv returned %d, absurd!\n", nbytes);
  54. exit(1);
  55. }
  56. for (j = 0; j < nbytes; j++) {
  57. if (buf[j] != 2*i) {
  58. fprintf(stderr, "byte %d should be %d but is %d\n",
  59. j, 2*i, buf[j]);
  60. exit(1);
  61. }
  62. }
  63. fprintf(stderr, "server: peeked expected data\n");
  64. memset(buf, 0, sizeof(buf));
  65. nbytes = PR_Recv(sock, buf, recv_amount[i],
  66. PR_MSG_PEEK, PR_INTERVAL_NO_TIMEOUT);
  67. if (-1 == nbytes) {
  68. fprintf(stderr, "PR_Recv failed\n");
  69. exit(1);
  70. }
  71. if (send_amount[i] != nbytes) {
  72. fprintf(stderr, "PR_Recv returned %d, absurd!\n", nbytes);
  73. exit(1);
  74. }
  75. for (j = 0; j < nbytes; j++) {
  76. if (buf[j] != 2*i) {
  77. fprintf(stderr, "byte %d should be %d but is %d\n",
  78. j, 2*i, buf[j]);
  79. exit(1);
  80. }
  81. }
  82. fprintf(stderr, "server: peeked expected data\n");
  83. memset(buf, 0, sizeof(buf));
  84. nbytes = PR_Recv(sock, buf, recv_amount[i],
  85. 0, PR_INTERVAL_NO_TIMEOUT);
  86. if (-1 == nbytes) {
  87. fprintf(stderr, "PR_Recv failed\n");
  88. exit(1);
  89. }
  90. if (send_amount[i] != nbytes) {
  91. fprintf(stderr, "PR_Recv returned %d, absurd!\n", nbytes);
  92. exit(1);
  93. }
  94. for (j = 0; j < nbytes; j++) {
  95. if (buf[j] != 2*i) {
  96. fprintf(stderr, "byte %d should be %d but is %d\n",
  97. j, 2*i, buf[j]);
  98. exit(1);
  99. }
  100. }
  101. fprintf(stderr, "server: received expected data\n");
  102. PR_Sleep(PR_SecondsToInterval(1));
  103. memset(buf, 2*i+1, send_amount[i]);
  104. nbytes = PR_Send(sock, buf, send_amount[i],
  105. 0, PR_INTERVAL_NO_TIMEOUT);
  106. if (-1 == nbytes) {
  107. fprintf(stderr, "PR_Send failed\n");
  108. exit(1);
  109. }
  110. if (send_amount[i] != nbytes) {
  111. fprintf(stderr, "PR_Send returned %d, absurd!\n", nbytes);
  112. exit(1);
  113. }
  114. }
  115. if (PR_Close(sock) == PR_FAILURE) {
  116. fprintf(stderr, "PR_Close failed\n");
  117. exit(1);
  118. }
  119. }
  120. /* Non-blocking I/O */
  121. static void ClientNB(void *arg)
  122. {
  123. PRFileDesc *sock;
  124. PRSocketOptionData opt;
  125. PRUint16 port = (PRUint16) arg;
  126. PRNetAddr addr;
  127. char buf[BUFFER_SIZE];
  128. PRPollDesc pd;
  129. PRInt32 npds;
  130. PRInt32 nbytes;
  131. int i;
  132. int j;
  133. sock = PR_OpenTCPSocket(PR_AF_INET6);
  134. if (NULL == sock) {
  135. fprintf(stderr, "PR_OpenTCPSocket failed\n");
  136. exit(1);
  137. }
  138. opt.option = PR_SockOpt_Nonblocking;
  139. opt.value.non_blocking = PR_TRUE;
  140. if (PR_SetSocketOption(sock, &opt) == PR_FAILURE) {
  141. fprintf(stderr, "PR_SetSocketOption failed\n");
  142. exit(1);
  143. }
  144. memset(&addr, 0, sizeof(addr));
  145. if (PR_SetNetAddr(PR_IpAddrLoopback, PR_AF_INET6, port, &addr)
  146. == PR_FAILURE) {
  147. fprintf(stderr, "PR_SetNetAddr failed\n");
  148. exit(1);
  149. }
  150. if (PR_Connect(sock, &addr, PR_INTERVAL_NO_TIMEOUT) == PR_FAILURE) {
  151. if (PR_GetError() != PR_IN_PROGRESS_ERROR) {
  152. fprintf(stderr, "PR_Connect failed\n");
  153. exit(1);
  154. }
  155. pd.fd = sock;
  156. pd.in_flags = PR_POLL_WRITE | PR_POLL_EXCEPT;
  157. npds = PR_Poll(&pd, 1, PR_INTERVAL_NO_TIMEOUT);
  158. if (-1 == npds) {
  159. fprintf(stderr, "PR_Poll failed\n");
  160. exit(1);
  161. }
  162. if (1 != npds) {
  163. fprintf(stderr, "PR_Poll returned %d, absurd!\n", npds);
  164. exit(1);
  165. }
  166. if (PR_GetConnectStatus(&pd) == PR_FAILURE) {
  167. fprintf(stderr, "PR_GetConnectStatus failed\n");
  168. exit(1);
  169. }
  170. }
  171. for (i = 0; i < iterations; i++) {
  172. PR_Sleep(PR_SecondsToInterval(1));
  173. memset(buf, 2*i, send_amount[i]);
  174. while ((nbytes = PR_Send(sock, buf, send_amount[i],
  175. 0, PR_INTERVAL_NO_TIMEOUT)) == -1) {
  176. if (PR_GetError() != PR_WOULD_BLOCK_ERROR) {
  177. fprintf(stderr, "PR_Send failed\n");
  178. exit(1);
  179. }
  180. pd.fd = sock;
  181. pd.in_flags = PR_POLL_WRITE;
  182. npds = PR_Poll(&pd, 1, PR_INTERVAL_NO_TIMEOUT);
  183. if (-1 == npds) {
  184. fprintf(stderr, "PR_Poll failed\n");
  185. exit(1);
  186. }
  187. if (1 != npds) {
  188. fprintf(stderr, "PR_Poll returned %d, absurd!\n", npds);
  189. exit(1);
  190. }
  191. }
  192. if (send_amount[i] != nbytes) {
  193. fprintf(stderr, "PR_Send returned %d, absurd!\n", nbytes);
  194. exit(1);
  195. }
  196. memset(buf, 0, sizeof(buf));
  197. while ((nbytes = PR_Recv(sock, buf, recv_amount[i],
  198. PR_MSG_PEEK, PR_INTERVAL_NO_TIMEOUT)) == -1) {
  199. if (PR_GetError() != PR_WOULD_BLOCK_ERROR) {
  200. fprintf(stderr, "PR_Recv failed\n");
  201. exit(1);
  202. }
  203. pd.fd = sock;
  204. pd.in_flags = PR_POLL_READ;
  205. npds = PR_Poll(&pd, 1, PR_INTERVAL_NO_TIMEOUT);
  206. if (-1 == npds) {
  207. fprintf(stderr, "PR_Poll failed\n");
  208. exit(1);
  209. }
  210. if (1 != npds) {
  211. fprintf(stderr, "PR_Poll returned %d, absurd!\n", npds);
  212. exit(1);
  213. }
  214. }
  215. if (send_amount[i] != nbytes) {
  216. fprintf(stderr, "PR_Recv returned %d, absurd!\n", nbytes);
  217. exit(1);
  218. }
  219. for (j = 0; j < nbytes; j++) {
  220. if (buf[j] != 2*i+1) {
  221. fprintf(stderr, "byte %d should be %d but is %d\n",
  222. j, 2*i+1, buf[j]);
  223. exit(1);
  224. }
  225. }
  226. fprintf(stderr, "client: peeked expected data\n");
  227. memset(buf, 0, sizeof(buf));
  228. nbytes = PR_Recv(sock, buf, recv_amount[i],
  229. PR_MSG_PEEK, PR_INTERVAL_NO_TIMEOUT);
  230. if (-1 == nbytes) {
  231. fprintf(stderr, "PR_Recv failed\n");
  232. exit(1);
  233. }
  234. if (send_amount[i] != nbytes) {
  235. fprintf(stderr, "PR_Recv returned %d, absurd!\n", nbytes);
  236. exit(1);
  237. }
  238. for (j = 0; j < nbytes; j++) {
  239. if (buf[j] != 2*i+1) {
  240. fprintf(stderr, "byte %d should be %d but is %d\n",
  241. j, 2*i+1, buf[j]);
  242. exit(1);
  243. }
  244. }
  245. fprintf(stderr, "client: peeked expected data\n");
  246. memset(buf, 0, sizeof(buf));
  247. nbytes = PR_Recv(sock, buf, recv_amount[i],
  248. 0, PR_INTERVAL_NO_TIMEOUT);
  249. if (-1 == nbytes) {
  250. fprintf(stderr, "PR_Recv failed\n");
  251. exit(1);
  252. }
  253. if (send_amount[i] != nbytes) {
  254. fprintf(stderr, "PR_Recv returned %d, absurd!\n", nbytes);
  255. exit(1);
  256. }
  257. for (j = 0; j < nbytes; j++) {
  258. if (buf[j] != 2*i+1) {
  259. fprintf(stderr, "byte %d should be %d but is %d\n",
  260. j, 2*i+1, buf[j]);
  261. exit(1);
  262. }
  263. }
  264. fprintf(stderr, "client: received expected data\n");
  265. }
  266. if (PR_Close(sock) == PR_FAILURE) {
  267. fprintf(stderr, "PR_Close failed\n");
  268. exit(1);
  269. }
  270. }
  271. static void
  272. RunTest(PRThreadScope scope, PRFileDesc *listenSock, PRUint16 port)
  273. {
  274. PRThread *server, *client;
  275. server = PR_CreateThread(PR_USER_THREAD, ServerB, listenSock,
  276. PR_PRIORITY_NORMAL, scope, PR_JOINABLE_THREAD, 0);
  277. if (NULL == server) {
  278. fprintf(stderr, "PR_CreateThread failed\n");
  279. exit(1);
  280. }
  281. client = PR_CreateThread(
  282. PR_USER_THREAD, ClientNB, (void *) port,
  283. PR_PRIORITY_NORMAL, scope, PR_JOINABLE_THREAD, 0);
  284. if (NULL == client) {
  285. fprintf(stderr, "PR_CreateThread failed\n");
  286. exit(1);
  287. }
  288. if (PR_JoinThread(server) == PR_FAILURE) {
  289. fprintf(stderr, "PR_JoinThread failed\n");
  290. exit(1);
  291. }
  292. if (PR_JoinThread(client) == PR_FAILURE) {
  293. fprintf(stderr, "PR_JoinThread failed\n");
  294. exit(1);
  295. }
  296. }
  297. int main(int argc, char **argv)
  298. {
  299. PRFileDesc *listenSock;
  300. PRNetAddr addr;
  301. PRUint16 port;
  302. listenSock = PR_OpenTCPSocket(PR_AF_INET6);
  303. if (NULL == listenSock) {
  304. fprintf(stderr, "PR_OpenTCPSocket failed\n");
  305. exit(1);
  306. }
  307. memset(&addr, 0, sizeof(addr));
  308. if (PR_SetNetAddr(PR_IpAddrAny, PR_AF_INET6, 0, &addr) == PR_FAILURE) {
  309. fprintf(stderr, "PR_SetNetAddr failed\n");
  310. exit(1);
  311. }
  312. if (PR_Bind(listenSock, &addr) == PR_FAILURE) {
  313. fprintf(stderr, "PR_Bind failed\n");
  314. exit(1);
  315. }
  316. if (PR_GetSockName(listenSock, &addr) == PR_FAILURE) {
  317. fprintf(stderr, "PR_GetSockName failed\n");
  318. exit(1);
  319. }
  320. port = PR_ntohs(addr.ipv6.port);
  321. if (PR_Listen(listenSock, 5) == PR_FAILURE) {
  322. fprintf(stderr, "PR_Listen failed\n");
  323. exit(1);
  324. }
  325. fprintf(stderr, "Running the test with local threads\n");
  326. RunTest(PR_LOCAL_THREAD, listenSock, port);
  327. fprintf(stderr, "Running the test with global threads\n");
  328. RunTest(PR_GLOBAL_THREAD, listenSock, port);
  329. if (PR_Close(listenSock) == PR_FAILURE) {
  330. fprintf(stderr, "PR_Close failed\n");
  331. exit(1);
  332. }
  333. printf("PASS\n");
  334. return 0;
  335. }