Browse Source

Add preliminary support for MTFTP.

tags/v0.9.3
Michael Brown 17 years ago
parent
commit
f3265b4bf8
2 changed files with 297 additions and 112 deletions
  1. 2
    0
      src/include/gpxe/tftp.h
  2. 295
    112
      src/net/udp/tftp.c

+ 2
- 0
src/include/gpxe/tftp.h View File

29
 #define TFTP_ERR_UNKNOWN_USER	7 /**< No such user */
29
 #define TFTP_ERR_UNKNOWN_USER	7 /**< No such user */
30
 #define TFTP_ERR_BAD_OPTS	8 /**< Option negotiation failed */
30
 #define TFTP_ERR_BAD_OPTS	8 /**< Option negotiation failed */
31
 
31
 
32
+#define MTFTP_PORT	     1759 /**< Default MTFTP server port */
33
+
32
 /** A TFTP read request (RRQ) packet */
34
 /** A TFTP read request (RRQ) packet */
33
 struct tftp_rrq {
35
 struct tftp_rrq {
34
 	uint16_t opcode;
36
 	uint16_t opcode;

+ 295
- 112
src/net/udp/tftp.c View File

74
 	 * "tsize" option, this value will be zero.
74
 	 * "tsize" option, this value will be zero.
75
 	 */
75
 	 */
76
 	unsigned long tsize;
76
 	unsigned long tsize;
77
-	/** Multicast address
78
-	 *
79
-	 * This is the destination address for multicast data
80
-	 * transmissions.
81
-	 */
82
-	struct sockaddr_tcpip multicast;
83
-	/** Master client
77
+	
78
+	/** Server port
84
 	 *
79
 	 *
85
-	 * True if this is the client responsible for sending ACKs.
80
+	 * This is the port to which RRQ packets are sent.
86
 	 */
81
 	 */
87
-	int master;
88
-	
82
+	unsigned int port;
89
 	/** Peer address
83
 	/** Peer address
90
 	 *
84
 	 *
91
 	 * The peer address is determined by the first response
85
 	 * The peer address is determined by the first response
92
 	 * received to the TFTP RRQ.
86
 	 * received to the TFTP RRQ.
93
 	 */
87
 	 */
94
 	struct sockaddr_tcpip peer;
88
 	struct sockaddr_tcpip peer;
89
+	/** Request flags */
90
+	unsigned int flags;
91
+	/** MTFTP timeout count */
92
+	unsigned int mtftp_timeouts;
93
+
95
 	/** Block bitmap */
94
 	/** Block bitmap */
96
 	struct bitmap bitmap;
95
 	struct bitmap bitmap;
97
 	/** Maximum known length
96
 	/** Maximum known length
110
 	struct retry_timer timer;
109
 	struct retry_timer timer;
111
 };
110
 };
112
 
111
 
112
+/** TFTP request flags */
113
+enum {
114
+	/** Send ACK packets */
115
+	TFTP_FL_SEND_ACK = 0x0001,
116
+	/** Request blksize and tsize options */
117
+	TFTP_FL_RRQ_SIZES = 0x0002,
118
+	/** Request multicast option */
119
+	TFTP_FL_RRQ_MULTICAST = 0x0004,
120
+	/** Perform MTFTP recovery on timeout */
121
+	TFTP_FL_MTFTP_RECOVERY = 0x0008,
122
+};
123
+
124
+/** Maximum number of MTFTP open requests before falling back to TFTP */
125
+#define MTFTP_MAX_TIMEOUTS 3
126
+
113
 /**
127
 /**
114
  * Free TFTP request
128
  * Free TFTP request
115
  *
129
  *
147
 	xfer_close ( &tftp->xfer, rc );
161
 	xfer_close ( &tftp->xfer, rc );
148
 }
162
 }
149
 
163
 
164
+/**
165
+ * Reopen TFTP socket
166
+ *
167
+ * @v tftp		TFTP connection
168
+ * @ret rc		Return status code
169
+ */
170
+static int tftp_reopen ( struct tftp_request *tftp ) {
171
+	struct sockaddr_tcpip server;
172
+	int rc;
173
+
174
+	/* Close socket */
175
+	xfer_close ( &tftp->socket, 0 );
176
+
177
+	/* Disable ACK sending. */
178
+	tftp->flags &= ~TFTP_FL_SEND_ACK;
179
+
180
+	/* Reset peer address */
181
+	memset ( &tftp->peer, 0, sizeof ( tftp->peer ) );
182
+
183
+	/* Open socket */
184
+	memset ( &server, 0, sizeof ( server ) );
185
+	server.st_port = htons ( tftp->port );
186
+	if ( ( rc = xfer_open_named_socket ( &tftp->socket, SOCK_DGRAM,
187
+					     ( struct sockaddr * ) &server,
188
+					     tftp->uri->host, NULL ) ) != 0 ) {
189
+		DBGC ( tftp, "TFTP %p could not open socket: %s\n",
190
+		       tftp, strerror ( rc ) );
191
+		return rc;
192
+	}
193
+
194
+	return 0;
195
+}
196
+
197
+/**
198
+ * Reopen TFTP multicast socket
199
+ *
200
+ * @v tftp		TFTP connection
201
+ * @v local		Local socket address
202
+ * @ret rc		Return status code
203
+ */
204
+static int tftp_reopen_mc ( struct tftp_request *tftp,
205
+			    struct sockaddr *local ) {
206
+	int rc;
207
+
208
+	/* Close multicast socket */
209
+	xfer_close ( &tftp->mc_socket, 0 );
210
+
211
+	/* Open multicast socket.  We never send via this socket, so
212
+	 * use the local address as the peer address (since the peer
213
+	 * address cannot be NULL).
214
+	 */
215
+	if ( ( rc = xfer_open_socket ( &tftp->mc_socket, SOCK_DGRAM,
216
+				       local, local ) ) != 0 ) {
217
+		DBGC ( tftp, "TFTP %p could not open multicast "
218
+		       "socket: %s\n", tftp, strerror ( rc ) );
219
+		return rc;
220
+	}
221
+
222
+	return 0;
223
+}
224
+
150
 /**
225
 /**
151
  * Presize TFTP receive buffers and block bitmap
226
  * Presize TFTP receive buffers and block bitmap
152
  *
227
  *
201
 	tftp_request_blksize = blksize;
276
 	tftp_request_blksize = blksize;
202
 }
277
 }
203
 
278
 
279
+/**
280
+ * MTFTP multicast receive address
281
+ *
282
+ * This is treated as a global configuration parameter.
283
+ */
284
+static struct sockaddr_in tftp_mtftp_socket = {
285
+	.sin_family = AF_INET,
286
+	.sin_addr.s_addr = htonl ( 0xefff0101 ),
287
+	.sin_port = htons ( 3001 ),
288
+};
289
+
290
+/**
291
+ * Set MTFTP multicast address
292
+ *
293
+ * @v address		Multicast IPv4 address
294
+ */
295
+void tftp_set_mtftp_address ( struct in_addr address ) {
296
+	tftp_mtftp_socket.sin_addr = address;
297
+}
298
+
299
+/**
300
+ * Set MTFTP multicast port
301
+ *
302
+ * @v port		Multicast port
303
+ */
304
+void tftp_set_mtftp_port ( unsigned int port ) {
305
+	tftp_mtftp_socket.sin_port = htons ( port );
306
+}
307
+
204
 /**
308
 /**
205
  * Transmit RRQ
309
  * Transmit RRQ
206
  *
310
  *
227
 	/* Build request */
331
 	/* Build request */
228
 	rrq = iob_put ( iobuf, sizeof ( *rrq ) );
332
 	rrq = iob_put ( iobuf, sizeof ( *rrq ) );
229
 	rrq->opcode = htons ( TFTP_RRQ );
333
 	rrq->opcode = htons ( TFTP_RRQ );
230
-	iob_put ( iobuf,
231
-		  snprintf ( rrq->data, iob_tailroom ( iobuf ),
232
-			     "%s%coctet%cblksize%c%d%ctsize%c0%cmulticast%c",
233
-			     path, 0, 0, 0, tftp_request_blksize, 0,
234
-			     0, 0, 0 ) + 1 );
334
+	iob_put ( iobuf, snprintf ( iobuf->tail, iob_tailroom ( iobuf ),
335
+				    "%s%coctet", path, 0 ) + 1 );
336
+	if ( tftp->flags & TFTP_FL_RRQ_SIZES ) {
337
+		iob_put ( iobuf, snprintf ( iobuf->tail,
338
+					    iob_tailroom ( iobuf ),
339
+					    "blksize%c%d%ctsize%c0", 0,
340
+					    tftp_request_blksize, 0, 0 ) + 1 );
341
+	}
342
+	if ( tftp->flags & TFTP_FL_RRQ_MULTICAST ) {
343
+		iob_put ( iobuf, snprintf ( iobuf->tail,
344
+					    iob_tailroom ( iobuf ),
345
+					    "multicast%c", 0 ) + 1 );
346
+	}
235
 
347
 
236
 	/* RRQ always goes to the address specified in the initial
348
 	/* RRQ always goes to the address specified in the initial
237
 	 * xfer_open() call
349
 	 * xfer_open() call
283
 	stop_timer ( &tftp->timer );
395
 	stop_timer ( &tftp->timer );
284
 	start_timer ( &tftp->timer );
396
 	start_timer ( &tftp->timer );
285
 
397
 
286
-	/* If we are the master client, send RRQ or ACK as appropriate */
287
-	if ( tftp->master ) {
288
-		if ( ! tftp->peer.st_family ) {
289
-			return tftp_send_rrq ( tftp );
290
-		} else {
398
+	/* Send RRQ or ACK as appropriate */
399
+	if ( ! tftp->peer.st_family ) {
400
+		return tftp_send_rrq ( tftp );
401
+	} else {
402
+		if ( tftp->flags & TFTP_FL_SEND_ACK ) {
291
 			return tftp_send_ack ( tftp );
403
 			return tftp_send_ack ( tftp );
404
+		} else {
405
+			return 0;
292
 		}
406
 		}
293
-	} else {
294
-		/* Do nothing when not the master client */
295
-		return 0;
296
 	}
407
 	}
297
 }
408
 }
