audiohda: automatically pick inpin

This commit is contained in:
cinap_lenrek 2013-01-19 02:30:33 +01:00
parent 1f8313acc1
commit 8cafbf3184

View file

@ -887,6 +887,8 @@ connectpin(Ctlr *ctlr, Stream *s, int type, uint pin, uint cad)
{ {
Widget *jack, *conv; Widget *jack, *conv;
if(s->atag == 0)
return -1;
if(cad >= Maxcodecs || pin >= Maxwidgets || ctlr->codec[cad] == nil) if(cad >= Maxcodecs || pin >= Maxwidgets || ctlr->codec[cad] == nil)
return -1; return -1;
jack = ctlr->codec[cad]->widgets[pin]; jack = ctlr->codec[cad]->widgets[pin];
@ -929,12 +931,57 @@ connectpin(Ctlr *ctlr, Stream *s, int type, uint pin, uint cad)
} }
static int static int
bestpin(Ctlr *ctlr, int *pcad) scoreout(Widget *w)
{
int score;
uint r;
if((w->pincap & Pout) == 0)
return -1;
if(w->id.ctlr->sin.jack == w)
return -1;
score = 0;
r = w->pin;
if(((r >> 30) & 0x3) >= 2) /* fix or fix+jack */
score |= 32;
if(((r >> 12) & 0xf) == 4) /* green */
score |= 32;
if(((r >> 24) & 0xf) == 1) /* rear */
score |= 16;
if(((r >> 28) & 0x3) == 0) /* ext */
score |= 8;
if(((r >> 20) & 0xf) == 2) /* hpout */
score |= 4;
if(((r >> 20) & 0xf) == 0) /* lineout */
score |= 4;
return score;
}
static int
scorein(Widget *w)
{
int score;
uint r;
if((w->pincap & Pin) == 0)
return -1;
if(w->id.ctlr->sout.jack == w)
return -1;
score = 0;
r = w->pin;
if(((r >> 30) & 0x3) >= 2) /* fix or fix+jack */
score |= 4;
return score;
}
static int
bestpin(Ctlr *ctlr, int *pcad, int (*fscore)(Widget *))
{ {
Fungroup *fg; Fungroup *fg;
Widget *w; Widget *w;
int best, pin, score; int best, pin, score;
uint r;
int i; int i;
pin = -1; pin = -1;
@ -946,21 +993,8 @@ bestpin(Ctlr *ctlr, int *pcad)
for(w=fg->first; w; w=w->next){ for(w=fg->first; w; w=w->next){
if(w->type != Wpin) if(w->type != Wpin)
continue; continue;
if((w->pincap & Pout) == 0) score = (*fscore)(w);
continue; if(score >= 0 && score >= best){
score = 0;
r = w->pin;
if(((r >> 12) & 0xf) == 4) /* green */
score |= 32;
if(((r >> 24) & 0xf) == 1) /* rear */
score |= 16;
if(((r >> 28) & 0x3) == 0) /* ext */
score |= 8;
if(((r >> 20) & 0xf) == 2) /* hpout */
score |= 4;
if(((r >> 20) & 0xf) == 0) /* lineout */
score |= 4;
if(score >= best){
best = score; best = score;
pin = w->id.nid; pin = w->id.nid;
*pcad = i; *pcad = i;
@ -1156,7 +1190,7 @@ hdactl(Audio *adev, void *va, long n, vlong)
error("connectpin failed"); error("connectpin failed");
}else }else
if(cistrcmp(tok[0], "inpin") == 0 && ntok >= 2){ if(cistrcmp(tok[0], "inpin") == 0 && ntok >= 2){
cad = ctlr->sout.cad; cad = ctlr->sin.cad;
pin = strtoul(tok[1], 0, 0); pin = strtoul(tok[1], 0, 0);
if(ntok > 2) if(ntok > 2)
cad = strtoul(tok[2], 0, 0); cad = strtoul(tok[2], 0, 0);
@ -1243,6 +1277,8 @@ hdawrite(Audio *adev, void *vp, long n, vlong)
e = p + n; e = p + n;
ctlr = adev->ctlr; ctlr = adev->ctlr;
ring = &ctlr->sout.ring; ring = &ctlr->sout.ring;
if(ring->buf == nil || ctlr->sout.conv == nil)
return 0;
while(p < e) { while(p < e) {
if((n = writering(ring, p, e - p)) <= 0){ if((n = writering(ring, p, e - p)) <= 0){
hdakick(ctlr); hdakick(ctlr);
@ -1696,26 +1732,40 @@ Found:
print("#A%d: unable to start hda\n", ctlr->no); print("#A%d: unable to start hda\n", ctlr->no);
return -1; return -1;
} }
if(streamalloc(ctlr, &ctlr->sout, ctlr->iss) < 0){
/* iss + oss + bss */
if(streamalloc(ctlr, &ctlr->sout, ctlr->iss) < 0)
print("#A%d: output streamalloc failed\n", ctlr->no); print("#A%d: output streamalloc failed\n", ctlr->no);
return -1; if(ctlr->iss > 0){
}
if(ctlr->iss > 0)
if(streamalloc(ctlr, &ctlr->sin, 0) < 0) if(streamalloc(ctlr, &ctlr->sin, 0) < 0)
print("#A%d: input streamalloc failed\n", ctlr->no); print("#A%d: input streamalloc failed\n", ctlr->no);
}
else if(ctlr->bss > 0){
if(ctlr->oss > 0){
if(streamalloc(ctlr, &ctlr->sin, ctlr->oss) < 0)
print("#A%d: input streamalloc failed\n", ctlr->no);
} else if(ctlr->bss > 1) {
if(streamalloc(ctlr, &ctlr->sin, 1) < 0)
print("#A%d: input streamalloc failed\n", ctlr->no);
}
}
if(enumdev(ctlr) < 0){ if(enumdev(ctlr) < 0){
print("#A%d: no audio codecs found\n", ctlr->no); print("#A%d: no audio codecs found\n", ctlr->no);
return -1; return -1;
} }
best = bestpin(ctlr, &cad);
if(best < 0){ best = bestpin(ctlr, &cad, scoreout);
print("#A%d: no output pins found!\n", ctlr->no); if(best < 0)
return -1; print("#A%d: no output pins found\n", ctlr->no);
} else if(connectpin(ctlr, &ctlr->sout, Waout, best, cad) < 0)
if(connectpin(ctlr, &ctlr->sout, Waout, best, cad) < 0){ print("#A%d: error connecting output pin\n", ctlr->no);
print("#A%d: error connecting pin\n", ctlr->no);
return -1; best = bestpin(ctlr, &cad, scorein);
} if(best < 0)
print("#A%d: no input pins found\n", ctlr->no);
else if(connectpin(ctlr, &ctlr->sin, Wain, best, cad) < 0)
print("#A%d: error connecting input pin\n", ctlr->no);
adev->read = hdaread; adev->read = hdaread;
adev->write = hdawrite; adev->write = hdawrite;