/*************************************************************************** * __________ __ ___. * Open \______ \ ____ ____ | | _\_ |__ _______ ___ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ / * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < < * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \ * \/ \/ \/ \/ \/ * $Id$ * * Copyright (C) 2010 Thomas Martitz * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY * KIND, either express or implied. * ****************************************************************************/ #include /* snprintf */ #include #include #include "config.h" #include "rbpaths.h" #include "file.h" /* MAX_PATH */ #include "logf.h" #include "gcc_extensions.h" #include "string-extra.h" #include "filefuncs.h" #undef open #undef creat #undef remove #undef rename #undef opendir #undef mkdir #undef rmdir #if (CONFIG_PLATFORM & PLATFORM_ANDROID) || defined(SAMSUNG_YPR0) && !defined(__PCTOOL__) #include "dir-target.h" #define opendir _opendir #define mkdir _mkdir #if (CONFIG_PLATFORM & PLATFORM_ANDROID) static const char rbhome[] = "/sdcard"; #endif #elif (CONFIG_PLATFORM & (PLATFORM_SDL|PLATFORM_MAEMO|PLATFORM_PANDORA)) && !defined(__PCTOOL__) #define open sim_open #define remove sim_remove #define rename sim_rename #define opendir sim_opendir #define mkdir sim_mkdir #define rmdir sim_rmdir extern int sim_open(const char* name, int o, ...); extern int sim_remove(const char* name); extern int sim_rename(const char* old, const char* new); extern DIR* sim_opendir(const char* name); extern int sim_mkdir(const char* name); extern int sim_rmdir(const char* name); const char *rbhome; #endif #if !defined(SAMSUNG_YPR0) && !defined(__PCTOOL__) /* flags for get_user_file_path() */ /* whether you need write access to that file/dir, especially true * for runtime generated files (config.cfg) */ #define NEED_WRITE (1<<0) /* file or directory? */ #define IS_FILE (1<<1) void paths_init(void) { /* make sure $HOME/.config/rockbox.org exists, it's needed for config.cfg */ #if (CONFIG_PLATFORM & PLATFORM_ANDROID) mkdir("/sdcard/rockbox"); mkdir("/sdcard/rockbox/rocks.data"); #else char config_dir[MAX_PATH]; const char *home = getenv("RBROOT"); if (!home) { home = getenv("HOME"); } if (!home) { logf("HOME environment var not set. Can't write config"); return; } rbhome = home; snprintf(config_dir, sizeof(config_dir), "%s/.config", home); mkdir(config_dir); snprintf(config_dir, sizeof(config_dir), "%s/.config/rockbox.org", home); mkdir(config_dir); /* Plugin data directory */ snprintf(config_dir, sizeof(config_dir), "%s/.config/rockbox.org/rocks.data", home); mkdir(config_dir); #endif } static bool try_path(const char* filename, unsigned flags) { if (flags & IS_FILE) { if (file_exists(filename)) return true; } else { if (dir_exists(filename)) return true; } return false; } static const char* _get_user_file_path(const char *path, unsigned flags, char* buf, const size_t bufsize) { const char *ret = path; const char *pos = path; /* replace ROCKBOX_DIR in path with $HOME/.config/rockbox.org */ pos += ROCKBOX_DIR_LEN; if (*pos == '/') pos += 1; #if (CONFIG_PLATFORM & PLATFORM_ANDROID) if (snprintf(buf, bufsize, "/sdcard/rockbox/%s", pos) #else if (snprintf(buf, bufsize, "%s/.config/rockbox.org/%s", rbhome, pos) #endif >= (int)bufsize) return NULL; /* always return the replacement buffer (pointing to $HOME) if * write access is needed */ if (flags & NEED_WRITE) ret = buf; else if (try_path(buf, flags)) ret = buf; if (ret != buf) /* not found in $HOME, try ROCKBOX_BASE_DIR, !NEED_WRITE only */ { if (snprintf(buf, bufsize, ROCKBOX_SHARE_PATH "/%s", pos) >= (int)bufsize) return NULL; if (try_path(buf, flags)) ret = buf; } return ret; } static const char* handle_special_dirs(const char* dir, unsigned flags, char *buf, const size_t bufsize) { if (!strncmp(HOME_DIR, dir, HOME_DIR_LEN)) { const char *p = dir + HOME_DIR_LEN; while (*p == '/') p++; snprintf(buf, bufsize, "%s/%s", rbhome, p); return buf; } else if (!strncmp(ROCKBOX_DIR, dir, ROCKBOX_DIR_LEN)) return _get_user_file_path(dir, flags, buf, bufsize); return dir; } int app_open(const char *name, int o, ...) { char realpath[MAX_PATH]; va_list ap; int fd; int flags = IS_FILE; if (o & (O_CREAT|O_RDWR|O_TRUNC|O_WRONLY)) flags |= NEED_WRITE; name = handle_special_dirs(name, flags, realpath, sizeof(realpath)); va_start(ap, o); fd = open(name, o, va_arg(ap, unsigned int)); va_end(ap); return fd; } int app_creat(const char* name, mode_t mode) { return app_open(name, O_CREAT|O_WRONLY|O_TRUNC, mode); } int app_remove(const char *name) { char realpath[MAX_PATH]; const char *fname = handle_special_dirs(name, NEED_WRITE, realpath, sizeof(realpath)); return remove(fname); } int app_rename(const char *old, const char *new) { char realpath_old[MAX_PATH], realpath_new[MAX_PATH]; const char *final_old, *final_new; final_old = handle_special_dirs(old, NEED_WRITE, realpath_old, sizeof(realpath_old)); final_new = handle_special_dirs(new, NEED_WRITE, realpath_new, sizeof(realpath_new)); return rename(final_old, final_new); } DIR *app_opendir(const char *name) { char realpath[MAX_PATH]; const char *fname = handle_special_dirs(name, 0, realpath, sizeof(realpath)); return opendir(fname); } int app_mkdir(const char* name) { char realpath[MAX_PATH]; const char *fname = handle_special_dirs(name, NEED_WRITE, realpath, sizeof(realpath)); return mkdir(fname); } int app_rmdir(const char* name) { char realpath[MAX_PATH]; const char *fname = handle_special_dirs(name, NEED_WRITE, realpath, sizeof(realpath)); return rmdir(fname); } #else int app_open(const char *name, int o, ...) { if (o & O_CREAT) { int ret; va_list ap; va_start(ap, o); ret = open(name, o, va_arg(ap, mode_t)); va_end(ap); return ret; } return open(name, o); } int app_creat(const char* name, mode_t mode) { return creat(name, mode); } int app_remove(const char *name) { return remove(name); } int app_rename(const char *old, const char *new) { return rename(old,new); } DIR *app_opendir(const char *name) { return (DIR*)opendir(name); } /* cast to remove warning in checkwps */ int app_mkdir(const char* name) { return mkdir(name); } int app_rmdir(const char* name) { return rmdir(name); } #endif