298
 
409
 
305
 static void tftp_timer_expired ( struct retry_timer *timer, int fail ) {
416
 static void tftp_timer_expired ( struct retry_timer *timer, int fail ) {
306
 	struct tftp_request *tftp =
417
 	struct tftp_request *tftp =
307
 		container_of ( timer, struct tftp_request, timer );
418
 		container_of ( timer, struct tftp_request, timer );
419
+	int rc;
308
 
420
 
309
-	if ( fail ) {
310
-		tftp_done ( tftp, -ETIMEDOUT );
421
+	/* If we are doing MTFTP, attempt the various recovery strategies */
422
+	if ( tftp->flags & TFTP_FL_MTFTP_RECOVERY ) {
423
+		if ( tftp->peer.st_family ) {
424
+			/* If we have received any response from the server,
425
+			 * try resending the RRQ to restart the download.
426
+			 */
427
+			DBGC ( tftp, "TFTP %p attempting reopen\n", tftp );
428
+			if ( ( rc = tftp_reopen ( tftp ) ) != 0 )
429
+				goto err;
430
+		} else {
431
+			/* Fall back to plain TFTP after several attempts */
432
+			tftp->mtftp_timeouts++;
433
+			DBGC ( tftp, "TFTP %p timeout %d waiting for MTFTP "
434
+			       "open\n", tftp, tftp->mtftp_timeouts );
435
+
436
+			if ( tftp->mtftp_timeouts > MTFTP_MAX_TIMEOUTS ) {
437
+				DBGC ( tftp, "TFTP %p falling back to plain "
438
+				       "TFTP\n", tftp );
439
+				tftp->flags = TFTP_FL_RRQ_SIZES;
440
+
441
+				/* Close multicast socket */
442
+				xfer_close ( &tftp->mc_socket, 0 );
443
+
444
+				/* Reset retry timer */
445
+				start_timer_nodelay ( &tftp->timer );
446
+
447
+				/* The blocksize may change: discard
448
+				 * the block bitmap
449
+				 */
450
+				bitmap_free ( &tftp->bitmap );
451
+				memset ( &tftp->bitmap, 0,
452
+					 sizeof ( tftp->bitmap ) );
453
+
454
+				/* Reopen on standard TFTP port */
455
+				tftp->port = TFTP_PORT;
456
+				if ( ( rc = tftp_reopen ( tftp ) ) != 0 )
457
+					goto err;
458
+			}
459
+		}
311
 	} else {
460
 	} else {
312
-		tftp_send_packet ( tftp );
461
+		/* Not doing MTFTP (or have fallen back to plain
462
+		 * TFTP); fail as per normal.
463
+		 */
464
+		if ( fail ) {
465
+			rc = -ETIMEDOUT;
466
+			goto err;
467
+		}
313
 	}
468
 	}
469
+	tftp_send_packet ( tftp );
470
+	return;
471
+
472
+ err:
473
+	tftp_done ( tftp, rc );
314
 }
474
 }
