summaryrefslogtreecommitdiffstats
path: root/fs
diff options
context:
space:
mode:
authorSascha Hauer <s.hauer@pengutronix.de>2015-07-03 08:37:54 +0200
committerSascha Hauer <s.hauer@pengutronix.de>2015-07-03 08:37:54 +0200
commitc408a4ecbee60a9d512930972c09c6f9e365a671 (patch)
tree6bc428653d8da4cc181eb017e76601a2273e212d /fs
parent45019dc10b217d249c07cd47bc342f04317d1a7d (diff)
parent8bbc1064c78cdf658c837b996170b8eab8cbcb3d (diff)
downloadbarebox-c408a4ecbee60a9d512930972c09c6f9e365a671.tar.gz
barebox-c408a4ecbee60a9d512930972c09c6f9e365a671.tar.xz
Merge branch 'for-next/net'
Diffstat (limited to 'fs')
-rw-r--r--fs/tftp.c47
1 files changed, 26 insertions, 21 deletions
diff --git a/fs/tftp.c b/fs/tftp.c
index 72e4983a01..0de215e5e9 100644
--- a/fs/tftp.c
+++ b/fs/tftp.c
@@ -253,27 +253,24 @@ static void tftp_timer_reset(struct file_priv *priv)
priv->progress_timeout = priv->resend_timeout = get_time_ns();
}
-static void tftp_handler(void *ctx, char *packet, unsigned len)
+static void tftp_recv(struct file_priv *priv,
+ uint8_t *pkt, unsigned len, uint16_t uh_sport)
{
- struct file_priv *priv = ctx;
- uint16_t proto;
- uint16_t *s;
- char *pkt = net_eth_to_udp_payload(packet);
- struct udphdr *udp = net_eth_to_udphdr(packet);
+ uint16_t opcode;
- len = net_eth_to_udplen(packet);
- if (len < 2)
+ /* according to RFC1350 minimal tftp packet length is 4 bytes */
+ if (len < 4)
return;
- len -= 2;
+ opcode = ntohs(*(uint16_t *)pkt);
- s = (uint16_t *)pkt;
- proto = *s++;
- pkt = (unsigned char *)s;
+ /* skip tftp opcode 2-byte field */
+ len -= 2;
+ pkt += 2;
- debug("%s: proto 0x%04x\n", __func__, proto);
+ debug("%s: opcode 0x%04x\n", __func__, opcode);
- switch (ntohs(proto)) {
+ switch (opcode) {
case TFTP_RRQ:
case TFTP_WRQ:
default:
@@ -296,14 +293,14 @@ static void tftp_handler(void *ctx, char *packet, unsigned len)
priv->state = STATE_DONE;
break;
}
- priv->tftp_con->udp->uh_dport = udp->uh_sport;
+ priv->tftp_con->udp->uh_dport = uh_sport;
priv->state = STATE_WDATA;
break;
case TFTP_OACK:
tftp_parse_oack(priv, pkt, len);
- priv->server_port = ntohs(udp->uh_sport);
- priv->tftp_con->udp->uh_dport = udp->uh_sport;
+ priv->server_port = ntohs(uh_sport);
+ priv->tftp_con->udp->uh_dport = uh_sport;
if (priv->push) {
/* send first block */
@@ -318,16 +315,14 @@ static void tftp_handler(void *ctx, char *packet, unsigned len)
break;
case TFTP_DATA:
- if (len < 2)
- return;
len -= 2;
priv->block = ntohs(*(uint16_t *)pkt);
if (priv->state == STATE_RRQ || priv->state == STATE_OACK) {
/* first block received */
priv->state = STATE_RDATA;
- priv->tftp_con->udp->uh_dport = udp->uh_sport;
- priv->server_port = ntohs(udp->uh_sport);
+ priv->tftp_con->udp->uh_dport = uh_sport;
+ priv->server_port = ntohs(uh_sport);
priv->last_block = 0;
if (priv->block != 1) { /* Assertion */
@@ -376,6 +371,16 @@ static void tftp_handler(void *ctx, char *packet, unsigned len)
}
}
+static void tftp_handler(void *ctx, char *packet, unsigned len)
+{
+ struct file_priv *priv = ctx;
+ char *pkt = net_eth_to_udp_payload(packet);
+ struct udphdr *udp = net_eth_to_udphdr(packet);
+
+ (void)len;
+ tftp_recv(priv, pkt, net_eth_to_udplen(packet), udp->uh_sport);
+}
+
static struct file_priv *tftp_do_open(struct device_d *dev,
int accmode, const char *filename)
{