A modern Music Player Daemon based on Rockbox open source high quality audio player
libadwaita audio rust zig deno mpris rockbox mpd
at master 206 lines 5.3 kB view raw
1/*************************************************************************** 2 * __________ __ ___. 3 * Open \______ \ ____ ____ | | _\_ |__ _______ ___ 4 * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ / 5 * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < < 6 * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \ 7 * \/ \/ \/ \/ \/ 8 * $Id$ 9 * 10 * Copyright (C) 2002 Gilles Roux 11 * 2003 Garrett Derner 12 * 2010 Yoshihisa Uchida 13 * 14 * This program is free software; you can redistribute it and/or 15 * modify it under the terms of the GNU General Public License 16 * as published by the Free Software Foundation; either version 2 17 * of the License, or (at your option) any later version. 18 * 19 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY 20 * KIND, either express or implied. 21 * 22 ****************************************************************************/ 23#include "plugin.h" 24#include "tv_preferences.h" 25#include "tv_reader.h" 26 27#if PLUGIN_BUFFER_SIZE < 0x10000 28#define TV_MIN_BLOCK_SIZE 0x800 29#else 30#define TV_MIN_BLOCK_SIZE 0x1000 31#endif 32 33/* UTF-8 BOM */ 34#define BOM "\xef\xbb\xbf" 35#define BOM_SIZE 3 36 37static int fd = -1; 38 39static off_t file_pos; 40static off_t start_file_pos; 41 42static off_t file_size; 43 44static unsigned char *reader_buffer; 45static ssize_t buffer_size; 46static ssize_t block_size; 47 48static ssize_t buf_pos; 49static ssize_t read_size; 50 51off_t tv_get_file_size(void) 52{ 53 return file_size; 54} 55 56bool tv_is_eof(void) 57{ 58 return (file_pos + buf_pos >= file_size); 59} 60 61off_t tv_get_current_file_pos(void) 62{ 63 return file_pos + buf_pos; 64} 65 66const unsigned char *tv_get_buffer(ssize_t *bufsize) 67{ 68 *bufsize = read_size - buf_pos; 69 return reader_buffer + buf_pos; 70} 71 72static ssize_t tv_read(unsigned char *buf, ssize_t reqsize) 73{ 74 if (buf - reader_buffer + reqsize > buffer_size) 75 reqsize = buffer_size - (buf - reader_buffer); 76 77 return rb->read(fd, buf, reqsize); 78} 79 80void tv_seek(off_t offset, int whence) 81{ 82 ssize_t size; 83 84 switch (whence) 85 { 86 case SEEK_SET: 87 if (offset >= file_pos && offset < file_pos + read_size) 88 { 89 buf_pos = offset - file_pos; 90 return; 91 } 92 file_pos = offset; 93 break; 94 95 case SEEK_CUR: 96 buf_pos += offset; 97 if (buf_pos >= 0 && buf_pos < read_size) 98 { 99 if (buf_pos > block_size) 100 { 101 buf_pos -= block_size; 102 file_pos += block_size; 103 size = read_size - block_size; 104 rb->memcpy(reader_buffer, reader_buffer + block_size, size); 105 read_size = tv_read(reader_buffer + block_size, block_size); 106 if (read_size < 0) 107 read_size = 0; 108 109 read_size += size; 110 } 111 return; 112 } 113 file_pos += buf_pos; 114 break; 115 116 default: 117 return; 118 break; 119 } 120 121 if (whence == SEEK_SET) 122 { 123 if (file_pos < 0) 124 file_pos = 0; 125 else if (file_pos > file_size) 126 file_pos = file_size; 127 128 rb->lseek(fd, file_pos + start_file_pos, SEEK_SET); 129 buf_pos = 0; 130 read_size = tv_read(reader_buffer, buffer_size); 131 } 132} 133 134static int tv_change_preferences(const struct tv_preferences *oldp) 135{ 136 bool change_file = false; 137 138 /* open the new file */ 139 if (oldp == NULL || rb->strcmp(oldp->file_name, preferences->file_name)) 140 { 141 if (fd >= 0) 142 rb->close(fd); 143 144 fd = rb->open(preferences->file_name, O_RDONLY); 145 if (fd < 0) 146 return TV_CALLBACK_ERROR; 147 148 file_size = rb->filesize(fd); 149 change_file = true; 150 } 151 152 /* 153 * When a file is UTF-8 file with BOM, if encoding is UTF-8, 154 * then file size decreases only BOM_SIZE. 155 */ 156 if (change_file || oldp->encoding != preferences->encoding) 157 { 158 int old_start_file_pos = start_file_pos; 159 int delta_start_file_pos; 160 off_t cur_file_pos = file_pos + buf_pos; 161 162 file_pos = 0; 163 buf_pos = 0; 164 read_size = 0; 165 start_file_pos = 0; 166 167 if (preferences->encoding == UTF_8) 168 { 169 unsigned char bom[BOM_SIZE]; 170 171 rb->lseek(fd, 0, SEEK_SET); 172 rb->read(fd, bom, BOM_SIZE); 173 if (rb->memcmp(bom, BOM, BOM_SIZE) == 0) 174 start_file_pos = BOM_SIZE; 175 } 176 177 delta_start_file_pos = old_start_file_pos - start_file_pos; 178 file_size += delta_start_file_pos; 179 tv_seek(cur_file_pos + delta_start_file_pos, SEEK_SET); 180 } 181 182 return TV_CALLBACK_OK; 183} 184 185bool tv_init_reader(unsigned char **buf, size_t *size) 186{ 187 if (*size < 2 * TV_MIN_BLOCK_SIZE) 188 return false; 189 190 block_size = *size / 2; 191 buffer_size = 2 * block_size; 192 reader_buffer = *buf; 193 194 *buf += buffer_size; 195 *size -= buffer_size; 196 197 tv_add_preferences_change_listner(tv_change_preferences); 198 199 return true; 200} 201 202void tv_finalize_reader(void) 203{ 204 if (fd >= 0) 205 rb->close(fd); 206}