315
 
475
 
316
 /**
476
 /**
366
  */
526
  */
367
 static int tftp_process_multicast ( struct tftp_request *tftp,
527
 static int tftp_process_multicast ( struct tftp_request *tftp,
368
 				    const char *value ) {
528
 				    const char *value ) {
369
-	struct sockaddr_in *sin = ( struct sockaddr_in * ) &tftp->multicast;
529
+	union {
530
+		struct sockaddr sa;
531
+		struct sockaddr_in sin;
532
+	} socket;
370
 	char buf[ strlen ( value ) + 1 ];
533
 	char buf[ strlen ( value ) + 1 ];
371
 	char *addr;
534
 	char *addr;
372
 	char *port;
535
 	char *port;
373
 	char *port_end;
536
 	char *port_end;
374
 	char *mc;
537
 	char *mc;
375
 	char *mc_end;
538
 	char *mc_end;
376
-	struct sockaddr *mc_peer;
377
-	struct sockaddr *mc_local;
378
 	int rc;
539
 	int rc;
379
 
540
 
380
 	/* Split value into "addr,port,mc" fields */
541
 	/* Split value into "addr,port,mc" fields */
394
 	*(mc++) = '\0';
555
 	*(mc++) = '\0';
395
 
556
 
396
 	/* Parse parameters */
557
 	/* Parse parameters */
397
-	if ( *addr ) {
398
-		if ( inet_aton ( addr, &sin->sin_addr ) == 0 ) {
558
+	if ( strtoul ( mc, &mc_end, 0 ) == 0 )
559
+		tftp->flags &= ~TFTP_FL_SEND_ACK;
560
+	if ( *mc_end ) {
561
+		DBGC ( tftp, "TFTP %p multicast invalid mc %s\n", tftp, mc );
562
+		return -EINVAL;
563
+	}
564
+	DBGC ( tftp, "TFTP %p is%s the master client\n",
565
+	       tftp, ( ( tftp->flags & TFTP_FL_SEND_ACK ) ? "" : " not" ) );
566
+	if ( *addr && *port ) {
567
+		socket.sin.sin_family = AF_INET;
568
+		if ( inet_aton ( addr, &socket.sin.sin_addr ) == 0 ) {
399
 			DBGC ( tftp, "TFTP %p multicast invalid IP address "
569
 			DBGC ( tftp, "TFTP %p multicast invalid IP address "
400
 			       "%s\n", tftp, addr );
570
 			       "%s\n", tftp, addr );
401
 			return -EINVAL;
571
 			return -EINVAL;
402
 		}
572
 		}
403
 		DBGC ( tftp, "TFTP %p multicast IP address %s\n",
573
 		DBGC ( tftp, "TFTP %p multicast IP address %s\n",
404
-		       tftp, inet_ntoa ( sin->sin_addr ) );
405
-	}
406
-	if ( *port ) {
407
-		sin->sin_port = htons ( strtoul ( port, &port_end, 0 ) );
574
+		       tftp, inet_ntoa ( socket.sin.sin_addr ) );
575
+		socket.sin.sin_port = htons ( strtoul ( port, &port_end, 0 ) );
408
 		if ( *port_end ) {
576
 		if ( *port_end ) {
409
 			DBGC ( tftp, "TFTP %p multicast invalid port %s\n",
577
 			DBGC ( tftp, "TFTP %p multicast invalid port %s\n",
410
 			       tftp, port );
578
 			       tftp, port );
411
 			return -EINVAL;
579
 			return -EINVAL;
412
 		}
580
 		}
413
 		DBGC ( tftp, "TFTP %p multicast port %d\n",
581
 		DBGC ( tftp, "TFTP %p multicast port %d\n",
414
-		       tftp, ntohs ( sin->sin_port ) );
415
-	}
416
-	tftp->master = strtoul ( mc, &mc_end, 0 );
417
-	if ( *mc_end ) {
418
-		DBGC ( tftp, "TFTP %p multicast invalid mc %s\n", tftp, mc );
419
-		return -EINVAL;
420
-	}
421
-	DBGC ( tftp, "TFTP %p is%s the master client\n",
422
-	       tftp, ( tftp->master ? "" : " not" ) );
423
-
424
-	/* Open multicast socket, if new address specified */
425
-	if ( *addr || *port ) {
426
-		xfer_close ( &tftp->mc_socket, 0 );
427
-		mc_peer = ( ( struct sockaddr * ) &tftp->peer );
428
-		mc_local = ( ( struct sockaddr * ) &tftp->multicast );
429
-		mc_local->sa_family = mc_peer->sa_family;
430
-		if ( ( rc = xfer_open_socket ( &tftp->mc_socket, SOCK_DGRAM,
431
-					       mc_peer, mc_local ) ) != 0 ) {
432
-			DBGC ( tftp, "TFTP %p could not open multicast "
433
-			       "socket: %s\n", tftp, strerror ( rc ) );
582
+		       tftp, ntohs ( socket.sin.sin_port ) );
583
+		if ( ( rc = tftp_reopen_mc ( tftp, &socket.sa ) ) != 0 )
434
 			return rc;
584
 			return rc;
435
-		}
436
 	}
585
 	}
437
 
586
 
438
 	return 0;
587
 	return 0;
611
 }
760
 }
