sadfasdfsadf

 

  
  
  
  
  1. #include <string.h>  
  2. #ifdef __WIN32__  
  3. #include <winsock.h>  
  4. #include <wininet.h>  
  5. #define SHUT_RDWR SD_BOTH  
  6. #else  
  7. #include <sys/socket.h>  
  8. #include <arpa/inet.h>  
  9. #endif  
  10. #include "debug.h"  
  11. #include "server.h"  
  12. #include "conn.h"  
  13. #include "xmlparser.h"  
  14.  
  15. static void load_config(struct ftpserver* srv)  
  16. {  
  17.     struct XML_DATA* xml = xml_load("./config.xml");  
  18.  
  19.     strcpy(srv->server_root, xml_readstr(xml, "server_root"));  
  20.  
  21.     strcpy(srv->server_ip, xml_readstr(xml, "server_ip"));  
  22.     srv->server_port = xml_readnum(xml, "server_port");  
  23.     srv->max_ips = xml_readnum(xml, "max_ips");  
  24.     srv->max_conns = xml_readnum(xml, "max_conns");  
  25.     srv->max_conns_per_ip = xml_readnum(xml, "max_conns_per_ip");  
  26.     srv->conn_timeout = xml_readnum(xml, "conn_timeout");  
  27.     srv->reserved_ips = xml_readnum(xml, "reserved_ips");  
  28.     int terminal_log = xml_readnum(xml, "terminal_log");  
  29.     int file_log = xml_readnum(xml, "file_log");  
  30.     debug_set_dir( xml_readstr(xml, "file_log:directory") );  
  31.     if( terminal_log )  
  32.         debug_term_on();  
  33.     else 
  34.         debug_term_off();  
  35.     if( file_log )  
  36.         debug_file_on();  
  37.     else 
  38.         debug_file_off();  
  39.     xml_free(xml);  
  40. }  
  41.  
  42. int server_create(struct ftpserver* srv)  
  43. {  
  44.     /* Initialization  
  45.      * Why not check pointer?  
  46.      * Reference header files assert.h   
  47.     */ 
  48.  
  49.     /* initialization srv */ 
  50.     memset(srv, 0, sizeof(struct ftpserver));  
  51.  
  52.     /* set server name */ 
  53.     strcpy(srv->server_name, "xxftpd");  
  54.  
  55.     /* load config XML jump */ 
  56.  
  57.     load_config(srv);  
  58.       
  59.     if(!srv->server_ip[0]){  
  60.         /* get host ip */ 
  61.         char hostname[128];  
  62.         gethostname(hostname, 127);  
  63.         DBG("hostname: %s", hostname);  
  64.         struct hostent * pHostent;  
  65.         pHostent = gethostbyname(hostname);   
  66.           
  67.         struct in_addr iaddr;  
  68.         memcpy(&srv->server_ipnum, pHostent->h_addr_list[0], 4);  
  69.         iaddr.s_addr = srv->server_ipnum;  
  70.         strcpy(srv->server_ip, inet_ntoa(iaddr));   
  71.     }  
  72.     return 0;  
  73. }  
  74.  
  75. static int loop_ips_search_by_ip(const void* p, const void* q)  
  76. {  
  77.     if(((struct ftpip*)p)->ipnum == (unsigned int)q)  
  78.         return 1;  
  79.     return 0;  
  80. }  
  81.  
  82. static void * server_listen(void* data)  
  83. {  
  84.     struct sockaddr_in addr;  
  85.     unsigned int ipnum;   
  86.     unsigned len;  
  87.     struct ftpserver* srv = (struct ftpserver*)data;  
  88.     while(!srv->server_end){  
  89.         len = sizeof(addr);  
  90.         int fd = accept(srv->server_fd, (struct sockaddr *)&addr, &len);  
  91.         if(fd < 0){  
  92.             DBG("accept failed. fd=%d", fd);  
  93.             continue;  
  94.         }  
  95.         ipnum = ntohl(addr.sin_addr.s_addr);  
  96.           
  97.         struct ftpip* ip = (struct ftpip* )loop_search(&srv->loop_ips, (void*)ipnum, loop_ips_search_by_ip);  
  98.         if(ip == NULL){  
  99.             if(srv->cur_ips >= srv->max_ips){  
  100.                 shutdown(fd, SHUT_RDWR);  
  101.                 closesocket(fd);  
  102.             }else{  
  103.                 ip = (struct ftpip*)malloc(sizeof(struct ftpip));  
  104.                 ip->ipnum = ipnum;  
  105.                 ip->cur_conns = 0;  
  106.                 struct in_addr iaddr;  
  107.                 iaddr.s_addr = htonl( ipnum );  
  108.                 strcpy(ip->ipstr, inet_ntoa(iaddr));   
  109.                 ip->server = srv;  
  110.                 loop_push_to_head(&srv->loop_ips, ip);  
  111.                 srv->cur_ips ++;  
  112.             }  
  113.         }  
  114.         if(ip != NULL){  
  115.             if(ip->cur_conns >= srv->max_conns_per_ip || srv->cur_conns>=srv->max_conns){  
  116.                 shutdown(fd, SHUT_RDWR);  
  117.                 closesocket(fd);  
  118.             }else{  
  119.                 struct ftpconn* conn = (struct ftpconn*)malloc(sizeof(struct ftpconn));  
  120.                 memset(conn, 0, sizeof(struct ftpconn));  
  121.                 conn->conn_fd = fd;  
  122.                 conn->time_create =   
  123.                 conn->time_alive = time(NULL);  
  124.                 conn->ip = ip;  
  125.                 conn->server = srv;  
  126.                 conn->opts_utf8 = 1;  
  127.                 loop_push_to_head(&srv->loop_conns, conn);  
  128.                 ip->cur_conns ++;  
  129.                 srv->cur_conns ++;  
  130.                 DBG("new connection %d from %s, ips:%d conns:%d", fd, ip->ipstr, srv->cur_ips, srv->cur_conns);  
  131.                 int ret = pthread_create(&conn->thread_conn, &srv->thread_attr, conn_start, (void*)conn);  
  132.             }  
  133.         }  
  134.     }  
  135.     return NULL;  
  136. }  
  137.  
  138. static int loop_conns_iterator(const void*p, const void*q)  
  139. {  
  140.     struct ftpconn* conn = (struct ftpconn*)p;  
  141.     if((time_t)q - conn->time_alive >= conn->server->conn_timeout){  
  142.         DBG("timeout connection of user: %s",   
  143.             q, conn->time_alive, conn->username);  
  144.         conn->conn_end = 1;  
  145.         if(conn->dataconn_fd){  
  146.             shutdown(conn->dataconn_fd, SHUT_RDWR);  
  147.             closesocket(conn->dataconn_fd);  
  148.             conn->dataconn_fd = 0;  
  149.         }  
  150.         if(conn->conn_fd){  
  151.             shutdown(conn->conn_fd, SHUT_RDWR);  
  152.             closesocket(conn->conn_fd);  
  153.             conn->conn_fd = 0;  
  154.         }  
  155.     }  
  156.     return conn->conn_end;  
  157. }  
  158.  
  159. static int loop_ips_iterator(const void*p, const void*q)  
  160. {  
  161.     struct ftpip* ip = (struct ftpip*)p;  
  162.     if(ip->cur_conns == 0)  
  163.         return 1;  
  164.     return 0;  
  165. }  
  166.  
  167. static void * server_guard(void* data)  
  168. {  
  169.     struct ftpserver* srv = (struct ftpserver*)data;  
  170.     while(!srv->server_end){  
  171.         time_t t = time(NULL);  
  172.         struct ftpconn* conn;  
  173.         while( conn = loop_search(&srv->loop_conns, (void*)t, loop_conns_iterator) ){  
  174.             pthread_join(conn->thread_conn, NULL);  
  175.               
  176.             if(conn->dataconn_fd){  
  177.                 closesocket(conn->dataconn_fd);  
  178.                 conn->dataconn_fd = 0;  
  179.             }  
  180.             if(conn->conn_fd){  
  181.                 closesocket(conn->conn_fd);  
  182.                 conn->conn_fd = 0;  
  183.             }  
  184.             loop_remove(&srv->loop_conns, conn);  
  185.             conn->ip->cur_conns --;  
  186.             srv->cur_conns --;  
  187.             free(conn);  
  188.         }  
  189.         struct ftpip* ip;  
  190.         while( ip = loop_search(&srv->loop_ips, (void*)0, loop_ips_iterator) ){  
  191.             loop_remove(&srv->loop_ips, ip);  
  192.             srv->cur_ips --;  
  193.             free(ip);  
  194.         }  
  195.         SLEEP(1);  
  196.     }  
  197.     return NULL;  
  198. }  
  199.  
  200. int server_start(struct ftpserver* srv)  
  201. {  
  202.     int ret;  
  203.       
  204.     pthread_attr_init(&srv->thread_attr);  
  205.     pthread_attr_setstacksize(&srv->thread_attr, 128*1024);  
  206.       
  207.     srv->server_fd = socket( AF_INET, SOCK_STREAM, IPPROTO_TCP );  
  208.       
  209.     /* Socket Reuse */ 
  210.     ret = 1;  
  211.     setsockopt(srv->server_fd, SOL_SOCKET, SO_REUSEADDR, (void*)&ret, sizeof(ret));  
  212.       
  213.     /* Bind Socket */ 
  214.     struct sockaddr_in addr;  
  215.     memset(&addr,0, sizeof(addr));  
  216.     addr.sin_family = AF_INET;  
  217.     if( !srv->server_ip[0] )  
  218.         addr.sin_addr.s_addr = INADDR_ANY;  
  219.     else 
  220.         addr.sin_addr.s_addr = inet_addr( srv->server_ip );  
  221.     addr.sin_port = htons( srv->server_port );  
  222.     if ( bind( srv->server_fd, (struct sockaddr*)&addr, sizeof( addr ) ) <0 ){  
  223.         DBG("Bind Socket Error.");  
  224.         return 1;  
  225.     }  
  226.     //try to listen  
  227.     if( listen( srv->server_fd, srv->max_conns ) <0 ){  
  228.         DBG("Listen Socket Error.");  
  229.         return 1;  
  230.     }  
  231.     srv->time_start = time(NULL);  
  232.     DBG("listening on %s:%d", srv->server_ip, srv->server_port );  
  233.       
  234.     loop_create(&srv->loop_ips, srv->max_ips, NULL);  
  235.     loop_create(&srv->loop_conns, srv->max_conns, NULL);  
  236.       
  237.     /* create threads */ 
  238.     ret = pthread_create( &srv->thread_guard, &srv->thread_attr, (void*)server_guard, (void*)srv );  
  239.       
  240.     ret = pthread_create( &srv->thread_listen, &srv->thread_attr, (void*)server_listen, (void*)srv );  
  241.       
  242.     return ret;  
  243. }  
  244.  
  245. int server_stop(struct ftpserver* srv)  
  246. {  
  247.     srv->server_end = 1;  
  248.     shutdown(srv->server_fd, 0);  
  249.     closesocket(srv->server_fd);  
  250.       
  251.     pthread_join(srv->thread_listen, NULL);  
  252.     pthread_join(srv->thread_guard, NULL);  
  253.       
  254.     pthread_attr_destroy(&srv->thread_attr);  
  255. }  

 

你可能感兴趣的:(c,职场,休闲)