实验报告(实验五)

不问归期 提交于 2020-02-02 02:22:19

北京电子科技学院(BESTI)

             

课程:信息安全系统设计基础               班级:1353、1352班                        

姓名:刘浩晨、王玥                 学号:20135318、20135232

成绩:            指导教师:娄嘉鹏           实验日期:2015.12.01

实验密级:        预习程度:               实验时间:15:30-18:00

仪器组次:         必修/选修:必修             实验序号:五

实验名称:实验五:通讯协议设计                                                            

实验目的与要求:

1. 掌握在ARM开发板实现一个简单的WEB服务器的过程。
2. 学习在ARM开发板上的SOCKET网络编程。
3. 学习Linux下的signal()函数的使用。                                

实验仪器:

名称

型号

数量

PC机

Lenovo

1

嵌入式实验平台

UP-TECH S2410

1

实验内容、步骤与体会:

一、实验内容与步骤 

1、阅读理解源码

进入/arm2410cl/exp/basic/07_httpd 目录,使用 vi 编辑器或其他编辑器阅读理解源代码

  

  1 httpd.c代码分析
  2 
  3  / * httpd.c:  A very simple http server 
  4  * Copyfight (C) 2003  Zou jian guo <ah_zou@163.com> 
  5  * Copyright (C) 2000 Lineo, Inc.  (www.lineo.com) 
  6  * Copyright (c) 1997-1999 D. Jeff Dionne <jeff@lineo.ca> 
  7  * Copyright (c) 1998  Kenneth Albanowski <kjahds@kjahds.com> 
  8  * Copyright (c) 1999  Nick Brok <nick@nbrok.iaehv.nl> 
  9  * 
 10  * This program is free software; you can redistribute it and/or modify 
 11  * it under the terms of the GNU General Public License as published by 
 12  * the Free Software Foundation; either version 2 of the License, or 
 13  * (at your option) any later version. 
 14  * 
 15  */ 
 16  
 17 #include <stdio.h> 
 18 #include <stdlib.h> 
 19 #include <fcntl.h> 
 20 #include <string.h> 
 21 #include <sys/types.h> 
 22 #include <sys/socket.h> 
 23 #include <netinet/in.h> 
 24 #include <errno.h> 
 25 #include <sys/stat.h> 
 26 #include <dirent.h> 
 27 #include <signal.h> 
 28 #include <unistd.h> 
 29 #include <ctype.h> 
 30 #include "pthread.h" 
 31  
 32 #define DEBUG 
 33  
 34 int KEY_QUIT=0; 
 35 int TIMEOUT=30; //设置闹钟秒数; 
 36  
 37 #ifndef O_BINARY 
 38 #define O_BINARY 0 
 39 #endif 
 40  
 41 char referrer[128]; 
 42 int content_length; 
 43  
 44 #define SERVER_PORT 80 
 45  
 46 int PrintHeader(FILE *f, int content_type)   //发送HTTP协议数据头 
 47 { 
 48   alarm(TIMEOUT); 
 49   fprintf(f,"HTTP/1.0 200 OKn"); //服务器回应http协议数据头的状态行;发送请求成功; 
 50   switch (content_type) 
 51   {  
 52    case 't': 
 53 fprintf(f,"Content-type: text/plainn"); //发送纯文本文件信息; 
 54 break; 
 55    case 'g': 
 56 fprintf(f,"Content-type: image/gifn"); //发送gif格式图片信息; 
 57 break; 
 58    case 'j': 
 59 fprintf(f,"Content-type: image/jpegn"); //发送gpeg格式图片信息; 
 60 break; 
 61    case 'h': 
 62 fprintf(f,"Content-type: text/htmln"); //发送html信息; 
 63 break; 
 64   } 
 65   fprintf(f,"Server: uClinux-httpd 0.2.2n"); //发送服务器版本信息; 
 66   fprintf(f,"Expires: 0n"); //发送文件永不过期信息; 
 67   fprintf(f,"n"); //打印换行符; 
 68   alarm(0); 
 69   return(0); 
 70 } 
 71  
 72 int DoJpeg(FILE *f, char *name)  //对jpeg格式的文件进行处理; 
 73 { 
 74   char *buf; 
 75   FILE * infile; 
 76   int count; 
 77   
 78   if (!(infile = fopen(name, "r"))) { //通过文件名打开一个文件,只读属性; 
 79 alarm(TIMEOUT); 
 80 fprintf(stderr, "Unable to open JPEG file %s, %dn", name, errno); 
 81 fflush(f); 
 82 alarm(0); 
 83 return -1; 
 84   } 
 85   
 86   PrintHeader(f,'j');//发送j类型的http协议数据头信息; 
 87  
 88   
 89   copy(infile,f); /* prints the page */  
 90   
 91   alarm(TIMEOUT); 
 92   fclose(infile); 
 93   alarm(0); 
 94   
 95   return 0; 
 96 } 
 97  
 98 int DoGif(FILE *f, char *name)  //对gif格式的文件进行处理; 
 99 { 
100   char *buf; 
101   FILE * infile; 
102   int count; 
103  
104   if (!(infile = fopen(name, "r"))) { //通过文件名打开一个文件,只读属性; 
105 alarm(TIMEOUT); 
106 fprintf(stderr, "Unable to open GIF file %s, %dn", name, errno); 
107 fflush(f); 
108 alarm(0); 
109 return -1; 
110   } 
111    
112   PrintHeader(f,'g'); //发送g类型的http协议数据头信息 
113  
114   copy(infile,f); /* prints the page */   
115  
116   alarm(TIMEOUT); 
117   fclose(infile); 
118   alarm(0); 
119    
120   return 0; 
121 } 
122  
123 int DoDir(FILE *f, char *name) //对目录进行处理; 
124 { 
125   char *buf; 
126   DIR * dir; 
127   struct dirent * dirent; //dirent不仅仅指向目录,还指向目录中的具体文件,dirent结构体存储的关于文件的信息很少,所以dirent起着一个索引的作用 
128  
129   if ((dir = opendir(name))== 0) { //打开一个目录; 
130 fprintf(stderr, "Unable to open directory %s, %dn", name, errno); 
131 fflush(f); 
132 return -1; 
133   } 
134    
135   PrintHeader(f,'h'); //发送h类型的http协议数据头信息 
136    
137   alarm(TIMEOUT); 
138   fprintf(f, "<H1>Index of %s</H1>nn",name); 
139   alarm(0); 
140  
141   if (name[strlen(name)-1] != '/') { //若名字的后面没有/则默认加上 /; 
142 strcat(name, "/"); 
143   } 
144    
145   while(dirent = readdir(dir)) { //读取目录; 
146 alarm(TIMEOUT); 
147    
148 fprintf(f, "<p><a href="/%s%s">%s</p>n", name, dirent->d_name, dirent->d_name); 
149 alarm(0); //发送目录信息; 
150   } 
151    
152   closedir(dir); 
153   return 0; 
154 } 
155  
156 int DoHTML(FILE *f, char *name) 
157 { 
158   char *buf; 
159   FILE *infile; //定义文件流指针 
160   int count;  
161   char * dir = 0; 
162  
163   if (!(infile = fopen(name,"r"))) {   //通过文件名打开一个文件,只读属性; 
164 alarm(TIMEOUT);  
165 fprintf(stderr, "Unable to open HTML file %s, %dn", name, errno); //打印打开文件失败信息; 
166 fflush(f); 
167 alarm(0); 
168 return -1; 
169   } 
170  
171   PrintHeader(f,'h'); //发送http协议数据报;f表示客户连接的文件流指针用于写入http协议数据头信息; 
172   copy(infile,f); /* prints the page */  //将打开的文件内容通过发送回客户端; 
173  
174   alarm(TIMEOUT); 
175   fclose(infile); 
176   alarm(0); 
177  
178   return 0; 
179 } 
180  
181 int DoText(FILE *f, char *name) //纯文本文件的处理; 
182 { 
183   char *buf; 
184   FILE *infile; //定义文件流指针; 
185   int count; 
186  
187   if (!(infile = fopen(name,"r"))) { //通过文件名打开一个文件,只读属性 
188 alarm(TIMEOUT); 
189 fprintf(stderr, "Unable to open text file %s, %dn", name, errno); 
190 fflush(f); 
191 alarm(0); 
192 return -1; 
193   } 
194  
195   PrintHeader(f,'t'); //发送t类型的http协议数据头信息; 
196   copy(infile,f); /* prints the page */   
197  
198   alarm(TIMEOUT); 
199   fclose(infile); 
200   alarm(0); 
201  
202   return 0; 
203 } 
204  
205 int ParseReq(FILE *f, char *r) 
206 { 
207   char *bp; //定义指针bp; 
208   struct stat stbuf;  
209   char * arg; //参数指针; 
210   char * c; 
211   int e; 
212   int raw; 
213  
214 #ifdef DEBUG 
215   printf("req is '%s'n", r); //打印请求命令;例如:GET /img/baidu_sylogo1.gif HTTP/1.1rn 
216 #endif 
217    
218   while(*(++r) != ' ');  /*skip non-white space*/ //判断buf中的内容是否为空跳过非空白; 
219   while(isspace(*r))   //判断r所在位置的字符是否为空格若为空格则r指向下一个字符; 
220   r++; 
221    
222   while (*r == '/')  //判断r所在位置的字符是否为/若为空格则r指向下一个字符; 
223   r++; 
224   bp = r; //将r所指向的内容赋值给bp bp指向/之后的内容;img/baidu_sylogo1.gif HTTP/1.1rn 
225    
226   while(*r && (*(r) != ' ') && (*(r) != '?')) 
227   r++;//当r不为空,并求 r不为?时r指向下一个字符 
228    
229 #ifdef DEBUG 
230   printf("bp='%s' %x, r='%s' n", bp, *bp,r); //打印 r和bp的值; 
231 #endif 
232    
233   if (*r == '?')   //判断 r是否为 ?若为?则执行以下语句; 
234   { 
235   char * e; //定义指针变量; 
236   *r = 0;  //将r所在位置处的字符设为; 的ASCII码值是0 
237   arg = r+1; //arg指向下一个参数; 
238   if (e = strchr(arg,' '))  
239 { 
240   *e = '';  //如果arg为空则将arg所在位置置为复制给e; 
241   } 
242   } else  
243 { // 如果当前r指向字符不为 '?', 将r指向字符置为 '',  
244   arg = 0;  
245   *r = 0;   // r处设为; 
246 } 
247    
248   c = bp;//将bp赋值给c; 
249  
250 /*zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz*/ 
251   if (c[0] == 0x20){ //判断c中的字符内容是否为空格;若为空格 
252 c[0]='.'; //将.和放入c数组中; 
253 c[1]='';  
254 } 
255 /*zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz*/ 
256 if(c[0] == '') strcat(c,"."); //若 c中为则将.链接在c后; 
257  
258 if (c && !stat(c, &stbuf))  //通过文件名c获取文件信息,并保存在stbuf中
259 //返回值:  执行成功则返回0,失败返回-1,错误代码存于errno
260  
261   { 
262 if (S_ISDIR(stbuf.st_mode))//判断结果是否为特定的值 
263 {  
264 char * end = c + strlen(c); //end指向c的末尾; 
265 strcat(c, "/index.html"); //将/index.html加到c后,后面追加; 
266 if (!stat(c, &stbuf)) //通过文件名c获取文件信息,并保存在stbuf中 ;成功返回0; 
267 { 
268 DoHTML(f, c); //对html文件进行处理; 
269 }  
270 else  
271 { 
272   *end = ''; //将end指向; 
273 DoDir(f,c); //若c中没有"/index.html" 则跳到目录处理目录代码处去执行; 
274 } 
275 } 
276 else if (!strcmp(r - 4, ".gif")) //判断r中的后四个字符,即判断文件类型; 
277   DoGif(f,c);  //若是 gif格式的文件则跳转到DoGif对其进行处理; 
278 else if (!strcmp(r - 4, ".jpg") || !strcmp(r - 5, ".jpeg")) 
279   DoJpeg(f,c); //若是 jpg或jpeg格式的文件则跳转到DoJpeg对其进行处理; 
280 else if (!strcmp(r - 4, ".htm") || !strcmp(r - 5, ".html")) 
281 DoHTML(f,c); //若是 htm格式的文件则跳转到DoHTML处对其进行处理; 
282  else 
283   DoText(f,c);//若是 纯文本格式的文件则跳转到DoText对其进行处理 
284 }  
285 else{ 
286   PrintHeader(f,'h'); //发送h类型的http协议数据头 
287   alarm(TIMEOUT); 
288   fprintf(f, "<html><head><title>404 File Not Found</title></head>n"); //打印出错信息 
289 fprintf(f, "<body>The requested URL was not found on this server</body></html>n"); 
290   alarm(0); 
291 } 
292   return 0; 
293 } 
294  
295 void sigalrm(int signo) //定时器终止时发送给进程的信号; 
296 { 
297 /* got an alarm, exit & recycle */ 
298 exit(0); 
299 } 
300  
301 int HandleConnect(int fd) 
302 { 
303   FILE *f;//定义文件流FILE结构体指针用来表示与客户连接的文件流指针; 
304  
305   char buf[160];  //定义缓冲区buf用来存放客户端的请求命令; 
306   char buf1[160]; //定义缓冲区buf用来存放客户端的各字段信息; 
307  
308   f = fdopen(fd,"a+"); //以文件描述符的形式打开文件; a+ 以附加方式打开可读写的文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾后,即文件原先的内容会被保留。  
309   if (!f) {//若文件打开失败则打印出错信息; 
310 fprintf(stderr, "httpd: Unable to open httpd input fd, error %dn", errno); 
311 alarm(TIMEOUT); // 闹钟函数成功则返回上一个闹钟时间的剩余时间,否则返回0。 出错返回-1  
312 close(fd);//关闭文件描述符; 
313 alarm(0); //将闹钟时间清0; 
314 return 0; 
315   } 
316   setbuf(f, 0); //将关闭缓冲区; 
317  
318   alarm(TIMEOUT); //启用闹钟; 
319  
320   if (!fgets(buf, 150, f)) {   //直接通过f读取150个字符放入以buf为起始地址中,不成功时返回0则打印出错信息;否则fgets成功返回函数指针打印buf的内容; 
321 fprintf(stderr, "httpd: Error reading connection, error %dn", errno); 
322 fclose(f); //关闭文件描述符; 
323 alarm(0);   
324 return 0; 
325   } 
326 #ifdef DEBUG 
327   printf("buf = '%s'n", buf); //打印客户机发出的请求命令; 
328 #endif 
329  
330   alarm(0);  //将闹钟时间清0; 
331  
332   referrer[0] = '';//初始化referrer数组; 
333   content_length = -1;  //将信息长度初始化为-1; 
334  
335  alarm(TIMEOUT);  //设置定时器; 
336 //read other line to parse Rrferrer and content_length infomation 
337 while (fgets(buf1, 150, f) && (strlen(buf1) > 2)) {  //直接通过f读取150个字符放入以buf1为起始地址的空间中; 
338   alarm(TIMEOUT); 
339 #ifdef DEBUG 
340 printf("Got buf1 '%s'n", buf1); //打印buf1中的信息; 
341 #endif 
342 if (!strncasecmp(buf1, "Referer:", 8)) {  //将buf1中的前八个字符与字符串Referer:若相等则将将指针指向buf1中的Referer:之后; 
343   char * c = buf1+8; 
344   while (isspace(*c)) //判断c处是否为空格若为空格则c指向下一个字符; 
345 c++; 
346 strcpy(referrer, c); //将c所指的内存单元的内容复制到referrer数组中; 
347 }  
348 else if (!strncasecmp(buf1, "Referrer:", 9)) { //将buf1中的前九个字符与字符串Referrer:若相等则将将指针指向buf1中的Referrer:之后; 
349   char * c = buf1+8; 
350   char * c = buf1+9; 
351   while (isspace(*c))  //判断c处是否���空格若为空格则c指向下一个字符; 
352 c++; 
353   strcpy(referrer, c); //将c所指的内存单元的内容复制到referrer数组中; 
354 }  
355 else if (!strncasecmp(buf1, "Content-length:", 15)) { )) { //将buf1中的前15个字符与字符串Content-length:若相等则将将指针指向buf1中的Content-length:之后; 
356  
357   content_length = atoi(buf1+15); //atoi类型转换将buf1中的内容转换为整型赋值给content_length; 
358 }  
359   } 
360   alarm(0); 
361    
362   if (ferror(f)) {  //错误信息输出; 
363 fprintf(stderr, "http: Error continuing reading connection, error %dn", errno); 
364 fclose(f); 
365 return 0; 
366   } 
367  
368   ParseReq(f, buf); //解析客户请求函数; 
369  
370   alarm(TIMEOUT); //打开计时器; 
371   fflush(f); //刷新流; 
372   fclose(f); //关闭文件流; 
373   alarm(0); 
374   return 1; 
375 } 
376  
377  
378  
379 void* key(void* data) 
380 { 
381 int c; 
382 for(;;){ 
383 c=getchar(); //从键盘输入一个字符 
384 if(c == 'q' || c == 'Q'){ 
385 KEY_QUIT=1; 
386 exit(10); //若输入q则退出程序; 
387 break; 
388 } 
389 } 
390  
391 } 
392  
393 int main(int argc, char *argv[]) 
394 { 
395   int fd, s;   //定义套接字文件描述符作为客户机和服务器之间的通道; 
396   int len;  
397   volatile int true = 1;  //定义volatile类型的变量用来作为指向缓冲区的指针变量; 
398   struct sockaddr_in ec; 
399   struct sockaddr_in server_sockaddr; //定义结构体变量; 
400    
401   pthread_t th_key;//定义线程号; 
402   void * retval;   //用来存储被等待线程的返回值。  
403  
404  
405   signal(SIGCHLD, SIG_IGN); //忽略信号量; 
406   signal(SIGPIPE, SIG_IGN); 
407   signal(SIGALRM, sigalrm);  //设置时钟信号的对应动作; 
408  
409   chroot(HTTPD_DOCUMENT_ROOT);  //改变根目录;在makefile文件中指定; 
410   printf("starting httpd...n"); //打印启用服务器程序信息; 
411   printf("press q to quit.n"); 
412 //  chdir("/"); 
413  
414   if (argc > 1 && !strcmp(argv[1], "-i")) {// 若argv【1】等于-i strcmp返回0 并且 argc大于1  执行if下的语句快即关闭文件描述符; 
415 /* I'm running from inetd, handle the request on stdin */ 
416 fclose(stderr); 
417 HandleConnect(0); //向HandleConnect函数传入0文件描述符即标准输入; 
418 exit(0);  
419   } 
420  
421   if((s = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)) == -1) {  //若获取套接字出错则将错误信息输出到标准设备; 
422 perror("Unable to obtain network"); 
423 exit(1); 
424   } 
425    
426   if((setsockopt(s, SOL_SOCKET, SO_REUSEADDR, (void *)&true,  //此函数用于设置套接口,若成功返回0,否则返回错误 
427  sizeof(true))) == -1) { 
428 perror("setsockopt failed");   //输出错误信息; 
429 exit(1); 
430   } 
431  
432   server_sockaddr.sin_family = AF_INET; //设置ip地址类型; 
433   server_sockaddr.sin_port = htons(SERVER_PORT);  //设置网络端口; 
434   server_sockaddr.sin_addr.s_addr = htonl(INADDR_ANY); //INADDR_ANY表示本地任意ip; 
435    
436   if(bind(s, (struct sockaddr *)&server_sockaddr,  //将所监听的端口号与服务器的地址、端口绑定;
437  
438  sizeof(server_sockaddr)) == -1)  {  
439 perror("Unable to bind socket");//若绑定失败则打印出错信息; 
440 exit(1); 
441   } 
442  
443   if(listen(s, 8*3) == -1) { //listen()声明服务器处于监听状态,并且最多允许有24个客户端处于连接待状态; 
444 perror("Unable to listen"); 
445 exit(4); 
446   } 
447  
448    
449    pthread_create(&th_key, NULL, key, 0);   //创建线程; 
450   /* Wait until producer and consumer finish. */ 
451   printf("wait for connection.n"); //打印服务器等待链接信息; 
452   while (1) {   
453    
454 len = sizeof(ec);//ec结构体变量的长度; 
455 if((fd = accept(s, (void *)&ec, &len)) == -1) { //接受客户机的请求,与客户机建立链接; 
456   exit(5); 
457   close(s); 
458 } 
459 HandleConnect(fd); //处理链接函数调用fd 为客户连接文件描述符;; 
460  
461   } 
462   pthread_join(th_key, &retval); //以阻塞的方式等待thread指定的线程结束。当函数返回时,被等待线程的资源被收回。如果进程已经结束,那么该函数会立即返回。成功返回0;该语句不会执行到;
463    
464 } 

 

