summaryrefslogtreecommitdiff
Unidiff
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--core/multimedia/opieplayer/libmad/libmadplugin.cpp42
1 files changed, 21 insertions, 21 deletions
diff --git a/core/multimedia/opieplayer/libmad/libmadplugin.cpp b/core/multimedia/opieplayer/libmad/libmadplugin.cpp
index 9d1ff8c..1989b4a 100644
--- a/core/multimedia/opieplayer/libmad/libmadplugin.cpp
+++ b/core/multimedia/opieplayer/libmad/libmadplugin.cpp
@@ -1,869 +1,869 @@
1/********************************************************************** 1/**********************************************************************
2** Copyright (C) 2001 Trolltech AS. All rights reserved. 2** Copyright (C) 2001 Trolltech AS. All rights reserved.
3** 3**
4** This file is part of Qtopia Environment. 4** This file is part of Qtopia Environment.
5** 5**
6** This file may be distributed and/or modified under the terms of the 6** This file may be distributed and/or modified under the terms of the
7** GNU General Public License version 2 as published by the Free Software 7** GNU General Public License version 2 as published by the Free Software
8** Foundation and appearing in the file LICENSE.GPL included in the 8** Foundation and appearing in the file LICENSE.GPL included in the
9** packaging of this file. 9** packaging of this file.
10** 10**
11** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE 11** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE
12** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. 12** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
13** 13**
14** See http://www.trolltech.com/gpl/ for GPL licensing information. 14** See http://www.trolltech.com/gpl/ for GPL licensing information.
15** 15**
16** Contact info@trolltech.com if any conditions of this licensing are 16** Contact info@trolltech.com if any conditions of this licensing are
17** not clear to you. 17** not clear to you.
18** 18**
19**********************************************************************/ 19**********************************************************************/
20// largly modified by Maximilian Reiss <max.reiss@gmx.de> 20// largly modified by Maximilian Reiss <max.reiss@gmx.de>
21 21
22#include <stdio.h> 22#include <stdio.h>
23#include <stdarg.h> 23#include <stdarg.h>
24#include <stdlib.h> 24#include <stdlib.h>
25#include <sys/types.h> 25#include <sys/types.h>
26#include <sys/stat.h> 26#include <sys/stat.h>
27#include <fcntl.h> 27#include <fcntl.h>
28#include <unistd.h> 28#include <unistd.h>
29#include <string.h> 29#include <string.h>
30#include <ctype.h> 30#include <ctype.h>
31#include <errno.h> 31#include <errno.h>
32#include <time.h> 32#include <time.h>
33#include <locale.h> 33#include <locale.h>
34#include <math.h> 34#include <math.h>
35#include <assert.h> 35#include <assert.h>
36 36
37#include <qapplication.h> 37#include <qapplication.h>
38#include <qmessagebox.h> 38#include <qmessagebox.h>
39#include <qregexp.h> 39#include <qregexp.h>
40 40
41#include <qpe/config.h> 41#include <qpe/config.h>
42 42
43// for network handling 43// for network handling
44#include <netinet/in.h> 44#include <netinet/in.h>
45#include <netdb.h> 45#include <netdb.h>
46#include <linux/limits.h> 46#include <linux/limits.h>
47#include <sys/socket.h> 47#include <sys/socket.h>
48#include <arpa/inet.h> 48#include <arpa/inet.h>
49#include <unistd.h> 49#include <unistd.h>
50 50
51 51
52//#define HAVE_MMAP 52//#define HAVE_MMAP
53 53
54#if defined(HAVE_MMAP) 54#if defined(HAVE_MMAP)
55# include <sys/mman.h> 55# include <sys/mman.h>
56#endif 56#endif
57#include "libmadplugin.h" 57#include "libmadplugin.h"
58 58
59 59
60extern "C" { 60extern "C" {
61#include "mad.h" 61#include "mad.h"
62} 62}
63 63
64 64
65#define MPEG_BUFFER_SIZE 65536 65#define MPEG_BUFFER_SIZE 65536
66//#define MPEG_BUFFER_SIZE 32768 //16384 // 8192 66//#define MPEG_BUFFER_SIZE 32768 //16384 // 8192
67//#define debugMsg(a) qDebug(a) 67//#define debugMsg(a) qDebug(a)
68#define debugMsg(a) 68#define debugMsg(a)
69 69
70 70
71class Input { 71class Input {
72public: 72public:
73 char const *path; 73 char const *path;
74 int fd; 74 int fd;
75#if defined(HAVE_MMAP) 75#if defined(HAVE_MMAP)
76 void *fdm; 76 void *fdm;
77#endif 77#endif
78 unsigned long fileLength; 78 unsigned long fileLength;
79 unsigned char *data; 79 unsigned char *data;
80 unsigned long length; 80 unsigned long length;
81 int eof; 81 int eof;
82}; 82};
83 83
84 84
85class Output { 85class Output {
86public: 86public:
87 mad_fixed_t attenuate; 87 mad_fixed_t attenuate;
88 struct filter *filters; 88 struct filter *filters;
89 unsigned int channels_in; 89 unsigned int channels_in;
90 unsigned int channels_out; 90 unsigned int channels_out;
91 unsigned int speed_in; 91 unsigned int speed_in;
92 unsigned int speed_out; 92 unsigned int speed_out;
93 const char *path; 93 const char *path;
94}; 94};
95 95
96 96
97# if defined(HAVE_MMAP) 97# if defined(HAVE_MMAP)
98static void *map_file(int fd, unsigned long *length) 98static void *map_file(int fd, unsigned long *length)
99{ 99{
100 void *fdm; 100 void *fdm;
101 101
102 *length += MAD_BUFFER_GUARD; 102 *length += MAD_BUFFER_GUARD;
103 103
104 fdm = mmap(0, *length, PROT_READ, MAP_SHARED, fd, 0); 104 fdm = mmap(0, *length, PROT_READ, MAP_SHARED, fd, 0);
105 if (fdm == MAP_FAILED) 105 if (fdm == MAP_FAILED)
106 return 0; 106 return 0;
107 107
108# if defined(HAVE_MADVISE) 108# if defined(HAVE_MADVISE)
109 madvise(fdm, *length, MADV_SEQUENTIAL); 109 madvise(fdm, *length, MADV_SEQUENTIAL);
110# endif 110# endif
111 111
112 return fdm; 112 return fdm;
113} 113}
114 114
115 115
116static int unmap_file(void *fdm, unsigned long length) 116static int unmap_file(void *fdm, unsigned long length)
117{ 117{
118 if (munmap(fdm, length) == -1) 118 if (munmap(fdm, length) == -1)
119 return -1; 119 return -1;
120 120
121 return 0; 121 return 0;
122} 122}
123# endif 123# endif
124 124
125 125
126static inline QString tr( const char *str ) { 126static inline QString tr( const char *str ) {
127 // Apparently this is okay from a plugin as it runs in the process space of the owner of the plugin 127 // Apparently this is okay from a plugin as it runs in the process space of the owner of the plugin
128 return qApp->translate( "OpiePlayer", str, "libmad strings for mp3 file info" ); 128 return qApp->translate( "OpiePlayer", str, "libmad strings for mp3 file info" );
129} 129}
130 130
131 131
132class LibMadPluginData { 132class LibMadPluginData {
133public: 133public:
134 Input input; 134 Input input;
135 Output output; 135 Output output;
136 int bad_last_frame; 136 int bad_last_frame;
137 struct mad_stream stream; 137 struct mad_stream stream;
138 struct mad_frame frame; 138 struct mad_frame frame;
139 struct mad_synth synth; 139 struct mad_synth synth;
140 bool flush; 140 bool flush;
141}; 141};
142 142
143 143
144LibMadPlugin::LibMadPlugin() { 144LibMadPlugin::LibMadPlugin() {
145 d = new LibMadPluginData; 145 d = new LibMadPluginData;
146 d->input.fd = 0; 146 d->input.fd = 0;
147#if defined(HAVE_MMAP) 147#if defined(HAVE_MMAP)
148 d->input.fdm = 0; 148 d->input.fdm = 0;
149#endif 149#endif
150 d->input.data = 0; 150 d->input.data = 0;
151 d->flush = TRUE; 151 d->flush = TRUE;
152 info = tr( "No Song Open" ); 152 info = tr( "No Song Open" );
153} 153}
154 154
155 155
156LibMadPlugin::~LibMadPlugin() { 156LibMadPlugin::~LibMadPlugin() {
157 close(); 157 close();
158 delete d; 158 delete d;
159} 159}
160 160
161 161
162bool LibMadPlugin::isFileSupported( const QString& path ) { 162bool LibMadPlugin::isFileSupported( const QString& path ) {
163 debugMsg( "LibMadPlugin::isFileSupported" ); 163 debugMsg( "LibMadPlugin::isFileSupported" );
164 164
165 // Mpeg file extensions 165 // Mpeg file extensions
166 // "mp2","mp3","m1v","m2v","m2s","mpg","vob","mpeg","ac3" 166 // "mp2","mp3","m1v","m2v","m2s","mpg","vob","mpeg","ac3"
167 // Other media extensions 167 // Other media extensions
168 // "wav","mid","mod","s3m","ogg","avi","mov","sid" 168 // "wav","mid","mod","s3m","ogg","avi","mov","sid"
169 169
170 char *ext = strrchr( path.latin1(), '.' ); 170 char *ext = strrchr( path.latin1(), '.' );
171 171
172 // Test file extension 172 // Test file extension
173 if ( ext ) { 173 if ( ext ) {
174 if ( strncasecmp(ext, ".mp2", 4) == 0 ) 174 if ( strncasecmp(ext, ".mp2", 4) == 0 )
175 return TRUE; 175 return TRUE;
176 if ( strncasecmp(ext, ".mp3", 4) == 0 ) 176 if ( strncasecmp(ext, ".mp3", 4) == 0 )
177 return TRUE; 177 return TRUE;
178 } 178 }
179 179
180 // UGLY - just for fast testing 180 // UGLY - just for fast testing
181 if ( path.left(4) == "http") { 181 if ( path.left(4) == "http") {
182 return TRUE; 182 return TRUE;
183 } 183 }
184 184
185 return FALSE; 185 return FALSE;
186} 186}
187 187
188 188
189 189
190int LibMadPlugin::is_address_multicast(unsigned long address) { 190int LibMadPlugin::is_address_multicast(unsigned long address) {
191 if ((address & 255) >= 224 && (address & 255) <= 239) 191 if ((address & 255) >= 224 && (address & 255) <= 239)
192 return (1); 192 return (1);
193 return (0); 193 return (0);
194} 194}
195 195
196 196
197int LibMadPlugin::udp_open(char *address, int port) { 197int LibMadPlugin::udp_open(char *address, int port) {
198 198
199 int enable = 1L; 199 int enable = 1L;
200 struct sockaddr_in stAddr; 200 struct sockaddr_in stAddr;
201 struct sockaddr_in stLclAddr; 201 struct sockaddr_in stLclAddr;
202 struct ip_mreq stMreq; 202 struct ip_mreq stMreq;
203 struct hostent *host; 203 struct hostent *host;
204 int sock; 204 int sock;
205 205
206 stAddr.sin_family = AF_INET; 206 stAddr.sin_family = AF_INET;
207 stAddr.sin_port = htons(port); 207 stAddr.sin_port = htons(port);
208 208
209 if ((host = gethostbyname(address)) == NULL) { 209 if ((host = gethostbyname(address)) == NULL) {
210 return (0); 210 return (0);
211 } 211 }
212 212
213 stAddr.sin_addr = *((struct in_addr *)host->h_addr_list[0]); 213 stAddr.sin_addr = *((struct in_addr *)host->h_addr_list[0]);
214 214
215 /* Create a UDP socket */ 215 /* Create a UDP socket */
216 if ((sock = socket(AF_INET, SOCK_DGRAM, 0)) < 0) { 216 if ((sock = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
217 return (0); 217 return (0);
218 } 218 }
219 219
220 /* Allow multiple instance of the client to share the same address and port */ 220 /* Allow multiple instance of the client to share the same address and port */
221 if (setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, (char *)&enable, sizeof(unsigned long int)) < 0) { 221 if (setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, (char *)&enable, sizeof(unsigned long int)) < 0) {
222 return (0); 222 return (0);
223 } 223 }
224 224
225 /* If the address is multicast, register to the multicast group */ 225 /* If the address is multicast, register to the multicast group */
226 if (is_address_multicast(stAddr.sin_addr.s_addr)) { 226 if (is_address_multicast(stAddr.sin_addr.s_addr)) {
227 /* Bind the socket to port */ 227 /* Bind the socket to port */
228 stLclAddr.sin_family = AF_INET; 228 stLclAddr.sin_family = AF_INET;
229 stLclAddr.sin_addr.s_addr = htonl(INADDR_ANY); 229 stLclAddr.sin_addr.s_addr = htonl(INADDR_ANY);
230 stLclAddr.sin_port = stAddr.sin_port; 230 stLclAddr.sin_port = stAddr.sin_port;
231 if (bind(sock, (struct sockaddr *)&stLclAddr, sizeof(stLclAddr)) < 0) { 231 if (bind(sock, (struct sockaddr *)&stLclAddr, sizeof(stLclAddr)) < 0) {
232 return (0); 232 return (0);
233 } 233 }
234 234
235 /* Register to a multicast address */ 235 /* Register to a multicast address */
236 stMreq.imr_multiaddr.s_addr = stAddr.sin_addr.s_addr; 236 stMreq.imr_multiaddr.s_addr = stAddr.sin_addr.s_addr;
237 stMreq.imr_interface.s_addr = INADDR_ANY; 237 stMreq.imr_interface.s_addr = INADDR_ANY;
238 if (setsockopt(sock, IPPROTO_IP, IP_ADD_MEMBERSHIP, (char *)&stMreq, sizeof(stMreq)) < 0) { 238 if (setsockopt(sock, IPPROTO_IP, IP_ADD_MEMBERSHIP, (char *)&stMreq, sizeof(stMreq)) < 0) {
239 return (0); 239 return (0);
240 } 240 }
241 } else { 241 } else {
242 /* Bind the socket to port */ 242 /* Bind the socket to port */
243 stLclAddr.sin_family = AF_INET; 243 stLclAddr.sin_family = AF_INET;
244 stLclAddr.sin_addr.s_addr = htonl(INADDR_ANY); 244 stLclAddr.sin_addr.s_addr = htonl(INADDR_ANY);
245 stLclAddr.sin_port = htons(0); 245 stLclAddr.sin_port = htons(0);
246 if (bind(sock, (struct sockaddr *)&stLclAddr, sizeof(stLclAddr)) < 0) { 246 if (bind(sock, (struct sockaddr *)&stLclAddr, sizeof(stLclAddr)) < 0) {
247 return (0); 247 return (0);
248 } 248 }
249 } 249 }
250 return (sock); 250 return (sock);
251} 251}
252 252
253int LibMadPlugin::tcp_open(char *address, int port) { 253int LibMadPlugin::tcp_open(char *address, int port) {
254 struct sockaddr_in stAddr; 254 struct sockaddr_in stAddr;
255 struct hostent *host; 255 struct hostent *host;
256 int sock; 256 int sock;
257 struct linger l; 257 struct linger l;
258 258
259 memset(&stAddr, 0, sizeof(stAddr)); 259 memset(&stAddr, 0, sizeof(stAddr));
260 stAddr.sin_family = AF_INET; 260 stAddr.sin_family = AF_INET;
261 stAddr.sin_port = htons(port); 261 stAddr.sin_port = htons(port);
262 262
263 if ((host = gethostbyname(address)) == NULL) { 263 if ((host = gethostbyname(address)) == NULL) {
264 return (0); 264 return (0);
265 } 265 }
266 266
267 stAddr.sin_addr = *((struct in_addr *)host->h_addr_list[0]); 267 stAddr.sin_addr = *((struct in_addr *)host->h_addr_list[0]);
268 268
269 if ((sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)) < 0) { 269 if ((sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)) < 0) {
270 return (0); 270 return (0);
271 } 271 }
272 272
273 l.l_onoff = 1; 273 l.l_onoff = 1;
274 l.l_linger = 5; 274 l.l_linger = 5;
275 if (setsockopt(sock, SOL_SOCKET, SO_LINGER, (char *)&l, sizeof(l)) < 0) { 275 if (setsockopt(sock, SOL_SOCKET, SO_LINGER, (char *)&l, sizeof(l)) < 0) {
276 return (0); 276 return (0);
277 } 277 }
278 278
279 if (connect(sock, (struct sockaddr *)&stAddr, sizeof(stAddr)) < 0) { 279 if (connect(sock, (struct sockaddr *)&stAddr, sizeof(stAddr)) < 0) {
280 return (0); 280 return (0);
281 } 281 }
282 282
283 return (sock); 283 return (sock);
284} 284}
285 285
286 286
287/** 287/**
288 * Read a http line header. 288 * Read a http line header.
289 * This function read character by character. 289 * This function read character by character.
290 * @param tcp_sock the socket use to read the stream 290 * @param tcp_sock the socket use to read the stream
291 * @param buf a buffer to receive the data 291 * @param buf a buffer to receive the data
292 * @param size size of the buffer 292 * @param size size of the buffer
293 * @return the size of the stream read or -1 if an error occured 293 * @return the size of the stream read or -1 if an error occured
294 */ 294 */
295int LibMadPlugin::http_read_line(int tcp_sock, char *buf, int size) { 295int LibMadPlugin::http_read_line(int tcp_sock, char *buf, int size) {
296 int offset = 0; 296 int offset = 0;
297 297
298 do { 298 do {
299 if (::read(tcp_sock, buf + offset, 1) < 0) 299 if (::read(tcp_sock, buf + offset, 1) < 0)
300 return -1; 300 return -1;
301 if (buf[offset] != '\r') /* Strip \r from answer */ 301 if (buf[offset] != '\r') /* Strip \r from answer */
302 offset++; 302 offset++;
303 } while (offset < size - 1 && buf[offset - 1] != '\n'); 303 } while (offset < size - 1 && buf[offset - 1] != '\n');
304 304
305 buf[offset] = 0; 305 buf[offset] = 0;
306 return offset; 306 return offset;
307} 307}
308 308
309int LibMadPlugin::http_open(const QString& path ) { 309int LibMadPlugin::http_open(const QString& path ) {
310 char *host; 310 char *host;
311 int port; 311 int port;
312 char *request; 312 char *request;
313 int tcp_sock; 313 int tcp_sock;
314 char http_request[PATH_MAX]; 314 char http_request[PATH_MAX];
315 char filename[PATH_MAX]; 315 char filename[PATH_MAX];
316 //char c; 316 //char c;
317 char *arg =strdup(path.latin1()); 317 char *arg =strdup(path.latin1());
318 318
319 /* Check for URL syntax */ 319 /* Check for URL syntax */
320 if (strncmp(arg, "http://", strlen("http://"))) { 320 if (strncmp(arg, "http://", strlen("http://"))) {
321 return (0); 321 return (0);
322 } 322 }
323 323
324 /* Parse URL */ 324 /* Parse URL */
325 port = 80; 325 port = 80;
326 host = arg + strlen("http://"); 326 host = arg + strlen("http://");
327 if ((request = strchr(host, '/')) == NULL) { 327 if ((request = strchr(host, '/')) == NULL) {
328 return (0); 328 return (0);
329 } 329 }
330 330
331 *request++ = 0; 331 *request++ = 0;
332 332
333 if (strchr(host, ':') != NULL) { /* port is specified */ 333 if (strchr(host, ':') != NULL) { /* port is specified */
334 port = atoi(strchr(host, ':') + 1); 334 port = atoi(strchr(host, ':') + 1);
335 *strchr(host, ':') = 0; 335 *strchr(host, ':') = 0;
336 } 336 }
337 337
338 /* Open a TCP socket */ 338 /* Open a TCP socket */
339 if (!(tcp_sock = tcp_open(host, port))) { 339 if (!(tcp_sock = tcp_open(host, port))) {
340 perror("http_open"); 340 perror("http_open");
341 return (0); 341 return (0);
342 } 342 }
343 343
344 snprintf(filename, sizeof(filename) - strlen(host) - 75, "%s", request); 344 snprintf(filename, sizeof(filename) - strlen(host) - 75, "%s", request);
345 345
346 /* Send HTTP GET request */ 346 /* Send HTTP GET request */
347 /* Please don't use a Agent know by shoutcast (Lynx, Mozilla) seems to be reconized and print 347 /* Please don't use a Agent know by shoutcast (Lynx, Mozilla) seems to be reconized and print
348 * a html page and not the stream */ 348 * a html page and not the stream */
349 snprintf(http_request, sizeof(http_request), "GET /%s HTTP/1.0\r\n" 349 snprintf(http_request, sizeof(http_request), "GET /%s HTTP/1.0\r\n"
350 /* "User-Agent: Mozilla/2.0 (Win95; I)\r\n" */ 350 /* "User-Agent: Mozilla/2.0 (Win95; I)\r\n" */
351 "Pragma: no-cache\r\n" "Host: %s\r\n" "Accept: */*\r\n" "\r\n", filename, host); 351 "Pragma: no-cache\r\n" "Host: %s\r\n" "Accept: */*\r\n" "\r\n", filename, host);
352 352
353 send(tcp_sock, http_request, strlen(http_request), 0); 353 send(tcp_sock, http_request, strlen(http_request), 0);
354 354
355 /* Parse server reply */ 355 /* Parse server reply */
356#if 0 356#if 0
357 do 357 do
358 read(tcp_sock, &c, sizeof(char)); 358 read(tcp_sock, &c, sizeof(char));
359 while (c != ' '); 359 while (c != ' ');
360 read(tcp_sock, http_request, 4 * sizeof(char)); 360 read(tcp_sock, http_request, 4 * sizeof(char));
361 http_request[4] = 0; 361 http_request[4] = 0;
362 if (strcmp(http_request, "200 ")) { 362 if (strcmp(http_request, "200 ")) {
363 fprintf(stderr, "http_open: "); 363 fprintf(stderr, "http_open: ");
364 do { 364 do {
365 read(tcp_sock, &c, sizeof(char)); 365 read(tcp_sock, &c, sizeof(char));
366 fprintf(stderr, "%c", c); 366 fprintf(stderr, "%c", c);
367 } while (c != '\r'); 367 } while (c != '\r');
368 fprintf(stderr, "\n"); 368 fprintf(stderr, "\n");
369 return (0); 369 return (0);
370 } 370 }
371#endif 371#endif
372 372
373 QString name; 373 QString name;
374 QString genre; 374 QString genre;
375 QString bitrate; 375 QString bitrate;
376 QString url; 376 QString url;
377 QString message = tr("Info: "); 377 QString message = tr("Info: ");
378 do { 378 do {
379 379
380 int len; 380 int len;
381 381
382 len = http_read_line(tcp_sock, http_request, sizeof(http_request)); 382 len = http_read_line(tcp_sock, http_request, sizeof(http_request));
383 383
384 if (len == -1) { 384 if (len == -1) {
385 qDebug( "http_open: "+ QString(strerror(errno)) +"\n"); 385 // qDebug( "http_open: "+ QString(strerror(errno)) +"\n");
386 return 0; 386 return 0;
387 } 387 }
388 388
389 if (QString(http_request).left(9) == "Location:") { 389 if (QString(http_request).left(9) == "Location:") {
390 /* redirect */ 390 /* redirect */
391 ::close(tcp_sock); 391 ::close(tcp_sock);
392 http_request[strlen(http_request) - 1] = '\0'; 392 http_request[strlen(http_request) - 1] = '\0';
393 return http_open(&http_request[10]); 393 return http_open(&http_request[10]);
394 } 394 }
395 395
396 if (QString(http_request).left(4) == "ICY ") { 396 if (QString(http_request).left(4) == "ICY ") {
397 /* This is shoutcast/icecast streaming */ 397 /* This is shoutcast/icecast streaming */
398 if (strncmp(http_request + 4, "200 ", 4)) { 398 if (strncmp(http_request + 4, "200 ", 4)) {
399 qDebug("http_open: " + QString(http_request) + "\n"); 399 // qDebug("http_open: " + QString(http_request) + "\n");
400 return 0; 400 return 0;
401 } 401 }
402 } else if (QString(http_request).left(4) == "icy-") { 402 } else if (QString(http_request).left(4) == "icy-") {
403 /* we can have: icy-noticeX, icy-name, icy-genre, icy-url, icy-pub, icy-metaint, icy-br */ 403 /* we can have: icy-noticeX, icy-name, icy-genre, icy-url, icy-pub, icy-metaint, icy-br */
404 if ( QString( http_request ).left( 8 ) == "icy-name" ) { 404 if ( QString( http_request ).left( 8 ) == "icy-name" ) {
405 name = tr("Name: ") + QString(http_request).mid(9, (QString(http_request).length())- 9 ); 405 name = tr("Name: ") + QString(http_request).mid(9, (QString(http_request).length())- 9 );
406 } else if ( QString( http_request ).left( 9 ) == "icy-genre" ) { 406 } else if ( QString( http_request ).left( 9 ) == "icy-genre" ) {
407 genre = tr("Genre: ") + QString(http_request).mid(10, (QString(http_request).length())-10 ); 407 genre = tr("Genre: ") + QString(http_request).mid(10, (QString(http_request).length())-10 );
408 } else if ( QString( http_request ).left( 6 ) == "icy-br" ) { 408 } else if ( QString( http_request ).left( 6 ) == "icy-br" ) {
409 bitrate = tr("Bitrate: ") + QString(http_request).mid(7, (QString(http_request).length())- 7 ); 409 bitrate = tr("Bitrate: ") + QString(http_request).mid(7, (QString(http_request).length())- 7 );
410 } else if ( QString( http_request ).left( 7 ) == "icy-url" ) { 410 } else if ( QString( http_request ).left( 7 ) == "icy-url" ) {
411 url = tr("URL: ") + QString(http_request).mid(8, (QString(http_request).length())- 8 ); 411 url = tr("URL: ") + QString(http_request).mid(8, (QString(http_request).length())- 8 );
412 } else if ( QString( http_request ).left( 10 ) == "icy-notice" ) { 412 } else if ( QString( http_request ).left( 10 ) == "icy-notice" ) {
413 message += QString(http_request).mid(11, QString(http_request).length()-11 ) ; 413 message += QString(http_request).mid(11, QString(http_request).length()-11 ) ;
414 } 414 }
415 } 415 }
416 } while (strcmp(http_request, "\n") != 0); 416 } while (strcmp(http_request, "\n") != 0);
417 417
418 info = QString(name + genre + url + bitrate + message).replace( QRegExp("\n"), " : " ); 418 info = QString(name + genre + url + bitrate + message).replace( QRegExp("\n"), " : " );
419 419
420 qDebug("Stream info: " + info); 420 // qDebug("Stream info: " + info);
421 421
422 return (tcp_sock); 422 return (tcp_sock);
423} 423}
424 424
425 425
426 426
427bool LibMadPlugin::open( const QString& path ) { 427bool LibMadPlugin::open( const QString& path ) {
428 debugMsg( "LibMadPlugin::open" ); 428 debugMsg( "LibMadPlugin::open" );
429 Config cfg("OpiePlayer"); 429 Config cfg("OpiePlayer");
430 cfg.setGroup("Options"); 430 cfg.setGroup("Options");
431 bufferSize = cfg.readNumEntry("MPeg_BufferSize",MPEG_BUFFER_SIZE); 431 bufferSize = cfg.readNumEntry("MPeg_BufferSize",MPEG_BUFFER_SIZE);
432 qDebug("buffer size is %d", bufferSize); 432 // qDebug("buffer size is %d", bufferSize);
433 d->bad_last_frame = 0; 433 d->bad_last_frame = 0;
434 d->flush = TRUE; 434 d->flush = TRUE;
435 info = QString( "" ); 435 info = QString( "" );
436 436
437 //qDebug( "Opening %s", path.latin1() ); 437 //qDebug( "Opening %s", path.latin1() );
438 438
439 if (path.left( 4 ) == "http" ) { 439 if (path.left( 4 ) == "http" ) {
440 // in case of any error we get 0 here 440 // in case of any error we get 0 here
441 if ( !(http_open(path) == 0) ) { 441 if ( !(http_open(path) == 0) ) {
442 d->input.fd = http_open(path); 442 d->input.fd = http_open(path);
443 } else { 443 } else {
444 return FALSE; 444 return FALSE;
445 } 445 }
446 } else { 446 } else {
447 d->input.path = path.latin1(); 447 d->input.path = path.latin1();
448 d->input.fd = ::open( d->input.path, O_RDONLY ); 448 d->input.fd = ::open( d->input.path, O_RDONLY );
449 // thats a better place, since it should only seek for ID3 tags on mp3 files, not streams 449 // thats a better place, since it should only seek for ID3 tags on mp3 files, not streams
450 printID3Tags(); 450 printID3Tags();
451 } 451 }
452 if (d->input.fd == -1) { 452 if (d->input.fd == -1) {
453 qDebug("error opening %s", d->input.path ); 453 // qDebug("error opening %s", d->input.path );
454 return FALSE; 454 return FALSE;
455 } 455 }
456 456
457 struct stat stat; 457 struct stat stat;
458 if (fstat(d->input.fd, &stat) == -1) { 458 if (fstat(d->input.fd, &stat) == -1) {
459 qDebug("error calling fstat"); return FALSE; 459 // qDebug("error calling fstat"); return FALSE;
460 } 460 }
461 if (S_ISREG(stat.st_mode) && stat.st_size > 0) 461 if (S_ISREG(stat.st_mode) && stat.st_size > 0)
462 d->input.fileLength = stat.st_size; 462 d->input.fileLength = stat.st_size;
463 else 463 else
464 d->input.fileLength = 0; 464 d->input.fileLength = 0;
465 465
466#if defined(HAVE_MMAP) 466#if defined(HAVE_MMAP)
467 if (S_ISREG(stat.st_mode) && stat.st_size > 0) { 467 if (S_ISREG(stat.st_mode) && stat.st_size > 0) {
468 d->input.length = stat.st_size; 468 d->input.length = stat.st_size;
469 d->input.fdm = map_file(d->input.fd, &d->input.length); 469 d->input.fdm = map_file(d->input.fd, &d->input.length);
470 if (d->input.fdm == 0) { 470 if (d->input.fdm == 0) {
471 qDebug("error mmapping file"); return FALSE; 471 // qDebug("error mmapping file"); return FALSE;
472 } 472 }
473 d->input.data = (unsigned char *)d->input.fdm; 473 d->input.data = (unsigned char *)d->input.fdm;
474 } 474 }
475#endif 475#endif
476 476
477 if (d->input.data == 0) { 477 if (d->input.data == 0) {
478 d->input.data = (unsigned char *)malloc( bufferSize /*MPEG_BUFFER_SIZE*/); 478 d->input.data = (unsigned char *)malloc( bufferSize /*MPEG_BUFFER_SIZE*/);
479 if (d->input.data == 0) { 479 if (d->input.data == 0) {
480 qDebug("error allocating input buffer"); 480 // qDebug("error allocating input buffer");
481 return FALSE; 481 return FALSE;
482 } 482 }
483 d->input.length = 0; 483 d->input.length = 0;
484 } 484 }
485 485
486 d->input.eof = 0; 486 d->input.eof = 0;
487 487
488 mad_stream_init(&d->stream); 488 mad_stream_init(&d->stream);
489 mad_frame_init(&d->frame); 489 mad_frame_init(&d->frame);
490 mad_synth_init(&d->synth); 490 mad_synth_init(&d->synth);
491 491
492 return TRUE; 492 return TRUE;
493} 493}
494 494
495 495
496bool LibMadPlugin::close() { 496bool LibMadPlugin::close() {
497 debugMsg( "LibMadPlugin::close" ); 497 debugMsg( "LibMadPlugin::close" );
498 498
499 int result = TRUE; 499 int result = TRUE;
500 500
501 mad_synth_finish(&d->synth); 501 mad_synth_finish(&d->synth);
502 mad_frame_finish(&d->frame); 502 mad_frame_finish(&d->frame);
503 mad_stream_finish(&d->stream); 503 mad_stream_finish(&d->stream);
504 504
505#if defined(HAVE_MMAP) 505#if defined(HAVE_MMAP)
506 if (d->input.fdm) { 506 if (d->input.fdm) {
507 if (unmap_file(d->input.fdm, d->input.length) == -1) { 507 if (unmap_file(d->input.fdm, d->input.length) == -1) {
508 qDebug("error munmapping file"); 508 // qDebug("error munmapping file");
509 result = FALSE; 509 result = FALSE;
510 } 510 }
511 d->input.fdm = 0; 511 d->input.fdm = 0;
512 d->input.data = 0; 512 d->input.data = 0;
513 } 513 }
514#endif 514#endif
515 515
516 if (d->input.data) { 516 if (d->input.data) {
517 free(d->input.data); 517 free(d->input.data);
518 d->input.data = 0; 518 d->input.data = 0;
519 } 519 }
520 520
521 if (::close(d->input.fd) == -1) { 521 if (::close(d->input.fd) == -1) {
522 qDebug("error closing file %s", d->input.path); 522 // qDebug("error closing file %s", d->input.path);
523 result = FALSE; 523 result = FALSE;
524 } 524 }
525 525
526 d->input.fd = 0; 526 d->input.fd = 0;
527 527
528 return result; 528 return result;
529} 529}
530 530
531 531
532bool LibMadPlugin::isOpen() { 532bool LibMadPlugin::isOpen() {
533 debugMsg( "LibMadPlugin::isOpen" ); 533 debugMsg( "LibMadPlugin::isOpen" );
534 return ( d->input.fd != 0 ); 534 return ( d->input.fd != 0 );
535} 535}
536 536
537 537
538int LibMadPlugin::audioStreams() { 538int LibMadPlugin::audioStreams() {
539 debugMsg( "LibMadPlugin::audioStreams" ); 539 debugMsg( "LibMadPlugin::audioStreams" );
540 return 1; 540 return 1;
541} 541}
542 542
543 543
544int LibMadPlugin::audioChannels( int ) { 544int LibMadPlugin::audioChannels( int ) {
545 debugMsg( "LibMadPlugin::audioChannels" ); 545 debugMsg( "LibMadPlugin::audioChannels" );
546/* 546/*
547 long t; short t1[5]; audioReadSamples( t1, 2, 1, t, 0 ); 547 long t; short t1[5]; audioReadSamples( t1, 2, 1, t, 0 );
548 qDebug( "LibMadPlugin::audioChannels: %i", d->frame.header.mode > 0 ? 2 : 1 ); 548 qDebug( "LibMadPlugin::audioChannels: %i", d->frame.header.mode > 0 ? 2 : 1 );
549 return d->frame.header.mode > 0 ? 2 : 1; 549 return d->frame.header.mode > 0 ? 2 : 1;
550*/ 550*/
551 return 2; 551 return 2;
552} 552}
553 553
554 554
555int LibMadPlugin::audioFrequency( int ) { 555int LibMadPlugin::audioFrequency( int ) {
556 debugMsg( "LibMadPlugin::audioFrequency" ); 556 debugMsg( "LibMadPlugin::audioFrequency" );
557 long t; short t1[5]; audioReadSamples( t1, 2, 1, t, 0 ); 557 long t; short t1[5]; audioReadSamples( t1, 2, 1, t, 0 );
558 qDebug( "LibMadPlugin::audioFrequency: %i", d->frame.header.samplerate ); 558 // qDebug( "LibMadPlugin::audioFrequency: %i", d->frame.header.samplerate );
559 return d->frame.header.samplerate; 559 return d->frame.header.samplerate;
560} 560}
561 561
562 562
563int LibMadPlugin::audioSamples( int ) { 563int LibMadPlugin::audioSamples( int ) {
564 debugMsg( "LibMadPlugin::audioSamples" ); 564 debugMsg( "LibMadPlugin::audioSamples" );
565 565
566 long t; short t1[5]; audioReadSamples( t1, 2, 1, t, 0 ); 566 long t; short t1[5]; audioReadSamples( t1, 2, 1, t, 0 );
567 mad_header_decode( (struct mad_header *)&d->frame.header, &d->stream ); 567 mad_header_decode( (struct mad_header *)&d->frame.header, &d->stream );
568/* 568/*
569 qDebug( "LibMadPlugin::audioSamples: %i*%i", d->frame.header.duration.seconds, 569 qDebug( "LibMadPlugin::audioSamples: %i*%i", d->frame.header.duration.seconds,
570 d->frame.header.samplerate ); 570 d->frame.header.samplerate );
571 return d->frame.header.duration.seconds * d->frame.header.samplerate; 571 return d->frame.header.duration.seconds * d->frame.header.samplerate;
572*/ 572*/
573 if ( d->frame.header.bitrate == 0 ) 573 if ( d->frame.header.bitrate == 0 )
574 return 0; 574 return 0;
575 int samples = (d->input.fileLength / (d->frame.header.bitrate/8)) * d->frame.header.samplerate; 575 int samples = (d->input.fileLength / (d->frame.header.bitrate/8)) * d->frame.header.samplerate;
576 576
577 qDebug( "LibMadPlugin::audioSamples: %i * %i * 8 / %i", (int)d->input.fileLength, 577 // qDebug( "LibMadPlugin::audioSamples: %i * %i * 8 / %i", (int)d->input.fileLength,
578 (int)d->frame.header.samplerate, (int)d->frame.header.bitrate ); 578 // (int)d->frame.header.samplerate, (int)d->frame.header.bitrate );
579 qDebug( "LibMadPlugin::audioSamples: %i", samples ); 579 // qDebug( "LibMadPlugin::audioSamples: %i", samples );
580 580
581 return samples; 581 return samples;
582 582
583// return 10000000; 583// return 10000000;
584} 584}
585 585
586 586
587bool LibMadPlugin::audioSetSample( long, int ) { 587bool LibMadPlugin::audioSetSample( long, int ) {
588 debugMsg( "LibMadPlugin::audioSetSample" ); 588 debugMsg( "LibMadPlugin::audioSetSample" );
589 589
590// long totalSamples = audioSamples(0); 590// long totalSamples = audioSamples(0);
591// if ( totalSamples <= 1 ) 591// if ( totalSamples <= 1 )
592// return FALSE; 592// return FALSE;
593 593
594// // Seek to requested position 594// // Seek to requested position
595// qDebug( "seek pos: %i", (int)((double)pos * d->input.fileLength / totalSamples) ); 595// qDebug( "seek pos: %i", (int)((double)pos * d->input.fileLength / totalSamples) );
596// ::lseek( d->input.fd, (long)((double)pos * d->input.fileLength / totalSamples), SEEK_SET ); 596// ::lseek( d->input.fd, (long)((double)pos * d->input.fileLength / totalSamples), SEEK_SET );
597// mad_stream_sync(&d->stream); 597// mad_stream_sync(&d->stream);
598 598
599// mad_stream_init(&d->stream); 599// mad_stream_init(&d->stream);
600// mad_frame_init(&d->frame); 600// mad_frame_init(&d->frame);
601// mad_synth_init(&d->synth); 601// mad_synth_init(&d->synth);
602 602
603// return TRUE; 603// return TRUE;
604 debugMsg( "LibMadPlugin::audioSetSample" ); 604 debugMsg( "LibMadPlugin::audioSetSample" );
605 return FALSE; 605 return FALSE;
606} 606}
607 607
608 608
609long LibMadPlugin::audioGetSample( int ) { 609long LibMadPlugin::audioGetSample( int ) {
610 debugMsg( "LibMadPlugin::audioGetSample" ); 610 debugMsg( "LibMadPlugin::audioGetSample" );
611 return 0; 611 return 0;
612} 612}
613 613
614/* 614/*
615bool LibMadPlugin::audioReadSamples( short *, int, long, int ) { 615bool LibMadPlugin::audioReadSamples( short *, int, long, int ) {
616debugMsg( "LibMadPlugin::audioReadSamples" ); 616debugMsg( "LibMadPlugin::audioReadSamples" );
617return FALSE; 617return FALSE;
618} 618}
619 619
620 620
621bool LibMadPlugin::audioReReadSamples( short *, int, long, int ) { 621bool LibMadPlugin::audioReReadSamples( short *, int, long, int ) {
622debugMsg( "LibMadPlugin::audioReReadSamples" ); 622debugMsg( "LibMadPlugin::audioReReadSamples" );
623 return FALSE; 623 return FALSE;
624 } 624 }
625*/ 625*/
626 626
627bool LibMadPlugin::read() { 627bool LibMadPlugin::read() {
628 debugMsg( "LibMadPlugin::read" ); 628 debugMsg( "LibMadPlugin::read" );
629 int len; 629 int len;
630 630
631 if (d->input.eof) 631 if (d->input.eof)
632 return FALSE; 632 return FALSE;
633 633
634#if defined(HAVE_MMAP) 634#if defined(HAVE_MMAP)
635 if (d->input.fdm) { 635 if (d->input.fdm) {
636 unsigned long skip = 0; 636 unsigned long skip = 0;
637 637
638 if (d->stream.next_frame) { 638 if (d->stream.next_frame) {
639 struct stat stat; 639 struct stat stat;
640 640
641 if (fstat(d->input.fd, &stat) == -1) 641 if (fstat(d->input.fd, &stat) == -1)
642 return FALSE; 642 return FALSE;
643 643
644 if (stat.st_size + MAD_BUFFER_GUARD <= (signed)d->input.length) 644 if (stat.st_size + MAD_BUFFER_GUARD <= (signed)d->input.length)
645 return FALSE; 645 return FALSE;
646 646
647 // file size changed; update memory map 647 // file size changed; update memory map
648 skip = d->stream.next_frame - d->input.data; 648 skip = d->stream.next_frame - d->input.data;
649 649
650 if (unmap_file(d->input.fdm, d->input.length) == -1) { 650 if (unmap_file(d->input.fdm, d->input.length) == -1) {
651 d->input.fdm = 0; 651 d->input.fdm = 0;
652 d->input.data = 0; 652 d->input.data = 0;
653 return FALSE; 653 return FALSE;
654 } 654 }
655 655
656 d->input.length = stat.st_size; 656 d->input.length = stat.st_size;
657 657
658 d->input.fdm = map_file(d->input.fd, &d->input.length); 658 d->input.fdm = map_file(d->input.fd, &d->input.length);
659 if (d->input.fdm == 0) { 659 if (d->input.fdm == 0) {
660 d->input.data = 0; 660 d->input.data = 0;
661 return FALSE; 661 return FALSE;
662 } 662 }
663 663
664 d->input.data = (unsigned char *)d->input.fdm; 664 d->input.data = (unsigned char *)d->input.fdm;
665 } 665 }
666 666
667 mad_stream_buffer(&d->stream, d->input.data + skip, d->input.length - skip); 667 mad_stream_buffer(&d->stream, d->input.data + skip, d->input.length - skip);
668 668
669 } else 669 } else
670#endif 670#endif
671 { 671 {
672 if (d->stream.next_frame) { 672 if (d->stream.next_frame) {
673 memmove(d->input.data, d->stream.next_frame, 673 memmove(d->input.data, d->stream.next_frame,
674 d->input.length = &d->input.data[d->input.length] - d->stream.next_frame); 674 d->input.length = &d->input.data[d->input.length] - d->stream.next_frame);
675 } 675 }
676 676
677 do { 677 do {
678 len = ::read(d->input.fd, d->input.data + d->input.length, bufferSize /* MPEG_BUFFER_SIZE*/ - d->input.length); 678 len = ::read(d->input.fd, d->input.data + d->input.length, bufferSize /* MPEG_BUFFER_SIZE*/ - d->input.length);
679 } 679 }
680 while (len == -1 && errno == EINTR); 680 while (len == -1 && errno == EINTR);
681 681
682 if (len == -1) { 682 if (len == -1) {
683 qDebug("error reading audio"); 683 // qDebug("error reading audio");
684 return FALSE; 684 return FALSE;
685 } 685 }
686 else if (len == 0) { 686 else if (len == 0) {
687 d->input.eof = 1; 687 d->input.eof = 1;
688 688
689 assert(bufferSize /*MPEG_BUFFER_SIZE*/ - d->input.length >= MAD_BUFFER_GUARD); 689 assert(bufferSize /*MPEG_BUFFER_SIZE*/ - d->input.length >= MAD_BUFFER_GUARD);
690 690
691 while (len < MAD_BUFFER_GUARD) 691 while (len < MAD_BUFFER_GUARD)
692 d->input.data[d->input.length + len++] = 0; 692 d->input.data[d->input.length + len++] = 0;
693 } 693 }
694 694
695 mad_stream_buffer(&d->stream, d->input.data, d->input.length += len); 695 mad_stream_buffer(&d->stream, d->input.data, d->input.length += len);
696 } 696 }
697 697
698 return TRUE; 698 return TRUE;
699} 699}
700 700
701 701
702static mad_fixed_t left_err, right_err; 702static mad_fixed_t left_err, right_err;
703static const int bits = 16; 703static const int bits = 16;
704static const int shift = MAD_F_FRACBITS + 1 - bits; 704static const int shift = MAD_F_FRACBITS + 1 - bits;
705 705
706 706
707inline long audio_linear_dither( mad_fixed_t sample, mad_fixed_t& error ) { 707inline long audio_linear_dither( mad_fixed_t sample, mad_fixed_t& error ) {
708 sample += error; 708 sample += error;
709 mad_fixed_t quantized = (sample >= MAD_F_ONE) ? MAD_F_ONE - 1 : ( (sample < -MAD_F_ONE) ? -MAD_F_ONE : sample ); 709 mad_fixed_t quantized = (sample >= MAD_F_ONE) ? MAD_F_ONE - 1 : ( (sample < -MAD_F_ONE) ? -MAD_F_ONE : sample );
710 quantized &= ~((1L << shift) - 1); 710 quantized &= ~((1L << shift) - 1);
711 error = sample - quantized; 711 error = sample - quantized;
712 return quantized >> shift; 712 return quantized >> shift;
713} 713}
714 714
715 715
716inline void audio_pcm( short *data, unsigned int nsamples, mad_fixed_t *left, mad_fixed_t *right ) { 716inline void audio_pcm( short *data, unsigned int nsamples, mad_fixed_t *left, mad_fixed_t *right ) {
717 if ( right ) { 717 if ( right ) {
718 while (nsamples--) { 718 while (nsamples--) {
719 data[0] = audio_linear_dither( *left++, left_err ); 719 data[0] = audio_linear_dither( *left++, left_err );
720 data[1] = audio_linear_dither( *right++, right_err ); 720 data[1] = audio_linear_dither( *right++, right_err );
721 data += 2; 721 data += 2;
722 } 722 }
723 } else { 723 } else {
724 while (nsamples--) { 724 while (nsamples--) {
725 data[0] = data[1] = audio_linear_dither( *left++, left_err ); 725 data[0] = data[1] = audio_linear_dither( *left++, left_err );
726 data += 2; 726 data += 2;
727 } 727 }
728 } 728 }
729} 729}
730 730
731 731
732bool LibMadPlugin::decode( short *output, long samples, long& samplesMade ) { 732bool LibMadPlugin::decode( short *output, long samples, long& samplesMade ) {
733 debugMsg( "LibMadPlugin::decode" ); 733 debugMsg( "LibMadPlugin::decode" );
734 734
735 static int buffered = 0; 735 static int buffered = 0;
736 static mad_fixed_t buffer[2][65536 * 2]; 736 static mad_fixed_t buffer[2][65536 * 2];
737 int offset = buffered; 737 int offset = buffered;
738 samplesMade = 0; 738 samplesMade = 0;
739 739
740 static int maxBuffered = 8000; // 65536; 740 static int maxBuffered = 8000; // 65536;
741 741
742 if ( samples > maxBuffered ) { 742 if ( samples > maxBuffered ) {
743 samples = maxBuffered; 743 samples = maxBuffered;
744 } 744 }
745 745
746 if ( d->flush ) { 746 if ( d->flush ) {
747 buffered = 0; 747 buffered = 0;
748 offset = 0; 748 offset = 0;
749 d->flush = FALSE; 749 d->flush = FALSE;
750 } 750 }
751 751
752 while ( buffered < maxBuffered ) { 752 while ( buffered < maxBuffered ) {
753 753
754 while (mad_frame_decode(&d->frame, &d->stream) == -1) { 754 while (mad_frame_decode(&d->frame, &d->stream) == -1) {
755 if (!MAD_RECOVERABLE(d->stream.error)) { 755 if (!MAD_RECOVERABLE(d->stream.error)) {
756 debugMsg( "feed me" ); 756 debugMsg( "feed me" );
757 return FALSE; // Feed me 757 return FALSE; // Feed me
758 } 758 }
759 if ( d->stream.error == MAD_ERROR_BADCRC ) { 759 if ( d->stream.error == MAD_ERROR_BADCRC ) {
760 mad_frame_mute(&d->frame); 760 mad_frame_mute(&d->frame);
761 qDebug( "error decoding, bad crc" ); 761 // qDebug( "error decoding, bad crc" );
762 } 762 }
763 } 763 }
764 764
765 mad_synth_frame(&d->synth, &d->frame); 765 mad_synth_frame(&d->synth, &d->frame);
766 int decodedSamples = d->synth.pcm.length; 766 int decodedSamples = d->synth.pcm.length;
767 memcpy( &(buffer[0][offset]), d->synth.pcm.samples[0], decodedSamples * sizeof(mad_fixed_t) ); 767 memcpy( &(buffer[0][offset]), d->synth.pcm.samples[0], decodedSamples * sizeof(mad_fixed_t) );
768 if ( d->synth.pcm.channels == 2 ) 768 if ( d->synth.pcm.channels == 2 )
769 memcpy( &(buffer[1][offset]), d->synth.pcm.samples[1], decodedSamples * sizeof(mad_fixed_t) ); 769 memcpy( &(buffer[1][offset]), d->synth.pcm.samples[1], decodedSamples * sizeof(mad_fixed_t) );
770 offset += decodedSamples; 770 offset += decodedSamples;
771 buffered += decodedSamples; 771 buffered += decodedSamples;
772 } 772 }
773 773
774//qApp->processEvents(); 774//qApp->processEvents();
775 audio_pcm( output, samples, buffer[0], (d->synth.pcm.channels == 2) ? buffer[1] : 0 ); 775 audio_pcm( output, samples, buffer[0], (d->synth.pcm.channels == 2) ? buffer[1] : 0 );
776// audio_pcm( output, samples, buffer[1], buffer[0] ); 776// audio_pcm( output, samples, buffer[1], buffer[0] );
777// audio_pcm( output, samples, buffer[0], buffer[1] ); 777// audio_pcm( output, samples, buffer[0], buffer[1] );
778 samplesMade = samples; 778 samplesMade = samples;
779 memmove( buffer[0], &(buffer[0][samples]), (buffered - samples) * sizeof(mad_fixed_t) ); 779 memmove( buffer[0], &(buffer[0][samples]), (buffered - samples) * sizeof(mad_fixed_t) );
780 if ( d->synth.pcm.channels == 2 ) { 780 if ( d->synth.pcm.channels == 2 ) {
781 memmove( buffer[1], &(buffer[1][samples]), (buffered - samples) * sizeof(mad_fixed_t) ); 781 memmove( buffer[1], &(buffer[1][samples]), (buffered - samples) * sizeof(mad_fixed_t) );
782 } 782 }
783 buffered -= samples; 783 buffered -= samples;
784 784
785 return TRUE; 785 return TRUE;
786} 786}
787 787
788/*bool LibMadPlugin::audioReadStereoSamples( short *output, long samples, long& samplesMade, int ) { 788/*bool LibMadPlugin::audioReadStereoSamples( short *output, long samples, long& samplesMade, int ) {
789*/ 789*/
790bool LibMadPlugin::audioReadSamples( short *output, int /*channels*/, long samples, long& samplesMade, int ) { 790bool LibMadPlugin::audioReadSamples( short *output, int /*channels*/, long samples, long& samplesMade, int ) {
791 debugMsg( "LibMadPlugin::audioReadStereoSamples" ); 791 debugMsg( "LibMadPlugin::audioReadStereoSamples" );
792 792
793 static bool needInput = TRUE; 793 static bool needInput = TRUE;
794 794
795 if ( samples == 0 ) 795 if ( samples == 0 )
796 return FALSE; 796 return FALSE;
797 797
798 do { 798 do {
799 if ( needInput ) 799 if ( needInput )
800 if ( !read() ) { 800 if ( !read() ) {
801 return FALSE; 801 return FALSE;
802 } 802 }
803 803
804 needInput = FALSE; 804 needInput = FALSE;
805 805
806 if ( decode( output, samples, samplesMade ) ) 806 if ( decode( output, samples, samplesMade ) )
807 return TRUE; 807 return TRUE;
808 else 808 else
809 needInput = TRUE; 809 needInput = TRUE;
810 } 810 }
811 while ( ( samplesMade < samples ) && ( !d->input.eof ) ); 811 while ( ( samplesMade < samples ) && ( !d->input.eof ) );
812 812
813 return FALSE; 813 return FALSE;
814} 814}
815 815
816 816
817double LibMadPlugin::getTime() { 817double LibMadPlugin::getTime() {
818 debugMsg( "LibMadPlugin::getTime" ); 818 debugMsg( "LibMadPlugin::getTime" );
819 return 0.0; 819 return 0.0;
820} 820}
821 821
822 822
823void LibMadPlugin::printID3Tags() { 823void LibMadPlugin::printID3Tags() {
824 qDebug( "LibMadPlugin::printID3Tags" ); 824 // qDebug( "LibMadPlugin::printID3Tags" );
825 825
826 char id3v1[128 + 1]; 826 char id3v1[128 + 1];
827 827
828 if ( ::lseek( d->input.fd, -128, SEEK_END ) == -1 ) { 828 if ( ::lseek( d->input.fd, -128, SEEK_END ) == -1 ) {
829 qDebug( "error seeking to id3 tags" ); 829 // qDebug( "error seeking to id3 tags" );
830 return; 830 return;
831 } 831 }
832 832
833 if ( ::read( d->input.fd, id3v1, 128 ) != 128 ) { 833 if ( ::read( d->input.fd, id3v1, 128 ) != 128 ) {
834 qDebug( "error reading in id3 tags" ); 834 // qDebug( "error reading in id3 tags" );
835 return; 835 return;
836 } 836 }
837 837
838 if ( ::strncmp( (const char *)id3v1, "TAG", 3 ) != 0 ) { 838 if ( ::strncmp( (const char *)id3v1, "TAG", 3 ) != 0 ) {
839 debugMsg( "sorry, no id3 tags" ); 839 debugMsg( "sorry, no id3 tags" );
840 } else { 840 } else {
841 int len[5] = { 30, 30, 30, 4, 30 }; 841 int len[5] = { 30, 30, 30, 4, 30 };
842 QString label[5] = { tr( "Title" ), tr( "Artist" ), tr( "Album" ), tr( "Year" ), tr( "Comment" ) }; 842 QString label[5] = { tr( "Title" ), tr( "Artist" ), tr( "Album" ), tr( "Year" ), tr( "Comment" ) };
843 char *ptr = id3v1 + 3, *ptr2 = ptr + len[0]; 843 char *ptr = id3v1 + 3, *ptr2 = ptr + len[0];
844 qDebug( "ID3 tags in file:" ); 844 // qDebug( "ID3 tags in file:" );
845 info = ""; 845 info = "";
846 for ( int i = 0; i < 5; ptr += len[i], i++, ptr2 += len[i] ) { 846 for ( int i = 0; i < 5; ptr += len[i], i++, ptr2 += len[i] ) {
847 char push = *ptr2; 847 char push = *ptr2;
848 *ptr2 = '\0'; 848 *ptr2 = '\0';
849 char *ptr3 = ptr2; 849 char *ptr3 = ptr2;
850 while ( ptr3-1 >= ptr && isspace(ptr3[-1]) ) ptr3--; 850 while ( ptr3-1 >= ptr && isspace(ptr3[-1]) ) ptr3--;
851 char push2 = *ptr3; *ptr3 = '\0'; 851 char push2 = *ptr3; *ptr3 = '\0';
852 if ( strcmp( ptr, "" ) ) { 852 if ( strcmp( ptr, "" ) ) {
853 if( ((QString)ptr).find(" ") == -1) // don't add anything that has blanks 853 if( ((QString)ptr).find(" ") == -1) // don't add anything that has blanks
854 info += ( i != 0 ? ", " : "" ) + label[i] + ": " + ptr; 854 info += ( i != 0 ? ", " : "" ) + label[i] + ": " + ptr;
855 } 855 }
856// qDebug( info.latin1() ); 856// qDebug( info.latin1() );
857 *ptr3 = push2; 857 *ptr3 = push2;
858 *ptr2 = push; 858 *ptr2 = push;
859 } 859 }
860 if (id3v1[126] == 0 && id3v1[127] != 0) 860 if (id3v1[126] == 0 && id3v1[127] != 0)
861 info += tr( ", Track: " ) + id3v1[127]; 861 info += tr( ", Track: " ) + id3v1[127];
862 } 862 }
863 863
864 if ( ::lseek(d->input.fd, 0, SEEK_SET) == -1 ) { 864 if ( ::lseek(d->input.fd, 0, SEEK_SET) == -1 ) {
865 qDebug( "error seeking back to beginning" ); 865 // qDebug( "error seeking back to beginning" );
866 return; 866 return;
867 } 867 }
868} 868}
869 869