⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 tftp_server.c

📁 ecos实时嵌入式操作系统
💻 C
📖 第 1 页 / 共 2 页
字号:
    local_addr.sin_addr.s_addr = htonl(INADDR_ANY);    local_addr.sin_port = htons(INADDR_ANY);    if (bind(s, (struct sockaddr *)&local_addr, sizeof(local_addr)) < 0) {        // Problem setting up my end        diag_printf("TFTPD: can't bind to reply port for 'read_file'\n");        close(s);        return;    }    if ((fd = (server->ops->open)(hdr->th_stuff, O_RDONLY)) < 0) {        tftpd_send_error(s,reply,TFTP_ENOTFOUND,from_addr, from_len);        close(s);        return;    }    block = 0;    ok = true;    while (ok) {        // Read next chunk of file        len = (server->ops->read)(fd, reply->th_data, SEGSIZE);        reply->th_block = htons(++block); // preincrement        reply->th_opcode = htons(DATA);        for (tries = 0;  tries < TFTP_RETRIES_MAX;  tries++) {#ifdef CYGOPT_NET_TFTP_SERVER_INSTRUMENT            tftp_server_instrument.data.send++;#endif            if (sendto(s, reply, 4+len, 0,                       (struct sockaddr *)from_addr, from_len) < 0) {                // Something went wrong with the network!                ok = false;                break;            }        repeat_select:            timeout.tv_sec = TFTP_TIMEOUT_PERIOD;            timeout.tv_usec = 0;            FD_ZERO(&fds);            FD_SET(s, &fds);            if (select(s+1, &fds, 0, 0, &timeout) <= 0) {                if (++total_timeouts > TFTP_TIMEOUT_MAX) {#ifdef CYGOPT_NET_TFTP_SERVER_INSTRUMENT                    tftp_server_instrument.err_send++;#endif 		    tftpd_send_error(s,reply,TFTP_EBADOP,from_addr, from_len);                    ok = false;                    break;                }#ifdef CYGOPT_NET_TFTP_SERVER_INSTRUMENT                tftp_server_instrument.data.resend++;#endif                continue; // retry the send, using up one retry.            }            data_len = sizeof(data_in);            client_len = sizeof(client_addr);            if ((data_len = recvfrom(s, data_in, data_len, 0,                                      (struct sockaddr *)&client_addr,                                     &client_len)) < 0) {                // What happened?  Maybe someone lied to us...#ifdef CYGOPT_NET_TFTP_SERVER_INSTRUMENT                tftp_server_instrument.data.resend++;#endif                continue; // retry the send, using up one retry.            }            if ((ntohs(response->th_opcode) == ACK) &&                (ntohs(response->th_block) < block)) {#ifdef CYGOPT_NET_TFTP_SERVER_INSTRUMENT                tftp_server_instrument.ack.rx_repeat++;#endif                // Then it is a repeat ACK for an old block; listen again,                // but do not repeat sending the current block, and do not                // use up a retry count.  (we do re-send the data if                // subsequently we time out)                goto repeat_select;            }            if ((ntohs(response->th_opcode) == ACK) &&                (ntohs(response->th_block) == block)) {#ifdef CYGOPT_NET_TFTP_SERVER_INSTRUMENT                tftp_server_instrument.ack.rx++;#endif                // Happy!  Break out of the retries loop.                break;            }            // Otherwise, we got something we do not understand!  So repeat            // sending the current block, and use up a retry count.#ifdef CYGOPT_NET_TFTP_SERVER_INSTRUMENT            if ( (ntohs(response->th_opcode) == ACK) )                tftp_server_instrument.ack.rx_skip++;            tftp_server_instrument.data.resend++;#endif        } // End of the retries loop.        if (TFTP_RETRIES_MAX <= tries) {#ifdef CYGOPT_NET_TFTP_SERVER_INSTRUMENT            tftp_server_instrument.err_send++;#endif            tftpd_send_error(s,reply,TFTP_EBADOP,from_addr, from_len);            ok = false;        }        if (len < SEGSIZE) {            break; // That's end of file then.        }    }    close(s);    (server->ops->close)(fd);}//// Actual TFTP server//#define CYGSEM_TFTP_SERVER_MULTITHREADED#ifdef CYGSEM_TFTP_SERVER_MULTITHREADEDstatic cyg_sem_t tftp_server_sem;#endifstatic voidtftpd_server(cyg_addrword_t p){    struct tftp_server *server = (struct tftp_server *)p;    int s;    int data_len, recv_len, from_len;    struct sockaddr_in local_addr, from_addr;    struct servent *server_info;    char data[SEGSIZE+sizeof(struct tftphdr)];    struct tftphdr *hdr = (struct tftphdr *)data;#ifndef CYGPKG_NET_TESTS_USE_RT_TEST_HARNESS    // Otherwise routine printfs fail the test - interrupts disabled too long.    diag_printf("TFTPD [%x]: port %d\n", p, server->port);#endif    // Set up port    if (server->port == 0) {        server_info = getservbyname("tftp", "udp");        if (server_info == (struct servent *)0) {            diag_printf("TFTPD: can't get TFTP service information\n");            return;        }        // Network order in info; host order in server:        server->port = ntohs( server_info->s_port );    }    while (true) {#ifdef CYGOPT_NET_TFTP_SERVER_INSTRUMENT        struct info o = tftp_server_instrument;#endif        // Create socket        s = socket(AF_INET, SOCK_DGRAM, 0);        if (s < 0) {            diag_printf("TFTPD [%x]: can't open socket\n", p);            return;        }        memset((char *)&local_addr, 0, sizeof(local_addr));        local_addr.sin_family = AF_INET;        local_addr.sin_len = sizeof(local_addr);        local_addr.sin_addr.s_addr = htonl(INADDR_ANY);        local_addr.sin_port = htons(server->port);        if (bind(s, (struct sockaddr *)&local_addr, sizeof(local_addr)) < 0) {            // Problem setting up my end            close(s);#ifdef CYGSEM_TFTP_SERVER_MULTITHREADED#ifndef CYGPKG_NET_TESTS_USE_RT_TEST_HARNESS            diag_printf("TFTPD [%x]: waiting to bind to service port\n", p);#endif            // Wait until the socket is free...            cyg_semaphore_wait( &tftp_server_sem );            continue; // try re-opening and rebinding the socket.#else            diag_printf("TFTPD [%x]: can't bind to service port\n", p);            return;#endif        }        recv_len = sizeof(data);        from_len = sizeof(from_addr);        data_len = recvfrom(s, hdr, recv_len, 0,                            (struct sockaddr *)&from_addr, &from_len);        close(s); // so that other servers can bind to the TFTP socket#ifdef CYGSEM_TFTP_SERVER_MULTITHREADED        // The socket is free...        cyg_semaphore_post( &tftp_server_sem );#endif        if ( data_len < 0) {            diag_printf("TFTPD [%x]: can't read request\n", p);        } else {#ifndef CYGPKG_NET_TESTS_USE_RT_TEST_HARNESS            diag_printf("TFTPD [%x]: received %x from %s:%d\n", p,                        ntohs(hdr->th_opcode), inet_ntoa(from_addr.sin_addr),                        from_addr.sin_port);#endif            switch (ntohs(hdr->th_opcode)) {            case WRQ:                tftpd_write_file(server, hdr, &from_addr, from_len);                break;            case RRQ:                tftpd_read_file(server, hdr, &from_addr, from_len);                break;            case ACK:            case DATA:            case ERROR:                // Ignore                break;            default:                diag_printf("TFTPD [%x]: bogus request %x from %s:%d\n", p,                            ntohs(hdr->th_opcode),                            inet_ntoa(from_addr.sin_addr),                            from_addr.sin_port );                tftpd_send_error(s,hdr,TFTP_EBADOP,&from_addr,from_len);            }        }#ifdef CYGOPT_NET_TFTP_SERVER_INSTRUMENT        tftp_server_instrument.total_transactions++;        o.data.rx        -= tftp_server_instrument.data.rx       ;        o.data.rx_repeat -= tftp_server_instrument.data.rx_repeat;        o.data.rx_skip   -= tftp_server_instrument.data.rx_skip  ;        o.data.send      -= tftp_server_instrument.data.send     ;        o.data.resend    -= tftp_server_instrument.data.resend   ;        o.ack.rx         -= tftp_server_instrument.ack.rx        ;        o.ack.rx_repeat  -= tftp_server_instrument.ack.rx_repeat ;        o.ack.rx_skip    -= tftp_server_instrument.ack.rx_skip   ;        o.ack.send       -= tftp_server_instrument.ack.send      ;        o.ack.resend     -= tftp_server_instrument.ack.resend    ;        o.err_send       -= tftp_server_instrument.err_send      ;#ifndef CYGPKG_NET_TESTS_USE_RT_TEST_HARNESS        if ( o.data.rx        ) diag_printf( "data rx       %4d\n", -o.data.rx        );        if ( o.data.rx_repeat ) diag_printf( "data rx_repeat%4d\n", -o.data.rx_repeat );        if ( o.data.rx_skip   ) diag_printf( "data rx_skip  %4d\n", -o.data.rx_skip   );        if ( o.data.send      ) diag_printf( "data send     %4d\n", -o.data.send      );        if ( o.data.resend    ) diag_printf( "data resend   %4d\n", -o.data.resend    );        if ( o.ack.rx        ) diag_printf( " ack rx       %4d\n", -o.ack.rx        );        if ( o.ack.rx_repeat ) diag_printf( " ack rx_repeat%4d\n", -o.ack.rx_repeat );        if ( o.ack.rx_skip   ) diag_printf( " ack rx_skip  %4d\n", -o.ack.rx_skip   );        if ( o.ack.send      ) diag_printf( " ack send     %4d\n", -o.ack.send      );        if ( o.ack.resend    ) diag_printf( " ack resend   %4d\n", -o.ack.resend    );        if ( o.err_send      ) diag_printf( "*error sends  %4d\n", -o.err_send      );#endif // CYGPKG_NET_TESTS_USE_RT_TEST_HARNESS#endif // CYGOPT_NET_TFTP_SERVER_INSTRUMENT    }}//// This function is used to create a new server [thread] which supports// the TFTP protocol on the given port.  A server 'id' will be returned// which can later be used to destroy the server.  //// Note: all [memory] resources for the server thread will be allocated// dynamically.  If there are insufficient resources, an error will be// returned.//int tftpd_start(int port, struct tftpd_fileops *ops){    struct tftp_server *server;#ifdef CYGSEM_TFTP_SERVER_MULTITHREADED    static char init = 0;    if ( 0 == init ) {        init++;        cyg_semaphore_init( &tftp_server_sem, 0 );    }#endif    if ((server = malloc(sizeof(struct tftp_server)))) {        server->tag = TFTP_tag;        server->port = port;        server->ops = ops;        cyg_thread_create(CYGPKG_NET_TFTPD_THREAD_PRIORITY, // Priority                          tftpd_server,              // entry                          (cyg_addrword_t)server,    // entry parameter                          "TFTP server",             // Name                          &server->stack[0],         // Stack                          STACK_SIZE,                // Size                          &server->thread_handle,    // Handle                          &server->thread_data       // Thread data structure            );        cyg_thread_resume(server->thread_handle);  // Start it    }    return (int)server;}//// Destroy a TFTP server, using a previously created server 'id'.//int tftpd_stop(int p){    struct tftp_server *server = (struct tftp_server *)p;    // Simple sanity check    if (server->tag == TFTP_tag) {        cyg_thread_kill(server->thread_handle);        cyg_thread_set_priority(server->thread_handle, 0);        cyg_thread_delay(1);  // Make sure it gets to die...        if (cyg_thread_delete(server->thread_handle)) {            // Success shutting down the thread            free(server);  // Give up memory            return 1;        }    }    return 0;}// EOF tftp_server.c

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -