1 /**
2 * \File playlist-spl.c
3 *
4 * Playlist_t to Samsung (.spl) and back conversion functions.
5 *
6 * Copyright (C) 2008 Alistair Boyle <alistair.js.boyle@gmail.com>
7 *
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2 of the License, or (at your option) any later version.
12 *
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
17 *
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with this library; if not, write to the
20 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
21 * Boston, MA 02111-1307, USA.
22 */
23
24 #include <config.h>
25
26 #include <stdio.h>
27 #include <stdlib.h> // mkstmp()
28 #include <unistd.h>
29 #include <errno.h>
30 #include <sys/stat.h>
31 #include <sys/types.h>
32 #ifdef HAVE_SYS_UIO_H
33 #include <sys/uio.h>
34 #endif
35 #include <fcntl.h>
36
37 #include <string.h>
38
39 #include "libmtp.h"
40 #include "libusb-glue.h"
41 #include "ptp.h"
42 #include "unicode.h"
43
44 #include "playlist-spl.h"
45
46 // set this to 1 to add lots of messy debug output to the playlist code
47 #define DEBUG_ENABLED 0
48
49 // debug macro
50 // d = indenting depth
51 #define IF_DEBUG() if(DEBUG_ENABLED) {\
52 printf("%s:%u:%s(): ", __FILE__, __LINE__, __func__); \
53 } \
54 if(DEBUG_ENABLED)
55
56 // Internal singly linked list of strings
57 // used to hold .spl playlist in memory
58 typedef struct text_struct {
59 char* text; // String
60 struct text_struct *next; // Link to next line, NULL if end of list
61 } text_t;
62
63
64 /**
65 * Forward declarations of local (static) functions.
66 */
67 static text_t* read_into_spl_text_t(LIBMTP_mtpdevice_t *device, const int fd);
68 static void write_from_spl_text_t(LIBMTP_mtpdevice_t *device, const int fd, text_t* p);
69 static void free_spl_text_t(text_t* p);
70 static void print_spl_text_t(text_t* p);
71 static uint32_t trackno_spl_text_t(text_t* p);
72 static void tracks_from_spl_text_t(text_t* p, uint32_t* tracks, LIBMTP_folder_t* folders, LIBMTP_file_t* files);
73 static void spl_text_t_from_tracks(text_t** p, uint32_t* tracks, const uint32_t trackno, const uint32_t ver_major, const uint32_t ver_minor, char* dnse, LIBMTP_folder_t* folders, LIBMTP_file_t* files);
74
75 static uint32_t discover_id_from_filepath(const char* s, LIBMTP_folder_t* folders, LIBMTP_file_t* files); // TODO add file/dir cached args
76 static void discover_filepath_from_id(char** p, uint32_t track, LIBMTP_folder_t* folders, LIBMTP_file_t* files);
77 static void find_folder_name(LIBMTP_folder_t* folders, uint32_t* id, char** name);
78 static uint32_t find_folder_id(LIBMTP_folder_t* folders, uint32_t parent, char* name);
79
80 static void append_text_t(text_t** t, char* s);
81
82
83
84
85 /**
86 * Decides if the indicated object index is an .spl playlist.
87 *
88 * @param oi object we are deciding on
89 * @return 1 if this is a Samsung .spl object, 0 otherwise
90 */
is_spl_playlist(PTPObjectInfo * oi)91 int is_spl_playlist(PTPObjectInfo *oi)
92 {
93 return (oi->ObjectFormat == PTP_OFC_Undefined) &&
94 (strlen(oi->Filename) > 4) &&
95 (strcmp((oi->Filename + strlen(oi->Filename) -4), ".spl") == 0);
96 }
97
98 #ifndef HAVE_MKSTEMP
99 # ifdef __WIN32__
100 # include <fcntl.h>
101 # define mkstemp(_pattern) _open(_mktemp(_pattern), _O_CREAT | _O_SHORT_LIVED | _O_EXCL)
102 # else
103 # error Missing mkstemp() function.
104 # endif
105 #endif
106
107 /**
108 * Take an object ID, a .spl playlist on the MTP device,
109 * and convert it to a playlist_t object.
110 *
111 * @param device mtp device pointer
112 * @param oi object we are reading
113 * @param id .spl playlist id on MTP device
114 * @param pl the LIBMTP_playlist_t pointer to be filled with info from id
115 */
116
spl_to_playlist_t(LIBMTP_mtpdevice_t * device,PTPObjectInfo * oi,const uint32_t id,LIBMTP_playlist_t * const pl)117 void spl_to_playlist_t(LIBMTP_mtpdevice_t* device, PTPObjectInfo *oi,
118 const uint32_t id, LIBMTP_playlist_t * const pl)
119 {
120 // Fill in playlist metadata
121 // Use the Filename as the playlist name, dropping the ".spl" extension
122 pl->name = malloc(sizeof(char)*(strlen(oi->Filename) -4 +1));
123 memcpy(pl->name, oi->Filename, strlen(oi->Filename) -4);
124 // Set terminating character
125 pl->name[strlen(oi->Filename) - 4] = 0;
126 pl->playlist_id = id;
127 pl->parent_id = oi->ParentObject;
128 pl->storage_id = oi->StorageID;
129 pl->tracks = NULL;
130 pl->no_tracks = 0;
131
132 IF_DEBUG() printf("pl->name='%s'\n",pl->name);
133
134 // open a temporary file
135 char tmpname[] = "/tmp/mtp-spl2pl-XXXXXX";
136 int fd = mkstemp(tmpname);
137 if(fd < 0) {
138 printf("failed to make temp file for %s.spl -> %s, errno=%s\n", pl->name, tmpname, strerror(errno));
139 return;
140 }
141 // make sure the file will be deleted afterwards
142 if(unlink(tmpname) < 0)
143 printf("failed to delete temp file for %s.spl -> %s, errno=%s\n", pl->name, tmpname, strerror(errno));
144 int ret = LIBMTP_Get_File_To_File_Descriptor(device, pl->playlist_id, fd, NULL, NULL, NULL);
145 if( ret < 0 ) {
146 // FIXME add_ptp_error_to_errorstack(device, ret, "LIBMTP_Get_Playlist: Could not get .spl playlist file.");
147 close(fd);
148 printf("FIXME closed\n");
149 }
150
151 text_t* p = read_into_spl_text_t(device, fd);
152 close(fd);
153
154 // FIXME cache these somewhere else so we don't keep calling this!
155 LIBMTP_folder_t *folders;
156 LIBMTP_file_t *files;
157 folders = LIBMTP_Get_Folder_List(device);
158 files = LIBMTP_Get_Filelisting_With_Callback(device, NULL, NULL);
159
160 // convert the playlist listing to track ids
161 pl->no_tracks = trackno_spl_text_t(p);
162 IF_DEBUG() printf("%u track%s found\n", pl->no_tracks, pl->no_tracks==1?"":"s");
163 pl->tracks = malloc(sizeof(uint32_t)*(pl->no_tracks));
164 tracks_from_spl_text_t(p, pl->tracks, folders, files);
165
166 free_spl_text_t(p);
167
168 // debug: add a break since this is the top level function call
169 IF_DEBUG() printf("------------\n\n");
170 }
171
172
173 /**
174 * Push a playlist_t onto the device after converting it to a .spl format
175 *
176 * @param device mtp device pointer
177 * @param pl the LIBMTP_playlist_t to convert (pl->playlist_id will be updated
178 * with the newly created object's id)
179 * @return 0 on success, any other value means failure.
180 */
playlist_t_to_spl(LIBMTP_mtpdevice_t * device,LIBMTP_playlist_t * const pl)181 int playlist_t_to_spl(LIBMTP_mtpdevice_t *device,
182 LIBMTP_playlist_t * const pl)
183 {
184 text_t* t;
185 LIBMTP_folder_t *folders;
186 LIBMTP_file_t *files;
187 folders = LIBMTP_Get_Folder_List(device);
188 files = LIBMTP_Get_Filelisting_With_Callback(device, NULL, NULL);
189
190 char tmpname[] = "/tmp/mtp-spl2pl-XXXXXX"; // must be a var since mkstemp modifies it
191
192 IF_DEBUG() printf("pl->name='%s'\n",pl->name);
193
194 // open a file descriptor
195 int fd = mkstemp(tmpname);
196 if(fd < 0) {
197 printf("failed to make temp file for %s.spl -> %s, errno=%s\n", pl->name, tmpname, strerror(errno));
198 return -1;
199 }
200 // make sure the file will be deleted afterwards
201 if(unlink(tmpname) < 0)
202 printf("failed to delete temp file for %s.spl -> %s, errno=%s\n", pl->name, tmpname, strerror(errno));
203
204 // decide on which version of the .spl format to use
205 uint32_t ver_major;
206 uint32_t ver_minor = 0;
207 PTP_USB *ptp_usb = (PTP_USB*) device->usbinfo;
208 if(FLAG_PLAYLIST_SPL_V2(ptp_usb)) ver_major = 2;
209 else ver_major = 1; // FLAG_PLAYLIST_SPL_V1()
210
211 IF_DEBUG() printf("%u track%s\n", pl->no_tracks, pl->no_tracks==1?"":"s");
212 IF_DEBUG() printf(".spl version %d.%02d\n", ver_major, ver_minor);
213
214 // create the text for the playlist
215 spl_text_t_from_tracks(&t, pl->tracks, pl->no_tracks, ver_major, ver_minor, NULL, folders, files);
216 write_from_spl_text_t(device, fd, t);
217 free_spl_text_t(t); // done with the text
218
219 // create the file object for storing
220 LIBMTP_file_t* f = malloc(sizeof(LIBMTP_file_t));
221 f->item_id = 0;
222 f->parent_id = pl->parent_id;
223 f->storage_id = pl->storage_id;
224 f->filename = malloc(sizeof(char)*(strlen(pl->name)+5));
225 strcpy(f->filename, pl->name);
226 strcat(f->filename, ".spl"); // append suffix
227 f->filesize = lseek(fd, 0, SEEK_CUR); // file desc is currently at end of file
228 f->filetype = LIBMTP_FILETYPE_UNKNOWN;
229 f->next = NULL;
230
231 IF_DEBUG() printf("%s is %dB\n", f->filename, (int)f->filesize);
232
233 // push the playlist to the device
234 lseek(fd, 0, SEEK_SET); // reset file desc. to start of file
235 int ret = LIBMTP_Send_File_From_File_Descriptor(device, fd, f, NULL, NULL);
236 pl->playlist_id = f->item_id;
237 free(f->filename);
238 free(f);
239
240 // release the memory when we're done with it
241 close(fd);
242 // debug: add a break since this is the top level function call
243 IF_DEBUG() printf("------------\n\n");
244
245 return ret;
246 }
247
248
249
250 /**
251 * Update a playlist on the device. If only the playlist's name is being
252 * changed the pl->playlist_id will likely remain the same. An updated track
253 * list will result in the old playlist being replaced (ie: new playlist_id).
254 * NOTE: Other playlist metadata aside from playlist name and tracks are
255 * ignored.
256 *
257 * @param device mtp device pointer
258 * @param new the LIBMTP_playlist_t to convert (pl->playlist_id will be updated
259 * with the newly created object's id)
260 * @return 0 on success, any other value means failure.
261 */
update_spl_playlist(LIBMTP_mtpdevice_t * device,LIBMTP_playlist_t * const newlist)262 int update_spl_playlist(LIBMTP_mtpdevice_t *device,
263 LIBMTP_playlist_t * const newlist)
264 {
265 IF_DEBUG() printf("pl->name='%s'\n",newlist->name);
266
267 // read in the playlist of interest
268 LIBMTP_playlist_t * old = LIBMTP_Get_Playlist(device, newlist->playlist_id);
269
270 // check to see if we found it
271 if (!old)
272 return -1;
273
274 // check if the playlists match
275 int delta = 0;
276 int i;
277 if(old->no_tracks != newlist->no_tracks)
278 delta++;
279 for(i=0;i<newlist->no_tracks && delta==0;i++) {
280 if(old->tracks[i] != newlist->tracks[i])
281 delta++;
282 }
283
284 // if not, kill the playlist and replace it
285 if(delta) {
286 IF_DEBUG() printf("new tracks detected:\n");
287 IF_DEBUG() printf("delete old playlist and build a new one\n");
288 IF_DEBUG() printf(" NOTE: new playlist_id will result!\n");
289 if(LIBMTP_Delete_Object(device, old->playlist_id) != 0)
290 return -1;
291
292 IF_DEBUG() {
293 if(strcmp(old->name,newlist->name) == 0)
294 printf("name unchanged\n");
295 else
296 printf("name is changing too -> %s\n",newlist->name);
297 }
298
299 return LIBMTP_Create_New_Playlist(device, newlist);
300 }
301
302
303 // update the name only
304 if(strcmp(old->name,newlist->name) != 0) {
305 IF_DEBUG() printf("ONLY name is changing -> %s\n",newlist->name);
306 IF_DEBUG() printf("playlist_id will remain unchanged\n");
307 char* s = malloc(sizeof(char)*(strlen(newlist->name)+5));
308 strcpy(s, newlist->name);
309 strcat(s,".spl"); // FIXME check for success
310 int ret = LIBMTP_Set_Playlist_Name(device, newlist, s);
311 free(s);
312 return ret;
313 }
314
315 IF_DEBUG() printf("no change\n");
316 return 0; // nothing to be done, success
317 }
318
319
320 /**
321 * Load a file descriptor into a string.
322 *
323 * @param device a pointer to the current device.
324 * (needed for ucs2->utf8 charset conversion)
325 * @param fd the file descriptor to load
326 * @return text_t* a linked list of lines of text, id is left blank, NULL if nothing read in
327 */
read_into_spl_text_t(LIBMTP_mtpdevice_t * device,const int fd)328 static text_t* read_into_spl_text_t(LIBMTP_mtpdevice_t *device, const int fd)
329 {
330 // set MAXREAD to match STRING_BUFFER_LENGTH in unicode.h conversion function
331 const size_t MAXREAD = 1024*2;
332 char t[MAXREAD];
333 // upto 3 bytes per utf8 character, 2 bytes per ucs2 character,
334 // +1 for '\0' at end of string
335 const size_t WSIZE = MAXREAD/2*3+1;
336 char w[WSIZE];
337 char* it = t; // iterator on t
338 char* iw = w;
339 ssize_t rdcnt;
340 off_t offcnt;
341 text_t* head = NULL;
342 text_t* tail = NULL;
343 int eof = 0;
344
345 // reset file descriptor (fd) to start of file
346 offcnt = lseek(fd, 0, SEEK_SET);
347
348 while(!eof) {
349 // find the current offset in the file
350 // to allow us to determine how many bytes we read if we hit the EOF
351 // where returned rdcnt=0 from read()
352 offcnt = lseek(fd, 0, SEEK_CUR);
353 // read to refill buffer
354 // (there might be data left from an incomplete last string in t,
355 // hence start filling at it)
356 it = t; // set ptr to start of buffer
357 rdcnt = read(fd, it, sizeof(char)*MAXREAD);
358 if(rdcnt < 0)
359 printf("load_spl_fd read err %s\n", strerror(errno));
360 else if(rdcnt == 0) { // for EOF, fix rdcnt
361 if(it-t == MAXREAD)
362 printf("error -- buffer too small to read in .spl playlist entry\n");
363
364 rdcnt = lseek(fd, 0, SEEK_CUR) - offcnt;
365 eof = 1;
366 }
367
368 IF_DEBUG() printf("read buff= {%dB new, %dB old/left-over}%s\n",(int)rdcnt, (int)(iw-w), eof?", EOF":"");
369
370 // while more input bytes
371 char* it_end = t + rdcnt;
372 while(it < it_end) {
373 // copy byte, unless EOL (then replace with end-of-string \0)
374 if(*it == '\r' || *it == '\n')
375 *iw = '\0';
376 else
377 *iw = *it;
378
379 it++;
380 iw++;
381
382 // EOL -- store it
383 if( (iw-w) >= 2 && // we must have at least two bytes
384 *(iw-1) == '\0' && *(iw-2) == '\0' && // 0x0000 is end-of-string
385 // but it must be aligned such that we have an {odd,even} set of
386 // bytes since we are expecting to consume bytes two-at-a-time
387 !((iw-w)%2) ) {
388
389 // drop empty lines
390 // ... cast as a string of 2 byte characters
391 if(ucs2_strlen((uint16_t*)w) == 0) {
392 iw = w;
393 continue;
394 }
395
396 // create a new node in the list
397 if(head == NULL) {
398 head = malloc(sizeof(text_t));
399 tail = head;
400 }
401 else {
402 tail->next = malloc(sizeof(text_t));
403 tail = tail->next;
404 }
405 // fill in the data for the node
406 // ... cast as a string of 2 byte characters
407 tail->text = utf16_to_utf8(device, (uint16_t*) w);
408 iw = w; // start again
409
410 IF_DEBUG() printf("line: %s\n", tail->text);
411 }
412
413 // prevent buffer overflow
414 if(iw >= w + WSIZE) {
415 // if we ever see this error its BAD:
416 // we are dropping all the processed bytes for this line and
417 // proceeding on as if everything is okay, probably losing a track
418 // from the playlist
419 printf("ERROR %s:%u:%s(): buffer overflow! .spl line too long @ %zuB\n",
420 __FILE__, __LINE__, __func__, WSIZE);
421 iw = w; // reset buffer
422 }
423 }
424
425 // if the last thing we did was save our line, then we finished working
426 // on the input buffer and we can start fresh
427 // otherwise we need to save our partial work, if we're not quiting (eof).
428 // there is nothing special we need to do, to achieve this since the
429 // partially completed string will sit in 'w' until we return to complete
430 // the line
431
432 }
433
434 // set the next pointer at the end
435 // if there is any list
436 if(head != NULL)
437 tail->next = NULL;
438
439 // return the head of the list (NULL if no list)
440 return head;
441 }
442
443
444 /**
445 * Write a .spl text file to a file in preparation for pushing it
446 * to the device.
447 *
448 * @param fd file descriptor to write to
449 * @param p the text to output one line per string in the linked list
450 * @see playlist_t_to_spl()
451 */
write_from_spl_text_t(LIBMTP_mtpdevice_t * device,const int fd,text_t * p)452 static void write_from_spl_text_t(LIBMTP_mtpdevice_t *device,
453 const int fd,
454 text_t* p) {
455 ssize_t ret;
456 // write out BOM for utf16/ucs2 (byte order mark)
457 ret = write(fd,"\xff\xfe",2);
458 while(p != NULL) {
459 char *const t = (char*) utf8_to_utf16(device, p->text);
460 // note: 2 bytes per ucs2 character
461 const size_t len = ucs2_strlen((uint16_t*)t)*sizeof(uint16_t);
462 int i;
463
464 IF_DEBUG() {
465 printf("\nutf8=%s ",p->text);
466 for(i=0;i<strlen(p->text);i++)
467 printf("%02x ", p->text[i] & 0xff);
468 printf("\n");
469 printf("ucs2=");
470 for(i=0;i<ucs2_strlen((uint16_t*)t)*sizeof(uint16_t);i++)
471 printf("%02x ", t[i] & 0xff);
472 printf("\n");
473 }
474
475 // write: utf8 -> utf16
476 ret += write(fd, t, len);
477
478 // release the converted string
479 free(t);
480
481 // check for failures
482 if(ret < 0)
483 printf("write spl file failed: %s\n", strerror(errno));
484 else if(ret != len +2)
485 printf("write spl file wrong number of bytes ret=%d len=%d '%s'\n", (int)ret, (int)len, p->text);
486
487 // write carriage return, line feed in ucs2
488 ret = write(fd, "\r\0\n\0", 4);
489 if(ret < 0)
490 printf("write spl file failed: %s\n", strerror(errno));
491 else if(ret != 4)
492 printf("failed to write the correct number of bytes '\\n'!\n");
493
494 // fake out count (first time through has two extra bytes from BOM)
495 ret = 2;
496
497 // advance to the next line
498 p = p->next;
499 }
500 }
501
502 /**
503 * Destroy a linked-list of strings.
504 *
505 * @param p the list to destroy
506 * @see spl_to_playlist_t()
507 * @see playlist_t_to_spl()
508 */
free_spl_text_t(text_t * p)509 static void free_spl_text_t(text_t* p)
510 {
511 text_t* d;
512 while(p != NULL) {
513 d = p;
514 free(p->text);
515 p = p->next;
516 free(d);
517 }
518 }
519
520 /**
521 * Print a linked-list of strings to stdout.
522 *
523 * @param p the list to print
524 */
print_spl_text_t(text_t * p)525 static void print_spl_text_t(text_t* p)
526 {
527 while(p != NULL) {
528 printf("%s\n",p->text);
529 p = p->next;
530 }
531 }
532
533 /**
534 * Count the number of tracks in this playlist. A track will be counted as
535 * such if the line starts with a leading slash.
536 *
537 * @param p the text to search
538 * @return number of tracks in the playlist
539 * @see spl_to_playlist_t()
540 */
trackno_spl_text_t(text_t * p)541 static uint32_t trackno_spl_text_t(text_t* p) {
542 uint32_t c = 0;
543 while(p != NULL) {
544 if(p->text[0] == '\\' ) c++;
545 p = p->next;
546 }
547
548 return c;
549 }
550
551 /**
552 * Find the track ids for this playlist's files.
553 * (ie: \Music\song.mp3 -> 12345)
554 *
555 * @param p the text to search
556 * @param tracks returned list of track id's for the playlist_t, must be large
557 * enough to accomodate all the tracks as reported by
558 * trackno_spl_text_t()
559 * @param folders the folders list for the device
560 * @param fiels the files list for the device
561 * @see spl_to_playlist_t()
562 */
tracks_from_spl_text_t(text_t * p,uint32_t * tracks,LIBMTP_folder_t * folders,LIBMTP_file_t * files)563 static void tracks_from_spl_text_t(text_t* p,
564 uint32_t* tracks,
565 LIBMTP_folder_t* folders,
566 LIBMTP_file_t* files)
567 {
568 uint32_t c = 0;
569 while(p != NULL) {
570 if(p->text[0] == '\\' ) {
571 tracks[c] = discover_id_from_filepath(p->text, folders, files);
572 IF_DEBUG()
573 printf("track %d = %s (%u)\n", c+1, p->text, tracks[c]);
574 c++;
575 }
576 p = p->next;
577 }
578 }
579
580
581 /**
582 * Find the track names (including path) for this playlist's track ids.
583 * (ie: 12345 -> \Music\song.mp3)
584 *
585 * @param p the text to search
586 * @param tracks list of track id's to look up
587 * @param folders the folders list for the device
588 * @param fiels the files list for the device
589 * @see playlist_t_to_spl()
590 */
spl_text_t_from_tracks(text_t ** p,uint32_t * tracks,const uint32_t trackno,const uint32_t ver_major,const uint32_t ver_minor,char * dnse,LIBMTP_folder_t * folders,LIBMTP_file_t * files)591 static void spl_text_t_from_tracks(text_t** p,
592 uint32_t* tracks,
593 const uint32_t trackno,
594 const uint32_t ver_major,
595 const uint32_t ver_minor,
596 char* dnse,
597 LIBMTP_folder_t* folders,
598 LIBMTP_file_t* files)
599 {
600
601 // HEADER
602 text_t* c = NULL;
603 append_text_t(&c, "SPL PLAYLIST");
604 *p = c; // save the top of the list!
605
606 char vs[14]; // "VERSION 2.00\0"
607 sprintf(vs,"VERSION %d.%02d",ver_major,ver_minor);
608
609 append_text_t(&c, vs);
610 append_text_t(&c, "");
611
612 // TRACKS
613 int i;
614 char* f;
615 for(i=0;i<trackno;i++) {
616 discover_filepath_from_id(&f, tracks[i], folders, files);
617
618 if(f != NULL) {
619 append_text_t(&c, f);
620 IF_DEBUG()
621 printf("track %d = %s (%u)\n", i+1, f, tracks[i]);
622 }
623 else
624 printf("failed to find filepath for track=%d\n", tracks[i]);
625 }
626
627 // FOOTER
628 append_text_t(&c, "");
629 append_text_t(&c, "END PLAYLIST");
630 if(ver_major == 2) {
631 append_text_t(&c, "");
632 append_text_t(&c, "myDNSe DATA");
633 if(dnse != NULL) {
634 append_text_t(&c, dnse);
635 }
636 else {
637 append_text_t(&c, "");
638 append_text_t(&c, "");
639 }
640 append_text_t(&c, "END myDNSe");
641 }
642
643 c->next = NULL;
644
645 // debug
646 IF_DEBUG() {
647 printf(".spl playlist:\n");
648 print_spl_text_t(*p);
649 }
650 }
651
652
653 /**
654 * Find the track names (including path) given a fileid
655 * (ie: 12345 -> \Music\song.mp3)
656 *
657 * @param p returns the file path (ie: \Music\song.mp3),
658 * (*p) == NULL if the look up fails
659 * @param track track id to look up
660 * @param folders the folders list for the device
661 * @param files the files list for the device
662 * @see spl_text_t_from_tracks()
663 */
664
665 // returns p = NULL on failure, else the filepath to the track including track name, allocated as a correct length string
discover_filepath_from_id(char ** p,uint32_t track,LIBMTP_folder_t * folders,LIBMTP_file_t * files)666 static void discover_filepath_from_id(char** p,
667 uint32_t track,
668 LIBMTP_folder_t* folders,
669 LIBMTP_file_t* files)
670 {
671 // fill in a string from the right side since we don't know the root till the end
672 const int M = 1024;
673 char w[M];
674 char* iw = w + M; // iterator on w
675
676 // in case of failure return NULL string
677 *p = NULL;
678
679
680 // find the right file
681 while(files != NULL && files->item_id != track) {
682 files = files->next;
683 }
684 // if we didn't find a matching file, abort
685 if(files == NULL)
686 return;
687
688 // stuff the filename into our string
689 // FIXME: check for string overflow before it occurs
690 iw = iw - (strlen(files->filename) +1); // leave room for '\0' at the end
691 strcpy(iw,files->filename);
692
693 // next follow the directories to the root
694 // prepending folders to the path as we go
695 uint32_t id = files->parent_id;
696 char* f = NULL;
697 while(id != 0) {
698 find_folder_name(folders, &id, &f);
699 if(f == NULL) return; // fail if the next part of the path couldn't be found
700 iw = iw - (strlen(f) +1);
701 // FIXME: check for string overflow before it occurs
702 strcpy(iw, f);
703 iw[strlen(f)] = '\\';
704 free(f);
705 }
706
707 // prepend a slash
708 iw--;
709 iw[0] = '\\';
710
711 // now allocate a string of the right length to be returned
712 *p = strdup(iw);
713 }
714
715
716 /**
717 * Find the track id given a track's name (including path)
718 * (ie: \Music\song.mp3 -> 12345)
719 *
720 * @param s file path to look up (ie: \Music\song.mp3),
721 * (*p) == NULL if the look up fails
722 * @param folders the folders list for the device
723 * @param files the files list for the device
724 * @return track id, 0 means failure
725 * @see tracks_from_spl_text_t()
726 */
discover_id_from_filepath(const char * s,LIBMTP_folder_t * folders,LIBMTP_file_t * files)727 static uint32_t discover_id_from_filepath(const char* s, LIBMTP_folder_t* folders, LIBMTP_file_t* files)
728 {
729 // abort if this isn't a path
730 if(s[0] != '\\')
731 return 0;
732
733 int i;
734 uint32_t id = 0;
735 char* sc = strdup(s);
736 char* sci = sc +1; // iterator
737 // skip leading slash in path
738
739 // convert all \ to \0
740 size_t len = strlen(s);
741 for(i=0;i<len;i++) {
742 if(sc[i] == '\\') {
743 sc[i] = '\0';
744 }
745 }
746
747 // now for each part of the string, find the id
748 while(sci != sc + len +1) {
749 // if its the last part of the string, its the filename
750 if(sci + strlen(sci) == sc + len) {
751
752 while(files != NULL) {
753 // check parent matches id and name matches sci
754 if( (files->parent_id == id) &&
755 (strcmp(files->filename, sci) == 0) ) { // found it!
756 id = files->item_id;
757 break;
758 }
759 files = files->next;
760 }
761 }
762 else { // otherwise its part of the directory path
763 id = find_folder_id(folders, id, sci);
764 }
765
766 // move to next folder/file
767 sci += strlen(sci) +1;
768 }
769
770 // release our copied string
771 free(sc);
772
773 // FIXME check that we actually have a file
774
775 return id;
776 }
777
778
779
780 /**
781 * Find the folder name given the folder's id.
782 *
783 * @param folders the folders list for the device
784 * @param id the folder_id to look up, returns the folder's parent folder_id
785 * @param name returns the name of the folder or NULL on failure
786 * @see discover_filepath_from_id()
787 */
find_folder_name(LIBMTP_folder_t * folders,uint32_t * id,char ** name)788 static void find_folder_name(LIBMTP_folder_t* folders, uint32_t* id, char** name)
789 {
790
791 // FIXME this function is exactly LIBMTP_Find_Folder
792
793 LIBMTP_folder_t* f = LIBMTP_Find_Folder(folders, *id);
794 if(f == NULL) {
795 *name = NULL;
796 }
797 else { // found it!
798 *name = strdup(f->name);
799 *id = f->parent_id;
800 }
801 }
802
803
804 /**
805 * Find the folder id given the folder's name and parent id.
806 *
807 * @param folders the folders list for the device
808 * @param parent the folder's parent's id
809 * @param name the name of the folder
810 * @return the folder_id or 0 on failure
811 * @see discover_filepath_from_id()
812 */
find_folder_id(LIBMTP_folder_t * folders,uint32_t parent,char * name)813 static uint32_t find_folder_id(LIBMTP_folder_t* folders, uint32_t parent, char* name) {
814
815 if(folders == NULL)
816 return 0;
817
818 // found it!
819 else if( (folders->parent_id == parent) &&
820 (strcmp(folders->name, name) == 0) )
821 return folders->folder_id;
822
823 // no luck so far, search both siblings and children
824 else {
825 uint32_t id = 0;
826
827 if(folders->sibling != NULL)
828 id = find_folder_id(folders->sibling, parent, name);
829 if( (id == 0) && (folders->child != NULL) )
830 id = find_folder_id(folders->child, parent, name);
831
832 return id;
833 }
834 }
835
836
837 /**
838 * Append a string to a linked-list of strings.
839 *
840 * @param t the list-of-strings, returns with the added string
841 * @param s the string to append
842 * @see spl_text_t_from_tracks()
843 */
append_text_t(text_t ** t,char * s)844 static void append_text_t(text_t** t, char* s)
845 {
846 if(*t == NULL) {
847 *t = malloc(sizeof(text_t));
848 }
849 else {
850 (*t)->next = malloc(sizeof(text_t));
851 (*t) = (*t)->next;
852 }
853 (*t)->text = strdup(s);
854 }
855