ref: a6e5d4bae6075c741a39fcba62a365d9dffaed93
dir: /sys/src/cmd/audio/zuke/icy.c/
#include <u.h> #include <libc.h> #include <bio.h> #include <thread.h> #include "plist.h" #include "icy.h" typedef struct Icyaux Icyaux; struct Icyaux { Channel *newtitle; int outfd; int metaint; }; static long Breadn(Biobufhdr *bp, void *addr, long nbytes) { long n, r; u8int *p; for(n = 0, p = addr; n < nbytes; n += r, p += r){ if((r = Bread(bp, p, nbytes-n)) < 1) break; } return n; } static void icyproc(void *b_) { char *p, *s, *e; Biobuf *b, *out; int n, r, sz; Icyaux *aux; threadsetname("icy/pull"); b = b_; aux = b->aux; out = Bfdopen(aux->outfd, OWRITE); sz = aux->metaint > 4096 ? aux->metaint : 4096; p = malloc(sz); for(;;){ r = Breadn(b, p, aux->metaint > 0 ? aux->metaint : sz); if(r < 1 || Bwrite(out, p, r) != r) break; if(aux->metaint > 0){ if((n = 16*Bgetc(b)) < 0) break; if(Breadn(b, p, n) != n) break; p[n] = 0; if((s = strstr(p, "StreamTitle='")) != nil && (e = strstr(s+13, "';")) != nil && e != s+13){ *e = 0; if(sendp(aux->newtitle, strdup(s+13)) != 1){ free(s); break; } } } } free(p); Bterm(b); Bterm(out); chanclose(aux->newtitle); threadexits(nil); } int icyget(Meta *m, int outfd, Channel **newtitle) { char *s, *e, *p, *path, *d; int f, r, n, loc; Icyaux *aux; Biobuf *b; loc = 0; path = strdup(m->path); nextloc: s = strchr(path, ':')+3; if((e = strchr(s, '/')) != nil) *e++ = 0; if((p = strchr(s, ':')) != nil) *p = '!'; p = smprint("tcp!%s", s); f = -1; if((d = netmkaddr(p, "tcp", "80")) != nil) f = dial(d, nil, nil, nil); free(p); if(f < 0){ free(path); return -1; } fprint(f, "GET /%s HTTP/1.1\r\nHost: %s\r\nIcy-MetaData: 1\r\n\r\n", e ? e : "", s); free(path); b = Bfdopen(f, OREAD); aux = mallocz(sizeof(*aux), 1); aux->outfd = outfd; aux->newtitle = chancreate(sizeof(char*), 2); for(r = -1;;){ if((s = Brdline(b, '\n')) == nil) break; if((n = Blinelen(b)) < 2) break; if(n == 2 && *s == '\r'){ /* eof */ r = 0; break; } s[n-2] = 0; if(strncmp(s, "Location: ", 10) == 0){ if(++loc >= 10){ werrstr("too many links to follow"); r = -1; break; } Bterm(b); chanclose(aux->newtitle); path = strdup(s+10); goto nextloc; }else if(strncmp(s, "icy-name:", 9) == 0){ s += 9; if(newtitle != nil) sendp(aux->newtitle, strdup(s)); else if(m->title == nil) m->title = strdup(s); }else if(newtitle == nil && strncmp(s, "icy-url:", 8) == 0 && m->numartist == 0){ s += 8; m->artist[m->numartist++] = strdup(s); }else if(strncmp(s, "icy-metaint:", 12) == 0){ s += 12; aux->metaint = atoi(s); } } if(r < 0 || outfd < 0){ Bterm(b); b = nil; chanclose(aux->newtitle); free(aux); } if(b != nil){ assert(aux->newtitle != nil); b->aux = aux; *newtitle = aux->newtitle; proccreate(icyproc, b, 4096); } return r; }