612
 
761
 
613
 /** Translation between TFTP errors and internal error numbers */
762
 /** Translation between TFTP errors and internal error numbers */
614
-static const uint8_t tftp_errors[] = {
615
-	[TFTP_ERR_FILE_NOT_FOUND]	= PXENV_STATUS_TFTP_FILE_NOT_FOUND,
616
-	[TFTP_ERR_ACCESS_DENIED]	= PXENV_STATUS_TFTP_ACCESS_VIOLATION,
617
-	[TFTP_ERR_ILLEGAL_OP]		= PXENV_STATUS_TFTP_UNKNOWN_OPCODE,
763
+static const int tftp_errors[] = {
764
+	[TFTP_ERR_FILE_NOT_FOUND]	= ENOENT,
765
+	[TFTP_ERR_ACCESS_DENIED]	= EACCES,
766
+	[TFTP_ERR_ILLEGAL_OP]		= ENOTSUP,
618
 };
767
 };
619
 
768
 
620
 /**
769
 /**
645
 	if ( err < ( sizeof ( tftp_errors ) / sizeof ( tftp_errors[0] ) ) )
794
 	if ( err < ( sizeof ( tftp_errors ) / sizeof ( tftp_errors[0] ) ) )
646
 		rc = -tftp_errors[err];
795
 		rc = -tftp_errors[err];
647
 	if ( ! rc )
796
 	if ( ! rc )
648
-		rc = -PXENV_STATUS_TFTP_CANNOT_OPEN_CONNECTION;
797
+		rc = -ENOTSUP;
649
 
798
 
650
 	/* Close TFTP request */
