web01.lc 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. /* include web1 */
  2. #include "unpthread.h"## 1 ##src/threads/web01.c##
  3. #include <thread.h> /* Solaris threads */## 2 ##src/threads/web01.c##
  4. #define MAXFILES 20## 3 ##src/threads/web01.c##
  5. #define SERV "80" /* port number or service name */## 4 ##src/threads/web01.c##
  6. struct file {## 5 ##src/threads/web01.c##
  7. char *f_name; /* filename */## 6 ##src/threads/web01.c##
  8. char *f_host; /* hostname or IP address */## 7 ##src/threads/web01.c##
  9. int f_fd; /* descriptor */## 8 ##src/threads/web01.c##
  10. int f_flags; /* F_xxx below */## 9 ##src/threads/web01.c##
  11. pthread_t f_tid; /* thread ID */## 10 ##src/threads/web01.c##
  12. } file[MAXFILES];## 11 ##src/threads/web01.c##
  13. #define F_CONNECTING 1 /* connect() in progress */## 12 ##src/threads/web01.c##
  14. #define F_READING 2 /* connect() complete; now reading */## 13 ##src/threads/web01.c##
  15. #define F_DONE 4 /* all done */## 14 ##src/threads/web01.c##
  16. #define GET_CMD "GET %s HTTP/1.0\r\n\r\n"## 15 ##src/threads/web01.c##
  17. int nconn, nfiles, nlefttoconn, nlefttoread;## 16 ##src/threads/web01.c##
  18. void *do_get_read(void *);## 17 ##src/threads/web01.c##
  19. void home_page(const char *, const char *);## 18 ##src/threads/web01.c##
  20. void write_get_cmd(struct file *);## 19 ##src/threads/web01.c##
  21. int## 20 ##src/threads/web01.c##
  22. main(int argc, char **argv)## 21 ##src/threads/web01.c##
  23. {## 22 ##src/threads/web01.c##
  24. int i, n, maxnconn;## 23 ##src/threads/web01.c##
  25. pthread_t tid;## 24 ##src/threads/web01.c##
  26. struct file *fptr;## 25 ##src/threads/web01.c##
  27. if (argc < 5)## 26 ##src/threads/web01.c##
  28. err_quit("usage: web <#conns> <IPaddr> <homepage> file1 ...");## 27 ##src/threads/web01.c##
  29. maxnconn = atoi(argv[1]);## 28 ##src/threads/web01.c##
  30. nfiles = min(argc - 4, MAXFILES);## 29 ##src/threads/web01.c##
  31. for (i = 0; i < nfiles; i++) {## 30 ##src/threads/web01.c##
  32. file[i].f_name = argv[i + 4];## 31 ##src/threads/web01.c##
  33. file[i].f_host = argv[2];## 32 ##src/threads/web01.c##
  34. file[i].f_flags = 0;## 33 ##src/threads/web01.c##
  35. }## 34 ##src/threads/web01.c##
  36. printf("nfiles = %d\n", nfiles);## 35 ##src/threads/web01.c##
  37. home_page(argv[2], argv[3]);## 36 ##src/threads/web01.c##
  38. nlefttoread = nlefttoconn = nfiles;## 37 ##src/threads/web01.c##
  39. nconn = 0;## 38 ##src/threads/web01.c##
  40. /* end web1 */
  41. /* include web2 */
  42. while (nlefttoread > 0) {## 39 ##src/threads/web01.c##
  43. while (nconn < maxnconn && nlefttoconn > 0) {## 40 ##src/threads/web01.c##
  44. /* 4find a file to read */## 41 ##src/threads/web01.c##
  45. for (i = 0; i < nfiles; i++)## 42 ##src/threads/web01.c##
  46. if (file[i].f_flags == 0)## 43 ##src/threads/web01.c##
  47. break;## 44 ##src/threads/web01.c##
  48. if (i == nfiles)## 45 ##src/threads/web01.c##
  49. err_quit("nlefttoconn = %d but nothing found", nlefttoconn);## 46 ##src/threads/web01.c##
  50. file[i].f_flags = F_CONNECTING;## 47 ##src/threads/web01.c##
  51. Pthread_create(&tid, NULL, &do_get_read, &file[i]);## 48 ##src/threads/web01.c##
  52. file[i].f_tid = tid;## 49 ##src/threads/web01.c##
  53. nconn++;## 50 ##src/threads/web01.c##
  54. nlefttoconn--;## 51 ##src/threads/web01.c##
  55. }## 52 ##src/threads/web01.c##
  56. if ((n = thr_join(0, &tid, (void **) &fptr)) != 0)## 53 ##src/threads/web01.c##
  57. errno = n, err_sys("thr_join error");## 54 ##src/threads/web01.c##
  58. nconn--;## 55 ##src/threads/web01.c##
  59. nlefttoread--;## 56 ##src/threads/web01.c##
  60. printf("thread id %d for %s done\n", tid, fptr->f_name);## 57 ##src/threads/web01.c##
  61. }## 58 ##src/threads/web01.c##
  62. exit(0);## 59 ##src/threads/web01.c##
  63. }## 60 ##src/threads/web01.c##
  64. /* end web2 */
  65. /* include do_get_read */
  66. void *## 61 ##src/threads/web01.c##
  67. do_get_read(void *vptr)## 62 ##src/threads/web01.c##
  68. {## 63 ##src/threads/web01.c##
  69. int fd, n;## 64 ##src/threads/web01.c##
  70. char line[MAXLINE];## 65 ##src/threads/web01.c##
  71. struct file *fptr;## 66 ##src/threads/web01.c##
  72. fptr = (struct file *) vptr;## 67 ##src/threads/web01.c##
  73. fd = Tcp_connect(fptr->f_host, SERV);## 68 ##src/threads/web01.c##
  74. fptr->f_fd = fd;## 69 ##src/threads/web01.c##
  75. printf("do_get_read for %s, fd %d, thread %d\n",## 70 ##src/threads/web01.c##
  76. fptr->f_name, fd, fptr->f_tid);## 71 ##src/threads/web01.c##
  77. write_get_cmd(fptr); /* write() the GET command */## 72 ##src/threads/web01.c##
  78. /* 4Read server's reply */## 73 ##src/threads/web01.c##
  79. for (;;) {## 74 ##src/threads/web01.c##
  80. if ((n = Read(fd, line, MAXLINE)) == 0)## 75 ##src/threads/web01.c##
  81. break; /* server closed connection */## 76 ##src/threads/web01.c##
  82. printf("read %d bytes from %s\n", n, fptr->f_name);## 77 ##src/threads/web01.c##
  83. }## 78 ##src/threads/web01.c##
  84. printf("end-of-file on %s\n", fptr->f_name);## 79 ##src/threads/web01.c##
  85. Close(fd);## 80 ##src/threads/web01.c##
  86. fptr->f_flags = F_DONE; /* clears F_READING */## 81 ##src/threads/web01.c##
  87. return (fptr); /* terminate thread */## 82 ##src/threads/web01.c##
  88. }## 83 ##src/threads/web01.c##
  89. /* end do_get_read */
  90. /* include write_get_cmd */
  91. void## 84 ##src/threads/web01.c##
  92. write_get_cmd(struct file *fptr)## 85 ##src/threads/web01.c##
  93. {## 86 ##src/threads/web01.c##
  94. int n;## 87 ##src/threads/web01.c##
  95. char line[MAXLINE];## 88 ##src/threads/web01.c##
  96. n = snprintf(line, sizeof(line), GET_CMD, fptr->f_name);## 89 ##src/threads/web01.c##
  97. Writen(fptr->f_fd, line, n);## 90 ##src/threads/web01.c##
  98. printf("wrote %d bytes for %s\n", n, fptr->f_name);## 91 ##src/threads/web01.c##
  99. fptr->f_flags = F_READING; /* clears F_CONNECTING */## 92 ##src/threads/web01.c##
  100. }## 93 ##src/threads/web01.c##
  101. /* end write_get_cmd */
  102. /* include home_page */
  103. void## 94 ##src/threads/web01.c##
  104. home_page(const char *host, const char *fname)## 95 ##src/threads/web01.c##
  105. {## 96 ##src/threads/web01.c##
  106. int fd, n;## 97 ##src/threads/web01.c##
  107. char line[MAXLINE];## 98 ##src/threads/web01.c##
  108. fd = Tcp_connect(host, SERV); /* blocking connect() */## 99 ##src/threads/web01.c##
  109. n = snprintf(line, sizeof(line), GET_CMD, fname);##100 ##src/threads/web01.c##
  110. Writen(fd, line, n);##101 ##src/threads/web01.c##
  111. for (;;) {##102 ##src/threads/web01.c##
  112. if ((n = Read(fd, line, MAXLINE)) == 0)##103 ##src/threads/web01.c##
  113. break; /* server closed connection */##104 ##src/threads/web01.c##
  114. printf("read %d bytes of home page\n", n);##105 ##src/threads/web01.c##
  115. /* do whatever with data */##106 ##src/threads/web01.c##
  116. }##107 ##src/threads/web01.c##
  117. printf("end-of-file on home page\n");##108 ##src/threads/web01.c##
  118. Close(fd);##109 ##src/threads/web01.c##
  119. }##110 ##src/threads/web01.c##
  120. /* end home_page */