例如:客户端请求服务器端下载文件zImage,其扩展选项opt1 : timeout 5,opt 2 : blksize 1462
optcode filename mode opt1 opt2
---------------------------------------------------------------------------------
1 | zImage\0 | octet\0 | timeout\05\0 | blksize\01462\0 |
---------------------------------------------------------------------------------
注意:optcode占用2bytes,如果不带扩展选项,opt1,opt2就不需要添加了。如果使用扩展选项,每个扩展选项以NULL结束.
使用扩展选项可以使tftp包的收发更灵活,比如如果不使用扩展选项的话,blksize的默认值为512,即每次携带的数据大小为512byte。如果小于512就认为是最后一个数据包了。如果使用扩展选项的话,我们可以和服务器端商量,比如说让blksize的值大一点,常用的是1462。这样传输同样大小的文件,显然是使用扩展选项更快一点。
3 Data (DATA)
数据包:
2 bytes 2 bytes n bytes
----------------------------------
| Opcode | Block # | Data |
----------------------------------
例如:发送第一个数据包,内容为"hello word"
opcode Block # Data
------------------------------------------------
3 | 1 | hello word\0 |
-------------------------------------------------
//应答包
4 Acknowledgment (ACK)
2 bytes 2 bytes
---------------------
| Opcode | Block # |
---------------------
例如:服务器端发送完一个数据包给客户端,客户端收到后应做出应答
Opcode Block#
-----------------------------
4 | 1 |
------------------------------
5 Error (ERROR)
2 bytes 2 bytes string
-----------------------------------------
| Opcode | ErrorCode | ErrMsg \0 |
-----------------------------------------
一个ERROR包,它的操作码是5,它的格式如上所示。此包可以被其它任何类型的包确认。错误码指定错误的类型。错误的值和错误的意义在附录中。错误信息是供程序员使用的。
/*
*TFTP error code
*/
enum{
TFTP_ERR_UNDEFINED = 0,
TFTP_ERR_FILE_NOT_FOUND = 1,
TFTP_ERR_ACCESS_DENIED = 2,
TFTP_ERR_DISK_FULL = 3,
TFTP_ERR_UNEXPECTED_OPCODE = 4,
TFTP_ERR_UNKNOWN_TRANSFER_ID = 5,
TFTP_ERR_FILE_ALREADY_EXISTS = 6,
};
6 OACK(带扩展选项)
例如:
我们上面客户端用可选选项进行发读请求,服务器端收到后,做出应答如下:
optcode opt1 opt2
----------------------------------------------------
1 | timeout\05\0 | blksize\01462\0 |
----------------------------------------------------
注意:如果客户端请求时所带的选项,服务器端并不支持,此时服务器端将发出错误包
四、客户端和服务器端的交互
实例用wirshak抓包如下:
最后附上源码:
注意一点就是,第一次发出请求时,服务器端的端口号是69,服务器端收到请求之后,做出应答,在这个应答包中包含服务器端和客户端交互的端口号,客户端收到应答之后,要保留好服务器端的端口号。