2、编译应用程序

gcc编译,运行 make 产生可执行文件 httpd

3、下载调试

使用 NFS 服务方式将 HTTPD 下载到开发板上,并拷贝测试用的网页进行调试,本例中用的是 index 测试网页。下载结果如图一:

 

4、本机测试

在台式机的浏览器中输入 http://192.168.0.111(111 为 UP-CUP S2410 实验板的 IP地址),观察在客户机的浏览器中的连接请求结果(如图 2.7.8)和在开发板上的服务器的打印信息。实验结果如下图二:

图二 本机测试结果

 

二、遇到的问题与解决方法

1、在运行make命令产生可执行文件httpd时,我们运行make命令结束后,进入07_httpd文件夹内查看,发现并没有生成httpd这一文件,详细见下图三:

 

图三 问题一截图

解决办法:再次运行make命令后得到提示“can’t open output file ../bin/httpd”,可知,在我们当前所在的07_httpd文件夹的上级目录ws中并不存在bin这个文件夹,所以导致命令运行时无法进入ws/bin/httpd这一路径,由此,我们选择使用了mkdir这个创建指定名称目录的命令,返回到当前目录的上一级目录中,使用“mkdir bin”在ws文档中创建了bin目录,于是可顺利运行make命令产生httpd文件,解决过程如下图四:

 

