kde-extraapps/kmix/core/mixertoolbox.cpp
Ivailo Monev 45dcabf7f8 generic: prepare for Katie changes
Signed-off-by: Ivailo Monev <xakepa10@gmail.com>
2022-11-13 15:37:30 +02:00

389 lines
13 KiB
C++

/*
* KMix -- KDE's full featured mini mixer
*
*
* Copyright (C) 2004 Christian Esken <esken@kde.org>
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* 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
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with this program; if not, write to the Free
* Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#include "core/mixer.h"
#include <QDir>
#include <QWidget>
#include <QString>
//#include <kdebug.h>
#include <klocale.h>
#include <kstandarddirs.h>
#include "core/kmixdevicemanager.h"
#include "core/mixdevice.h"
#include "core/mixertoolbox.h"
MixerToolBox* MixerToolBox::s_instance = 0;
QRegExp MixerToolBox::s_ignoreMixerExpression( QLatin1String( "Modem" ));
//KLocale* MixerToolBox::s_whatsthisLocale = 0;
/***********************************************************************************
Attention:
This MixerToolBox is linked to the KMix Main Program, the KMix Applet and kmixctrl.
As we do not want to link in more than necessary to kmixctrl, you are asked
not to put any GUI classes in here.
In the case where it is unavoidable, please put them in KMixToolBox.
***********************************************************************************/
MixerToolBox* MixerToolBox::instance()
{
if ( s_instance == 0 ) {
s_instance = new MixerToolBox();
// if ( s_ignoreMixerExpression.isEmpty() )
// s_ignoreMixerExpression.setPattern("Modem");
}
return s_instance;
}
/**
* Scan for Mixers in the System. This is the method that implicitely fills the
* list of Mixer's, which is accessible via the static Mixer::mixer() method.
*
* This is run only once during the initialization phase of KMix. It has the following tasks:
* 1) Coldplug scan, to fill the initial mixer list
* 2) Rember UDI's, to match them when unplugging a device
* 3) Find out, which Backend to use (plugin events of other Backends are ignored).
*
* @deprecated TODO this method has to go away. Migrate to MultiDriverMode enum
*
* @par multiDriverMode Whether the Mixer scan should try more all backendends.
* 'true' means to scan all backends. 'false' means: After scanning the
* current backend the next backend is only scanned if no Mixers were found yet.
* @par backendList Activated backends (typically a value from the kmixrc or a default)
* @par ref_hwInfoString Here a descripitive text of the scan is returned (Hardware Information)
*/
void MixerToolBox::initMixer(bool multiDriverModeBool, QList<QString> backendList, QString& ref_hwInfoString)
{
MultiDriverMode multiDriverMode = multiDriverModeBool ? MULTI : SINGLE_PLUS_MPRIS2;
initMixer(multiDriverMode, backendList, ref_hwInfoString);
}
void MixerToolBox::initMixer(MultiDriverMode multiDriverMode, QList<QString> backendList, QString& ref_hwInfoString)
{
initMixerInternal(multiDriverMode, backendList, ref_hwInfoString);
if ( Mixer::mixers().isEmpty() )
initMixerInternal(multiDriverMode, QList<QString>(), ref_hwInfoString); // try again without filter
}
/**
*
*/
void MixerToolBox::initMixerInternal(MultiDriverMode multiDriverMode, QList<QString> backendList, QString& ref_hwInfoString)
{
bool useBackendFilter = ( ! backendList.isEmpty() );
bool backendMprisFound = false; // only for SINGLE_PLUS_MPRIS2
bool regularBackendFound = false; // only for SINGLE_PLUS_MPRIS2
kDebug() << "multiDriverMode=" << multiDriverMode << ", backendList=" << backendList;
// Find all mixers and initialize them
int drvNum = Mixer::numDrivers();
int driverWithMixer = -1;
bool multipleDriversActive = false;
QString driverInfo = "";
QString driverInfoUsed = "";
for( int drv1=0; drv1<drvNum; drv1++ )
{
QString driverName = Mixer::driverName(drv1);
if ( driverInfo.length() > 0 ) {
driverInfo += " + ";
}
driverInfo += driverName;
}
/* Run a loop over all drivers. The loop will terminate after the first driver which
has mixers. And here is the reason:
- If you run ALSA with ALSA-OSS-Emulation enabled, mixers will show up twice: once
as native ALSA mixer, once as OSS mixer (emulated by ALSA). This is bad and WILL
confuse users. So it is a design decision that we can compile in multiple drivers
but we can run only one driver.
- For special usage scenarios, people will still want to run both drivers at the
same time. We allow them to hack their Config-File, where they can enable a
multi-driver mode.
- Another remark: For KMix3.0 or so, we should allow multiple-driver, for allowing
addition of special-use drivers, e.g. an Jack-Mixer-Backend, or a CD-Rom volume Backend.
*/
bool autodetectionFinished = false;
for( int drv=0; drv<drvNum; drv++ )
{
if ( autodetectionFinished )
{
// inner loop indicates that we are finished => sane exit from outer loop
break;
}
QString driverName = Mixer::driverName(drv);
kDebug(67100) << "Looking for mixers with the : " << driverName << " driver";
if ( useBackendFilter && ! backendList.contains(driverName) )
{
kDebug() << "Skipping " << driverName << " (filtered)";
continue;
}
bool regularBackend = driverName != "MPRIS2";
if (regularBackend && regularBackendFound)
{
// Only accept one regular backend => skip this one
continue;
}
bool drvInfoAppended = false;
// The "19" below is just a "silly" number:
// (Old: The loop will break as soon as an error is detected - e.g. on 3rd loop when 2 soundcards are installed)
// New: We don't try be that clever anymore. We now blindly scan 20 cards, as the clever
// approach doesn't work for the one or other user (e.g. hotplugging might create holes in the list of soundcards).
int devNumMax = 19;
for( int dev=-1; dev<=devNumMax; dev++ )
{
Mixer *mixer = new Mixer( driverName, dev );
bool mixerAccepted = possiblyAddMixer(mixer);
/* Lets decide if the autoprobing shall end.
* If the user has configured a backend filter, we will use that as a plain list to obey. It overrides the
* multiDriverMode.
*/
if ( ! useBackendFilter )
{
bool foundSomethingAndLastControlReached = dev == devNumMax && ! Mixer::mixers().isEmpty();
switch ( multiDriverMode )
{
case SINGLE:
// In Single-Driver-mode we only need to check after we reached devNumMax
if ( foundSomethingAndLastControlReached )
autodetectionFinished = true; // highest device number of driver and a Mixer => finished
break;
case MULTI:
// In multiDriver mode we scan all devices, so we will simply continue
break;
case SINGLE_PLUS_MPRIS2:
if ( driverName == "MPRIS2" )
{
backendMprisFound = true;
}
else
{
// same check as in SINGLE
if ( foundSomethingAndLastControlReached )
regularBackendFound = true;
}
if ( backendMprisFound && regularBackendFound )
autodetectionFinished = true; // highest device number of driver and a Mixer => finished
break;
}
}
else
{
// Using backend filter. This is a plain list to obey.
// Simply continue (and filter at the start of the loop).
}
if ( mixerAccepted )
{
kDebug(67100) << "Success! Found a mixer with the : " << driverName << " driver";
// append driverName (used drivers)
if ( !drvInfoAppended )
{
drvInfoAppended = true;
if ( Mixer::mixers().count() > 1)
driverInfoUsed += " + ";
driverInfoUsed += driverName;
}
// Check whether there are mixers in different drivers, so that the user can be warned
if ( !multipleDriversActive )
{
if ( driverWithMixer == -1 )
{
// Aha, this is the very first detected device
driverWithMixer = drv;
}
else if ( driverWithMixer != drv )
{
// Got him: There are mixers in different drivers
multipleDriversActive = true;
}
} // !multipleDriversActive
} // mixerAccepted
} // loop over sound card devices of current driver
if (autodetectionFinished) {
break;
}
} // loop over soundcard drivers
// Add a master device (if we haven't defined one yet)
if ( !Mixer::getGlobalMasterMD(false) ) {
// We have no master card yet. This actually only happens when there was
// not one defined in the kmixrc.
// So lets just set the first card as master card.
if ( Mixer::mixers().count() > 0 ) {
std::shared_ptr<MixDevice> master = Mixer::mixers().first()->getLocalMasterMD();
if ( master ) {
QString controlId = master->id();
Mixer::setGlobalMaster( Mixer::mixers().first()->id(), controlId, true);
}
}
}
else {
// setGlobalMaster was already set after reading the configuration.
// So we must make the local master consistent
std::shared_ptr<MixDevice> md = Mixer::getGlobalMasterMD();
QString mdID = md->id();
md->mixer()->setLocalMasterMD(mdID);
}
if ( Mixer::mixers().count() == 0 )
{
// If there was no mixer found, we assume, that hotplugging will take place
// on the preferred driver (this is always the first in the backend list).
driverInfoUsed = Mixer::driverName(0);
}
ref_hwInfoString = i18n("Sound drivers supported:");
ref_hwInfoString.append(" ").append( driverInfo ).append( "\n").append(i18n("Sound drivers used:")) .append(" ").append(driverInfoUsed);
if ( multipleDriversActive )
{
// this will only be possible by hacking the config-file, as it will not be officially supported
ref_hwInfoString.append("\n").append(i18n("Experimental multiple-Driver mode activated"));
QString allDrivermatch("*");
KMixDeviceManager::instance()->setHotpluggingBackends(allDrivermatch);
}
else {
KMixDeviceManager::instance()->setHotpluggingBackends(driverInfoUsed);
}
kDebug(67100) << ref_hwInfoString << "\nTotal number of detected Mixers: " << Mixer::mixers().count();
//kDebug(67100) << "OUT MixerToolBox::initMixer()";
}
/**
* Opens and adds a mixer to the KMix wide Mixer array, if the given Mixer is valid.
* Otherwise the Mixer is deleted.
* This method can be used for adding "static" devices (at program start) and also for hotplugging.
*
* @arg mixer
* @returns true if the Mixer was added
*/
bool MixerToolBox::possiblyAddMixer(Mixer *mixer)
{
if ( mixer->openIfValid() )
{
if ( (!s_ignoreMixerExpression.isEmpty()) && mixer->id().contains(s_ignoreMixerExpression) )
{
// This Mixer should be ignored (default expression is "Modem").
// next 3 lines are duplicated code
delete mixer;
mixer = 0;
return false;
}
else
{
Mixer::mixers().append( mixer );
kDebug(67100) << "Added card " << mixer->id();
emit mixerAdded(mixer->id()); // TODO should we still use this, as we now have our publish/subscribe notification system?
return true;
}
} // valid
else
{
delete mixer;
mixer = 0;
return false;
} // invalid
}
/* This allows to set an expression form Mixers that should be ignored.
The default is "Modem", because most people don't want to control the modem volume. */
void MixerToolBox::setMixerIgnoreExpression(const QString& ignoreExpr)
{
s_ignoreMixerExpression.setPattern(ignoreExpr);
}
QString MixerToolBox::mixerIgnoreExpression() const
{
return s_ignoreMixerExpression.pattern( );
}
void MixerToolBox::removeMixer(Mixer *par_mixer)
{
for (int i=0; i<Mixer::mixers().count(); ++i) {
Mixer *mixer = (Mixer::mixers())[i];
if ( mixer == par_mixer ) {
kDebug(67100) << "Removing card " << mixer->id();
Mixer::mixers().removeAt(i);
delete mixer;
}
}
}
/*
* Clean up and free all resources of all found Mixers, which were found in the initMixer() call
*/
void MixerToolBox::deinitMixer()
{
//kDebug(67100) << "IN MixerToolBox::deinitMixer()";
int mixerCount = Mixer::mixers().count();
for ( int i=0; i<mixerCount; ++i)
{
Mixer* mixer = (Mixer::mixers())[i];
//kDebug(67100) << "MixerToolBox::deinitMixer() Remove Mixer";
mixer->close();
delete mixer;
}
Mixer::mixers().clear();
// kDebug(67100) << "OUT MixerToolBox::deinitMixer()";
}
/*
KLocale* MixerToolBox::whatsthisControlLocale()
{
if ( s_whatsthisLocale == 0 ) {
s_whatsthisLocale = new KLocale("kmix-controls");
}
return s_whatsthisLocale;
}
*/
#include "moc_mixertoolbox.cpp"