ref: 80067ce095cccd88ab3212819445a930355cd4bb
dir: /id3.c/
#include <u.h>
#include <libc.h>
#include <thread.h>
#include <draw.h>
#include "dat.h"
#include "fncs.h"
/* If we can't read the tag, at least default to file names */
void
fillnotag(ID3v1 *id, int fd)
{
char buf[512];
char *slash, *dot;
fd2path(fd, buf, 512);
slash = strrchr(buf, '/');
dot = strrchr(buf, '.');
*dot = '\0';
id->title = runesmprint("%s", slash+1);
}
/* The following function is stolen from https://git.sr.ht/~ft/libtags/ */
#define synchsafe(d) (uint)(((d)[0]&127)<<21 | ((d)[1]&127)<<14 | ((d)[2]&127)<<7 | ((d)[3]&127)<<0)
#define beuint(d) (uint)((d)[0]<<24 | (d)[1]<<16 | (d)[2]<<8 | (d)[3]<<0)
ID3v1*
readid3v2(int fd)
{
int sz, framesz;
int ver;
char nu[8];
uchar d[10];
uchar buf[256];
ID3v1 *id;
read(fd, d, sizeof d);
ver = d[3];
sz = synchsafe(&d[6]);
if(ver == 2 && (d[5] & (1<<6)) != 0) /* compression */
return nil;
if(ver > 2){
if((d[5] & (1<<4)) != 0) /* footer */
sz -= 10;
if((d[5] & (1<<6)) != 0){ /* we don't do extended header */
return nil;
}
}
framesz = (ver >= 3) ? 10 : 6;
id = emalloc(sizeof(ID3v1));
id->title = id->album = nil;
for(; sz > framesz;){
int tsz;
read(fd, d, framesz);
sz -= framesz;
if(memcmp(d, "\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00", framesz) == 0)
break;
if(ver >= 3){
tsz = (ver == 3) ? beuint(&d[4]) : synchsafe(&d[4]);
if(tsz < 0 || tsz > sz)
break;
d[4] = 0;
if((d[9] & 0x0c) != 0){ /* compression & encryption */
return nil;
}
}else{
tsz = beuint(&d[3]) >> 8;
if(tsz > sz)
return nil;
d[3] = 0;
}
sz -= tsz;
if(memcmp(d, "TIT2", 4) == 0){
read(fd, nu, 1);
switch(nu[0]){
case 3:
id->title = emalloc(sizeof(Rune)*tsz);
read(fd, buf, tsz-1);
runesprint(id->title, "%s", (char*)buf);
}
}else if(memcmp(d, "TALB", 4) == 0){
read(fd, nu, 1);
switch(nu[0]){
case 3:
id->album = emalloc(sizeof(Rune)*tsz);
read(fd, buf, tsz-1);
runesprint(id->album, "%s", (char*)buf);
}
}
}
if(id->title == nil || id->album == nil)
return nil;
return id;
}
ID3v1*
readid3(int fd)
{
Dir *d;
ID3v1 *id;
char buf[128];
char fieldbuf[31];
int length;
d = dirfstat(fd);
if(d == nil)
return nil;
length = d->length;
free(d);
if(pread(fd, buf, 128, length-128) != 128)
return nil;
id = emalloc(sizeof(ID3v1));
if(memcmp(buf, "TAG", 3) != 0){
fillnotag(id, fd);
return id;
}
memcpy(fieldbuf, buf+3, 30);
fieldbuf[30] = '\0';
id->title = emalloc(sizeof(Rune) * 31);
runesprint(id->title, "%s", fieldbuf);
memcpy(fieldbuf, buf+33, 30);
id->artist = emalloc(sizeof(Rune) * 31);
runesprint(id->artist, "%s", fieldbuf);
memcpy(fieldbuf, buf+63, 30);
id->album = emalloc(sizeof(Rune) * 31);
runesprint(id->album, "%s", fieldbuf);
memcpy(fieldbuf, buf+93, 4);
fieldbuf[4] = '\0';
id->year = atoi(fieldbuf);
memcpy(fieldbuf, buf+97, 30);
fieldbuf[30] = '\0';
id->comment = emalloc(sizeof(Rune) * 31);
runesprint(id->comment, "%s", fieldbuf);
id->genre = buf[127];
return id;
}
void
destroyid3(ID3v1 *id)
{
free(id->title);
free(id->artist);
free(id->album);
free(id->comment);
free(id);
}