图四 问题一解决过程

 

2、在将httpd服务下载到开发板上这一步骤中,挂载超级终端与共享文件建立通讯时,输入“mount  -t nfs  -o nolock  192.168.0.234:/root/bc  /host”时,提示“No such file or directory”错误提示,具体问题见下图五:

 

图五 问题二截图

 

解决方法:再次查看共享文件目录,确定我们并不存在/root/home/bc这一目录,将命令行修改为“mount  -t nfs  -o nolock  192.168.0.234:/home/bc  /host”即可解决所提示错误,解决过程如图六:

图六 问题二解决过程

三、实验体会

这次实验是简单嵌入式WEB服务器实验,首先进行代码的阅读和理解,然后再实验。通过这些我们对在ARM开发板上开发一个简单的WEB的过程,以及在ARM开发板上SOCKET网络编程和Linux下signal()函数的使用有了一定的了解。在实验开始ARM开发板的搭建以及超级终端的建立较为顺利,因为有了前几次实验的经验。所以不管是学书本上的知识还是动手操作,都是熟能生巧,应该多看多练多巩固,这样才能真正吸收。这次实验也是最后一次实验了,通过这五次实验,我们更加深入了解了linux和有趣的东西。对我们的学习生活也有一定的帮助。我们以后将会继续努力学习。

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!