799
 	/* Close TFTP request */
651
 	tftp_done ( tftp, rc );
800
 	tftp_done ( tftp, rc );
736
 	struct tftp_request *tftp =
885
 	struct tftp_request *tftp =
737
 		container_of ( socket, struct tftp_request, socket );
886
 		container_of ( socket, struct tftp_request, socket );
738
 
887
 
739
-	return tftp_rx ( tftp, iobuf, meta );
740
-}
741
-
742
-/**
743
- * TFTP connection closed by network stack
744
- *
745
- * @v socket		Transport layer interface
746
- * @v rc		Reason for close
747
- */
748
-static void tftp_socket_close ( struct xfer_interface *socket, int rc ) {
749
-	struct tftp_request *tftp =
750
-		container_of ( socket, struct tftp_request, socket );
751
-
752
-	DBGC ( tftp, "TFTP %p socket closed: %s\n",
753
-	       tftp, strerror ( rc ) );
754
-
755
-	/* Any close counts as an error */
756
-	if ( ! rc )
757
-		rc = -ECONNRESET;
888
+	/* Enable sending ACKs when we receive a unicast packet.  This
889
+	 * covers three cases:
890
+	 *
891
+	 * 1. Standard TFTP; we should always send ACKs, and will
892
+	 *    always receive a unicast packet before we need to send the
893
+	 *    first ACK.
894
+	 *
895
+	 * 2. RFC2090 multicast TFTP; the only unicast packets we will
896
+         *    receive are the OACKs; enable sending ACKs here (before
897
+         *    processing the OACK) and disable it when processing the
898
+         *    multicast option if we are not the master client.
899
+	 *
900
+	 * 3. MTFTP; receiving a unicast datagram indicates that we
901
+	 *    are the "master client" and should send ACKs.
902
+	 */
903
+	tftp->flags |= TFTP_FL_SEND_ACK;
758
 
904
 
759
-	tftp_done ( tftp, rc );
905
+	return tftp_rx ( tftp, iobuf, meta );
760
 }
