aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJan Kiszka <jan.kiszka@siemens.com>2009-06-24 14:42:30 +0200
committerAnthony Liguori <aliguori@us.ibm.com>2009-06-29 08:52:47 -0500
commit806cd7b29df5e9208b91b11a504ff198345493d0 (patch)
tree1f26a0ffee7b4df73f404aea6e62608d2833e8c7
parent6e111e20a2822507ab345c274705ed10dfc1eb80 (diff)
downloadslirp-806cd7b29df5e9208b91b11a504ff198345493d0.zip
slirp-806cd7b29df5e9208b91b11a504ff198345493d0.tar.gz
slirp-806cd7b29df5e9208b91b11a504ff198345493d0.tar.bz2
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>
-rw-r--r--tftp.c48
1 files changed, 21 insertions, 27 deletions
diff --git a/tftp.c b/tftp.c
index 1dc2358..bf27585 100644
--- a/tftp.c
+++ b/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;
@@ -271,8 +271,8 @@ static int tftp_send_data(struct tftp_session *spt, u_int16_t block_nr,
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);
@@ -288,36 +288,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;
}
@@ -334,29 +329,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) {