1 /***************************************************************************
3 * Open \______ \ ____ ____ | | _\_ |__ _______ ___
4 * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
5 * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
6 * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
10 * Copyright (C) 2002 Linus Nielsen Feltzing
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation; either version 2
15 * of the License, or (at your option) any later version.
17 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
18 * KIND, either express or implied.
20 ****************************************************************************/
23 /****************************************************************************
26 * We allocate the MP3 buffer for storing the text to be sorted, and then
27 * search the buffer for newlines and store an array of character pointers
30 * The pointer array grows from the top of the buffer and downwards:
33 * | pointers[2] |--------|
34 * | pointers[1] |------| |
35 * | pointers[0] |----| | |
36 * |-------------| | | |
39 * | free space | | | |
42 * |-------------| | | |
44 * | line 3\0 |<---| | |
45 * | line 2\0 |<-----| |
46 * | line 1\0 |<-------|
49 * The advantage of this method is that we optimize the buffer usage.
51 * The disadvantage is that the string pointers will be loaded in reverse
52 * order. We therefore sort the strings in reverse order as well, so we
53 * don't have to sort an already sorted buffer.
54 ****************************************************************************/
56 /***************************************************************************
57 * TODO: Implement a merge sort for files larger than the buffer
58 ****************************************************************************/
63 static char *filename
;
64 static int num_entries
;
65 static char **pointers
;
66 static char *stringbuffer
;
67 static char crlf
[2] = "\r\n";
69 /* Compare function for sorting backwards */
70 static int compare(const void* p1
, const void* p2
)
72 char *s1
= *(char **)p1
;
73 char *s2
= *(char **)p2
;
75 return rb
->strcasecmp(s2
, s1
);
78 static void sort_buffer(void)
80 rb
->qsort(pointers
, num_entries
, sizeof(char *), compare
);
83 int read_buffer(int offset
)
90 fd
= rb
->open(filename
, O_RDONLY
);
94 /* Fill the buffer from the file */
95 rb
->lseek(fd
, offset
, SEEK_SET
);
96 readsize
= rb
->read(fd
, stringbuffer
, buf_size
);
100 return readsize
* 10 - 2;
102 /* Temporary fix until we can do merged sorting */
103 if(readsize
== buf_size
)
104 return buf_size
; /* File too big */
106 buf_ptr
= stringbuffer
;
111 while(*buf_ptr
!= '\n' && buf_ptr
< (char *)pointers
) {
112 /* Terminate the string with CR... */
121 return tmp_ptr
- stringbuffer
; /* Buffer is full, return
122 the point to resume at */
129 } while(buf_ptr
< stringbuffer
+ readsize
);
134 static int write_file(void)
136 char tmpfilename
[MAX_PATH
+1];
141 /* Create a temporary file */
142 rb
->snprintf(tmpfilename
, MAX_PATH
+1, "%s.tmp", filename
);
143 fd
= rb
->creat(tmpfilename
);
147 /* Write the sorted strings, with appended CR/LF, to the temp file,
149 for(i
= num_entries
-1;i
>= 0;i
--) {
150 rc
= rb
->write(fd
, pointers
[i
], rb
->strlen(pointers
[i
]));
156 rc
= rb
->write(fd
, crlf
, 2);
165 /* Remove the original file */
166 rc
= rb
->remove(filename
);
171 /* Replace the old file with the new */
172 rc
= rb
->rename(tmpfilename
, filename
);
179 enum plugin_status
plugin_start(const void* parameter
)
184 filename
= (char *)parameter
;
186 buf
= rb
->plugin_get_audio_buffer((size_t *)&buf_size
); /* start munching memory */
189 pointers
= (char **)(buf
+ buf_size
- sizeof(int));
191 rb
->lcd_clear_display();
192 rb
->splash(0, "Loading...");
196 rb
->lcd_clear_display();
197 rb
->splash(0, "Sorting...");
200 rb
->lcd_clear_display();
201 rb
->splash(0, "Writing...");
205 rb
->lcd_clear_display();
206 rb
->splashf(HZ
, "Can't write file: %d", rc
);
208 rb
->lcd_clear_display();
209 rb
->splash(HZ
, "Done");
213 rb
->lcd_clear_display();
214 rb
->splashf(HZ
, "Can't read file: %d", rc
);
216 rb
->lcd_clear_display();
217 rb
->splash(HZ
, "The file is too big");