From d7c8bee088cdb2d2225475cf5a30ab802bfd12a9 Mon Sep 17 00:00:00 2001 From: Antoine Jacquet Date: Sat, 14 Feb 2004 00:00:00 +0100 Subject: [PATCH] version 0.2 * more compatible with ID3 headers (some bug fixes) * faster to parse MP3 files (nicer buffering) * end of line are in DOS format when using the --windows flag (thank you Lukasz Wiechec) --- COPYING | 0 Makefile | 0 README | 4 +- fapg.c | 774 ++++++++++++++++++++++++++++--------------------------- 4 files changed, 395 insertions(+), 383 deletions(-) mode change 100755 => 100644 COPYING mode change 100755 => 100644 Makefile mode change 100755 => 100644 README mode change 100755 => 100644 fapg.c diff --git a/COPYING b/COPYING old mode 100755 new mode 100644 diff --git a/Makefile b/Makefile old mode 100755 new mode 100644 diff --git a/README b/README old mode 100755 new mode 100644 index 4913345..29bec50 --- a/README +++ b/README @@ -1,4 +1,4 @@ -FAPG 0.1 (Fast Audio Playlist Generator) +FAPG 0.2 (Fast Audio Playlist Generator) site: http://royale.zerezo.com/fapg/ mail: royale@zerezo.com @@ -20,4 +20,4 @@ links : http://geek.scorpiorising.ca/scripts.html http://id3lib.sourceforge.net/ http://id3.org/ -http://www.xiph.org/ogg/vorbis/docs.html \ No newline at end of file +http://www.xiph.org/ogg/vorbis/docs.html diff --git a/fapg.c b/fapg.c old mode 100755 new mode 100644 index 4dc2cc1..9e6063a --- a/fapg.c +++ b/fapg.c @@ -1,381 +1,393 @@ -/* - * FAPG 0.1 released under GPL - * http://royale.zerezo.com/fapg/ - */ - -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#define MAX 10240 - -int debug=0; -int format=0; /* 0 = m3u ; 1 = pls ; 2 = html */ -unsigned char *prefix=""; -int recursive=0; -int separator='/'; -int skip=0; -int windows=0; -unsigned char buffer[MAX]; - -int counter=0; - -unsigned char artist[1024]; -unsigned char title[1024]; -int duration; - -unsigned char unix2dos[256]={0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,33,70,35,36,37,38,39,40,41,82,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,84,59,36,61,65,71,64,65,66,67,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,84,85,86,87,88,89,90,91,36,93,94,95,96,97,98,99,100,101,102,103,104,105,106,107,108,109,110,111,112,113,114,115,116,117,118,119,120,121,122,123,36,125,126,127,199,252,233,226,228,224,229,231,234,235,232,239,238,236,196,197,201,230,198,244,246,242,251,249,255,214,220,248,163,216,215,131,225,237,243,250,241,209,170,186,191,174,172,189,188,161,171,187,166,166,166,166,166,193,194,192,169,166,166,43,43,162,165,43,43,45,45,43,45,43,227,195,43,43,45,45,166,45,43,164,240,208,202,203,200,105,205,206,207,43,43,166,220,166,204,175,211,223,212,210,245,213,181,254,222,218,219,217,253,221,175,180,173,177,61,190,182,167,247,184,176,168,183,185,179,178,166,160}; - -void usage() -{ - fprintf(stderr,"Usage >> fapg [-b|--backslash] [-d|--debug] [-f|--format=m3u|pls|html] [-o|--output=/path/to/file.m3u] [-p|--prefix=/the/prefix] [-r|--recursive] [-w|--windows] /path/to/mp3/dir1 [/path/to/mp3/dir2 ...]\n"); - exit(1); -} - -void parse_options(int argc,char **argv) -{ - static char const short_options[]="bdf:o:p:rw"; - static struct option long_options[]= - { - {"backslash",no_argument,NULL,'b'}, - {"debug",no_argument,NULL,'d'}, - {"format",required_argument,NULL,'f'}, - {"output",required_argument,NULL,'o'}, - {"prefix",required_argument,NULL,'p'}, - {"recursive",no_argument,NULL,'r'}, - {"windows",no_argument,NULL,'w'} - }; - int c; - int option_index=0; - while ((c=getopt_long(argc,argv,short_options,long_options,&option_index))!=-1) - { - switch(c) - { - case 'b': - separator='\\'; - break; - case 'd': - debug=1; - break; - case 'f': - if (strcmp(optarg,"m3u")==0) format=0; else - if (strcmp(optarg,"pls")==0) format=1; else - if (strcmp(optarg,"html")==0) format=2; else - usage(); - break; - case 'o': - close(1); - if (fopen(optarg,"w")==NULL) { fprintf(stderr,"Error >> unable to open output file : %s\n",optarg); exit(2); } - break; - case 'p': - prefix=malloc(strlen(optarg)+1); - strcpy(prefix,optarg); - break; - case 'r': - recursive=1; - break; - case 'w': - windows=1; - break; - default: - usage(); - } - } -} - -void parse_mp3(unsigned char *file) -{ - int bitrates[2][3][15]= - {{{0, 32, 64, 96, 128, 160, 192, 224, 256, 288, 320, 352, 384, 416, 448}, - {0, 32, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224, 256, 320, 384}, - {0, 32, 40, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224, 256, 320}}, - {{0, 32, 48, 56, 64, 80, 96, 112, 128, 144, 160, 176, 192, 224, 256}, - {0, 8, 16, 24, 32, 40, 48, 56, 64, 80, 96, 112, 128, 144, 160}, - {0, 8, 16, 24, 32, 40, 48, 56, 64, 80, 96, 112, 128, 144, 160}}}; - FILE *fic; - unsigned char *c; - int lus; - - if (debug) fprintf(stderr,"Debug >> parsing mp3 : %s\n",file); - - /* read header */ - if ((fic=fopen(file,"r"))==NULL) { fprintf(stderr,"Warning >> can't open file : %s\n",file); return; } - lus=fread(buffer,1,MAX,fic); - c=buffer; - - /* try ID3v2 */ - if (buffer[0]=='I' && buffer[1]=='D' && buffer[2]=='3') - { - int size; - int version; - version=*(buffer+3); - if (version<2 || version>4) - fprintf(stderr,"Warning >> ID3 v2.%d not implemented ! trying anyway : %s\n",version,file); - if (*(buffer+5)!=0) - fprintf(stderr,"Warning >> specials headers not implemented (%d) ! trying anyway : %s\n",*(buffer+5),file); - c=buffer+6; - size=(*c<<24)+(*(c+1)<<16)+(*(c+2)<<8)+(*(c+3)); - if (size>lus) size=lus; - c+=4; - if (version==2) while (c>3&1); - lay=4-(*(c+1)>>1&3); - bitrate_index=*(c+2)>>4&0xF; - if (version>=1 && version<=2 && lay-1>=0 && lay-1<=2 && bitrate_index>=0 && bitrate_index<=14) - bitrate=bitrates[version-1][lay-1][bitrate_index]; - else - bitrate=0; - if (bitrate!=0) - { - fseek(fic,0,SEEK_END); - duration=(ftell(fic)+buffer-c)/125/bitrate; - } - else - duration=0; - break; - } - c++; - } - - /* try ID3v1 */ - if (strlen(artist)==0 && strlen(title)==0) - { - fseek(fic,-128,SEEK_END); - lus=fread(buffer,1,128,fic); - if (lus==128 && buffer[0]=='T' && buffer[1]=='A' && buffer[2]=='G') - { - strncpy(title,buffer+3,30); - title[30]='\0'; - c=title+29; - while (c>title && *c==' ') *(c--)='\0'; - strncpy(artist,buffer+33,30); - artist[30]='\0'; - c=artist+29; - while (c>artist && *c==' ') *(c--)='\0'; - } - } - - fclose(fic); -} - -void parse_ogg(unsigned char *file) -{ - FILE *fic; - unsigned char *c; - int lus; - int sample_rate; - int samples; - - if (debug) fprintf(stderr,"Debug >> parsing ogg : %s\n",file); - - /* read header */ - if ((fic=fopen(file,"r"))==NULL) { fprintf(stderr,"Warning >> can't open file : %s\n",file); return; } - lus=fread(buffer,1,MAX,fic); - - /* try Ogg */ - if (buffer[0]!='O' && buffer[1]!='g' && buffer[2]!='g') - { - fprintf(stderr,"Warning >> not a Ogg header : %s\n",file); - return; - } - - c=buffer+0x28; - sample_rate=(*c)+(*(c+1)<<8)+(*(c+2)<<16)+(*(c+3)<<24); - - while (cbuffer) c--; - if (c!=buffer) - { - c+=6; - samples=(*c)+(*(c+1)<<8)+(*(c+2)<<16)+(*(c+3)<<24); - duration=samples/sample_rate; - } - - fclose(fic); -} - -void parse_directory(unsigned char *path) -{ - int i,n; - struct dirent **namelist; - unsigned char newpath[PATH_MAX]; - unsigned char *c; - struct stat infos; - - void print_faketitle() - { - c=newpath+strlen(newpath); - while (c>newpath && *c!='/') c--; - while (c> parsing directory : %s\n",path); - if ((n=scandir(path,&namelist,0,alphasort))<0) { fprintf(stderr,"Warning >> can't open directory : %s\n",path); return; } - for (i=0;id_name); - if (stat(newpath,&infos)!=0) { fprintf(stderr,"Warning >> can't stat file : %s\n",newpath); continue; } - if (recursive && S_ISDIR(infos.st_mode) && strcmp(namelist[i]->d_name,".")!=0 && strcmp(namelist[i]->d_name,"..")!=0) parse_directory(newpath); - if (S_ISREG(infos.st_mode)) - { - unsigned char ext[5]; - int j; - for (j=0;j<5;j++) ext[j]=tolower(namelist[i]->d_name[strlen(namelist[i]->d_name)-4+j]); - artist[0]='\0'; - title[0]='\0'; - duration=-2; - if (strcmp(".mp3",ext)==0) { duration=-1; parse_mp3(newpath); } - if (strcmp(".ogg",ext)==0) { duration=-1; parse_ogg(newpath); } - if (strcmp(".wav",ext)==0) { duration=-1; /* parse_wav(newpath); */ } - - if (duration!=-2) /* is it an audio file ? */ - { - counter++; - switch (format) - { - case 0: - if (duration!=-1) - { - printf("#EXTINF:%d,",duration); - if (strlen(artist)==0 && strlen(title)==0) print_faketitle(); - else printf("%s - %s",artist,title); - putchar('\n'); - } - print_path(); - putchar('\n'); - break; - case 1: - printf("File%d=",counter); - print_path(); - putchar('\n'); - printf("Title%d=",counter); - if (strlen(artist)==0 && strlen(title)==0) print_faketitle(); - else printf("%s - %s",artist,title); - putchar('\n'); - if (duration!=-1) printf("Length%d=%d\n",counter,duration); - break; - case 2: - printf("%d%s%s",counter,artist,title); - if (duration==-1) printf("?"); else printf("%d:%s%d",duration/60,duration%60<10?"0":"",duration%60); - printf("\n"); - break; - } - } - } - free(namelist[i]); - } - free(namelist); -} - -int main(int argc,char **argv) -{ - parse_options(argc,argv); - if (optind==argc) usage(); - switch (format) - { - case 0: - printf("#EXTM3U\n"); - break; - case 1: - printf("[playlist]\n"); - break; - case 2: - printf("\n\n\n\n\nPlaylist generated by FAPG 0.1\n\n\n\n\n\n\n

