mirror of
				https://github.com/ElvishArtisan/rivendell.git
				synced 2025-10-31 22:24:02 +01:00 
			
		
		
		
	* Reimplemented the '--autogen' directive in rdalsaconfig(8). * Added a '--rewrite' directive in rdalsaconfig(8).
		
			
				
	
	
		
			328 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			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++;
 | |
|   }
 | |
| }
 |