#include #include #include #include #define PORT_OUT 69 #define PORT_IN 7642 enum { TFTP_RRQ = 1, /* Read request */ TFTP_WRQ = 2, /* Write request */ TFTP_DATA = 3, /* Data */ TFTP_ACK = 4, /* Acknowledgment */ TFTP_ERROR = 5, /* Error */ }; #define BLOCK_SIZE 512 /* block size in bytes */ static int format_request(uint8_t *buf, uint16_t op, const char *filename) { int len = strlen(filename); *buf++ = op >> 8; /* Opcode */ *buf++ = op; memcpy(buf, filename, len); buf += len; *buf++ = 0x00; *buf++ = 'o'; *buf++ = 'c'; *buf++ = 't'; *buf++ = 'e'; *buf++ = 't'; *buf++ = 0x00; return 9+strlen(filename); } static int format_ack(uint8_t *buf, uint16_t block) { *buf++ = 0x00; /* Opcode: Ack */ *buf++ = TFTP_ACK; *buf++ = (block & 0xff00) >> 8; *buf++ = (block & 0x00ff); return 4; } static int format_data(uint8_t *buf, uint16_t block, const void *data, int len) { *buf++ = 0x00; /* Opcode: Data*/ *buf++ = TFTP_DATA; *buf++ = (block & 0xff00) >> 8; *buf++ = (block & 0x00ff); memcpy(buf, data, len); return len+4; } static uint8_t *packet_data; static int total_length; static int transfer_finished; static uint8_t *dst_buffer; static int last_ack; /* signed, so we can use -1 */ static uint16_t data_port; static void rx_callback(uint32_t src_ip, uint16_t src_port, uint16_t dst_port, void *_data, unsigned int length) { uint8_t *data = _data; uint16_t opcode; uint16_t block; int i; int offset; if(length < 4) return; if(dst_port != PORT_IN) return; opcode = data[0] << 8 | data[1]; block = data[2] << 8 | data[3]; if(opcode == TFTP_ACK) { /* Acknowledgement */ data_port = src_port; last_ack = block; return; } if(block < 1) return; if(opcode == TFTP_DATA) { /* Data */ length -= 4; offset = (block-1)*BLOCK_SIZE; for(i=0;i 0) || transfer_finished) break; } if((total_length > 0) || transfer_finished) break; tries--; if(tries == 0) { microudp_set_callback(NULL); return -1; } } length_before = total_length; while(!transfer_finished) { if(length_before != total_length) { i = 12000000; length_before = total_length; } if(i-- == 0) { microudp_set_callback(NULL); return -1; } microudp_service(); } microudp_set_callback(NULL); return total_length; } int tftp_put(uint32_t ip, const char *filename, const void *buffer, int size) { int len, send; int tries; int i; int block = 0, sent = 0; if(!microudp_arp_resolve(ip)) return -1; microudp_set_callback(rx_callback); packet_data = microudp_get_tx_buffer(); total_length = 0; transfer_finished = 0; tries = 5; while(1) { len = format_request(packet_data, TFTP_WRQ, filename); microudp_send(PORT_IN, PORT_OUT, len); for(i=0;i<2000000;i++) { last_ack = -1; microudp_service(); if(last_ack == block) goto send_data; if(transfer_finished) goto fail; } tries--; if(tries == 0) goto fail; } send_data: do { block++; send = sent+BLOCK_SIZE > size ? size-sent : BLOCK_SIZE; tries = 5; while(1) { len = format_data(packet_data, block, buffer, send); microudp_send(PORT_IN, data_port, len); for(i=0;i<12000000;i++) { microudp_service(); if(transfer_finished) goto fail; if(last_ack == block) goto next; } if (!--tries) goto fail; } next: sent += send; buffer += send; } while (send == BLOCK_SIZE); microudp_set_callback(NULL); return sent; fail: microudp_set_callback(NULL); return -1; }