Playlist

\n\n\n\n"); - break; - } - for (;optind\n\n

Playlist generated by FAPG 0.1

\n\n\n\n"); - break; - } - exit(0); -} +/* + * FAPG 0.2 released under GPL + * http://royale.zerezo.com/fapg/ + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#define MP3_BASE 1024 +#define OGG_BASE 1024*10 +#define MAX 1024*200 /* 200ko for ID3 with JPEG images in it */ + +int debug=0; +int format=0; /* 0 = m3u ; 1 = pls ; 2 = html */ +unsigned char *prefix=""; +int recursive=0; +int separator='/'; +int skip=0; +int windows=0; +unsigned char *eol="\n"; +unsigned char buffer[MAX]; + +int counter=0; + +unsigned char artist[1024]; +unsigned char title[1024]; +int duration; + +unsigned char unix2dos[256]={0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,33,70,35,36,37,38,39,40,41,82,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,84,59,36,61,65,71,64,65,66,67,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,84,85,86,87,88,89,90,91,36,93,94,95,96,97,98,99,100,101,102,103,104,105,106,107,108,109,110,111,112,113,114,115,116,117,118,119,120,121,122,123,36,125,126,127,199,252,233,226,228,224,229,231,234,235,232,239,238,236,196,197,201,230,198,244,246,242,251,249,255,214,220,248,163,216,215,131,225,237,243,250,241,209,170,186,191,174,172,189,188,161,171,187,166,166,166,166,166,193,194,192,169,166,166,43,43,162,165,43,43,45,45,43,45,43,227,195,43,43,45,45,166,45,43,164,240,208,202,203,200,105,205,206,207,43,43,166,220,166,204,175,211,223,212,210,245,213,181,254,222,218,219,217,253,221,175,180,173,177,61,190,182,167,247,184,176,168,183,185,179,178,166,160}; + +void usage() +{ + fprintf(stderr,"Usage >> fapg [-b|--backslash] [-d|--debug] [-f|--format=m3u|pls|html] [-o|--output=/path/to/file.m3u] [-p|--prefix=/the/prefix] [-r|--recursive] [-w|--windows] /path/to/mp3/dir1 [/path/to/mp3/dir2 ...]\n"); + exit(1); +} + +void parse_options(int argc,char **argv) +{ + static char const short_options[]="bdf:o:p:rw"; + static struct option long_options[]= + { + {"backslash",no_argument,NULL,'b'}, + {"debug",no_argument,NULL,'d'}, + {"format",required_argument,NULL,'f'}, + {"output",required_argument,NULL,'o'}, + {"prefix",required_argument,NULL,'p'}, + {"recursive",no_argument,NULL,'r'}, + {"windows",no_argument,NULL,'w'} + }; + int c; + int option_index=0; + while ((c=getopt_long(argc,argv,short_options,long_options,&option_index))!=-1) + { + switch(c) + { + case 'b': + separator='\\'; + break; + case 'd': + debug=1; + break; + case 'f': + if (strcmp(optarg,"m3u")==0) format=0; else + if (strcmp(optarg,"pls")==0) format=1; else + if (strcmp(optarg,"html")==0) format=2; else + usage(); + break; + case 'o': + close(1); + if (fopen(optarg,"w")==NULL) { fprintf(stderr,"Error >> unable to open output file : %s\n",optarg); exit(2); } + break; + case 'p': + prefix=malloc(strlen(optarg)+1); + strcpy(prefix,optarg); + break; + case 'r': + recursive=1; + break; + case 'w': + windows=1; + eol="\r\n"; + break; + default: + usage(); + } + } +} + +void parse_mp3(unsigned char *file) +{ + int bitrates[2][3][15]= + {{{0, 32, 64, 96, 128, 160, 192, 224, 256, 288, 320, 352, 384, 416, 448}, + {0, 32, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224, 256, 320, 384}, + {0, 32, 40, 48, 56, 64, 80, 96, 112, 128, 160, 192, 224, 256, 320}}, + {{0, 32, 48, 56, 64, 80, 96, 112, 128, 144, 160, 176, 192, 224, 256}, + {0, 8, 16, 24, 32, 40, 48, 56, 64, 80, 96, 112, 128, 144, 160}, + {0, 8, 16, 24, 32, 40, 48, 56, 64, 80, 96, 112, 128, 144, 160}}}; + FILE *fic; + unsigned char *c; + int lus; + + if (debug) fprintf(stderr,"Debug >> parsing mp3 : %s\n",file); + + /* read header */ + if ((fic=fopen(file,"r"))==NULL) { fprintf(stderr,"Warning >> can't open file : %s\n",file); return; } + lus=fread(buffer,1,MP3_BASE,fic); + c=buffer; + + /* try ID3v2 */ + if (buffer[0]=='I' && buffer[1]=='D' && buffer[2]=='3') + { + int size; + int version; + version=*(buffer+3); + if (version<2 || version>4) + fprintf(stderr,"Warning >> ID3 v2.%d not implemented ! trying anyway : %s\n",version,file); + if (*(buffer+5)!=0) + fprintf(stderr,"Warning >> specials headers not implemented (%d) ! trying anyway : %s\n",*(buffer+5),file); + c=buffer+6; + size=(*c<<21)+(*(c+1)<<14)+(*(c+2)<<7)+(*(c+3)); + /* read more header */ + if (size+lus>MAX) + { + lus+=fread(buffer+lus,1,MAX-lus,fic); + fprintf(stderr,"Warning >> ID3 header is huge (%d bytes) ! trying anyway : %s\n",size,file); + } + else + lus+=fread(buffer+lus,1,size,fic); + if (size>lus) size=lus; + c+=4; + if (version==2) while (c>3&1); + lay=4-(*(c+1)>>1&3); + bitrate_index=*(c+2)>>4&0xF; + if (version>=1 && version<=2 && lay-1>=0 && lay-1<=2 && bitrate_index>=0 && bitrate_index<=14) + bitrate=bitrates[version-1][lay-1][bitrate_index]; + else + bitrate=0; + if (bitrate!=0) + { + fseek(fic,0,SEEK_END); + duration=(ftell(fic)+buffer-c)/125/bitrate; + } + else + duration=0; + break; + } + c++; + } + + /* try ID3v1 */ + if (strlen(artist)==0 && strlen(title)==0) + { + fseek(fic,-128,SEEK_END); + lus=fread(buffer,1,128,fic); + if (lus==128 && buffer[0]=='T' && buffer[1]=='A' && buffer[2]=='G') + { + strncpy(title,buffer+3,30); + title[30]='\0'; + c=title+29; + while (c>title && *c==' ') *(c--)='\0'; + strncpy(artist,buffer+33,30); + artist[30]='\0'; + c=artist+29; + while (c>artist && *c==' ') *(c--)='\0'; + } + } + + fclose(fic); +} + +void parse_ogg(unsigned char *file) +{ + FILE *fic; + unsigned char *c; + int lus; + int sample_rate; + int samples; + + if (debug) fprintf(stderr,"Debug >> parsing ogg : %s\n",file); + + /* read header */ + if ((fic=fopen(file,"r"))==NULL) { fprintf(stderr,"Warning >> can't open file : %s\n",file); return; } + lus=fread(buffer,1,OGG_BASE,fic); + + /* try Ogg */ + if (buffer[0]!='O' && buffer[1]!='g' && buffer[2]!='g') + { + fprintf(stderr,"Warning >> not a Ogg header : %s\n",file); + return; + } + + c=buffer+0x28; + sample_rate=(*c)+(*(c+1)<<8)+(*(c+2)<<16)+(*(c+3)<<24); + + while (cbuffer) c--; + if (c!=buffer) + { + c+=6; + samples=(*c)+(*(c+1)<<8)+(*(c+2)<<16)+(*(c+3)<<24); + duration=samples/sample_rate; + } + + fclose(fic); +} + +void parse_directory(unsigned char *path) +{ + int i,n; + struct dirent **namelist; + unsigned char newpath[PATH_MAX]; + unsigned char *c; + struct stat infos; + + void print_faketitle() + { + c=newpath+strlen(newpath); + while (c>newpath && *c!='/') c--; + while (c> parsing directory : %s\n",path); + if ((n=scandir(path,&namelist,0,alphasort))<0) { fprintf(stderr,"Warning >> can't open directory : %s\n",path); return; } + for (i=0;id_name); + if (stat(newpath,&infos)!=0) { fprintf(stderr,"Warning >> can't stat file : %s\n",newpath); continue; } + if (recursive && S_ISDIR(infos.st_mode) && strcmp(namelist[i]->d_name,".")!=0 && strcmp(namelist[i]->d_name,"..")!=0) parse_directory(newpath); + if (S_ISREG(infos.st_mode)) + { + unsigned char ext[5]; + int j; + for (j=0;j<5;j++) ext[j]=tolower(namelist[i]->d_name[strlen(namelist[i]->d_name)-4+j]); + artist[0]='\0'; + title[0]='\0'; + duration=-2; + if (strcmp(".mp3",ext)==0) { duration=-1; parse_mp3(newpath); } + if (strcmp(".ogg",ext)==0) { duration=-1; parse_ogg(newpath); } + if (strcmp(".wav",ext)==0) { duration=-1; /* parse_wav(newpath); */ } + + if (duration!=-2) /* is it an audio file ? */ + { + counter++; + switch (format) + { + case 0: + if (duration!=-1) + { + printf("#EXTINF:%d,",duration); + if (strlen(artist)==0 && strlen(title)==0) print_faketitle(); + else printf("%s - %s",artist,title); + printf("%s",eol); + } + print_path(); + printf("%s",eol); + break; + case 1: + printf("File%d=",counter); + print_path(); + printf("%s",eol); + printf("Title%d=",counter); + if (strlen(artist)==0 && strlen(title)==0) print_faketitle(); + else printf("%s - %s",artist,title); + printf("%s",eol); + if (duration!=-1) printf("Length%d=%d%s",counter,duration,eol); + break; + case 2: + printf("
%s",eol); + break; + } + } + } + free(namelist[i]); + } + free(namelist); +} + +int main(int argc,char **argv) +{ + parse_options(argc,argv); + if (optind==argc) usage(); + switch (format) + { + case 0: + printf("#EXTM3U%s",eol); + break; + case 1: + printf("[playlist]%s,eol"); + break; + case 2: + printf("%s%s%s%s%sPlaylist generated by FAPG 0.2%s%s%s%s%s%s%s

Playlist

%s%s
EntryArtistTitleLength
%d%s%s",counter,artist,title); + if (duration==-1) printf("?"); else printf("%d:%s%d",duration/60,duration%60<10?"0":"",duration%60); + printf("
%s%s",eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol,eol); + break; + } + for (;optind%s%s

Playlist generated by FAPG 0.2

%s%s%s%s",eol,eol,eol,eol,eol,eol); + break; + } + exit(0); +} -- 2.20.1
EntryArtistTitleLength