Path: blob/a-new-beginning/SharedDependencies/Sources/libslirp/tftp.c
2 views
/* SPDX-License-Identifier: MIT */1/*2* tftp.c - a simple, read-only tftp server for qemu3*4* Copyright (c) 2004 Magnus Damm <[email protected]>5*6* Permission is hereby granted, free of charge, to any person obtaining a copy7* of this software and associated documentation files (the "Software"), to deal8* in the Software without restriction, including without limitation the rights9* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell10* copies of the Software, and to permit persons to whom the Software is11* furnished to do so, subject to the following conditions:12*13* The above copyright notice and this permission notice shall be included in14* all copies or substantial portions of the Software.15*16* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR17* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,18* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL19* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER20* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,21* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN22* THE SOFTWARE.23*/2425#include "slirp.h"2627#include <sys/types.h>28#include <sys/stat.h>29#include <fcntl.h>3031static inline int tftp_session_in_use(struct tftp_session *spt)32{33return (spt->slirp != NULL);34}3536static inline void tftp_session_update(struct tftp_session *spt)37{38spt->timestamp = curtime;39}4041static void tftp_session_terminate(struct tftp_session *spt)42{43if (spt->fd >= 0) {44close(spt->fd);45spt->fd = -1;46}47g_free(spt->filename);48spt->slirp = NULL;49}5051static int tftp_session_allocate(Slirp *slirp, struct sockaddr_storage *srcsas,52struct tftphdr *hdr)53{54struct tftp_session *spt;55int k;5657for (k = 0; k < TFTP_SESSIONS_MAX; k++) {58spt = &slirp->tftp_sessions[k];5960if (!tftp_session_in_use(spt))61goto found;6263/* sessions time out after 5 inactive seconds */64if ((int)(curtime - spt->timestamp) > 5000) {65tftp_session_terminate(spt);66goto found;67}68}6970return -1;7172found:73memset(spt, 0, sizeof(*spt));74memcpy(&spt->client_addr, srcsas, sockaddr_size(srcsas));75spt->fd = -1;76spt->block_size = 512;77spt->client_port = hdr->udp.uh_sport;78spt->slirp = slirp;7980tftp_session_update(spt);8182return k;83}8485static int tftp_session_find(Slirp *slirp, struct sockaddr_storage *srcsas,86struct tftphdr *hdr)87{88struct tftp_session *spt;89int k;9091for (k = 0; k < TFTP_SESSIONS_MAX; k++) {92spt = &slirp->tftp_sessions[k];9394if (tftp_session_in_use(spt)) {95if (sockaddr_equal(&spt->client_addr, srcsas)) {96if (spt->client_port == hdr->udp.uh_sport) {97return k;98}99}100}101}102103return -1;104}105106void tftp_cleanup(Slirp *slirp)107{108struct tftp_session *spt;109int k;110111for (k = 0; k < TFTP_SESSIONS_MAX; k++) {112spt = &slirp->tftp_sessions[k];113114if (tftp_session_in_use(spt)) {115tftp_session_terminate(spt);116}117}118}119120static int tftp_read_data(struct tftp_session *spt, uint32_t block_nr,121uint8_t *buf, int len)122{123int bytes_read = 0;124125if (spt->fd < 0) {126spt->fd = open(spt->filename, O_RDONLY | O_BINARY);127}128129if (spt->fd < 0) {130return -1;131}132133if (len) {134if (lseek(spt->fd, block_nr * spt->block_size, SEEK_SET) == (off_t)-1) {135return -1;136}137138bytes_read = read(spt->fd, buf, len);139}140141return bytes_read;142}143144static struct tftp_t *tftp_prep_mbuf_data(struct tftp_session *spt,145struct mbuf *m)146{147struct tftp_t *tp;148149memset(m->m_data, 0, m->m_size);150151m->m_data += IF_MAXLINKHDR;152if (spt->client_addr.ss_family == AF_INET6) {153m->m_data += sizeof(struct ip6);154} else {155m->m_data += sizeof(struct ip);156}157tp = (void *)m->m_data;158m->m_data += sizeof(struct udphdr);159160return tp;161}162163static void tftp_udp_output(struct tftp_session *spt, struct mbuf *m,164struct tftphdr *hdr)165{166if (spt->client_addr.ss_family == AF_INET6) {167struct sockaddr_in6 sa6, da6;168169sa6.sin6_addr = spt->slirp->vhost_addr6;170sa6.sin6_port = hdr->udp.uh_dport;171da6.sin6_addr = ((struct sockaddr_in6 *)&spt->client_addr)->sin6_addr;172da6.sin6_port = spt->client_port;173174udp6_output(NULL, m, &sa6, &da6);175} else {176struct sockaddr_in sa4, da4;177178sa4.sin_addr = spt->slirp->vhost_addr;179sa4.sin_port = hdr->udp.uh_dport;180da4.sin_addr = ((struct sockaddr_in *)&spt->client_addr)->sin_addr;181da4.sin_port = spt->client_port;182183udp_output(NULL, m, &sa4, &da4, IPTOS_LOWDELAY);184}185}186187static int tftp_send_oack(struct tftp_session *spt, const char *keys[],188uint32_t values[], int nb, struct tftp_t *recv_tp)189{190struct mbuf *m;191struct tftp_t *tp;192int i, n = 0;193194m = m_get(spt->slirp);195196if (!m)197return -1;198199tp = tftp_prep_mbuf_data(spt, m);200201tp->hdr.tp_op = htons(TFTP_OACK);202for (i = 0; i < nb; i++) {203n += slirp_fmt0(tp->x.tp_buf + n, sizeof(tp->x.tp_buf) - n, "%s", keys[i]);204n += slirp_fmt0(tp->x.tp_buf + n, sizeof(tp->x.tp_buf) - n, "%u", values[i]);205}206207m->m_len = G_SIZEOF_MEMBER(struct tftp_t, hdr.tp_op) + n;208tftp_udp_output(spt, m, &recv_tp->hdr);209210return 0;211}212213static void tftp_send_error(struct tftp_session *spt, uint16_t errorcode,214const char *msg, struct tftp_t *recv_tp)215{216struct mbuf *m;217struct tftp_t *tp;218219DEBUG_TFTP("tftp error msg: %s", msg);220221m = m_get(spt->slirp);222223if (!m) {224goto out;225}226227tp = tftp_prep_mbuf_data(spt, m);228229tp->hdr.tp_op = htons(TFTP_ERROR);230tp->x.tp_error.tp_error_code = htons(errorcode);231slirp_pstrcpy((char *)tp->x.tp_error.tp_msg, sizeof(tp->x.tp_error.tp_msg),232msg);233234m->m_len = sizeof(struct tftp_t) - (TFTP_BLOCKSIZE_MAX + 2) + 3 +235strlen(msg) - sizeof(struct udphdr);236tftp_udp_output(spt, m, &recv_tp->hdr);237238out:239tftp_session_terminate(spt);240}241242static void tftp_send_next_block(struct tftp_session *spt,243struct tftphdr *hdr)244{245struct mbuf *m;246struct tftp_t *tp;247int nobytes;248249m = m_get(spt->slirp);250251if (!m) {252return;253}254255tp = tftp_prep_mbuf_data(spt, m);256257tp->hdr.tp_op = htons(TFTP_DATA);258tp->x.tp_data.tp_block_nr = htons((spt->block_nr + 1) & 0xffff);259260nobytes = tftp_read_data(spt, spt->block_nr, tp->x.tp_data.tp_buf,261spt->block_size);262263if (nobytes < 0) {264/* send "file not found" error back */265266tftp_send_error(spt, 1, "File not found", tp);267268m_free(m);269270return;271}272273m->m_len = sizeof(struct tftp_t) - (TFTP_BLOCKSIZE_MAX - nobytes) -274sizeof(struct udphdr);275tftp_udp_output(spt, m, hdr);276277if (nobytes == spt->block_size) {278tftp_session_update(spt);279} else {280tftp_session_terminate(spt);281}282283spt->block_nr++;284}285286static void tftp_handle_rrq(Slirp *slirp, struct sockaddr_storage *srcsas,287struct tftp_t *tp, int pktlen)288{289struct tftp_session *spt;290int s, k;291size_t prefix_len;292char *req_fname;293const char *option_name[2];294uint32_t option_value[2];295int nb_options = 0;296297/* check if a session already exists and if so terminate it */298s = tftp_session_find(slirp, srcsas, &tp->hdr);299if (s >= 0) {300tftp_session_terminate(&slirp->tftp_sessions[s]);301}302303s = tftp_session_allocate(slirp, srcsas, &tp->hdr);304305if (s < 0) {306return;307}308309spt = &slirp->tftp_sessions[s];310311/* unspecified prefix means service disabled */312if (!slirp->tftp_prefix) {313tftp_send_error(spt, 2, "Access violation", tp);314return;315}316317/* skip header fields */318k = 0;319pktlen -= offsetof(struct tftp_t, x.tp_buf);320321/* prepend tftp_prefix */322prefix_len = strlen(slirp->tftp_prefix);323spt->filename = g_malloc(prefix_len + TFTP_FILENAME_MAX + 2);324memcpy(spt->filename, slirp->tftp_prefix, prefix_len);325spt->filename[prefix_len] = '/';326327/* get name */328req_fname = spt->filename + prefix_len + 1;329330while (1) {331if (k >= TFTP_FILENAME_MAX || k >= pktlen) {332tftp_send_error(spt, 2, "Access violation", tp);333return;334}335req_fname[k] = tp->x.tp_buf[k];336if (req_fname[k++] == '\0') {337break;338}339}340341DEBUG_TFTP("tftp rrq file: %s", req_fname);342343/* check mode */344if ((pktlen - k) < 6) {345tftp_send_error(spt, 2, "Access violation", tp);346return;347}348349if (g_ascii_strcasecmp(&tp->x.tp_buf[k], "octet") != 0) {350tftp_send_error(spt, 4, "Unsupported transfer mode", tp);351return;352}353354k += 6; /* skipping octet */355356/* do sanity checks on the filename */357if (358#ifdef G_OS_WIN32359strstr(req_fname, "..\\") ||360req_fname[strlen(req_fname) - 1] == '\\' ||361#endif362strstr(req_fname, "../") ||363req_fname[strlen(req_fname) - 1] == '/') {364tftp_send_error(spt, 2, "Access violation", tp);365return;366}367368/* check if the file exists */369if (tftp_read_data(spt, 0, NULL, 0) < 0) {370tftp_send_error(spt, 1, "File not found", tp);371return;372}373374if (tp->x.tp_buf[pktlen - 1] != 0) {375tftp_send_error(spt, 2, "Access violation", tp);376return;377}378379while (k < pktlen && nb_options < G_N_ELEMENTS(option_name)) {380const char *key, *value;381382key = &tp->x.tp_buf[k];383k += strlen(key) + 1;384385if (k >= pktlen) {386tftp_send_error(spt, 2, "Access violation", tp);387return;388}389390value = &tp->x.tp_buf[k];391k += strlen(value) + 1;392393if (g_ascii_strcasecmp(key, "tsize") == 0) {394int tsize = atoi(value);395struct stat stat_p;396397if (tsize == 0) {398if (stat(spt->filename, &stat_p) == 0)399tsize = stat_p.st_size;400else {401tftp_send_error(spt, 1, "File not found", tp);402return;403}404}405406option_name[nb_options] = "tsize";407option_value[nb_options] = tsize;408nb_options++;409} else if (g_ascii_strcasecmp(key, "blksize") == 0) {410int blksize = atoi(value);411412/* Accept blksize up to our maximum size */413if (blksize > 0) {414spt->block_size = MIN(blksize, TFTP_BLOCKSIZE_MAX);415option_name[nb_options] = "blksize";416option_value[nb_options] = spt->block_size;417nb_options++;418}419}420}421422if (nb_options > 0) {423assert(nb_options <= G_N_ELEMENTS(option_name));424tftp_send_oack(spt, option_name, option_value, nb_options, tp);425return;426}427428spt->block_nr = 0;429tftp_send_next_block(spt, &tp->hdr);430}431432static void tftp_handle_ack(Slirp *slirp, struct sockaddr_storage *srcsas,433struct tftphdr *hdr)434{435int s;436437s = tftp_session_find(slirp, srcsas, hdr);438439if (s < 0) {440return;441}442443tftp_send_next_block(&slirp->tftp_sessions[s], hdr);444}445446static void tftp_handle_error(Slirp *slirp, struct sockaddr_storage *srcsas,447struct tftphdr *hdr)448{449int s;450451s = tftp_session_find(slirp, srcsas, hdr);452453if (s < 0) {454return;455}456457tftp_session_terminate(&slirp->tftp_sessions[s]);458}459460void tftp_input(struct sockaddr_storage *srcsas, struct mbuf *m)461{462struct tftphdr *hdr = mtod_check(m, sizeof(struct tftphdr));463464if (hdr == NULL) {465return;466}467468switch (ntohs(hdr->tp_op)) {469case TFTP_RRQ:470tftp_handle_rrq(m->slirp, srcsas,471mtod(m, struct tftp_t *),472m->m_len);473break;474475case TFTP_ACK:476tftp_handle_ack(m->slirp, srcsas, hdr);477break;478479case TFTP_ERROR:480tftp_handle_error(m->slirp, srcsas, hdr);481break;482}483}484485486