906
 }
761
 
907
 
762
 /** TFTP socket operations */
908
 /** TFTP socket operations */
763
 static struct xfer_interface_operations tftp_socket_operations = {
909
 static struct xfer_interface_operations tftp_socket_operations = {
764
-	.close		= tftp_socket_close,
910
+	.close		= ignore_xfer_close,
765
 	.vredirect	= xfer_vopen,
911
 	.vredirect	= xfer_vopen,
766
 	.seek		= ignore_xfer_seek,
912
 	.seek		= ignore_xfer_seek,
767
 	.window		= unlimited_xfer_window,
913
 	.window		= unlimited_xfer_window,
787
 	return tftp_rx ( tftp, iobuf, meta );
933
 	return tftp_rx ( tftp, iobuf, meta );
788
 }
934
 }
789
 
935
 
790
-/**
791
- * TFTP multicast connection closed by network stack
792
- *
793
- * @v socket		Multicast transport layer interface
794
- * @v rc		Reason for close
795
- */
796
-static void tftp_mc_socket_close ( struct xfer_interface *mc_socket,
797
-				   int rc ) {
798
-	struct tftp_request *tftp =
799
-		container_of ( mc_socket, struct tftp_request, mc_socket );
800
-
801
-	DBGC ( tftp, "TFTP %p multicast socket closed: %s\n",
802
-	       tftp, strerror ( rc ) );
803
-
804
-	/* The multicast socket may be closed when we receive a new
805
-	 * OACK and open/reopen the socket; we should not call
806
-	 * tftp_done() at this point.
807
-	 */
808
-}
809
- 
810
 /** TFTP multicast socket operations */
936
 /** TFTP multicast socket operations */
811
 static struct xfer_interface_operations tftp_mc_socket_operations = {
937
 static struct xfer_interface_operations tftp_mc_socket_operations = {
812
-	.close		= tftp_mc_socket_close,
938
+	.close		= ignore_xfer_close,
813
 	.vredirect	= xfer_vopen,
939
 	.vredirect	= xfer_vopen,
814
 	.seek		= ignore_xfer_seek,
940
 	.seek		= ignore_xfer_seek,
815
 	.window		= unlimited_xfer_window,
941
 	.window		= unlimited_xfer_window,
846
 };
972
 };
847
 
973
 
848
 /**
974
 /**
849
- * Initiate TFTP download
975
+ * Initiate TFTP/TFTM/MTFTP download
850
  *
976
  *
851
  * @v xfer		Data transfer interface
977
  * @v xfer		Data transfer interface
852
  * @v uri		Uniform Resource Identifier
978
  * @v uri		Uniform Resource Identifier
853
  * @ret rc		Return status code
979
  * @ret rc		Return status code
854
  */
980
  */
