Fred Gleason a658fda4fc 2019-08-26 Fred Gleason <fredg@paravelsystems.com>
* Reimplemented the '--autogen' directive in rdalsaconfig(8).
	* Added a '--rewrite' directive in rdalsaconfig(8).
2019-08-26 16:17:24 -04:00

328 lines
7.0 KiB
C++

// rdalsamodel.cpp
//
// Abstract an ALSA configuration.
//
// (C) Copyright 2009-2019 Fred Gleason <fredg@paravelsystems.com>
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License version 2 as
// published by the Free Software Foundation.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public
// License along with this program; if not, write to the Free Software
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
//
#include <stdlib.h>
#include <stdio.h>
#include <alsa/asoundlib.h>
#include <qstringlist.h>
#include <rdapplication.h>
#include <rdalsamodel.h>
RDAlsaModel::RDAlsaModel(unsigned samprate,QObject *parent)
: QAbstractListModel(parent)
{
model_sample_rate=samprate;
LoadSystemConfig();
}
int RDAlsaModel::rowCount(const QModelIndex &parent) const
{
int rows=0;
for(int i=0;i<model_alsa_cards.size();i++) {
rows+=model_alsa_cards.at(i)->pcmQuantity();
}
return rows;
}
Qt::ItemFlags RDAlsaModel::flags(const QModelIndex &index) const
{
Qt::ItemFlags flags=QAbstractListModel::flags(index);
if((model_alsa_cards.at(model_card_index.at(index.row()))->id()=="Axia")&&
(model_sample_rate!=48000)) {
flags=flags&Qt::ItemIsEnabled;
}
return flags;
}
QVariant RDAlsaModel::data(const QModelIndex &index,int role) const
{
int row=index.row();
switch((Qt::ItemDataRole)role) {
case Qt::DisplayRole:
return QVariant(model_alsa_cards.at(model_card_index.at(row))->name()+" - "+
model_alsa_cards.at(model_card_index.at(row))->
pcmName(model_pcm_index.at(row)));
break;
case Qt::DecorationRole:
case Qt::EditRole:
case Qt::ToolTipRole:
case Qt::StatusTipRole:
case Qt::WhatsThisRole:
case Qt::SizeHintRole:
case Qt::FontRole:
case Qt::TextAlignmentRole:
case Qt::BackgroundColorRole:
case Qt::TextColorRole:
case Qt::CheckStateRole:
case Qt::AccessibleTextRole:
case Qt::AccessibleDescriptionRole:
case Qt::InitialSortOrderRole:
case Qt::DisplayPropertyRole:
case Qt::DecorationPropertyRole:
case Qt::ToolTipPropertyRole:
case Qt::StatusTipPropertyRole:
case Qt::WhatsThisPropertyRole:
case Qt::UserRole:
break;
}
return QVariant();
}
QVariant RDAlsaModel::headerData(int section,Qt::Orientation orient,
int role) const
{
switch(orient) {
case Qt::Horizontal:
return QVariant(tr("ALSA Devices"));
case Qt::Vertical:
break;
}
return QVariant();
}
QModelIndex RDAlsaModel::indexOf(const QString &card_id,int pcm_num) const
{
bool ok=false;
int cardnum=card_id.toUInt(&ok);
if(ok) {
for(int i=0;i<model_card_index.size();i++) {
if((model_card_index.at(i)==cardnum)&&
(model_pcm_index.at(i)==pcm_num)) {
return createIndex(i,0);
}
}
}
else {
for(int i=0;i<model_card_index.size();i++) {
if((model_alsa_cards.at(model_card_index.at(i))->id()==card_id)&&
(model_pcm_index.at(i)==pcm_num)) {
return createIndex(i,0);
}
}
}
return QModelIndex();
}
RDAlsaCard *RDAlsaModel::card(const QModelIndex &index) const
{
return model_alsa_cards.at(model_card_index.at(index.row()));
}
int RDAlsaModel::pcmNumber(const QModelIndex &index) const
{
return model_pcm_index.at(index.row());
}
bool RDAlsaModel::isEnabled(int row) const
{
return model_alsa_cards.at(model_card_index.at(row))->
isEnabled(model_pcm_index.at(row));
}
void RDAlsaModel::setEnabled(int row,bool state)
{
return model_alsa_cards.at(model_card_index.at(row))->
setEnabled(model_pcm_index.at(row),state);
}
bool RDAlsaModel::loadConfig(const QString &filename)
{
FILE *f=NULL;
char line[1024];
int istate=0;
int port=0;
QString card_id=0;
int card_num=0;
bool ok=false;
int device=0;
QStringList list;
bool active_line=false;
QModelIndex index;
if((f=fopen(filename.toUtf8(),"r"))==NULL) {
return false;
}
while(fgets(line,1024,f)!=NULL) {
QString str=line;
str.replace("\n","");
if(str==START_MARKER) {
active_line=true;
}
if(str==END_MARKER) {
active_line=false;
}
if((str!=START_MARKER)&&(str!=END_MARKER)) {
if(active_line) {
switch(istate) {
case 0:
if(str.left(6)=="pcm.rd") {
port=str.mid(6,1).toInt();
istate=1;
}
else {
if(str.left(6)=="ctl.rd") {
istate=10;
}
else {
model_other_lines.push_back(str+"\n");
}
}
break;
case 1:
list=str.split(" ",QString::SkipEmptyParts);
if(list[0]=="}") {
if((port>=0)&&(port<RD_MAX_CARDS)) {
for(int i=0;i<model_alsa_cards.size();i++) {
RDAlsaCard *card=model_alsa_cards.at(i);
card_num=card_id.toUInt(&ok);
if(ok) {
if(card_num==card->index()) {
if((device>=0)&&(device<card->pcmQuantity())) {
card->setEnabled(device,true);
}
}
}
else {
if(card_id==card->id()) {
if((device>=0)&&(device<card->pcmQuantity())) {
card->setEnabled(device,true);
}
}
}
}
}
card_id="";
device=0;
istate=0;
}
else {
if(list.size()==2) {
if(list[0]=="card") {
card_id=list[1].trimmed();
}
if(list[0]=="device") {
device=list[1].toInt();
}
}
}
break;
case 10:
if(str.left(1)=="}") {
istate=0;
}
break;
}
}
else {
model_other_lines.push_back(str+"\n");
}
}
}
fclose(f);
return true;
}
bool RDAlsaModel::saveConfig(const QString &filename)
{
QString tempfile=filename+"-temp";
FILE *f=NULL;
int index=0;
if((f=fopen(tempfile.toUtf8(),"w"))==NULL) {
return false;
}
for(int i=0;i<model_other_lines.size();i++) {
fprintf(f,model_other_lines.at(i));
}
fprintf(f,"%s\n",START_MARKER);
for(int i=0;i<model_alsa_cards.size();i++) {
RDAlsaCard *card=model_alsa_cards.at(i);
for(int j=0;j<card->pcmQuantity();j++) {
if(card->isEnabled(j)) {
fprintf(f,"pcm.rd%d {\n",index);
fprintf(f," type hw\n");
fprintf(f," card %s\n",(const char *)card->id().toUtf8());
fprintf(f," device %d\n",j);
fprintf(f," rate %u\n",rda->system()->sampleRate());
if(card->id()=="Axia") {
fprintf(f," channels 2\n");
}
fprintf(f,"}\n");
fprintf(f,"ctl.rd%d {\n",index);
fprintf(f," type hw\n");
fprintf(f," card %s\n",(const char *)card->id().toUtf8());
fprintf(f,"}\n");
index++;
}
}
}
fprintf(f,"%s\n",END_MARKER);
fclose(f);
rename(tempfile.toUtf8(),filename.toUtf8());
return true;
}
void RDAlsaModel::LoadSystemConfig()
{
snd_ctl_t *snd_ctl=NULL;
int index=0;
while(snd_ctl_open(&snd_ctl,QString().sprintf("hw:%d",index),0)>=0) {
model_alsa_cards.push_back(new RDAlsaCard(snd_ctl,index));
for(int i=0;i<model_alsa_cards.back()->pcmQuantity();i++) {
model_card_index.push_back(index);
model_pcm_index.push_back(i);
}
snd_ctl_close(snd_ctl);
index++;
}
}