/*************************************************************************** * Copyright (C) 2006-2019 by Ilya Kotov * * forkotov02@ya.ru * * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU 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 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., * * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * ***************************************************************************/ #include #include extern "C" { #include } #include "settingsdialog.h" SettingsDialog::SettingsDialog ( QWidget *parent ) : QDialog ( parent ) { ui.setupUi (this); setAttribute (Qt::WA_DeleteOnClose); ui.deviceComboBox->setEditable (true); getCards(); getSoftDevices(); connect (ui.deviceComboBox, SIGNAL(activated(int)),SLOT(setText(int))); connect(ui.mixerCardComboBox, SIGNAL(activated(int)), SLOT(showMixerDevices(int))); QSettings settings(Qmmp::configFile(), QSettings::IniFormat); settings.beginGroup("ALSA"); ui.deviceComboBox->setEditText(settings.value("device","default").toString()); ui.bufferSpinBox->setValue(settings.value("buffer_time",500).toInt()); ui.periodSpinBox->setValue(settings.value("period_time",100).toInt()); int d = m_cards.indexOf(settings.value("mixer_card","hw:0").toString()); if (d >= 0) ui.mixerCardComboBox->setCurrentIndex(d); showMixerDevices(ui.mixerCardComboBox->currentIndex ()); d = ui.mixerDeviceComboBox->findText(settings.value("mixer_device", "PCM").toString()); if (d >= 0) ui.mixerDeviceComboBox->setCurrentIndex(d); ui.mmapCheckBox->setChecked(settings.value("use_mmap", false).toBool()); ui.pauseCheckBox->setChecked(settings.value("use_snd_pcm_pause", false).toBool()); settings.endGroup(); } SettingsDialog::~SettingsDialog() {} void SettingsDialog::getCards() { int card = -1, err; m_devices.clear(); m_devices << "default"; ui.deviceComboBox->addItem("Default PCM device (default)"); if ((err = snd_card_next(&card)) !=0) qWarning("SettingsDialog (ALSA): snd_next_card() failed: %s", snd_strerror(-err)); while (card > -1) { getCardDevices(card); m_cards << QString("hw:%1").arg(card); if ((err = snd_card_next(&card)) !=0) { qWarning("SettingsDialog (ALSA): snd_next_card() failed: %s", snd_strerror(-err)); break; } } } void SettingsDialog::getSoftDevices() { void **hints = nullptr; int i = 0; if(snd_device_name_hint(-1, "pcm", &hints) < 0) return; while(hints && hints[i]) { char *type = snd_device_name_get_hint (hints[i], "IOID"); if (!type || !strcmp (type, "Output")) { char *device_name = snd_device_name_get_hint (hints[i], "NAME"); char *device_desc = snd_device_name_get_hint (hints[i], "DESC"); m_devices << QString(device_name); QString str = QString("%1 (%2)").arg(device_desc).arg(device_name); qDebug("%s", qPrintable(str)); ui.deviceComboBox->addItem(str); free (device_name); free (device_desc); } if(type) free (type); ++i; } if (hints) snd_device_name_free_hint (hints); } void SettingsDialog::getCardDevices(int card) { int pcm_device = -1, err; snd_pcm_info_t *pcm_info; snd_ctl_t *ctl; char dev[64], *card_name; sprintf(dev, "hw:%i", card); if ((err = snd_ctl_open(&ctl, dev, 0)) < 0) { qWarning("SettingsDialog (ALSA): snd_ctl_open() failed: %s", snd_strerror(-err)); return; } if ((err = snd_card_get_name(card, &card_name)) != 0) { qWarning("SettingsDialog (ALSA): snd_card_get_name() failed: %s", snd_strerror(-err)); card_name = strdup("Unknown soundcard"); } ui.mixerCardComboBox->addItem(QString(card_name)); snd_pcm_info_alloca(&pcm_info); qDebug("SettingsDialog (ALSA): detected sound cards:"); for (;;) { QString device; if ((err = snd_ctl_pcm_next_device(ctl, &pcm_device)) < 0) { qWarning("SettingsDialog (ALSA): snd_ctl_pcm_next_device() failed: %s", snd_strerror(-err)); pcm_device = -1; } if (pcm_device < 0) break; snd_pcm_info_set_device(pcm_info, pcm_device); snd_pcm_info_set_subdevice(pcm_info, 0); snd_pcm_info_set_stream(pcm_info, SND_PCM_STREAM_PLAYBACK); if ((err = snd_ctl_pcm_info(ctl, pcm_info)) < 0) { if (err != -ENOENT) qWarning("SettingsDialog (ALSA): get_devices_for_card(): " "snd_ctl_pcm_info() " "failed (%d:%d): %s.", card, pcm_device, snd_strerror(-err)); } device = QString("hw:%1,%2").arg(card).arg(pcm_device); m_devices << device; QString str; str = QString(card_name) + ": "+ snd_pcm_info_get_name(pcm_info)+" ("+device+")"; qDebug("%s",qPrintable(str)); ui.deviceComboBox->addItem(str); } free(card_name); snd_ctl_close(ctl); } void SettingsDialog::getMixerDevices(QString card) { ui.mixerDeviceComboBox->clear(); snd_mixer_t *mixer; snd_mixer_elem_t *current; if (getMixer(&mixer, card) < 0) return; current = snd_mixer_first_elem(mixer); while (current) { const char *sname = snd_mixer_selem_get_name(current); if (snd_mixer_selem_is_active(current) && snd_mixer_selem_has_playback_volume(current)) ui.mixerDeviceComboBox->addItem(QString(sname)); current = snd_mixer_elem_next(current); } } void SettingsDialog::setText(int n) { ui.deviceComboBox->setEditText(m_devices.at(n)); } void SettingsDialog::accept() { qDebug("SettingsDialog (ALSA):: writeSettings()"); QSettings settings(Qmmp::configFile(), QSettings::IniFormat); settings.beginGroup("ALSA"); settings.setValue("device", ui.deviceComboBox->currentText ()); settings.setValue("buffer_time",ui.bufferSpinBox->value()); settings.setValue("period_time",ui.periodSpinBox->value()); if(ui.mixerCardComboBox->currentIndex() >= 0) { QString card = m_cards.at(ui.mixerCardComboBox->currentIndex()); settings.setValue("mixer_card", card); } settings.setValue("mixer_device", ui.mixerDeviceComboBox->currentText ()); settings.setValue("use_mmap", ui.mmapCheckBox->isChecked()); settings.setValue("use_snd_pcm_pause", ui.pauseCheckBox->isChecked()); settings.endGroup(); QDialog::accept(); } int SettingsDialog::getMixer(snd_mixer_t **mixer, QString card) { int err; if ((err = snd_mixer_open(mixer, 0)) < 0) { qWarning("SettingsDialog (ALSA): alsa_get_mixer(): " "Failed to open empty mixer: %s", snd_strerror(-err)); mixer = NULL; return -1; } if ((err = snd_mixer_attach(*mixer, card.toLatin1().constData())) < 0) { qWarning("SettingsDialog (ALSA): alsa_get_mixer(): " "Attaching to mixer %s failed: %s", qPrintable(card), snd_strerror(-err)); return -1; } if ((err = snd_mixer_selem_register(*mixer, nullptr, nullptr)) < 0) { qWarning("SettingsDialog (ALSA): alsa_get_mixer(): " "Failed to register mixer: %s", snd_strerror(-err)); return -1; } if ((err = snd_mixer_load(*mixer)) < 0) { qWarning("SettingsDialog (ALSA): alsa_get_mixer(): Failed to load mixer: %s", snd_strerror(-err)); return -1; } return (*mixer != nullptr); } void SettingsDialog::showMixerDevices(int d) { if (0<=d && d