text viewer: some fixes related to font.
[kugel-rb.git] / apps / plugins / text_viewer / tv_reader.c
blob6dc66ef5671d794bfa875359892fdc4b36223b9e
1 /***************************************************************************
2 * __________ __ ___.
3 * Open \______ \ ____ ____ | | _\_ |__ _______ ___
4 * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
5 * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
6 * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
7 * \/ \/ \/ \/ \/
8 * $Id$
10 * Copyright (C) 2002 Gilles Roux
11 * 2003 Garrett Derner
12 * 2010 Yoshihisa Uchida
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.
19 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
20 * KIND, either express or implied.
22 ****************************************************************************/
23 #include "plugin.h"
24 #include "tv_preferences.h"
25 #include "tv_reader.h"
27 #if PLUGIN_BUFFER_SIZE < 0x10000
28 #define TV_MIN_BLOCK_SIZE 0x800
29 #else
30 #define TV_MIN_BLOCK_SIZE 0x1000
31 #endif
33 /* UTF-8 BOM */
34 #define BOM "\xef\xbb\xbf"
35 #define BOM_SIZE 3
37 static int fd = -1;
39 static off_t file_pos;
40 static off_t start_file_pos;
42 static off_t file_size;
44 static unsigned char *reader_buffer;
45 static ssize_t buffer_size;
46 static ssize_t block_size;
48 static ssize_t buf_pos;
49 static ssize_t read_size;
51 off_t tv_get_file_size(void)
53 return file_size;
56 bool tv_is_eof(void)
58 return (file_pos + buf_pos >= file_size);
61 off_t tv_get_current_file_pos(void)
63 return file_pos + buf_pos;
66 const unsigned char *tv_get_buffer(ssize_t *bufsize)
68 *bufsize = read_size - buf_pos;
69 return reader_buffer + buf_pos;
72 static ssize_t tv_read(unsigned char *buf, ssize_t reqsize)
74 if (buf - reader_buffer + reqsize > buffer_size)
75 reqsize = buffer_size - (buf - reader_buffer);
77 return rb->read(fd, buf, reqsize);
80 void tv_seek(off_t offset, int whence)
82 ssize_t size;
84 switch (whence)
86 case SEEK_SET:
87 if (offset >= file_pos && offset < file_pos + read_size)
89 buf_pos = offset - file_pos;
90 return;
92 file_pos = offset;
93 break;
95 case SEEK_CUR:
96 buf_pos += offset;
97 if (buf_pos >= 0 && buf_pos < read_size)
99 if (buf_pos > block_size)
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;
109 read_size += size;
111 return;
113 file_pos += buf_pos;
114 whence = SEEK_SET;
115 break;
117 default:
118 return;
119 break;
122 if (whence == SEEK_SET)
124 if (file_pos < 0)
125 file_pos = 0;
126 else if (file_pos > file_size)
127 file_pos = file_size;
129 rb->lseek(fd, file_pos + start_file_pos, SEEK_SET);
130 buf_pos = 0;
131 read_size = tv_read(reader_buffer, buffer_size);
135 static void tv_change_preferences(const struct tv_preferences *oldp)
137 unsigned char bom[BOM_SIZE];
138 const struct tv_preferences *prefs = tv_get_preferences();
139 int cur_start_file_pos = start_file_pos;
140 off_t cur_file_pos = file_pos + buf_pos;
142 file_pos = 0;
143 buf_pos = 0;
144 read_size = 0;
145 start_file_pos = 0;
147 /* open the new file */
148 if (oldp == NULL || rb->strcmp(oldp->file_name, prefs->file_name))
150 if (fd >= 0)
151 rb->close(fd);
153 fd = rb->open(prefs->file_name, O_RDONLY);
154 if (fd < 0)
155 return;
159 * When a file is UTF-8 file with BOM, if prefs.encoding is UTF-8,
160 * then file size decreases only BOM_SIZE.
162 if (prefs->encoding == UTF_8)
164 rb->lseek(fd, 0, SEEK_SET);
165 rb->read(fd, bom, BOM_SIZE);
166 if (rb->memcmp(bom, BOM, BOM_SIZE) == 0)
167 start_file_pos = BOM_SIZE;
170 file_size = rb->filesize(fd) - start_file_pos;
171 tv_seek(cur_file_pos + cur_start_file_pos - start_file_pos, SEEK_SET);
174 bool tv_init_reader(unsigned char *buf, size_t bufsize, size_t *used_size)
176 if (bufsize < 2 * TV_MIN_BLOCK_SIZE)
177 return false;
179 reader_buffer = buf;
180 block_size = bufsize / 2;
181 buffer_size = 2 * block_size;
182 *used_size = buffer_size;
183 tv_add_preferences_change_listner(tv_change_preferences);
184 return true;
187 void tv_finalize_reader(void)
189 if (fd >= 0)
190 rb->close(fd);