855
-int tftp_open ( struct xfer_interface *xfer, struct uri *uri ) {
981
+static int tftp_core_open ( struct xfer_interface *xfer, struct uri *uri,
982
+			    unsigned int default_port,
983
+			    struct sockaddr *multicast,
984
+			    unsigned int flags ) {
856
 	struct tftp_request *tftp;
985
 	struct tftp_request *tftp;
857
-	struct sockaddr_tcpip server;
858
 	int rc;
986
 	int rc;
859
 
987
 
860
 	/* Sanity checks */
988
 	/* Sanity checks */
874
 	xfer_init ( &tftp->mc_socket, &tftp_mc_socket_operations,
1002
 	xfer_init ( &tftp->mc_socket, &tftp_mc_socket_operations,
875
 		    &tftp->refcnt );
1003
 		    &tftp->refcnt );
876
 	tftp->blksize = TFTP_DEFAULT_BLKSIZE;
1004
 	tftp->blksize = TFTP_DEFAULT_BLKSIZE;
877
-	tftp->master = 1;
1005
+	tftp->flags = flags;
878
 	tftp->timer.expired = tftp_timer_expired;
1006
 	tftp->timer.expired = tftp_timer_expired;
879
 
1007
 
880
 	/* Open socket */
1008
 	/* Open socket */
881
-	memset ( &server, 0, sizeof ( server ) );
882
-	server.st_port = htons ( uri_port ( tftp->uri, TFTP_PORT ) );
883
-	if ( ( rc = xfer_open_named_socket ( &tftp->socket, SOCK_DGRAM,
884
-					     ( struct sockaddr * ) &server,
885
-					     uri->host, NULL ) ) != 0 )
1009
+	tftp->port = uri_port ( tftp->uri, default_port );
1010
+	if ( ( rc = tftp_reopen ( tftp ) ) != 0 )
886
 		goto err;
1011
 		goto err;
887
 
1012
 
1013
+	/* Open multicast socket */
1014
+	if ( multicast ) {
1015
+		if ( ( rc = tftp_reopen_mc ( tftp, multicast ) ) != 0 )
1016
+			goto err;
1017
+	}
1018
+
888
 	/* Start timer to initiate RRQ */
1019
 	/* Start timer to initiate RRQ */
889
 	start_timer_nodelay ( &tftp->timer );
1020
 	start_timer_nodelay ( &tftp->timer );
890
 
1021
 
901
 	return rc;
1032
 	return rc;
902
 }
1033
 }
903
 
1034
 
1035
+/**
1036
+ * Initiate TFTP download
1037
+ *
1038
+ * @v xfer		Data transfer interface
1039
+ * @v uri		Uniform Resource Identifier
1040
+ * @ret rc		Return status code
1041
+ */
1042
+static int tftp_open ( struct xfer_interface *xfer, struct uri *uri ) {
1043
+	return tftp_core_open ( xfer, uri, TFTP_PORT, NULL,
1044
+				TFTP_FL_RRQ_SIZES );
1045
+
1046
+}
1047
+
904
 /** TFTP URI opener */
1048
 /** TFTP URI opener */
905
 struct uri_opener tftp_uri_opener __uri_opener = {
1049
 struct uri_opener tftp_uri_opener __uri_opener = {
906
 	.scheme	= "tftp",
1050
 	.scheme	= "tftp",
907
 	.open	= tftp_open,
1051
 	.open	= tftp_open,
908
 };
1052
 };
1053
+
1054
+/**
1055
+ * Initiate TFTM download
1056
+ *
1057
+ * @v xfer		Data transfer interface
1058
+ * @v uri		Uniform Resource Identifier
1059
+ * @ret rc		Return status code
1060
+ */
1061
+static int tftm_open ( struct xfer_interface *xfer, struct uri *uri ) {
1062
+	return tftp_core_open ( xfer, uri, TFTP_PORT, NULL,
1063
+				( TFTP_FL_RRQ_SIZES |
1064
+				  TFTP_FL_RRQ_MULTICAST ) );
1065
+
1066
+}
1067
+
1068
+/** TFTM URI opener */
1069
+struct uri_opener tftm_uri_opener __uri_opener = {
1070
+	.scheme	= "tftm",
1071
+	.open	= tftm_open,
1072
+};
1073
+
1074
+/**
1075
+ * Initiate MTFTP download
1076
+ *
1077
+ * @v xfer		Data transfer interface
1078
+ * @v uri		Uniform Resource Identifier
1079
+ * @ret rc		Return status code
1080
+ */
1081
+static int mtftp_open ( struct xfer_interface *xfer, struct uri *uri ) {
1082
+	return tftp_core_open ( xfer, uri, MTFTP_PORT,
1083
+				( struct sockaddr * ) &tftp_mtftp_socket,
1084
+				TFTP_FL_MTFTP_RECOVERY );
1085
+}
1086
+
1087
+/** MTFTP URI opener */
1088
+struct uri_opener mtftp_uri_opener __uri_opener = {
1089
+	.scheme	= "mtftp",
1090
+	.open	= mtftp_open,
1091
+};

Loading…
Cancel
Save