#include "file_io.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "osd.h" #include "fpga_io.h" #include "menu.h" #include "errno.h" #include "DiskImage.h" #include "user_io.h" #include "cfg.h" #include "input.h" #include "scheduler.h" typedef std::vector DirentVector; static const size_t YieldIterations = 128; DirentVector DirItem; int iSelectedEntry = 0; // selected entry index int iFirstEntry = 0; static char full_path[1200]; void FileClose(fileTYPE *file) { if (file->fd > 0) { //printf("closing %d\n", file->fd); close(file->fd); if (file->type == 1) { if (file->name[0] == '/') { shm_unlink(file->name); } file->type = 0; } } file->fd = -1; } int FileOpenEx(fileTYPE *file, const char *name, int mode, char mute) { const char *root = getRootDir(); if (strncasecmp(getRootDir(), name, strlen(root))) { sprintf(full_path, "%s/%s", (mode == -1) ? "" : root, name); } else { sprintf(full_path, name); } FileClose(file); file->mode = 0; file->type = 0; char *p = strrchr(full_path, '/'); strcpy(file->name, (mode == -1) ? full_path : p+1); file->fd = (mode == -1) ? shm_open("/vtrd", O_CREAT | O_RDWR | O_TRUNC, 0777) : open(full_path, mode, 0777); if (file->fd <= 0) { if(!mute) printf("FileOpenEx(open) File:%s, error: %d.\n", full_path, file->fd); file->fd = -1; return 0; } if (mode == -1) { file->type = 1; file->size = 0; file->offset = 0; file->mode = O_CREAT | O_RDWR | O_TRUNC; } else { struct stat64 st; int ret = fstat64(file->fd, &st); if (ret < 0) { if (!mute) printf("FileOpenEx(fstat) File:%s, error: %d.\n", full_path, ret); FileClose(file); return 0; } file->size = st.st_size; file->offset = 0; file->mode = mode; } //printf("opened %s, size %lu\n", full_path, file->size); return 1; } __off64_t FileGetSize(fileTYPE *file) { if (file->fd <= 0) return 0; struct stat64 st; int ret = fstat64(file->fd, &st); return (ret < 0) ? 0 : st.st_size; } int FileOpen(fileTYPE *file, const char *name, char mute) { return FileOpenEx(file, name, O_RDONLY, mute); } int FileNextSector(fileTYPE *file) { __off64_t newoff = lseek64(file->fd, file->offset + 512, SEEK_SET); if (newoff != file->offset + 512) { //printf("Fail to seek to next sector. File: %s.\n", file->name); lseek64(file->fd, file->offset, SEEK_SET); return 0; } file->offset = newoff; return 1; } int FileSeek(fileTYPE *file, __off64_t offset, int origin) { __off64_t newoff = lseek64(file->fd, offset, origin); if(newoff<0) { printf("Fail to seek the file.\n"); return 0; } file->offset = newoff; return 1; } int FileSeekLBA(fileTYPE *file, uint32_t offset) { __off64_t off64 = offset; off64 <<= 9; return FileSeek(file, off64, SEEK_SET); } // Read. MiST compatible. Avoid to use it. int FileRead(fileTYPE *file, void *pBuffer) { return FileReadEx(file, pBuffer, 1); } int FileReadEx(fileTYPE *file, void *pBuffer, int nSize) { static uint8_t tmpbuff[512]; if (!FileSeek(file, file->offset, SEEK_SET)) { printf("FileRead error(seek).\n"); return 0; } if (!pBuffer) { for (int i = 0; i < nSize; i++) { int ret = read(file->fd, tmpbuff, 512); if (ret < 0) { printf("FileRead error(%d).\n", ret); return 0; } EnableDMode(); spi_block_write(tmpbuff, 0); DisableDMode(); } } else { int ret = read(file->fd, pBuffer, nSize * 512); if (ret < 0) { printf("FileRead error(%d).\n", ret); return 0; } } return 1; } // Write. MiST compatible. Avoid to use it. int FileWrite(fileTYPE *file, void *pBuffer) { if (!FileSeek(file, file->offset, SEEK_SET)) { printf("FileWrite error(seek).\n"); return 0; } int ret = write(file->fd, pBuffer, 512); if (ret < 0) { printf("FileWrite error(%d).\n", ret); return 0; } return 1; } // Read with offset advancing int FileReadAdv(fileTYPE *file, void *pBuffer, int length) { ssize_t ret = read(file->fd, pBuffer, length); if (ret < 0) { printf("FileReadAdv error(%d).\n", ret); return 0; } file->offset += ret; return ret; } int FileReadSec(fileTYPE *file, void *pBuffer) { return FileReadAdv(file, pBuffer, 512); } // Write with offset advancing int FileWriteAdv(fileTYPE *file, void *pBuffer, int length) { int ret = write(file->fd, pBuffer, length); if (ret < 0) { printf("FileWriteAdv error(%d).\n", ret); return 0; } file->offset += ret; return ret; } int FileWriteSec(fileTYPE *file, void *pBuffer) { return FileWriteAdv(file, pBuffer, 512); } int FileSave(const char *name, void *pBuffer, int size) { sprintf(full_path, "%s/%s", getRootDir(), name); int fd = open(full_path, O_WRONLY | O_CREAT | O_TRUNC | O_SYNC, S_IRWXU | S_IRWXG | S_IRWXO); if (fd < 0) { printf("FileSave(open) File:%s, error: %d.\n", full_path, fd); return 0; } int ret = write(fd, pBuffer, size); close(fd); if (ret < 0) { printf("FileSave(write) File:%s, error: %d.\n", full_path, ret); return 0; } return ret; } int FileSaveConfig(const char *name, void *pBuffer, int size) { char path[256] = { CONFIG_DIR"/" }; strcat(path, name); return FileSave(path, pBuffer, size); } int FileLoad(const char *name, void *pBuffer, int size) { sprintf(full_path, "%s/%s", getRootDir(), name); int fd = open(full_path, O_RDONLY); if (fd < 0) { printf("FileLoad(open) File:%s, error: %d.\n", full_path, fd); return 0; } struct stat64 st; int ret = fstat64(fd, &st); if (ret < 0) { printf("FileLoad(fstat) File:%s, error: %d.\n", full_path, ret); close(fd); return 0; } if (!pBuffer) { close(fd); return (int)st.st_size; } ret = read(fd, pBuffer, size ? size : st.st_size); close(fd); if (ret < 0) { printf("FileLoad(read) File:%s, error: %d.\n", full_path, ret); return 0; } return ret; } int FileLoadConfig(const char *name, void *pBuffer, int size) { char path[256] = { CONFIG_DIR"/" }; strcat(path, name); return FileLoad(path, pBuffer, size); } int FileCanWrite(const char *name) { sprintf(full_path, "%s/%s", getRootDir(), name); struct stat64 st; int ret = stat64(full_path, &st); if (ret < 0) { printf("FileCanWrite(stat) File:%s, error: %d.\n", full_path, ret); return 0; } //printf("FileCanWrite: mode=%04o.\n", st.st_mode); return ((st.st_mode & S_IWUSR) != 0); } uint32_t getFileType(const char *name) { sprintf(full_path, "%s/%s", getRootDir(), name); struct stat64 st; if (stat64(full_path, &st)) return 0; return st.st_mode; } static int device = 0; static int usbnum = 0; const char *getStorageDir(int dev) { static char path[32]; if (!dev) return "/media/fat"; sprintf(path, "/media/usb%d", usbnum); return path; } const char *getRootDir() { return getStorageDir(device); } const char *getFullPath(const char *name) { sprintf(full_path, "%s/%s", getRootDir(), name); return full_path; } void setStorage(int dev) { device = 0; FileSave(CONFIG_DIR"/device.bin", &dev, sizeof(int)); fpga_load_rbf("menu.rbf"); } static int orig_device = 0; int getStorage(int from_setting) { return from_setting ? orig_device : device; } int isPathMounted(int n) { char path[32]; sprintf(path, "/media/usb%d", n); struct stat file_stat; struct stat parent_stat; if (-1 == stat(path, &file_stat)) { printf("failed to stat %s\n", path); return 0; } if (!(file_stat.st_mode & S_IFDIR)) { printf("%s is not a directory.\n", path); return 0; } if (-1 == stat("/media", &parent_stat)) { printf("failed to stat /media\n"); return 0; } if (file_stat.st_dev != parent_stat.st_dev || (file_stat.st_dev == parent_stat.st_dev && file_stat.st_ino == parent_stat.st_ino)) { printf("%s IS a mountpoint.\n", path); struct statfs fs_stat; if (!statfs(path, &fs_stat)) { printf("%s is FS: 0x%08X\n", path, fs_stat.f_type); if (fs_stat.f_type != EXT4_SUPER_MAGIC) { printf("%s is not EXT2/3/4.\n", path); return 1; } } } printf("%s is NOT a VFAT mountpoint.\n", path); return 0; } int isUSBMounted() { for (int i = 0; i < 4; i++) { if (isPathMounted(i)) { usbnum = i; return 1; } } return 0; } void FindStorage(void) { char str[128]; printf("Looking for root device...\n"); device = 0; FileLoad(CONFIG_DIR"/device.bin", &device, sizeof(int)); orig_device = device; if(device && !isUSBMounted()) { int saveddev = device; device = 0; MiSTer_ini_parse(); device = saveddev; parse_video_mode(); user_io_send_buttons(1); printf("Waiting for USB...\n"); int btn = 0; int done = 0; for (int i = 30; i >= 0; i--) { sprintf(str, "\n Waiting for USB...\n\n %d \n\n\n OSD/USER or ESC to cancel", i); InfoMessage(str); if (isUSBMounted()) { done = 1; break; } for (int i = 0; i < 10; i++) { btn = fpga_get_buttons(); if (!btn) btn = input_poll(1); if (btn) { printf("Button has been pressed %d\n", btn); InfoMessage("\n\n Canceled!\n"); usleep(500000); setStorage(0); break; } usleep(100000); } if (done) break; } if (!done) { InfoMessage("\n\n No USB storage found\n Falling back to SD card\n"); usleep(2000000); setStorage(0); } } if (device) { printf("Using USB as a root device\n"); } else { printf("Using SD card as a root device\n"); } sprintf(full_path, "%s/" CONFIG_DIR, getRootDir()); DIR* dir = opendir(full_path); if (dir) closedir(dir); else if (ENOENT == errno) mkdir(full_path, S_IRWXU | S_IRWXG | S_IRWXO); } struct DirentComp { bool operator()(const dirent& de1, const dirent& de2) { if (++iterations % YieldIterations == 0) { scheduler_yield(); } if ((de1.d_type == DT_DIR) && !strcmp(de1.d_name, "..")) return true; if ((de2.d_type == DT_DIR) && !strcmp(de2.d_name, "..")) return false; if ((de1.d_type == DT_DIR) && (de2.d_type == DT_REG)) return true; if ((de1.d_type == DT_REG) && (de2.d_type == DT_DIR)) return false; if ((de1.d_type == DT_REG) && (de2.d_type == DT_REG)) { int len1 = strlen(de1.d_name); int len2 = strlen(de2.d_name); if ((len1 > 4) && (de1.d_name[len1 - 4] == '.')) len1 -= 4; if ((len2 > 4) && (de2.d_name[len2 - 4] == '.')) len2 -= 4; int len = (len1 < len2) ? len1 : len2; int ret = strncasecmp(de1.d_name, de2.d_name, len); if (!ret) { return len1 < len2; } return ret < 0; } return strcasecmp(de1.d_name, de2.d_name) < 0; } size_t iterations = 0; }; static int get_stmode(const char *path) { sprintf(full_path, "%s/%s", getRootDir(), path); struct stat64 st; return (stat64(full_path, &st) < 0) ? 0 : st.st_mode; } void AdjustDirectory(char *path) { int stmode = get_stmode(path); if (!stmode) { printf("AdjustDirectory(stat) path:%s, error.\n", path); path[0] = 0; return; } if (stmode & S_IFDIR) return; char *p = strrchr(path, '/'); if (p) { *p = 0; } else { path[0] = 0; } } int ScanDirectory(char* path, int mode, const char *extension, int options, const char *prefix) { static char file_name[1024]; int has_trd = 0; const char *ext = extension; while (*ext) { if (!strncasecmp(ext, "TRD", 3)) has_trd = 1; ext += 3; } int extlen = strlen(extension); //printf("scan dir\n"); if (mode == SCANF_INIT) { file_name[0] = 0; int stmode = get_stmode(path); if (!(stmode & S_IFDIR)) { char *p = strrchr(path, '/'); if (p) { strcpy(file_name, p + 1); *p = 0; } else { strcpy(file_name, path); path[0] = 0; } if (!(stmode & S_IFREG)) file_name[0] = 0; } if (!(get_stmode(path) & S_IFDIR)) { path[0] = 0; file_name[0] = 0; } sprintf(full_path, "%s/%s", getRootDir(), path); printf("Start to scan dir: %s\n", full_path); iFirstEntry = 0; iSelectedEntry = 0; DirItem.clear(); DIR *d = opendir(full_path); if (!d) { printf("Couldn't open dir: %s\n", full_path); return 0; } struct dirent *de; for (size_t i = 0; (de = readdir(d)); i++) { if (0 < i && i % YieldIterations == 0) { scheduler_yield(); } if (de->d_type == DT_DIR) { if (!strcmp(de->d_name, ".")) continue; if (!strcmp(de->d_name, "..")) { if(!strlen(path)) continue; } if (!(options & SCANO_DIR)) { if (de->d_name[0] != '_' && strcmp(de->d_name, "..")) continue; if (!(options & SCANO_CORES)) continue; } } else if (de->d_type == DT_REG) { //skip non-selectable files if (!strcasecmp(de->d_name, "menu.rbf")) continue; if (!strcasecmp(de->d_name, "boot.rom")) continue; //check the prefix if given if (prefix && strncasecmp(prefix, de->d_name, strlen(prefix))) continue; if (extlen > 0) { int len = strlen(de->d_name); const char *ext = extension; int found = (has_trd && x2trd_ext_supp(de->d_name)); if (!found && is_minimig() && !memcmp(extension, "HDF", 3)) { found = !strcasecmp(de->d_name + strlen(de->d_name) - 4, ".iso"); } while(!found && *ext) { char e[5]; memcpy(e+1, ext, 3); if (e[3] == 0x20) { e[3] = 0; if (e[2] == 0x20) { e[2] = 0; } } e[0] = '.'; e[4] = 0; int l = strlen(e); if (len > l) { char *p = de->d_name + len - l; found = 1; for (int i = 0; i < l; i++) { if (e[i] == '?') continue; if (tolower(e[i]) != tolower(p[i])) found = 0; } } if (found) break; if (strlen(ext) < 3) break; ext += 3; } if (!found) continue; } } else { continue; } DirItem.push_back(*de); } closedir(d); printf("Got %d dir entries\n", flist_nDirEntries()); if (!flist_nDirEntries()) return 0; std::sort(DirItem.begin(), DirItem.end(), DirentComp()); if (file_name[0]) { for (int i = 0; i < flist_nDirEntries(); i++) { if (!strcmp(file_name, DirItem[i].d_name)) { iSelectedEntry = i; if (iSelectedEntry + (OsdGetSize() / 2) - 1 >= flist_nDirEntries()) iFirstEntry = flist_nDirEntries() - OsdGetSize(); else iFirstEntry = iSelectedEntry - (OsdGetSize() / 2) + 1; if (iFirstEntry < 0) iFirstEntry = 0; break; } } } return flist_nDirEntries(); } else { if (flist_nDirEntries() == 0) // directory is empty so there is no point in searching for any entry return 0; if (mode == SCANF_NEXT) { if(iSelectedEntry + 1 < flist_nDirEntries()) // scroll within visible items { iSelectedEntry++; if (iSelectedEntry > iFirstEntry + OsdGetSize() - 1) iFirstEntry = iSelectedEntry - OsdGetSize() + 1; } return 0; } else if (mode == SCANF_PREV) { if (iSelectedEntry > 0) // scroll within visible items { iSelectedEntry--; if (iSelectedEntry < iFirstEntry) iFirstEntry = iSelectedEntry; } return 0; } else if (mode == SCANF_NEXT_PAGE) { if (iSelectedEntry < iFirstEntry + OsdGetSize() - 1) { iSelectedEntry = iFirstEntry + OsdGetSize() - 1; if (iSelectedEntry >= flist_nDirEntries()) iSelectedEntry = flist_nDirEntries() - 1; } else { iSelectedEntry += OsdGetSize(); iFirstEntry += OsdGetSize(); if (iSelectedEntry >= flist_nDirEntries()) { iSelectedEntry = flist_nDirEntries() - 1; iFirstEntry = iSelectedEntry - OsdGetSize() + 1; if (iFirstEntry < 0) iFirstEntry = 0; } else if (iFirstEntry + OsdGetSize() > flist_nDirEntries()) { iFirstEntry = flist_nDirEntries() - OsdGetSize(); } } return 0; } else if (mode == SCANF_PREV_PAGE) { if(iSelectedEntry != iFirstEntry) { iSelectedEntry = iFirstEntry; } else { iFirstEntry -= OsdGetSize(); if (iFirstEntry < 0) iFirstEntry = 0; iSelectedEntry = iFirstEntry; } } else if (mode == SCANF_SET_ITEM) { for (int i = 0; i < flist_nDirEntries(); i++) { if((DirItem[i].d_type == DT_DIR) && !strcmp(DirItem[i].d_name, extension)) { iSelectedEntry = i; if (iSelectedEntry + (OsdGetSize() / 2) - 1 >= flist_nDirEntries()) iFirstEntry = flist_nDirEntries() - OsdGetSize(); else iFirstEntry = iSelectedEntry - (OsdGetSize() / 2) + 1; if (iFirstEntry < 0) iFirstEntry = 0; break; } } } else { //printf("dir scan for key: %x/%c\n", mode, mode); mode = toupper(mode); if ((mode >= '0' && mode <= '9') || (mode >= 'A' && mode <= 'Z')) { int found = -1; for (int i = iSelectedEntry+1; i < flist_nDirEntries(); i++) { if (toupper(DirItem[i].d_name[0]) == mode) { found = i; break; } } if (found < 0) { for (int i = 0; i < flist_nDirEntries(); i++) { if (toupper(DirItem[i].d_name[0]) == mode) { found = i; break; } } } if (found >= 0) { iSelectedEntry = found; if (iSelectedEntry + (OsdGetSize() / 2) >= flist_nDirEntries()) iFirstEntry = flist_nDirEntries() - OsdGetSize(); else iFirstEntry = iSelectedEntry - (OsdGetSize()/2) + 1; if (iFirstEntry < 0) iFirstEntry = 0; } } } } return 0; } int flist_nDirEntries() { return DirItem.size(); } int flist_iFirstEntry() { return iFirstEntry; } int flist_iSelectedEntry() { return iSelectedEntry; } dirent* flist_DirItem(int n) { return &DirItem[n]; } dirent* flist_SelectedItem() { return &DirItem[iSelectedEntry]; }