From: Olivier Valentin <valen...@free.fr> When static buffers are used, PATH_MAX becomes the size of the provided buffer, and is passed as a parameters to the underlying implementation. --- frontends/framebuffer/fetch.c | 2 +- frontends/gtk/fetch.c | 2 +- frontends/gtk/gui.c | 2 +- frontends/monkey/fetch.c | 2 +- frontends/monkey/main.c | 2 +- frontends/windows/fetch.c | 2 +- utils/filepath.c | 143 ++++++++++++++++++---------------- utils/filepath.h | 39 +++------- 8 files changed, 92 insertions(+), 102 deletions(-)
diff --git a/frontends/framebuffer/fetch.c b/frontends/framebuffer/fetch.c index d58a82205..f917bddf3 100644 --- a/frontends/framebuffer/fetch.c +++ b/frontends/framebuffer/fetch.c @@ -54,7 +54,7 @@ static nsurl *get_resource_url(const char *path) if (strcmp(path, "favicon.ico") == 0) path = "favicon.png"; - if (filepath_sfind(respaths, buf, path) == NSERROR_OK) + if (filepath_sfind(respaths, buf, PATH_MAX, path) == NSERROR_OK) netsurf_path_to_nsurl(buf, &url); return url; diff --git a/frontends/gtk/fetch.c b/frontends/gtk/fetch.c index 36ee36c17..98682ffc4 100644 --- a/frontends/gtk/fetch.c +++ b/frontends/gtk/fetch.c @@ -257,7 +257,7 @@ static nsurl *nsgtk_get_resource_url(const char *path) if (strcmp(path, "favicon.ico") == 0) { nsurl_create("resource:favicon.png", &url); } else { - if (filepath_sfind(respaths, buf, path) == NSERROR_OK) + if (filepath_sfind(respaths, buf, PATH_MAX, path) == NSERROR_OK) netsurf_path_to_nsurl(buf, &url); } diff --git a/frontends/gtk/gui.c b/frontends/gtk/gui.c index fa0c4063b..2c5558ddb 100644 --- a/frontends/gtk/gui.c +++ b/frontends/gtk/gui.c @@ -987,7 +987,7 @@ static nserror nsgtk_setup(int argc, char** argv, char **respath) .pma = true, }); - filepath_sfinddef(respath, buf, "mime.types", "/etc/"); + filepath_sfinddef(respath, buf, PATH_MAX, "mime.types", "/etc/"); gtk_fetch_filetype_init(buf); save_complete_init(); diff --git a/frontends/monkey/fetch.c b/frontends/monkey/fetch.c index 96ba28d67..f6d554915 100644 --- a/frontends/monkey/fetch.c +++ b/frontends/monkey/fetch.c @@ -39,7 +39,7 @@ static nsurl *gui_get_resource_url(const char *path) char buf[PATH_MAX]; nsurl *url = NULL; - if (filepath_sfind(respaths, buf, path) == NSERROR_OK) + if (filepath_sfind(respaths, buf, PATH_MAX, path) == NSERROR_OK) netsurf_path_to_nsurl(buf, &url); return url; diff --git a/frontends/monkey/main.c b/frontends/monkey/main.c index 2d6c1bd1b..35af25235 100644 --- a/frontends/monkey/main.c +++ b/frontends/monkey/main.c @@ -452,7 +452,7 @@ main(int argc, char **argv) die("NetSurf failed to initialise"); } - filepath_sfinddef(respaths, buf, "mime.types", "/etc/"); + filepath_sfinddef(respaths, buf, PATH_MAX, "mime.types", "/etc/"); monkey_fetch_filetype_init(buf); urldb_load(nsoption_charp(url_file)); diff --git a/frontends/windows/fetch.c b/frontends/windows/fetch.c index cad99446b..bb5290e59 100644 --- a/frontends/windows/fetch.c +++ b/frontends/windows/fetch.c @@ -79,7 +79,7 @@ static nsurl *nsw32_get_resource_url(const char *path) char buf[PATH_MAX]; nsurl *url = NULL; - if (filepath_sfind(G_resource_pathv, buf, path) == NSERROR_OK) + if (filepath_sfind(G_resource_pathv, buf, PATH_MAX, path) == NSERROR_OK) netsurf_path_to_nsurl(buf, &url); return url; diff --git a/utils/filepath.c b/utils/filepath.c index 5cfcba548..82568e298 100644 --- a/utils/filepath.c +++ b/utils/filepath.c @@ -33,7 +33,6 @@ #include <unistd.h> #include <string.h> -#include "utils/dirent.h" /** \todo why is this necessary for atari to get PATH_MAX and is there a better way */ #include "utils/utils.h" #include "utils/config.h" #include "utils/filepath.h" @@ -41,119 +40,121 @@ /** maximum number of elements in the resource vector */ #define MAX_RESPATH 128 -/* exported interface documented in filepath.h */ -nserror filepath_vsfindfile(char *str, const char *format, va_list ap) +static nserror filepath_vprintpath(const char *format, va_list ap, char **retstr) { - char *realpathname; - char *pathname; - int len; + char *pathname = NULL; + int len = 0; - pathname = malloc(PATH_MAX); - if (pathname == NULL) - return NSERROR_NOMEM; /* unable to allocate memory */ + do { + len = vsnprintf(pathname, len, format, ap); + if (len <= 0) + return NSERROR_INVALID; - len = vsnprintf(pathname, PATH_MAX, format, ap); + if (pathname != NULL) break; - if ((len < 0) || (len >= PATH_MAX)) { - /* error or output exceeded PATH_MAX length so - * operation is doomed to fail. - */ - free(pathname); - NSERROR_BAD_SIZE; - } + pathname = malloc(len + 1); + } while (pathname != NULL); - realpathname = realpath(pathname, str); + if (pathname == NULL) + return NSERROR_NOMEM; /* unable to allocate memory */ - free(pathname); + *retstr = realpath(pathname, NULL); - if (realpathname != NULL) { - /* sucessfully expanded pathname */ - if (access(realpathname, R_OK) != 0) { - /* unable to read the file */ - return NSERROR_NOT_FOUND; - } - } + if (*retstr == NULL) + *retstr = pathname; + else + free(pathname); return NSERROR_OK; } - -/* exported interface documented in filepath.h */ -nserror filepath_sfindfile(char *str, const char *format, ...) -{ - va_list ap; +static nserror filepath_printpath(char **retstr, const char *format, ...) { nserror ret; + va_list ap; va_start(ap, format); - ret = filepath_vsfindfile(str, format, ap); + ret = filepath_vprintpath(format, ap, retstr); va_end(ap); return ret; } - /* exported interface documented in filepath.h */ nserror filepath_findfile(char **retstr, const char *format, ...) { nserror ret; va_list ap; - *retstr = malloc(PATH_MAX); - if (*retstr == NULL) - return NSERROR_NOMEM; /* unable to allocate memory */ - va_start(ap, format); - ret = filepath_vsfindfile(*retstr, format, ap); + ret = filepath_vprintpath(format, ap, retstr); va_end(ap); - if (ret != NSERROR_OK) { - free(*retstr); - *retstr = NULL; + if (ret == NSERROR_OK) { + if (access(*retstr, F_OK) != 0) { + /* File not found */ + free(*retstr); + *retstr = NULL; + + return NSERROR_NOT_FOUND; + } + + if (access(*retstr, R_OK) != 0) { + /* unable to read the file */ + free(*retstr); + *retstr = NULL; + + return NSERROR_PERMISSION; + } } return ret; } /* exported interface documented in filepath.h */ -nserror filepath_sfind(char **respathv, char *filepath, const char *filename) +nserror filepath_sfind(char **respathv, char *filepath, size_t size, const char *filename) { - int respathc = 0; + nserror ret; + char *allocatedfilepath; - if ((respathv == NULL) || (respathv[0] == NULL) || (filepath == NULL)) + if (filepath == NULL || size == 0) return NSERROR_INVALID; - while (respathv[respathc] != NULL) { - if (filepath_sfindfile(filepath, "%s/%s", respathv[respathc], filename) == NSERROR_OK) { - return NSERROR_OK; - } + ret = filepath_find(respathv, filename, &allocatedfilepath); + if (ret == NSERROR_OK) { + if (strlen(allocatedfilepath) >= size) + ret = NSERROR_NOSPACE; - respathc++; + strncpy(filepath, allocatedfilepath, size); + filepath[size - 1] = '\0'; + + free(allocatedfilepath); } - return NSERROR_NOT_FOUND; + return ret; } /* exported interface documented in filepath.h */ nserror filepath_find(char **respathv, const char *filename, char **retstr) { - nserror ret; + int respathc = 0; if ((respathv == NULL) || (respathv[0] == NULL)) return NSERROR_INVALID; - *retstr = malloc(PATH_MAX); - if (*retstr == NULL) - return NSERROR_NOMEM; + while (respathv[respathc] != NULL) { + nserror ret; - ret = filepath_sfind(respathv, *retstr, filename); + ret = filepath_findfile(retstr, "%s/%s", respathv[respathc], filename); + if (ret == NSERROR_OK) + return NSERROR_OK; + if (ret != NSERROR_NOT_FOUND) + return ret; - if (ret == NSERROR_OK) { - free(*retstr); - *retstr = NULL; + respathc++; } - return ret; + return NSERROR_NOT_FOUND; } @@ -161,29 +162,37 @@ nserror filepath_find(char **respathv, const char *filename, char **retstr) nserror filepath_sfinddef(char **respathv, char *filepath, + size_t size, const char *filename, const char *def) { - char t[PATH_MAX]; + char *allocatedfilepath; nserror ret; if ((respathv == NULL) || (respathv[0] == NULL) || (filepath == NULL)) return NSERROR_INVALID; - ret = filepath_sfind(respathv, filepath, filename); + ret = filepath_find(respathv, filename, &allocatedfilepath); if ((ret != NSERROR_OK) && (def != NULL)) { /* search failed, return the path specified */ - ret = NSERROR_OK; if (def[0] == '~') { - snprintf(t, PATH_MAX, "%s/%s/%s", getenv("HOME"), def + 1, filename); + ret = filepath_printpath(&allocatedfilepath, "%s/%s/%s", getenv("HOME"), def + 1, filename); } else { - snprintf(t, PATH_MAX, "%s/%s", def, filename); - } - if (realpath(t, filepath) == NULL) { - strncpy(filepath, t, PATH_MAX); + ret = filepath_printpath(&allocatedfilepath, "%s/%s", def, filename); } } + + if (ret == NSERROR_OK) { + if (strlen(allocatedfilepath) >= size) + ret = NSERROR_NOSPACE; + + strncpy(filepath, allocatedfilepath, size); + filepath[size - 1] = '\0'; + + free(allocatedfilepath); + } + return ret; } diff --git a/utils/filepath.h b/utils/filepath.h index e87437d90..c44f5af7e 100644 --- a/utils/filepath.h +++ b/utils/filepath.h @@ -32,32 +32,13 @@ * Create a normalised file name. * * If the file described by the format exists and is accessible the - * normalised path is placed in str and a pointer to str returned - * otherwise NULL is returned. The string in str is always modified. + * normalised path is placed in retstr. The returned string is allocated + * and it is the responsibility of the caller to free it via free(). * - * @param str A buffer to contain the normalised file name must be at - * least PATH_MAX bytes long. + * @param retstr A pointer the the allocated result string. * @param format A printf format for the filename. - * @param ap The list of arguments for the format. * @return NSERROR_OK on success or error code on failure. */ -nserror filepath_vsfindfile(char *str, const char *format, va_list ap); - - -/** - * Create a normalised file name. - * - * Similar to vsfindfile but takes variadic (printf like) parameters - */ -nserror filepath_sfindfile(char *str, const char *format, ...); - - -/** - * Create a normalised file name. - * - * Similar to sfindfile but allocates its own storage for the - * returned string in retstr. The caller must free this storage. - */ nserror filepath_findfile(char **retstr, const char *format, ...); @@ -65,22 +46,21 @@ nserror filepath_findfile(char **retstr, const char *format, ...); * Searches an array of resource paths for a file. * * Iterates through a vector of resource paths and returns the - * normalised file name of the first acessible file or NULL if no file - * can be found in any of the resource paths. + * normalised file name of the first acessible file. * * @param respathv The resource path vector to iterate. * @param filepath The buffer to place the result in. * @param filename The filename of the resource to search for. * @return NSERROR_OK on success or error code on failure. */ -nserror filepath_sfind(char **respathv, char *filepath, const char *filename); +nserror filepath_sfind(char **respathv, char *filepath, size_t size, const char *filename); /** * Searches an array of resource paths for a file. * * Similar to filepath_sfind except it allocates its own storage for - * the returned string. The caller must free this sorage. + * the returned string. The caller must free this storage. */ nserror filepath_find(char **respathv, const char *filename, char **retstr); @@ -94,13 +74,14 @@ nserror filepath_find(char **respathv, const char *filename, char **retstr); * path and the filename. * * @param respathv The resource path vector to iterate. - * @param filepath The buffer to place the result in. Must have space for PATH_MAX bytes. + * @param filepath The buffer to place the result in. + * @param size The size of the filepath buffer. * @param filename The filename of the resource to search for. * @param def The default path to use * @return NSERROR_OK on success or error code on failure. */ -nserror filepath_sfinddef(char **respathv, char *filepath, const char *filename, - const char *def); +nserror filepath_sfinddef(char **respathv, char *filepath, size_t size, + const char *filename, const char *def); /** -- 2.45.2