diff options
author | Jan Kiszka <jan.kiszka@siemens.com> | 2009-06-24 14:42:30 +0200 |
---|---|---|
committer | Anthony Liguori <aliguori@us.ibm.com> | 2009-06-29 08:52:47 -0500 |
commit | 20c24bf24ddf91fd9ab5d827fdd0fe14397de412 (patch) | |
tree | d9fd9b243004bf10f25c717b3d35b6c0c2402f85 /slirp/tftp.c | |
parent | ef2d54d8df7e4a515b603fe101e73f716d81e778 (diff) | |
download | qemu-20c24bf24ddf91fd9ab5d827fdd0fe14397de412.tar.gz qemu-20c24bf24ddf91fd9ab5d827fdd0fe14397de412.tar.bz2 qemu-20c24bf24ddf91fd9ab5d827fdd0fe14397de412.zip |
slirp: tftp: Refactor tftp_handle_rrq
Specifically make the filename extraction more readable, and always
report errors back to the client.
Signed-off-by: Jan Kiszka <jan.kiszka@siemens.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
Diffstat (limited to 'slirp/tftp.c')
-rw-r--r-- | slirp/tftp.c | 49 |
1 files changed, 21 insertions, 28 deletions
diff --git a/slirp/tftp.c b/slirp/tftp.c index 75aa4b24f6..83519088f5 100644 --- a/slirp/tftp.c +++ b/slirp/tftp.c @@ -27,7 +27,7 @@ struct tftp_session { int in_use; - unsigned char filename[TFTP_FILENAME_MAX]; + char filename[TFTP_FILENAME_MAX]; struct in_addr client_ip; u_int16_t client_port; @@ -273,8 +273,8 @@ static int tftp_send_data(struct tftp_session *spt, static void tftp_handle_rrq(struct tftp_t *tp, int pktlen) { struct tftp_session *spt; - int s, k, n; - u_int8_t *src, *dst; + int s, k; + char *req_fname; s = tftp_session_allocate(tp); @@ -290,37 +290,31 @@ static void tftp_handle_rrq(struct tftp_t *tp, int pktlen) return; } - src = tp->x.tp_buf; - dst = spt->filename; - n = pktlen - ((uint8_t *)&tp->x.tp_buf[0] - (uint8_t *)tp); + /* skip header fields */ + k = 0; + pktlen -= ((uint8_t *)&tp->x.tp_buf[0] - (uint8_t *)tp); /* get name */ + req_fname = spt->filename; - for (k = 0; k < n; k++) { - if (k < TFTP_FILENAME_MAX) { - dst[k] = src[k]; - } - else { + while (1) { + if (k >= TFTP_FILENAME_MAX || k >= pktlen) { + tftp_send_error(spt, 2, "Access violation", tp); return; } - - if (src[k] == '\0') { + req_fname[k] = (char)tp->x.tp_buf[k]; + if (req_fname[k++] == '\0') { break; } } - if (k >= n) { - return; - } - - k++; - /* check mode */ - if ((n - k) < 6) { + if ((pktlen - k) < 6) { + tftp_send_error(spt, 2, "Access violation", tp); return; } - if (memcmp(&src[k], "octet\0", 6) != 0) { + if (memcmp(&tp->x.tp_buf[k], "octet\0", 6) != 0) { tftp_send_error(spt, 4, "Unsupported transfer mode", tp); return; } @@ -337,29 +331,28 @@ static void tftp_handle_rrq(struct tftp_t *tp, int pktlen) } /* check if the file exists */ - - if (tftp_read_data(spt, 0, spt->filename, 0) < 0) { + if (tftp_read_data(spt, 0, NULL, 0) < 0) { tftp_send_error(spt, 1, "File not found", tp); return; } - if (src[n - 1] != 0) { + if (tp->x.tp_buf[pktlen - 1] != 0) { tftp_send_error(spt, 2, "Access violation", tp); return; } - while (k < n) { + while (k < pktlen) { const char *key, *value; - key = (char *)src + k; + key = (const char *)&tp->x.tp_buf[k]; k += strlen(key) + 1; - if (k >= n) { + if (k >= pktlen) { tftp_send_error(spt, 2, "Access violation", tp); return; } - value = (char *)src + k; + value = (const char *)&tp->x.tp_buf[k]; k += strlen(value) + 1; if (strcmp(key, "tsize") == 0) { |